mirror of
https://github.com/ohwgiles/laminar.git
synced 2024-10-27 20:34:20 +00:00
assign run numbers at queue time
This allows build chains to be traced in the common case where an upstream job calls `laminarc queue' instead of `laminarc start'. Incomplete job runs now have database entries, which requires some adjustments in queries. Queued jobs can now be viewed in the frontend and there is a corresponding status icon.
This commit is contained in:
parent
6d2c0b208b
commit
06a5f3d8ef
@ -127,30 +127,30 @@ chmod +x /var/lib/laminar/cfg/jobs/hello.run
|
|||||||
|
|
||||||
# Triggering a run
|
# Triggering a run
|
||||||
|
|
||||||
When triggering a run, the job is first added to a queue of upcoming tasks. If the server is busy, the job may wait in this queue for a while. It will only be assigned a job number when it leaves this queue and starts executing. The job number may be useful to the client that triggers the run, so there are a few ways to trigger a run.
|
To queue execution of the `hello` job, run
|
||||||
|
|
||||||
To add the `hello` job to the queue ("fire-and-forget"), execute
|
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
laminarc queue hello
|
laminarc queue hello
|
||||||
```
|
```
|
||||||
|
|
||||||
In this case, laminarc returns immediately, with its error code indicating whether adding the job to the queue was sucessful.
|
In this case, `laminarc` returns immediately, with its error code indicating whether adding the job to the queue was sucessful. The run number will be printed to standard output.
|
||||||
|
|
||||||
To queue the job and wait until it leaves the queue and starts executing, use
|
If the server is busy, a run may wait in the queue for some time. To have `laminarc` instead block until the run leaves the queue and starts executing, use
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
laminarc start hello
|
laminarc start hello
|
||||||
```
|
```
|
||||||
|
|
||||||
In this case, laminarc blocks until the job starts executing, or returns immediately if queueing failed. The run number will be printed to standard output.
|
In this case, `laminarc` blocks until the job starts executing, or returns immediately if queueing failed. The run number will be printed to standard output.
|
||||||
|
|
||||||
To launch and run the `hello` job to completion, execute
|
Finally, to launch and run the `hello` job to completion, execute
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
laminarc run hello
|
laminarc run hello
|
||||||
```
|
```
|
||||||
|
|
||||||
|
In this case, laminarc's return value indicates whether the run completed successfully.
|
||||||
|
|
||||||
In all cases, a started run means the `/var/lib/laminar/cfg/jobs/hello.run` script will be executed, with a working directory of `/var/lib/laminar/run/hello/1` (or current run number)
|
In all cases, a started run means the `/var/lib/laminar/cfg/jobs/hello.run` script will be executed, with a working directory of `/var/lib/laminar/run/hello/1` (or current run number)
|
||||||
|
|
||||||
The result and log output should be visible in the Web UI at http://localhost:8080/jobs/hello/1
|
The result and log output should be visible in the Web UI at http://localhost:8080/jobs/hello/1
|
||||||
|
@ -147,7 +147,8 @@ int main(int argc, char** argv) {
|
|||||||
if(resp.getResult() != LaminarCi::MethodResult::SUCCESS) {
|
if(resp.getResult() != LaminarCi::MethodResult::SUCCESS) {
|
||||||
fprintf(stderr, "Failed to queue job '%s'\n", argv[jobNameIndex]);
|
fprintf(stderr, "Failed to queue job '%s'\n", argv[jobNameIndex]);
|
||||||
ret = EXIT_OPERATION_FAILED;
|
ret = EXIT_OPERATION_FAILED;
|
||||||
}
|
} else
|
||||||
|
printTriggerLink(argv[jobNameIndex], resp.getBuildNum());
|
||||||
}));
|
}));
|
||||||
jobNameIndex += n + 1;
|
jobNameIndex += n + 1;
|
||||||
} while(jobNameIndex < argc);
|
} while(jobNameIndex < argc);
|
||||||
|
@ -2,7 +2,7 @@
|
|||||||
|
|
||||||
interface LaminarCi {
|
interface LaminarCi {
|
||||||
|
|
||||||
queue @0 (jobName :Text, params :List(JobParam)) -> (result :MethodResult);
|
queue @0 (jobName :Text, params :List(JobParam)) -> (result :MethodResult, buildNum :UInt32);
|
||||||
start @1 (jobName :Text, params :List(JobParam)) -> (result :MethodResult, buildNum :UInt32);
|
start @1 (jobName :Text, params :List(JobParam)) -> (result :MethodResult, buildNum :UInt32);
|
||||||
run @2 (jobName :Text, params :List(JobParam)) -> (result :JobResult, buildNum :UInt32);
|
run @2 (jobName :Text, params :List(JobParam)) -> (result :JobResult, buildNum :UInt32);
|
||||||
listQueued @3 () -> (result :List(Text));
|
listQueued @3 () -> (result :List(Text));
|
||||||
|
@ -136,18 +136,9 @@ void Laminar::loadCustomizations() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
uint Laminar::latestRun(std::string job) {
|
uint Laminar::latestRun(std::string job) {
|
||||||
auto it = activeJobs.byJobName().equal_range(job);
|
if(auto it = buildNums.find(job); it != buildNums.end())
|
||||||
if(it.first == it.second) {
|
return it->second;
|
||||||
uint result = 0;
|
return 0;
|
||||||
db->stmt("SELECT MAX(number) FROM builds WHERE name = ?")
|
|
||||||
.bind(job)
|
|
||||||
.fetch<uint>([&](uint x){
|
|
||||||
result = x;
|
|
||||||
});
|
|
||||||
return result;
|
|
||||||
} else {
|
|
||||||
return (*--it.second)->build;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Laminar::handleLogRequest(std::string name, uint num, std::string& output, bool& complete) {
|
bool Laminar::handleLogRequest(std::string name, uint num, std::string& output, bool& complete) {
|
||||||
@ -231,30 +222,24 @@ std::string Laminar::getStatus(MonitorScope scope) {
|
|||||||
j.set("time", time(nullptr));
|
j.set("time", time(nullptr));
|
||||||
j.startObject("data");
|
j.startObject("data");
|
||||||
if(scope.type == MonitorScope::RUN) {
|
if(scope.type == MonitorScope::RUN) {
|
||||||
db->stmt("SELECT queuedAt,startedAt,completedAt,result,reason,parentJob,parentBuild FROM builds WHERE name = ? AND number = ?")
|
db->stmt("SELECT queuedAt,startedAt,completedAt,result,reason,parentJob,parentBuild,q.lr IS NOT NULL,q.lr FROM builds "
|
||||||
|
"LEFT JOIN (SELECT name n, MAX(number), completedAt-startedAt lr FROM builds WHERE result IS NOT NULL GROUP BY n) q ON q.n = name "
|
||||||
|
"WHERE name = ? AND number = ?")
|
||||||
.bind(scope.job, scope.num)
|
.bind(scope.job, scope.num)
|
||||||
.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) {
|
.fetch<time_t, time_t, time_t, int, std::string, std::string, uint, uint, uint>([&](time_t queued, time_t started, time_t completed, int result, std::string reason, std::string parentJob, uint parentBuild, uint lastRuntimeKnown, uint lastRuntime) {
|
||||||
j.set("queued", started-queued);
|
j.set("queued", queued);
|
||||||
j.set("started", started);
|
j.set("started", started);
|
||||||
j.set("completed", completed);
|
if(completed)
|
||||||
j.set("result", to_string(RunState(result)));
|
j.set("completed", completed);
|
||||||
|
j.set("result", to_string(completed ? RunState(result) : started ? RunState::RUNNING : RunState::QUEUED));
|
||||||
j.set("reason", reason);
|
j.set("reason", reason);
|
||||||
j.startObject("upstream").set("name", parentJob).set("num", parentBuild).EndObject(2);
|
j.startObject("upstream").set("name", parentJob).set("num", parentBuild).EndObject(2);
|
||||||
|
if(lastRuntimeKnown)
|
||||||
|
j.set("etc", started + lastRuntime);
|
||||||
});
|
});
|
||||||
if(const Run* run = activeRun(scope.job, scope.num)) {
|
|
||||||
j.set("queued", run->startedAt - run->queuedAt);
|
|
||||||
j.set("started", run->startedAt);
|
|
||||||
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")
|
|
||||||
.bind(run->name)
|
|
||||||
.fetch<uint>([&](uint lastRuntime){
|
|
||||||
j.set("etc", run->startedAt + lastRuntime);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
if(auto it = buildNums.find(scope.job); it != buildNums.end())
|
if(auto it = buildNums.find(scope.job); it != buildNums.end())
|
||||||
j.set("latestNum", int(it->second));
|
j.set("latestNum", int(it->second));
|
||||||
|
|
||||||
j.startArray("artifacts");
|
j.startArray("artifacts");
|
||||||
populateArtifacts(j, scope.job, scope.num);
|
populateArtifacts(j, scope.job, scope.num);
|
||||||
j.EndArray();
|
j.EndArray();
|
||||||
@ -274,7 +259,8 @@ std::string Laminar::getStatus(MonitorScope scope) {
|
|||||||
order_by = "(completedAt-startedAt) " + direction + ", number DESC";
|
order_by = "(completedAt-startedAt) " + direction + ", number DESC";
|
||||||
else
|
else
|
||||||
order_by = "number DESC";
|
order_by = "number DESC";
|
||||||
std::string stmt = "SELECT number,startedAt,completedAt,result,reason FROM builds WHERE name = ? ORDER BY "
|
std::string stmt = "SELECT number,startedAt,completedAt,result,reason FROM builds "
|
||||||
|
"WHERE name = ? AND result IS NOT NULL ORDER BY "
|
||||||
+ order_by + " LIMIT ?,?";
|
+ order_by + " LIMIT ?,?";
|
||||||
db->stmt(stmt.c_str())
|
db->stmt(stmt.c_str())
|
||||||
.bind(scope.job, scope.page * runsPerPage, runsPerPage)
|
.bind(scope.job, scope.page * runsPerPage, runsPerPage)
|
||||||
@ -288,7 +274,7 @@ std::string Laminar::getStatus(MonitorScope scope) {
|
|||||||
.EndObject();
|
.EndObject();
|
||||||
});
|
});
|
||||||
j.EndArray();
|
j.EndArray();
|
||||||
db->stmt("SELECT COUNT(*),AVG(completedAt-startedAt) FROM builds WHERE name = ?")
|
db->stmt("SELECT COUNT(*),AVG(completedAt-startedAt) FROM builds WHERE name = ? AND result IS NOT NULL")
|
||||||
.bind(scope.job)
|
.bind(scope.job)
|
||||||
.fetch<uint,uint>([&](uint nRuns, uint averageRuntime){
|
.fetch<uint,uint>([&](uint nRuns, uint averageRuntime){
|
||||||
j.set("averageRuntime", averageRuntime);
|
j.set("averageRuntime", averageRuntime);
|
||||||
@ -319,14 +305,17 @@ std::string Laminar::getStatus(MonitorScope scope) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
j.set("nQueued", nQueued);
|
j.set("nQueued", nQueued);
|
||||||
db->stmt("SELECT number,startedAt FROM builds WHERE name = ? AND result = ? ORDER BY completedAt DESC LIMIT 1")
|
db->stmt("SELECT number,startedAt FROM builds WHERE name = ? AND result = ? "
|
||||||
|
"ORDER BY completedAt DESC LIMIT 1")
|
||||||
.bind(scope.job, int(RunState::SUCCESS))
|
.bind(scope.job, int(RunState::SUCCESS))
|
||||||
.fetch<int,time_t>([&](int build, time_t started){
|
.fetch<int,time_t>([&](int build, time_t started){
|
||||||
j.startObject("lastSuccess");
|
j.startObject("lastSuccess");
|
||||||
j.set("number", build).set("started", started);
|
j.set("number", build).set("started", started);
|
||||||
j.EndObject();
|
j.EndObject();
|
||||||
});
|
});
|
||||||
db->stmt("SELECT number,startedAt FROM builds WHERE name = ? AND result <> ? ORDER BY completedAt DESC LIMIT 1")
|
db->stmt("SELECT number,startedAt FROM builds "
|
||||||
|
"WHERE name = ? AND result <> ? "
|
||||||
|
"ORDER BY completedAt DESC LIMIT 1")
|
||||||
.bind(scope.job, int(RunState::SUCCESS))
|
.bind(scope.job, int(RunState::SUCCESS))
|
||||||
.fetch<int,time_t>([&](int build, time_t started){
|
.fetch<int,time_t>([&](int build, time_t started){
|
||||||
j.startObject("lastFailed");
|
j.startObject("lastFailed");
|
||||||
@ -337,7 +326,9 @@ std::string Laminar::getStatus(MonitorScope scope) {
|
|||||||
j.set("description", desc == jobDescriptions.end() ? "" : desc->second);
|
j.set("description", desc == jobDescriptions.end() ? "" : desc->second);
|
||||||
} else if(scope.type == MonitorScope::ALL) {
|
} else if(scope.type == MonitorScope::ALL) {
|
||||||
j.startArray("jobs");
|
j.startArray("jobs");
|
||||||
db->stmt("SELECT name,number,startedAt,completedAt,result FROM builds b JOIN (SELECT name n,MAX(number) l FROM builds GROUP BY n) q ON b.name = q.n AND b.number = q.l")
|
db->stmt("SELECT name,number,startedAt,completedAt,result FROM builds b "
|
||||||
|
"JOIN (SELECT name n,MAX(number) latest FROM builds WHERE result IS NOT NULL GROUP BY n) q "
|
||||||
|
"ON b.name = q.n AND b.number = latest")
|
||||||
.fetch<str,uint,time_t,time_t,int>([&](str name,uint number, time_t started, time_t completed, int result){
|
.fetch<str,uint,time_t,time_t,int>([&](str name,uint number, time_t started, time_t completed, int result){
|
||||||
j.StartObject();
|
j.StartObject();
|
||||||
j.set("name", name);
|
j.set("name", name);
|
||||||
@ -364,7 +355,7 @@ std::string Laminar::getStatus(MonitorScope scope) {
|
|||||||
j.EndObject();
|
j.EndObject();
|
||||||
} else { // Home page
|
} else { // Home page
|
||||||
j.startArray("recent");
|
j.startArray("recent");
|
||||||
db->stmt("SELECT * FROM builds ORDER BY completedAt DESC LIMIT 20")
|
db->stmt("SELECT * FROM builds WHERE completedAt IS NOT NULL ORDER BY completedAt DESC LIMIT 20")
|
||||||
.fetch<str,uint,str,time_t,time_t,time_t,int>([&](str name,uint build,str context,time_t,time_t started,time_t completed,int result){
|
.fetch<str,uint,str,time_t,time_t,time_t,int>([&](str name,uint build,str context,time_t,time_t started,time_t completed,int result){
|
||||||
j.StartObject();
|
j.StartObject();
|
||||||
j.set("name", name)
|
j.set("name", name)
|
||||||
@ -383,7 +374,9 @@ std::string Laminar::getStatus(MonitorScope scope) {
|
|||||||
j.set("number", run->build);
|
j.set("number", run->build);
|
||||||
j.set("context", run->context->name);
|
j.set("context", run->context->name);
|
||||||
j.set("started", run->startedAt);
|
j.set("started", run->startedAt);
|
||||||
db->stmt("SELECT completedAt - startedAt FROM builds WHERE name = ? ORDER BY completedAt DESC LIMIT 1")
|
db->stmt("SELECT completedAt - startedAt FROM builds "
|
||||||
|
"WHERE completedAt IS NOT NULL AND name = ? "
|
||||||
|
"ORDER BY completedAt DESC LIMIT 1")
|
||||||
.bind(run->name)
|
.bind(run->name)
|
||||||
.fetch<uint>([&](uint lastRuntime){
|
.fetch<uint>([&](uint lastRuntime){
|
||||||
j.set("etc", run->startedAt + lastRuntime);
|
j.set("etc", run->startedAt + lastRuntime);
|
||||||
@ -586,14 +579,19 @@ std::shared_ptr<Run> Laminar::queueJob(std::string name, ParamMap params) {
|
|||||||
if(jobContexts[name].empty())
|
if(jobContexts[name].empty())
|
||||||
jobContexts.at(name).insert("default");
|
jobContexts.at(name).insert("default");
|
||||||
|
|
||||||
std::shared_ptr<Run> run = std::make_shared<Run>(name, kj::mv(params), homePath.clone());
|
std::shared_ptr<Run> run = std::make_shared<Run>(name, ++buildNums[name], kj::mv(params), homePath.clone());
|
||||||
queuedJobs.push_back(run);
|
queuedJobs.push_back(run);
|
||||||
|
|
||||||
|
db->stmt("INSERT INTO builds(name,number,queuedAt,parentJob,parentBuild,reason) VALUES(?,?,?,?,?,?)")
|
||||||
|
.bind(run->name, run->build, run->queuedAt, run->parentName, run->parentBuild, run->reason())
|
||||||
|
.exec();
|
||||||
|
|
||||||
// notify clients
|
// notify clients
|
||||||
Json j;
|
Json j;
|
||||||
j.set("type", "job_queued")
|
j.set("type", "job_queued")
|
||||||
.startObject("data")
|
.startObject("data")
|
||||||
.set("name", name)
|
.set("name", name)
|
||||||
|
.set("number", run->build)
|
||||||
.EndObject();
|
.EndObject();
|
||||||
http->notifyEvent(j.str(), name.c_str());
|
http->notifyEvent(j.str(), name.c_str());
|
||||||
|
|
||||||
@ -620,14 +618,19 @@ bool Laminar::tryStartRun(std::shared_ptr<Run> run, int queueIndex) {
|
|||||||
if(ctx->canQueue(jobContexts.at(run->name))) {
|
if(ctx->canQueue(jobContexts.at(run->name))) {
|
||||||
RunState lastResult = RunState::UNKNOWN;
|
RunState lastResult = RunState::UNKNOWN;
|
||||||
|
|
||||||
// set the last known result if exists
|
// set the last known result if exists. Runs which haven't started yet should
|
||||||
|
// have completedAt == NULL and thus be at the end of a DESC ordered query
|
||||||
db->stmt("SELECT result FROM builds WHERE name = ? ORDER BY completedAt DESC LIMIT 1")
|
db->stmt("SELECT result FROM builds WHERE name = ? ORDER BY completedAt DESC LIMIT 1")
|
||||||
.bind(run->name)
|
.bind(run->name)
|
||||||
.fetch<int>([&](int result){
|
.fetch<int>([&](int result){
|
||||||
lastResult = RunState(result);
|
lastResult = RunState(result);
|
||||||
});
|
});
|
||||||
|
|
||||||
kj::Promise<RunState> onRunFinished = run->start(buildNums[run->name] + 1, lastResult, ctx, *fsHome,[this](kj::Maybe<pid_t>& pid){return srv.onChildExit(pid);});
|
kj::Promise<RunState> onRunFinished = run->start(lastResult, ctx, *fsHome,[this](kj::Maybe<pid_t>& pid){return srv.onChildExit(pid);});
|
||||||
|
|
||||||
|
db->stmt("UPDATE builds SET node = ?, startedAt = ? WHERE name = ? AND number = ?")
|
||||||
|
.bind(ctx->name, run->startedAt, run->name, run->build)
|
||||||
|
.exec();
|
||||||
|
|
||||||
ctx->busyExecutors++;
|
ctx->busyExecutors++;
|
||||||
|
|
||||||
@ -650,16 +653,13 @@ bool Laminar::tryStartRun(std::shared_ptr<Run> run, int queueIndex) {
|
|||||||
srv.addTask(kj::mv(exec));
|
srv.addTask(kj::mv(exec));
|
||||||
LLOG(INFO, "Started job", run->name, run->build, ctx->name);
|
LLOG(INFO, "Started job", run->name, run->build, ctx->name);
|
||||||
|
|
||||||
// update next build number
|
|
||||||
buildNums[run->name]++;
|
|
||||||
|
|
||||||
// notify clients
|
// notify clients
|
||||||
Json j;
|
Json j;
|
||||||
j.set("type", "job_started")
|
j.set("type", "job_started")
|
||||||
.startObject("data")
|
.startObject("data")
|
||||||
.set("queueIndex", queueIndex)
|
.set("queueIndex", queueIndex)
|
||||||
.set("name", run->name)
|
.set("name", run->name)
|
||||||
.set("queued", run->startedAt - run->queuedAt)
|
.set("queued", run->queuedAt)
|
||||||
.set("started", run->startedAt)
|
.set("started", run->startedAt)
|
||||||
.set("number", run->build)
|
.set("number", run->build)
|
||||||
.set("reason", run->reason());
|
.set("reason", run->reason());
|
||||||
@ -708,10 +708,8 @@ void Laminar::handleRunFinished(Run * r) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
std::string reason = r->reason();
|
db->stmt("UPDATE builds SET completedAt = ?, result = ?, output = ?, outputLen = ? WHERE name = ? AND number = ?")
|
||||||
db->stmt("INSERT INTO builds VALUES(?,?,?,?,?,?,?,?,?,?,?,?)")
|
.bind(completedAt, int(r->result), maybeZipped, logsize, r->name, r->build)
|
||||||
.bind(r->name, r->build, ctx->name, r->queuedAt, r->startedAt, completedAt, int(r->result),
|
|
||||||
maybeZipped, logsize, r->parentName, r->parentBuild, reason)
|
|
||||||
.exec();
|
.exec();
|
||||||
|
|
||||||
// notify clients
|
// notify clients
|
||||||
@ -720,7 +718,7 @@ void Laminar::handleRunFinished(Run * r) {
|
|||||||
.startObject("data")
|
.startObject("data")
|
||||||
.set("name", r->name)
|
.set("name", r->name)
|
||||||
.set("number", r->build)
|
.set("number", r->build)
|
||||||
.set("queued", r->startedAt - r->queuedAt)
|
.set("queued", r->queuedAt)
|
||||||
.set("completed", completedAt)
|
.set("completed", completedAt)
|
||||||
.set("started", r->startedAt)
|
.set("started", r->startedAt)
|
||||||
.set("result", to_string(r->result))
|
.set("result", to_string(r->result))
|
||||||
|
@ -118,7 +118,8 @@
|
|||||||
<th class="text-center vp-sm-hide">Reason <a class="sort" :class="(sort.field=='reason'?sort.order:'')" v-on:click="do_sort('reason')"> </a></th>
|
<th class="text-center vp-sm-hide">Reason <a class="sort" :class="(sort.field=='reason'?sort.order:'')" v-on:click="do_sort('reason')"> </a></th>
|
||||||
</tr></thead>
|
</tr></thead>
|
||||||
<tr v-show="nQueued">
|
<tr v-show="nQueued">
|
||||||
<td colspan="5"><i>{{nQueued}} run(s) queued</i></td>
|
<td style="width:1px"><span v-html="runIcon('queued')"></span></td>
|
||||||
|
<td colspan="4"><i>{{nQueued}} run(s) queued</i></td>
|
||||||
</tr>
|
</tr>
|
||||||
<tr v-for="job in jobsRunning.concat(jobsRecent)" track-by="$index">
|
<tr v-for="job in jobsRunning.concat(jobsRecent)" track-by="$index">
|
||||||
<td style="width:1px"><span v-html="runIcon(job.result)"></span></td>
|
<td style="width:1px"><span v-html="runIcon(job.result)"></span></td>
|
||||||
@ -152,10 +153,10 @@
|
|||||||
<dl>
|
<dl>
|
||||||
<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 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>{{formatDuration(job.queued, job.started ? job.started : Math.floor(Date.now()/1000))}}</dd>
|
||||||
<dt>Started</dt><dd>{{formatDate(job.started)}}</dd>
|
<dt v-show="job.started">Started</dt><dd v-show="job.started">{{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>
|
||||||
<dt>Duration</dt><dd>{{formatDuration(job.started, job.completed)}}</dd>
|
<dt v-show="job.started">Duration</dt><dd v-show="job.started">{{formatDuration(job.started, job.completed)}}</dd>
|
||||||
</dl>
|
</dl>
|
||||||
<dl v-show="job.artifacts.length">
|
<dl v-show="job.artifacts.length">
|
||||||
<dt>Artifacts</dt>
|
<dt>Artifacts</dt>
|
||||||
|
@ -23,10 +23,10 @@ const timeScale = function(max){
|
|||||||
: { scale:function(v){return v;}, label:'Seconds' };
|
: { scale:function(v){return v;}, label:'Seconds' };
|
||||||
}
|
}
|
||||||
const ServerEventHandler = function() {
|
const ServerEventHandler = function() {
|
||||||
function setupEventSource(path, query, next, comp) {
|
function setupEventSource(to, query, next, comp) {
|
||||||
const es = new EventSource(document.head.baseURI + path.substr(1) + query);
|
const es = new EventSource(document.head.baseURI + to.path.substr(1) + query);
|
||||||
es.comp = comp;
|
es.comp = comp;
|
||||||
es.path = path; // save for later in case we need to add query params
|
es.path = to.path; // save for later in case we need to add query params
|
||||||
es.onmessage = function(msg) {
|
es.onmessage = function(msg) {
|
||||||
msg = JSON.parse(msg.data);
|
msg = JSON.parse(msg.data);
|
||||||
// "status" is the first message the server always delivers.
|
// "status" is the first message the server always delivers.
|
||||||
@ -55,7 +55,7 @@ const ServerEventHandler = function() {
|
|||||||
comp.$root.clockSkew = msg.time - Math.floor((new Date()).getTime()/1000);
|
comp.$root.clockSkew = msg.time - Math.floor((new Date()).getTime()/1000);
|
||||||
comp.$root.connected = true;
|
comp.$root.connected = true;
|
||||||
// Component-specific callback handler
|
// Component-specific callback handler
|
||||||
comp[msg.type](msg.data);
|
comp[msg.type](msg.data, to.params);
|
||||||
});
|
});
|
||||||
} else {
|
} else {
|
||||||
// at this point, the component must be defined
|
// at this point, the component must be defined
|
||||||
@ -72,7 +72,7 @@ const ServerEventHandler = function() {
|
|||||||
es.onerror = function(e) {
|
es.onerror = function(e) {
|
||||||
this.comp.$root.connected = false;
|
this.comp.$root.connected = false;
|
||||||
setTimeout(() => {
|
setTimeout(() => {
|
||||||
this.comp.es = setupEventSource(path, query, null, this.comp);
|
this.comp.es = setupEventSource(to, query, null, this.comp);
|
||||||
}, this.comp.esReconnectInterval);
|
}, this.comp.esReconnectInterval);
|
||||||
if(this.comp.esReconnectInterval < 7500)
|
if(this.comp.esReconnectInterval < 7500)
|
||||||
this.comp.esReconnectInterval *= 1.5;
|
this.comp.esReconnectInterval *= 1.5;
|
||||||
@ -82,11 +82,11 @@ const ServerEventHandler = function() {
|
|||||||
}
|
}
|
||||||
return {
|
return {
|
||||||
beforeRouteEnter(to, from, next) {
|
beforeRouteEnter(to, from, next) {
|
||||||
setupEventSource(to.path, '', (fn) => { next(fn); });
|
setupEventSource(to, '', (fn) => { next(fn); });
|
||||||
},
|
},
|
||||||
beforeRouteUpdate(to, from, next) {
|
beforeRouteUpdate(to, from, next) {
|
||||||
this.es.close();
|
this.es.close();
|
||||||
setupEventSource(to.path, '', (fn) => { fn(this); next(); });
|
setupEventSource(to, '', (fn) => { fn(this); next(); });
|
||||||
},
|
},
|
||||||
beforeRouteLeave(to, from, next) {
|
beforeRouteLeave(to, from, next) {
|
||||||
this.es.close();
|
this.es.close();
|
||||||
@ -115,6 +115,11 @@ const Utils = {
|
|||||||
21,-26 5,5 11,15 15,20 8,-2 15,-9 20,-15 -3,-3 -17,-18 -20,-24 3,-5 23,-26 30,-33 -3,-5 -8,-9
|
21,-26 5,5 11,15 15,20 8,-2 15,-9 20,-15 -3,-3 -17,-18 -20,-24 3,-5 23,-26 30,-33 -3,-5 -8,-9
|
||||||
-12,-12 -6,5 -26,26 -29,30 -6,-8 -11,-15 -15,-23 -3,0 -12,5 -15,7 z" />
|
-12,-12 -6,5 -26,26 -29,30 -6,-8 -11,-15 -15,-23 -3,0 -12,5 -15,7 z" />
|
||||||
</svg>`
|
</svg>`
|
||||||
|
: (result == 'queued') ? /* clock */
|
||||||
|
`<svg class="status queued" viewBox="0 0 100 100">
|
||||||
|
<circle r="50" cy="50" cx="50" />
|
||||||
|
<path d="m 50,15 0,35 17,17" stroke-width="10" fill="none" />
|
||||||
|
</svg>`
|
||||||
: /* spinner */
|
: /* spinner */
|
||||||
`<svg class="status running" viewBox="0 0 100 100">
|
`<svg class="status running" viewBox="0 0 100 100">
|
||||||
<circle cx="50" cy="50" r="40" stroke-width="15" fill="none" stroke-dasharray="175">
|
<circle cx="50" cy="50" r="40" stroke-width="15" fill="none" stroke-dasharray="175">
|
||||||
@ -656,49 +661,46 @@ const Run = function() {
|
|||||||
return state;
|
return state;
|
||||||
},
|
},
|
||||||
methods: {
|
methods: {
|
||||||
status: function(data) {
|
status: function(data, params) {
|
||||||
// Check for the /latest endpoint. An intuitive check might be
|
// Check for the /latest endpoint
|
||||||
// if(this.$route.params.number == 'latest'), but unfortunately
|
if(params.number === 'latest')
|
||||||
// after calling $router.replace, we re-enter status() before
|
return this.$router.replace('/jobs/' + params.name + '/' + data.latestNum);
|
||||||
// $route.params is updated. Instead, assume that if there is
|
|
||||||
// no 'started' field, we should redirect to the latest number
|
|
||||||
if(!('started' in data) && 'latestNum' in data)
|
|
||||||
return this.$router.replace('/jobs/' + this.$route.params.name + '/' + data.latestNum);
|
|
||||||
|
|
||||||
|
state.number = parseInt(params.number);
|
||||||
state.jobsRunning = [];
|
state.jobsRunning = [];
|
||||||
state.job = data;
|
state.job = data;
|
||||||
state.latestNum = data.latestNum;
|
state.latestNum = data.latestNum;
|
||||||
state.jobsRunning = [data];
|
state.jobsRunning = [data];
|
||||||
|
state.log = '';
|
||||||
|
if(this.logstream)
|
||||||
|
this.logstream.abort();
|
||||||
|
if(data.started)
|
||||||
|
this.logstream = logFetcher(this, params.name, params.number);
|
||||||
|
},
|
||||||
|
job_queued: function(data) {
|
||||||
|
state.latestNum = data.number;
|
||||||
|
this.$forceUpdate();
|
||||||
},
|
},
|
||||||
job_started: function(data) {
|
job_started: function(data) {
|
||||||
state.latestNum++;
|
if(data.number === state.number) {
|
||||||
this.$forceUpdate();
|
state.job = Object.assign(state.job, data);
|
||||||
|
state.job.result = 'running';
|
||||||
|
if(this.logstream)
|
||||||
|
this.logstream.abort();
|
||||||
|
this.logstream = logFetcher(this, data.name, data.number);
|
||||||
|
this.$forceUpdate();
|
||||||
|
}
|
||||||
},
|
},
|
||||||
job_completed: function(data) {
|
job_completed: function(data) {
|
||||||
state.job = Object.assign(state.job, data);
|
if(data.number === state.number) {
|
||||||
state.jobsRunning = [];
|
state.job = Object.assign(state.job, data);
|
||||||
this.$forceUpdate();
|
state.jobsRunning = [];
|
||||||
|
this.$forceUpdate();
|
||||||
|
}
|
||||||
},
|
},
|
||||||
runComplete: function(run) {
|
runComplete: function(run) {
|
||||||
return !!run && (run.result === 'aborted' || run.result === 'failed' || run.result === 'success');
|
return !!run && (run.result === 'aborted' || run.result === 'failed' || run.result === 'success');
|
||||||
},
|
},
|
||||||
},
|
|
||||||
beforeRouteEnter(to, from, next) {
|
|
||||||
next(vm => {
|
|
||||||
state.log = '';
|
|
||||||
vm.logstream = logFetcher(vm, to.params.name, to.params.number);
|
|
||||||
});
|
|
||||||
},
|
|
||||||
beforeRouteUpdate(to, from, next) {
|
|
||||||
var vm = this;
|
|
||||||
vm.logstream.abort();
|
|
||||||
state.log = '';
|
|
||||||
vm.logstream = logFetcher(vm, to.params.name, to.params.number);
|
|
||||||
next();
|
|
||||||
},
|
|
||||||
beforeRouteLeave(to, from, next) {
|
|
||||||
this.logstream.abort();
|
|
||||||
next();
|
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
}();
|
}();
|
||||||
|
@ -85,6 +85,8 @@ canvas {
|
|||||||
svg.success path { fill: var(--success); }
|
svg.success path { fill: var(--success); }
|
||||||
svg.failed path { fill: var(--failure); }
|
svg.failed path { fill: var(--failure); }
|
||||||
svg.running circle { stroke: var(--running); }
|
svg.running circle { stroke: var(--running); }
|
||||||
|
svg.queued circle { fill: var(--nav-fg); }
|
||||||
|
svg.queued path { stroke: white; }
|
||||||
|
|
||||||
/* sort indicators */
|
/* sort indicators */
|
||||||
a.sort {
|
a.sort {
|
||||||
|
11
src/rpc.cpp
11
src/rpc.cpp
@ -53,10 +53,13 @@ public:
|
|||||||
kj::Promise<void> queue(QueueContext context) override {
|
kj::Promise<void> queue(QueueContext context) override {
|
||||||
std::string jobName = context.getParams().getJobName();
|
std::string jobName = context.getParams().getJobName();
|
||||||
LLOG(INFO, "RPC queue", jobName);
|
LLOG(INFO, "RPC queue", jobName);
|
||||||
LaminarCi::MethodResult result = laminar.queueJob(jobName, params(context.getParams().getParams()))
|
std::shared_ptr<Run> run = laminar.queueJob(jobName, params(context.getParams().getParams()));
|
||||||
? LaminarCi::MethodResult::SUCCESS
|
if(Run* r = run.get()) {
|
||||||
: LaminarCi::MethodResult::FAILED;
|
context.getResults().setResult(LaminarCi::MethodResult::SUCCESS);
|
||||||
context.getResults().setResult(result);
|
context.getResults().setBuildNum(r->build);
|
||||||
|
} else {
|
||||||
|
context.getResults().setResult(LaminarCi::MethodResult::FAILED);
|
||||||
|
}
|
||||||
return kj::READY_NOW;
|
return kj::READY_NOW;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
12
src/run.cpp
12
src/run.cpp
@ -33,7 +33,7 @@ inline kj::Path operator/(const kj::Path& p, const T& ext) {
|
|||||||
|
|
||||||
std::string to_string(const RunState& rs) {
|
std::string to_string(const RunState& rs) {
|
||||||
switch(rs) {
|
switch(rs) {
|
||||||
case RunState::PENDING: return "pending";
|
case RunState::QUEUED: return "queued";
|
||||||
case RunState::RUNNING: return "running";
|
case RunState::RUNNING: return "running";
|
||||||
case RunState::ABORTED: return "aborted";
|
case RunState::ABORTED: return "aborted";
|
||||||
case RunState::FAILED: return "failed";
|
case RunState::FAILED: return "failed";
|
||||||
@ -44,9 +44,10 @@ std::string to_string(const RunState& rs) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
Run::Run(std::string name, ParamMap pm, kj::Path&& rootPath) :
|
Run::Run(std::string name, uint num, ParamMap pm, kj::Path&& rootPath) :
|
||||||
result(RunState::SUCCESS),
|
result(RunState::SUCCESS),
|
||||||
name(name),
|
name(name),
|
||||||
|
build(num),
|
||||||
params(kj::mv(pm)),
|
params(kj::mv(pm)),
|
||||||
queuedAt(time(nullptr)),
|
queuedAt(time(nullptr)),
|
||||||
rootPath(kj::mv(rootPath)),
|
rootPath(kj::mv(rootPath)),
|
||||||
@ -83,7 +84,7 @@ static void setEnvFromFile(const kj::Path& rootPath, kj::Path file) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
kj::Promise<RunState> Run::start(uint buildNum, RunState lastResult, std::shared_ptr<Context> ctx, const kj::Directory &fsHome, std::function<kj::Promise<int>(kj::Maybe<pid_t>&)> getPromise)
|
kj::Promise<RunState> Run::start(RunState lastResult, std::shared_ptr<Context> ctx, const kj::Directory &fsHome, std::function<kj::Promise<int>(kj::Maybe<pid_t>&)> getPromise)
|
||||||
{
|
{
|
||||||
kj::Path cfgDir{"cfg"};
|
kj::Path cfgDir{"cfg"};
|
||||||
|
|
||||||
@ -130,7 +131,7 @@ kj::Promise<RunState> Run::start(uint buildNum, RunState lastResult, std::shared
|
|||||||
PATH.append(p);
|
PATH.append(p);
|
||||||
}
|
}
|
||||||
|
|
||||||
std::string runNumStr = std::to_string(buildNum);
|
std::string runNumStr = std::to_string(build);
|
||||||
|
|
||||||
setenv("PATH", PATH.c_str(), true);
|
setenv("PATH", PATH.c_str(), true);
|
||||||
setenv("RUN", runNumStr.c_str(), true);
|
setenv("RUN", runNumStr.c_str(), true);
|
||||||
@ -151,14 +152,13 @@ kj::Promise<RunState> Run::start(uint buildNum, RunState lastResult, std::shared
|
|||||||
// enough. Instead, we'll just exec ourselves and handle that in laminard's
|
// enough. Instead, we'll just exec ourselves and handle that in laminard's
|
||||||
// main() by calling leader_main()
|
// main() by calling leader_main()
|
||||||
char* procName;
|
char* procName;
|
||||||
if(asprintf(&procName, "{laminar} %s:%d", name.data(), buildNum) > 0)
|
if(asprintf(&procName, "{laminar} %s:%d", name.data(), build) > 0)
|
||||||
execl("/proc/self/exe", procName, NULL); // does not return
|
execl("/proc/self/exe", procName, NULL); // does not return
|
||||||
_exit(EXIT_FAILURE);
|
_exit(EXIT_FAILURE);
|
||||||
}
|
}
|
||||||
|
|
||||||
// All good, we've "started"
|
// All good, we've "started"
|
||||||
startedAt = time(nullptr);
|
startedAt = time(nullptr);
|
||||||
build = buildNum;
|
|
||||||
context = ctx;
|
context = ctx;
|
||||||
|
|
||||||
output_fd = plog[0];
|
output_fd = plog[0];
|
||||||
|
@ -34,7 +34,7 @@ typedef unsigned int uint;
|
|||||||
|
|
||||||
enum class RunState {
|
enum class RunState {
|
||||||
UNKNOWN,
|
UNKNOWN,
|
||||||
PENDING,
|
QUEUED,
|
||||||
RUNNING,
|
RUNNING,
|
||||||
ABORTED,
|
ABORTED,
|
||||||
FAILED,
|
FAILED,
|
||||||
@ -50,14 +50,14 @@ typedef std::unordered_map<std::string, std::string> ParamMap;
|
|||||||
// Represents an execution of a job.
|
// Represents an execution of a job.
|
||||||
class Run {
|
class Run {
|
||||||
public:
|
public:
|
||||||
Run(std::string name, ParamMap params, kj::Path&& rootPath);
|
Run(std::string name, uint num, ParamMap params, kj::Path&& rootPath);
|
||||||
~Run();
|
~Run();
|
||||||
|
|
||||||
// copying this class would be asking for trouble...
|
// copying this class would be asking for trouble...
|
||||||
Run(const Run&) = delete;
|
Run(const Run&) = delete;
|
||||||
Run& operator=(const Run&) = delete;
|
Run& operator=(const Run&) = delete;
|
||||||
|
|
||||||
kj::Promise<RunState> start(uint buildNum, RunState lastResult, std::shared_ptr<Context> ctx, const kj::Directory &fsHome, std::function<kj::Promise<int>(kj::Maybe<pid_t>&)> getPromise);
|
kj::Promise<RunState> start(RunState lastResult, std::shared_ptr<Context> ctx, const kj::Directory &fsHome, std::function<kj::Promise<int>(kj::Maybe<pid_t>&)> getPromise);
|
||||||
|
|
||||||
// aborts this run
|
// aborts this run
|
||||||
bool abort();
|
bool abort();
|
||||||
|
Loading…
Reference in New Issue
Block a user