1
0
mirror of https://github.com/ohwgiles/laminar.git synced 2024-10-27 20:34:20 +00:00

resolves #61: clickable up/downstream triggers

Recognises triggers in build logs and converts them to
hyperlinks. Also separates upstream job from reason string
and allows both to be provided
This commit is contained in:
Oliver Giles 2018-09-30 09:04:17 +03:00
parent f5e719ac02
commit 63301c73d9
5 changed files with 29 additions and 27 deletions

View File

@ -1,5 +1,5 @@
/// ///
/// Copyright 2015-2017 Oliver Giles /// Copyright 2015-2018 Oliver Giles
/// ///
/// This file is part of Laminar /// This file is part of Laminar
/// ///
@ -36,20 +36,13 @@ static int setParams(int argc, char** argv, T& request) {
n++; n++;
} }
int argsConsumed = n;
char* job = getenv("JOB"); char* job = getenv("JOB");
char* num = getenv("RUN"); char* num = getenv("RUN");
char* reason = getenv("LAMINAR_REASON"); char* reason = getenv("LAMINAR_REASON");
if(job && num) n+=2; auto params = request.initParams(n + (job&&num?2:0) + (reason?1:0));
else if(reason) n++;
if(n == 0) return argsConsumed; for(int i = 0; i < n; ++i) {
auto params = request.initParams(n);
for(int i = 0; i < argsConsumed; ++i) {
char* name = argv[i]; char* name = argv[i];
char* val = strchr(name, '='); char* val = strchr(name, '=');
*val++ = '\0'; *val++ = '\0';
@ -57,19 +50,28 @@ static int setParams(int argc, char** argv, T& request) {
params[i].setValue(val); params[i].setValue(val);
} }
int argsConsumed = n;
if(job && num) { if(job && num) {
params[argsConsumed].setName("=parentJob"); params[n].setName("=parentJob");
params[argsConsumed].setValue(job); params[n++].setValue(job);
params[argsConsumed+1].setName("=parentBuild"); params[n].setName("=parentBuild");
params[argsConsumed+1].setValue(num); params[n++].setValue(num);
} else if(reason) { }
params[argsConsumed].setName("=reason"); if(reason) {
params[argsConsumed].setValue(reason); params[n].setName("=reason");
params[n].setValue(reason);
} }
return argsConsumed; return argsConsumed;
} }
static void printTriggerLink(const char* job, uint run) {
// use a private ANSI CSI sequence to mark the JOB:NUM so the
// frontend can recognise it and generate a hyperlink.
printf("\033[{%s:%d\033\\\n", job, run);
}
int main(int argc, char** argv) { int main(int argc, char** argv) {
if(argc < 2) { if(argc < 2) {
fprintf(stderr, "Usage: %s <command> [parameters...]\n", argv[0]); fprintf(stderr, "Usage: %s <command> [parameters...]\n", argv[0]);
@ -129,7 +131,7 @@ int main(int argc, char** argv) {
fprintf(stderr, "Failed to start job '%s'\n", argv[2]); fprintf(stderr, "Failed to start job '%s'\n", argv[2]);
ret = ENOENT; ret = ENOENT;
} }
printf("%s:%d\n", argv[jobNameIndex], resp.getBuildNum()); printTriggerLink(argv[jobNameIndex], resp.getBuildNum());
})); }));
jobNameIndex += n + 1; jobNameIndex += n + 1;
} while(jobNameIndex < argc); } while(jobNameIndex < argc);
@ -150,7 +152,7 @@ int main(int argc, char** argv) {
req.setJobName(argv[jobNameIndex]); req.setJobName(argv[jobNameIndex]);
int n = setParams(argc - jobNameIndex - 1, &argv[jobNameIndex + 1], req); int n = setParams(argc - jobNameIndex - 1, &argv[jobNameIndex + 1], req);
ts.add(req.send().then([&ret,argv,jobNameIndex](capnp::Response<LaminarCi::RunResults> resp){ ts.add(req.send().then([&ret,argv,jobNameIndex](capnp::Response<LaminarCi::RunResults> resp){
printf("%s:%d\n", argv[jobNameIndex], resp.getBuildNum()); printTriggerLink(argv[jobNameIndex], resp.getBuildNum());
if(resp.getResult() != LaminarCi::JobResult::SUCCESS) { if(resp.getResult() != LaminarCi::JobResult::SUCCESS) {
ret = EFAILED; ret = EFAILED;
} }

View File

@ -178,20 +178,22 @@ void Laminar::sendStatus(LaminarClient* client) {
j.set("time", time(nullptr)); j.set("time", time(nullptr));
j.startObject("data"); j.startObject("data");
if(client->scope.type == MonitorScope::RUN) { if(client->scope.type == MonitorScope::RUN) {
db->stmt("SELECT queuedAt,startedAt,completedAt, result, reason FROM builds WHERE name = ? AND number = ?") db->stmt("SELECT queuedAt,startedAt,completedAt,result,reason,parentJob,parentBuild FROM builds WHERE name = ? AND number = ?")
.bind(client->scope.job, client->scope.num) .bind(client->scope.job, client->scope.num)
.fetch<time_t, time_t, time_t, int, std::string>([&](time_t queued, time_t started, time_t completed, int result, std::string reason) { .fetch<time_t, time_t, time_t, int, std::string, std::string, uint>([&](time_t queued, time_t started, time_t completed, int result, std::string reason, std::string parentJob, uint parentBuild) {
j.set("queued", started-queued); j.set("queued", started-queued);
j.set("started", started); j.set("started", started);
j.set("completed", completed); j.set("completed", completed);
j.set("result", to_string(RunState(result))); j.set("result", to_string(RunState(result)));
j.set("reason", reason); j.set("reason", reason);
j.startObject("upstream").set("name", parentJob).set("num", parentBuild).EndObject(2);
}); });
if(const Run* run = activeRun(client->scope.job, client->scope.num)) { if(const Run* run = activeRun(client->scope.job, client->scope.num)) {
j.set("queued", run->startedAt - run->queuedAt); j.set("queued", run->startedAt - run->queuedAt);
j.set("started", run->startedAt); j.set("started", run->startedAt);
j.set("reason", run->reason());
j.set("result", to_string(RunState::RUNNING)); j.set("result", to_string(RunState::RUNNING));
j.set("reason", run->reason());
j.startObject("upstream").set("name", run->parentName).set("num", run->parentBuild).EndObject(2);
db->stmt("SELECT completedAt - startedAt FROM builds WHERE name = ? ORDER BY completedAt DESC LIMIT 1") db->stmt("SELECT completedAt - startedAt FROM builds WHERE name = ? ORDER BY completedAt DESC LIMIT 1")
.bind(run->name) .bind(run->name)
.fetch<uint>([&](uint lastRuntime){ .fetch<uint>([&](uint lastRuntime){

View File

@ -294,6 +294,7 @@
<div style="clear:both;"></div> <div style="clear:both;"></div>
<dl class="dl-horizontal"> <dl class="dl-horizontal">
<dt>Reason</dt><dd>{{job.reason}}</dd> <dt>Reason</dt><dd>{{job.reason}}</dd>
<dt v-show="job.upstream.num > 0">Upstream</dt><dd v-show="job.upstream.num > 0"><router-link :to="'/jobs/'+job.upstream.name">{{job.upstream.name}}</router-link> <router-link :to="'/jobs/'+job.upstream.name+'/'+job.upstream.num">#{{job.upstream.num}}</router-link></li></dd>
<dt>Queued for</dt><dd>{{job.queued}}s</dd> <dt>Queued for</dt><dd>{{job.queued}}s</dd>
<dt>Started</dt><dd>{{formatDate(job.started)}}</dd> <dt>Started</dt><dd>{{formatDate(job.started)}}</dd>
<dt v-show="runComplete(job)">Completed</dt><dd v-show="job.completed">{{formatDate(job.completed)}}</dd> <dt v-show="runComplete(job)">Completed</dt><dd v-show="job.completed">{{formatDate(job.completed)}}</dd>

View File

@ -655,14 +655,14 @@ var Job = function() {
const Run = function() { const Run = function() {
var state = { var state = {
job: { artifacts: [] }, job: { artifacts: [], upstream: {} },
latestNum: null, latestNum: null,
log: '', log: '',
autoscroll: false autoscroll: false
}; };
var firstLog = false; var firstLog = false;
var logHandler = function(vm, d) { var logHandler = function(vm, d) {
state.log += ansi_up.ansi_to_html(d.replace(/</g,'&lt;').replace(/>/g,'&gt;')); state.log += ansi_up.ansi_to_html(d.replace(/</g,'&lt;').replace(/>/g,'&gt;').replace(/\033\[\{([^:]+):(\d+)\033\\/g, (m,$1,$2)=>{return '<a href="/jobs/'+$1+'" onclick="return vroute(this);">'+$1+'</a>:<a href="/jobs/'+$1+'/'+$2+'" onclick="return vroute(this);">#'+$2+'</a>';}));
vm.$forceUpdate(); vm.$forceUpdate();
if (!firstLog) { if (!firstLog) {
firstLog = true; firstLog = true;

View File

@ -159,9 +159,6 @@ bool Run::configure(uint buildNum, std::shared_ptr<Node> nd, const kj::Directory
} }
std::string Run::reason() const { std::string Run::reason() const {
if(!parentName.empty()) {
return std::string("Triggered by upstream ") + parentName + " #" + std::to_string(parentBuild);
}
return reasonMsg; return reasonMsg;
} }