#! /usr/bin/perl # ex:ts=8 sw=4: # $OpenBSD: pkg_create,v 1.66 2007/04/30 09:07:16 espie Exp $ # # Copyright (c) 2003-2007 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::PackingList; use OpenBSD::PackageInfo; use OpenBSD::Getopt; use OpenBSD::md5; use OpenBSD::Temp; use OpenBSD::Error; use OpenBSD::Ustar; use OpenBSD::ArcCheck; use Symbol; use File::Basename; # Extra stuff needed to archive files package OpenBSD::PackingElement; sub create_package { my ($self, $arc, $base, $verbose) = @_; $self->archive($arc, $base); if ($verbose) { $self->comment_create_package(); } } sub archive {} sub comment_create_package {} sub count_elements { ${$_[1]}++; } sub print_file {} sub avert_duplicates_and_other_checks { my ($self, $allfiles) = @_; return unless $self->NoDuplicateNames(); my $n = $self->fullname(); if (defined $allfiles->{$n}) { print STDERR "Error in packing-list: duplicate item $n\n"; $main::errors++; } $allfiles->{$n} = 1; } my $warned; sub warn_once { my $self = shift; my $k = $self->keyword; if (!$warned->{$k}) { print STDERR "Error: \@$k is deprecated\n"; $main::errors++; $warned->{$k} = 1; } } sub makesum_plist { my ($self, $plist, $base, $stash) = @_; $self->add_object($plist); } package OpenBSD::PackingElement::PkgConflict; sub avert_duplicates_and_other_checks { shift->warn_once; } package OpenBSD::PackingElement::PkgDep; sub avert_duplicates_and_other_checks { shift->warn_once; } package OpenBSD::PackingElement::DirRm; sub avert_duplicates_and_other_checks { shift->warn_once; } package OpenBSD::PackingElement::SpecialFile; sub archive { my ($self, $arc, $base) = @_; my $o = $arc->prepare($self->{name}); $o->write(); } sub comment_create_package { my ($self) = @_; print "Adding ", $self->{name}, "\n"; } package OpenBSD::PackingElement::FileBase; use POSIX; sub archive { my ($self, $arc, $base) = @_; my $o = $arc->prepare_long($self); if (!defined $self->{owner} && !$o->isSymLink()) { if ($o->{uname} ne 'root' && $o->{uname} ne 'bin') { print STDERR "Error: no \@owner for ", $self->fullname(), " (", $o->{uname}, ")\n"; $main::errors++; } } if (!defined $self->{group} && !$o->isSymLink()) { if ($o->{gname} ne 'bin' && $o->{gname} ne 'wheel') { print STDERR "Warning: no \@group for ", $self->fullname(), " (", $o->{gname}, ")\n"; # $main::errors++; } } if (!defined $self->{mode} && $o->isFile()) { if (($o->{mode} & (S_ISUID | S_ISGID | S_IWOTH)) != 0) { print STDERR "Error: weird mode for ", $self->fullname(), ": ", sprintf("%4o", $o->{mode} & (S_IRWXU | S_IRWXG | S_IRWXO | S_ISUID | S_ISGID)), "\n"; $main::errors++; } } $o->write() unless $main::errors; } sub comment_create_package { my ($self) = @_; print "Adding ", $self->{name}, "\n"; } sub print_file { my ($item) = @_; print '@', $item->keyword(), " ", $item->fullname(), "\n"; } package OpenBSD::PackingElement::Cwd; use OpenBSD::Temp; sub archive { my ($self, $arc, $base) = @_; $arc->destdir($base."/".$self->{name}); } sub comment_create_package { my ($self) = @_; print "Cwd: ", $self->{name}, "\n"; } package OpenBSD::PackingElement; sub create_checksum { } sub verify_checksum { } package OpenBSD::PackingElement::FileBase; use OpenBSD::md5; sub compute_checksum { my ($self, $result, $base, $stash) = @_; my $fname = $self->fullname(); if (-l "$base/$fname") { my $value = readlink "$base/$fname"; $result->make_symlink($value); } elsif (-f _) { my ($dev, $ino, $size) = (stat _)[0,1,7]; if (defined $stash->{"$dev/$ino"}) { $result->make_hardlink($stash->{"$dev/$ino"}); } else { $stash->{"$dev/$ino"} = $fname; $result->add_md5(OpenBSD::md5::fromfile("$base/$fname")); $result->add_size($size); } } else { print STDERR "Error in package: $base/$fname does not exist\n"; $main::errors++; } } sub makesum_plist { my ($self, $plist, $base, $stash) = @_; $self->compute_checksum($self, $base, $stash); $self->add_object($plist); } sub verify_checksum { my ($self, $base, $stash) = @_; my $fname = $self->fullname(); my $check = ref($self)->new($self->{name}); $self->compute_checksum($check, $base, $stash); 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 makesum_plist { my ($self, $plist, $base, $stash) = @_; $self->SUPER::makesum_plist($plist, $base, $stash); my $fname = $self->fullname(); for (my $i = 1; ; $i++) { if (-e "$base/$fname-$i") { my $e = OpenBSD::PackingElement::File->add($plist, $self->{name}."-".$i); $e->compute_checksum($e, $base, $stash); } else { last; } } } package main; my %defines; sub dosubst { local $_ = shift; while (my ($k, $v) = each %defines) { s/\$\{\Q$k\E\}/$v/g; } s/\$\\/\$/g; return $_; } sub copy_subst { my ($srcname, $mode, $destname) = @_; open my $src, '<', $srcname or die "can't open $srcname"; open my $dest, $mode, $destname or die "can't open $destname"; local $_; while (<$src>) { print $dest dosubst($_); } } 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_M, $opt_U, $opt_P, $opt_W, $opt_n, $opt_B, $opt_q, $opt_Q); sub deduce_name { my ($o, $frag, $not) = @_; my $noto = $o; my $nofrag = "no-$frag"; $o =~ s/PFRAG\./PFRAG.$frag-/ or $o =~ s/PLIST/PFRAG.$frag/; $noto =~ s/PFRAG\./PFRAG.no-$frag-/ or $noto =~ s/PLIST/PFRAG.no-$frag/; unless (-e $o or -e $noto) { die "Missing fragments for $frag: $o and $noto don't exist"; } if ($not) { print "Switching to $noto\n" if !defined $opt_q; return $noto if -e $noto; } else { print "Switching to $o\n" if !defined $opt_q; return $o if -e $o; } return; } our $errors = 0; my @contents; my $regen_package = 0; set_usage( 'pkg_create [-hnQqv] [-A arches] [-B pkg-destdir] [-D name=value]', '[-i iscript] [-k dscript] [-L localbase] [-M module] [-M displayfile]', '[-P pkg-dependency] [-p prefix] [-r rscript] [-S pkg-destdir]', '[-U undisplayfile] [-W wantedlib] -c desc -d desc -f packinglist pkg-name'); my $plist = new OpenBSD::PackingList; try { getopts('hp:f:c:d:vi:k:r:m:M:U:S:hs:OA:L:B:D:P:W:nqQ', {'D' => sub { local $_ = shift; if (m/\=/) { $defines{$`} = $'; } else { $defines{$_} = 1; } }, 'f' => sub { push(@contents, shift); }, 'h' => sub { Usage(); }, 'P' => sub { OpenBSD::PackingElement::Dependency->add($plist, shift); }, 'W' => sub { OpenBSD::PackingElement::Wantlib->add($plist, shift); } }); } catchall { Usage($_); }; if (@ARGV == 0) { $regen_package = 1; } elsif (@ARGV != 1) { Usage "Exactly one single package name is required: ", join(' ', @ARGV); } try { my $dir; $dir = OpenBSD::Temp::dir() unless $opt_q; if (defined $opt_s) { Usage "Option s is no longer supported"; } if (defined $opt_O) { Usage "Option O is no longer supported"; } if (defined $opt_Q) { $opt_q = 1; } if (!@contents) { Usage "Packing list required"; } if (defined $opt_q) { for my $special (info_names()) { if ($special eq DESC or $special eq INSTALL and (defined $opt_i) or $special eq DEINSTALL and (defined $opt_k) or $special eq REQUIRE and (defined $opt_r) or $special eq DISPLAY and (defined $opt_M) or $special eq MODULE and (defined $opt_m) or $special eq UNDISPLAY and (defined $opt_U)) { OpenBSD::PackingElement::File->add($plist, $special); } } } else { if (defined $opt_c) { if ($opt_c =~ /^\-/) { open(my $fh, '>', $dir.DESC) or die "Can't write COMMENT to DESC file: $!"; print $fh $'; close($fh); } else { copy_subst($opt_c, '>', $dir.DESC); } } else { Usage "Comment required" unless $regen_package; } if (defined $opt_d) { if ($opt_d =~ /^\-/) { open(my $fh, '>>', $dir.DESC) or die "Can't write to DESC: $!"; print $fh $'; close($fh); } else { copy_subst($opt_d, '>>', $dir.DESC); } } else { Usage "Description required" unless $regen_package; } print "Creating package $ARGV[0]\n" if $opt_v && !$regen_package; if (defined $opt_i) { copy_subst($opt_i, '>', $dir.INSTALL); } if (defined $opt_k) { copy_subst($opt_k, '>', $dir.DEINSTALL); } if (defined $opt_r) { copy_subst($opt_r, '>', $dir.REQUIRE); } if (defined $opt_M) { copy_subst($opt_M, '>', $dir.DISPLAY); } if (defined $opt_m) { copy_subst($opt_m, '>', $dir.MODULE); } if (defined $opt_U) { copy_subst($opt_U, '>', $dir.UNDISPLAY); } for my $special (info_names()) { next unless -f $dir.$special; my $f = OpenBSD::PackingElement::File->add($plist, $special); $f->add_md5(OpenBSD::md5::fromfile($dir.$special)); $f->add_size((stat $dir.$special)[7]); } OpenBSD::PackingElement::File->add($plist, CONTENTS); } if (defined $opt_p) { OpenBSD::PackingElement::Cwd->add($plist, $opt_p); } elsif (!$regen_package) { Usage "Prefix required"; } if (defined $opt_A) { OpenBSD::PackingElement::Arch->add($plist, $opt_A); } if (defined $opt_L) { OpenBSD::PackingElement::LocalBase->add($plist, $opt_L); } if ($regen_package) { my $v = 0; $plist->count_elements(\$v); if ($v != 1 || @contents != 1) { Usage "Exactly one single packing list is required"; } $dir = dirname($contents[0]); } for my $contentsfile (@contents) { $plist->fromfile($contentsfile, sub { my ($fh, $cont) = @_; local $_; my (@fhstack, @namestack); push(@fhstack, $fh); push(@namestack, $contentsfile); while($fh = pop @fhstack) { my $fname = pop @namestack; GETLINE: while (<$fh>) { if (m/^(\!)?\%\%(.*)\%\%$/) { my ($not, $frag) = ($1, $2); my $def = $frag; if ($frag eq 'SHARED') { $def = 'SHARED_LIBS'; $frag = 'shared'; } if (!defined $defines{$def}) { die "Error: unknown fragment $frag"; } elsif ($defines{$def} == 1) { next GETLINE if defined $not; } elsif ($defines{$def} == 0) { next GETLINE unless defined $not; } else { die "Incorrect define for $frag"; } my $newname = deduce_name($fname, $frag, $not); if (defined $newname) { push(@fhstack, $fh); push(@namestack, $fname); $fname = $newname; $fh = gensym; open($fh, '<', $fname) or die "missing file $fname"; } next GETLINE; } if (m/^(\@comment\s+\$(?:Open)BSD\$)$/) { $_ = '@comment $'.'OpenBSD: '.basename($fname).',v$'; } if (m,^\@lib\s+.*/lib[^/]+\.so\.\d+\.\d+$,) { Warn "Shared library without SHARED_LIBS: $_"; $main::errors++; } &$cont(dosubst($_)); } } } ) or Fatal "Can't open packing list $contentsfile"; } if (!$plist->has('name') && $ARGV[0] =~ m|([^/]+)$|) { my $pkgname = $1; $pkgname =~ s/\.tgz$//; OpenBSD::PackingElement::Name->add($plist, $pkgname); } my $base = '/'; if (defined $opt_B) { $base = $opt_B; } elsif (defined $opt_S) { $base = $opt_S; } elsif (defined $ENV{'PKG_PREFIX'}) { $base = $ENV{'PKG_PREFIX'}; } my $fullpkgpath = $defines{'FULLPKGPATH'}; my $cdrom = $defines{'PERMIT_PACKAGE_CDROM'}; my $ftp = $defines{'PERMIT_PACKAGE_FTP'}; if (!defined $plist->{extrainfo} && defined $fullpkgpath && defined $cdrom && defined $ftp) { $cdrom = 'yes' if $cdrom =~ m/^yes$/i; $ftp = 'yes' if $ftp =~ m/^yes$/i; OpenBSD::PackingElement::ExtraInfo->add($plist, $fullpkgpath, $cdrom, $ftp); } if ($plist->{need_modules}) { print STDERR "Error: some needed modules were not found\n"; $errors++; } unless (defined $opt_q && defined $opt_n) { if ($regen_package) { $plist->verify_checksum($base, {}); } else { my $p2 = OpenBSD::PackingList->new; $plist->makesum_plist($p2, $base, {}); $plist = $p2; } } $plist->avert_duplicates_and_other_checks({}); if ($errors) { exit 1; } if (!defined $plist->{name}) { print STDERR "Can't write unnamed packing list\n"; exit 1; } if (defined $opt_q) { if (defined $opt_Q) { $plist->print_file; } else { $plist->write(\*STDOUT); } exit 0 if defined $opt_n; } my $wname; if ($regen_package) { $wname = $plist->pkgname().".tgz"; } else { $plist->tofile($dir.CONTENTS) or Fatal "Can't write packing list"; $wname = $ARGV[0]; } if ($opt_n) { $plist->comment_create_package(); } else { print "Creating gzip'd tar ball in '$wname'\n" if $opt_v; my $h = sub { unlink $wname; my $caught = shift; $SIG{$caught} = 'DEFAULT'; kill $caught, $$; }; local $SIG{'INT'} = $h; local $SIG{'QUIT'} = $h; local $SIG{'HUP'} = $h; local $SIG{'KILL'} = $h; local $SIG{'TERM'} = $h; open(my $fh, "|gzip >$wname"); my $wrarc = OpenBSD::Ustar->new($fh, $dir); $plist->create_package($wrarc, $base, $opt_v); $wrarc->pad(); close($fh); if ($errors) { unlink($wname); exit(1); } } } catch { print STDERR "$0: $_\n"; exit(1); };