diff options
author | Marc Espie <espie@cvs.openbsd.org> | 2004-12-01 12:17:41 +0000 |
---|---|---|
committer | Marc Espie <espie@cvs.openbsd.org> | 2004-12-01 12:17:41 +0000 |
commit | 4ff3611b0a379fac85b444b803fce1a853eea56b (patch) | |
tree | e304e48b1081cf2324862645dc83cae44641b5e0 /usr.sbin/pkg_add/pkg_create | |
parent | 46c1d1257400dbb4ad62574916323b6d9189ebd7 (diff) |
allow stuff to recreate a package from an installed CONTENTS: since
we have all the md5 and stuff, we just need to make sure the package is
correct (hence verify_checksum).
Add a -n option to pkg_create: this is consistent with pkg_add/pkg_delete,
and with this new verify_checksum, pkg_create can be used to check that a
package still matches its contents.
Diffstat (limited to 'usr.sbin/pkg_add/pkg_create')
-rw-r--r-- | usr.sbin/pkg_add/pkg_create | 110 |
1 files changed, 94 insertions, 16 deletions
diff --git a/usr.sbin/pkg_add/pkg_create b/usr.sbin/pkg_add/pkg_create index 50902db65e6..a7178173996 100644 --- a/usr.sbin/pkg_add/pkg_create +++ b/usr.sbin/pkg_add/pkg_create @@ -1,6 +1,6 @@ #! /usr/bin/perl # ex:ts=8 sw=4: -# $OpenBSD: pkg_create,v 1.25 2004/11/18 21:51:43 espie Exp $ +# $OpenBSD: pkg_create,v 1.26 2004/12/01 12:17:40 espie Exp $ # # Copyright (c) 2003-2004 Marc Espie <espie@openbsd.org> # @@ -31,6 +31,8 @@ use File::Basename; package OpenBSD::PackingElement; sub archive_cmd { () } +sub anything { ${$_[1]}++; } + package OpenBSD::PackingElement::FileBase; sub archive_cmd { @@ -75,6 +77,10 @@ sub compute_checksum { } +sub verify_checksum +{ +} + package OpenBSD::PackingElement::FileBase; use OpenBSD::md5; @@ -105,6 +111,37 @@ sub compute_checksum } } +sub verify_checksum +{ + my ($self, $base, $stash) = @_; + my $fname = $self->fullname(); + my $check = {}; + if (-l "$base/$fname") { + my $value = readlink "$base/$fname"; + $check->{symlink} = $value; + } elsif (-f _) { + my ($dev, $ino, $size) = (stat _)[0,1,7]; + if (defined $stash->{"$dev/$ino"}) { + $check->{link} = $stash->{"$dev/$ino"}; + } else { + $stash->{"$dev/$ino"} = $fname; + $check->{md5} = OpenBSD::md5::fromfile("$base/$fname"); + $check->{size} = $size; + } + } else { + print STDERR "Error in package: $base/$fname does not exist\n"; + $main::errors++; + } + for my $field (qw(symlink link md5 size)) { + if ((defined $check->{$field} && defined $self->{$field} && + $check->{$field} ne $self->{$field}) || + (defined $check->{$field} xor defined $self->{$field})) { + print STDERR "Error: $field inconsistency for $fname\n"; + $main::errors++; + } + } +} + package OpenBSD::PackingElement::InfoFile; sub compute_checksum { @@ -148,6 +185,16 @@ sub makesum } } +sub checksum +{ + my ($self, $base) = @_; + my $stash = {}; + for my $item (@{$self->{items}}) { + $self->{state}->{cwd} = $item->{cwd} if defined $item->{cwd}; + $item->verify_checksum($base, $stash); + } +} + sub avert_duplicates { my ($self) = @_; @@ -219,13 +266,14 @@ our $errors = 0; our ($opt_p, $opt_f, $opt_c, $opt_d, $opt_v, $opt_i, $opt_k, $opt_r, $opt_S, $opt_h, $opt_s, $opt_O, $opt_A, $opt_L, - $opt_M, $opt_U, $opt_P, $opt_W, + $opt_M, $opt_U, $opt_P, $opt_W, $opt_n, $opt_B); my @contents; +my $regen_package = 0; my $plist = new OpenBSD::PackingList; -getopts('p:f:c:d:vi:k:r:M:U:S:hs:OA:L:B:D:P:W:', +getopts('p:f:c:d:vi:k:r:M:U:S:hs:OA:L:B:D:P:W:n', {'D' => sub { local $_ = shift; @@ -247,11 +295,14 @@ getopts('p:f:c:d:vi:k:r:M:U:S:hs:OA:L:B:D:P:W:', } }); -if (@ARGV != 1) { +if (@ARGV == 0) { + $regen_package = 1; +} elsif (@ARGV != 1) { die "Exactly one single package name is required"; } my $dir = OpenBSD::Temp::dir(); +my $dir2 = $dir; if (defined $opt_s) { die "Option s is no longer supported"; @@ -274,7 +325,7 @@ if (defined $opt_c) { copy_subst($opt_c, $dir.COMMENT); } } else { - die "Comment required"; + die "Comment required" unless $regen_package; } if (defined $opt_d) { @@ -286,10 +337,10 @@ if (defined $opt_d) { copy_subst($opt_d, $dir.DESC); } } else { - die "Description required"; + die "Description required" unless $regen_package; } -print "Creating package $ARGV[0]\n" if $opt_v; +print "Creating package $ARGV[0]\n" if $opt_v && !$regen_package; if (defined $opt_i) { copy_subst($opt_i, $dir.INSTALL); @@ -322,11 +373,11 @@ for my $special (info_names()) { if (defined $opt_p) { OpenBSD::PackingElement::Cwd->add($plist, $opt_p); -} else { +} elsif (!$regen_package) { die "Prefix required"; } -if ($ARGV[0] =~ m|([^/]+)$|) { +if (!$regen_package && $ARGV[0] =~ m|([^/]+)$|) { my $pkgname = $1; $pkgname =~ s/\.tgz$//; OpenBSD::PackingElement::Name->add($plist, $pkgname); @@ -340,6 +391,15 @@ if (defined $opt_L) { OpenBSD::PackingElement::LocalBase->add($plist, $opt_L); } +if ($regen_package) { + my $v = 0; + $plist->visit('anything', \$v); + if ($v != 0 || @contents != 1) { + die "Exactly one single package name is required"; + } + $dir = dirname($contents[0]); +} + for my $contentsfile (@contents) { $plist->fromfile($contentsfile, sub { @@ -408,7 +468,12 @@ if (!defined $plist->{extrainfo} && } -$plist->makesum($base); +if ($regen_package) { + $plist->checksum($base); + $dir2 = OpenBSD::Temp::dir(); +} else { + $plist->makesum($base); +} $plist->avert_duplicates(); if (defined $plist->{pkgcfl}) { print STDERR "Error: \@pkgcfl is obsolete, use \@conflict instead\n"; @@ -421,16 +486,29 @@ if ($errors) { exit(1); } -my @cmd = $plist->archive_cmd($dir, $base); +my @cmd = $plist->archive_cmd($dir2, $base); if (!defined $plist->{name}) { print STDERR "Can't write unnamed packing list\n"; exit 1; } -$plist->tofile($dir.CONTENTS) or die "Can't write packing list"; -print "Creating gzip'd tar ball in '$ARGV[0]'\n" if $opt_v; -System('tar', $opt_h ? "zcfh" : "zcf", $ARGV[0], "-C", $dir, CONTENTS, - @extra_files, @cmd) == 0 or - die "tar failed"; +my $wname; +if ($regen_package) { + $wname = $plist->pkgname().".tgz"; +} else { + $plist->tofile($dir.CONTENTS) or die "Can't write packing list"; + $wname = $ARGV[0]; +} + +@cmd = ('tar', $opt_h ? "zcfh" : "zcf", $wname, "-C", $dir, CONTENTS, + @extra_files, @cmd); + +if ($opt_n) { + print join(' ', @cmd), "\n"; + exit(0); +} else { + print "Creating gzip'd tar ball in '$wname'\n" if $opt_v; + System(@cmd) == 0 or die "tar failed"; +} |