openbsd-ports/infrastructure/lib/DPB/Job/Port.pm

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;