# ex:ts=8 sw=4: # $OpenBSD: PackageRepository.pm,v 1.38 2007/05/19 09:18:55 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; # XXX load extra class, grab match from Base class, and tweak inheritance # to get all methods. use OpenBSD::PackageRepository::Installed; $OpenBSD::PackageRepository::Installed::ISA=(qw(OpenBSD::PackageRepository)); package OpenBSD::PackageRepository; our @ISA=(qw(OpenBSD::PackageRepositoryBase)); use OpenBSD::PackageLocation; sub _new { my ($class, $address) = @_; bless { baseurl => $address }, $class; } sub new { my ($class, $baseurl) = @_; if ($baseurl =~ m/^ftp\:/i) { return OpenBSD::PackageRepository::FTP->_new($'); } elsif ($baseurl =~ m/^http\:/i) { return OpenBSD::PackageRepository::HTTP->_new($'); } elsif ($baseurl =~ m/^scp\:/i) { require OpenBSD::PackageRepository::SCP; return OpenBSD::PackageRepository::SCP->_new($'); } elsif ($baseurl =~ m/^src\:/i) { require OpenBSD::PackageRepository::Source; return OpenBSD::PackageRepository::Source->_new($'); } elsif ($baseurl =~ m/^file\:/i) { return OpenBSD::PackageRepository::Local->_new($'); } elsif ($baseurl =~ m/^inst\:/i) { return OpenBSD::PackageRepository::Installed->_new($'); } else { return OpenBSD::PackageRepository::Local->_new($baseurl); } } sub available { my $self = shift; return @{$self->list}; } sub stemlist { my $self = shift; if (!defined $self->{stemlist}) { require OpenBSD::PackageName; $self->{stemlist} = OpenBSD::PackageName::avail2stems($self->available); } return $self->{stemlist}; } sub wipe_info { my ($self, $pkg) = @_; require File::Path; my $dir = $pkg->{dir}; if (defined $dir) { File::Path::rmtree($dir); delete $pkg->{dir}; } } # by default, all objects may exist sub may_exist { return 1; } # by default, we don't track opened files for this key sub opened { undef; } # hint: 0 premature close, 1 real error. undef, normal ! sub close { my ($self, $object, $hint) = @_; close($object->{fh}) if defined $object->{fh}; waitpid($object->{pid2}, 0) if defined $object->{pid2}; $self->parse_problems($object->{errors}, $hint) if defined $object->{errors}; undef $object->{errors}; $object->deref(); } sub finish_and_close { my ($self, $object) = @_; $self->close($object); } sub close_now { my ($self, $object) = @_; $self->close($object, 0); } sub close_after_error { my ($self, $object) = @_; $self->close($object, 1); } sub close_with_client_error { my ($self, $object) = @_; $self->close($object, 1); } sub make_room { my $self = shift; # kill old files if too many my $already = $self->opened; if (defined $already) { # gc old objects if (@$already >= $self->maxcount) { @$already = grep { defined $_->{fh} } @$already; } while (@$already >= $self->maxcount) { my $o = shift @$already; $self->close_now($o); } } return $already; } # open method that tracks opened files per-host. sub open { my ($self, $object) = @_; return unless $self->may_exist($object->{name}); # kill old files if too many my $already = $self->make_room; my $fh = $self->open_pipe($object); if (!defined $fh) { return; } $object->{fh} = $fh; if (defined $already) { push @$already, $object; } return $fh; } sub find { my ($repository, $name, $arch) = @_; my $self = OpenBSD::PackageLocation->new($repository, $name, $arch); return $self->openPackage; } sub grabPlist { my ($repository, $name, $arch, $code) = @_; my $self = OpenBSD::PackageLocation->new($repository, $name, $arch); return $self->grabPlist($code); } sub parse_problems { my ($self, $filename, $hint) = @_; unlink $filename; } sub cleanup { # nothing to do } sub relative_url { my ($self, $name) = @_; if (defined $name) { return $self->{baseurl}.$name.".tgz"; } else { return $self->{baseurl}; } } package OpenBSD::PackageRepository::Local; our @ISA=qw(OpenBSD::PackageRepository); sub urlscheme { return 'file'; } sub open_pipe { my ($self, $object) = @_; my $pid = open(my $fh, "-|"); if (!defined $pid) { die "Cannot fork: $!"; } if ($pid) { return $fh; } else { open STDERR, ">/dev/null"; exec {"/usr/bin/gzip"} "gzip", "-d", "-c", "-q", "-f", $self->relative_url($object->{name}) or die "Can't run gzip"; } } sub may_exist { my ($self, $name) = @_; return -r $self->relative_url($name); } sub list { my $self = shift; my $l = []; my $dname = $self->{baseurl}; opendir(my $dir, $dname) or return $l; while (my $e = readdir $dir) { next unless $e =~ m/\.tgz$/; next unless -f "$dname/$e"; push(@$l, $`); } close($dir); return $l; } package OpenBSD::PackageRepository::Local::Pipe; our @ISA=qw(OpenBSD::PackageRepository::Local); sub urlscheme { return 'pipe'; } sub relative_url { return ''; } sub may_exist { return 1; } sub open_pipe { my ($self, $object) = @_; my $pid = open(my $fh, "-|"); if (!defined $pid) { die "Cannot fork: $!"; } if ($pid) { return $fh; } else { open STDERR, ">/dev/null"; exec {"/usr/bin/gzip"} "gzip", "-d", "-c", "-q", "-f", "-" or die "can't run gzip"; } } package OpenBSD::PackageRepository::Distant; our @ISA=qw(OpenBSD::PackageRepository); my $buffsize = 2 * 1024 * 1024; sub pkg_copy { my ($self, $in, $object) = @_; require OpenBSD::Temp; my $name = $object->{name}; my $dir = $object->{cache_dir}; my ($copy, $filename) = OpenBSD::Temp::permanent_file($dir, $name) or die "Can't write copy to cache"; chmod 0644, $filename; $object->{tempname} = $filename; my $handler = sub { my ($sig) = @_; unlink $filename; close($in); $SIG{$sig} = 'DEFAULT'; kill $sig, $$; }; my $nonempty = 0; my $error = 0; { local $SIG{'PIPE'} = $handler; local $SIG{'INT'} = $handler; local $SIG{'HUP'} = $handler; local $SIG{'QUIT'} = $handler; local $SIG{'KILL'} = $handler; local $SIG{'TERM'} = $handler; my ($buffer, $n); # copy stuff over do { $n = sysread($in, $buffer, $buffsize); if (!defined $n) { die "Error reading\n"; } if ($n > 0) { $nonempty = 1; } if (!$error) { my $r = syswrite $copy, $buffer; if (!defined $r || $r < $n) { $error = 1; } } syswrite STDOUT, $buffer; } while ($n != 0); close($copy); } if ($nonempty && !$error) { rename $filename, "$dir/$name.tgz"; } else { unlink $filename; } close($in); } sub open_pipe { require OpenBSD::Temp; my ($self, $object) = @_; $object->{errors} = OpenBSD::Temp::file(); $object->{cache_dir} = $ENV{'PKG_CACHE'}; $object->{parent} = $$; my ($rdfh, $wrfh); pipe($rdfh, $wrfh); my $pid = open(my $fh, "-|"); if (!defined $pid) { die "Cannot fork: $!"; } if ($pid) { $object->{pid} = $pid; } else { open(STDIN, '<&', $rdfh) or die "Bad dup"; close($rdfh); close($wrfh); exec {"/usr/bin/gzip"} "gzip", "-d", "-c", "-q", "-" or die "can't run gzip"; } my $pid2 = fork(); if (!defined $pid2) { die "Cannot fork: $!"; } if ($pid2) { $object->{pid2} = $pid2; } else { open STDERR, '>', $object->{errors}; open(STDOUT, '>&', $wrfh) or die "Bad dup"; close($rdfh); close($wrfh); close($fh); if (defined $object->{cache_dir}) { my $pid3 = open(my $in, "-|"); if (!defined $pid3) { die "Cannot fork: $!"; } if ($pid3) { $self->pkg_copy($in, $object); } else { $self->grab_object($object); } } else { $self->grab_object($object); } exit(0); } close($rdfh); close($wrfh); return $fh; } sub _list { my ($self, $cmd) = @_; my $l =[]; local $_; open(my $fh, '-|', "$cmd") or return; while(<$fh>) { chomp; next if m/^d.*\s+\S/; next unless m/([^\s]+)\.tgz\s*$/; push(@$l, $1); } close($fh); return $l; } sub finish_and_close { my ($self, $object) = @_; if (defined $object->{cache_dir}) { while (defined $object->intNext) { } } $self->SUPER::finish_and_close($object); } package OpenBSD::PackageRepository::HTTPorFTP; our @ISA=qw(OpenBSD::PackageRepository::Distant); our %distant = (); sub grab_object { my ($self, $object) = @_; my $ftp = defined $ENV{'FETCH_CMD'} ? $ENV{'FETCH_CMD'} : "/usr/bin/ftp"; exec {$ftp} "ftp", "-o", "-", $self->url($object->{name}) or die "can't run ftp"; } sub maxcount { return 1; } sub opened { my $self = $_[0]; my $k = $self->{key}; if (!defined $distant{$k}) { $distant{$k} = []; } return $distant{$k}; } sub _new { my ($class, $baseurl) = @_; my $distant_host; if ($baseurl =~ m/^\/\/(.*?)\//i) { $distant_host = $&; } bless { baseurl => $baseurl, key => $distant_host }, $class; } sub should_have { my ($self, $pkgname) = @_; if (defined $self->{lasterror} && $self->{lasterror} == 421) { return (defined $self->{list}) && grep { $_ eq $pkgname } @{$self->{list}}; } else { return 0; } } sub try_until_success { my ($self, $pkgname, $code) = @_; for (my $retry = 5; $retry < 60; $retry *= 2) { undef $self->{lasterror}; my $o = &$code; if (defined $o) { return $o; } if ($self->should_have($pkgname)) { print STDERR "Temporary error, sleeping $retry seconds\n"; sleep($retry); } } return undef; } sub find { my ($self, $pkgname, @extra) = @_; return $self->try_until_success($pkgname, sub { return $self->SUPER::find($pkgname, @extra); }); } sub grabPlist { my ($self, $pkgname, @extra) = @_; return $self->try_until_success($pkgname, sub { return $self->SUPER::grabPlist($pkgname, @extra); }); } sub parse_problems { my ($self, $filename, $hint) = @_; CORE::open(my $fh, '<', $filename) or return; my $baseurl = $self->url; local $_; my $notyet = 1; while(<$fh>) { next if m/^(?:200|220|221|226|229|230|227|250|331|500|150)[\s\-]/; next if m/^EPSV command not understood/; next if m/^Trying [\da-f\.\:]+\.\.\./; next if m/^Requesting \Q$baseurl\E/; next if m/^Remote system type is\s+/; next if m/^Connected to\s+/; next if m/^remote\:\s+/; next if m/^Using binary mode to transfer files/; next if m/^Retrieving\s+/; next if m/^Success?fully retrieved file/; next if m/^\d+\s+bytes\s+received\s+in/; next if m/^ftp: connect to address.*: No route to host/; if (defined $hint && $hint == 0) { next if m/^ftp: -: short write/; next if m/^421\s+/; } if ($notyet) { print STDERR "Error from $baseurl:\n" if $notyet; $notyet = 0; } if (m/^421\s+/ || m/^ftp: connect: Connection timed out/ || m/^ftp: Can't connect or login to host/) { $self->{lasterror} = 421; } if (m/^550\s+/) { $self->{lasterror} = 550; } print STDERR $_; } CORE::close($fh); $self->SUPER::parse_problems($filename, $hint); } package OpenBSD::PackageRepository::HTTP; our @ISA=qw(OpenBSD::PackageRepository::HTTPorFTP); sub urlscheme { return 'http'; } sub list { my ($self) = @_; if (!defined $self->{list}) { my $error = OpenBSD::Temp::file(); $self->make_room; my $fullname = $self->url; my $l = $self->{list} = []; local $_; open(my $fh, '-|', "ftp -o - $fullname 2>$error") or return; # XXX assumes a pkg HREF won't cross a line. Is this the case ? while(<$fh>) { chomp; for my $pkg (m/\/gi) { next if $pkg =~ m|/|; push(@$l, $pkg); } } close($fh); $self->parse_problems($error); } return $self->{list}; } package OpenBSD::PackageRepository::FTP; our @ISA=qw(OpenBSD::PackageRepository::HTTPorFTP); sub urlscheme { return 'ftp'; } sub list { my ($self) = @_; if (!defined $self->{list}) { require OpenBSD::Temp; my $error = OpenBSD::Temp::file(); $self->make_room; my $fullname = $self->url; $self->{list} = $self->_list("echo 'nlist *.tgz'|ftp -o - $fullname 2>$error"); $self->parse_problems($error); } return $self->{list}; } 1;