# ex:ts=8 sw=4: # $OpenBSD: CollisionReport.pm,v 1.15 2007/06/20 13:44:39 espie Exp $ # # Copyright (c) 2003-2006 Marc Espie # # Permission to use, copy, modify, and distribute this software for any # purpose with or without fee is hereby granted, provided that the above # copyright notice and this permission notice appear in all copies. # # THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES # WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF # MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR # ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES # WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN # ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF # OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. use strict; use warnings; package OpenBSD::CollisionReport; use OpenBSD::PackingList; use OpenBSD::PackageInfo; use OpenBSD::Vstat; sub collision_report($$) { my ($list, $state) = @_; if ($state->{forced}->{removecollisions}) { for my $f (@$list) { print "rm ", $f->fullname, "\n"; unlink($f->fullname); } return; } my %todo = map {($_->fullname, $_->{md5})} @$list; my $bypkg = {}; my $clueless_bat = 0; my $clueless_bat2 = 0; print "Collision: the following files already exist\n"; for my $name (keys %todo) { my $p = OpenBSD::Vstat::vexists $name; if (ref $p) { my $pkg = $$p; $bypkg->{$pkg} = [] unless defined $bypkg->{$pkg}; push(@{$bypkg->{$pkg}}, $name); delete $todo{$name}; } } if (%todo) { BIGLOOP: { for my $pkg (installed_packages()) { print "Looking for collisions in $pkg\n" if $state->{verbose}; my $plist = OpenBSD::PackingList->from_installation($pkg, \&OpenBSD::PackingList::FilesOnly); next if !defined $plist; for my $item (@{$plist->{items}}) { next unless $item->IsFile; my $name = $item->fullname; if (defined $todo{$name}) { $bypkg->{$pkg} = [] unless defined $bypkg->{$pkg}; push(@{$bypkg->{$pkg}}, $name); delete $todo{$name}; last BIGLOOP if !%todo; } } } } } for my $pkg (sort keys %$bypkg) { for my $item (sort @{$bypkg->{$pkg}}) { print "\t$item ($pkg)\n"; } if ($pkg =~ m/^(?:partial\-|borked\.\d+$)/o) { $clueless_bat = $pkg; } if ($pkg =~ m/^\.libs\d*-*$/o) { $clueless_bat2 = $pkg; } } if (%todo) { require OpenBSD::md5; my $destdir = $state->{destdir}; for my $item (sort keys %todo) { if (defined $todo{$item}) { my $md5 = OpenBSD::md5::fromfile($destdir.$item); if ($md5 eq $todo{$item}) { print "\t$item (same md5)\n"; } else { print "\t$item (different md5)\n"; } } else { print "\t$item\n"; } } } if ($clueless_bat) { print "The package name $clueless_bat suggests that a former installation\n"; print "of a similar package got interrupted. It is likely that\n"; print "\tpkg_delete $clueless_bat\n"; print "will solve the problem\n"; } if ($clueless_bat2) { print "The package name $clueless_bat2 suggests remaining libraries\n"; print "from a former package update. It is likely that\n"; print "\tpkg_delete $clueless_bat2\n"; print "will solve the problem\n"; } } 1;