123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277 |
- #!/usr/bin/perl
- # $OpenBSD: find-all-conflicts,v 1.20 2010/06/30 11:11:19 espie Exp $
- # Copyright (c) 2000-2005
- # Marc Espie. All rights reserved.
- # Redistribution and use in source and binary forms, with or without
- # modification, are permitted provided that the following conditions
- # are met:
- # 1. Redistributions of code must retain the above copyright
- # notice, this list of conditions and the following disclaimer.
- # 2. Neither the name of OpenBSD nor the names of its contributors
- # may be used to endorse or promote products derived from this software
- # without specific prior written permission.
- #
- # THIS SOFTWARE IS PROVIDED BY ITS AUTHOR AND THE OpenBSD project ``AS IS'' AND
- # ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
- # IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
- # ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
- # FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
- # DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
- # OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
- # HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
- # LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
- # OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
- # SUCH DAMAGE.
- # check all packages in the current directory, and report conflicts which
- # are not apparent in @pkgcfl.
- use strict;
- use File::Spec;
- use File::Path;
- use OpenBSD::PackageInfo;
- use OpenBSD::PackingList;
- use OpenBSD::AddCreateDelete;
- use OpenBSD::PkgCfl;
- package OpenBSD::PackingElement;
- sub register
- {
- }
- package OpenBSD::PackingElement::FileBase;
- my $pkg_list = {};
- my $seen = {};
- sub register
- {
- my ($self, $all_conflict, $all_deps, $pkgname) = @_;
- my $file = File::Spec->canonpath($self->fullname);
- # build one single list for each pkgnames combination
- if (exists $all_conflict->{$file}) {
- $pkg_list->{$all_conflict->{$file}}->{$pkgname} ||=
- [@{$all_conflict->{$file}}, $pkgname ];
- $all_conflict->{$file} =
- $pkg_list->{$all_conflict->{$file}}->{$pkgname};
- } elsif (exists $seen->{$file}) {
- $pkg_list->{$seen->{$file}}->{$pkgname} ||=
- [ @{$seen->{$file}}, $pkgname ];
- $all_conflict->{$file} =
- $pkg_list->{$seen->{$file}}->{$pkgname};
- delete $seen->{$file};
- } else {
- $pkg_list->{$pkgname} ||= [$pkgname];
- $seen->{$file} = $pkg_list->{$pkgname};
- }
- }
- package OpenBSD::PackingElement::Depend;
- sub register
- {
- my ($self, $all_conflict, $all_deps, $pkgname) = @_;
- if (defined $self->{def}) {
- push @{$all_deps->{$pkgname}}, $self->{def};
- }
- }
- package main;
- my $cache = {};
- my $cache2 = {};
- my $cache3 = {};
- my $cache4 = {};
- sub direct_conflict
- {
- my ($conflicts, $pkg, $pkg2) = @_;
- return $cache3->{$pkg}{$pkg2} //= $conflicts->{$pkg}->conflicts_with($pkg2);
- }
- sub has_a_conflict
- {
- my ($conflicts, $deps, $pkg, $pkg2) = @_;
- return $cache4->{$pkg}{$pkg2} //= find_a_conflict($conflicts, $deps, $pkg, $pkg2);
- }
- sub find_a_conflict
- {
- my ($conflicts, $deps, $pkg, $pkg2) = @_;
- return 0 if $pkg eq $pkg2;
-
- if (defined $conflicts->{$pkg} && direct_conflict($conflicts, $pkg, $pkg2)) {
- return 1;
- }
- if (defined $deps->{$pkg}) {
- for my $dep (@{$deps->{$pkg}}) {
- if (has_a_conflict($conflicts, $deps, $dep, $pkg2)) {
- return 1;
- }
- }
- }
- if (defined $deps->{$pkg2}) {
- for my $dep (@{$deps->{$pkg2}}) {
- if (has_a_conflict($conflicts, $deps, $pkg, $dep)) {
- return 1;
- }
- }
- }
- return 0;
- }
- sub compute_true_conflicts
- {
- my ($l, $conflicts, $deps) = @_;
- # create a list of unconflicting packages.
- my $l2 = [];
- for my $pkg (@$l) {
- my $keepit = 0;
- for my $pkg2 (@$l) {
- next if $pkg eq $pkg2;
- if (!(has_a_conflict($conflicts, $deps, $pkg, $pkg2) ||
- has_a_conflict($conflicts, $deps, $pkg2, $pkg))) {
- $keepit = 1;
- last;
- }
- }
- if ($keepit) {
- push(@$l2, $pkg);
- }
- }
- return $l2;
- }
- sub compute_problems
- {
- my ($ui, $h, $conflicts, $deps) = @_;
- my $c = {};
- my $c2 = {};
- my $total = scalar(keys %$h);
- my $i =0;
- while (my ($key, $l) = each %$h) {
- $ui->progress->show(++$i, $total);
- if (!defined $c->{$l}) {
- my %s = map {($_, 1)} @$l;
- $c->{$l} = [sort keys %s];
- $c2->{$l} = join(',', @{$c->{$l}});
- }
- my $hv = $c2->{$l};
- $l = $c->{$l};
- next if @$l == 1;
- if (!defined $cache->{$hv}) {
- $cache->{$hv} = compute_true_conflicts($l, $conflicts, $deps);
- }
- my $result = $cache->{$hv};
- if (@$result != 0) {
- my $newkey = join(',', @$result);
- if (@$result == 1) {
- $newkey.="-> was ".join(',', @$l);
- }
- push(@{$cache2->{$newkey}}, $key);
- }
- }
- }
- my $filehash={};
- my %dirhash=();
- my $conflicts={};
- my $dephash={};
- our ($opt_d, $opt_p, $opt_v);
- sub handle_plist
- {
- my ($ui, $filename, $plist) = @_;
- if (!defined $plist) {
- $ui->errsay("Error reading #1", $filename);
- return;
- }
- $ui->say("#1 -> #2", $filename, $plist->pkgname) if $ui->verbose;
- $plist->forget;
- $conflicts->{$plist->pkgname} =
- OpenBSD::PkgCfl->make_conflict_list($plist);
- $plist->register($filehash, $dephash, $plist->pkgname);
- }
- sub handle_file
- {
- my ($ui, $filename) = @_;
- my $plist = OpenBSD::PackingList->fromfile($filename);
- handle_plist($ui, $filename, $plist);
- }
- sub handle_portsdir
- {
- my ($ui, $dir) = @_;
- my $make = $ENV{MAKE} || 'make';
- open(my $input, "cd $dir && $make print-plist-all |");
- my $done = 0;
- while (!$done) {
- my $plist = OpenBSD::PackingList->read($input, sub {
- my ($fh, $cont) = @_;
- local $_;
- while (<$fh>) {
- return if m/^\=\=\=\> /o;
- next unless m/^\@(?:cwd|name|info|man|file|lib|shell|bin|conflict|comment\s+subdir\=)\b/o || !m/^\@/o;
- &$cont($_);
- }
- $done = 1;
- });
- if (defined $plist && $plist->pkgname()) {
- handle_plist($ui, $dir, $plist);
- $ui->progress->working(10);
- }
- }
- }
- my $ui = OpenBSD::AddCreateDelete::State->new('find-all-conflicts');
- $ui->handle_options('d:p:', '[-v] [-d plist_dir] [-p ports_dir] [pkgname ...]');
- $ui->progress->set_header("Scanning");
- $opt_d = $ui->opt('d');
- $opt_p = $ui->opt('p');
- if ($opt_d) {
- opendir(my $dir, $opt_d);
- my @l = readdir $dir;
- closedir($dir);
- $ui->progress->for_list("Scanning", \@l,
- sub {
- my $pkgname = shift;
- return if $pkgname eq '.' or $pkgname eq '..';
- handle_file($ui, "$opt_d/$pkgname");
- });
- } elsif ($opt_p) {
- handle_portsdir($ui, $opt_p);
- } elsif (@ARGV==0) {
- @ARGV=(<*.tgz>);
- }
- $ui->progress->for_list("Scanning", \@ARGV,
- sub {
- my $pkgname = shift;
- my $true_package = $ui->repo->find($pkgname);
- return unless $true_package;
- my $dir = $true_package->info;
- $true_package->close;
- handle_file($ui, $dir.CONTENTS);
- rmtree($dir);
- });
- $ui->progress->next;
- $ui->progress->set_header("File problems");
- compute_problems($ui, $filehash, $conflicts, $dephash);
- for my $cfl (sort keys %$cache2) {
- $ui->say("#1", $cfl);
- for my $f (sort @{$cache2->{$cfl}}) {
- $ui->say("\t#1", $f);
- }
- }
|