682 lines
13 KiB
Perl
682 lines
13 KiB
Perl
# ex:ts=8 sw=4:
|
|
# $OpenBSD: Port.pm,v 1.25 2011/12/05 21:27:53 espie Exp $
|
|
#
|
|
# Copyright (c) 2010 Marc Espie <espie@openbsd.org>
|
|
#
|
|
# 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 DPB::Job;
|
|
use DPB::Clock;
|
|
package DPB::Task::Port;
|
|
use OpenBSD::Paths;
|
|
|
|
our @ISA = qw(DPB::Task::Clocked);
|
|
sub new
|
|
{
|
|
my ($class, $phase) = @_;
|
|
bless {phase => $phase}, $class;
|
|
}
|
|
|
|
sub name
|
|
{
|
|
my $self = shift;
|
|
return $self->{phase};
|
|
}
|
|
|
|
sub fork
|
|
{
|
|
my ($self, $core) = @_;
|
|
|
|
$core->job->{current} = $self->{phase};
|
|
return $self->SUPER::fork($core);
|
|
}
|
|
|
|
sub finalize
|
|
{
|
|
my ($self, $core) = @_;
|
|
$self->SUPER::finalize($core);
|
|
$core->job->finished_task($self);
|
|
return $core->{status} == 0;
|
|
}
|
|
|
|
sub handle_output
|
|
{
|
|
my ($self, $job) = @_;
|
|
$self->redirect($job->{log});
|
|
}
|
|
|
|
sub run
|
|
{
|
|
my ($self, $core) = @_;
|
|
my $job = $core->job;
|
|
my $t = $self->{phase};
|
|
my $builder = $job->{builder};
|
|
my $ports = $builder->ports;
|
|
my $fullpkgpath = $job->{v}->fullpkgpath;
|
|
my $sudo = OpenBSD::Paths->sudo;
|
|
my $shell = $core->{shell};
|
|
$self->handle_output($job);
|
|
close STDIN;
|
|
open STDIN, '</dev/null';
|
|
print ">>> Running $t in $fullpkgpath\n";
|
|
my @args = ($t, "TRUST_PACKAGES=Yes",
|
|
"FETCH_PACKAGES=No",
|
|
"PREPARE_CHECK_ONLY=Yes",
|
|
"REPORT_PROBLEM='exit 1'", "BULK=No");
|
|
if ($job->{special}) {
|
|
push(@args, "WRKOBJDIR=/tmp/ports");
|
|
}
|
|
if ($builder->{fetch}) {
|
|
push(@args, "NO_CHECKSUM=Yes");
|
|
}
|
|
if (defined $shell) {
|
|
unshift(@args, $builder->make_args);
|
|
if ($self->{sudo}) {
|
|
unshift(@args, $sudo, "-E");
|
|
}
|
|
$shell->run("cd $ports && SUBDIR=".
|
|
$fullpkgpath." ".join(' ', @args));
|
|
} else {
|
|
chdir($ports) or
|
|
die "Wrong ports tree $ports";
|
|
$ENV{SUBDIR} = $fullpkgpath;
|
|
if ($self->{sudo}) {
|
|
exec {$sudo}("sudo", "-E", $builder->make_args, @args);
|
|
} else {
|
|
exec {$builder->make} ($builder->make_args, @args);
|
|
}
|
|
}
|
|
exit(1);
|
|
}
|
|
|
|
sub notime { 0 }
|
|
|
|
package DPB::Task::Port::Serialized;
|
|
our @ISA = qw(DPB::Task::Port);
|
|
|
|
sub junk_lock
|
|
{
|
|
my ($self, $core) = @_;
|
|
my $job = $core->job;
|
|
my $locker = $job->{builder}->locker;
|
|
|
|
while (1) {
|
|
my $fh = $locker->lock($core);
|
|
if ($fh) {
|
|
print $fh "path=".$job->{v}->fullpkgpath, "\n";
|
|
return;
|
|
}
|
|
sleep 1;
|
|
}
|
|
}
|
|
|
|
sub junk_unlock
|
|
{
|
|
my ($self, $core) = @_;
|
|
|
|
$core->job->{builder}->locker->unlock($core);
|
|
}
|
|
|
|
sub finalize
|
|
{
|
|
my ($self, $core) = @_;
|
|
if ($core->{status} != 0) {
|
|
$self->junk_unlock($core);
|
|
}
|
|
$self->SUPER::finalize($core);
|
|
}
|
|
|
|
package DPB::Task::Port::Signature;
|
|
our @ISA =qw(DPB::Task::Port);
|
|
|
|
sub notime { 1 }
|
|
|
|
sub run
|
|
{
|
|
my ($self, $core) = @_;
|
|
my $job = $core->job;
|
|
exit($job->{builder}->check_signature($core, $job->{v}));
|
|
}
|
|
|
|
sub finalize
|
|
{
|
|
my ($self, $core) = @_;
|
|
$self->SUPER::finalize($core);
|
|
my $job = $core->job;
|
|
if ($core->{status} == 0) {
|
|
my $v = $job->{v};
|
|
my $builder = $job->{builder};
|
|
$job->add_normal_tasks($builder->{dontclean}{$v->pkgpath});
|
|
} else {
|
|
$job->{signature_only} = 1;
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
package DPB::Task::Port::Checksum;
|
|
our @ISA = qw(DPB::Task::Port);
|
|
sub run
|
|
{
|
|
my ($self, $core) = @_;
|
|
my $job = $core->job;
|
|
$self->handle_output($job);
|
|
my $exit = 0;
|
|
for my $dist (values %{$job->{v}{info}{DIST}}) {
|
|
if (!$dist->checksum($dist->filename)) {
|
|
$exit = 1;
|
|
} else {
|
|
unlink($dist->tempfilename);
|
|
}
|
|
}
|
|
exit($exit);
|
|
}
|
|
|
|
package DPB::Task::Port::Depends;
|
|
our @ISA=qw(DPB::Task::Port::Serialized);
|
|
|
|
sub notime { 1 }
|
|
|
|
sub run
|
|
{
|
|
my ($self, $core) = @_;
|
|
my $job = $core->job;
|
|
my $dep = {};
|
|
my $v = $job->{v};
|
|
if (exists $v->{info}{BDEPENDS}) {
|
|
for my $d (values %{$v->{info}{BDEPENDS}}) {
|
|
$dep->{$d->fullpkgname} = 1;
|
|
}
|
|
}
|
|
# recurse for extra stuff
|
|
if (exists $v->{info}{BEXTRA}) {
|
|
for my $two (values %{$v->{info}{BEXTRA}}) {
|
|
if (exists $two->{info}{BDEPENDS}) {
|
|
for my $d (values %{$two->{info}{BDEPENDS}}) {
|
|
$dep->{$d->fullpkgname} = 1;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
$self->junk_lock($core);
|
|
|
|
exit(0) unless %$dep;
|
|
my $sudo = OpenBSD::Paths->sudo;
|
|
my $shell = $core->{shell};
|
|
$self->handle_output($job);
|
|
my @cmd = ('/usr/sbin/pkg_add', '-a');
|
|
if ($job->{builder}->{update}) {
|
|
push(@cmd, "-rqU", "-Dupdate", "-Dupdatedepends");
|
|
}
|
|
if ($job->{builder}->{forceupdate}) {
|
|
push(@cmd, "-Dinstalled");
|
|
}
|
|
print join(' ', @cmd, (sort keys %$dep)), "\n";
|
|
my $path = $job->{builder}->{fullrepo}.'/';
|
|
if (defined $shell) {
|
|
$shell->run(join(' ', "PKG_PATH=$path", $sudo, @cmd,
|
|
(sort keys %$dep)));
|
|
} else {
|
|
$ENV{PKG_PATH} = $path;
|
|
exec{$sudo}($sudo, @cmd, sort keys %$dep);
|
|
}
|
|
exit(1);
|
|
}
|
|
|
|
sub finalize
|
|
{
|
|
my ($self, $core) = @_;
|
|
$core->{status} = 0;
|
|
$self->SUPER::finalize($core);
|
|
return 1;
|
|
}
|
|
|
|
package DPB::Task::Port::Install;
|
|
our @ISA=qw(DPB::Task::Port);
|
|
|
|
sub notime { 1 }
|
|
|
|
sub run
|
|
{
|
|
my ($self, $core) = @_;
|
|
my $job = $core->job;
|
|
my $v = $job->{v};
|
|
|
|
my $sudo = OpenBSD::Paths->sudo;
|
|
$self->handle_output($job);
|
|
my @cmd = ('/usr/sbin/pkg_add');
|
|
if ($job->{builder}->{update}) {
|
|
push(@cmd, "-rqU", "-Dupdate", "-Dupdatedepends");
|
|
}
|
|
if ($job->{builder}->{forceupdate}) {
|
|
push(@cmd, "-Dinstalled");
|
|
}
|
|
print join(' ', @cmd, $v->fullpkgname, "\n");
|
|
my $path = $job->{builder}->{fullrepo}.'/';
|
|
$ENV{PKG_PATH} = $path;
|
|
exec{$sudo}($sudo, @cmd, $v->fullpkgname);
|
|
exit(1);
|
|
}
|
|
|
|
sub finalize
|
|
{
|
|
my ($self, $core) = @_;
|
|
$core->{status} = 0;
|
|
$self->SUPER::finalize($core);
|
|
return 1;
|
|
}
|
|
|
|
package DPB::Task::Port::Prepare;
|
|
our @ISA = qw(DPB::Task::Port::Serialized);
|
|
|
|
package DPB::Task::Port::PrepareResults;
|
|
our @ISA = qw(DPB::Task::Port::Serialized);
|
|
|
|
sub result_filename
|
|
{
|
|
my ($self, $job) = @_;
|
|
return $job->{builder}->logger->log_pkgpath($job->{v}).".tmp";
|
|
}
|
|
|
|
sub handle_output
|
|
{
|
|
my ($self, $job) = @_;
|
|
$self->redirect($self->result_filename($job));
|
|
}
|
|
|
|
sub finalize
|
|
{
|
|
my ($self, $core) = @_;
|
|
|
|
my $job = $core->{job};
|
|
my $v = $job->{v};
|
|
my $file = $self->result_filename($job);
|
|
if (open my $fh, '<', $file) {
|
|
my @r;
|
|
while (<$fh>) {
|
|
# zap headers
|
|
next if m/^\>\>\>\s/ || m/^\=\=\=\>\s/;
|
|
chomp;
|
|
push(@r, $_);
|
|
}
|
|
print {$job->{lock}} "needed=", join(' ', sort @r), "\n";
|
|
close $fh;
|
|
unlink($file);
|
|
} else {
|
|
$core->{status} = 1;
|
|
}
|
|
$self->junk_unlock($core);
|
|
$self->SUPER::finalize($core);
|
|
}
|
|
|
|
package DPB::Task::Port::Uninstall;
|
|
our @ISA=qw(DPB::Task::Port::Serialized);
|
|
|
|
sub notime { 1 }
|
|
|
|
sub run
|
|
{
|
|
my ($self, $core) = @_;
|
|
my $job = $core->job;
|
|
my $v = $job->{v};
|
|
|
|
my $sudo = OpenBSD::Paths->sudo;
|
|
$self->handle_output($job);
|
|
|
|
$self->junk_lock($core);
|
|
my @d = $core->job->{builder}->locker->find_dependencies(
|
|
$core->hostname);
|
|
my @cmd = ('/usr/sbin/pkg_delete', '-aX', @d);
|
|
print join(' ', @cmd, "\n");
|
|
my $shell = $core->{shell};
|
|
if (defined $shell) {
|
|
$shell->run(join(' ', $sudo, @cmd));
|
|
} else {
|
|
exec{$sudo}($sudo, @cmd);
|
|
}
|
|
exit(1);
|
|
}
|
|
|
|
sub finalize
|
|
{
|
|
my ($self, $core) = @_;
|
|
$core->{status} = 0;
|
|
$self->junk_unlock($core);
|
|
$self->SUPER::finalize($core);
|
|
return 1;
|
|
}
|
|
|
|
package DPB::Task::Port::ShowSize;
|
|
our @ISA = qw(DPB::Task::Port);
|
|
|
|
sub fork
|
|
{
|
|
my ($self, $core) = @_;
|
|
$self->{sudo} = 1;
|
|
open($self->{fh}, "-|");
|
|
}
|
|
|
|
sub handle_output
|
|
{
|
|
}
|
|
|
|
sub finalize
|
|
{
|
|
my ($self, $core) = @_;
|
|
my $fh = $self->{fh};
|
|
if ($core->{status} == 0) {
|
|
my $line = <$fh>;
|
|
$line = <$fh>;
|
|
if ($line =~ m/^\s*(\d+)\s+/) {
|
|
my $sz = $1;
|
|
my $job = $core->job;
|
|
$core->job->{wrkdir} = $sz;
|
|
}
|
|
}
|
|
close($fh);
|
|
return 1;
|
|
}
|
|
|
|
package DPB::Task::Port::ShowFakeSize;
|
|
our @ISA = qw(DPB::Task::Port::ShowSize);
|
|
|
|
sub finalize
|
|
{
|
|
my ($self, $core) = @_;
|
|
my $fh = $self->{fh};
|
|
if ($core->{status} == 0) {
|
|
my $line = <$fh>;
|
|
$line = <$fh>;
|
|
if ($line =~ m/^\s*(\d+)\s+/) {
|
|
my $sz = $1;
|
|
my $job = $core->job;
|
|
my $f2 = $job->{builder}->logger->open("size");
|
|
print $f2 $job->{v}->fullpkgpath, " $job->{wrkdir} $sz\n";
|
|
}
|
|
}
|
|
close($fh);
|
|
return 1;
|
|
}
|
|
|
|
|
|
package DPB::Task::Port::Fetch;
|
|
our @ISA = qw(DPB::Task::Port);
|
|
|
|
sub notime { 1 }
|
|
|
|
sub finalize
|
|
{
|
|
my ($self, $core) = @_;
|
|
|
|
# if there's a watch file, then we remove the current size,
|
|
# so that we DON'T take prepare into account.
|
|
my $job = $core->job;
|
|
if (defined $job->{watched}) {
|
|
$job->{watched}->reset_offset;
|
|
}
|
|
$self->SUPER::finalize($core);
|
|
}
|
|
|
|
package DPB::Task::Port::Clean;
|
|
our @ISA = qw(DPB::Task::Port);
|
|
|
|
sub notime { 1 }
|
|
|
|
sub finalize
|
|
{
|
|
my ($self, $core) = @_;
|
|
# didn't clean right, and no sudo yet:
|
|
# run ourselves again (but log the problem)
|
|
if ($core->{status} != 0 && !$self->{sudo}) {
|
|
$self->{sudo} = 1;
|
|
my $job = $core->job;
|
|
unshift(@{$job->{tasks}}, $self);
|
|
my $fh = $job->{builder}->logger->open("clean");
|
|
print $fh $job->{v}->fullpkgpath, "\n";
|
|
$core->{status} = 0;
|
|
return 1;
|
|
}
|
|
$self->SUPER::finalize($core);
|
|
}
|
|
|
|
package DPB::Port::TaskFactory;
|
|
my $repo = {
|
|
default => 'DPB::Task::Port',
|
|
checksum => 'DPB::Task::Port::Checksum',
|
|
clean => 'DPB::Task::Port::Clean',
|
|
prepare => 'DPB::Task::Port::Prepare',
|
|
'show-prepare-results' => 'DPB::Task::Port::PrepareResults',
|
|
fetch => 'DPB::Task::Port::Fetch',
|
|
depends => 'DPB::Task::Port::Depends',
|
|
'show-size' => 'DPB::Task::Port::ShowSize',
|
|
'show-fake-size' => 'DPB::Task::Port::ShowFakeSize',
|
|
'junk' => 'DPB::Task::Port::Uninstall',
|
|
};
|
|
|
|
sub create
|
|
{
|
|
my ($class, $k) = @_;
|
|
my $fw = $repo->{$k};
|
|
$fw //= $repo->{default};
|
|
$fw->new($k);
|
|
}
|
|
|
|
package DPB::Job::Port;
|
|
our @ISA = qw(DPB::Job::Normal);
|
|
|
|
use Time::HiRes qw(time);
|
|
|
|
sub new
|
|
{
|
|
my ($class, $log, $v, $builder, $special, $endcode) = @_;
|
|
my $e;
|
|
if ($builder->{rebuild}) {
|
|
$e = sub { $builder->register_built($v); &$endcode; };
|
|
} else {
|
|
$e = $endcode;
|
|
}
|
|
my $job = bless {
|
|
tasks => [],
|
|
log => $log, v => $v,
|
|
special => $special, current => '',
|
|
builder => $builder, endcode => $e},
|
|
$class;
|
|
|
|
if ($builder->{rebuild}) {
|
|
push(@{$job->{tasks}},
|
|
DPB::Task::Port::Signature->new('signature'));
|
|
} else {
|
|
$job->add_normal_tasks($builder->{dontclean}{$v->pkgpath});
|
|
}
|
|
return $job;
|
|
}
|
|
|
|
sub add_normal_tasks
|
|
{
|
|
my ($self, $dontclean) = @_;
|
|
|
|
my @todo;
|
|
my $builder = $self->{builder};
|
|
if ($builder->{clean}) {
|
|
push @todo, "clean";
|
|
}
|
|
push(@todo, qw(depends prepare show-prepare-results));
|
|
if ($builder->{junk}) {
|
|
if ($builder->{junk_count}++ >= $builder->{junk}) {
|
|
$builder->{junk_count} = 0;
|
|
push(@todo, 'junk');
|
|
}
|
|
}
|
|
if ($builder->{fetch}) {
|
|
push(@todo, qw(checksum));
|
|
} else {
|
|
push(@todo, qw(fetch));
|
|
}
|
|
push(@todo, qw(patch configure build));
|
|
if ($builder->{size}) {
|
|
push @todo, 'show-size';
|
|
}
|
|
push(@todo, qw(fake package));
|
|
if ($builder->{size}) {
|
|
push @todo, 'show-fake-size';
|
|
}
|
|
if (!$dontclean) {
|
|
push @todo, 'clean';
|
|
}
|
|
$self->add_tasks(map {DPB::Port::TaskFactory->create($_)} @todo);
|
|
}
|
|
|
|
sub current_task
|
|
{
|
|
my $self = shift;
|
|
if (@{$self->{tasks}} > 0) {
|
|
return $self->{tasks}[0]{phase};
|
|
} else {
|
|
return "<nothing>";
|
|
}
|
|
}
|
|
|
|
sub pkgpath
|
|
{
|
|
my $self = shift;
|
|
return $self->{v};
|
|
}
|
|
|
|
sub name
|
|
{
|
|
my $self = shift;
|
|
return $self->{v}->fullpkgpath."(".$self->{task}->name.")";
|
|
}
|
|
|
|
sub finished_task
|
|
{
|
|
my ($self, $task) = @_;
|
|
push(@{$self->{done}}, $task);
|
|
}
|
|
|
|
sub finalize
|
|
{
|
|
my $self = shift;
|
|
if ($self->{stuck}) {
|
|
open my $fh, ">>", $self->{log};
|
|
print $fh $self->{stuck}, "\n";
|
|
}
|
|
$self->SUPER::finalize(@_);
|
|
}
|
|
|
|
sub totaltime
|
|
{
|
|
my $self = shift;
|
|
my $t = 0;
|
|
for my $plus (@{$self->{done}}) {
|
|
next if $plus->notime;
|
|
$t += $plus->elapsed;
|
|
}
|
|
return sprintf("%.2f", $t);
|
|
}
|
|
|
|
sub timings
|
|
{
|
|
my $self = shift;
|
|
return join('/', map {sprintf("%s=%.2f", $_->name, $_->elapsed)} @{$self->{done}});
|
|
}
|
|
|
|
my $logsize = {};
|
|
|
|
sub add_build_info
|
|
{
|
|
my ($class, $pkgpath, $host, $time, $sz) = @_;
|
|
$logsize->{$pkgpath} = $sz;
|
|
}
|
|
|
|
sub equates
|
|
{
|
|
my ($class, $h) = @_;
|
|
for my $v (values %$h) {
|
|
next unless defined $logsize->{$v};
|
|
for my $w (values %$h) {
|
|
$logsize->{$w} //= $logsize->{$v};
|
|
}
|
|
return;
|
|
}
|
|
}
|
|
|
|
sub set_watch
|
|
{
|
|
my ($self, $logger, $v) = @_;
|
|
my $expected;
|
|
for my $w ($logger->pathlist($v)) {
|
|
if (defined $logsize->{$w}) {
|
|
$expected = $logsize->{$w};
|
|
last;
|
|
}
|
|
}
|
|
$self->{watched} = DPB::Watch->new($logger->log_pkgpath($v),
|
|
$expected, $self->{offset}, $self->{started});
|
|
}
|
|
|
|
sub watched
|
|
{
|
|
my ($self, $current, $core) = @_;
|
|
return "" unless defined $self->{watched};
|
|
my $diff = $self->{watched}->check_change($current);
|
|
my $msg = $self->{watched}->change_message($diff);
|
|
my $stuck = $core->stuck_timeout;
|
|
if (defined $stuck) {
|
|
if ($diff > $stuck) {
|
|
$self->{stuck} =
|
|
"KILLED: $self->{current} stuck at $msg";
|
|
kill 9, $core->{pid};
|
|
return $self->{stuck};
|
|
}
|
|
}
|
|
return $msg;
|
|
}
|
|
|
|
sub really_watch
|
|
{
|
|
my ($self, $current) = @_;
|
|
return "" unless defined $self->{watched};
|
|
my $diff = $self->{watched}->check_change($current);
|
|
$self->{lastdiff} //= 5;
|
|
if ($diff > $self->{lastdiff} * 2) {
|
|
$self->{lastdiff} = $diff;
|
|
return 1;
|
|
} elsif ($diff < $self->{lastdiff}) {
|
|
$self->{lastdiff} = 5;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
package DPB::Job::Port::Install;
|
|
our @ISA = qw(DPB::Job::Port);
|
|
|
|
sub new
|
|
{
|
|
my ($class, $log, $v, $builder, $e) = @_;
|
|
my $job = bless {
|
|
tasks => [],
|
|
log => $log, v => $v,
|
|
builder => $builder, endcode => $e},
|
|
$class;
|
|
|
|
push(@{$job->{tasks}},
|
|
DPB::Task::Port::Install->new('install'));
|
|
return $job;
|
|
}
|
|
|
|
1;
|
|
|