forked from lix-project/hydra
Merge pull request #280 from shlevy/github-status-api
Add a plugin to interact with the github status API.
This commit is contained in:
commit
ef72569cc3
|
@ -85,9 +85,16 @@ State::StepResult State::doBuildStep(nix::ref<Store> destStore, Step::ptr step,
|
||||||
return sMaybeCancelled;
|
return sMaybeCancelled;
|
||||||
}
|
}
|
||||||
|
|
||||||
for (auto build2 : dependents)
|
for (auto build2 : dependents) {
|
||||||
if (build2->drvPath == step->drvPath) { build = build2; break; }
|
if (build2->drvPath == step->drvPath) {
|
||||||
|
build = build2;
|
||||||
|
{
|
||||||
|
auto notificationSenderQueue_(notificationSenderQueue.lock());
|
||||||
|
notificationSenderQueue_->push(NotificationItem{NotificationItem::Type::Started, build->id});
|
||||||
|
}
|
||||||
|
notificationSenderWakeup.notify_one();
|
||||||
|
}
|
||||||
|
}
|
||||||
if (!build) build = *dependents.begin();
|
if (!build) build = *dependents.begin();
|
||||||
|
|
||||||
printMsg(lvlInfo, format("performing step ‘%1%’ on ‘%2%’ (needed by build %3% and %4% others)")
|
printMsg(lvlInfo, format("performing step ‘%1%’ on ‘%2%’ (needed by build %3% and %4% others)")
|
||||||
|
@ -265,7 +272,7 @@ State::StepResult State::doBuildStep(nix::ref<Store> destStore, Step::ptr step,
|
||||||
for (auto id : buildIDs) {
|
for (auto id : buildIDs) {
|
||||||
{
|
{
|
||||||
auto notificationSenderQueue_(notificationSenderQueue.lock());
|
auto notificationSenderQueue_(notificationSenderQueue.lock());
|
||||||
notificationSenderQueue_->push(NotificationItem(id, std::vector<BuildID>()));
|
notificationSenderQueue_->push(NotificationItem{NotificationItem::Type::Finished, id});
|
||||||
}
|
}
|
||||||
notificationSenderWakeup.notify_one();
|
notificationSenderWakeup.notify_one();
|
||||||
}
|
}
|
||||||
|
@ -387,7 +394,7 @@ State::StepResult State::doBuildStep(nix::ref<Store> destStore, Step::ptr step,
|
||||||
/* Send notification about this build and its dependents. */
|
/* Send notification about this build and its dependents. */
|
||||||
{
|
{
|
||||||
auto notificationSenderQueue_(notificationSenderQueue.lock());
|
auto notificationSenderQueue_(notificationSenderQueue.lock());
|
||||||
notificationSenderQueue_->push(NotificationItem(build->id, dependentIDs));
|
notificationSenderQueue_->push(NotificationItem{NotificationItem::Type::Finished, build->id, dependentIDs});
|
||||||
}
|
}
|
||||||
notificationSenderWakeup.notify_one();
|
notificationSenderWakeup.notify_one();
|
||||||
|
|
||||||
|
|
|
@ -480,11 +480,11 @@ void State::notificationSender()
|
||||||
notificationSenderQueue_->pop();
|
notificationSenderQueue_->pop();
|
||||||
}
|
}
|
||||||
|
|
||||||
printMsg(lvlChatty, format("sending notification about build %1%") % item.first);
|
printMsg(lvlChatty, format("sending notification about build %1%") % item.id);
|
||||||
|
|
||||||
Pid pid = startProcess([&]() {
|
Pid pid = startProcess([&]() {
|
||||||
Strings argv({"hydra-notify", "build", std::to_string(item.first)});
|
Strings argv({"hydra-notify", item.type == NotificationItem::Type::Started ? "build-started" : "build-finished", std::to_string(item.id)});
|
||||||
for (auto id : item.second)
|
for (auto id : item.dependentIds)
|
||||||
argv.push_back(std::to_string(id));
|
argv.push_back(std::to_string(id));
|
||||||
execvp("hydra-notify", (char * *) stringsToCharPtrs(argv).data()); // FIXME: remove cast
|
execvp("hydra-notify", (char * *) stringsToCharPtrs(argv).data()); // FIXME: remove cast
|
||||||
throw SysError("cannot start hydra-notify");
|
throw SysError("cannot start hydra-notify");
|
||||||
|
@ -494,7 +494,7 @@ void State::notificationSender()
|
||||||
|
|
||||||
if (res != 0)
|
if (res != 0)
|
||||||
throw Error(format("hydra-build returned exit code %1% notifying about build %2%")
|
throw Error(format("hydra-build returned exit code %1% notifying about build %2%")
|
||||||
% res % item.first);
|
% res % item.id);
|
||||||
|
|
||||||
} catch (std::exception & e) {
|
} catch (std::exception & e) {
|
||||||
printMsg(lvlError, format("notification sender: %1%") % e.what());
|
printMsg(lvlError, format("notification sender: %1%") % e.what());
|
||||||
|
|
|
@ -318,7 +318,16 @@ private:
|
||||||
killed before it has finished sending notifications about a
|
killed before it has finished sending notifications about a
|
||||||
build, then the notifications may be lost. It would be better
|
build, then the notifications may be lost. It would be better
|
||||||
to mark builds with pending notification in the database. */
|
to mark builds with pending notification in the database. */
|
||||||
typedef std::pair<BuildID, std::vector<BuildID>> NotificationItem;
|
struct NotificationItem
|
||||||
|
{
|
||||||
|
enum class Type : char {
|
||||||
|
Started,
|
||||||
|
Finished
|
||||||
|
};
|
||||||
|
Type type;
|
||||||
|
BuildID id;
|
||||||
|
std::vector<BuildID> dependentIds;
|
||||||
|
};
|
||||||
nix::Sync<std::queue<NotificationItem>> notificationSenderQueue;
|
nix::Sync<std::queue<NotificationItem>> notificationSenderQueue;
|
||||||
std::condition_variable notificationSenderWakeup;
|
std::condition_variable notificationSenderWakeup;
|
||||||
|
|
||||||
|
|
|
@ -5,8 +5,21 @@ use Exporter;
|
||||||
|
|
||||||
our @ISA = qw(Exporter);
|
our @ISA = qw(Exporter);
|
||||||
our @EXPORT = qw(
|
our @EXPORT = qw(
|
||||||
|
notifyBuildStarted
|
||||||
notifyBuildFinished);
|
notifyBuildFinished);
|
||||||
|
|
||||||
|
sub notifyBuildStarted {
|
||||||
|
my ($plugins, $build) = @_;
|
||||||
|
foreach my $plugin (@{$plugins}) {
|
||||||
|
eval {
|
||||||
|
$plugin->buildStarted($build);
|
||||||
|
};
|
||||||
|
if ($@) {
|
||||||
|
print STDERR "$plugin->buildStarted: $@\n";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
sub notifyBuildFinished {
|
sub notifyBuildFinished {
|
||||||
my ($plugins, $build, $dependents) = @_;
|
my ($plugins, $build, $dependents) = @_;
|
||||||
foreach my $plugin (@{$plugins}) {
|
foreach my $plugin (@{$plugins}) {
|
||||||
|
|
|
@ -20,6 +20,11 @@ sub instantiate {
|
||||||
return @$plugins;
|
return @$plugins;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# Called when build $build has started.
|
||||||
|
sub buildStarted {
|
||||||
|
my ($self, $build) = @_;
|
||||||
|
}
|
||||||
|
|
||||||
# Called when build $build has finished. If the build failed, then
|
# Called when build $build has finished. If the build failed, then
|
||||||
# $dependents is an array ref to a list of builds that have also
|
# $dependents is an array ref to a list of builds that have also
|
||||||
# failed as a result (i.e. because they depend on $build or a failed
|
# failed as a result (i.e. because they depend on $build or a failed
|
||||||
|
|
78
src/lib/Hydra/Plugin/GithubStatus.pm
Normal file
78
src/lib/Hydra/Plugin/GithubStatus.pm
Normal file
|
@ -0,0 +1,78 @@
|
||||||
|
package Hydra::Plugin::GithubStatus;
|
||||||
|
|
||||||
|
use strict;
|
||||||
|
use parent 'Hydra::Plugin';
|
||||||
|
use HTTP::Request;
|
||||||
|
use JSON;
|
||||||
|
use LWP::UserAgent;
|
||||||
|
use Hydra::Helper::CatalystUtils;
|
||||||
|
|
||||||
|
sub toGithubState {
|
||||||
|
my ($buildStatus) = @_;
|
||||||
|
if ($buildStatus == 0) {
|
||||||
|
return "success";
|
||||||
|
} elsif ($buildStatus == 3 || $buildStatus == 4 || $buildStatus == 8 || $buildStatus == 10 || $buildStatus == 11) {
|
||||||
|
return "error";
|
||||||
|
} else {
|
||||||
|
return "failure";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
sub common {
|
||||||
|
my ($self, $build, $dependents, $finished) = @_;
|
||||||
|
my $cfg = $self->{config}->{githubstatus};
|
||||||
|
my @config = defined $cfg ? ref $cfg eq "ARRAY" ? @$cfg : ($cfg) : ();
|
||||||
|
my $baseurl = $self->{config}->{'base_uri'} || "http://localhost:3000";
|
||||||
|
|
||||||
|
# Find matching configs
|
||||||
|
foreach my $b ($build, @{$dependents}) {
|
||||||
|
my $jobName = showJobName $b;
|
||||||
|
my $evals = $build->jobsetevals;
|
||||||
|
my $ua = LWP::UserAgent->new();
|
||||||
|
|
||||||
|
foreach my $conf (@config) {
|
||||||
|
next unless $jobName =~ /^$conf->{jobs}$/;
|
||||||
|
|
||||||
|
my $contextTrailer = $conf->{excludeBuildFromContext} ? "" : (":" . $b->id);
|
||||||
|
my $body = encode_json(
|
||||||
|
{
|
||||||
|
state => $finished ? toGithubState($b->buildstatus) : "pending",
|
||||||
|
target_url => "$baseurl/build/" . $b->id,
|
||||||
|
description => "Hydra build #" . $b->id . " of $jobName",
|
||||||
|
context => "continuous-integration/hydra:" . $jobName . $contextTrailer
|
||||||
|
});
|
||||||
|
my $inputs_cfg = $conf->{inputs};
|
||||||
|
my @inputs = defined $inputs_cfg ? ref $inputs_cfg eq "ARRAY" ? @$inputs_cfg : ($inputs_cfg) : ();
|
||||||
|
my %seen = map { $_ => {} } @inputs;
|
||||||
|
while (my $eval = $evals->next) {
|
||||||
|
foreach my $input (@inputs) {
|
||||||
|
my $i = $eval->jobsetevalinputs->find({ name => $input, altnr => 0 });
|
||||||
|
next unless defined $i;
|
||||||
|
my $uri = $i->uri;
|
||||||
|
my $rev = $i->revision;
|
||||||
|
my $key = $uri . "-" . $rev;
|
||||||
|
next if exists $seen{$input}->{$key};
|
||||||
|
$seen{$input}->{$key} = 1;
|
||||||
|
$uri =~ m![:/]([^/]+)/([^/]+?)(?:.git)?$!;
|
||||||
|
my $req = HTTP::Request->new('POST', "https://api.github.com/repos/$1/$2/statuses/$rev");
|
||||||
|
$req->header('Content-Type' => 'application/json');
|
||||||
|
$req->header('Accept' => 'application/vnd.github.v3+json');
|
||||||
|
$req->header('Authorization' => $conf->{authorization});
|
||||||
|
$req->content($body);
|
||||||
|
my $res = $ua->request($req);
|
||||||
|
print STDERR $res->status_line, ": ", $res->decoded_content, "\n" unless $res->is_success;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
sub buildStarted {
|
||||||
|
common(@_, [], 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
sub buildFinished {
|
||||||
|
common(@_, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
1;
|
|
@ -15,12 +15,12 @@ my $db = Hydra::Model::DB->new();
|
||||||
|
|
||||||
my @plugins = Hydra::Plugin->instantiate(db => $db, config => $config);
|
my @plugins = Hydra::Plugin->instantiate(db => $db, config => $config);
|
||||||
|
|
||||||
my $cmd = shift @ARGV or die "Syntax: hydra-notify build BUILD-ID [BUILD-IDs...]\n";
|
my $cmd = shift @ARGV or die "Syntax: hydra-notify CMD BUILD-ID [BUILD-IDs...]\n";
|
||||||
|
|
||||||
if ($cmd eq "build") {
|
my $buildId = shift @ARGV or die;
|
||||||
my $buildId = shift @ARGV or die;
|
my $build = $db->resultset('Builds')->find($buildId)
|
||||||
my $build = $db->resultset('Builds')->find($buildId)
|
|
||||||
or die "build $buildId does not exist\n";
|
or die "build $buildId does not exist\n";
|
||||||
|
if ($cmd eq "build-finished") {
|
||||||
my @dependents;
|
my @dependents;
|
||||||
foreach my $id (@ARGV) {
|
foreach my $id (@ARGV) {
|
||||||
my $dep = $db->resultset('Builds')->find($id)
|
my $dep = $db->resultset('Builds')->find($id)
|
||||||
|
@ -28,6 +28,8 @@ if ($cmd eq "build") {
|
||||||
push @dependents, $dep;
|
push @dependents, $dep;
|
||||||
}
|
}
|
||||||
notifyBuildFinished(\@plugins, $build, [@dependents]);
|
notifyBuildFinished(\@plugins, $build, [@dependents]);
|
||||||
|
} elsif ($cmd eq "build-started") {
|
||||||
|
notifyBuildStarted(\@plugins, $build);
|
||||||
}
|
}
|
||||||
|
|
||||||
else {
|
else {
|
||||||
|
|
Loading…
Reference in a new issue