forked from lix-project/hydra
Merge pull request #1051 from DeterminateSystems/restart-cancel-resultset-p2
Validate and fix canceling builds
This commit is contained in:
commit
9ae676072c
3 changed files with 201 additions and 1 deletions
|
@ -144,7 +144,7 @@ sub create_jobset : Chained('evalChain') PathPart('create-jobset') Args(0) {
|
||||||
sub cancel : Chained('evalChain') PathPart('cancel') Args(0) {
|
sub cancel : Chained('evalChain') PathPart('cancel') Args(0) {
|
||||||
my ($self, $c) = @_;
|
my ($self, $c) = @_;
|
||||||
requireCancelBuildPrivileges($c, $c->stash->{project});
|
requireCancelBuildPrivileges($c, $c->stash->{project});
|
||||||
my $n = cancelBuilds($c->model('DB')->schema, $c->stash->{eval}->builds);
|
my $n = cancelBuilds($c->model('DB')->schema, $c->stash->{eval}->builds->search_rs({}));
|
||||||
$c->flash->{successMsg} = "$n builds have been cancelled.";
|
$c->flash->{successMsg} = "$n builds have been cancelled.";
|
||||||
$c->res->redirect($c->uri_for($c->controller('JobsetEval')->action_for('view'), $c->req->captures));
|
$c->res->redirect($c->uri_for($c->controller('JobsetEval')->action_for('view'), $c->req->captures));
|
||||||
}
|
}
|
||||||
|
|
133
t/Controller/Admin/clear-queue-non-current.t
Normal file
133
t/Controller/Admin/clear-queue-non-current.t
Normal file
|
@ -0,0 +1,133 @@
|
||||||
|
use strict;
|
||||||
|
use warnings;
|
||||||
|
use Setup;
|
||||||
|
use JSON qw(decode_json encode_json);
|
||||||
|
use File::Copy;
|
||||||
|
|
||||||
|
my %ctx = test_init(
|
||||||
|
hydra_config => q|
|
||||||
|
# No caching for PathInput plugin, otherwise we get wrong values
|
||||||
|
# (as it has a 30s window where no changes to the file are considered).
|
||||||
|
path_input_cache_validity_seconds = 0
|
||||||
|
|
|
||||||
|
);
|
||||||
|
|
||||||
|
require Hydra::Schema;
|
||||||
|
require Hydra::Model::DB;
|
||||||
|
require Hydra::Helper::Nix;
|
||||||
|
|
||||||
|
use Test2::V0;
|
||||||
|
require Catalyst::Test;
|
||||||
|
Catalyst::Test->import('Hydra');
|
||||||
|
use HTTP::Request::Common qw(POST PUT GET DELETE);
|
||||||
|
|
||||||
|
my $db = Hydra::Model::DB->new;
|
||||||
|
hydra_setup($db);
|
||||||
|
|
||||||
|
# Create a user to log in to
|
||||||
|
my $user = $db->resultset('Users')->create({ username => 'alice', emailaddress => 'root@invalid.org', password => '!' });
|
||||||
|
$user->setPassword('foobar');
|
||||||
|
$user->userroles->update_or_create({ role => 'admin' });
|
||||||
|
|
||||||
|
my $project = $db->resultset('Projects')->create({name => 'tests', displayname => 'Tests', owner => 'alice'});
|
||||||
|
|
||||||
|
my $scratchdir = $ctx{tmpdir} . "/scratch";
|
||||||
|
my $jobset = createBaseJobset("basic", "default.nix", $scratchdir);
|
||||||
|
|
||||||
|
subtest "Create and evaluate our job at version 1" => sub {
|
||||||
|
mkdir $scratchdir or die "mkdir($scratchdir): $!\n";
|
||||||
|
|
||||||
|
# Note: this recreates the raw derivation and skips
|
||||||
|
# the generated config.nix because we never actually
|
||||||
|
# build anything.
|
||||||
|
open(my $fh, ">", "$scratchdir/default.nix");
|
||||||
|
print $fh <<EOF;
|
||||||
|
{
|
||||||
|
example = derivation {
|
||||||
|
builder = "./builder.sh";
|
||||||
|
name = "example";
|
||||||
|
system = builtins.currentSystem;
|
||||||
|
version = 1;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
EOF
|
||||||
|
close($fh);
|
||||||
|
|
||||||
|
ok(evalSucceeds($jobset), "Evaluating our default.nix should exit with return code 0");
|
||||||
|
is(nrQueuedBuildsForJobset($jobset), 1, "Evaluating our default.nix should result in 1 builds");
|
||||||
|
};
|
||||||
|
|
||||||
|
subtest "Update and evaluate our job to version 2" => sub {
|
||||||
|
open(my $fh, ">", "$scratchdir/default.nix");
|
||||||
|
print $fh <<EOF;
|
||||||
|
{
|
||||||
|
example = derivation {
|
||||||
|
builder = "./builder.sh";
|
||||||
|
name = "example";
|
||||||
|
system = builtins.currentSystem;
|
||||||
|
version = 2;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
EOF
|
||||||
|
close($fh);
|
||||||
|
|
||||||
|
|
||||||
|
ok(evalSucceeds($jobset), "Evaluating our default.nix should exit with return code 0");
|
||||||
|
is(nrQueuedBuildsForJobset($jobset), 2, "Evaluating our default.nix should result in 1 more build, resulting in 2 queued builds");
|
||||||
|
};
|
||||||
|
|
||||||
|
my ($firstBuild, $secondBuild, @builds) = queuedBuildsForJobset($jobset)->search(
|
||||||
|
{},
|
||||||
|
{ order_by => { -asc => 'id' }}
|
||||||
|
);
|
||||||
|
subtest "Validating the first build" => sub {
|
||||||
|
isnt($firstBuild, undef, "We have our first build");
|
||||||
|
is($firstBuild->id, 1, "The first build is ID 1");
|
||||||
|
is($firstBuild->finished, 0, "The first build is not yet finished");
|
||||||
|
is($firstBuild->buildstatus, undef, "The first build status is null");
|
||||||
|
};
|
||||||
|
|
||||||
|
subtest "Validating the second build" => sub {
|
||||||
|
isnt($secondBuild, undef, "We have our second build");
|
||||||
|
is($secondBuild->id, 2, "The second build is ID 2");
|
||||||
|
is($secondBuild->finished, 0, "The second build is not yet finished");
|
||||||
|
is($secondBuild->buildstatus, undef, "The second build status is null");
|
||||||
|
};
|
||||||
|
|
||||||
|
is(@builds, 0, "No other builds were created");
|
||||||
|
|
||||||
|
# Login and save cookie for future requests
|
||||||
|
my $req = request(POST '/login',
|
||||||
|
Referer => 'http://localhost/',
|
||||||
|
Content => {
|
||||||
|
username => 'alice',
|
||||||
|
password => 'foobar'
|
||||||
|
}
|
||||||
|
);
|
||||||
|
is($req->code, 302, "Logging in gets a 302");
|
||||||
|
my $cookie = $req->header("set-cookie");
|
||||||
|
|
||||||
|
subtest 'Cancel queued, non-current builds' => sub {
|
||||||
|
my $restart = request(PUT '/admin/clear-queue-non-current',
|
||||||
|
Accept => 'application/json',
|
||||||
|
Content_Type => 'application/json',
|
||||||
|
Referer => '/admin/example-referer',
|
||||||
|
Cookie => $cookie,
|
||||||
|
);
|
||||||
|
is($restart->code, 302, "Canceling 302's back to the build");
|
||||||
|
is($restart->header("location"), "/admin/example-referer", "We're redirected back to the referer");
|
||||||
|
};
|
||||||
|
|
||||||
|
subtest "Validating the first build is canceled" => sub {
|
||||||
|
my $build = $db->resultset('Builds')->find($firstBuild->id);
|
||||||
|
is($build->finished, 1, "Build should be 'finished'.");
|
||||||
|
is($build->buildstatus, 4, "Build should be canceled.");
|
||||||
|
};
|
||||||
|
|
||||||
|
subtest "Validating the second build is not canceled" => sub {
|
||||||
|
my $build = $db->resultset('Builds')->find($secondBuild->id);
|
||||||
|
is($build->finished, 0, "Build should be unfinished.");
|
||||||
|
is($build->buildstatus, undef, "Build status should be null.");
|
||||||
|
};
|
||||||
|
|
||||||
|
done_testing;
|
67
t/Controller/JobsetEval/cancel.t
Normal file
67
t/Controller/JobsetEval/cancel.t
Normal file
|
@ -0,0 +1,67 @@
|
||||||
|
use feature 'unicode_strings';
|
||||||
|
use strict;
|
||||||
|
use warnings;
|
||||||
|
use Setup;
|
||||||
|
use JSON qw(decode_json encode_json);
|
||||||
|
|
||||||
|
my %ctx = test_init();
|
||||||
|
|
||||||
|
require Hydra::Schema;
|
||||||
|
require Hydra::Model::DB;
|
||||||
|
require Hydra::Helper::Nix;
|
||||||
|
|
||||||
|
use Test2::V0;
|
||||||
|
require Catalyst::Test;
|
||||||
|
Catalyst::Test->import('Hydra');
|
||||||
|
use HTTP::Request::Common qw(POST PUT GET DELETE);
|
||||||
|
|
||||||
|
my $db = Hydra::Model::DB->new;
|
||||||
|
hydra_setup($db);
|
||||||
|
|
||||||
|
# Create a user to log in to
|
||||||
|
my $user = $db->resultset('Users')->create({ username => 'alice', emailaddress => 'root@invalid.org', password => '!' });
|
||||||
|
$user->setPassword('foobar');
|
||||||
|
$user->userroles->update_or_create({ role => 'admin' });
|
||||||
|
|
||||||
|
my $project = $db->resultset('Projects')->create({name => 'tests', displayname => 'Tests', owner => 'alice'});
|
||||||
|
|
||||||
|
my $jobset = createBaseJobset("basic", "basic.nix", $ctx{jobsdir});
|
||||||
|
|
||||||
|
ok(evalSucceeds($jobset), "Evaluating jobs/basic.nix should exit with return code 0");
|
||||||
|
is(nrQueuedBuildsForJobset($jobset), 3, "Evaluating jobs/basic.nix should result in 3 builds");
|
||||||
|
|
||||||
|
my ($eval, @evals) = $jobset->jobsetevals;
|
||||||
|
isnt($eval, undef, "We have an evaluation to restart");
|
||||||
|
|
||||||
|
my ($build, @builds) = queuedBuildsForJobset($jobset);
|
||||||
|
is($build->finished, 0, "Unbuilt build should not be finished.");
|
||||||
|
is($build->buildstatus, undef, "Unbuilt build should be undefined.");
|
||||||
|
|
||||||
|
|
||||||
|
# Login and save cookie for future requests
|
||||||
|
my $req = request(POST '/login',
|
||||||
|
Referer => 'http://localhost/',
|
||||||
|
Content => {
|
||||||
|
username => 'alice',
|
||||||
|
password => 'foobar'
|
||||||
|
}
|
||||||
|
);
|
||||||
|
is($req->code, 302, "Logging in gets a 302");
|
||||||
|
my $cookie = $req->header("set-cookie");
|
||||||
|
|
||||||
|
|
||||||
|
subtest 'Cancel the JobsetEval builds' => sub {
|
||||||
|
my $restart = request(PUT '/eval/' . $eval->id . '/cancel',
|
||||||
|
Accept => 'application/json',
|
||||||
|
Content_Type => 'application/json',
|
||||||
|
Cookie => $cookie,
|
||||||
|
);
|
||||||
|
is($restart->code, 302, "Canceling 302's back to the build");
|
||||||
|
is($restart->header("location"), "http://localhost/eval/" . $eval->id, "We're redirected back to the eval page");
|
||||||
|
|
||||||
|
my $newbuild = $db->resultset('Builds')->find($build->id);
|
||||||
|
is($newbuild->finished, 1, "Build 'fails' from jobs/basic.nix should be 'finished'.");
|
||||||
|
is($newbuild->buildstatus, 4, "Build 'fails' from jobs/basic.nix should be canceled.");
|
||||||
|
};
|
||||||
|
|
||||||
|
done_testing;
|
Loading…
Reference in a new issue