2016-01-06 20:24:41 +00:00
|
|
|
'use strict';
|
|
|
|
|
|
|
|
var Steppy = require('twostep').Steppy,
|
2016-01-07 07:52:01 +00:00
|
|
|
_ = require('underscore'),
|
2016-01-06 20:24:41 +00:00
|
|
|
EventEmitter = require('events').EventEmitter,
|
|
|
|
inherits = require('util').inherits;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Facade entity which accumulates operations with currently running and
|
|
|
|
* db saved builds.
|
|
|
|
*/
|
|
|
|
function BuildsCollection(params) {
|
|
|
|
this.db = params.db;
|
|
|
|
this.distributor = params.distributor;
|
2016-01-07 07:52:01 +00:00
|
|
|
|
|
|
|
this._proxyDistributorEvent('buildUpdate', 'buildUpdated');
|
|
|
|
this._proxyDistributorEvent('buildCancel', 'buildCanceled');
|
|
|
|
this._proxyDistributorEvent('buildLogLines', 'buildLogLines');
|
2016-01-06 20:24:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
exports.BuildsCollection = BuildsCollection;
|
|
|
|
|
|
|
|
inherits(BuildsCollection, EventEmitter);
|
|
|
|
|
2016-01-07 07:52:01 +00:00
|
|
|
BuildsCollection.prototype._proxyDistributorEvent = function(source, dest) {
|
|
|
|
var self = this;
|
|
|
|
|
|
|
|
self.distributor.on(source, function() {
|
|
|
|
self.emit.apply(self, [dest].concat(_(arguments).toArray()));
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2016-01-06 20:24:41 +00:00
|
|
|
BuildsCollection.prototype.create = function(params, callback) {
|
|
|
|
this.distributor.run(params, callback);
|
|
|
|
};
|
|
|
|
|
|
|
|
BuildsCollection.prototype.cancel = function(params, callback) {
|
|
|
|
this.distributor.cancel(params, callback);
|
|
|
|
};
|
|
|
|
|
|
|
|
BuildsCollection.prototype.get = function(id, callback) {
|
2016-01-07 19:27:03 +00:00
|
|
|
this.db.builds.find({start: {id: id}}, function(err, builds) {
|
2016-01-06 20:24:41 +00:00
|
|
|
callback(err, builds && builds[0]);
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
BuildsCollection.prototype.getLogLines = function(params, callback) {
|
|
|
|
var self = this;
|
|
|
|
|
|
|
|
Steppy(
|
|
|
|
function() {
|
|
|
|
var findParams = {
|
|
|
|
start: {buildId: params.buildId},
|
|
|
|
end: {buildId: params.buildId}
|
|
|
|
};
|
|
|
|
if (params.from) findParams.start.number = params.from;
|
|
|
|
if (params.to) findParams.end.number = params.to;
|
|
|
|
|
|
|
|
var count = params.from && params.to ? params.to - params.from + 1: 0;
|
|
|
|
|
|
|
|
self.db.logLines.find(findParams, this.slot());
|
|
|
|
|
|
|
|
this.pass(count);
|
|
|
|
},
|
|
|
|
function(err, logLines, count) {
|
|
|
|
this.pass({
|
|
|
|
lines: logLines,
|
|
|
|
isLast: count ? logLines.length < count : true
|
|
|
|
});
|
|
|
|
},
|
|
|
|
callback
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
BuildsCollection.prototype.getLogLinesTail = function(params, callback) {
|
|
|
|
var self = this;
|
|
|
|
|
|
|
|
Steppy(
|
|
|
|
function() {
|
|
|
|
var findParams = {
|
|
|
|
reverse: true,
|
|
|
|
start: {buildId: params.buildId},
|
|
|
|
limit: params.limit
|
|
|
|
};
|
|
|
|
|
|
|
|
self.db.logLines.find(findParams, this.slot());
|
|
|
|
},
|
|
|
|
function(err, logLines) {
|
|
|
|
var lines = logLines.reverse(),
|
|
|
|
total = logLines.length ? logLines[logLines.length - 1].number : 0;
|
|
|
|
|
|
|
|
this.pass({lines: lines, total: total});
|
|
|
|
},
|
|
|
|
callback
|
|
|
|
);
|
|
|
|
};
|
2016-01-07 08:25:06 +00:00
|
|
|
|
2016-01-07 19:02:59 +00:00
|
|
|
BuildsCollection.prototype.getAvgBuildDuration = function(builds) {
|
|
|
|
var durationsSum = _(builds).reduce(function(sum, build) {
|
|
|
|
return sum + (build.endDate - build.startDate);
|
|
|
|
}, 0);
|
|
|
|
return Math.round(durationsSum / builds.length);
|
2016-01-07 08:25:06 +00:00
|
|
|
};
|
2016-01-07 08:57:45 +00:00
|
|
|
|
|
|
|
BuildsCollection.prototype.getRecent = function(params, callback) {
|
|
|
|
params.limit = params.limit || 20;
|
|
|
|
var self = this;
|
|
|
|
|
|
|
|
Steppy(
|
|
|
|
function() {
|
|
|
|
var findParams = {start: {}, limit: params.limit};
|
|
|
|
|
|
|
|
// such condition for match one of projections:
|
|
|
|
// projectName, descCreateDate
|
|
|
|
// projectName, status, descCreateDate
|
|
|
|
// or just descCreateDate projection if project name is not set
|
|
|
|
if (params.projectName) {
|
|
|
|
findParams.start.projectName = params.projectName;
|
|
|
|
if (params.status) findParams.start.status = params.status;
|
|
|
|
}
|
|
|
|
|
2016-01-07 19:23:35 +00:00
|
|
|
findParams.start.descCreateDate = '';
|
2016-01-07 08:57:45 +00:00
|
|
|
|
|
|
|
self.db.builds.find(findParams, this.slot());
|
|
|
|
},
|
|
|
|
callback
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
BuildsCollection.prototype.getDoneStreak = function(params, callback) {
|
|
|
|
var self = this;
|
|
|
|
|
|
|
|
Steppy(
|
|
|
|
function() {
|
|
|
|
var start = {};
|
|
|
|
|
|
|
|
if (params.projectName) start.projectName = params.projectName;
|
|
|
|
|
|
|
|
start.descCreateDate = '';
|
|
|
|
|
|
|
|
// tricky but effective streak counting inside filter goes below
|
|
|
|
var doneBuildsStreakCallback = _(this.slot()).once(),
|
2016-01-07 16:34:41 +00:00
|
|
|
doneBuildsStreak = {buildsCount: 0};
|
2016-01-07 08:57:45 +00:00
|
|
|
|
|
|
|
self.db.builds.find({
|
|
|
|
start: start,
|
|
|
|
filter: function(build) {
|
|
|
|
// error exits streak
|
|
|
|
if (build.status === 'error') {
|
|
|
|
doneBuildsStreakCallback(null, doneBuildsStreak);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
if (build.status === 'done') {
|
2016-01-07 16:34:41 +00:00
|
|
|
doneBuildsStreak.buildsCount++;
|
2016-01-07 08:57:45 +00:00
|
|
|
}
|
|
|
|
},
|
|
|
|
limit: 1
|
|
|
|
}, function(err) {
|
|
|
|
doneBuildsStreakCallback(err, doneBuildsStreak);
|
|
|
|
});
|
|
|
|
},
|
|
|
|
callback
|
|
|
|
);
|
|
|
|
};
|