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

frontend love

This commit is contained in:
Oliver Giles 2015-09-26 22:54:27 +02:00
parent 0df97e95fd
commit 1e0a2ebc36
11 changed files with 304 additions and 151 deletions

View File

@ -57,7 +57,7 @@ add_custom_command(OUTPUT laminar.capnp.c++ laminar.capnp.h
# Zip and compile statically served resources # Zip and compile statically served resources
generate_compressed_bins(${CMAKE_SOURCE_DIR}/src/resources index.html js/app.js generate_compressed_bins(${CMAKE_SOURCE_DIR}/src/resources index.html js/app.js
tpl/home.html tpl/job.html tpl/run.html tpl/log.html tpl/browse.html tpl/home.html tpl/job.html tpl/run.html tpl/browse.html
favicon.ico favicon-152.png icon.png) favicon.ico favicon-152.png icon.png)
# Download 3rd-party frontend JS libs... # Download 3rd-party frontend JS libs...
file(DOWNLOAD https://ajax.googleapis.com/ajax/libs/angularjs/1.3.14/angular.min.js file(DOWNLOAD https://ajax.googleapis.com/ajax/libs/angularjs/1.3.14/angular.min.js

View File

@ -8,6 +8,13 @@
### ###
#LAMINAR_HOME=/var/lib/laminar #LAMINAR_HOME=/var/lib/laminar
###
### LAMINAR_TITLE
###
### Page title to show in web frontend
###
#LAMINAR_TITLE=
### ###
### LAMINAR_KEEP_WORKDIR ### LAMINAR_KEEP_WORKDIR
### ###

View File

@ -118,48 +118,52 @@ void Laminar::sendStatus(LaminarClient* client) {
client->sendMessage(log); client->sendMessage(log);
}); });
} }
} else if(client->scope.type == MonitorScope::RUN) { return;
Json j; }
j.set("type", "status");
j.startObject("data"); Json j;
db->stmt("SELECT startedAt, result, reason FROM builds WHERE name = ? AND number = ?") j.set("type", "status");
j.startObject("data");
j.set("title", getenv("LAMINAR_TITLE") ?: "");
if(client->scope.type == MonitorScope::RUN) {
db->stmt("SELECT queuedAt,startedAt,completedAt, result, reason FROM builds WHERE name = ? AND number = ?")
.bind(client->scope.job, client->scope.num) .bind(client->scope.job, client->scope.num)
.fetch<time_t, int, std::string>([&](time_t started, int result, std::string reason) { .fetch<time_t, time_t, time_t, int, std::string>([&](time_t queued, time_t started, time_t completed, int result, std::string reason) {
j.set("queued", started-queued);
j.set("started", started); j.set("started", started);
j.set("completed", completed);
j.set("duration", completed-started);
j.set("result", to_string(RunState(result))); j.set("result", to_string(RunState(result)));
j.set("reason", reason); j.set("reason", reason);
}); });
j.set("latestNum", int(buildNums[client->scope.job]));
j.startArray("artifacts"); j.startArray("artifacts");
fs::path dir(fs::path(homeDir)/"archive"/client->scope.job/std::to_string(client->scope.num)); fs::path dir(fs::path(homeDir)/"archive"/client->scope.job/std::to_string(client->scope.num));
fs::recursive_directory_iterator rdt(dir); if(fs::is_directory(dir)) {
int prefixLen = (fs::path(homeDir)/"archive").string().length(); fs::recursive_directory_iterator rdt(dir);
int scopeLen = dir.string().length(); int prefixLen = (fs::path(homeDir)/"archive").string().length();
for(fs::directory_entry e : rdt) { int scopeLen = dir.string().length();
if(!fs::is_regular_file(e)) for(fs::directory_entry e : rdt) {
continue; if(!fs::is_regular_file(e))
j.StartObject(); continue;
j.set("url", archiveUrl + e.path().string().substr(prefixLen)); j.StartObject();
j.set("filename", e.path().string().substr(scopeLen+1)); j.set("url", archiveUrl + e.path().string().substr(prefixLen));
j.EndObject(); j.set("filename", e.path().string().substr(scopeLen+1));
j.EndObject();
}
} }
j.EndArray(); j.EndArray();
j.EndObject();
client->sendMessage(j.str());
} else if(client->scope.type == MonitorScope::JOB) { } else if(client->scope.type == MonitorScope::JOB) {
Json j;
j.set("type", "status");
j.startObject("data");
j.startArray("recent"); j.startArray("recent");
db->stmt("SELECT * FROM builds WHERE name = ? ORDER BY completedAt DESC LIMIT 5") db->stmt("SELECT number,startedAt,completedAt,result,reason FROM builds WHERE name = ? ORDER BY completedAt DESC LIMIT 25")
.bind(client->scope.job) .bind(client->scope.job)
.fetch<str,int,str,time_t,time_t,time_t,int>([&](str name,int build,str node,time_t,time_t started,time_t completed,int result){ .fetch<int,time_t,time_t,int,str>([&](int build,time_t started,time_t completed,int result,str reason){
j.StartObject(); j.StartObject();
j.set("name", name) j.set("number", build)
.set("number", build)
.set("node", node)
.set("duration", completed - started) .set("duration", completed - started)
.set("started", started) .set("started", started)
.set("result", to_string(RunState(result))) .set("result", to_string(RunState(result)))
.set("reason", reason)
.EndObject(); .EndObject();
}); });
j.EndArray(); j.EndArray();
@ -168,45 +172,52 @@ void Laminar::sendStatus(LaminarClient* client) {
for(auto it = p.first; it != p.second; ++it) { for(auto it = p.first; it != p.second; ++it) {
const std::shared_ptr<Run> run = *it; const std::shared_ptr<Run> run = *it;
j.StartObject(); j.StartObject();
j.set("name", run->name);
j.set("number", run->build); j.set("number", run->build);
j.set("node", run->node->name); j.set("node", run->node->name);
j.set("started", run->startedAt); j.set("started", run->startedAt);
j.EndObject(); j.EndObject();
} }
j.EndArray(); j.EndArray();
j.startArray("queued"); int nQueued = 0;
for(const std::shared_ptr<Run> run : queuedJobs) { for(const std::shared_ptr<Run> run : queuedJobs) {
if (run->name == client->scope.job) { if (run->name == client->scope.job) {
j.StartObject(); nQueued++;
j.set("name", run->name);
j.EndObject();
} }
} }
j.EndArray(); j.set("nQueued", nQueued);
j.EndObject(); db->stmt("SELECT number,startedAt FROM builds WHERE name = ? AND result = ? ORDER BY completedAt DESC LIMIT 1")
client->sendMessage(j.str()); .bind(client->scope.job, int(RunState::SUCCESS))
.fetch<int,time_t>([&](int build, time_t started){
j.startObject("lastSuccess");
j.set("number", build).set("started", started);
j.EndObject();
});
db->stmt("SELECT number,startedAt FROM builds WHERE name = ? AND result <> ? ORDER BY completedAt DESC LIMIT 1")
.bind(client->scope.job, int(RunState::SUCCESS))
.fetch<int,time_t>([&](int build, time_t started){
j.startObject("lastFailed");
j.set("number", build).set("started", started);
j.EndObject();
});
} else if(client->scope.type == MonitorScope::ALL) { } else if(client->scope.type == MonitorScope::ALL) {
Json j;
j.set("type", "status");
j.startObject("data");
j.startArray("jobs"); j.startArray("jobs");
db->stmt("SELECT name FROM builds GROUP BY name") db->stmt("SELECT name,number,startedAt,result FROM builds GROUP BY name ORDER BY number DESC")
.fetch<str>([&](str name){ .fetch<str,int,time_t,int>([&](str name,int number, time_t started, int result){
j.StartObject(); j.StartObject();
j.set("name", name) j.set("name", name);
.EndObject(); j.set("number", number).set("result", to_string(RunState(result))).set("started", started);
j.startArray("tags");
for(const str& t: jobTags[name]) {
j.String(t.c_str());
}
j.EndArray();
j.EndObject();
}); });
j.EndArray(); j.EndArray();
j.EndObject();
client->sendMessage(j.str());
} else { // Home page } else { // Home page
Json j;
j.set("type", "status");
j.startObject("data");
j.startArray("recent"); j.startArray("recent");
db->stmt("SELECT * FROM builds ORDER BY completedAt DESC LIMIT 5") db->stmt("SELECT * FROM builds ORDER BY completedAt DESC LIMIT 15")
.fetch<str,int,str,time_t,time_t,time_t,int>([&](str name,int build,str node,time_t,time_t started,time_t completed,int result){ .fetch<str,int,str,time_t,time_t,time_t,int>([&](str name,int build,str node,time_t,time_t started,time_t completed,int result){
j.StartObject(); j.StartObject();
j.set("name", name) j.set("name", name)
@ -262,10 +273,18 @@ void Laminar::sendStatus(LaminarClient* client) {
j.set(job.c_str(), count); j.set(job.c_str(), count);
}); });
j.EndObject(); j.EndObject();
j.startObject("timePerJob");
db->stmt("SELECT name, AVG(completedAt-startedAt) FROM builds WHERE completedAt > ? GROUP BY name")
.bind(time(0) - 7 * 86400)
.fetch<str, int>([&](str job, int time){
j.set(job.c_str(), time);
});
j.EndObject(); j.EndObject();
client->sendMessage(j.str());
} }
j.EndObject();
client->sendMessage(j.str());
} }
Laminar::~Laminar() { Laminar::~Laminar() {

View File

@ -37,7 +37,6 @@ Resources::Resources()
INIT_RESOURCE("/tpl/home.html", tpl_home_html); INIT_RESOURCE("/tpl/home.html", tpl_home_html);
INIT_RESOURCE("/tpl/job.html", tpl_job_html); INIT_RESOURCE("/tpl/job.html", tpl_job_html);
INIT_RESOURCE("/tpl/run.html", tpl_run_html); INIT_RESOURCE("/tpl/run.html", tpl_run_html);
INIT_RESOURCE("/tpl/log.html", tpl_log_html);
INIT_RESOURCE("/tpl/browse.html", tpl_browse_html); INIT_RESOURCE("/tpl/browse.html", tpl_browse_html);
INIT_RESOURCE("/js/angular.min.js", js_angular_min_js); INIT_RESOURCE("/js/angular.min.js", js_angular_min_js);
INIT_RESOURCE("/js/angular-route.min.js", js_angular_route_min_js); INIT_RESOURCE("/js/angular-route.min.js", js_angular_route_min_js);

View File

@ -17,11 +17,14 @@
<script src="/js/app.js"></script> <script src="/js/app.js"></script>
<style> <style>
body, html { height: 100%; } body, html { height: 100%; }
.navbar { margin-bottom: 0; }
.navbar-brand { margin: 0 -15px; padding: 7px 15px } .navbar-brand { margin: 0 -15px; padding: 7px 15px }
.navbar-brand>img { display: inline; }
a.navbar-btn { color: #9d9d9d; } a.navbar-btn { color: #9d9d9d; }
a.navbar-btn.active { color: #fff; } a.navbar-btn.active { color: #fff; }
a.navbar-btn:hover { color: #fff; text-decoration: none; } a.navbar-btn:hover { color: #fff; text-decoration: none; }
a.navbar-btn:focus { color: #fff; } a.navbar-btn:focus { color: #fff; }
dt,dd { line-height: 2; }
canvas { canvas {
width: 100% !important; width: 100% !important;
max-width: 800px; max-width: 800px;
@ -37,11 +40,15 @@
<nav class="navbar navbar-inverse"> <nav class="navbar navbar-inverse">
<div class="container-fluid"> <div class="container-fluid">
<div> <div>
<a class="navbar-brand" href="/"><img src="/icon.png"></a> <a class="navbar-brand" href="/"><img src="/icon.png">{{title}}</a>
<a class="btn navbar-btn" href="/jobs">Jobs</a> <a class="btn navbar-btn pull-right" href="/jobs">Jobs</a>
</div> </div>
</div> </div>
</nav> </nav>
<ol class="breadcrumb">
<li ng-repeat="n in bc.nodes track by $index"><a href="{{n.href}}">{{n.label}}</a></li>
<li class="active">{{bc.current}}</li>
</ol>
<div ng-view></div> <div ng-view></div>
</body> </body>
</html> </html>

View File

@ -1,13 +1,3 @@
Laminar = {
runIcon: function(result) {
return result === "success" ? '<span style="color:forestgreen">✔</span>' : '<span style="color:crimson;">✘</span>';
},
jobFormatter: function(o) {
o.duration = o.duration + "s"
o.when = (new Date(1000 * o.started)).toLocaleString();
return o;
}
};
angular.module('laminar',['ngRoute','ngSanitize']) angular.module('laminar',['ngRoute','ngSanitize'])
.config(function($routeProvider, $locationProvider, $sceProvider) { .config(function($routeProvider, $locationProvider, $sceProvider) {
$routeProvider $routeProvider
@ -27,10 +17,6 @@ angular.module('laminar',['ngRoute','ngSanitize'])
templateUrl: 'tpl/run.html', templateUrl: 'tpl/run.html',
controller: 'RunController' controller: 'RunController'
}) })
.when('/jobs/:name/:num/log', {
templateUrl: 'tpl/log.html',
controller: 'LogController'
})
$locationProvider.html5Mode(true); $locationProvider.html5Mode(true);
$sceProvider.enabled(false); $sceProvider.enabled(false);
}) })
@ -44,19 +30,23 @@ angular.module('laminar',['ngRoute','ngSanitize'])
}; };
}, },
logListener: function(callback) { logListener: function(callback) {
var ws = new WebSocket("ws://" + location.host + $location.path()); var ws = new WebSocket("ws://" + location.host + $location.path() + '/log');
ws.onmessage = function(message) { ws.onmessage = function(message) {
callback(message.data); callback(message.data);
}; };
} }
}; };
}) })
.controller('mainController', function($scope, $ws, $interval){ .controller('mainController', function($rootScope, $scope, $ws, $interval){
$rootScope.bc = {
nodes: [],
current: 'Home'
};
$scope.jobsQueued = []; $scope.jobsQueued = [];
$scope.jobsRunning = []; $scope.jobsRunning = [];
$scope.jobsRecent = []; $scope.jobsRecent = [];
var chtUtilization, chtBuildsPerDay, chtBuildsPerJob, chtTimePerJob;
var chtUtilization, chtBuildsPerDay, chtBuildsPerJob;
var updateUtilization = function(busy) { var updateUtilization = function(busy) {
chtUtilization.segments[0].value += busy ? 1 : -1; chtUtilization.segments[0].value += busy ? 1 : -1;
@ -66,16 +56,18 @@ angular.module('laminar',['ngRoute','ngSanitize'])
$ws.statusListener({ $ws.statusListener({
status: function(data) { status: function(data) {
$rootScope.title = data.title;
// populate jobs // populate jobs
$scope.jobsQueued = data.queued; $scope.jobsQueued = data.queued;
$scope.jobsRunning = data.running; $scope.jobsRunning = data.running;
$scope.jobsRecent = data.recent.map(Laminar.jobFormatter); $scope.jobsRecent = data.recent;
$scope.$apply(); $scope.$apply();
// setup charts // setup charts
chtUtilization = new Chart(document.getElementById("chartUtil").getContext("2d")).Pie( chtUtilization = new Chart(document.getElementById("chartUtil").getContext("2d")).Pie(
[{value: data.executorsBusy, color:"sandybrown", label: "Busy"}, [{value: data.executorsBusy, color:"tan", label: "Busy"},
{value: data.executorsTotal, color: "steelblue", label: "Idle"}], {value: data.executorsTotal, color: "darkseagreen", label: "Idle"}],
{animationEasing: 'easeInOutQuad'} {animationEasing: 'easeInOutQuad'}
); );
chtBuildsPerDay = new Chart(document.getElementById("chartBpd").getContext("2d")).Line({ chtBuildsPerDay = new Chart(document.getElementById("chartBpd").getContext("2d")).Line({
@ -104,10 +96,17 @@ angular.module('laminar',['ngRoute','ngSanitize'])
chtBuildsPerJob = new Chart(document.getElementById("chartBpj").getContext("2d")).HorizontalBar({ chtBuildsPerJob = new Chart(document.getElementById("chartBpj").getContext("2d")).HorizontalBar({
labels: Object.keys(data.buildsPerJob), labels: Object.keys(data.buildsPerJob),
datasets: [{ datasets: [{
fillColor: "steelblue", fillColor: "lightsteelblue",
data: Object.keys(data.buildsPerJob).map(function(e){return data.buildsPerJob[e];}) data: Object.keys(data.buildsPerJob).map(function(e){return data.buildsPerJob[e];})
}] }]
},{}); },{});
chtTimePerJob = new Chart(document.getElementById("chartTpj").getContext("2d")).HorizontalBar({
labels: Object.keys(data.timePerJob),
datasets: [{
fillColor: "lightsteelblue",
data: Object.keys(data.timePerJob).map(function(e){return data.timePerJob[e];})
}]
},{});
}, },
job_queued: function(data) { job_queued: function(data) {
$scope.jobsQueued.splice(0,0,data); $scope.jobsQueued.splice(0,0,data);
@ -130,7 +129,7 @@ angular.module('laminar',['ngRoute','ngSanitize'])
var job = $scope.jobsRunning[i]; var job = $scope.jobsRunning[i];
if(job.name == data.name && job.number == data.number) { if(job.name == data.name && job.number == data.number) {
$scope.jobsRunning.splice(i,1); $scope.jobsRunning.splice(i,1);
$scope.jobsRecent.splice(0,0,Laminar.jobFormatter(data)); $scope.jobsRecent.splice(0,0,data);
$scope.$apply(); $scope.$apply();
break; break;
@ -146,7 +145,6 @@ angular.module('laminar',['ngRoute','ngSanitize'])
} }
} }
}); });
$scope.runIcon = Laminar.runIcon;
timeUpdater = $interval(function() { timeUpdater = $interval(function() {
$scope.jobsRunning.forEach(function(o){ $scope.jobsRunning.forEach(function(o){
if(o.etc) { if(o.etc) {
@ -166,34 +164,82 @@ angular.module('laminar',['ngRoute','ngSanitize'])
$interval.cancel(timeUpdater); $interval.cancel(timeUpdater);
}); });
}) })
.controller('BrowseController', function($scope, $ws, $interval){ .controller('BrowseController', function($rootScope, $scope, $ws, $interval){
$rootScope.bc = {
nodes: [{ href: '/', label: 'Home' }],
current: 'Jobs'
};
$scope.currentTag = null;
$scope.activeTag = function(t) {
return $scope.currentTag === t;
};
$scope.bytag = function(job) {
if($scope.currentTag === null) return true;
return job.tags.indexOf($scope.currentTag) >= 0;
};
$scope.jobs = []; $scope.jobs = [];
$ws.statusListener({ $ws.statusListener({
status: function(data) { status: function(data) {
$rootScope.title = data.title;
$scope.jobs = data.jobs; $scope.jobs = data.jobs;
var tags = {};
for(var i in data.jobs) {
for(var j in data.jobs[i].tags) {
tags[data.jobs[i].tags[j]] = true;
}
}
$scope.tags = Object.keys(tags);
$scope.$apply(); $scope.$apply();
}, },
}); });
}) })
.controller('JobController', function($scope, $routeParams, $ws) { .controller('JobController', function($rootScope, $scope, $routeParams, $ws) {
$rootScope.bc = {
nodes: [{ href: '/', label: 'Home' },{ href: '/jobs', label: 'Jobs' }],
current: $routeParams.name
};
$scope.name = $routeParams.name; $scope.name = $routeParams.name;
$scope.jobsQueued = [];
$scope.jobsRunning = []; $scope.jobsRunning = [];
$scope.jobsRecent = []; $scope.jobsRecent = [];
$ws.statusListener({ $ws.statusListener({
status: function(data) { status: function(data) {
$scope.jobsQueued = data.queued.filter(function(e){return e.name == $routeParams.name;}); $rootScope.title = data.title;
$scope.jobsRunning = data.running.filter(function(e){return e.name == $routeParams.name;});
$scope.jobsRecent = data.recent.filter(function(e){return e.name == $routeParams.name;}); $scope.jobsRunning = data.running;
$scope.jobsRecent = data.recent;
$scope.lastSuccess = data.lastSuccess;
$scope.lastFailed = data.lastFailed;
$scope.$apply(); $scope.$apply();
},
job_queued: function(data) { var chtBt = new Chart(document.getElementById("chartBt").getContext("2d")).Bar({
if(data.name == $routeParams.name) { labels: data.recent.map(function(e){return '#' + e.number;}),
$scope.jobsQueued.splice(0,0,data); datasets: [{
$scope.$apply(); fillColor: "darkseagreen",
strokeColor: "forestgreen",
data: data.recent.map(function(e){return e.duration;})
}]
},
{barValueSpacing: 1,barStrokeWidth: 1,barDatasetSpacing:0}
);
for(var i = 0; i < data.recent.length; ++i) {
if(data.recent[i].result != "success") {
chtBt.datasets[0].bars[i].fillColor = "darksalmon";
chtBt.datasets[0].bars[i].strokeColor = "crimson";
}
} }
chtBt.update();
},
job_queued: function() {
$scope.nQueued++;
}, },
job_started: function(data) { job_started: function(data) {
$scope.nQueued--;
if(data.name == $routeParams.name) { if(data.name == $routeParams.name) {
$scope.jobsQueued.splice($scope.jobsQueued.length - 1,1); $scope.jobsQueued.splice($scope.jobsQueued.length - 1,1);
$scope.jobsRunning.splice(0,0,data); $scope.jobsRunning.splice(0,0,data);
@ -212,36 +258,62 @@ angular.module('laminar',['ngRoute','ngSanitize'])
} }
} }
}); });
$scope.runIcon = Laminar.runIcon;
}) })
.controller('RunController', function($scope, $routeParams, $ws) { .controller('RunController', function($rootScope, $scope, $routeParams, $ws) {
$rootScope.bc = {
nodes: [{ href: '/', label: 'Home' },
{ href: '/jobs', label: 'Jobs' },
{ href: '/jobs/'+$routeParams.name, label: $routeParams.name }
],
current: '#' + $routeParams.num
};
$scope.name = $routeParams.name; $scope.name = $routeParams.name;
$scope.num = $routeParams.num; $scope.num = parseInt($routeParams.num);
$ws.statusListener({ $ws.statusListener({
status: function(data) { status: function(data) {
$scope.job = Laminar.jobFormatter(data); $rootScope.title = data.title;
$scope.job = data;
$scope.$apply();
},
job_started: function() {
$scope.job.latestNum++;
$scope.$apply(); $scope.$apply();
}, },
job_completed: function(data) { job_completed: function(data) {
$scope.job = Laminar.jobFormatter(data); $scope.job = data;
$scope.$apply(); $scope.$apply();
} }
}); });
$scope.runIcon = Laminar.runIcon;
}) $scope.log = ""
.controller('LogController', function($scope, $routeParams, $ws) { $scope.autoscroll = false;
$scope.name = $routeParams.name; var firstLog = false;
$scope.num = $routeParams.num;
$scope._log = ""
$ws.logListener(function(data) { $ws.logListener(function(data) {
$scope._log += ansi_up.ansi_to_html(data); $scope.log += ansi_up.ansi_to_html(data.replace('<','&lt;').replace('>','&gt;'));
$scope.$apply(); $scope.$apply();
window.scrollTo(0, document.body.scrollHeight); if(!firstLog) {
firstLog = true;
} else if($scope.autoscroll) {
window.scrollTo(0, document.body.scrollHeight);
}
}); });
$scope.log = function() {
// TODO sanitize
return ansi_up.ansi_to_html($scope._log);
}
}) })
.run(function() {}); .run(function($rootScope) {
angular.extend($rootScope, {
runIcon: function(result) {
return result === "success" ? '<span style="color:forestgreen;font-family:\'Zapf Dingbats\';">✔</span>' : result === "failed" ? '<span style="color:crimson;">✘</span>' : '';
},
formatDate: function(unix) {
// TODO reimplement when toLocaleDateString() accepts formatting
// options on most browsers
var d = new Date(1000 * unix);
return d.getHours() + ':' + d.getMinutes() + ' on ' +
['Sun','Mon','Tue','Wed','Thu','Fri','Sat'][d.getDay()] + ' '
+ d.getDate() + '. ' + ['Jan','Feb','Mar','Apr','May','Jun',
'Jul','Aug','Sep', 'Oct','Nov','Dec'][d.getMonth()] + ' '
+ d.getFullYear();
}
});
});

View File

@ -1,14 +1,19 @@
<div class="container-fluid"> <div class="container-fluid">
<div class="row"> <div class="row">
<div class="col-xs-12"> <div class="col-xs-12">
<h3>Browse jobs</h3> <div class="pull-right">
<div class="form-inline form-group"> <input class="form-control" id="jobFilter" ng-model="search.name" placeholder="Filter...">
<label for="jobFilter">Filter</label>
<input id="jobFilter" ng-model="search.name">
</div> </div>
<table class="table table-bordered"> <ul class="nav nav-tabs">
<tr class="animate-repeat" ng-repeat="job in jobs | filter:search:strict"> <li ng-class="{active:activeTag(null)}"><a href ng-click="currentTag = null">All Jobs</a></li>
<li ng-repeat="tag in tags" ng-class="{active:activeTag(tag)}"><a href ng-click="$parent.currentTag = tag">{{tag}}</a></li>
</ul>
<style>table#joblist tr:first-child td { border-top: 0; }</style>
<table class="table table-striped" id="joblist">
<tr class="animate-repeat" ng-repeat="job in jobs | filter:bytag | filter:search">
<td><a href="jobs/{{job.name}}">{{job.name}}</a></td> <td><a href="jobs/{{job.name}}">{{job.name}}</a></td>
<td class="text-center"><span ng-bind-html="runIcon(job.result)"></span> <a href="jobs/{{job.name}}/{{job.numberumber}}">#{{job.number}}</a></td>
<td class="text-center">{{formatDate(job.started)}}</a></td>
</tr> </tr>
</table> </table>
</div> </div>

View File

@ -1,7 +1,6 @@
<div class="container-fluid"> <div class="container-fluid">
<div class="row"> <div class="row">
<div class="col-sm-5 col-md-4 col-lg-3 dash"> <div class="col-sm-5 col-md-4 col-lg-3 dash">
<h3>Recent Builds</h3>
<table class="table table-bordered"> <table class="table table-bordered">
<tr class="animate-repeat" ng-repeat="job in jobsQueued track by $index"> <tr class="animate-repeat" ng-repeat="job in jobsQueued track by $index">
<td><a href="jobs/{{job.name}}">{{job.name}}</a> <i>queued</i></td> <td><a href="jobs/{{job.name}}">{{job.name}}</a> <i>queued</i></td>
@ -13,16 +12,15 @@
</td> </td>
</tr> </tr>
<tr class="animate-repeat" ng-repeat="job in jobsRecent track by $index"> <tr class="animate-repeat" ng-repeat="job in jobsRecent track by $index">
<td><span ng-bind-html="runIcon(job.result)"></span> <a href="jobs/{{job.name}}">{{job.name}}</a> <a href="jobs/{{job.name}}/{{job.number}}">#{{job.number}}</a><br><small>Took {{job.duration}} at {{job.when}}</small></td> <td><span ng-bind-html="runIcon(job.result)"></span> <a href="jobs/{{job.name}}">{{job.name}}</a> <a href="jobs/{{job.name}}/{{job.number}}">#{{job.number}}</a><br><small>Took {{job.duration}}s at {{formatDate(job.started)}}</small></td>
</tr> </tr>
</table> </table>
</div> </div>
<div class="col-sm-7 col-md-8 col-lg-9"> <div class="col-sm-7 col-md-8 col-lg-9">
<h3>Dashboard</h3>
<div class="row"> <div class="row">
<div class="col-md-6"> <div class="col-md-6">
<div class="panel panel-default"> <div class="panel panel-default">
<div class="panel-heading">Builds per day</div> <div class="panel-heading">Total builds per day this week</div>
<div class="panel-body"> <div class="panel-body">
<canvas id="chartBpd"></canvas> <canvas id="chartBpd"></canvas>
</div> </div>
@ -38,16 +36,18 @@
</div> </div>
<div class="col-md-6"> <div class="col-md-6">
<div class="panel panel-default"> <div class="panel panel-default">
<div class="panel-heading">Current executor utilization</div> <div class="panel-heading">Average build time per job this week</div>
<div class="panel-body"> <div class="panel-body">
<canvas id="chartUtil"></canvas> <canvas id="chartTpj"></canvas>
</div> </div>
</div> </div>
</div> </div>
<div class="col-md-6"> <div class="col-md-6">
<div class="panel panel-default"> <div class="panel panel-default">
<div class="panel-heading">what to put here?</div> <div class="panel-heading">Current executor utilization</div>
<div class="panel-body"> <div class="panel-body">
<canvas id="chartUtil"></canvas>
</div> </div>
</div> </div>
</div> </div>

View File

@ -1,22 +1,48 @@
<div class="container-fluid"> <div class="container-fluid">
<div class="row"> <div class="row">
<div class="col-xs-12">
<h3>{{name}}</h3> <div class="col-sm-5 col-md-6 col-lg-7">
<table class="table table-bordered"> <h3>{{name}}</h3>
<tr class="animate-repeat" ng-repeat="job in jobsQueued track by $index"> <dl class="dl-horizontal">
<td><i>queued</i></td> <dt>Last Successful Run</dt><dd>
</tr> <a ng-show="lastSuccess" href="jobs/{{name}}/{{lastSuccess.number}}">#{{lastSuccess.number}}</a>
<tr class="animate-repeat" ng-repeat="job in jobsRunning track by $index"> {{lastSuccess?" - at "+formatDate(lastSuccess.started):"never"}}</dd>
<td><a href="jobs/{{job.name}}/{{job.number}}">#{{job.number}}</a> progressbar?</td> <dt>Last Failed Run</dt><dd>
</tr> <a ng-show="lastFailed" href="jobs/{{name}}/{{lastFailed.number}}">#{{lastFailed.number}}</a>
<tr class="animate-repeat" ng-repeat="job in jobsRecent track by $index"> {{lastFailed?" - at "+formatDate(lastFailed.started):"never"}}</dd>
<td><span ng-bind-html="runIcon(job.result)"></span> <a href="jobs/{{job.name}}/{{job.number}}">#{{job.number}}</a></td> </dl>
</tr>
</table> </div>
<div class="col-sm-7 col-md-6 col-lg-5">
<div class="panel panel-default">
<div class="panel-heading">Build time</div>
<div class="panel-body">
<canvas id="chartBt"></canvas>
</div>
</div>
</div>
</div> </div>
<div class="col-sm-7 col-md-8 col-lg-9"> <div class="row">
<div class="col-xs-12">
<table class="table table-striped"><thead>
<tr><th>Run</th><th class="text-center">Started</th><th class="text-center">Duration</th><th class="text-center hidden-xs">Reason</th></tr></thead>
<tr ng-show="nQueued">
<td colspan="4"><i>{{nQueued}} run(s) queued</i></td>
</tr>
<tr class="animate-repeat" ng-repeat="job in jobsRunning track by $index">
<td><a href="jobs/{{job.name}}/{{job.number}}">#{{job.number}}</a> progressbar?</td>
</tr>
<tr class="animate-repeat" ng-repeat="job in jobsRecent track by $index">
<td><span ng-bind-html="runIcon(job.result)"></span> <a href="jobs/{{name}}/{{job.number}}">#{{job.number}}</a></td>
<td class="text-center">{{formatDate(job.started)}}</td>
<td class="text-center">{{job.duration + " seconds"}}</td>
<td class="text-center hidden-xs">{{job.reason}}</td>
</tr>
</table>
</div>
</div> </div>
</div> </div>
</div>

View File

@ -1,8 +0,0 @@
<div class="container-fluid">
<div class="row">
<div class="col-xs-12">
<h3>Log output for {{name}} #{{num}}</h3>
<pre ng-bind-html="log()"></pre>
</div>
</div>
</div>

View File

@ -1,14 +1,40 @@
<div class="container-fluid"> <div class="container-fluid">
<div class="row"> <div class="row">
<div class="col-xs-12"> <div class="col-sm-5 col-md-6 col-lg-7">
<h3 style="float:left"><span ng-bind-html="runIcon(job.result)"></span> {{name}} #{{num}}</h3>
<nav class="pull-left">
<ul class="pagination" style="margin:15px 20px">
<li><a href="jobs/{{name}}/{{num-1}}">&laquo;</a></li>
<li ng-show="job.latestNum > num"><a ng-href="jobs/{{name}}/{{num+1}}">&raquo;</a></li>
</ul>
</nav>
<div style="clear:both;"></div>
<dl class="dl-horizontal"> <dl class="dl-horizontal">
<dt style="vertical-align:bottom;"></dt><dd><h3><span ng-bind-html="runIcon(job.result)"></span> {{name}} #{{num}}</h3></dd>
<dt><a class="btn btn-default" href="jobs/{{name}}">&lt; Job</a></dt><dd><a class="btn btn-default" href="jobs/{{name}}/{{num}}/log">Log output</a></dd>
<dt></dt><dd>&nbsp;</dd>
<dt>Reason</dt><dd>{{job.reason}}</dd> <dt>Reason</dt><dd>{{job.reason}}</dd>
<dt>Started</dt><dd>{{job.when}}</dd> <dt>Queued for</dt><dd>{{job.queued}}s</dd>
<dt>Artifacts</dt><dd><ul><li ng-repeat="art in job.artifacts"><a href="{{art.url}}" target="_self">{{art.filename}}</a></li></ul></dd> <dt>Started</dt><dd>{{formatDate(job.started)}}</dd>
<dt>Completed</dt><dd>{{formatDate(job.completed)}}</dd>
<dt>Duration</dt><dd>{{job.duration}}s</dd>
</dl> </dl>
</div> </div>
<div class="col-sm-7 col-md-6 col-lg-5">
<div class="panel panel-default" ng-show="job.artifacts.length">
<div class="panel-heading">Artifacts</div>
<div class="panel-body">
<ul class="list-unstyled" style="margin-bottom: 0">
<li ng-repeat="art in job.artifacts">
<a href="{{art.url}}" target="_self">{{art.filename}}</a>
</li>
</ul>
</div>
</div>
</div>
</div>
<div class="row">
<div class="col-xs-12">
<button type="button" class="btn btn-default btn-xs pull-right" ng-class="{active:autoscroll}" ng-click="autoscroll = !autoscroll" style="margin-top:10px">Autoscroll</button>
<h4>Console output</h4>
<pre ng-bind-html="log"></pre>
</div>
</div> </div>
</div> </div>