#! /usr/bin/perl # ex:ts=8 sw=4: # $OpenBSD: PkgCheck.pm,v 1.45 2014/01/10 16:09:08 espie Exp $ # # Copyright (c) 2003-2014 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; use OpenBSD::AddCreateDelete; use OpenBSD::SharedLibs; package Installer::State; our @ISA = qw(OpenBSD::PkgAdd::State); sub new { my ($class, $cmd) = @_; my $state = $class->SUPER::new($cmd); $state->{localbase} = OpenBSD::Paths->localbase; return $state; } package Installer; our @ISA = qw(OpenBSD::PkgAdd); sub new { my ($class, $mystate) = @_; my $state = Installer::State->new("pkg_check"); $state->{v} = $mystate->{v}; $state->{interactive} = $mystate->{interactive}; $state->{destdir} = $mystate->{destdir}; bless { state => $state}, $class; } sub install { my ($self, $pkg) = @_; my $state = $self->{state}; push(@{$state->{setlist}}, $state->updateset->add_hints2($pkg)); $self->framework($state); return $state->{bad} != 0; } package OpenBSD::PackingElement; sub thorough_check { my ($self, $state) = @_; $self->basic_check($state); } sub basic_check { } sub find_dependencies { } package OpenBSD::PackingElement::FileBase; use File::Basename; sub basic_check { my ($self, $state) = @_; my $name = $state->{destdir}.$self->fullname; $state->{known}{dirname($name)}{basename($name)} = 1; if ($self->{symlink}) { if (!-l $name) { if (!-e $name) { $state->log("#1 should be a symlink but does not exist", $name); } else { $state->log("#1 is not a symlink", $name); } } else { if (readlink($name) ne $self->{symlink}) { $state->log("#1 should point to #2 but points to #3 instead", $name, $self->{symlink}, readlink($name)); } } return; } if (!-e $name) { if (-l $name) { $state->log("#1 points to non-existent #2", $name, readlink($name)); } else { $state->log("#1 should exist", $name); } } if (!-f _) { $state->log("#1 is not a file", $name); } if ($self->{link}) { my ($a, $b) = (stat _)[0, 1]; if (!-f $state->{destdir}.$self->{link}) { $state->log("#1 should link to non-existent #2", $name, $self->{link}); } else { my ($c, $d) = (stat _)[0, 1]; if (defined $a && defined $c) { if ($a != $c || $b != $d) { $state->log("#1 doesn't link to #2", $name, $self->{link}); } } } } } sub thorough_check { my ($self, $state) = @_; my $name = $state->{destdir}.$self->fullname; $self->basic_check($state); return if $self->{link} or $self->{symlink} or $self->{nochecksum}; if (!-r $name) { $state->log("can't read #1", $name); return; } if (!defined $self->{d}) { $state->log("no checksum for #1", $name); return; } my $d = $self->compute_digest($name, ref($self->{d})); if (!$d->equals($self->{d})) { $state->log("checksum for #1 does not match", $name); } } package OpenBSD::PackingElement::SpecialFile; sub basic_check { &OpenBSD::PackingElement::FileBase::basic_check; } sub thorough_check { &OpenBSD::PackingElement::FileBase::basic_check; } package OpenBSD::PackingElement::DirlikeObject; sub basic_check { my ($self, $state) = @_; my $name = $state->{destdir}.$self->fullname; $state->{known}{$name} //= {}; if (!-e $name) { $state->log("#1 should exist", $name); } if (!-d _) { $state->log("#1 is not a directory", $name); } } package OpenBSD::PackingElement::Sample; use File::Basename; sub basic_check { my ($self, $state) = @_; my $name = $state->{destdir}.$self->fullname; $state->{known}{dirname($name)}{basename($name)} = 1; } package OpenBSD::PackingElement::Sampledir; sub basic_check { my ($self, $state) = @_; my $name = $state->{destdir}.$self->fullname; $state->{known}{$name} //= {}; } package OpenBSD::PackingElement::Mandir; sub basic_check { my ($self, $state) = @_; $self->SUPER::basic_check($state); my $name = $state->{destdir}.$self->fullname; for my $file (OpenBSD::Paths::man_cruft()) { $state->{known}{$name}{$file} = 1; } } package OpenBSD::PackingElement::Fontdir; sub basic_check { my ($self, $state) = @_; $self->SUPER::basic_check($state); my $name = $state->{destdir}.$self->fullname; for my $i (qw(fonts.alias fonts.scale fonts.dir)) { $state->{known}{$name}{$i} = 1; } } package OpenBSD::PackingElement::Infodir; sub basic_check { my ($self, $state) = @_; $self->SUPER::basic_check($state); my $name = $state->{destdir}.$self->fullname; $state->{known}{$name}{'dir'} = 1; } package OpenBSD::PackingElement::Dependency; sub find_dependencies { my ($self, $state, $l, $checker) = @_; # several ways to failure if (!$self->spec->is_valid) { $state->log("invalid \@", $self->keyword, " ", $self->stringize); return; } my @deps = $self->spec->filter(@$l); if (@deps == 0) { $state->log("dependency #1 does not match any installed package", $self->stringize); return; } my $okay = 0; for my $i (@deps) { if ($checker->find($i)) { $okay = 1; } } if (!$okay) { $checker->not_found($deps[0]); } } package OpenBSD::PackingElement::Wantlib; sub find_dependencies { my ($self, $state, $l, $checker) = @_; my $r = OpenBSD::SharedLibs::lookup_libspec($state->{localbase}, $self->spec); if (defined $r && @$r != 0) { my $okay = 0; for my $lib (@$r) { my $i = $lib->origin; if ($i eq 'system') { $okay = 1; next; } if ($checker->find($i)) { $okay = 1; } } if (!$okay) { $checker->not_found($r->[0]->origin); } } else { $state->log("#1 not found", $self->stringize); } } package OpenBSD::PkgCheck::State; our @ISA = qw(OpenBSD::AddCreateDelete::State); use OpenBSD::Log; sub init { my $self = shift; $self->{l} = OpenBSD::Log->new($self); $self->SUPER::init; } sub log { my $self = shift; if (@_ == 0) { return $self->{l}; } else { $self->{l}->say(@_); } } sub safe { my ($self, $_) = @_; s/[^\w\d\s\+\-\.\>\<\=\/\;\:\,\(\)\[\]]/?/g; return $_; } sub handle_options { my $self = shift; $self->{no_exports} = 1; $self->SUPER::handle_options('fB:Iiq', '[-fIimnqvx] [-B pkg-destdir] [-D value]'); if ($self->opt('i')) { $self->{interactive} = 1; } elsif ($self->opt('I')) { $self->{interactive} = 0; } else { $self->{interactive} = -t STDIN; } $self->{force} = $self->opt('f'); $self->{quick} = $self->opt('q'); if (defined $self->opt('B')) { $self->{destdir} = $self->opt('B'); } if (defined $self->{destdir}) { $self->{destdir} .= '/'; } else { $self->{destdir} = ''; } } package OpenBSD::DependencyCheck; sub new { my ($class, $state, $name, $req) = @_; my $o = bless { not_yet => {}, possible => {}, others => {}, name => $name, req => $req }, $class; for my $pkg ($req->list) { $o->{not_yet}{$pkg} = 1; if ($state->{exists}{$pkg}) { $o->{possible}{$pkg} = 1; } else { $state->errsay("#1: bogus #2", $name, $o->string($state->safe($pkg))); } } return $o; } sub find { my ($self, $name) = @_; if ($self->{possible}{$name}) { delete $self->{not_yet}{$name}; return 1; } else { return 0; } } sub not_found { my ($self, $name) = @_; $self->{others}{$name} = 1; } sub ask_delete_deps { my ($self, $state, $l) = @_; if ($state->{force}) { $self->{req}->delete(@$l); } elsif ($state->{interactive}) { require OpenBSD::Interactive; if (OpenBSD::Interactive::confirm("Remove missing ". $state->safe($self->string(@$l)))) { $self->{req}->delete(@$l); } } } sub ask_add_deps { my ($self, $state, $l) = @_; if ($state->{force}) { $self->{req}->add(@$l); } elsif ($state->{interactive}) { require OpenBSD::Interactive; if (OpenBSD::Interactive::confirm("Add missing ". $self->string(@$l))) { $self->{req}->add(@$l); } } } sub adjust { my ($self, $state) = @_; if (keys %{$self->{not_yet}} > 0) { my @todo = sort keys %{$self->{not_yet}}; unless ($state->{subst}->value("weed_libs")) { @todo = grep {!/^\.libs/} @todo; } if (@todo != 0) { $state->errsay("#1 has too many #2", $self->{name}, $state->safe($self->string(@todo))); $self->ask_delete_deps($state, \@todo); } } if (keys %{$self->{others}} > 0) { my @todo = sort keys %{$self->{others}}; $state->errsay("#1 is missing #2", $self->{name}, $self->string(@todo)); if ($self->{name} =~ m/^partial/) { $state->errsay("not a problem, since this is a partial- package"); } else { $self->ask_add_deps($state, \@todo); } } } package OpenBSD::DirectDependencyCheck; our @ISA = qw(OpenBSD::DependencyCheck); use OpenBSD::RequiredBy; sub string { my $self = shift; return "dependencies: ". join(' ', @_); } sub new { my ($class, $state, $name) = @_; return $class->SUPER::new($state, $name, OpenBSD::Requiring->new($name)); } package OpenBSD::ReverseDependencyCheck; our @ISA = qw(OpenBSD::DependencyCheck); use OpenBSD::RequiredBy; sub string { my $self = shift; return "reverse dependencies: ". join(' ', @_); } sub new { my ($class, $state, $name) = @_; return $class->SUPER::new($state, $name, OpenBSD::RequiredBy->new($name)); } package OpenBSD::Pkglocate; sub new { my ($class, $state) = @_; bless {state => $state, result => {}, params => []}, $class; } sub add_param { my ($self, @p) = @_; push(@{$self->{params}}, @p); while (@{$self->{params}} > 50) { $self->run_command; } } sub run_command { my $self = shift; if (@{$self->{params}} == 0) { return; } my %h = map {($_, 1)} @{$self->{params}}; open(my $cmd, '-|', 'pkg_locate', map {"*:$_"} @{$self->{params}}); while (<$cmd>) { chomp; my ($pkgname, $pkgpath, $path) = split(':', $_); # pkglocate will return false positives, so trim them if ($h{$path}) { push(@{$self->{result}{"$pkgname:$pkgpath"} }, $path); delete $h{$path}; } } close($cmd); for my $k (keys %h) { push(@{$self->{result}{unknown}}, $k); } $self->{params} = []; } sub result { my $self = shift; while (@{$self->{params}} > 0) { $self->run_command; } my $state = $self->{state}; my $r = $self->{result}; my $u = $r->{unknown}; delete $r->{unknown}; $state->say("Not found:"); for my $e (sort @$u) { $state->say("\t#1", $e); } for my $k (sort keys %{$r}) { $state->say("In #1:", $k); for my $e (sort @{$r->{$k}}) { $state->say("\t#1", $e); } } } package OpenBSD::PkgCheck; our @ISA = qw(OpenBSD::AddCreateDelete); use OpenBSD::PackageInfo; use OpenBSD::PackingList; use File::Find; use OpenBSD::Paths; use OpenBSD::Mtree; sub remove { my ($self, $state, $name) = @_; $state->{removed}{$name} = 1; my $dir = installed_info($name); for my $i (@OpenBSD::PackageInfo::info) { if (-e $dir.$i) { if ($state->verbose) { $state->say("unlink(#1)", $dir.$i); } unless ($state->{not}) { unlink($dir.$i) or $state->errsay("#1: Couldn't delete #2: #3", $name, $dir.$i, $!); } } } if (-f $dir) { if ($state->verbose) { $state->say("unlink(#1)", $dir); } unless ($state->{not}) { unlink($dir) or $state->errsay("#1: Couldn't delete #2: #3", $name, $dir, $!); } } elsif (-d $dir) { if ($state->verbose) { $state->say("rmdir(#1)", $dir); } unless ($state->{not}) { rmdir($dir) or $state->errsay("#1: Couldn't delete #2: #3", $name, $dir, $!); } } } sub may_remove { my ($self, $state, $name) = @_; if ($state->{force}) { $self->remove($state, $name); } elsif ($state->{interactive}) { require OpenBSD::Interactive; if (OpenBSD::Interactive::confirm("Remove wrong package $name")) { $self->remove($state, $name); } } $state->{bogus}{$name} = 1; } sub for_all_packages { my ($self, $state, $l, $msg, $code) = @_; $state->progress->for_list($msg, $l, sub { return if $state->{removed}{$_[0]}; if ($state->{bogus}{$_[0]}) { $state->errsay("skipping #1", $_[0]); return; } &$code; }); } sub sanity_check { my ($self, $state, $l) = @_; $self->for_all_packages($state, $l, "Packing-list sanity", sub { my $name = shift; my $info = installed_info($name); if (-f $info) { $state->errsay("#1: #2 should be a directory", $state->safe($name), $state->safe($info)); if ($info =~ m/\.core$/) { $state->errsay("looks like a core dump, ". "removing"); $self->remove($state, $name); } else { $self->may_remove($state, $name); } return; } my $contents = $info.OpenBSD::PackageInfo::CONTENTS; unless (-f $contents) { $state->errsay("#1: missing #2", $state->safe($name), $state->safe($contents)); $self->may_remove($state, $name); return; } my $plist; eval { $plist = OpenBSD::PackingList->fromfile($contents); }; if ($@ || !defined $plist) { $state->errsay("#1: bad packing-list", $state->safe($name)); $self->may_remove($state, $name); return; } if (!defined $plist->pkgname) { $state->errsay("#1: no pkgname in plist", $state->safe($name)); $self->may_remove($state, $name); return; } if ($plist->pkgname ne $name) { $state->errsay("#1: pkgname does not match", $state->safe($name)); $self->may_remove($state, $name); } $plist->mark_available_lib($plist->pkgname, $state); $state->{exists}{$plist->pkgname} = 1; }); } sub dependencies_check { my ($self, $state, $l) = @_; OpenBSD::SharedLibs::add_libs_from_system($state->{destdir}, $state); $self->for_all_packages($state, $l, "Direct dependencies", sub { my $name = shift; my $plist = OpenBSD::PackingList->from_installation($name, \&OpenBSD::PackingList::DependOnly); my $checker = OpenBSD::DirectDependencyCheck->new($state, $name); $state->{localbase} = $plist->localbase; $plist->find_dependencies($state, $l, $checker); $checker->adjust($state); for my $dep ($checker->{req}->list) { push(@{$state->{reverse}{$dep}}, $name); } }); } sub reverse_dependencies_check { my ($self, $state, $l) = @_; $self->for_all_packages($state, $l, "Reverse dependencies", sub { my $name = shift; my $checker = OpenBSD::ReverseDependencyCheck->new($state, $name); for my $i (@{$state->{reverse}{$name}}) { $checker->find($i) or $checker->not_found($i); } $checker->adjust($state); }); } sub package_files_check { my ($self, $state, $l) = @_; $self->for_all_packages($state, $l, "Files from packages", sub { my $name = shift; my $plist = OpenBSD::PackingList->from_installation($name); $state->log->set_context($name); if ($state->{quick}) { $plist->basic_check($state); } else { $plist->thorough_check($state); } $plist->mark_available_lib($plist->pkgname, $state); }); } sub install_pkglocate { my ($self, $state) = @_; my $spec = 'pkglocatedb->=0.4'; my @l = installed_stems()->find('pkglocatedb'); require OpenBSD::PkgSpec; if (OpenBSD::PkgSpec->new($spec)->match_ref(\@l)) { return 1; } require OpenBSD::Interactive; unless (OpenBSD::Interactive::confirm( "Unknown file system entries.\n". "Do you want to install $spec to look them up")) { return 0; } require OpenBSD::PkgAdd; $state->{installer} //= Installer->new($state); if ($state->{installer}->install('pkglocatedb--')) { return 1; } else { $state->errsay("Couldn't install #1", $spec); return 0; } } # non fancy display of unknown objects sub display_unknown { my ($self, $state) = @_; if (defined $state->{unknown}{file}) { $state->say("Unknown files:"); for my $e (sort @{$state->{unknown}{file}}) { $state->say("\t#1", $e); } } if (defined $state->{unknown}{dir}) { $state->say("Unknown directories:"); for my $e (sort {$b cmp $a } @{$state->{unknown}{dir}}) { $state->say("\t#1", $e); } } } sub locate_unknown { my ($self, $state) = @_; my $locator = OpenBSD::Pkglocate->new($state); if (defined $state->{unknown}{file}) { $state->progress->for_list("Locating unknown files", $state->{unknown}{file}, sub { $locator->add_param($_[0]); }); } if (defined $state->{unknown}{dir}) { $state->progress->for_list("Locating unknown directories", $state->{unknown}{dir}, sub { $locator->add_param($_[0]); }); } $locator->result; } sub localbase_check { my ($self, $state) = @_; $state->{known} //= {}; my $base = $state->{destdir}.OpenBSD::Paths->localbase; for my $file (OpenBSD::Paths::man_cruft()) { $state->{known}{$base."/man"}{$file} = 1; } $state->{known}{$base."/info"}{'dir'} = 1; $state->{known}{$base."/lib/X11"}{'app-defaults'} = 1; $state->{known}{$base."/libdata"} = {}; $state->{known}{$base."/libdata/perl5"} = {}; # XXX OpenBSD::Mtree::parse($state->{known}, $base, "/etc/mtree/BSD.local.dist", 1); $state->progress->set_header("Checking file system"); find(sub { $state->progress->working(1024); if (-d $_) { if ($File::Find::name eq OpenBSD::Paths->localbase."/lost+found") { $state->say("fsck(8) info found: #1", $File::Find::name); $File::Find::prune = 1; return; } return if defined $state->{known}{$File::Find::name}; if (-l $_) { return if $state->{known}{$File::Find::dir}{$_}; } push(@{$state->{unknown}{dir}}, $File::Find::name); } else { return if $state->{known}{$File::Find::dir}{$_}; push(@{$state->{unknown}{file}}, $File::Find::name); } }, OpenBSD::Paths->localbase); if (defined $state->{unknown}) { if ($self->install_pkglocate($state)) { $self->locate_unknown($state); } else { $self->display_unknown($state); } } } sub run { my ($self, $state) = @_; my @list = installed_packages(); $self->sanity_check($state, \@list); $self->dependencies_check($state, \@list); $state->log->dump; $self->reverse_dependencies_check($state, \@list); $state->log->dump; $self->package_files_check($state, \@list); $state->log->dump; $self->localbase_check($state); $state->progress->next; } sub parse_and_run { my ($self, $cmd) = @_; my $state = OpenBSD::PkgCheck::State->new($cmd); $state->handle_options; if (@ARGV != 0) { $state->usage; } lock_db(0, $state) unless $state->{subst}->value('nolock'); $self->run($state); return 0; } 1;