dpkg-scanpackages.pl 7.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289
  1. #!/usr/bin/perl
  2. use warnings;
  3. use strict;
  4. use IO::Handle;
  5. use IO::File;
  6. my $version= '1.2.6'; # This line modified by Makefile
  7. my $dpkglibdir= "."; # This line modified by Makefile
  8. ($0) = $0 =~ m:.*/(.+):;
  9. push(@INC,$dpkglibdir);
  10. require 'controllib.pl';
  11. require 'dpkg-gettext.pl';
  12. textdomain("dpkg-dev");
  13. our @pkg_dep_fields;
  14. my (@samemaint, @changedmaint);
  15. my %packages;
  16. my %overridden;
  17. my %kmap= (optional => 'suggests',
  18. recommended => 'recommends',
  19. class => 'priority',
  20. package_revision => 'revision',
  21. );
  22. my @fieldpri = (qw(Package Source Version Architecture Essential Origin Bugs
  23. Maintainer Installed-Size), @pkg_dep_fields, qw(Filename
  24. Size MD5sum Section Priority Description));
  25. # This maps the fields into the proper case
  26. my %field_case;
  27. @field_case{map{lc($_)} @fieldpri} = @fieldpri;
  28. use Getopt::Long qw(:config bundling);
  29. my %options = (help => sub { &usage; exit 0; },
  30. version => \&version,
  31. udeb => 0,
  32. arch => undef,
  33. multiversion => 0,
  34. );
  35. my $result = GetOptions(\%options,'help|h|?','version','udeb|u!','arch|a=s','multiversion|m!');
  36. sub version {
  37. printf _g("Debian %s version %s.\n"), $0, $version;
  38. exit;
  39. }
  40. sub usage {
  41. printf _g(
  42. "Usage: %s [<option> ...] <binarypath> [<overridefile> [<pathprefix>]] > Packages
  43. Options:
  44. -u, --udeb scan for udebs.
  45. -a, --arch <arch> architecture to scan for.
  46. -m, --multiversion allow multiple versions of a single package.
  47. -h, --help show this help message.
  48. --version show the version.
  49. "), $0;
  50. }
  51. sub load_override
  52. {
  53. my $override = shift;
  54. my $override_fh = new IO::File $override, 'r' or
  55. die sprintf(_g("Couldn't open override file %s: %s"), $override, $!)."\n";
  56. while (<$override_fh>) {
  57. s/\#.*//;
  58. s/\s+$//;
  59. next unless $_;
  60. my ($p, $priority, $section, $maintainer) = split(/\s+/, $_, 4);
  61. next unless defined($packages{$p});
  62. for my $package (@{$packages{$p}}) {
  63. if ($maintainer) {
  64. if ($maintainer =~ m/(.+?)\s*=\>\s*(.+)/) {
  65. my $oldmaint = $1;
  66. my $newmaint = $2;
  67. my $debmaint = $$package{Maintainer};
  68. if (!grep($debmaint eq $_, split(m:\s*//\s*:, $oldmaint))) {
  69. push(@changedmaint,
  70. " $p (package says $$package{Maintainer}, not $oldmaint)\n");
  71. } else {
  72. $$package{Maintainer} = $newmaint;
  73. }
  74. } elsif ($$package{Maintainer} eq $maintainer) {
  75. push(@samemaint, " $p ($maintainer)\n");
  76. } else {
  77. printf(STDERR _g(" * Unconditional maintainer override for %s *")."\n", $p) || die $!;
  78. $$package{Maintainer} = $maintainer;
  79. }
  80. }
  81. $$package{Priority} = $priority;
  82. $$package{Section} = $section;
  83. }
  84. $overridden{$p} = 1;
  85. }
  86. close($override_fh);
  87. }
  88. usage() and exit 1 if not $result;
  89. if (not @ARGV >= 1 && @ARGV <= 3) {
  90. warn _g("1 to 3 args expected\n");
  91. usage();
  92. exit 1;
  93. }
  94. my $udeb = $options{udeb};
  95. my $arch = $options{arch};
  96. my $ext = $options{udeb} ? 'udeb' : 'deb';
  97. my @find_args;
  98. if ($options{arch}) {
  99. @find_args = ('(','-name',"*_all.$ext",'-o','-name',"*_${arch}.$ext",')',);
  100. }
  101. else {
  102. @find_args = ('-name',"*.$ext");
  103. }
  104. push @find_args, '-follow';
  105. #push @ARGV, undef if @ARGV < 2;
  106. #push @ARGV, '' if @ARGV < 3;
  107. my ($binarydir, $override, $pathprefix) = @ARGV;
  108. -d $binarydir or die sprintf(_g("Binary dir %s not found"),
  109. $binarydir)."\n";
  110. defined $override and -e $override or
  111. die sprintf(_g("Override file %s not found"), $override)."\n";
  112. $pathprefix = '' if not defined $pathprefix;
  113. our %vercache;
  114. sub vercmp {
  115. my ($a,$b)=@_;
  116. return $vercache{$a}{$b} if exists $vercache{$a}{$b};
  117. system('dpkg','--compare-versions',$a,'le',$b);
  118. $vercache{$a}{$b}=$?;
  119. return $?;
  120. }
  121. my $find_h = new IO::Handle;
  122. open($find_h,'-|','find',"$binarydir/",@find_args,'-print')
  123. or die sprintf(_g("Couldn't open %s for reading: %s"),
  124. $binarydir, $!)."\n";
  125. FILE:
  126. while (<$find_h>) {
  127. chomp;
  128. my $fn = $_;
  129. my $control = `dpkg-deb -I $fn control`;
  130. if ($control eq "") {
  131. warn sprintf(_g("Couldn't call dpkg-deb on %s: %s, skipping package"), $fn, $!)."\n";
  132. next;
  133. }
  134. if ($?) {
  135. warn sprintf(_g("\`dpkg-deb -I %s control' exited with %d, skipping package"), $fn, $?)."\n";
  136. next;
  137. }
  138. my %tv = ();
  139. my $temp = $control;
  140. while ($temp =~ s/^\n*(\S+):[ \t]*(.*(\n[ \t].*)*)\n//) {
  141. my ($key,$value)= (lc $1,$2);
  142. if (defined($kmap{$key})) { $key= $kmap{$key}; }
  143. if (defined($field_case{$key})) { $key= $field_case{$key}; }
  144. $value =~ s/\s+$//;
  145. $tv{$key}= $value;
  146. }
  147. $temp =~ /^\n*$/
  148. or die sprintf(_g("Unprocessed text from %s control file; info:\n%s / %s\n"), $fn, $control, $temp);
  149. defined($tv{'Package'})
  150. or die sprintf(_g("No Package field in control file of %s"), $fn)."\n";
  151. my $p= $tv{'Package'}; delete $tv{'Package'};
  152. if (defined($packages{$p}) and not $options{multiversion}) {
  153. foreach (@{$packages{$p}}) {
  154. if (&vercmp($tv{'Version'}, $_->{'Version'})) {
  155. printf(STDERR _g(
  156. " ! Package %s (filename %s) is repeat but newer version;\n".
  157. " used that one and ignored data from %s !\n"), $p, $fn, $_->{Filename})
  158. || die $!;
  159. $packages{$p} = [];
  160. } else {
  161. printf(STDERR _g(
  162. " ! Package %s (filename %s) is repeat;\n".
  163. " ignored that one and using data from %s !\n"), $p, $fn, $_->{Filename})
  164. or die $!;
  165. next FILE;
  166. }
  167. }
  168. }
  169. printf(STDERR _g(" ! Package %s (filename %s) has Filename field!\n"), $p, $fn) || die $!
  170. if defined($tv{'Filename'});
  171. $tv{'Filename'}= "$pathprefix$fn";
  172. open(C,"md5sum <$fn |") || die "$fn $!";
  173. chop($_=<C>); close(C); $? and die sprintf(_g("\`md5sum < %s' exited with %d"), $fn, $?)."\n";
  174. /^([0-9a-f]{32})\s*-?\s*$/ or die sprintf(_g("Strange text from \`md5sum < %s': \`%s'"), $fn, $_)."\n";
  175. $tv{'MD5sum'}= $1;
  176. my @stat= stat($fn) or die sprintf(_g("Couldn't stat %s: %s"), $fn, $!)."\n";
  177. $stat[7] or die sprintf(_g("%s is empty"), $fn)."\n";
  178. $tv{'Size'}= $stat[7];
  179. if (defined $tv{Revision} and length($tv{Revision})) {
  180. $tv{Version}.= '-'.$tv{Revision};
  181. delete $tv{Revision};
  182. }
  183. push @{$packages{$p}}, {%tv};
  184. }
  185. close($find_h);
  186. select(STDERR); $= = 1000; select(STDOUT);
  187. sub writelist {
  188. my $title= shift(@_);
  189. return unless @_;
  190. print(STDERR " $title\n") || die $!;
  191. my $packages= join(' ',sort @_);
  192. format STDERR =
  193. ^<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<
  194. $packages
  195. .
  196. while (length($packages)) { write(STDERR) || die $!; }
  197. print(STDERR "\n") || die $!;
  198. }
  199. load_override($override) if defined $override;
  200. my @missingover=();
  201. my $records_written = 0;
  202. for my $p (sort keys %packages) {
  203. if (not defined($overridden{$p})) {
  204. push(@missingover,$p);
  205. }
  206. for my $package (@{$packages{$p}}) {
  207. my $record= "Package: $p\n";
  208. for my $key (@fieldpri) {
  209. next unless defined $$package{$key};
  210. $record .= "$key: $$package{$key}\n";
  211. }
  212. $record .= "\n";
  213. $records_written++;
  214. print(STDOUT $record) or die sprintf(_g("Failed when writing stdout: %s"), $!)."\n";
  215. }
  216. }
  217. close(STDOUT) or die sprintf(_g("Couldn't close stdout: %s"), $!)."\n";
  218. my @spuriousover= grep(!defined($packages{$_}),sort keys %overridden);
  219. &writelist(_g("** Packages in archive but missing from override file: **"),
  220. @missingover);
  221. if (@changedmaint) {
  222. print(STDERR
  223. _g(" ++ Packages in override file with incorrect old maintainer value: ++")."\n",
  224. @changedmaint,
  225. "\n") || die $!;
  226. }
  227. if (@samemaint) {
  228. print(STDERR
  229. _g(" -- Packages specifying same maintainer as override file: --")."\n",
  230. @samemaint,
  231. "\n") || die $!;
  232. }
  233. if (@spuriousover) {
  234. print(STDERR
  235. _g(" -- Packages in override file but not in archive: --")."\n",
  236. @spuriousover,
  237. "\n") || die $!;
  238. }
  239. printf(STDERR _g(" Wrote %s entries to output Packages file.")."\n", $records_written) || die $!;