mirror of the now-defunct rocklinux.org
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

235 lines
6.0 KiB

  1. #!/usr/bin/perl -w
  2. use strict;
  3. use English;
  4. my (@pkg, %opr, %pri, %dep, %rep, %rdp, %ign, %lop, %deldeps, %bas);
  5. my @useddeps; # deps used for visualization of cross dependencies
  6. my %pkg_redone; # packages built in stages 6-8
  7. my $config = "";
  8. while ($_ = shift @ARGV) {
  9. if ( $_ eq "-cfg" ) {
  10. $config = shift @ARGV;
  11. } elsif ( /^-/ ) {
  12. print "\n";
  13. print "Usgage: scripts/Check-Deps-2 [ -cfg config-name ]\n";
  14. print "\n";
  15. print "This script does some dependency checking and suggests\n";
  16. print "package priority reorderings (if neccessary).\n";
  17. print "\n";
  18. print "The data from scripts/dep_db.txt and scripts/dep_fixes.txt\n";
  19. print "is used for the dependency analysis.\n";
  20. print "\n";
  21. exit 1;
  22. } else {
  23. $ign{$_} = 1;
  24. }
  25. }
  26. print "Reading package priorities ...\n";
  27. if ( $config eq "" ) {
  28. open(F, "./scripts/Create-PkgList |") || die $!;
  29. } else {
  30. open(F, "< config/$config/packages") || die $!;
  31. }
  32. while (<F>) {
  33. @_ = split /\s+/;
  34. next if $_[1] =~ /[1234]/ or $_[1] !~ /5/;
  35. $pkg_redone{$_[4]} = 1 if $_[1] =~ /[678]/;
  36. next if defined $ign{$_[3]} || defined $ign{$_[4]};
  37. my ($b, $p) = ($_[4], $_[4]);
  38. ($b, $p) = ($1, $2) if $_[4] =~ /(.*)=(.*)/;
  39. $opr{$p} = $_[2];
  40. $pri{$p} = $_[2];
  41. $rep{$p} = $_[3];
  42. $bas{$p} = $b;
  43. $pkg[$#pkg+1] = $p;
  44. }
  45. close F;
  46. print "Reading dependency fixes ...\n";
  47. open(F, "scripts/dep_fixes.txt") or die $!;
  48. while (<F>) {
  49. chomp;
  50. if (/^([^#\s]\S*)\s+del\s+(.*)$/) {
  51. my ($p, $l) = ($1, $2);
  52. $deldeps{$p}{$_} = 1 foreach (split /\s+/, $l);
  53. next;
  54. }
  55. if (/^([^#\s]\S*)\s+add\s+(.*)$/) {
  56. my ($p, $l) = ($1, $2);
  57. foreach ( split /\s+/, $l ) {
  58. push @{$dep{$p}}, $_;
  59. push @{$rdp{$_}}, $p;
  60. }
  61. next;
  62. }
  63. }
  64. close F;
  65. print "Reading package dependencies ...\n";
  66. open(F, "scripts/dep_db.txt") || die $!;
  67. while (<F>) {
  68. chomp;
  69. if ( ! /^(\S+): \d+ \d+ (.*)$/ ) {
  70. print "Format Error: $_\n";
  71. exit 1;
  72. }
  73. my ($p, $l) = ($1, $2);
  74. next if defined $pkg_redone{$p};
  75. foreach ( split /\s+/, $l ) {
  76. next if defined $deldeps{$p}{$_};
  77. push @{$dep{$p}}, $_;
  78. push @{$rdp{$_}}, $p;
  79. }
  80. }
  81. close F;
  82. sub count_errs($) {
  83. my $package = $_[0];
  84. my $dependency;
  85. my $errors = 0;
  86. foreach $dependency (@{$dep{$package}}) {
  87. next unless defined $pri{$dependency};
  88. $errors++ if $pri{$package} < $pri{$dependency};
  89. }
  90. foreach $dependency (@{$rdp{$package}}) {
  91. next unless defined $pri{$dependency};
  92. $errors++ if $pri{$package} > $pri{$dependency};
  93. }
  94. return $errors;
  95. }
  96. my ($iteration, $package, $dependency, $a, $b);
  97. my $did_something=0;
  98. print "\nLoop Old/New Errors Package Dependency\n".
  99. "------------------------------------------------------------------------\n";
  100. for $iteration (1..99) {
  101. my $looplog = '';
  102. foreach $package (@pkg) {
  103. foreach $dependency (@{$dep{$package}}) {
  104. next unless defined $pri{$dependency};
  105. if ( $pri{$package} < $pri{$dependency} ) {
  106. $a = count_errs($package) + count_errs($dependency);
  107. $_ = $pri{$dependency};
  108. $pri{$dependency} = $pri{$package};
  109. $pri{$package} = $_;
  110. $b = count_errs($package) + count_errs($dependency);
  111. $looplog.="[$package,$dependency]";
  112. $_ = sprintf "[%02d] %-7d %-7d %-25s %s\n",
  113. $iteration, $a, $b, $pri{$dependency}." ".$package,
  114. $pri{$package}." ".$dependency;
  115. $useddeps[$iteration]{$package}{$dependency} = 1;
  116. s/ / . /g; s/\. /.. /g; s/\. /.. /g;
  117. s/\. (\s*)\./..$1./g; s/\. (\s*)\./..$1./g;
  118. print; $did_something=1;
  119. }
  120. }
  121. }
  122. last if $looplog eq "";
  123. if (defined $lop{$looplog}) {
  124. my %crossdeps;
  125. print "[XX] Detected endless-loop ".
  126. "(cross-dependency) -> Aborting now.\n";
  127. print "[XX] Debug graph printed to dependencies.dot.\n";
  128. foreach my $i ($lop{$looplog} .. $iteration) {
  129. foreach my $p (keys %{$useddeps[$i]}) {
  130. foreach my $d (keys %{$useddeps[$i]{$p}}) {
  131. $crossdeps{$d}{$p} = 1;
  132. }
  133. }
  134. }
  135. open(F, ">dependencies.dot") || die $!;
  136. print F "# run this thru e.g. 'dot -Tps dependencies.dot -o dependencies.ps'\n";
  137. print F "digraph \"Cross-Dependencies Graph\" {\n";
  138. print F " Package_X -> Has_X_in_Dep_List;\n";
  139. foreach my $p (sort keys %crossdeps) {
  140. foreach my $d (sort keys %{$crossdeps{$p}}) {
  141. my $p_ = $p; $p_ =~ s/[^a-z0-9]/_/g;
  142. my $d_ = $d; $d_ =~ s/[^a-z0-9]/_/g;
  143. print F "\t$p_ -> $d_;\n";
  144. }
  145. }
  146. print F "}\n";
  147. close F;
  148. open(F, ">dependencies.dbg") || die $!;
  149. foreach my $p (sort keys %crossdeps) {
  150. foreach my $d (sort keys %{$crossdeps{$p}}) {
  151. print F "$p $d\n";
  152. }
  153. }
  154. close F;
  155. system("dot -Tps dependencies.dot -o dependencies.ps");
  156. system("convert dependencies.ps dependencies.png");
  157. last;
  158. }
  159. $lop{$looplog} = $iteration;
  160. }
  161. sub patchfile($$$$) {
  162. my ($tmpfile, $descfile, $re1, $re2) = @_;
  163. if ( ! open(IN, $descfile) )
  164. { print "ERROR: $descfile: $!\n"; return; }
  165. if ( ! open(OUT, ">$tmpfile") )
  166. { print "ERROR: $descfile: $!\n"; close IN; return; }
  167. $did_something = 0;
  168. while (<IN>) {
  169. $did_something = 1 if eval "s/$re1/$re2/i";
  170. print OUT;
  171. }
  172. close IN; close OUT;
  173. if (not $did_something) {
  174. print "ERROR: Can't patch $descfile!\n";
  175. print "ERROR: Regex was s/$re1/$re2/\n";
  176. }
  177. system("diff -U 0 ./$descfile $tmpfile >> dependencies.patch");
  178. }
  179. sub setpri($$$$$$) {
  180. my ($pri, $opr, $rep, $bas, $package, $tmpfile) = @_;
  181. if ($bas eq "cpan") {
  182. my $r = $package; $r =~ s/^cpan-//g; $r =~ s/-/(-|::)/g;
  183. patchfile($tmpfile, "package/import/cpan/hosted_cpan.txt",
  184. "$opr ($r)", "$pri \$1");
  185. patchfile($tmpfile, "package/import/cpan/hosted_cpan.cfg",
  186. "(pkgfork cpan $package .*) $opr;", "\$1 $pri;");
  187. return;
  188. }
  189. patchfile($tmpfile, "package/$rep/$bas/$package.desc",
  190. "(\\[P\\] . \\S+) $opr", "\$1 $pri");
  191. }
  192. if ( $did_something ) {
  193. print "\nCreate dependencies.patch ...\n";
  194. my $tmpfile = `mktemp`; chomp $tmpfile;
  195. unlink "dependencies.patch";
  196. foreach $package (@pkg) {
  197. if ($pri{$package} != $opr{$package}) {
  198. print "Setting priority $pri{$package} on package $rep{$package}/$bas{$package}=$package.\n";
  199. setpri($pri{$package}, $opr{$package}, $rep{$package}, $bas{$package}, $package, $tmpfile);
  200. }
  201. }
  202. unlink $tmpfile;
  203. print "Done. Please check moves manually bofore applying the patch.\n";
  204. } else {
  205. print "No unresolved dependencies found.\n";
  206. }