Merge pull request #1145 from DeterminateSystems/eval-status
`eval_*` events: change the API to include IDs and use \t separators
This commit is contained in:
commit
1abe7f4d80
14 changed files with 753 additions and 17 deletions
|
@ -57,25 +57,25 @@ It is possible for subsequent deliveries of the same `build_finished` data to im
|
|||
|
||||
### `eval_started`
|
||||
|
||||
* **Payload:** Exactly three values, separated by the two-character string `\t` (ie: not a tab): an opaque trace ID representing this evaluation, the name of the project, and the name of the jobset.
|
||||
* **Payload:** Exactly two values, tab separated: an opaque trace ID representing this evaluation, and the ID of the jobset.
|
||||
* **When:** At the beginning of the evaluation phase for the jobset, before any work is done.
|
||||
* **Delivery Semantics:** Ephemeral. `hydra-notify` must be running to react to this event. No record of this event is stored.
|
||||
|
||||
### `eval_added`
|
||||
|
||||
* **Payload:** Exactly two values, separated by the two-character string `\t` (ie: not a tab): an opaque trace ID representing this evaluation, and the ID of the JobsetEval record.
|
||||
* **Payload:** Exactly three values, tab separated: an opaque trace ID representing this evaluation, the ID of the jobset, and the ID of the JobsetEval record.
|
||||
* **When:** After the evaluator fetches inputs and completes the evaluation successfully.
|
||||
* **Delivery Semantics:** Ephemeral. `hydra-notify` must be running to react to this event. No record of this event is stored.
|
||||
|
||||
### `eval_cached`
|
||||
|
||||
* **Payload:** Exactly one value: an opaque trace ID representing this evaluation.
|
||||
* **Payload:** Exactly three values: an opaque trace ID representing this evaluation, the ID of the jobset, and the ID of the previous identical evaluation.
|
||||
* **When:** After the evaluator fetches inputs, if none of the inputs changed.
|
||||
* **Delivery Semantics:** Ephemeral. `hydra-notify` must be running to react to this event. No record of this event is stored.
|
||||
|
||||
### `eval_failed`
|
||||
|
||||
* **Payload:** Exactly one value: an opaque trace ID representing this evaluation.
|
||||
* **Payload:** Exactly two values: an opaque trace ID representing this evaluation, and the ID of the jobset.
|
||||
* **When:** After any fetching any input fails, or any other evaluation error occurs.
|
||||
* **Delivery Semantics:** Ephemeral. `hydra-notify` must be running to react to this event. No record of this event is stored.
|
||||
|
||||
|
|
|
@ -7,6 +7,10 @@ use Hydra::Event::BuildQueued;
|
|||
use Hydra::Event::BuildStarted;
|
||||
use Hydra::Event::CachedBuildFinished;
|
||||
use Hydra::Event::CachedBuildQueued;
|
||||
use Hydra::Event::EvalAdded;
|
||||
use Hydra::Event::EvalCached;
|
||||
use Hydra::Event::EvalFailed;
|
||||
use Hydra::Event::EvalStarted;
|
||||
use Hydra::Event::StepFinished;
|
||||
|
||||
my %channels_to_events = (
|
||||
|
@ -15,6 +19,10 @@ my %channels_to_events = (
|
|||
build_started => \&Hydra::Event::BuildStarted::parse,
|
||||
cached_build_finished => \&Hydra::Event::CachedBuildFinished::parse,
|
||||
cached_build_queued => \&Hydra::Event::CachedBuildQueued::parse,
|
||||
eval_added => \&Hydra::Event::EvalAdded::parse,
|
||||
eval_cached => \&Hydra::Event::EvalCached::parse,
|
||||
eval_failed => \&Hydra::Event::EvalFailed::parse,
|
||||
eval_started => \&Hydra::Event::EvalStarted::parse,
|
||||
step_finished => \&Hydra::Event::StepFinished::parse,
|
||||
);
|
||||
|
||||
|
|
63
src/lib/Hydra/Event/EvalAdded.pm
Normal file
63
src/lib/Hydra/Event/EvalAdded.pm
Normal file
|
@ -0,0 +1,63 @@
|
|||
package Hydra::Event::EvalAdded;
|
||||
|
||||
use strict;
|
||||
use warnings;
|
||||
|
||||
sub parse :prototype(@) {
|
||||
unless (@_ == 3) {
|
||||
die "eval_added: payload takes exactly three arguments, but ", scalar(@_), " were given";
|
||||
}
|
||||
|
||||
my ($trace_id, $jobset_id, $evaluation_id) = @_;
|
||||
|
||||
unless ($jobset_id =~ /^\d+$/) {
|
||||
die "eval_added: payload argument jobset_id should be an integer, but '", $jobset_id, "' was given"
|
||||
}
|
||||
unless ($evaluation_id =~ /^\d+$/) {
|
||||
die "eval_added: payload argument evaluation_id should be an integer, but '", $evaluation_id, "' was given"
|
||||
}
|
||||
|
||||
return Hydra::Event::EvalAdded->new($trace_id, int($jobset_id), int($evaluation_id));
|
||||
}
|
||||
|
||||
sub new {
|
||||
my ($self, $trace_id, $jobset_id, $evaluation_id) = @_;
|
||||
return bless {
|
||||
"trace_id" => $trace_id,
|
||||
"jobset_id" => $jobset_id,
|
||||
"evaluation_id" => $evaluation_id,
|
||||
"jobset" => undef,
|
||||
"evaluation" => undef
|
||||
}, $self;
|
||||
}
|
||||
|
||||
sub interestedIn {
|
||||
my ($self, $plugin) = @_;
|
||||
return int(defined($plugin->can('evalAdded')));
|
||||
}
|
||||
|
||||
sub load {
|
||||
my ($self, $db) = @_;
|
||||
|
||||
if (!defined($self->{"jobset"})) {
|
||||
$self->{"jobset"} = $db->resultset('Jobsets')->find({ id => $self->{"jobset_id"}})
|
||||
or die "Jobset $self->{'jobset_id'} does not exist\n";
|
||||
}
|
||||
|
||||
if (!defined($self->{"evaluation"})) {
|
||||
$self->{"evaluation"} = $db->resultset('JobsetEvals')->find({ id => $self->{"evaluation_id"}})
|
||||
or die "Jobset $self->{'jobset_id'} does not exist\n";
|
||||
}
|
||||
}
|
||||
|
||||
sub execute {
|
||||
my ($self, $db, $plugin) = @_;
|
||||
|
||||
$self->load($db);
|
||||
|
||||
$plugin->evalAdded($self->{"trace_id"}, $self->{"jobset"}, $self->{"evaluation"});
|
||||
|
||||
return 1;
|
||||
}
|
||||
|
||||
1;
|
63
src/lib/Hydra/Event/EvalCached.pm
Normal file
63
src/lib/Hydra/Event/EvalCached.pm
Normal file
|
@ -0,0 +1,63 @@
|
|||
package Hydra::Event::EvalCached;
|
||||
|
||||
use strict;
|
||||
use warnings;
|
||||
|
||||
sub parse :prototype(@) {
|
||||
unless (@_ == 3) {
|
||||
die "eval_cached: payload takes exactly three arguments, but ", scalar(@_), " were given";
|
||||
}
|
||||
|
||||
my ($trace_id, $jobset_id, $evaluation_id) = @_;
|
||||
|
||||
unless ($jobset_id =~ /^\d+$/) {
|
||||
die "eval_cached: payload argument jobset_id should be an integer, but '", $jobset_id, "' was given"
|
||||
}
|
||||
unless ($evaluation_id =~ /^\d+$/) {
|
||||
die "eval_cached: payload argument evaluation_id should be an integer, but '", $evaluation_id, "' was given"
|
||||
}
|
||||
|
||||
return Hydra::Event::EvalCached->new($trace_id, int($jobset_id), int($evaluation_id));
|
||||
}
|
||||
|
||||
sub new {
|
||||
my ($self, $trace_id, $jobset_id, $evaluation_id) = @_;
|
||||
return bless {
|
||||
"trace_id" => $trace_id,
|
||||
"jobset_id" => $jobset_id,
|
||||
"evaluation_id" => $evaluation_id,
|
||||
"jobset" => undef,
|
||||
"evaluation" => undef
|
||||
}, $self;
|
||||
}
|
||||
|
||||
sub interestedIn {
|
||||
my ($self, $plugin) = @_;
|
||||
return int(defined($plugin->can('evalCached')));
|
||||
}
|
||||
|
||||
sub load {
|
||||
my ($self, $db) = @_;
|
||||
|
||||
if (!defined($self->{"jobset"})) {
|
||||
$self->{"jobset"} = $db->resultset('Jobsets')->find({ id => $self->{"jobset_id"}})
|
||||
or die "Jobset $self->{'jobset_id'} does not exist\n";
|
||||
}
|
||||
|
||||
if (!defined($self->{"evaluation"})) {
|
||||
$self->{"evaluation"} = $db->resultset('JobsetEvals')->find({ id => $self->{"evaluation_id"}})
|
||||
or die "Jobset $self->{'jobset_id'} does not exist\n";
|
||||
}
|
||||
}
|
||||
|
||||
sub execute {
|
||||
my ($self, $db, $plugin) = @_;
|
||||
|
||||
$self->load($db);
|
||||
|
||||
$plugin->evalCached($self->{"trace_id"}, $self->{"jobset"}, $self->{"evaluation"});
|
||||
|
||||
return 1;
|
||||
}
|
||||
|
||||
1;
|
53
src/lib/Hydra/Event/EvalFailed.pm
Normal file
53
src/lib/Hydra/Event/EvalFailed.pm
Normal file
|
@ -0,0 +1,53 @@
|
|||
package Hydra::Event::EvalFailed;
|
||||
|
||||
use strict;
|
||||
use warnings;
|
||||
|
||||
sub parse :prototype(@) {
|
||||
unless (@_ == 2) {
|
||||
die "eval_failed: payload takes two arguments, but ", scalar(@_), " were given";
|
||||
}
|
||||
|
||||
my ($trace_id, $jobset_id) = @_;
|
||||
|
||||
unless ($jobset_id =~ /^\d+$/) {
|
||||
die "eval_failed: payload argument should be an integer, but '", $jobset_id, "' was given"
|
||||
}
|
||||
|
||||
return Hydra::Event::EvalFailed->new($trace_id, int($jobset_id));
|
||||
}
|
||||
|
||||
sub new {
|
||||
my ($self, $trace_id, $jobset_id) = @_;
|
||||
return bless {
|
||||
"trace_id" => $trace_id,
|
||||
"jobset_id" => $jobset_id,
|
||||
"jobset" => undef
|
||||
}, $self;
|
||||
}
|
||||
|
||||
sub interestedIn {
|
||||
my ($self, $plugin) = @_;
|
||||
return int(defined($plugin->can('evalFailed')));
|
||||
}
|
||||
|
||||
sub load {
|
||||
my ($self, $db) = @_;
|
||||
|
||||
if (!defined($self->{"jobset"})) {
|
||||
$self->{"jobset"} = $db->resultset('Jobsets')->find({ id => $self->{"jobset_id"}})
|
||||
or die "Jobset $self->{'jobset_id'} does not exist\n";
|
||||
}
|
||||
}
|
||||
|
||||
sub execute {
|
||||
my ($self, $db, $plugin) = @_;
|
||||
|
||||
$self->load($db);
|
||||
|
||||
$plugin->evalFailed($self->{"trace_id"}, $self->{"jobset"});
|
||||
|
||||
return 1;
|
||||
}
|
||||
|
||||
1;
|
53
src/lib/Hydra/Event/EvalStarted.pm
Normal file
53
src/lib/Hydra/Event/EvalStarted.pm
Normal file
|
@ -0,0 +1,53 @@
|
|||
package Hydra::Event::EvalStarted;
|
||||
|
||||
use strict;
|
||||
use warnings;
|
||||
|
||||
sub parse :prototype(@) {
|
||||
unless (@_ == 2) {
|
||||
die "eval_started: payload takes two arguments, but ", scalar(@_), " were given";
|
||||
}
|
||||
|
||||
my ($trace_id, $jobset_id) = @_;
|
||||
|
||||
unless ($jobset_id =~ /^\d+$/) {
|
||||
die "eval_started: payload argument should be an integer, but '", $jobset_id, "' was given"
|
||||
}
|
||||
|
||||
return Hydra::Event::EvalStarted->new($trace_id, int($jobset_id));
|
||||
}
|
||||
|
||||
sub new {
|
||||
my ($self, $trace_id, $jobset_id) = @_;
|
||||
return bless {
|
||||
"trace_id" => $trace_id,
|
||||
"jobset_id" => $jobset_id,
|
||||
"jobset" => undef
|
||||
}, $self;
|
||||
}
|
||||
|
||||
sub interestedIn {
|
||||
my ($self, $plugin) = @_;
|
||||
return int(defined($plugin->can('evalStarted')));
|
||||
}
|
||||
|
||||
sub load {
|
||||
my ($self, $db) = @_;
|
||||
|
||||
if (!defined($self->{"jobset"})) {
|
||||
$self->{"jobset"} = $db->resultset('Jobsets')->find({ id => $self->{"jobset_id"}})
|
||||
or die "Jobset $self->{'jobset_id'} does not exist\n";
|
||||
}
|
||||
}
|
||||
|
||||
sub execute {
|
||||
my ($self, $db, $plugin) = @_;
|
||||
|
||||
$self->load($db);
|
||||
|
||||
$plugin->evalStarted($self->{"trace_id"}, $self->{"jobset"});
|
||||
|
||||
return 1;
|
||||
}
|
||||
|
||||
1;
|
|
@ -31,6 +31,27 @@ sub instantiate {
|
|||
# See the tests in t/Event/*.t for arguments, and the documentation for
|
||||
# notify events for semantics.
|
||||
#
|
||||
|
||||
# # Called when an evaluation of $jobset has begun.
|
||||
# sub evalStarted {
|
||||
# my ($self, $traceID, $jobset) = @_;
|
||||
# }
|
||||
|
||||
# # Called when an evaluation of $jobset determined the inputs had not changed.
|
||||
# sub evalCached {
|
||||
# my ($self, $traceID, $jobset, $evaluation) = @_;
|
||||
# }
|
||||
|
||||
# # Called when an evaluation of $jobset failed.
|
||||
# sub evalFailed {
|
||||
# my ($self, $traceID, $jobset) = @_;
|
||||
# }
|
||||
|
||||
# # Called when $evaluation of $jobset has completed successfully.
|
||||
# sub evalAdded {
|
||||
# my ($self, $traceID, $jobset, $evaluation) = @_;
|
||||
# }
|
||||
|
||||
# # Called when build $build has been queued.
|
||||
# sub buildQueued {
|
||||
# my ($self, $build) = @_;
|
||||
|
|
|
@ -654,7 +654,7 @@ sub checkJobsetWrapped {
|
|||
print STDERR $fetchError;
|
||||
$db->txn_do(sub {
|
||||
$jobset->update({ lastcheckedtime => time, fetcherrormsg => $fetchError }) if !$dryRun;
|
||||
$db->storage->dbh->do("notify eval_failed, ?", undef, join('\t', $tmpId));
|
||||
$db->storage->dbh->do("notify eval_failed, ?", undef, join("\t", $tmpId, $jobset->get_column('id')));
|
||||
});
|
||||
return;
|
||||
}
|
||||
|
@ -670,7 +670,11 @@ sub checkJobsetWrapped {
|
|||
Net::Statsd::increment("hydra.evaluator.unchanged_checkouts");
|
||||
$db->txn_do(sub {
|
||||
$jobset->update({ lastcheckedtime => time, fetcherrormsg => undef });
|
||||
$db->storage->dbh->do("notify eval_cached, ?", undef, join('\t', $tmpId));
|
||||
$db->storage->dbh->do("notify eval_cached, ?", undef, join("\t",
|
||||
$tmpId,
|
||||
$jobset->get_column('id'),
|
||||
$prevEval->get_column('id'))
|
||||
);
|
||||
});
|
||||
return;
|
||||
}
|
||||
|
@ -767,7 +771,7 @@ sub checkJobsetWrapped {
|
|||
});
|
||||
|
||||
$db->storage->dbh->do("notify eval_added, ?", undef,
|
||||
join('\t', $tmpId, $ev->id));
|
||||
join("\t", $tmpId, $jobset->get_column('id'), $ev->id));
|
||||
|
||||
if ($jobsetChanged) {
|
||||
# Create JobsetEvalMembers mappings.
|
||||
|
@ -861,7 +865,7 @@ sub checkJobset {
|
|||
my $tmpId = "${startTime}.$$";
|
||||
|
||||
$db->storage->dbh->do("notify eval_started, ?", undef,
|
||||
join('\t', $tmpId, $jobset->get_column('project'), $jobset->name));
|
||||
join("\t", $tmpId, $jobset->get_column('id')));
|
||||
|
||||
eval {
|
||||
checkJobsetWrapped($jobset, $tmpId);
|
||||
|
@ -878,7 +882,7 @@ sub checkJobset {
|
|||
$db->txn_do(sub {
|
||||
$jobset->update({lastcheckedtime => $eventTime});
|
||||
setJobsetError($jobset, $checkError, $eventTime);
|
||||
$db->storage->dbh->do("notify eval_failed, ?", undef, join('\t', $tmpId));
|
||||
$db->storage->dbh->do("notify eval_failed, ?", undef, join("\t", $tmpId, $jobset->get_column('id')));
|
||||
}) if !$dryRun;
|
||||
$failed = 1;
|
||||
}
|
||||
|
|
|
@ -98,9 +98,14 @@ $listener->subscribe("build_queued");
|
|||
$listener->subscribe("build_started");
|
||||
$listener->subscribe("cached_build_finished");
|
||||
$listener->subscribe("cached_build_queued");
|
||||
$listener->subscribe("eval_added");
|
||||
$listener->subscribe("eval_cached");
|
||||
$listener->subscribe("eval_failed");
|
||||
$listener->subscribe("eval_started");
|
||||
$listener->subscribe("hydra_notify_dump_metrics");
|
||||
$listener->subscribe("step_finished");
|
||||
|
||||
|
||||
# Process builds that finished while hydra-notify wasn't running.
|
||||
for my $build ($db->resultset('Builds')->search(
|
||||
{ notificationpendingsince => { '!=', undef } }))
|
||||
|
|
112
t/Hydra/Event/EvalAdded.t
Normal file
112
t/Hydra/Event/EvalAdded.t
Normal file
|
@ -0,0 +1,112 @@
|
|||
use strict;
|
||||
use warnings;
|
||||
use Setup;
|
||||
use Hydra::Event;
|
||||
use Hydra::Event::EvalAdded;
|
||||
use Test2::V0;
|
||||
use Test2::Tools::Exception;
|
||||
use Test2::Tools::Mock qw(mock_obj);
|
||||
|
||||
|
||||
my $ctx = test_context();
|
||||
my $builds = $ctx->makeAndEvaluateJobset(
|
||||
expression => "basic.nix",
|
||||
build => 1
|
||||
);
|
||||
|
||||
|
||||
subtest "Parsing eval_added" => sub {
|
||||
like(
|
||||
dies { Hydra::Event::parse_payload("eval_added", "") },
|
||||
qr/three arguments/,
|
||||
"empty payload"
|
||||
);
|
||||
like(
|
||||
dies { Hydra::Event::parse_payload("eval_added", "abc123") },
|
||||
qr/three arguments/,
|
||||
"one argument"
|
||||
);
|
||||
like(
|
||||
dies { Hydra::Event::parse_payload("eval_added", "abc123\tabc123") },
|
||||
qr/three arguments/,
|
||||
"two arguments"
|
||||
);
|
||||
like(
|
||||
dies { Hydra::Event::parse_payload("eval_added", "abc123\tabc123\tabc123\tabc123") },
|
||||
qr/three arguments/,
|
||||
"four arguments"
|
||||
);
|
||||
like(
|
||||
dies { Hydra::Event::parse_payload("eval_added", "abc123\tabc123\t123") },
|
||||
qr/should be an integer/,
|
||||
"not an integer: second position"
|
||||
);
|
||||
like(
|
||||
dies { Hydra::Event::parse_payload("eval_added", "abc123\t123\tabc123") },
|
||||
qr/should be an integer/,
|
||||
"not an integer: third position"
|
||||
);
|
||||
is(
|
||||
Hydra::Event::parse_payload("eval_added", "abc123\t123\t456"),
|
||||
Hydra::Event::EvalAdded->new("abc123", 123, 456)
|
||||
);
|
||||
};
|
||||
|
||||
subtest "interested" => sub {
|
||||
my $event = Hydra::Event::EvalAdded->new("abc123", 123, 456);
|
||||
|
||||
subtest "A plugin which does not implement the API" => sub {
|
||||
my $plugin = {};
|
||||
my $mock = mock_obj $plugin => ();
|
||||
|
||||
is($event->interestedIn($plugin), 0, "The plugin is not interesting.");
|
||||
};
|
||||
|
||||
subtest "A plugin which does implement the API" => sub {
|
||||
my $plugin = {};
|
||||
my $mock = mock_obj $plugin => (
|
||||
add => [
|
||||
"evalAdded" => sub {}
|
||||
]
|
||||
);
|
||||
|
||||
is($event->interestedIn($plugin), 1, "The plugin is interesting.");
|
||||
};
|
||||
};
|
||||
|
||||
subtest "load" => sub {
|
||||
my $jobset = $builds->{"empty_dir"}->jobset;
|
||||
my $evaluation = $builds->{"empty_dir"}->jobsetevals->first();
|
||||
|
||||
my $event = Hydra::Event::EvalAdded->new("traceID", $jobset->id, $evaluation->id);
|
||||
|
||||
$event->load($ctx->db());
|
||||
is($event->{"trace_id"}, "traceID", "The Trace ID matches");
|
||||
is($event->{"jobset_id"}, $jobset->id, "The Jobset ID matches");
|
||||
is($event->{"evaluation_id"}, $evaluation->id, "The Evaluation ID matches");
|
||||
|
||||
|
||||
# Create a fake "plugin" with a evalAdded sub, the sub sets these
|
||||
# "globals"
|
||||
my $passedTraceID;
|
||||
my $passedJobset;
|
||||
my $passedEvaluation;
|
||||
my $plugin = {};
|
||||
my $mock = mock_obj $plugin => (
|
||||
add => [
|
||||
"evalAdded" => sub {
|
||||
my ($self, $traceID, $jobset, $evaluation) = @_;
|
||||
$passedTraceID = $traceID;
|
||||
$passedJobset = $jobset;
|
||||
$passedEvaluation = $evaluation;
|
||||
}
|
||||
]
|
||||
);
|
||||
|
||||
$event->execute($ctx->db(), $plugin);
|
||||
is($passedTraceID, "traceID", "We get the expected trace ID");
|
||||
is($passedJobset->id, $jobset->id, "The correct jobset is passed");
|
||||
is($passedEvaluation->id, $evaluation->id, "The correct evaluation is passed");
|
||||
};
|
||||
|
||||
done_testing;
|
112
t/Hydra/Event/EvalCached.t
Normal file
112
t/Hydra/Event/EvalCached.t
Normal file
|
@ -0,0 +1,112 @@
|
|||
use strict;
|
||||
use warnings;
|
||||
use Setup;
|
||||
use Hydra::Event;
|
||||
use Hydra::Event::EvalCached;
|
||||
use Test2::V0;
|
||||
use Test2::Tools::Exception;
|
||||
use Test2::Tools::Mock qw(mock_obj);
|
||||
|
||||
|
||||
my $ctx = test_context();
|
||||
my $builds = $ctx->makeAndEvaluateJobset(
|
||||
expression => "basic.nix",
|
||||
build => 1
|
||||
);
|
||||
|
||||
|
||||
subtest "Parsing eval_cached" => sub {
|
||||
like(
|
||||
dies { Hydra::Event::parse_payload("eval_cached", "") },
|
||||
qr/three arguments/,
|
||||
"empty payload"
|
||||
);
|
||||
like(
|
||||
dies { Hydra::Event::parse_payload("eval_cached", "abc123") },
|
||||
qr/three arguments/,
|
||||
"one argument"
|
||||
);
|
||||
like(
|
||||
dies { Hydra::Event::parse_payload("eval_cached", "abc123\tabc123") },
|
||||
qr/three arguments/,
|
||||
"two arguments"
|
||||
);
|
||||
like(
|
||||
dies { Hydra::Event::parse_payload("eval_cached", "abc123\tabc123\tabc123\tabc123") },
|
||||
qr/three arguments/,
|
||||
"four arguments"
|
||||
);
|
||||
like(
|
||||
dies { Hydra::Event::parse_payload("eval_cached", "abc123\tabc123\t123") },
|
||||
qr/should be an integer/,
|
||||
"not an integer: second position"
|
||||
);
|
||||
like(
|
||||
dies { Hydra::Event::parse_payload("eval_cached", "abc123\t123\tabc123") },
|
||||
qr/should be an integer/,
|
||||
"not an integer: third position"
|
||||
);
|
||||
is(
|
||||
Hydra::Event::parse_payload("eval_cached", "abc123\t123\t456"),
|
||||
Hydra::Event::EvalCached->new("abc123", 123, 456)
|
||||
);
|
||||
};
|
||||
|
||||
subtest "interested" => sub {
|
||||
my $event = Hydra::Event::EvalCached->new("abc123", 123, 456);
|
||||
|
||||
subtest "A plugin which does not implement the API" => sub {
|
||||
my $plugin = {};
|
||||
my $mock = mock_obj $plugin => ();
|
||||
|
||||
is($event->interestedIn($plugin), 0, "The plugin is not interesting.");
|
||||
};
|
||||
|
||||
subtest "A plugin which does implement the API" => sub {
|
||||
my $plugin = {};
|
||||
my $mock = mock_obj $plugin => (
|
||||
add => [
|
||||
"evalCached" => sub {}
|
||||
]
|
||||
);
|
||||
|
||||
is($event->interestedIn($plugin), 1, "The plugin is interesting.");
|
||||
};
|
||||
};
|
||||
|
||||
subtest "load" => sub {
|
||||
my $jobset = $builds->{"empty_dir"}->jobset;
|
||||
my $evaluation = $builds->{"empty_dir"}->jobsetevals->first();
|
||||
|
||||
my $event = Hydra::Event::EvalCached->new("traceID", $jobset->id, $evaluation->id);
|
||||
|
||||
$event->load($ctx->db());
|
||||
is($event->{"trace_id"}, "traceID", "The Trace ID matches");
|
||||
is($event->{"jobset_id"}, $jobset->id, "The Jobset ID matches");
|
||||
is($event->{"evaluation_id"}, $evaluation->id, "The Evaluation ID matches");
|
||||
|
||||
|
||||
# Create a fake "plugin" with a evalCached sub, the sub sets these
|
||||
# "globals"
|
||||
my $passedTraceID;
|
||||
my $passedJobset;
|
||||
my $passedEvaluation;
|
||||
my $plugin = {};
|
||||
my $mock = mock_obj $plugin => (
|
||||
add => [
|
||||
"evalCached" => sub {
|
||||
my ($self, $traceID, $jobset, $evaluation) = @_;
|
||||
$passedTraceID = $traceID;
|
||||
$passedJobset = $jobset;
|
||||
$passedEvaluation = $evaluation;
|
||||
}
|
||||
]
|
||||
);
|
||||
|
||||
$event->execute($ctx->db(), $plugin);
|
||||
is($passedTraceID, "traceID", "We get the expected trace ID");
|
||||
is($passedJobset->id, $jobset->id, "The correct jobset is passed");
|
||||
is($passedEvaluation->id, $evaluation->id, "The correct evaluation is passed");
|
||||
};
|
||||
|
||||
done_testing;
|
94
t/Hydra/Event/EvalFailed.t
Normal file
94
t/Hydra/Event/EvalFailed.t
Normal file
|
@ -0,0 +1,94 @@
|
|||
use strict;
|
||||
use warnings;
|
||||
use Setup;
|
||||
use Hydra::Event;
|
||||
use Hydra::Event::EvalFailed;
|
||||
use Test2::V0;
|
||||
use Test2::Tools::Exception;
|
||||
use Test2::Tools::Mock qw(mock_obj);
|
||||
|
||||
my $ctx = test_context();
|
||||
|
||||
my $builds = $ctx->makeAndEvaluateJobset(
|
||||
expression => "basic.nix",
|
||||
build => 1
|
||||
);
|
||||
|
||||
subtest "Parsing eval_failed" => sub {
|
||||
like(
|
||||
dies { Hydra::Event::parse_payload("eval_failed", "") },
|
||||
qr/two arguments/,
|
||||
"empty payload"
|
||||
);
|
||||
like(
|
||||
dies { Hydra::Event::parse_payload("eval_failed", "abc123") },
|
||||
qr/two arguments/,
|
||||
"one argument"
|
||||
);
|
||||
like(
|
||||
dies { Hydra::Event::parse_payload("eval_failed", "abc123\tabc123\tabc123") },
|
||||
qr/two arguments/,
|
||||
"three arguments"
|
||||
);
|
||||
like(
|
||||
dies { Hydra::Event::parse_payload("eval_failed", "abc123\tabc123") },
|
||||
qr/should be an integer/,
|
||||
"not an integer: second argument"
|
||||
);
|
||||
is(
|
||||
Hydra::Event::parse_payload("eval_failed", "abc123\t456"),
|
||||
Hydra::Event::EvalFailed->new("abc123", 456)
|
||||
);
|
||||
};
|
||||
|
||||
subtest "interested" => sub {
|
||||
my $event = Hydra::Event::EvalFailed->new(123, []);
|
||||
|
||||
subtest "A plugin which does not implement the API" => sub {
|
||||
my $plugin = {};
|
||||
my $mock = mock_obj $plugin => ();
|
||||
|
||||
is($event->interestedIn($plugin), 0, "The plugin is not interesting.");
|
||||
};
|
||||
|
||||
subtest "A plugin which does implement the API" => sub {
|
||||
my $plugin = {};
|
||||
my $mock = mock_obj $plugin => (
|
||||
add => [
|
||||
"evalFailed" => sub {}
|
||||
]
|
||||
);
|
||||
|
||||
is($event->interestedIn($plugin), 1, "The plugin is interesting.");
|
||||
};
|
||||
};
|
||||
|
||||
subtest "load" => sub {
|
||||
my $jobset = $builds->{"empty_dir"}->jobset;
|
||||
|
||||
my $event = Hydra::Event::EvalFailed->new("traceID", $jobset->id);
|
||||
|
||||
$event->load($ctx->db());
|
||||
is($event->{"jobset"}->get_column("id"), $jobset->id, "The jobset record matches.");
|
||||
|
||||
# Create a fake "plugin" with a evalFailed sub, the sub sets this
|
||||
# "global" passedTraceID, passedJobset
|
||||
my $passedTraceID;
|
||||
my $passedJobset;
|
||||
my $plugin = {};
|
||||
my $mock = mock_obj $plugin => (
|
||||
add => [
|
||||
"evalFailed" => sub {
|
||||
my ($self, $traceID, $jobset) = @_;
|
||||
$passedTraceID = $traceID;
|
||||
$passedJobset = $jobset;
|
||||
}
|
||||
]
|
||||
);
|
||||
|
||||
$event->execute($ctx->db(), $plugin);
|
||||
is($passedTraceID, "traceID", "The plugin is told what the trace ID was");
|
||||
is($passedJobset->get_column("id"), $jobset->id, "The plugin's evalFailed hook is called with the right jobset");
|
||||
};
|
||||
|
||||
done_testing;
|
94
t/Hydra/Event/EvalStarted.t
Normal file
94
t/Hydra/Event/EvalStarted.t
Normal file
|
@ -0,0 +1,94 @@
|
|||
use strict;
|
||||
use warnings;
|
||||
use Setup;
|
||||
use Hydra::Event;
|
||||
use Hydra::Event::EvalStarted;
|
||||
use Test2::V0;
|
||||
use Test2::Tools::Exception;
|
||||
use Test2::Tools::Mock qw(mock_obj);
|
||||
|
||||
my $ctx = test_context();
|
||||
|
||||
my $builds = $ctx->makeAndEvaluateJobset(
|
||||
expression => "basic.nix",
|
||||
build => 1
|
||||
);
|
||||
|
||||
subtest "Parsing eval_started" => sub {
|
||||
like(
|
||||
dies { Hydra::Event::parse_payload("eval_started", "") },
|
||||
qr/two arguments/,
|
||||
"empty payload"
|
||||
);
|
||||
like(
|
||||
dies { Hydra::Event::parse_payload("eval_started", "abc123") },
|
||||
qr/two arguments/,
|
||||
"one argument"
|
||||
);
|
||||
like(
|
||||
dies { Hydra::Event::parse_payload("eval_started", "abc123\tabc123\tabc123") },
|
||||
qr/two arguments/,
|
||||
"three arguments"
|
||||
);
|
||||
like(
|
||||
dies { Hydra::Event::parse_payload("eval_started", "abc123\tabc123") },
|
||||
qr/should be an integer/,
|
||||
"not an integer: second argument"
|
||||
);
|
||||
is(
|
||||
Hydra::Event::parse_payload("eval_started", "abc123\t456"),
|
||||
Hydra::Event::EvalStarted->new("abc123", 456)
|
||||
);
|
||||
};
|
||||
|
||||
subtest "interested" => sub {
|
||||
my $event = Hydra::Event::EvalStarted->new(123, []);
|
||||
|
||||
subtest "A plugin which does not implement the API" => sub {
|
||||
my $plugin = {};
|
||||
my $mock = mock_obj $plugin => ();
|
||||
|
||||
is($event->interestedIn($plugin), 0, "The plugin is not interesting.");
|
||||
};
|
||||
|
||||
subtest "A plugin which does implement the API" => sub {
|
||||
my $plugin = {};
|
||||
my $mock = mock_obj $plugin => (
|
||||
add => [
|
||||
"evalStarted" => sub {}
|
||||
]
|
||||
);
|
||||
|
||||
is($event->interestedIn($plugin), 1, "The plugin is interesting.");
|
||||
};
|
||||
};
|
||||
|
||||
subtest "load" => sub {
|
||||
my $jobset = $builds->{"empty_dir"}->jobset;
|
||||
|
||||
my $event = Hydra::Event::EvalStarted->new("traceID", $jobset->id);
|
||||
|
||||
$event->load($ctx->db());
|
||||
is($event->{"jobset"}->get_column("id"), $jobset->id, "The jobset record matches.");
|
||||
|
||||
# Create a fake "plugin" with a evalStarted sub, the sub sets this
|
||||
# "global" passedTraceID, passedJobset
|
||||
my $passedTraceID;
|
||||
my $passedJobset;
|
||||
my $plugin = {};
|
||||
my $mock = mock_obj $plugin => (
|
||||
add => [
|
||||
"evalStarted" => sub {
|
||||
my ($self, $traceID, $jobset) = @_;
|
||||
$passedTraceID = $traceID;
|
||||
$passedJobset = $jobset;
|
||||
}
|
||||
]
|
||||
);
|
||||
|
||||
$event->execute($ctx->db(), $plugin);
|
||||
is($passedTraceID, "traceID", "The plugin is told what the trace ID was");
|
||||
is($passedJobset->get_column("id"), $jobset->id, "The plugin's evalStarted hook is called with the right jobset");
|
||||
};
|
||||
|
||||
done_testing;
|
|
@ -5,6 +5,24 @@ use Setup;
|
|||
use Test2::V0;
|
||||
use File::Copy;
|
||||
use Hydra::PostgresListener;
|
||||
use Hydra::Event;
|
||||
|
||||
# expectEvent(Hydra::PostgresLister, name of the channel to expect, a sub which gets the parsed event)
|
||||
sub expectEvent {
|
||||
my ($listener, $expectedChannel, $then) = @_;
|
||||
my $message = $listener->block_for_messages(0)->();
|
||||
|
||||
my $channel = $message->{"channel"};
|
||||
|
||||
if ($channel eq $expectedChannel) {
|
||||
my $event = Hydra::Event->new_event($message->{"channel"}, $message->{"payload"});
|
||||
local $_ = $event->{event};
|
||||
$then->();
|
||||
} else {
|
||||
is($expectedChannel, $channel, "Expecting a message on channel $channel");
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
my $ctx = test_context(
|
||||
hydra_config => q|
|
||||
|
@ -36,22 +54,45 @@ my $builds = $ctx->makeAndEvaluateJobset(
|
|||
jobsdir => $jobsetdir,
|
||||
build => 0
|
||||
);
|
||||
my $jobset = $builds->{"stable-job-queued"}->jobset;
|
||||
my $evaluation = $builds->{"stable-job-queued"}->jobsetevals->first();
|
||||
|
||||
subtest "on the initial evaluation" => sub {
|
||||
is($listener->block_for_messages(0)->()->{"channel"}, "eval_started", "every eval starts with a notification");
|
||||
expectEvent($listener, "eval_started", sub {
|
||||
isnt($_->{"trace_id"}, "", "We got a trace ID");
|
||||
is($_->{"jobset_id"}, $jobset->get_column('id'), "the jobset ID matches");
|
||||
});
|
||||
|
||||
is($listener->block_for_messages(0)->()->{"channel"}, "build_queued", "expect 1/4 builds being queued");
|
||||
is($listener->block_for_messages(0)->()->{"channel"}, "build_queued", "expect 2/4 builds being queued");
|
||||
is($listener->block_for_messages(0)->()->{"channel"}, "build_queued", "expect 3/4 builds being queued");
|
||||
is($listener->block_for_messages(0)->()->{"channel"}, "build_queued", "expect 4/4 builds being queued");
|
||||
is($listener->block_for_messages(0)->()->{"channel"}, "eval_added", "the evaluation has completed");
|
||||
|
||||
expectEvent($listener, "eval_added", sub {
|
||||
is($_->{"jobset_id"}, $jobset->get_column('id'), "the jobset ID matches");
|
||||
is($_->{"evaluation_id"}, $evaluation->get_column('id'), "the evaluation ID matches");
|
||||
});
|
||||
is($listener->block_for_messages(0)->()->{"channel"}, "builds_added", "new builds have been scheduled");
|
||||
is($listener->block_for_messages(0)->(), undef, "there are no more messages from the evaluator");
|
||||
};
|
||||
|
||||
subtest "on a subsequent, totally cached / unchanged evaluation" => sub {
|
||||
ok(evalSucceeds($builds->{"variable-job"}->jobset), "evaluating for the second time");
|
||||
is($listener->block_for_messages(0)->()->{"channel"}, "eval_started", "an evaluation has started");
|
||||
is($listener->block_for_messages(0)->()->{"channel"}, "eval_cached", "the evaluation finished and nothing changed");
|
||||
ok(evalSucceeds($jobset), "evaluating for the second time");
|
||||
my $evaluation = $builds->{"stable-job-queued"}->jobsetevals->first();
|
||||
|
||||
my $traceID;
|
||||
expectEvent($listener, "eval_started", sub {
|
||||
isnt($_->{"trace_id"}, "", "We got a trace ID");
|
||||
$traceID = $_->{"trace_id"};
|
||||
is($_->{"jobset_id"}, $jobset->get_column('id'), "the jobset ID matches");
|
||||
});
|
||||
|
||||
expectEvent($listener, "eval_cached", sub {
|
||||
is($_->{"trace_id"}, $traceID, "Trace ID matches");
|
||||
is($_->{"jobset_id"}, $jobset->get_column('id'), "the jobset ID matches");
|
||||
is($_->{"evaluation_id"}, $evaluation->get_column('id'), "the evaluation ID matches");
|
||||
});
|
||||
|
||||
is($listener->block_for_messages(0)->(), undef, "there are no more messages from the evaluator");
|
||||
};
|
||||
|
||||
|
@ -67,7 +108,9 @@ subtest "on a fresh evaluation with changed sources" => sub {
|
|||
$builds->{"stable-job-failing"}->discard_changes();
|
||||
|
||||
ok(evalSucceeds($builds->{"variable-job"}->jobset), "evaluating for the third time");
|
||||
is($listener->block_for_messages(0)->()->{"channel"}, "eval_started", "the evaluation started");
|
||||
expectEvent($listener, "eval_started", sub {
|
||||
is($_->{"jobset_id"}, $jobset->get_column('id'), "the jobset ID matches");
|
||||
});
|
||||
|
||||
# The order of builds is randomized when writing to the database,
|
||||
# so we can't expect the list in any specific order here.
|
||||
|
@ -106,8 +149,19 @@ subtest "on a fresh evaluation with corrupted sources" => sub {
|
|||
close $fh;
|
||||
|
||||
ok(evalFails($builds->{"variable-job"}->jobset), "evaluating the corrupted job");
|
||||
is($listener->block_for_messages(0)->()->{"channel"}, "eval_started", "the evaluation started");
|
||||
is($listener->block_for_messages(0)->()->{"channel"}, "eval_failed", "the evaluation failed");
|
||||
|
||||
my $traceID;
|
||||
expectEvent($listener, "eval_started", sub {
|
||||
isnt($_->{"trace_id"}, "", "We got a trace ID");
|
||||
$traceID = $_->{"trace_id"};
|
||||
is($_->{"jobset_id"}, $jobset->get_column('id'), "the jobset ID matches");
|
||||
});
|
||||
|
||||
expectEvent($listener, "eval_failed", sub {
|
||||
is($_->{"trace_id"}, $traceID, "Trace ID matches");
|
||||
is($_->{"jobset_id"}, $jobset->get_column('id'), "the jobset ID matches");
|
||||
});
|
||||
|
||||
is($listener->block_for_messages(0)->(), undef, "there are no more messages from the evaluator");
|
||||
|
||||
};
|
||||
|
|
Loading…
Reference in a new issue