dpkg-gencontrol.pl 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341
  1. #!/usr/bin/perl
  2. use strict;
  3. use warnings;
  4. our $version = '1.3.0'; # This line modified by Makefile
  5. our $dpkglibdir = "."; # This line modified by Makefile
  6. our $pkgdatadir = ".."; # This line modified by Makefile
  7. use POSIX;
  8. use POSIX qw(:errno_h);
  9. push(@INC,$dpkglibdir);
  10. require 'controllib.pl';
  11. our $progname;
  12. our %substvar;
  13. our (%f, %fi);
  14. our %fieldimps;
  15. our %p2i;
  16. our @pkg_dep_fields;
  17. our $sourcepackage;
  18. our $host_arch;
  19. require 'dpkg-gettext.pl';
  20. textdomain("dpkg-dev");
  21. my @control_fields = (qw(Package Source Version Architecture Essential Origin
  22. Bugs Maintainer Installed-Size), @pkg_dep_fields,
  23. qw(Section Priority Description));
  24. my $controlfile = 'debian/control';
  25. my $changelogfile = 'debian/changelog';
  26. my $changelogformat;
  27. my $fileslistfile = 'debian/files';
  28. my $varlistfile = 'debian/substvars';
  29. my $packagebuilddir = 'debian/tmp';
  30. my $sourceversion;
  31. my $forceversion;
  32. my $forcefilename;
  33. my $stdout;
  34. my %remove;
  35. my %override;
  36. my (%spvalue, %spdefault);
  37. my $oppackage;
  38. sub version {
  39. printf _g("Debian %s version %s.\n"), $progname, $version;
  40. printf _g("
  41. Copyright (C) 1996 Ian Jackson.
  42. Copyright (C) 2000,2002 Wichert Akkerman.");
  43. printf _g("
  44. This is free software; see the GNU General Public Licence version 2 or
  45. later for copying conditions. There is NO warranty.
  46. ");
  47. }
  48. sub usage {
  49. printf _g(
  50. "Usage: %s [<option> ...]
  51. Options:
  52. -p<package> print control file for package.
  53. -c<controlfile> get control info from this file.
  54. -l<changelogfile> get per-version info from this file.
  55. -F<changelogformat> force change log format.
  56. -v<forceversion> set version of binary package.
  57. -f<fileslistfile> write files here instead of debian/files.
  58. -P<packagebuilddir> temporary build dir instead of debian/tmp.
  59. -n<filename> assume the package filename will be <filename>.
  60. -O write to stdout, not .../DEBIAN/control.
  61. -is, -ip, -isp, -ips deprecated, ignored for compatibility.
  62. -D<field>=<value> override or add a field and value.
  63. -U<field> remove a field.
  64. -V<name>=<value> set a substitution variable.
  65. -T<varlistfile> read variables here, not debian/substvars.
  66. -h, --help show this help message.
  67. --version show the version.
  68. "), $progname;
  69. }
  70. while (@ARGV) {
  71. $_=shift(@ARGV);
  72. if (m/^-p([-+0-9a-z.]+)$/) {
  73. $oppackage= $1;
  74. } elsif (m/^-p(.*)/) {
  75. &error(sprintf(_g("Illegal package name \`%s'"), $1));
  76. } elsif (m/^-c/) {
  77. $controlfile= $';
  78. } elsif (m/^-l/) {
  79. $changelogfile= $';
  80. } elsif (m/^-P/) {
  81. $packagebuilddir= $';
  82. } elsif (m/^-f/) {
  83. $fileslistfile= $';
  84. } elsif (m/^-v(.+)$/) {
  85. $forceversion= $1;
  86. } elsif (m/^-O$/) {
  87. $stdout= 1;
  88. } elsif (m/^-i[sp][sp]?$/) {
  89. # ignored for backwards compatibility
  90. } elsif (m/^-F([0-9a-z]+)$/) {
  91. $changelogformat=$1;
  92. } elsif (m/^-D([^\=:]+)[=:]/) {
  93. $override{$1}= $';
  94. } elsif (m/^-U([^\=:]+)$/) {
  95. $remove{$1}= 1;
  96. } elsif (m/^-V(\w[-:0-9A-Za-z]*)[=:]/) {
  97. $substvar{$1}= $';
  98. } elsif (m/^-T/) {
  99. $varlistfile= $';
  100. } elsif (m/^-n/) {
  101. $forcefilename= $';
  102. } elsif (m/^-(h|-help)$/) {
  103. &usage; exit(0);
  104. } elsif (m/^--version$/) {
  105. &version; exit(0);
  106. } else {
  107. &usageerr(sprintf(_g("unknown option \`%s'"), $_));
  108. }
  109. }
  110. parsechangelog($changelogfile, $changelogformat);
  111. parsesubstvars($varlistfile);
  112. parsecontrolfile($controlfile);
  113. my $myindex;
  114. if (defined($oppackage)) {
  115. defined($p2i{"C $oppackage"}) || &error(sprintf(_g("package %s not in control info"), $oppackage));
  116. $myindex= $p2i{"C $oppackage"};
  117. } else {
  118. my @packages = grep(m/^C /, keys %p2i);
  119. @packages==1 ||
  120. &error(sprintf(_g("must specify package since control info has many (%s)"), "@packages"));
  121. $myindex=1;
  122. }
  123. #print STDERR "myindex $myindex\n";
  124. my %pkg_dep_fields = map { $_ => 1 } @pkg_dep_fields;
  125. for $_ (keys %fi) {
  126. my $v = $fi{$_};
  127. if (s/^C //) {
  128. #print STDERR "G key >$_< value >$v<\n";
  129. if (m/^(Origin|Bugs|Maintainer)$/) {
  130. $f{$_} = $v;
  131. } elsif (m/^Source$/) {
  132. setsourcepackage($v);
  133. }
  134. elsif (s/^X[CS]*B[CS]*-//i) { $f{$_}= $v; }
  135. elsif (m/^X[CS]+-|^(Standards-Version|Uploaders)$|^Build-(Depends|Conflicts)(-Indep)?$/i) { }
  136. elsif (m/^Section$|^Priority$/) { $spdefault{$_}= $v; }
  137. else { $_ = "C $_"; &unknown(_g('general section of control info file')); }
  138. } elsif (s/^C$myindex //) {
  139. #print STDERR "P key >$_< value >$v<\n";
  140. if (m/^(Package|Description|Essential|Optional)$/) {
  141. $f{$_}= $v;
  142. } elsif (exists($pkg_dep_fields{$_})) {
  143. } elsif (m/^Section$|^Priority$/) {
  144. $spvalue{$_}= $v;
  145. } elsif (m/^Architecture$/) {
  146. my $host_arch = get_host_arch();
  147. if (debarch_eq('all', $v)) {
  148. $f{$_}= $v;
  149. } else {
  150. my @archlist = split(/\s+/, $v);
  151. my @invalid_archs = grep m/[^\w-]/, @archlist;
  152. warning(sprintf(ngettext(
  153. "`%s' is not a legal architecture string.",
  154. "`%s' are not legal architecture strings.",
  155. scalar(@invalid_archs)),
  156. join("' `", @invalid_archs)))
  157. if @invalid_archs >= 1;
  158. grep(debarch_is($host_arch, $_), @archlist) ||
  159. &error(sprintf(_g("current build architecture %s does not".
  160. " appear in package's list (%s)"),
  161. $host_arch, "@archlist"));
  162. $f{$_} = $host_arch;
  163. }
  164. } elsif (s/^X[CS]*B[CS]*-//i) {
  165. $f{$_}= $v;
  166. } elsif (!m/^X[CS]+-/i) {
  167. $_ = "C$myindex $_"; &unknown(_g("package's section of control info file"));
  168. }
  169. } elsif (m/^C\d+ /) {
  170. #print STDERR "X key >$_< value not shown<\n";
  171. } elsif (s/^L //) {
  172. #print STDERR "L key >$_< value >$v<\n";
  173. if (m/^Source$/) {
  174. setsourcepackage($v);
  175. } elsif (m/^Version$/) {
  176. $sourceversion= $v;
  177. $f{$_} = $v unless defined($forceversion);
  178. } elsif (m/^(Maintainer|Changes|Urgency|Distribution|Date|Closes)$/) {
  179. } elsif (s/^X[CS]*B[CS]*-//i) {
  180. $f{$_}= $v;
  181. } elsif (!m/^X[CS]+-/i) {
  182. $_ = "L $_"; &unknown(_g("parsed version of changelog"));
  183. }
  184. } elsif (m/o:/) {
  185. } else {
  186. &internerr(sprintf(_g("value from nowhere, with key >%s< and value >%s<"), $_, $v));
  187. }
  188. }
  189. $f{'Version'} = $forceversion if defined($forceversion);
  190. &init_substvars;
  191. init_substvar_arch();
  192. for $_ (keys %fi) {
  193. my $v = $fi{$_};
  194. if (s/^C //) {
  195. } elsif (s/^C$myindex //) {
  196. if (m/^(Package|Description|Essential|Optional)$/) {
  197. } elsif (exists($pkg_dep_fields{$_})) {
  198. my $dep = parsedep(substvars($v), 1, 1);
  199. &error(sprintf(_g("error occurred while parsing %s"), $_)) unless defined $dep;
  200. $f{$_}= showdep($dep, 0);
  201. } elsif (m/^Section$|^Priority$/) {
  202. } elsif (m/^Architecture$/) {
  203. } elsif (s/^X[CS]*B[CS]*-//i) {
  204. } elsif (!m/^X[CS]+-/i) {
  205. }
  206. } elsif (m/^C\d+ /) {
  207. } elsif (s/^L //) {
  208. } elsif (m/o:/) {
  209. } else {
  210. }
  211. }
  212. for my $f (qw(Section Priority)) {
  213. $spvalue{$f} = $spdefault{$f} unless defined($spvalue{$f});
  214. $f{$f} = $spvalue{$f} if defined($spvalue{$f});
  215. }
  216. for my $f (qw(Package Version)) {
  217. defined($f{$f}) || &error(sprintf(_g("missing information for output field %s"), $f));
  218. }
  219. for my $f (qw(Maintainer Description Architecture)) {
  220. defined($f{$f}) || warning(sprintf(_g("missing information for output field %s"), $f));
  221. }
  222. $oppackage= $f{'Package'};
  223. my $verdiff = $f{'Version'} ne $substvar{'source:Version'} ||
  224. $f{'Version'} ne $sourceversion;
  225. if ($oppackage ne $sourcepackage || $verdiff) {
  226. $f{'Source'}= $sourcepackage;
  227. $f{'Source'}.= " ($substvar{'source:Version'})" if $verdiff;
  228. }
  229. if (!defined($substvar{'Installed-Size'})) {
  230. defined(my $c = open(DU, "-|")) || syserr(_g("fork for du"));
  231. if (!$c) {
  232. chdir("$packagebuilddir") || &syserr(sprintf(_g("chdir for du to \`%s'"), $packagebuilddir));
  233. exec("du","-k","-s",".") or &syserr(_g("exec du"));
  234. }
  235. my $duo = '';
  236. while (<DU>) {
  237. $duo .= $_;
  238. }
  239. close(DU); $? && &subprocerr(sprintf(_g("du in \`%s'"), $packagebuilddir));
  240. $duo =~ m/^(\d+)\s+\.$/ || &failure(sprintf(_g("du gave unexpected output \`%s'"), $duo));
  241. $substvar{'Installed-Size'}= $1;
  242. }
  243. if (defined($substvar{'Extra-Size'})) {
  244. $substvar{'Installed-Size'} += $substvar{'Extra-Size'};
  245. }
  246. if (defined($substvar{'Installed-Size'})) {
  247. $f{'Installed-Size'}= $substvar{'Installed-Size'};
  248. }
  249. for my $f (keys %override) {
  250. $f{capit($f)} = $override{$f};
  251. }
  252. for my $f (keys %remove) {
  253. delete $f{capit($f)};
  254. }
  255. $fileslistfile="./$fileslistfile" if $fileslistfile =~ m/^\s/;
  256. open(Y,"> $fileslistfile.new") || &syserr(_g("open new files list file"));
  257. binmode(Y);
  258. chown(getfowner(), "$fileslistfile.new")
  259. || &syserr(_g("chown new files list file"));
  260. if (open(X,"< $fileslistfile")) {
  261. binmode(X);
  262. while (<X>) {
  263. chomp;
  264. next if m/^([-+0-9a-z.]+)_[^_]+_([\w-]+)\.deb /
  265. && ($1 eq $oppackage)
  266. && (debarch_eq($2, $f{'Architecture'})
  267. || debarch_eq($2, 'all'));
  268. print(Y "$_\n") || &syserr(_g("copy old entry to new files list file"));
  269. }
  270. close(X) || &syserr(_g("close old files list file"));
  271. } elsif ($! != ENOENT) {
  272. &syserr(_g("read old files list file"));
  273. }
  274. my $sversion = $f{'Version'};
  275. $sversion =~ s/^\d+://;
  276. $forcefilename=sprintf("%s_%s_%s.deb", $oppackage,$sversion,$f{'Architecture'})
  277. unless ($forcefilename);
  278. print(Y &substvars(sprintf("%s %s %s\n", $forcefilename,
  279. &spfileslistvalue('Section'), &spfileslistvalue('Priority'))))
  280. || &syserr(_g("write new entry to new files list file"));
  281. close(Y) || &syserr(_g("close new files list file"));
  282. rename("$fileslistfile.new",$fileslistfile) || &syserr(_g("install new files list file"));
  283. my $cf;
  284. if (!$stdout) {
  285. $cf= "$packagebuilddir/DEBIAN/control";
  286. $cf= "./$cf" if $cf =~ m/^\s/;
  287. open(STDOUT,"> $cf.new") ||
  288. &syserr(sprintf(_g("cannot open new output control file \`%s'"), "$cf.new"));
  289. binmode(STDOUT);
  290. }
  291. set_field_importance(@control_fields);
  292. outputclose($varlistfile);
  293. if (!$stdout) {
  294. rename("$cf.new","$cf") || &syserr(sprintf(_g("cannot install output control file \`%s'"), $cf));
  295. }
  296. sub spfileslistvalue {
  297. my $r = $spvalue{$_[0]};
  298. $r = '-' if !defined($r);
  299. return $r;
  300. }