#!/usr/bin/perl # ex:ts=8 sw=4: # $OpenBSD: pkg_delete,v 1.61 2004/10/20 11:38:57 espie Exp $ # # Copyright (c) 2003-2004 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::Getopt; use OpenBSD::PackingList; use OpenBSD::PackingOld; use OpenBSD::PackageInfo; use OpenBSD::RequiredBy; use OpenBSD::Logger; use OpenBSD::Vstat; use OpenBSD::PackageInfo; use OpenBSD::Error; use OpenBSD::ProgressMeter; our %forced = (); sub ensure_ldconfig { my $state = shift; return if $state->{not}; return unless defined $OpenBSD::PackingElement::Lib::todo; VSystem($state->{very_verbose}, @OpenBSD::PackingElement::Lib::ldconfig, "-R"); undef $OpenBSD::PackingElement::Lib::todo; } sub record_all_shared_items { my $db = {dirs=>{}, users=>{}, groups=>{}}; my @list = installed_packages(); my $total = @list; OpenBSD::ProgressMeter::set_header("Read shared items"); my $done = 0; for my $e (@list) { OpenBSD::ProgressMeter::show($done, $total); my $plist = OpenBSD::PackingList->fromfile(installed_info($e).CONTENTS, \&OpenBSD::PackingList::DirrmOnly) or next; for my $item (@{$plist->{users}}, @{$plist->{groups}}, @{$plist->{items}}) { $item->record_shared_item($e, $db); } $done++; } return $db; } sub cleanup_shared_items { my $state = shift; my $h = $state->{dirs_to_rm}; my $u = $state->{users_to_rm}; my $g = $state->{groups_to_rm}; return unless defined $h or defined $u or defined $g; my $remaining = record_all_shared_items(); OpenBSD::ProgressMeter::clear(); OpenBSD::ProgressMeter::set_header("Clean shared items"); my $total = 0; $total += keys %$h if defined $h; if ($state->{extra}) { $total += keys %$u if defined $u; $total += keys %$g if defined $g; } my $done = 0; if (defined $h) { for my $d (sort {$b cmp $a} keys %$h) { OpenBSD::ProgressMeter::show($done, $total); my $realname = $state->{destdir}.$d; if ($remaining->{dirs}->{$realname}) { for my $i (@{$h->{$d}}) { $i->reload($state); } } else { for my $i (@{$h->{$d}}) { $i->cleanup($state); } if (!rmdir $realname) { print "Error deleting directory $realname: $!\n"; } } $done++; } } if ($state->{extra}) { if (defined $u) { for my $user (keys %$u) { OpenBSD::ProgressMeter::show($done, $total); next if $remaining->{users}->{$user}; System("/usr/sbin/userdel", $user); $done++; } } if (defined $g) { for my $group (keys %$g) { OpenBSD::ProgressMeter::show($done, $total); next if $remaining->{groups}->{$group}; System("/usr/sbin/groupdel", $group); $done++; } } } OpenBSD::ProgressMeter::next(); } package OpenBSD::PackingElement; sub delete { } sub record_shared_item { } sub cleanup { } sub reload { } sub log_pkgname { my ($self, $state) = @_; if (defined $state->{pkgname_tolog}) { OpenBSD::Logger::log "# package ", $state->{pkgname_tolog}, "\n"; $state->{pkgname_tolog} = undef; } } package OpenBSD::PackingElement::NewUser; sub record_shared_item { my ($self, $pkgname, $db) = @_; my $k = $self->{name}; $db->{users}->{$k} = 1; } sub delete { my ($self, $state) = @_; my $name = $self->{name}; if ($state->{beverbose}) { print "rmuser: $name\n"; } $state->{users_to_rm} = {} unless defined $state->{users_to_rm}; my $h = $state->{users_to_rm}; $h->{$name} = 1; } package OpenBSD::PackingElement::NewGroup; sub record_shared_item { my ($self, $pkgname, $db) = @_; my $k = $self->{name}; $db->{groups}->{$k} = 1; } sub delete { my ($self, $state) = @_; my $name = $self->{name}; if ($state->{beverbose}) { print "rmgroup: $name\n"; } $state->{groups_to_rm} = {} unless defined $state->{groups_to_rm}; my $h = $state->{groups_to_rm}; $h->{$name} = 1; } package OpenBSD::PackingElement::DirBase; sub record_shared_item { my ($self, $pkgname, $db) = @_; my $k = $self->fullname(); $db->{dirs}->{$k} = 1; } sub delete { my ($self, $state) = @_; my $name = $self->fullname(); if ($state->{beverbose}) { print "dirrm: $name\n"; } $state->{dirs_to_rm} = {} unless defined $state->{dirs_to_rm}; my $h = $state->{dirs_to_rm}; $h->{$name} = [] unless defined $h->{$name}; push(@{$h->{$name}}, $self->clone()); } package OpenBSD::PackingElement::DirRm; sub record_shared_item { &OpenBSD::PackingElement::DirBase::record_shared_item; } sub delete { &OpenBSD::PackingElement::DirBase::delete; } package OpenBSD::PackingElement::Unexec; sub delete { my ($self, $state) = @_; $self->run($state); } package OpenBSD::PackingElement::FileBase; use OpenBSD::md5; sub delete { my ($self, $state) = @_; my $name = $self->fullname(); my $realname = $state->{destdir}.$name; if (-l $realname) { if ($state->{beverbose}) { print "deleting symlink: $realname\n"; } } else { if (! -f $realname) { print "File $realname does not exist\n"; return; } unless (defined($self->{link}) or $self->{nochecksum} or $state->{quick}) { if (!defined $self->{md5}) { print "Problem: $name does not have an md5 checksum\n"; print "NOT deleting: $realname\n"; $self->log_pkgname($state); OpenBSD::Logger::log "rm $state->{destdirname}$name\n"; return; } my $md5 = OpenBSD::md5::fromfile($realname); if ($md5 ne $self->{md5}) { print "Problem: md5 doesn't match for $name\n"; print "NOT deleting: $realname\n"; $self->log_pkgname($state); OpenBSD::Logger::log "rm $state->{destdirname}$name #MD5\n"; return; } } if ($state->{beverbose}) { print "deleting: $realname\n"; } } return if $state->{not}; if (!unlink $realname) { print "Problem deleting $realname\n"; $self->log_pkgname($state); OpenBSD::Logger::log "rm $state->{destdirname}$name\n"; } } package OpenBSD::PackingElement::Sample; use OpenBSD::md5; use OpenBSD::Error; sub delete { my ($self, $state) = @_; my $name = $self->fullname(); my $realname = $state->{destdir}.$name; my $orig = $self->{copyfrom}; if (!defined $orig) { Fatal "\@sample element does not reference a valid file\n"; } my $origname = $state->{destdir}.$orig->fullname(); if (! -e $realname) { print "File $realname does not exist\n"; return; } if (! -f $realname) { print "File $realname is not a file\n"; return; } if (!defined $orig->{md5}) { print "Problem: file $name does not have an md5 checksum\n"; print "NOT deleting: $realname\n"; $self->log_pkgname($state); OpenBSD::Logger::log "rm $state->{destdirname}$name\n"; return; } if ($state->{quick}) { unless ($state->{extra}) { print "NOT deleting file $realname\n"; return; } } else { my $md5 = OpenBSD::md5::fromfile($realname); if ($md5 eq $orig->{md5}) { print "File $realname identical to sample\n" if $state->{not} or $state->{verbose}; } else { print "File $realname NOT identical to sample\n"; unless ($state->{extra}) { print "NOT deleting $realname\n"; return; } } } return if $state->{not}; print "deleting $realname\n" if $state->{verbose}; if (!unlink $realname) { print "Problem deleting $realname\n"; $self->log_pkgname($state); OpenBSD::Logger::log "rm $state->{destdirname}$name\n"; } } package OpenBSD::PackingElement::InfoFile; use File::Basename; use OpenBSD::Error; sub delete { my ($self, $state) = @_; unless ($state->{not}) { my $fullname = $state->{destdir}.$self->fullname(); VSystem($state->{very_verbose}, "install-info", "--delete", "--info-dir=".dirname($fullname), $fullname); } $self->SUPER::delete($state); } package OpenBSD::PackingElement::Shell; sub delete { my ($self, $state) = @_; unless ($state->{not}) { my $destdir = $state->{destdir}; my $fullname = $self->fullname(); my @l=(); if (open(my $shells, '<', $destdir.'/etc/shells')) { local $_; while(<$shells>) { push(@l, $_); s/^\#.*//; if ($_ =~ m/^\Q$fullname\E\s*$/) { pop(@l); } } close($shells); open(my $shells2, '>', $destdir.'/etc/shells'); print $shells2 @l; close $shells2; print "Shell $fullname removed from $destdir/etc/shells\n"; } } $self->SUPER::delete($state); } package OpenBSD::PackingElement::Extra; sub delete { my ($self, $state) = @_; return unless $state->{extra}; my $name = $self->fullname(); my $realname = $state->{destdir}.$name; if ($state->{beverbose}) { print "deleting extra file: $realname\n"; } return if $state->{not}; return unless -e $realname; unlink($realname) or print "problem deleting extra file $realname\n"; } package OpenBSD::PackingElement::Extradir; sub delete { my ($self, $state) = @_; return unless $state->{extra}; return unless -e $state->{destdir}.$self->fullname(); $self->SUPER::delete($state); } package OpenBSD::PackingElement::Mandir; sub cleanup { my ($self, $state) = @_; my $fullname = $state->{destdir}.$self->fullname(); $state->print("You may wish to remove ", $fullname, " from man.conf\n"); unlink("$fullname/whatis.db"); } package OpenBSD::PackingElement::Fontdir; sub cleanup { my ($self, $state) = @_; my $fullname = $state->{destdir}.$self->fullname(); print "You may wish to remove ", $fullname, " from your font path\n"; unlink("$fullname/fonts.alias"); unlink("$fullname/fonts.dir"); unlink("$fullname/fonts.cache-1"); } package OpenBSD::PackingElement::Infodir; sub cleanup { my ($self, $state) = @_; my $fullname = $state->{destdir}.$self->fullname(); unlink("$fullname/dir"); } package OpenBSD::PackingElement::ExtraUnexec; use OpenBSD::Error; sub delete { my ($self, $state) = @_; return unless $state->{extra}; $self->run($state); } package OpenBSD::PackingElement::Lib; sub delete { my ($self, $state) = @_; $self->SUPER::delete($state); $self->mark_ldconfig_directory($state->{destdir}); } package OpenBSD::PackingElement::FREQUIRE; use OpenBSD::PackageInfo; use OpenBSD::Error; sub delete { my ($self, $state) = @_; $self->run($state, "DEINSTALL"); } package OpenBSD::PackingElement::FDEINSTALL; use OpenBSD::PackageInfo; use OpenBSD::Error; sub delete { my ($self, $state) = @_; $self->run($state, "DEINSTALL"); } package OpenBSD::PackingElement::FUNDISPLAY; use OpenBSD::Error; use OpenBSD::PackageInfo; sub delete { my ($self, $state) = @_; my $dir = $state->{dir}; my $pager = $ENV{'PAGER'} || "/usr/bin/more"; System("$pager $dir".UNDISPLAY); } package main; our ($opt_v, $opt_D, $opt_d, $opt_n, $opt_f, $opt_q, $opt_p, $opt_c, $opt_L, $opt_B, $opt_I, $opt_x); $opt_v = 0; sub remove_packing_info { my $dir = shift; for my $fname (info_names()) { unlink($dir.$fname); } rmdir($dir) or Fatal "Can't finish removing directory $dir: $!"; } sub manpages_unindex { my ($state) = @_; return unless defined $state->{mandirs}; my $destdir = $state->{destdir}; require OpenBSD::Makewhatis; while (my ($k, $v) = each %{$state->{mandirs}}) { my @l = map { $destdir.$_ } @$v; eval { OpenBSD::Makewhatis::remove($destdir.$k, \@l); }; if ($@) { print STDERR "Error in makewhatis: $@\n"; } } undef $state->{mandirs}; } sub validate_plist($$) { my ($plist, $destdir) = @_; my $problems = 0; my $totsize = 0; for my $item (@{$plist->{items}}) { next unless $item->IsFile(); my $fname = $destdir.$item->fullname(); $totsize += $item->{size} if defined $item->{size}; my $s = OpenBSD::Vstat::remove($fname, $item->{size}); next unless defined $s; if ($s->{ro}) { Warn "Error: ", $s->{mnt}, " is read-only ($fname)\n"; $problems++; } } Fatal "fatal issues" if $problems; return $totsize; } sub delete_package { my ($pkgname, $state) = @_; $state->{pkgname} = $pkgname; my $dir = installed_info($pkgname); $state->{dir} = $dir; my $plist = OpenBSD::PackingList->fromfile($dir.CONTENTS) or Fatal "Bad package"; if (!defined $plist->pkgname()) { Fatal "Package $pkgname has no name"; } if ($plist->pkgname() ne $pkgname) { Fatal "Package $pkgname real name does not match"; } my $totsize = validate_plist($plist, $state->{destdir}); $ENV{'PKG_PREFIX'} = $plist->pkgbase(); if ($plist->has(REQUIRE)) { $plist->get(REQUIRE)->delete($state); } if ($plist->has(DEINSTALL)) { $plist->get(DEINSTALL)->delete($state); } $plist->visit('register_manpage', $state); manpages_unindex($state); my $donesize = 0; for my $item (@{$plist->{groups}}, @{$plist->{users}}, @{$plist->{items}}) { $item->delete($state); if (defined $item->{size}) { $donesize += $item->{size}; OpenBSD::ProgressMeter::show($donesize, $totsize); } } OpenBSD::ProgressMeter::next(); if ($plist->has(UNDISPLAY)) { $plist->get(UNDISPLAY)->prepare($state); } # guard against duplicate pkgdep my $removed = {}; for my $item (@{$plist->{pkgdep}}) { my $name = $item->{name}; next if defined $removed->{$name}; print "remove dependency in $name\n" if $opt_v or $opt_n; local $@; eval { OpenBSD::RequiredBy->new($name)->delete($pkgname) unless $opt_n; }; if ($@) { print STDERR "$@\n"; } $removed->{$name} = 1; } remove_packing_info($dir) unless $opt_n; } getopts('vcxDdnf:qpS:L:B:I', {'v' => sub {++$opt_v;} }); if ($opt_D) { $opt_I = 1; } $opt_B = $ENV{'PKG_DESTDIR'} unless defined $opt_B; $opt_B = '' unless defined $opt_B; if ($opt_B ne '') { $opt_B.='/' unless $opt_B =~ m/\/$/; } $ENV{'PKG_DESTDIR'} = $opt_B; $opt_L = '/usr/local' unless defined $opt_L; if (defined $opt_p) { Fatal "Option p is obsolete"; } if (defined $opt_d) { Fatal "Option d is obsolete"; } if ($opt_x) { OpenBSD::ProgressMeter::enable(); } if ($opt_f) { %forced = map {($_, 1)} split(/,/, $opt_f); } if ($< && !$forced{nonroot}) { Fatal "$0 must be run as root"; } my %done; my $removed; my $state = new OpenBSD::Error; $state->{not} = $opt_n; $state->{quick} = $opt_q; $state->{verbose} = $opt_v >= 2; $state->{very_verbose} = $opt_v; $state->{beverbose} = $opt_n || ($opt_v >= 2); $state->{extra} = $opt_c; $state->{dont_run_scripts} = $opt_I; $ENV{'PKG_DELETE_EXTRA'} = $state->{extra} ? "Yes" : "No"; # First, resolve pkg names my @realnames; my $bad; for my $pkgname (@ARGV) { $pkgname =~ s/\.tgz$//; if (is_installed($pkgname)) { push(@realnames, installed_name($pkgname)); } else { if (OpenBSD::PackageName::is_stem($pkgname)) { my @l = OpenBSD::PackageName::findstem($pkgname, installed_packages()); if (@l == 0) { print "Can't resolve $pkgname to an installed package name\n"; $bad = 1 unless $forced{uninstalled}; } elsif (@l == 1) { push(@realnames, $l[0]); } elsif (@l != 0) { print "Ambiguous: $pkgname could be ", join(' ', @l),"\n"; if ($forced{ambiguous}) { print "(removing them all)\n"; push(@realnames, @l); } else { $bad = 1; } } } } } # Then check that dependencies are okay my (%toremove, %extra_rm); my @todo; for my $pkgname (@realnames) { $toremove{$pkgname} = 1; } push(@todo, @realnames); OpenBSD::Logger::log_as("pkg_delete"); while (my $pkgname = pop @todo) { my $deps = OpenBSD::RequiredBy->new($pkgname)->list(); if (@$deps > 0) { for my $dep (@$deps) { next if defined $toremove{$dep}; next if defined $extra_rm{$dep}; $extra_rm{$dep}=$pkgname; push(@todo, $dep); } } } if (keys(%extra_rm) != 0) { print "Can't remove ", join(' ', @ARGV), " without also removing:\n", join(' ', keys(%extra_rm)), "\n"; if ($forced{dependencies}) { print "(removing them as well)\n"; push(@realnames, keys(%extra_rm)); } else { $bad = 1; } } if ($bad) { exit(1); } $state->{destdir} = $opt_B; if ($opt_B eq '') { $state->{destdirname} = ''; } else { OpenBSD::Logger::annotate("PKG_DESTDIR=\"$opt_B\"; export PKG_DESTDIR\n"); $state->{destdirname} = '${PKG_DESTDIR}'; } eval { # and finally, handle the removal { do { $removed = 0; for my $pkgname (@realnames) { next if $done{$pkgname}; unless (is_installed($pkgname)) { print "$pkgname was not installed\n"; $done{$pkgname} = 1; $removed++; next; } my $deps = OpenBSD::RequiredBy->new($pkgname)->list(); next if @$deps > 0; if (!OpenBSD::ProgressMeter::set_header($pkgname)) { print $opt_n ? "Pretending to delete " : "Deleting ", "$pkgname\n"; } $state->set_pkgname($pkgname); $state->{pkgname_tolog} = $pkgname; delete_package($pkgname, $state); delete_installed($pkgname); $done{$pkgname} = 1; $removed++; } # we're not actually doing anything, so we can't expect this loop # to ever finish last if $opt_n; } while ($removed); } }; my $dielater = $@; ensure_ldconfig($state); # delayed directory/user/group removal cleanup_shared_items($state); OpenBSD::PackingElement::Fontdir::finish_fontdirs(); if (defined $state->{display}) { close $state->{display}; my $pager = $ENV{'PAGER'} || "/usr/bin/more"; System("$pager ".$state->{displayname}); } my $logname = OpenBSD::Logger::logname(); if (defined $logname) { print "Problems logged as $logname\n"; } $state->delayed_output(); if ($dielater) { die $dielater; }