#! /usr/bin/perl # $OpenBSD: find-plist-issues,v 1.10 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 issues apparent # in packing-lists use strict; use warnings; use File::Spec; use File::Path; use File::Basename; use OpenBSD::PackageInfo; use OpenBSD::PackingList; use OpenBSD::Mtree; use OpenBSD::Getopt; use OpenBSD::State; use OpenBSD::PkgCfl; package OpenBSD::PackingElement; use OpenBSD::PkgSpec; sub register { } sub fix { my ($self, $l) = @_; if ($self->{def} eq 'def') { my @m = OpenBSD::PkgSpec->new($self->{pattern})->match_ref($l); if (@m > 0) { $self->{def} = $m[0]; } else { $self->{def} = $self->{pattern}; } } } sub check_common_dirs { } package OpenBSD::PackingElement::FileBase; use File::Basename; sub register_dir { my ($self, $d, $h) = @_; return if defined $h->{$d}; $h->{$d} = 1; $self->register_dir(dirname($d), $h); } sub register { my ($self, $all_conflict, $all_deps, $pkgname, $avail) = @_; my $file= File::Spec->canonpath($self->fullname()); unless (defined $all_conflict->{$file}) { $all_conflict->{$file} = []; } push @{$all_conflict->{$file}}, $pkgname; } sub check_common_dirs { my ($item, $t) = @_; my $d = File::Spec->canonpath($item->fullname()); $item->register_dir(dirname($d), $t->{need_dirs}); } package OpenBSD::PackingElement::DirlikeObject; sub check_common_dirs { my ($item, $t) = @_; my $d = File::Spec->canonpath($item->fullname()); $t->{dirs}->{$d} = 1; } package OpenBSD::PackingElement::Depend; sub register { my ($self, $all_conflict, $all_deps, $pkgname, $avail) = @_; if (defined $self->{def}) { unless (defined $all_deps->{$pkgname}) { $all_deps->{$pkgname} = []; } $self->fix($avail); push @{$all_deps->{$pkgname}}, $self->{def}; } } sub check_common_dirs { my ($item, $t) = @_; $item->fix($t->{avail}); $t->{deps}->{$item->{def}} = 1; } package OpenBSD::PackingElement::PkgDep; sub check_common_dirs { my ($item, $t) = @_; $t->{deps}->{$item->{name}} = 1; } package OpenBSD::PackingElement::Wantlib; sub check_common_dirs { } package main; my $cache = {}; my $cache2 = {}; my @available = (); my $conflicts_cache = {}; sub find_a_conflict { my ($conflicts, $deps, $pkg, $pkg2) = @_; return 0 if $pkg2 eq $pkg; my $h = "$pkg/$pkg2"; if (defined $conflicts_cache->{$h}) { return $conflicts_cache->{$h}; } if (defined $conflicts->{$pkg} && $conflicts->{$pkg}->conflicts_with($pkg2)) { $conflicts_cache->{$h} = 1; return 1; } if (defined $deps->{$pkg}) { for my $dep (@{$deps->{$pkg}}) { if (find_a_conflict($conflicts, $deps, $dep, $pkg2)) { $conflicts_cache->{$h} = 1; return 1; } } } if (defined $deps->{$pkg2}) { for my $dep (@{$deps->{$pkg2}}) { if (find_a_conflict($conflicts, $deps, $pkg, $dep)) { $conflicts_cache->{$h} = 1; return 1; } } } $conflicts_cache->{$h} = 0; return 0; } sub compute_conflicts { my ($h, $conflicts, $deps) = @_; while (my ($key, $l) = each %$h) { my %s = map {($_, 1)} @$l; @$l = sort keys %s; if (@$l > 1) { my $hv = join(',', @$l); if (!defined $cache->{$hv}) { # 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 (!(find_a_conflict($conflicts, $deps, $pkg, $pkg2) || find_a_conflict($conflicts, $deps, $pkg2, $pkg))) { $keepit = 1; last; } } if ($keepit) { push(@$l2, $pkg); } } $cache->{$hv} = $l2; } my $result = $cache->{$hv}; if (@$result != 0) { my $newkey = join(',', @$result); if (@$result == 1) { $newkey.="-> was ".join(',', @$l); } $cache2->{$newkey} = [] unless defined($cache2->{$newkey}); push(@{$cache2->{$newkey}}, $key); } } } } sub analyze_dirs { my ($plist, $db) = @_; my $pkgname = $plist->pkgname(); $db->{$pkgname} = { pkgname => $pkgname, missing_deps => {}, dirs => {}, need_dirs => {}, deps => {}, problems => {}, avail => \@available } unless defined $db->{$pkgname}; my $t = $db->{$pkgname}; $plist->check_common_dirs($t) } sub parent_has_dir { my ($db, $t, $dir) = @_; for my $dep (keys %{$t->{deps}}) { if (!defined $db->{$dep}) { if (!defined $t->{missing_deps}->{$dep}) { print $t->{pkgname}, ": $dep not found\n"; $t->{missing_deps}->{$dep} = 1; } next; } if ($db->{$dep}->{dirs}->{$dir} || parent_has_dir($db, $db->{$dep}, $dir)) { $t->{dirs}->{$dir} = 1; return 1; } } return 0; } sub parent_has_dir_issue { my ($db, $t, $dir) = @_; for my $dep (keys %{$t->{deps}}) { next if !defined $db->{$dep}; if ($db->{$dep}->{problems}->{$dir}) { return 1; } } return 0; } sub build_dir_results { my ($db, $mtree) = @_; for my $pkgname (keys %$db) { my $t = $db->{$pkgname}; for my $dir (keys(%{$t->{need_dirs}})) { next if $t->{dirs}->{$dir}; next if $mtree->{$dir}; next if parent_has_dir($db, $t, $dir); $t->{problems}->{$dir} = 1; } } } sub show_dir_results { my ($db, $mtree) = @_; # first reverse the results my $dir_db = {}; for my $pkgname (keys %$db) { my @l=(); my $t = $db->{$pkgname}; for my $dir (keys %{$t->{problems}}) { next if parent_has_dir_issue($db, $t, $dir); $dir_db->{$dir} = [] if !defined $dir_db->{$dir}; push(@{$dir_db->{$dir}}, $pkgname); } } # and print the resulting table: for my $dir (sort keys %$dir_db) { print $dir, ": ", join(',', sort @{$dir_db->{$dir}}), "\n"; } } my $filehash={}; my %dirhash=(); my $conflicts={}; my $dephash={}; my $db = {}; my $mtree = {}; our ($opt_d, $opt_v, $opt_C, $opt_D, $opt_f); sub handle_plist { my $plist = shift; print $plist->pkgname(), "\n" if $opt_v; $plist->forget(); if ($opt_C) { $conflicts->{$plist->pkgname()} = OpenBSD::PkgCfl->make_conflict_list($plist); $plist->register($filehash, $dephash, $plist->pkgname(), \@available); } if ($opt_D) { analyze_dirs($plist, $db); } } sub handle_file { my $filename = shift; my $plist = OpenBSD::PackingList->fromfile($filename); if (!defined $plist) { print STDERR "Error reading $filename\n"; return; } handle_plist($plist); } my $ui = OpenBSD::State->new('find-plist-issues'); $ui->usage_is('[-vCDf] [-d plist_dir] [pkgname ...]'); $ui->do_options(sub { getopts('d:vCDf'); }); OpenBSD::Mtree::parse($mtree, '/usr/local', '/etc/mtree/BSD.local.dist'); OpenBSD::Mtree::parse($mtree, '/', '/etc/mtree/4.4BSD.dist'); OpenBSD::Mtree::parse($mtree, '/', '/etc/mtree/BSD.x11.dist'); $mtree->{'/usr/local/lib/X11'} = 1; $mtree->{'/usr/local/include/X11'} = 1; $mtree->{'/usr/local/lib/X11/app-defaults'} = 1; print "Scanning packages\n" if $opt_v; print "-----------------\n" if $opt_v; if ($opt_d) { for my $dirname (split(/:/, $opt_d)) { opendir(my $dir, $dirname) or next; push(@available, grep { $_ ne '.' && $_ ne '..' } readdir($dir)); closedir($dir); } for my $dirname (split(/:/, $opt_d)) { if (opendir(my $dir, $dirname)) { while (my $pkgname = readdir($dir)) { next if $pkgname eq '.' or $pkgname eq '..'; handle_file("$dirname/$pkgname"); } closedir($dir); } else { print STDERR "No such dir: $dirname\n"; } } } elsif (@ARGV==0) { @ARGV=(<*.tgz>); } my @pkgs = @ARGV; push(@available, map { s,.*/,,; s/\.tgz$//; } @pkgs); for my $pkgname (@ARGV) { print STDERR "$pkgname\n"; if ($opt_f) { handle_file($pkgname); } else { my $plist = $ui->repo->grabPlist($pkgname); next unless $plist; handle_plist($plist); } } print "File problems:\n"; print "-------------\n"; if ($opt_C) { compute_conflicts($filehash, $conflicts, $dephash); for my $cfl (sort keys %$cache2) { print "$cfl\n"; for my $f (sort @{$cache2->{$cfl}}) { print "\t$f\n"; } } } if ($opt_D) { build_dir_results($db, $mtree); show_dir_results($db); }