| 
									
										
										
										
											2011-11-30 17:32:50 +01:00
										 |  |  | #! /var/run/current-system/sw/bin/perl -w | 
					
						
							| 
									
										
										
										
											2008-11-10 13:33:12 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | use strict; | 
					
						
							|  |  |  | use File::Basename; | 
					
						
							| 
									
										
										
										
											2008-11-12 14:29:32 +00:00
										 |  |  | use File::stat; | 
					
						
							| 
									
										
										
										
											2011-11-30 17:22:31 +01:00
										 |  |  | use Nix::Store; | 
					
						
							| 
									
										
										
										
											2008-11-25 11:09:15 +00:00
										 |  |  | use Hydra::Schema; | 
					
						
							| 
									
										
										
										
											2008-11-28 14:36:04 +00:00
										 |  |  | use Hydra::Helper::Nix; | 
					
						
							| 
									
										
										
										
											2010-09-01 08:52:54 +00:00
										 |  |  | use Hydra::Helper::AddBuilds; | 
					
						
							| 
									
										
										
										
											2009-07-07 16:15:38 +00:00
										 |  |  | use Email::Sender::Simple qw(sendmail); | 
					
						
							|  |  |  | use Email::Sender::Transport::SMTP; | 
					
						
							|  |  |  | use Email::Simple; | 
					
						
							|  |  |  | use Email::Simple::Creator; | 
					
						
							| 
									
										
										
										
											2009-07-08 15:52:55 +00:00
										 |  |  | use Sys::Hostname::Long; | 
					
						
							| 
									
										
										
										
											2009-07-10 09:47:42 +00:00
										 |  |  | use Config::General; | 
					
						
							| 
									
										
										
										
											2009-08-12 13:31:27 +00:00
										 |  |  | use Text::Table; | 
					
						
							|  |  |  | use POSIX qw(strftime); | 
					
						
							| 
									
										
										
										
											2009-10-15 13:23:15 +00:00
										 |  |  | use Net::Twitter::Lite; | 
					
						
							| 
									
										
										
										
											2010-01-19 14:15:31 +00:00
										 |  |  | use Data::Dump qw(dump); | 
					
						
							| 
									
										
										
										
											2012-03-12 20:41:45 +01:00
										 |  |  | use feature qw/switch/; | 
					
						
							| 
									
										
										
										
											2008-11-10 13:33:12 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-04-22 22:59:54 +00:00
										 |  |  | STDOUT->autoflush(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-11-28 14:36:04 +00:00
										 |  |  | my $db = openHydraDB; | 
					
						
							| 
									
										
										
										
											2008-11-10 13:33:12 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-03-07 15:06:32 +00:00
										 |  |  | my %config = new Config::General(getHydraConf)->getall; | 
					
						
							| 
									
										
										
										
											2009-07-10 09:47:42 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-15 13:23:15 +00:00
										 |  |  | sub sendTwitterNotification { | 
					
						
							|  |  |  |     my ($build) = @_; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return unless (defined $ENV{'TWITTER_USER'} && defined $ENV{'TWITTER_PASS'}); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-08 10:45:42 +00:00
										 |  |  |     my $addURL = defined $config{'base_uri'}; | 
					
						
							| 
									
										
										
										
											2009-10-15 13:23:15 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     my $jobName  = $build->project->name . ":" . $build->jobset->name . ":" . $build->job->name; | 
					
						
							| 
									
										
										
										
											2012-03-05 21:52:47 +01:00
										 |  |  |     my $status   = $build->buildstatus == 0 ? "SUCCEEDED" : "FAILED"; | 
					
						
							| 
									
										
										
										
											2009-10-15 13:23:15 +00:00
										 |  |  |     my $system   = $build->system; | 
					
						
							| 
									
										
										
										
											2012-03-05 21:52:47 +01:00
										 |  |  |     my $duration = ($build->stoptime - $build->starttime) . " seconds";  | 
					
						
							| 
									
										
										
										
											2010-10-08 10:45:42 +00:00
										 |  |  |     my $url = $config{'base_uri'}."/build/".$build->id ; | 
					
						
							| 
									
										
										
										
											2009-10-15 13:23:15 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     my $nt = Net::Twitter::Lite->new( | 
					
						
							|  |  |  |         username => $ENV{'TWITTER_USER'}, | 
					
						
							| 
									
										
										
										
											2009-10-15 13:35:52 +00:00
										 |  |  |         password => $ENV{'TWITTER_PASS'}, | 
					
						
							|  |  |  |         clientname => "Hydra Build Daemon" | 
					
						
							| 
									
										
										
										
											2009-10-15 13:23:15 +00:00
										 |  |  |       ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     my $tag = $build->project->name; | 
					
						
							|  |  |  |     my $msg = "$jobName ($system): $status in $duration #$tag";  | 
					
						
							|  |  |  |     if (length($msg) + 1 + length($url) <= 140) { | 
					
						
							|  |  |  |       $msg = "$msg $url" ; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     eval { | 
					
						
							|  |  |  |         my $result = eval { $nt->update($msg) }; | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  |     warn "$@\n" if $@; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2009-03-26 12:53:39 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-04-26 14:36:56 +00:00
										 |  |  | sub statusDescription { | 
					
						
							|  |  |  |     my ($buildstatus) = @_; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-03-15 11:51:00 +00:00
										 |  |  |     my $status = "Failed"; | 
					
						
							| 
									
										
										
										
											2012-03-12 20:41:45 +01:00
										 |  |  |     given ($buildstatus) { | 
					
						
							|  |  |  |         when (0) { $status = "Success"; } | 
					
						
							|  |  |  |         when (1) { $status = "Failed with non-zero exit code"; } | 
					
						
							|  |  |  |         when (2) { $status = "Dependency failed"; } | 
					
						
							|  |  |  |         when (4) { $status = "Cancelled"; } | 
					
						
							| 
									
										
										
										
											2010-04-26 14:36:56 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |    return $status; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-07 16:15:38 +00:00
										 |  |  | sub sendEmailNotification { | 
					
						
							|  |  |  |     my ($build) = @_; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-05 21:52:47 +01:00
										 |  |  |     die unless $build->finished; | 
					
						
							| 
									
										
										
										
											2009-07-08 15:52:55 +00:00
										 |  |  |          | 
					
						
							| 
									
										
										
										
											2010-09-01 10:50:57 +00:00
										 |  |  |     return if ! ( $build->jobset->enableemail && ($build->maintainers ne "" || $build->jobset->emailoverride ne "") ); | 
					
						
							| 
									
										
										
										
											2009-07-07 16:15:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 15:52:55 +00:00
										 |  |  |     # Do we want to send mail? | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-06 13:07:59 +00:00
										 |  |  |     my $prevBuild; | 
					
						
							|  |  |  |     ($prevBuild) = $db->resultset('Builds')->search( | 
					
						
							|  |  |  | 	{ project => $build->project->name | 
					
						
							|  |  |  |         , jobset => $build->jobset->name | 
					
						
							|  |  |  |         , job => $build->job->name | 
					
						
							|  |  |  |         , system => $build->system  | 
					
						
							|  |  |  |         , finished => 1 | 
					
						
							| 
									
										
										
										
											2012-03-07 19:37:45 +01:00
										 |  |  |         , id => { '<', $build->id } | 
					
						
							|  |  |  | 	}, { order_by => ["id DESC"] } | 
					
						
							| 
									
										
										
										
											2010-01-06 13:07:59 +00:00
										 |  |  |         ); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-03-15 12:12:56 +00:00
										 |  |  |     # if there is a previous build with same buildstatus, do not send email | 
					
						
							| 
									
										
										
										
											2012-03-05 21:52:47 +01:00
										 |  |  |     if (defined $prevBuild && ($build->buildstatus == $prevBuild->buildstatus)) { | 
					
						
							| 
									
										
										
										
											2010-01-06 13:07:59 +00:00
										 |  |  |         return;  | 
					
						
							| 
									
										
										
										
											2009-07-08 15:52:55 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-14 11:48:44 +01:00
										 |  |  |     # if buildstatus of this build or the previous one is aborted, do | 
					
						
							|  |  |  |     # not send email | 
					
						
							| 
									
										
										
										
											2012-03-05 21:52:47 +01:00
										 |  |  |     if ($build->buildstatus == 3 || (defined $prevBuild && ($prevBuild->buildstatus == 3))) { | 
					
						
							| 
									
										
										
										
											2011-03-15 12:12:56 +00:00
										 |  |  |         return;  | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 15:52:55 +00:00
										 |  |  |     # Send mail. | 
					
						
							|  |  |  |     # !!! should use the Template Toolkit here. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     print STDERR "sending mail notification to ", $build->maintainers, "\n"; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     my $jobName = $build->project->name . ":" . $build->jobset->name . ":" . $build->job->name; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-05 21:52:47 +01:00
										 |  |  |     my $status =  statusDescription($build->buildstatus); | 
					
						
							| 
									
										
										
										
											2009-07-08 15:52:55 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-04-27 08:01:38 +00:00
										 |  |  |     my $baseurl = hostname_long ; | 
					
						
							| 
									
										
										
										
											2009-07-24 18:06:34 +00:00
										 |  |  |     my $sender = $config{'notification_sender'} || | 
					
						
							| 
									
										
										
										
											2010-04-27 08:01:38 +00:00
										 |  |  |         (($ENV{'USER'} || "hydra") .  "@" . $baseurl); | 
					
						
							| 
									
										
										
										
											2009-07-08 15:52:55 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-10 09:47:42 +00:00
										 |  |  |     my $selfURI = $config{'base_uri'} || "http://localhost:3000"; | 
					
						
							| 
									
										
										
										
											2009-07-08 15:52:55 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-12 13:31:27 +00:00
										 |  |  |     sub showTime { my ($x) = @_; return strftime('%Y-%m-%d %H:%M:%S', localtime($x)); } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-12 13:50:59 +00:00
										 |  |  |     my $infoTable = Text::Table->new({ align => "left" }, \ " | ", { align => "left" }); | 
					
						
							| 
									
										
										
										
											2009-08-12 13:31:27 +00:00
										 |  |  |     my @lines = ( | 
					
						
							|  |  |  |         [ "Build ID:", $build->id ], | 
					
						
							|  |  |  |         [ "Nix name:", $build->nixname ], | 
					
						
							|  |  |  |         [ "Short description:", $build->description || '(not given)' ], | 
					
						
							|  |  |  |         [ "Maintainer(s):", $build->maintainers ], | 
					
						
							|  |  |  |         [ "System:", $build->system ], | 
					
						
							|  |  |  |         [ "Derivation store path:", $build->drvpath ], | 
					
						
							|  |  |  |         [ "Output store path:", $build->outpath ], | 
					
						
							|  |  |  |         [ "Time added:", showTime $build->timestamp ], | 
					
						
							|  |  |  |         ); | 
					
						
							|  |  |  |     push @lines, ( | 
					
						
							| 
									
										
										
										
											2012-03-05 21:52:47 +01:00
										 |  |  |         [ "Build started:", showTime $build->starttime ], | 
					
						
							|  |  |  |         [ "Build finished:", showTime $build->stoptime ], | 
					
						
							|  |  |  |         [ "Duration:", $build->stoptime - $build->starttime . "s" ], | 
					
						
							|  |  |  |     ) if $build->starttime; | 
					
						
							| 
									
										
										
										
											2009-08-12 13:31:27 +00:00
										 |  |  |     $infoTable->load(@lines); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     my $inputsTable = Text::Table->new( | 
					
						
							| 
									
										
										
										
											2009-08-12 13:50:59 +00:00
										 |  |  |         { title => "Name", align => "left" }, \ " | ", | 
					
						
							|  |  |  |         { title => "Type", align => "left" }, \ " | ", | 
					
						
							| 
									
										
										
										
											2009-08-12 13:31:27 +00:00
										 |  |  |         { title => "Value", align => "left" }); | 
					
						
							|  |  |  |     @lines = (); | 
					
						
							|  |  |  |     foreach my $input ($build->inputs) { | 
					
						
							|  |  |  |         my $type = $input->type; | 
					
						
							|  |  |  |         push @lines, | 
					
						
							|  |  |  |             [ $input->name | 
					
						
							|  |  |  |             , $input->type | 
					
						
							| 
									
										
										
										
											2010-01-19 14:15:31 +00:00
										 |  |  |             , ( $input->type eq "build" || $input->type eq "sysbuild") | 
					
						
							| 
									
										
										
										
											2009-08-12 13:31:27 +00:00
										 |  |  |               ? $input->dependency->id | 
					
						
							|  |  |  |               : ($input->type eq "string" || $input->type eq "boolean") | 
					
						
							|  |  |  |               ? $input->value : ($input->uri . ':' . $input->revision) | 
					
						
							|  |  |  |             ]; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     $inputsTable->load(@lines); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-06 13:57:41 +00:00
										 |  |  |     my $loglines = 50; | 
					
						
							| 
									
										
										
										
											2012-03-05 21:52:47 +01:00
										 |  |  |     my $logfile = $build->logfile; | 
					
						
							| 
									
										
										
										
											2011-06-10 10:48:51 +00:00
										 |  |  |     my $logtext = defined $logfile && -e $logfile ? `tail -$loglines $logfile` : "No logfile available.\n"; | 
					
						
							|  |  |  |     $logtext = removeAsciiEscapes($logtext); | 
					
						
							| 
									
										
										
										
											2010-01-06 13:57:41 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 15:52:55 +00:00
										 |  |  |     my $body = "Hi,\n" | 
					
						
							|  |  |  |         . "\n" | 
					
						
							| 
									
										
										
										
											2010-02-10 12:27:16 +00:00
										 |  |  |         . "This is to let you know that Hydra build " . $build->id | 
					
						
							| 
									
										
										
										
											2012-03-05 21:52:47 +01:00
										 |  |  |         . " of job " . $jobName . " "  . (defined $prevBuild ? "has changed from '" . statusDescription($prevBuild->buildstatus) . "' to '$status'" : "is '$status'" ) .".\n" | 
					
						
							| 
									
										
										
										
											2009-07-08 15:52:55 +00:00
										 |  |  |         . "\n" | 
					
						
							| 
									
										
										
										
											2009-08-12 13:50:59 +00:00
										 |  |  |         . "Complete build information can be found on this page: " | 
					
						
							| 
									
										
										
										
											2009-07-08 15:52:55 +00:00
										 |  |  |         . "$selfURI/build/" . $build->id . "\n" | 
					
						
							| 
									
										
										
										
											2012-03-05 21:52:47 +01:00
										 |  |  |         . ($build->buildstatus != 0 ? "\nThe last $loglines lines of the build log are shown at the bottom of this email.\n" : "") | 
					
						
							| 
									
										
										
										
											2009-07-08 15:52:55 +00:00
										 |  |  |         . "\n" | 
					
						
							| 
									
										
										
										
											2009-08-12 13:31:27 +00:00
										 |  |  |         . "A summary of the build information follows:\n" | 
					
						
							|  |  |  |         . "\n" | 
					
						
							|  |  |  |         . $infoTable->body | 
					
						
							|  |  |  |         . "\n" | 
					
						
							|  |  |  |         . "The build inputs were:\n" | 
					
						
							|  |  |  |         . "\n" | 
					
						
							|  |  |  |         . $inputsTable->title | 
					
						
							|  |  |  |         . $inputsTable->rule('-', '+') | 
					
						
							|  |  |  |         . $inputsTable->body | 
					
						
							|  |  |  |         . "\n" | 
					
						
							| 
									
										
										
										
											2010-01-06 13:57:41 +00:00
										 |  |  |         . "Regards,\n\nThe Hydra build daemon.\n" | 
					
						
							| 
									
										
										
										
											2012-03-05 21:52:47 +01:00
										 |  |  |         . ($build->buildstatus != 0 ? "\n---\n$logtext" : "");  | 
					
						
							| 
									
										
										
										
											2010-01-06 13:57:41 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-06 13:07:59 +00:00
										 |  |  |     # stripping trailing spaces from lines | 
					
						
							|  |  |  |     $body =~ s/[\ ]+$//gm; | 
					
						
							| 
									
										
										
										
											2009-07-08 15:52:55 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-06 13:07:59 +00:00
										 |  |  |     my $to = (!$build->jobset->emailoverride eq "") ? $build->jobset->emailoverride : $build->maintainers; | 
					
						
							| 
									
										
										
										
											2009-07-07 16:15:38 +00:00
										 |  |  |     my $email = Email::Simple->create( | 
					
						
							|  |  |  |         header => [ | 
					
						
							| 
									
										
										
										
											2010-01-06 13:07:59 +00:00
										 |  |  |             To      => $to, | 
					
						
							| 
									
										
										
										
											2009-07-08 15:52:55 +00:00
										 |  |  |             From    => "Hydra Build Daemon <$sender>", | 
					
						
							| 
									
										
										
										
											2012-01-31 18:27:33 +01:00
										 |  |  |             Subject => "Hydra job $jobName on " . $build->system . ", build " . $build->id . ": $status", | 
					
						
							| 
									
										
										
										
											2010-04-27 08:01:38 +00:00
										 |  |  |             'X-Hydra-Instance' => $baseurl, | 
					
						
							| 
									
										
										
										
											2010-04-26 14:36:56 +00:00
										 |  |  |             'X-Hydra-Project'  => $build->project->name, | 
					
						
							|  |  |  |             'X-Hydra-Jobset'   => $build->jobset->name, | 
					
						
							| 
									
										
										
										
											2012-01-31 18:27:33 +01:00
										 |  |  |             'X-Hydra-Job'      => $build->job->name, | 
					
						
							|  |  |  |             'X-Hydra-System'   => $build->system | 
					
						
							| 
									
										
										
										
											2009-07-07 16:15:38 +00:00
										 |  |  |         ], | 
					
						
							| 
									
										
										
										
											2010-01-07 13:53:05 +00:00
										 |  |  |         body => "", | 
					
						
							| 
									
										
										
										
											2009-07-07 16:15:38 +00:00
										 |  |  |     ); | 
					
						
							| 
									
										
										
										
											2010-01-07 13:53:05 +00:00
										 |  |  |     $email->body_set($body); | 
					
						
							| 
									
										
										
										
											2009-07-07 16:15:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 15:52:55 +00:00
										 |  |  |     print $email->as_string if $ENV{'HYDRA_MAIL_TEST'}; | 
					
						
							| 
									
										
										
										
											2009-07-07 16:15:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-31 14:08:59 +00:00
										 |  |  |     sendmail($email); | 
					
						
							| 
									
										
										
										
											2009-07-07 16:15:38 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-11-11 12:54:37 +00:00
										 |  |  | sub doBuild { | 
					
						
							|  |  |  |     my ($build) = @_; | 
					
						
							| 
									
										
										
										
											2008-11-10 13:33:12 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-05-26 08:03:59 +00:00
										 |  |  |     my $drvPath   = $build->drvpath; | 
					
						
							|  |  |  |     my $outPath   = $build->outpath; | 
					
						
							|  |  |  |     my $maxsilent = $build->maxsilent; | 
					
						
							|  |  |  |     my $timeout   = $build->timeout; | 
					
						
							| 
									
										
										
										
											2008-11-10 13:33:12 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     my $isCachedBuild = 1; | 
					
						
							|  |  |  |     my $outputCreated = 1; # i.e., the Nix build succeeded (but it could be a positive failure) | 
					
						
							|  |  |  |     my $startTime = 0; | 
					
						
							|  |  |  |     my $stopTime = 0; | 
					
						
							| 
									
										
										
										
											2008-11-25 00:38:16 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     my $buildStatus = 0; # = succeeded | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     my $errormsg = undef; | 
					
						
							| 
									
										
										
										
											2009-03-09 17:21:10 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-11-10 13:33:12 +00:00
										 |  |  |     if (!isValidPath($outPath)) { | 
					
						
							|  |  |  |         $isCachedBuild = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-03-09 16:22:41 +00:00
										 |  |  |         # Do the build. | 
					
						
							| 
									
										
										
										
											2008-11-10 13:33:12 +00:00
										 |  |  |         $startTime = time(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-11-25 00:38:16 +00:00
										 |  |  |         my $thisBuildFailed = 0; | 
					
						
							|  |  |  |         my $someBuildFailed = 0; | 
					
						
							|  |  |  |          | 
					
						
							| 
									
										
										
										
											2008-11-11 17:49:50 +00:00
										 |  |  |         # Run Nix to perform the build, and monitor the stderr output | 
					
						
							|  |  |  |         # to get notifications about specific build steps, the | 
					
						
							|  |  |  |         # associated log files, etc. | 
					
						
							| 
									
										
										
										
											2011-07-04 13:55:02 +00:00
										 |  |  | 	# Note: `--timeout' was added in Nix 1.0pre27564, June 2011. | 
					
						
							| 
									
										
										
										
											2010-11-22 12:20:04 +00:00
										 |  |  |         my $cmd = "nix-store --realise $drvPath " . | 
					
						
							| 
									
										
										
										
											2011-07-04 13:55:02 +00:00
										 |  |  | 	    "--timeout $timeout " . | 
					
						
							| 
									
										
										
										
											2010-05-26 08:03:59 +00:00
										 |  |  |             "--max-silent-time $maxsilent --keep-going --fallback " . | 
					
						
							| 
									
										
										
										
											2009-10-08 11:51:49 +00:00
										 |  |  |             "--no-build-output --log-type flat --print-build-trace " . | 
					
						
							| 
									
										
										
										
											2009-03-15 11:56:11 +00:00
										 |  |  |             "--add-root " . gcRootFor $outPath . " 2>&1"; | 
					
						
							| 
									
										
										
										
											2010-01-06 13:57:41 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-26 13:55:54 +00:00
										 |  |  |         my $max = $build->buildsteps->find( | 
					
						
							|  |  |  |             {}, {select => {max => 'stepnr + 1'}, as => ['max']}); | 
					
						
							| 
									
										
										
										
											2011-09-08 22:48:38 +00:00
										 |  |  |         my $buildStepNr = (defined $max && defined $max->get_column('max')) ? $max->get_column('max') : 1; | 
					
						
							| 
									
										
										
										
											2009-03-16 16:56:47 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         my %buildSteps; | 
					
						
							| 
									
										
										
										
											2008-11-11 17:49:50 +00:00
										 |  |  |          | 
					
						
							|  |  |  |         open OUT, "$cmd |" or die; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         while (<OUT>) { | 
					
						
							| 
									
										
										
										
											2008-11-25 00:38:16 +00:00
										 |  |  |             $errormsg .= $_; | 
					
						
							|  |  |  |              | 
					
						
							| 
									
										
										
										
											2008-11-11 17:49:50 +00:00
										 |  |  |             unless (/^@\s+/) { | 
					
						
							|  |  |  |                 print STDERR "$_"; | 
					
						
							|  |  |  |                 next; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |              | 
					
						
							|  |  |  |             if (/^@\s+build-started\s+(\S+)\s+(\S+)\s+(\S+)\s+(\S+)$/) { | 
					
						
							| 
									
										
										
										
											2009-03-16 17:46:46 +00:00
										 |  |  | 		my $drvPathStep = $1; | 
					
						
							| 
									
										
										
										
											2009-04-22 22:43:04 +00:00
										 |  |  |                 txn_do($db, sub { | 
					
						
							| 
									
										
										
										
											2009-03-12 14:18:30 +00:00
										 |  |  |                     $build->buildsteps->create( | 
					
						
							| 
									
										
										
										
											2009-03-16 17:46:46 +00:00
										 |  |  |                         { stepnr => ($buildSteps{$drvPathStep} = $buildStepNr++) | 
					
						
							| 
									
										
										
										
											2008-11-11 17:49:50 +00:00
										 |  |  |                         , type => 0 # = build | 
					
						
							| 
									
										
										
										
											2009-03-16 17:46:46 +00:00
										 |  |  |                         , drvpath => $drvPathStep | 
					
						
							| 
									
										
										
										
											2008-11-11 17:49:50 +00:00
										 |  |  |                         , outpath => $2 | 
					
						
							| 
									
										
										
										
											2010-08-31 15:27:46 +00:00
										 |  |  |                         , system => $3 | 
					
						
							| 
									
										
										
										
											2008-11-11 17:49:50 +00:00
										 |  |  |                         , logfile => $4 | 
					
						
							|  |  |  |                         , busy => 1 | 
					
						
							| 
									
										
										
										
											2008-11-12 11:09:21 +00:00
										 |  |  |                         , starttime => time | 
					
						
							| 
									
										
										
										
											2008-11-11 17:49:50 +00:00
										 |  |  |                         }); | 
					
						
							|  |  |  |                 }); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |              | 
					
						
							| 
									
										
										
										
											2010-08-31 14:08:59 +00:00
										 |  |  |             elsif (/^@\s+build-remote\s+(\S+)\s+(\S+)$/) { | 
					
						
							|  |  |  |                 my $drvPathStep = $1; | 
					
						
							|  |  |  | 		my $machine = $2; | 
					
						
							|  |  |  |                 txn_do($db, sub { | 
					
						
							|  |  |  |                     my $step = $build->buildsteps->find({stepnr => $buildSteps{$drvPathStep}}) or die; | 
					
						
							|  |  |  |                     $step->update({machine => $machine}); | 
					
						
							|  |  |  |                 }); | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  |              | 
					
						
							| 
									
										
										
										
											2008-11-12 13:00:56 +00:00
										 |  |  |             elsif (/^@\s+build-succeeded\s+(\S+)\s+(\S+)$/) { | 
					
						
							| 
									
										
										
										
											2009-03-16 17:46:46 +00:00
										 |  |  |                 my $drvPathStep = $1; | 
					
						
							| 
									
										
										
										
											2009-04-22 22:43:04 +00:00
										 |  |  |                 txn_do($db, sub { | 
					
						
							| 
									
										
										
										
											2009-03-16 17:46:46 +00:00
										 |  |  |                     my $step = $build->buildsteps->find({stepnr => $buildSteps{$drvPathStep}}) or die; | 
					
						
							| 
									
										
										
										
											2010-12-22 14:55:05 +00:00
										 |  |  |                     my $stepOutpath = $step->outpath; | 
					
						
							|  |  |  |                     my $stepStatus = 0; | 
					
						
							|  |  |  |   | 
					
						
							|  |  |  |                     $step->update({busy => 0, status => $stepStatus, stoptime => time}); | 
					
						
							| 
									
										
										
										
											2008-11-11 17:49:50 +00:00
										 |  |  |                 }); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-11-12 13:00:56 +00:00
										 |  |  |             elsif (/^@\s+build-failed\s+(\S+)\s+(\S+)\s+(\S+)\s+(.*)$/) { | 
					
						
							| 
									
										
										
										
											2008-11-25 00:38:16 +00:00
										 |  |  |                 my $drvPathStep = $1; | 
					
						
							|  |  |  |                 $someBuildFailed = 1; | 
					
						
							|  |  |  |                 $thisBuildFailed = 1 if $drvPath eq $drvPathStep; | 
					
						
							| 
									
										
										
										
											2009-03-26 12:53:39 +00:00
										 |  |  |                 my $errorMsg = $4; | 
					
						
							|  |  |  |                 $errorMsg = "build failed previously (cached)" if $3 eq "cached"; | 
					
						
							| 
									
										
										
										
											2009-04-22 22:43:04 +00:00
										 |  |  |                 txn_do($db, sub { | 
					
						
							| 
									
										
										
										
											2009-03-16 16:56:47 +00:00
										 |  |  |                     if ($buildSteps{$drvPathStep}) { | 
					
						
							|  |  |  |                         my $step = $build->buildsteps->find({stepnr => $buildSteps{$drvPathStep}}) or die; | 
					
						
							| 
									
										
										
										
											2009-03-26 12:53:39 +00:00
										 |  |  |                         $step->update({busy => 0, status => 1, errormsg => $errorMsg, stoptime => time}); | 
					
						
							| 
									
										
										
										
											2009-04-22 14:41:12 +00:00
										 |  |  |                     } | 
					
						
							|  |  |  |                     # Don't write a record if this derivation already | 
					
						
							|  |  |  |                     # failed previously.  This can happen if this is a | 
					
						
							|  |  |  |                     # restarted build. | 
					
						
							|  |  |  |                     elsif (scalar $build->buildsteps->search({drvpath => $drvPathStep, type => 0, busy => 0, status => 1}) == 0) { | 
					
						
							| 
									
										
										
										
											2009-03-12 14:18:30 +00:00
										 |  |  |                         $build->buildsteps->create( | 
					
						
							| 
									
										
										
										
											2009-03-26 12:53:39 +00:00
										 |  |  |                             { stepnr => ($buildSteps{$drvPathStep} = $buildStepNr++) | 
					
						
							| 
									
										
										
										
											2008-11-12 11:09:21 +00:00
										 |  |  |                             , type => 0 # = build | 
					
						
							| 
									
										
										
										
											2008-11-25 00:38:16 +00:00
										 |  |  |                             , drvpath => $drvPathStep | 
					
						
							| 
									
										
										
										
											2008-11-12 11:09:21 +00:00
										 |  |  |                             , outpath => $2 | 
					
						
							| 
									
										
										
										
											2009-03-26 12:53:39 +00:00
										 |  |  |                             , logfile => getBuildLog($drvPathStep) | 
					
						
							| 
									
										
										
										
											2008-11-12 11:09:21 +00:00
										 |  |  |                             , busy => 0 | 
					
						
							|  |  |  |                             , status => 1 | 
					
						
							|  |  |  |                             , starttime => time | 
					
						
							|  |  |  |                             , stoptime => time | 
					
						
							| 
									
										
										
										
											2009-03-26 12:53:39 +00:00
										 |  |  |                             , errormsg => $errorMsg | 
					
						
							| 
									
										
										
										
											2008-11-12 11:09:21 +00:00
										 |  |  |                             }); | 
					
						
							|  |  |  |                     } | 
					
						
							| 
									
										
										
										
											2008-11-11 17:49:50 +00:00
										 |  |  |                 }); | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2008-11-12 13:00:56 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |             elsif (/^@\s+substituter-started\s+(\S+)\s+(\S+)$/) { | 
					
						
							|  |  |  |                 my $outPath = $1; | 
					
						
							| 
									
										
										
										
											2009-04-22 22:43:04 +00:00
										 |  |  |                 txn_do($db, sub { | 
					
						
							| 
									
										
										
										
											2009-03-12 14:18:30 +00:00
										 |  |  |                     $build->buildsteps->create( | 
					
						
							| 
									
										
										
										
											2009-03-16 16:56:47 +00:00
										 |  |  |                         { stepnr => ($buildSteps{$outPath} = $buildStepNr++) | 
					
						
							| 
									
										
										
										
											2008-11-12 13:00:56 +00:00
										 |  |  |                         , type => 1 # = substitution | 
					
						
							|  |  |  |                         , outpath => $1 | 
					
						
							|  |  |  |                         , busy => 1 | 
					
						
							|  |  |  |                         , starttime => time | 
					
						
							|  |  |  |                         }); | 
					
						
							|  |  |  |                 }); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             elsif (/^@\s+substituter-succeeded\s+(\S+)$/) { | 
					
						
							|  |  |  |                 my $outPath = $1; | 
					
						
							| 
									
										
										
										
											2009-04-22 22:43:04 +00:00
										 |  |  |                 txn_do($db, sub { | 
					
						
							| 
									
										
										
										
											2009-03-16 16:56:47 +00:00
										 |  |  |                     my $step = $build->buildsteps->find({stepnr => $buildSteps{$outPath}}) or die; | 
					
						
							| 
									
										
										
										
											2009-03-09 16:22:41 +00:00
										 |  |  |                     $step->update({busy => 0, status => 0, stoptime => time}); | 
					
						
							| 
									
										
										
										
											2008-11-12 13:00:56 +00:00
										 |  |  |                 }); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             elsif (/^@\s+substituter-failed\s+(\S+)\s+(\S+)\s+(\S+)$/) { | 
					
						
							|  |  |  |                 my $outPath = $1; | 
					
						
							| 
									
										
										
										
											2009-04-22 22:43:04 +00:00
										 |  |  |                 txn_do($db, sub { | 
					
						
							| 
									
										
										
										
											2009-03-16 16:56:47 +00:00
										 |  |  |                     my $step = $build->buildsteps->find({stepnr => $buildSteps{$outPath}}) or die; | 
					
						
							| 
									
										
										
										
											2009-03-09 16:22:41 +00:00
										 |  |  |                     $step->update({busy => 0, status => 1, errormsg => $3, stoptime => time}); | 
					
						
							| 
									
										
										
										
											2008-11-12 13:00:56 +00:00
										 |  |  |                 }); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             else { | 
					
						
							|  |  |  |                 print STDERR "unknown Nix trace message: $_"; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2008-11-11 17:49:50 +00:00
										 |  |  |         } | 
					
						
							|  |  |  |          | 
					
						
							|  |  |  |         close OUT; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         my $res = $?; | 
					
						
							| 
									
										
										
										
											2008-11-10 13:33:12 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         $stopTime = time(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-11-25 00:38:16 +00:00
										 |  |  |         if ($res != 0) { | 
					
						
							|  |  |  |             if ($thisBuildFailed) { $buildStatus = 1; } | 
					
						
							|  |  |  |             elsif ($someBuildFailed) { $buildStatus = 2; } | 
					
						
							|  |  |  |             else { $buildStatus = 3; } | 
					
						
							| 
									
										
										
										
											2011-03-07 13:55:43 +00:00
										 |  |  |         }  | 
					
						
							| 
									
										
										
										
											2008-11-10 13:33:12 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-11-25 00:38:16 +00:00
										 |  |  |         # Only store the output of running Nix if we have a miscellaneous error. | 
					
						
							|  |  |  |         $errormsg = undef unless $buildStatus == 3; | 
					
						
							| 
									
										
										
										
											2008-11-10 13:33:12 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-03-09 17:21:10 +00:00
										 |  |  |   done: | 
					
						
							| 
									
										
										
										
											2010-11-11 11:03:50 +00:00
										 |  |  |     my $logfile = getBuildLog($drvPath);  | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-19 15:44:20 +00:00
										 |  |  |     my $logsize = defined $logfile ? stat($logfile)->size : 0; | 
					
						
							| 
									
										
										
										
											2010-11-11 11:03:50 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-19 15:44:20 +00:00
										 |  |  |     my $size = 0; | 
					
						
							| 
									
										
										
										
											2010-11-11 11:03:50 +00:00
										 |  |  |     my $closuresize = 0; | 
					
						
							| 
									
										
										
										
											2010-11-19 15:44:20 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-11 11:03:50 +00:00
										 |  |  |     if (isValidPath($outPath)) { | 
					
						
							| 
									
										
										
										
											2011-11-30 17:22:31 +01:00
										 |  |  |         my ($deriver, $hash, $time, $narSize, $refs) = queryPathInfo($outPath); | 
					
						
							| 
									
										
										
										
											2010-11-19 15:44:20 +00:00
										 |  |  |         $size = $narSize; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-30 17:22:31 +01:00
										 |  |  |         my @closure = computeFSClosure(0, 0, $outPath); | 
					
						
							| 
									
										
										
										
											2010-11-19 15:44:20 +00:00
										 |  |  |         foreach my $path (@closure) { | 
					
						
							| 
									
										
										
										
											2011-11-30 17:22:31 +01:00
										 |  |  | 	    my ($deriver, $hash, $time, $narSize, $refs) = queryPathInfo($path); | 
					
						
							| 
									
										
										
										
											2010-11-19 15:44:20 +00:00
										 |  |  |  	    $closuresize += $narSize;     | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-11-11 11:03:50 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-03-09 17:21:10 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-04-22 22:43:04 +00:00
										 |  |  |     txn_do($db, sub { | 
					
						
							| 
									
										
										
										
											2010-09-01 10:50:57 +00:00
										 |  |  |         my $releaseName = getReleaseName($outPath); | 
					
						
							| 
									
										
										
										
											2008-11-25 00:38:16 +00:00
										 |  |  |          | 
					
						
							| 
									
										
										
										
											2012-03-05 21:52:47 +01:00
										 |  |  |         $buildStatus = 6 if $buildStatus == 0 && -f "$outPath/nix-support/failed"; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $build->update( | 
					
						
							|  |  |  |             { finished => 1 | 
					
						
							|  |  |  |             , busy => 0 | 
					
						
							|  |  |  |             , locker => '' | 
					
						
							|  |  |  |             , logfile => '' | 
					
						
							|  |  |  |             , timestamp => time # !!! Why change the timestamp? | 
					
						
							| 
									
										
										
										
											2008-11-10 13:33:12 +00:00
										 |  |  |             , iscachedbuild => $isCachedBuild | 
					
						
							|  |  |  |             , buildstatus => $buildStatus | 
					
						
							|  |  |  |             , starttime => $startTime | 
					
						
							|  |  |  |             , stoptime => $stopTime | 
					
						
							| 
									
										
										
										
											2010-11-11 11:03:50 +00:00
										 |  |  |             , logfile => $logfile | 
					
						
							|  |  |  |             , logsize => $logsize | 
					
						
							|  |  |  |             , size => $size | 
					
						
							|  |  |  |             , closuresize => $closuresize | 
					
						
							| 
									
										
										
										
											2008-11-25 00:38:16 +00:00
										 |  |  |             , errormsg => $errormsg | 
					
						
							| 
									
										
										
										
											2008-11-25 16:13:22 +00:00
										 |  |  |             , releasename => $releaseName | 
					
						
							| 
									
										
										
										
											2008-11-10 13:33:12 +00:00
										 |  |  |             }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-22 14:55:05 +00:00
										 |  |  |         if ($buildStatus == 0 || $buildStatus == 6) { | 
					
						
							| 
									
										
										
										
											2010-09-07 11:29:52 +00:00
										 |  |  |             addBuildProducts($db, $build); | 
					
						
							| 
									
										
										
										
											2008-11-10 13:33:12 +00:00
										 |  |  |         } | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2009-07-07 16:15:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 15:52:55 +00:00
										 |  |  |     sendEmailNotification $build; | 
					
						
							| 
									
										
										
										
											2009-10-15 13:23:15 +00:00
										 |  |  |     sendTwitterNotification $build; | 
					
						
							| 
									
										
										
										
											2008-11-10 13:33:12 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-30 17:32:50 +01:00
										 |  |  | my $buildId = $ARGV[0] or die "syntax: $0 BUILD-ID\n"; | 
					
						
							| 
									
										
										
										
											2008-11-11 17:49:50 +00:00
										 |  |  | print STDERR "performing build $buildId\n"; | 
					
						
							| 
									
										
										
										
											2008-11-10 13:33:12 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 15:52:55 +00:00
										 |  |  | if ($ENV{'HYDRA_MAIL_TEST'}) { | 
					
						
							|  |  |  |     sendEmailNotification $db->resultset('Builds')->find($buildId); | 
					
						
							|  |  |  |     exit 0; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2009-10-15 13:23:15 +00:00
										 |  |  | if ($ENV{'HYDRA_TWITTER_TEST'}) { | 
					
						
							|  |  |  |     sendTwitterNotification $db->resultset('Builds')->find($buildId); | 
					
						
							|  |  |  |     exit 0; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2009-07-07 16:15:38 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-11-11 12:54:37 +00:00
										 |  |  | # Lock the build.  If necessary, steal the lock from the parent | 
					
						
							|  |  |  | # process (runner.pl).  This is so that if the runner dies, the | 
					
						
							|  |  |  | # children (i.e. the build.pl instances) can continue to run and won't | 
					
						
							|  |  |  | # have the lock taken away. | 
					
						
							|  |  |  | my $build; | 
					
						
							| 
									
										
										
										
											2009-04-22 22:43:04 +00:00
										 |  |  | txn_do($db, sub { | 
					
						
							| 
									
										
										
										
											2008-11-27 15:16:06 +00:00
										 |  |  |     $build = $db->resultset('Builds')->find($buildId); | 
					
						
							| 
									
										
										
										
											2008-11-11 12:54:37 +00:00
										 |  |  |     die "build $buildId doesn't exist" unless defined $build; | 
					
						
							| 
									
										
										
										
											2012-03-05 21:52:47 +01:00
										 |  |  |     die "build $buildId already done" if $build->finished; | 
					
						
							| 
									
										
										
										
											2012-02-29 02:22:49 +01:00
										 |  |  |     if ($build->busy != 0 && $build->locker != getppid) { | 
					
						
							| 
									
										
										
										
											2008-11-11 12:54:37 +00:00
										 |  |  |         die "build $buildId is already being built"; | 
					
						
							| 
									
										
										
										
											2008-11-10 13:33:12 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2012-02-29 02:22:49 +01:00
										 |  |  |     $build->update({busy => 1, locker => $$}); | 
					
						
							| 
									
										
										
										
											2009-03-16 16:56:47 +00:00
										 |  |  |     $build->buildsteps->search({busy => 1})->delete_all; | 
					
						
							| 
									
										
										
										
											2009-03-06 13:34:53 +00:00
										 |  |  |     $build->buildproducts->delete_all; | 
					
						
							| 
									
										
										
										
											2008-11-10 13:33:12 +00:00
										 |  |  | }); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-11-11 12:54:37 +00:00
										 |  |  | die unless $build; | 
					
						
							| 
									
										
										
										
											2008-11-10 13:33:12 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-11-11 12:54:37 +00:00
										 |  |  | # Do the build.  If it throws an error, unlock the build so that it | 
					
						
							|  |  |  | # can be retried. | 
					
						
							| 
									
										
										
										
											2008-11-10 13:33:12 +00:00
										 |  |  | eval { | 
					
						
							| 
									
										
										
										
											2008-11-11 12:54:37 +00:00
										 |  |  |     doBuild $build; | 
					
						
							| 
									
										
										
										
											2008-11-12 14:29:32 +00:00
										 |  |  |     print "done\n"; | 
					
						
							| 
									
										
										
										
											2008-11-10 13:33:12 +00:00
										 |  |  | }; | 
					
						
							|  |  |  | if ($@) { | 
					
						
							|  |  |  |     warn $@; | 
					
						
							| 
									
										
										
										
											2009-04-22 22:43:04 +00:00
										 |  |  |     txn_do($db, sub { | 
					
						
							| 
									
										
										
										
											2012-02-29 02:22:49 +01:00
										 |  |  |         $build->update({busy => 0, locker => $$}); | 
					
						
							| 
									
										
										
										
											2008-11-10 13:33:12 +00:00
										 |  |  |     }); | 
					
						
							|  |  |  | } |