From d19890a4e18af4a710889e0b022bb191f211657c Mon Sep 17 00:00:00 2001 From: Graham Christensen Date: Wed, 27 Oct 2021 11:05:00 -0400 Subject: [PATCH 1/2] JobsetEval/cancel: test & fix --- src/lib/Hydra/Controller/JobsetEval.pm | 2 +- t/Controller/JobsetEval/cancel.t | 67 ++++++++++++++++++++++++++ 2 files changed, 68 insertions(+), 1 deletion(-) create mode 100644 t/Controller/JobsetEval/cancel.t diff --git a/src/lib/Hydra/Controller/JobsetEval.pm b/src/lib/Hydra/Controller/JobsetEval.pm index 21c4e25f..2635adee 100644 --- a/src/lib/Hydra/Controller/JobsetEval.pm +++ b/src/lib/Hydra/Controller/JobsetEval.pm @@ -144,7 +144,7 @@ sub create_jobset : Chained('evalChain') PathPart('create-jobset') Args(0) { sub cancel : Chained('evalChain') PathPart('cancel') Args(0) { my ($self, $c) = @_; 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->res->redirect($c->uri_for($c->controller('JobsetEval')->action_for('view'), $c->req->captures)); } diff --git a/t/Controller/JobsetEval/cancel.t b/t/Controller/JobsetEval/cancel.t new file mode 100644 index 00000000..03f23e05 --- /dev/null +++ b/t/Controller/JobsetEval/cancel.t @@ -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; From c1630e6d8053e228074de19692e366cae2a3841b Mon Sep 17 00:00:00 2001 From: Graham Christensen Date: Wed, 27 Oct 2021 11:43:58 -0400 Subject: [PATCH 2/2] Admin/clear-queue-non-current: test behavior --- t/Controller/Admin/clear-queue-non-current.t | 133 +++++++++++++++++++ 1 file changed, 133 insertions(+) create mode 100644 t/Controller/Admin/clear-queue-non-current.t diff --git a/t/Controller/Admin/clear-queue-non-current.t b/t/Controller/Admin/clear-queue-non-current.t new file mode 100644 index 00000000..630abbdc --- /dev/null +++ b/t/Controller/Admin/clear-queue-non-current.t @@ -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 < sub { + open(my $fh, ">", "$scratchdir/default.nix"); + print $fh <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;