From 26008105519a62fd1daecf352215a783529c2aa8 Mon Sep 17 00:00:00 2001 From: Cole Helbling Date: Tue, 27 Apr 2021 15:51:17 -0700 Subject: [PATCH 1/7] hydra-api: flesh out Jobset schema * made all columns available via the API (except for forceeval) * renamed flakeref to flake to unify the API with the database schema * renamed inputs to jobsetinputs to unify the API with the database schema --- hydra-api.yaml | 98 ++++++++++++++++++------------ src/lib/Hydra/Controller/Jobset.pm | 6 +- src/lib/Hydra/Schema/Jobsets.pm | 26 ++++++-- src/root/edit-jobset.tt | 20 +++--- 4 files changed, 95 insertions(+), 55 deletions(-) diff --git a/hydra-api.yaml b/hydra-api.yaml index 2443b679..d01387ec 100644 --- a/hydra-api.yaml +++ b/hydra-api.yaml @@ -306,35 +306,7 @@ paths: content: application/json: schema: - type: object - properties: - 'description': - description: a description of the jobset - type: string - checkinterval: - description: interval (in seconds) in which to check for evaluation - type: integer - enabled: - description: when true the jobset gets scheduled for evaluation - type: boolean - visible: - description: when true the jobset is visible in the web frontend - type: boolean - keepnr: - description: number or evaluations to keep - type: integer - nixexprinput: - description: the name of the jobset input which contains the nixexprpath - type: string - nixexprpath: - nullable: true - description: the path to the file to evaluate - type: string - inputs: - description: inputs for this jobset - type: object - additionalProperties: - $ref: '#/components/schemas/JobsetInput' + $ref: '#/components/schemas/Jobset' responses: '201': description: jobset creation response @@ -590,26 +562,76 @@ components: Jobset: type: object properties: - fetcherrormsg: + name: + description: the name of the jobset + type: string + project: + description: the project this jobset belongs to + type: string + description: nullable: true - description: contains the error message when there was a problem fetching sources for a jobset + description: a description of the jobset type: string nixexprinput: + nullable: true description: the name of the jobset input which contains the nixexprpath type: string - errormsg: - description: contains the stderr output of the nix-instantiate command - type: string - emailoverride: - description: email address to send notices to instead of the package maintainer (can be a comma separated list) - type: string nixexprpath: nullable: true description: the path to the file to evaluate type: string + errormsg: + nullable: true + description: contains the stderr output of the nix-instantiate command + type: string + errortime: + nullable: true + description: timestamp associated with errormsg + type: integer + lastcheckedtime: + nullable: true + description: the last time the evaluator looked at this jobset + type: integer + triggertime: + nullable: true + description: set to the time we were triggered by a push event + type: integer enabled: - description: when set to true the jobset gets scheduled for evaluation + description: 0 is disabled, 1 is enabled, 2 is one-shot, and 3 is one-at-a-time + type: integer + enableemail: + description: when true the jobset sends emails when previously-successful builds fail type: boolean + hidden: + description: when false the jobset is visible in the web frontend + type: boolean + emailoverride: + description: email address to send notices to instead of the package maintainer (can be a comma separated list) + type: string + keepnr: + description: number or evaluations to keep + type: integer + checkinterval: + description: interval (in seconds) in which to check for evaluation + type: integer + schedulingshares: + description: how many shares to be allocated to the jobset + type: integer + fetcherrormsg: + nullable: true + description: contains the error message when there was a problem fetching sources for a jobset + type: string + startime: + nullable: true + description: set to the time the latest evaluation started (if one is currently running) + type: integer + type: + description: the type of the jobset + type: string + flake: + nullable: true + description: the flake uri to evaluate + type: string jobsetinputs: description: inputs configured for this jobset type: object diff --git a/src/lib/Hydra/Controller/Jobset.pm b/src/lib/Hydra/Controller/Jobset.pm index 8178c6b7..b391e231 100644 --- a/src/lib/Hydra/Controller/Jobset.pm +++ b/src/lib/Hydra/Controller/Jobset.pm @@ -231,7 +231,7 @@ sub updateJobset { if ($type == 0) { ($nixExprPath, $nixExprInput) = nixExprPathFromParams $c; } elsif ($type == 1) { - $flake = trim($c->stash->{params}->{"flakeref"}); + $flake = trim($c->stash->{params}->{"flake"}); error($c, "Invalid flake URI ‘$flake’.") if $flake !~ /^[a-zA-Z]/; } else { error($c, "Invalid jobset type."); @@ -270,8 +270,8 @@ sub updateJobset { $jobset->jobsetinputs->delete; if ($type == 0) { - foreach my $name (keys %{$c->stash->{params}->{inputs}}) { - my $inputData = $c->stash->{params}->{inputs}->{$name}; + foreach my $name (keys %{$c->stash->{params}->{jobsetinputs}}) { + my $inputData = $c->stash->{params}->{jobsetinputs}->{$name}; my $type = $inputData->{type}; my $value = $inputData->{value}; my $emailresponsible = defined $inputData->{emailresponsible} ? 1 : 0; diff --git a/src/lib/Hydra/Schema/Jobsets.pm b/src/lib/Hydra/Schema/Jobsets.pm index b2dc0131..ee93f349 100644 --- a/src/lib/Hydra/Schema/Jobsets.pm +++ b/src/lib/Hydra/Schema/Jobsets.pm @@ -412,12 +412,30 @@ __PACKAGE__->add_column( my %hint = ( columns => [ + "errortime", + "lastcheckedtime", + "triggertime", "enabled", - "errormsg", - "fetcherrormsg", - "emailoverride", + "keepnr", + "checkinterval", + "schedulingshares", + "starttime" + ], + string_columns => [ + "name", + "project", + "description", + "nixexprinput", "nixexprpath", - "nixexprinput" + "errormsg", + "emailoverride", + "fetcherrormsg", + "type", + "flake" + ], + boolean_columns => [ + "enableemail", + "hidden" ], eager_relations => { jobsetinputs => "name" diff --git a/src/root/edit-jobset.tt b/src/root/edit-jobset.tt index 324c7a87..4243f1fa 100644 --- a/src/root/edit-jobset.tt +++ b/src/root/edit-jobset.tt @@ -46,8 +46,8 @@ Input nameTypeValueNotify committers - - [% inputs = createFromEval ? eval.jobsetevalinputs : jobset.jobsetinputs; FOREACH input IN inputs %] + + [% jobsetinputs = createFromEval ? eval.jobsetevalinputs : jobset.jobsetinputs; FOREACH input IN jobsetinputs %] [% INCLUDE renderJobsetInput input=input baseName="input-$input.name" %] [% END %] @@ -111,9 +111,9 @@
- +
- jobset.flake) %]/> + jobset.flake) %]/>
@@ -220,8 +220,8 @@ $("#submit-jobset").click(function() { var formElements = $(this).parents("form").serializeArray(); - var data = { 'inputs': {} }; - var inputs = {}; + var data = { 'jobsetinputs': {} }; + var jobsetinputs = {}; for (var i = 0; formElements.length > i; i++) { var elem = formElements[i]; var match = elem.name.match(/^input-([\w-]+)-(\w+)$/); @@ -233,13 +233,13 @@ if (baseName === "template") continue; - if (!(baseName in inputs)) - inputs[baseName] = {}; + if (!(baseName in jobsetinputs)) + jobsetinputs[baseName] = {}; if (param === "name") - data.inputs[elem.value] = inputs[baseName]; + data.jobsetinputs[elem.value] = jobsetinputs[baseName]; else - inputs[baseName][param] = elem.value; + jobsetinputs[baseName][param] = elem.value; } } redirectJSON({ From 72fec31dbbbc087d9a64552241110ab89e92efff Mon Sep 17 00:00:00 2001 From: Cole Helbling Date: Tue, 27 Apr 2021 16:06:49 -0700 Subject: [PATCH 2/7] hydra-api: flesh out JobsetInput schema --- hydra-api.yaml | 9 +++++++++ src/lib/Hydra/Schema/JobsetInputs.pm | 7 +++++++ 2 files changed, 16 insertions(+) diff --git a/hydra-api.yaml b/hydra-api.yaml index d01387ec..9e67d207 100644 --- a/hydra-api.yaml +++ b/hydra-api.yaml @@ -553,6 +553,15 @@ components: JobsetInput: type: object properties: + name: + description: name of the input + type: string + type: + description: type of input + type: string + emailresponsible: + description: whether or not to email responsible parties + type: boolean jobsetinputalts: type: array description: ??? diff --git a/src/lib/Hydra/Schema/JobsetInputs.pm b/src/lib/Hydra/Schema/JobsetInputs.pm index d0964ab4..d0d1665d 100644 --- a/src/lib/Hydra/Schema/JobsetInputs.pm +++ b/src/lib/Hydra/Schema/JobsetInputs.pm @@ -135,6 +135,13 @@ __PACKAGE__->has_many( # DO NOT MODIFY THIS OR ANYTHING ABOVE! md5sum:5uKwEhDXso4IR1TFmwRxiA my %hint = ( + string_columns => [ + "name", + "type" + ], + boolean_columns => [ + "emailresponsible" + ], relations => { "jobsetinputalts" => "value" } From bcd3bbb6806c59c3f7130d623d55b688c6f888a0 Mon Sep 17 00:00:00 2001 From: Cole Helbling Date: Tue, 27 Apr 2021 16:12:52 -0700 Subject: [PATCH 3/7] hydra-api: implement DELETE /jobset/{project-id}/{jobset-id} --- hydra-api.yaml | 33 +++++++++++++++++++++++++++++++++ 1 file changed, 33 insertions(+) diff --git a/hydra-api.yaml b/hydra-api.yaml index 9e67d207..acbf835f 100644 --- a/hydra-api.yaml +++ b/hydra-api.yaml @@ -370,6 +370,39 @@ paths: schema: $ref: '#/components/schemas/Error' + delete: + summary: Deletes a jobset designated by project and jobset id + parameters: + - name: project-id + in: path + description: name of the project the jobset belongs to + required: true + schema: + type: string + - name: jobset-id + in: path + description: name of the jobset to retrieve + required: true + schema: + type: string + responses: + '200': + description: jobset successfully deleted + content: + application/json: + schema: + type: object + properties: + redirect: + type: string + description: root of the Hydra instance + '404': + description: jobset couldn't be found + content: + application/json: + schema: + $ref: '#/components/schemas/Error' + /jobset/{project-id}/{jobset-id}/evals: get: summary: Retrieves all evaluations of a jobset From 42ef3b7b72f06a0079cefc24103bd2a3085b56a0 Mon Sep 17 00:00:00 2001 From: Cole Helbling Date: Tue, 27 Apr 2021 23:06:45 -0700 Subject: [PATCH 4/7] hydra-api: update Project and Jobset examples with the new schema For future reference, this was generated by sending the request and piping it to `yq -y` so that it would spit out YAML. --- hydra-api.yaml | 83 ++++++++++++++++++++++++++++++++------------------ 1 file changed, 53 insertions(+), 30 deletions(-) diff --git a/hydra-api.yaml b/hydra-api.yaml index acbf835f..766cfd0b 100644 --- a/hydra-api.yaml +++ b/hydra-api.yaml @@ -935,49 +935,72 @@ components: examples: projects-success: value: - - enabled: 1 - name: example-hello - hidden: 0 - description: hello - owner: hydra-user + - displayname: Foo Bar + description: Foo Bar Baz Qux + enabled: true + owner: alice jobsets: - - hello - displayname: example-hello - - displayname: foo - jobsets: - - foobar - owner: hydra-user - name: foo - enabled: 1 - description: foo project - hidden: 0 + - bar-jobset + hidden: false + homepage: https://example.com/ + name: foobar + - jobsets: + - test-jobset + hidden: false + name: hello + homepage: https://example.com/ + description: Hi There + displayname: Hello + enabled: true + owner: alice project-success: value: - name: foo - enabled: 1 - hidden: 0 - description: foo project - displayname: foo - owner: gilligan jobsets: - - foobar + - bar-jobset + homepage: https://example.com/ + name: foobar + hidden: false + enabled: true + displayname: Foo Bar + description: Foo Bar Baz Qux + owner: alice jobset-success: value: - nixexprpath: examples/hello.nix - enabled: 1 + triggertime: null + enableemail: false jobsetinputs: - hydra: - jobsetinputalts: - - 'https://github.com/gilligan/hydra extend-readme' nixpkgs: + type: git + name: nixpkgs + emailresponsible: false jobsetinputalts: - - 'https://github.com/nixos/nixpkgs-channels nixos-20.03' + - https://github.com/NixOS/nixpkgs.git + officialRelease: + jobsetinputalts: + - 'false' + emailresponsible: false + name: officialRelease + type: boolean + fetcherrormsg: '' + hidden: false + schedulingshares: 1 emailoverride: '' + starttime: null + description: '' errormsg: '' - nixexprinput: hydra - fetcherrormsg: null + lastcheckedtime: null + nixexprinput: nixpkgs + checkinterval: 0 + project: foobar + flake: '' + type: 0 + enabled: 1 + name: bar-jobset + keepnr: 0 + nixexprpath: pkgs/top-level/release.nix + errortime: null evals-success: value: From d589db2ed9b915b9f326c06a3405d5791ec910fd Mon Sep 17 00:00:00 2001 From: Graham Christensen Date: Tue, 27 Apr 2021 21:10:15 -0400 Subject: [PATCH 5/7] login: missing parameters are 400s --- src/lib/Hydra/Controller/User.pm | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/lib/Hydra/Controller/User.pm b/src/lib/Hydra/Controller/User.pm index b3512a1b..a07a3c42 100644 --- a/src/lib/Hydra/Controller/User.pm +++ b/src/lib/Hydra/Controller/User.pm @@ -27,8 +27,8 @@ sub login_POST { my $username = $c->stash->{params}->{username} // ""; my $password = $c->stash->{params}->{password} // ""; - error($c, "You must specify a user name.") if $username eq ""; - error($c, "You must specify a password.") if $password eq ""; + badRequest($c, "You must specify a user name.") if $username eq ""; + badRequest($c, "You must specify a password.") if $password eq ""; if ($c->get_auth_realm('ldap') && $c->authenticate({username => $username, password => $password}, 'ldap')) { doLDAPLogin($self, $c, $username); From 725c9c2f81b6c3a2082b985bf25772c60c1b4f37 Mon Sep 17 00:00:00 2001 From: Graham Christensen Date: Tue, 27 Apr 2021 21:34:22 -0400 Subject: [PATCH 6/7] login: redirect to the current-user page --- src/lib/Hydra/Controller/User.pm | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/src/lib/Hydra/Controller/User.pm b/src/lib/Hydra/Controller/User.pm index a07a3c42..0ce5b2ff 100644 --- a/src/lib/Hydra/Controller/User.pm +++ b/src/lib/Hydra/Controller/User.pm @@ -37,7 +37,11 @@ sub login_POST { accessDenied($c, "Bad username or password.") } - currentUser_GET($self, $c); + $self->status_found( + $c, + location => $c->uri_for("current-user"), + entity => {} + ); } From 948b3cf0734a52f33b48ed6d6c3ffdf33376cdc2 Mon Sep 17 00:00:00 2001 From: Cole Helbling Date: Wed, 28 Apr 2021 09:25:26 -0700 Subject: [PATCH 7/7] Jobset: add HTTP API test --- t/Controller/Jobset/http.t | 176 +++++++++++++++++++++++++++++++++++++ 1 file changed, 176 insertions(+) create mode 100644 t/Controller/Jobset/http.t diff --git a/t/Controller/Jobset/http.t b/t/Controller/Jobset/http.t new file mode 100644 index 00000000..0561f3ac --- /dev/null +++ b/t/Controller/Jobset/http.t @@ -0,0 +1,176 @@ +use feature 'unicode_strings'; +use strict; +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); + +# This test verifies that creating, reading, updating, and deleting a jobset via +# the HTTP API works as expected. + +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'}); + +# Login and save cookie for future requests +my $req = request(POST '/login', + Referer => 'http://localhost/', + Content => { + username => 'alice', + password => 'foobar' + } +); +is($req->code, 302); +my $cookie = $req->header("set-cookie"); + + +subtest 'Create new jobset "job" as flake type' => sub { + my $jobsetcreate = request(PUT '/jobset/tests/job', + Accept => 'application/json', + Content_Type => 'application/json', + Cookie => $cookie, + Content => encode_json({ + enabled => 2, + visible => 1, + name => "job", + type => 1, + description => "test jobset", + flake => "github:nixos/nix", + checkinterval => 0, + schedulingshares => 100, + keepnr => 3 + }) + ); + ok($jobsetcreate->is_success); + is($jobsetcreate->header("location"), "http://localhost/jobset/tests/job"); +}; + + +subtest 'Read newly-created jobset "job"' => sub { + my $jobsetinfo = request(GET '/jobset/tests/job', + Accept => 'application/json', + ); + ok($jobsetinfo->is_success); + is(decode_json($jobsetinfo->content), { + checkinterval => 0, + description => "test jobset", + emailoverride => "", + enabled => 2, + enableemail => JSON::false, + errortime => undef, + errormsg => "", + fetcherrormsg => "", + flake => "github:nixos/nix", + hidden => JSON::false, + jobsetinputs => {}, + keepnr => 3, + lastcheckedtime => undef, + name => "job", + nixexprinput => "", + nixexprpath => "", + project => "tests", + schedulingshares => 100, + starttime => undef, + triggertime => undef, + type => 1 + }); +}; + + +subtest 'Update jobset "job" to legacy type' => sub { + my $jobsetupdate = request(PUT '/jobset/tests/job', + Accept => 'application/json', + Content_Type => 'application/json', + Cookie => $cookie, + Content => encode_json({ + enabled => 3, + visible => 1, + name => "job", + type => 0, + nixexprinput => "ofborg", + nixexprpath => "release.nix", + jobsetinputs => { + ofborg => { + name => "ofborg", + type => "git", + value => "https://github.com/NixOS/ofborg.git released" + } + }, + description => "test jobset", + checkinterval => 0, + schedulingshares => 50, + keepnr => 1 + }) + ); + ok($jobsetupdate->is_success); + + # Read newly-updated jobset "job" + my $jobsetinfo = request(GET '/jobset/tests/job', + Accept => 'application/json', + ); + ok($jobsetinfo->is_success); + is(decode_json($jobsetinfo->content), { + checkinterval => 0, + description => "test jobset", + emailoverride => "", + enabled => 3, + enableemail => JSON::false, + errortime => undef, + errormsg => "", + fetcherrormsg => "", + flake => "", + hidden => JSON::false, + jobsetinputs => { + ofborg => { + name => "ofborg", + type => "git", + emailresponsible => JSON::false, + jobsetinputalts => [ + "https://github.com/NixOS/ofborg.git released" + ] + } + }, + keepnr => 1, + lastcheckedtime => undef, + name => "job", + nixexprinput => "ofborg", + nixexprpath => "release.nix", + project => "tests", + schedulingshares => 50, + starttime => undef, + triggertime => undef, + type => 0 + }); +}; + + +subtest 'Delete jobset "job"' => sub { + my $jobsetinfo = request(DELETE '/jobset/tests/job', + Accept => 'application/json', + Cookie => $cookie + ); + ok($jobsetinfo->is_success); + + # Jobset "job" should no longer exist. + $jobsetinfo = request(GET '/jobset/tests/job', + Accept => 'application/json', + ); + ok(!$jobsetinfo->is_success); +}; + +done_testing;