mirror of
https://github.com/ansible/awx.git
synced 2026-02-18 03:30:02 -03:30
AC-1183 Job detail page performance
Reorganized initial data load so that it no longer reads all job events. Using objects in place of arrays.
This commit is contained in:
@@ -13,19 +13,22 @@ function JobDetailController ($scope, $compile, $routeParams, $log, ClearScope,
|
|||||||
ClearScope();
|
ClearScope();
|
||||||
|
|
||||||
var job_id = $routeParams.id,
|
var job_id = $routeParams.id,
|
||||||
event_socket, job,
|
event_socket,
|
||||||
event_queue = [],
|
event_queue = [],
|
||||||
processed_events = [],
|
|
||||||
scope = $scope,
|
scope = $scope,
|
||||||
api_complete = false,
|
api_complete = false,
|
||||||
refresh_count = 0;
|
refresh_count = 0,
|
||||||
|
lastEventId = 0;
|
||||||
|
|
||||||
scope.plays = [];
|
scope.plays = {};
|
||||||
scope.tasks = [];
|
scope.tasks = {};
|
||||||
scope.hosts = [];
|
scope.hosts = [];
|
||||||
|
scope.hostResults = [];
|
||||||
|
scope.hostResultsMap = {};
|
||||||
|
scope.hostsMap = {};
|
||||||
|
|
||||||
scope.search_all_tasks = [];
|
scope.search_all_tasks = [];
|
||||||
scope.search_all_plays = [];
|
scope.search_all_plays = [];
|
||||||
scope.hostResults = [];
|
|
||||||
scope.job_status = {};
|
scope.job_status = {};
|
||||||
scope.job_id = job_id;
|
scope.job_id = job_id;
|
||||||
scope.auto_scroll = false;
|
scope.auto_scroll = false;
|
||||||
@@ -55,114 +58,236 @@ function JobDetailController ($scope, $compile, $routeParams, $log, ClearScope,
|
|||||||
|
|
||||||
event_socket.init();
|
event_socket.init();
|
||||||
|
|
||||||
// Evaluate elements of an array, returning the set of elements that
|
|
||||||
// match a condition as expressed in a function
|
|
||||||
//
|
|
||||||
// matches = myarray.find(function(x) { return x.id === 5 });
|
|
||||||
//
|
|
||||||
Array.prototype.find = function(parameterFunction) {
|
|
||||||
var results = [];
|
|
||||||
this.forEach(function(row) {
|
|
||||||
if (parameterFunction(row)) {
|
|
||||||
results.push(row);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
return results;
|
|
||||||
};
|
|
||||||
|
|
||||||
// Reduce an array of objects down to just the bits we want from each object by
|
|
||||||
// passing in a function that returns just those parts.
|
|
||||||
//
|
|
||||||
// new_array = myarray.reduce(function(x) { return { blah: x.blah, foo: x.foo } });
|
|
||||||
//
|
|
||||||
Array.prototype.reduce = function(parameterFunction) {
|
|
||||||
var results= [];
|
|
||||||
this.forEach(function(row) {
|
|
||||||
results.push(parameterFunction(row));
|
|
||||||
});
|
|
||||||
return results;
|
|
||||||
};
|
|
||||||
|
|
||||||
|
|
||||||
// Apply each event to the view
|
|
||||||
if (scope.removeEventsReady) {
|
|
||||||
scope.removeEventsReady();
|
|
||||||
}
|
|
||||||
scope.removeEventsReady = scope.$on('EventsReady', function(e, events) {
|
|
||||||
DigestEvents({
|
|
||||||
scope: scope,
|
|
||||||
events: events
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
event_socket.on("job_events-" + job_id, function(data) {
|
event_socket.on("job_events-" + job_id, function(data) {
|
||||||
var matches;
|
|
||||||
data.event = data.event_name;
|
data.event = data.event_name;
|
||||||
if (api_complete) {
|
if (api_complete) {
|
||||||
matches = processed_events.find(function(x) { return x === data.id; });
|
// api loading is complete, process incoming event
|
||||||
if (matches.length === 0) {
|
DigestEvents({
|
||||||
// event not processed
|
scope: scope,
|
||||||
scope.$emit('EventsReady', [ data ]);
|
events: [ data ]
|
||||||
}
|
});
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
|
// waiting on api load, queue incoming event
|
||||||
event_queue.push(data);
|
event_queue.push(data);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
||||||
if (scope.removeAPIComplete) {
|
if (scope.removeAPIComplete) {
|
||||||
scope.removeAPIComplete();
|
scope.removeAPIComplete();
|
||||||
}
|
}
|
||||||
scope.removeAPIComplete = scope.$on('APIComplete', function() {
|
scope.removeAPIComplete = scope.$on('APIComplete', function() {
|
||||||
var events;
|
// process any events sitting in the queue
|
||||||
|
var events = [];
|
||||||
if (event_queue.length > 0) {
|
if (event_queue.length > 0) {
|
||||||
// Events arrived while we were processing API results
|
event_queue.forEach(function(event) {
|
||||||
events = event_queue.find(function(event) {
|
if (event.id > lastEventId) {
|
||||||
var matched = false;
|
events.push(event);
|
||||||
processed_events.every(function(event_id) {
|
}
|
||||||
if (event_id === event.id) {
|
|
||||||
matched = true;
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
return true;
|
|
||||||
});
|
|
||||||
return (!matched); //return true when event.id not in the list of processed_events
|
|
||||||
});
|
});
|
||||||
if (events.length > 0) {
|
if (events.legnt > 0) {
|
||||||
scope.$emit('EventsReady', events);
|
DigestEvents({
|
||||||
api_complete = true;
|
scope: scope,
|
||||||
|
events: events
|
||||||
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else {
|
api_complete = true;
|
||||||
api_complete = true;
|
|
||||||
}
|
|
||||||
});
|
});
|
||||||
|
|
||||||
// Get events, 50 at a time. When done, emit APIComplete
|
if (scope.removeJobHostSummariesReady) {
|
||||||
if (scope.removeJobReady) {
|
scope.removeJobHostSummariesReady();
|
||||||
scope.removeJobReady();
|
|
||||||
}
|
}
|
||||||
scope.removeJobReady = scope.$on('JobReady', function(e, next) {
|
scope.removeJobHostSummariesReady = scope.$on('JobHostSummariesReady', function() {
|
||||||
Rest.setUrl(next);
|
var url = scope.job.related.job_events + '?parent=' + scope.activeTask + '&host__isnull=false&order_by=host__name';
|
||||||
|
Rest.setUrl(url);
|
||||||
Rest.get()
|
Rest.get()
|
||||||
.success(function(data) {
|
.success(function(data) {
|
||||||
processed_events = processed_events.concat( data.results.reduce(function(x) { return x.id; }) );
|
data.results.forEach(function(event) {
|
||||||
scope.$emit('EventsReady', data.results);
|
scope.hostResults.push({
|
||||||
if (data.next) {
|
id: event.id,
|
||||||
scope.$emit('JobReady', data.next);
|
status: ( (event.failed) ? 'failed': (event.changed) ? 'changed' : 'successful' ),
|
||||||
}
|
host_id: event.host,
|
||||||
else {
|
task_id: event.parent,
|
||||||
Wait('stop');
|
name: event.event_data.host,
|
||||||
scope.$emit('APIComplete');
|
created: event.created,
|
||||||
}
|
msg: ( (event.event_data && event.event_data.res) ? event.event_data.res.msg : '' )
|
||||||
|
});
|
||||||
|
scope.hostResultsMap[event.id] = scope.hostResults.length - 1;
|
||||||
|
if (event.id > lastEventId) {
|
||||||
|
lastEventId = event.id;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
scope.$emit('APIComplete');
|
||||||
|
Wait('stop');
|
||||||
})
|
})
|
||||||
.error(function(data, status) {
|
.error(function(data, status) {
|
||||||
ProcessErrors(scope, data, status, null, { hdr: 'Error!',
|
ProcessErrors(scope, data, status, null, { hdr: 'Error!',
|
||||||
msg: 'Failed to retrieve job events: ' + next + ' GET returned: ' + status });
|
msg: 'Call to ' + url + '. GET returned: ' + status });
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
if (scope.removeTasksReady) {
|
||||||
|
scope.removeTasksReady();
|
||||||
|
}
|
||||||
|
scope.removeTasksReady = scope.$on('TasksReady', function() {
|
||||||
|
var url = scope.job.related.job_host_summaries + '?page_size=' + scope.hostSummaryTableRows + '&order_by=host__name';
|
||||||
|
Rest.setUrl(url);
|
||||||
|
Rest.get()
|
||||||
|
.success(function(data) {
|
||||||
|
data.results.forEach(function(event) {
|
||||||
|
scope.hosts.push({
|
||||||
|
id: event.host,
|
||||||
|
name: event.summary_fields.host.name,
|
||||||
|
ok: event.ok,
|
||||||
|
changed: event.changed,
|
||||||
|
unreachable: event.dark,
|
||||||
|
failed: event.failures
|
||||||
|
});
|
||||||
|
scope.hostsMap[event.id] = scope.hosts.length - 1;
|
||||||
|
});
|
||||||
|
scope.$emit('JobHostSummariesReady');
|
||||||
|
})
|
||||||
|
.error(function(data, status) {
|
||||||
|
ProcessErrors(scope, data, status, null, { hdr: 'Error!',
|
||||||
|
msg: 'Call to ' + url + '. GET returned: ' + status });
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
if (scope.removePlaysReady) {
|
||||||
|
scope.removePlaysReady();
|
||||||
|
}
|
||||||
|
scope.removePlaysReady = scope.$on('PlaysReady', function(e, events_url) {
|
||||||
|
var url = events_url + '?event__in=playbook_on_task_start,playbook_on_setup&parent=' + scope.activePlay + '&order_by=id',
|
||||||
|
task;
|
||||||
|
|
||||||
|
Rest.setUrl(url);
|
||||||
|
Rest.get()
|
||||||
|
.success( function(data) {
|
||||||
|
data.results.forEach(function(event, idx) {
|
||||||
|
var end, play_id, elapsed;
|
||||||
|
|
||||||
|
if (idx < data.results.length - 1) {
|
||||||
|
// end date = starting date of the next event
|
||||||
|
end = data.results[idx + 1].created;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
// no next event (task), get the end time of the play
|
||||||
|
end = scope.plays[scope.activePlay].finished;
|
||||||
|
}
|
||||||
|
if (end) {
|
||||||
|
elapsed = GetElapsed({
|
||||||
|
start: event.created,
|
||||||
|
end: end
|
||||||
|
});
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
elapsed = '00:00:00';
|
||||||
|
}
|
||||||
|
|
||||||
|
play_id = (event.parent) ? event.parent : scope.activePlay;
|
||||||
|
|
||||||
|
if (!scope.tasks[play_id]) {
|
||||||
|
scope.tasks[play_id] = {};
|
||||||
|
}
|
||||||
|
|
||||||
|
scope.tasks[play_id][event.id] = {
|
||||||
|
id: event.id,
|
||||||
|
play_id: play_id,
|
||||||
|
name: event.event_display,
|
||||||
|
status: ( (event.failed) ? 'failed' : (event.changed) ? 'changed' : 'successful' ),
|
||||||
|
created: event.created,
|
||||||
|
modified: event.modified,
|
||||||
|
finished: end,
|
||||||
|
elapsed: elapsed,
|
||||||
|
hostCount: 0, // hostCount,
|
||||||
|
reportedHosts: 0,
|
||||||
|
successfulCount: 0,
|
||||||
|
failedCount: 0,
|
||||||
|
changedCount: 0,
|
||||||
|
skippedCount: 0,
|
||||||
|
successfulStyle: { display: 'none'},
|
||||||
|
failedStyle: { display: 'none' },
|
||||||
|
changedStyle: { display: 'none' },
|
||||||
|
skippedStyle: { display: 'none' },
|
||||||
|
taskActiveClass: ''
|
||||||
|
};
|
||||||
|
});
|
||||||
|
if (data.results.length > 0) {
|
||||||
|
task = data.results[data.results.length - 1];
|
||||||
|
lastEventId = task.id;
|
||||||
|
scope.tasks[scope.activePlay][task.id].taskActiveClass = 'active';
|
||||||
|
scope.activeTask = task.id;
|
||||||
|
scope.activeTaskName = task.name;
|
||||||
|
}
|
||||||
|
scope.$emit('TasksReady', events_url);
|
||||||
|
})
|
||||||
|
.error( function(data, status) {
|
||||||
|
ProcessErrors(scope, data, status, null, { hdr: 'Error!',
|
||||||
|
msg: 'Call to ' + url + '. GET returned: ' + status });
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
if (scope.removeJobReady) {
|
||||||
|
scope.removeJobReady();
|
||||||
|
}
|
||||||
|
scope.removeJobReady = scope.$on('JobReady', function(e, events_url) {
|
||||||
|
// Job finished loading. Now get the set of plays
|
||||||
|
var url = events_url + '?event=playbook_on_play_start&order_by=id',
|
||||||
|
play;
|
||||||
|
Rest.setUrl(url);
|
||||||
|
Rest.get()
|
||||||
|
.success( function(data) {
|
||||||
|
data.results.forEach(function(event, idx) {
|
||||||
|
var status = (event.failed) ? 'failed' : (event.changed) ? 'changed' : 'none',
|
||||||
|
start = event.created,
|
||||||
|
end,
|
||||||
|
elapsed;
|
||||||
|
if (idx < data.results.length - 1) {
|
||||||
|
// end date = starting date of the next event
|
||||||
|
end = data.results[idx + 1].created;
|
||||||
|
}
|
||||||
|
else if (scope.job_status.status === 'successful' || scope.job_status.status === 'failed' || scope.job_status.status === 'error' || scope.job_status.status === 'canceled') {
|
||||||
|
// this is the last play and the job already finished
|
||||||
|
end = scope.job_status.finished;
|
||||||
|
}
|
||||||
|
if (end) {
|
||||||
|
elapsed = GetElapsed({
|
||||||
|
start: start,
|
||||||
|
end: end
|
||||||
|
});
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
elapsed = '00:00:00';
|
||||||
|
}
|
||||||
|
scope.plays[event.id] = {
|
||||||
|
id: event.id,
|
||||||
|
name: event.play,
|
||||||
|
created: event.created,
|
||||||
|
finished: end,
|
||||||
|
status: status,
|
||||||
|
elapsed: elapsed,
|
||||||
|
children: [],
|
||||||
|
playActiveClass: ''
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
|
if (data.results.length > 0) {
|
||||||
|
play = data.results[data.results.length - 1];
|
||||||
|
lastEventId = play.id;
|
||||||
|
scope.plays[play.id].playActiveClass = 'active';
|
||||||
|
scope.activePlay = play.id;
|
||||||
|
scope.activePlayName = play.play;
|
||||||
|
}
|
||||||
|
scope.$emit('PlaysReady', events_url);
|
||||||
|
})
|
||||||
|
.error( function(data, status) {
|
||||||
|
ProcessErrors(scope, data, status, null, { hdr: 'Error!',
|
||||||
|
msg: 'Call to ' + url + '. GET returned: ' + status });
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
|
||||||
if (scope.removeGetCredentialNames) {
|
if (scope.removeGetCredentialNames) {
|
||||||
scope.removeGetCredentialNames();
|
scope.removeGetCredentialNames();
|
||||||
}
|
}
|
||||||
@@ -206,7 +331,7 @@ function JobDetailController ($scope, $compile, $routeParams, $log, ClearScope,
|
|||||||
Rest.setUrl(GetBasePath('jobs') + job_id + '/');
|
Rest.setUrl(GetBasePath('jobs') + job_id + '/');
|
||||||
Rest.get()
|
Rest.get()
|
||||||
.success(function(data) {
|
.success(function(data) {
|
||||||
job = data;
|
scope.job = data;
|
||||||
scope.job_template_name = data.name;
|
scope.job_template_name = data.name;
|
||||||
scope.project_name = (data.summary_fields.project) ? data.summary_fields.project.name : '';
|
scope.project_name = (data.summary_fields.project) ? data.summary_fields.project.name : '';
|
||||||
scope.inventory_name = (data.summary_fields.inventory) ? data.summary_fields.inventory.name : '';
|
scope.inventory_name = (data.summary_fields.inventory) ? data.summary_fields.inventory.name : '';
|
||||||
@@ -229,6 +354,7 @@ function JobDetailController ($scope, $compile, $routeParams, $log, ClearScope,
|
|||||||
scope.job_status.status_class = ((data.status === 'error' || data.status === 'failed') && data.job_explanation) ? "alert alert-danger" : "";
|
scope.job_status.status_class = ((data.status === 'error' || data.status === 'failed') && data.job_explanation) ? "alert alert-danger" : "";
|
||||||
scope.job_status.finished = data.finished;
|
scope.job_status.finished = data.finished;
|
||||||
scope.job_status.explanation = data.job_explanation;
|
scope.job_status.explanation = data.job_explanation;
|
||||||
|
|
||||||
if (data.started && data.finished) {
|
if (data.started && data.finished) {
|
||||||
scope.job_status.elapsed = GetElapsed({
|
scope.job_status.elapsed = GetElapsed({
|
||||||
start: data.started,
|
start: data.started,
|
||||||
@@ -240,7 +366,7 @@ function JobDetailController ($scope, $compile, $routeParams, $log, ClearScope,
|
|||||||
}
|
}
|
||||||
|
|
||||||
scope.setSearchAll('host');
|
scope.setSearchAll('host');
|
||||||
scope.$emit('JobReady', data.related.job_events + '?page_size=50&order_by=id');
|
scope.$emit('JobReady', data.related.job_events);
|
||||||
scope.$emit('GetCredentialNames', data);
|
scope.$emit('GetCredentialNames', data);
|
||||||
})
|
})
|
||||||
.error(function(data, status) {
|
.error(function(data, status) {
|
||||||
|
|||||||
@@ -39,9 +39,9 @@
|
|||||||
|
|
||||||
angular.module('JobDetailHelper', ['Utilities', 'RestServices'])
|
angular.module('JobDetailHelper', ['Utilities', 'RestServices'])
|
||||||
|
|
||||||
.factory('DigestEvents', ['UpdatePlayStatus', 'UpdateHostStatus', 'UpdatePlayChild', 'AddHostResult', 'SelectPlay', 'SelectTask',
|
.factory('DigestEvents', ['UpdatePlayStatus', 'UpdateHostStatus', 'AddHostResult', 'SelectPlay', 'SelectTask',
|
||||||
'GetHostCount', 'GetElapsed', 'UpdateTaskStatus', 'DrawGraph', 'LoadHostSummary',
|
'GetHostCount', 'GetElapsed', 'UpdateTaskStatus', 'DrawGraph', 'LoadHostSummary',
|
||||||
function(UpdatePlayStatus, UpdateHostStatus, UpdatePlayChild, AddHostResult, SelectPlay, SelectTask, GetHostCount, GetElapsed,
|
function(UpdatePlayStatus, UpdateHostStatus, AddHostResult, SelectPlay, SelectTask, GetHostCount, GetElapsed,
|
||||||
UpdateTaskStatus, DrawGraph, LoadHostSummary) {
|
UpdateTaskStatus, DrawGraph, LoadHostSummary) {
|
||||||
return function(params) {
|
return function(params) {
|
||||||
|
|
||||||
@@ -49,7 +49,7 @@ function(UpdatePlayStatus, UpdateHostStatus, UpdatePlayChild, AddHostResult, Sel
|
|||||||
events = params.events;
|
events = params.events;
|
||||||
|
|
||||||
events.forEach(function(event) {
|
events.forEach(function(event) {
|
||||||
var hostCount;
|
var hostCount, play_id;
|
||||||
|
|
||||||
if (event.event === 'playbook_on_start') {
|
if (event.event === 'playbook_on_start') {
|
||||||
if (scope.job_status.status!== 'failed' && scope.job_status.status !== 'canceled' &&
|
if (scope.job_status.status!== 'failed' && scope.job_status.status !== 'canceled' &&
|
||||||
@@ -60,14 +60,14 @@ function(UpdatePlayStatus, UpdateHostStatus, UpdatePlayChild, AddHostResult, Sel
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (event.event === 'playbook_on_play_start') {
|
if (event.event === 'playbook_on_play_start') {
|
||||||
scope.plays.push({
|
scope.plays[event.id] = {
|
||||||
id: event.id,
|
id: event.id,
|
||||||
name: event.play,
|
name: event.play,
|
||||||
created: event.created,
|
created: event.created,
|
||||||
status: (event.failed) ? 'failed' : (event.changed) ? 'changed' : 'none',
|
status: (event.failed) ? 'failed' : (event.changed) ? 'changed' : 'none',
|
||||||
elapsed: '00:00:00',
|
elapsed: '00:00:00',
|
||||||
children: []
|
children: []
|
||||||
});
|
};
|
||||||
SelectPlay({
|
SelectPlay({
|
||||||
scope: scope,
|
scope: scope,
|
||||||
id: event.id
|
id: event.id
|
||||||
@@ -78,10 +78,13 @@ function(UpdatePlayStatus, UpdateHostStatus, UpdatePlayChild, AddHostResult, Sel
|
|||||||
scope: scope,
|
scope: scope,
|
||||||
play_id: event.parent
|
play_id: event.parent
|
||||||
});
|
});
|
||||||
scope.tasks.push({
|
play_id = (event.parent) ? event.parent : scope.activePlay;
|
||||||
|
if (!scope.tasks[play_id]) {
|
||||||
|
scope.tasks[play_id] = {};
|
||||||
|
}
|
||||||
|
scope.tasks[play_id][event.id] = {
|
||||||
id: event.id,
|
id: event.id,
|
||||||
name: event.event_display,
|
name: event.event_display,
|
||||||
play_id: ( (event.parent) ? event.parent : scope.activePlay ),
|
|
||||||
status: ( (event.failed) ? 'failed' : (event.changed) ? 'changed' : 'successful' ),
|
status: ( (event.failed) ? 'failed' : (event.changed) ? 'changed' : 'successful' ),
|
||||||
created: event.created,
|
created: event.created,
|
||||||
modified: event.modified,
|
modified: event.modified,
|
||||||
@@ -95,7 +98,7 @@ function(UpdatePlayStatus, UpdateHostStatus, UpdatePlayChild, AddHostResult, Sel
|
|||||||
failedStyle: { display: 'none' },
|
failedStyle: { display: 'none' },
|
||||||
changedStyle: { display: 'none' },
|
changedStyle: { display: 'none' },
|
||||||
skippedStyle: { display: 'none' }
|
skippedStyle: { display: 'none' }
|
||||||
});
|
};
|
||||||
UpdatePlayStatus({
|
UpdatePlayStatus({
|
||||||
scope: scope,
|
scope: scope,
|
||||||
play_id: event.parent,
|
play_id: event.parent,
|
||||||
@@ -238,25 +241,16 @@ function(UpdatePlayStatus, UpdateHostStatus, UpdatePlayChild, AddHostResult, Sel
|
|||||||
};
|
};
|
||||||
}])
|
}])
|
||||||
|
|
||||||
|
//Get the # of expected hosts for a task by looking at the number
|
||||||
|
//on the very first task for a play
|
||||||
.factory('GetHostCount', function() {
|
.factory('GetHostCount', function() {
|
||||||
return function(params) {
|
return function(params) {
|
||||||
var scope = params.scope,
|
var scope = params.scope,
|
||||||
play_id = params.play_id,
|
play_id = params.play_id,
|
||||||
tasks = [];
|
taskIds;
|
||||||
// Get the known set of tasks for a given play
|
taskIds = Object.keys(scope.tasks[play_id]);
|
||||||
if (scope.tasks.length > 0) {
|
if (taskIds.length > 0) {
|
||||||
scope.tasks.forEach(function(task) {
|
return scope.tasks[play_id][taskIds.length - 1].hostCount;
|
||||||
if (task.play_id === play_id) {
|
|
||||||
tasks.push(task);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
// sort by ascending event.id
|
|
||||||
if (tasks.length > 0) {
|
|
||||||
tasks.sort(function(a, b) {
|
|
||||||
return a.id - b.id;
|
|
||||||
});
|
|
||||||
return tasks[0].hostCount;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
return 0;
|
return 0;
|
||||||
};
|
};
|
||||||
@@ -266,75 +260,15 @@ function(UpdatePlayStatus, UpdateHostStatus, UpdatePlayChild, AddHostResult, Sel
|
|||||||
return function(params) {
|
return function(params) {
|
||||||
var scope = params.scope,
|
var scope = params.scope,
|
||||||
play_id = params.play_id,
|
play_id = params.play_id,
|
||||||
tasks = [];
|
taskIds;
|
||||||
// Get the known set of tasks for a given play
|
taskIds = Object.keys(scope.tasks[play_id]);
|
||||||
if (scope.tasks.length > 0) {
|
if (taskIds.length > 0) {
|
||||||
scope.tasks.forEach(function(task) {
|
return scope.tasks[play_id][taskIds.length - 1].id;
|
||||||
if (task.play_id === play_id) {
|
|
||||||
tasks.push(task);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
// sort by ascending event.id
|
|
||||||
if (tasks.length > 0) {
|
|
||||||
tasks.sort(function(a, b) {
|
|
||||||
return a.id - b.id;
|
|
||||||
});
|
|
||||||
return tasks[0].id;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
return 0;
|
return null;
|
||||||
};
|
};
|
||||||
})
|
})
|
||||||
|
|
||||||
.factory('MakeLastRowActive', [ function() {
|
|
||||||
return function(params) {
|
|
||||||
var scope = params.scope,
|
|
||||||
list = params.list,
|
|
||||||
field = params.field,
|
|
||||||
set = params.set;
|
|
||||||
list.forEach(function(row, idx) {
|
|
||||||
list[idx][field] = '';
|
|
||||||
});
|
|
||||||
if (list.length > 0) {
|
|
||||||
list[list.length - 1][field] = 'active';
|
|
||||||
scope[set] = list[list.length - 1].id;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
}])
|
|
||||||
|
|
||||||
.factory('UpdatePlayChild', [ function() {
|
|
||||||
return function(params) {
|
|
||||||
var scope = params.scope,
|
|
||||||
id = params.id,
|
|
||||||
play_id = params.play_id,
|
|
||||||
failed = params.failed,
|
|
||||||
name = params.name,
|
|
||||||
found_child = false;
|
|
||||||
scope.plays.every(function(play, i) {
|
|
||||||
if (play.id === play_id) {
|
|
||||||
scope.plays[i].children.every(function(child, j) {
|
|
||||||
if (child.id === id) {
|
|
||||||
scope.plays[i].children[j].status = (failed) ? 'failed' : 'successful';
|
|
||||||
found_child = true;
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
return true;
|
|
||||||
});
|
|
||||||
if (!found_child) {
|
|
||||||
scope.plays[i].children.push({
|
|
||||||
id: id,
|
|
||||||
name: name,
|
|
||||||
status: (failed) ? 'failed' : 'successful'
|
|
||||||
});
|
|
||||||
}
|
|
||||||
scope.plays[i].status = (failed) ? 'failed' : 'successful';
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
return true;
|
|
||||||
});
|
|
||||||
};
|
|
||||||
}])
|
|
||||||
|
|
||||||
.factory('GetElapsed', [ function() {
|
.factory('GetElapsed', [ function() {
|
||||||
return function(params) {
|
return function(params) {
|
||||||
var start = params.start,
|
var start = params.start,
|
||||||
@@ -396,35 +330,31 @@ function(UpdatePlayStatus, UpdateHostStatus, UpdatePlayChild, AddHostResult, Sel
|
|||||||
id = params.play_id,
|
id = params.play_id,
|
||||||
modified = params.modified,
|
modified = params.modified,
|
||||||
no_hosts = params.no_hosts,
|
no_hosts = params.no_hosts,
|
||||||
status_text = params.status_text;
|
status_text = params.status_text,
|
||||||
scope.plays.every(function(play,idx) {
|
play = scope.plays[id];
|
||||||
if (play.id === id) {
|
|
||||||
if (failed) {
|
if (failed) {
|
||||||
scope.plays[idx].status = 'failed';
|
scope.plays[id].status = 'failed';
|
||||||
}
|
}
|
||||||
else if (play.status !== 'changed' && play.status !== 'failed') {
|
else if (play.status !== 'changed' && play.status !== 'failed') {
|
||||||
// once the status becomes 'changed' or 'failed' don't modify it
|
// once the status becomes 'changed' or 'failed' don't modify it
|
||||||
if (no_hosts) {
|
if (no_hosts) {
|
||||||
scope.plays[idx].status = 'no-matching-hosts';
|
scope.plays[id].status = 'no-matching-hosts';
|
||||||
}
|
|
||||||
else {
|
|
||||||
scope.plays[idx].status = (changed) ? 'changed' : (failed) ? 'failed' : 'successful';
|
|
||||||
}
|
|
||||||
}
|
|
||||||
scope.plays[idx].finished = modified;
|
|
||||||
scope.plays[idx].elapsed = GetElapsed({
|
|
||||||
start: play.created,
|
|
||||||
end: modified
|
|
||||||
});
|
|
||||||
scope.plays[idx].status_text = (status_text) ? status_text : scope.plays[idx].status;
|
|
||||||
UpdateJobStatus({
|
|
||||||
scope: scope,
|
|
||||||
failed: null,
|
|
||||||
modified: modified
|
|
||||||
});
|
|
||||||
return false;
|
|
||||||
}
|
}
|
||||||
return true;
|
else {
|
||||||
|
scope.plays[id].status = (changed) ? 'changed' : (failed) ? 'failed' : 'successful';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
scope.plays[id].finished = modified;
|
||||||
|
scope.plays[id].elapsed = GetElapsed({
|
||||||
|
start: play.created,
|
||||||
|
end: modified
|
||||||
|
});
|
||||||
|
scope.plays[id].status_text = (status_text) ? status_text : scope.plays[id].status;
|
||||||
|
UpdateJobStatus({
|
||||||
|
scope: scope,
|
||||||
|
failed: null,
|
||||||
|
modified: modified
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
}])
|
}])
|
||||||
@@ -436,35 +366,30 @@ function(UpdatePlayStatus, UpdateHostStatus, UpdatePlayChild, AddHostResult, Sel
|
|||||||
changed = params.changed,
|
changed = params.changed,
|
||||||
id = params.task_id,
|
id = params.task_id,
|
||||||
modified = params.modified,
|
modified = params.modified,
|
||||||
no_hosts = params.no_hosts;
|
no_hosts = params.no_hosts,
|
||||||
scope.tasks.every(function (task, i) {
|
task = scope.tasks[scope.activePlay][id];
|
||||||
if (task.id === id) {
|
if (no_hosts){
|
||||||
if (no_hosts){
|
task.status = 'no-matching-hosts';
|
||||||
scope.tasks[i].status = 'no-matching-hosts';
|
}
|
||||||
}
|
else if (failed) {
|
||||||
else if (failed) {
|
task.status = 'failed';
|
||||||
scope.tasks[i].status = 'failed';
|
}
|
||||||
}
|
else if (task.status !== 'changed' && task.status !== 'failed') {
|
||||||
else if (task.status !== 'changed' && task.status !== 'failed') {
|
// once the status becomes 'changed' or 'failed' don't modify it
|
||||||
// once the status becomes 'changed' or 'failed' don't modify it
|
task.status = (failed) ? 'failed' : (changed) ? 'changed' : 'successful';
|
||||||
scope.tasks[i].status = (failed) ? 'failed' : (changed) ? 'changed' : 'successful';
|
}
|
||||||
}
|
task.finished = params.modified;
|
||||||
scope.tasks[i].finished = params.modified;
|
task.elapsed = GetElapsed({
|
||||||
scope.tasks[i].elapsed = GetElapsed({
|
start: task.created,
|
||||||
start: task.created,
|
end: modified
|
||||||
end: modified
|
});
|
||||||
});
|
UpdatePlayStatus({
|
||||||
UpdatePlayStatus({
|
scope: scope,
|
||||||
scope: scope,
|
failed: failed,
|
||||||
failed: failed,
|
changed: changed,
|
||||||
changed: changed,
|
play_id: task.play_id,
|
||||||
play_id: task.play_id,
|
modified: modified,
|
||||||
modified: modified,
|
no_hosts: no_hosts
|
||||||
no_hosts: no_hosts
|
|
||||||
});
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
return true;
|
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
}])
|
}])
|
||||||
@@ -482,21 +407,16 @@ function(UpdatePlayStatus, UpdateHostStatus, UpdatePlayChild, AddHostResult, Sel
|
|||||||
modified = params.modified,
|
modified = params.modified,
|
||||||
created = params.created,
|
created = params.created,
|
||||||
msg = params.message,
|
msg = params.message,
|
||||||
host_found = false;
|
host;
|
||||||
|
|
||||||
scope.hosts.every(function(host, i) {
|
if (scope.hostsMap[host_id]) {
|
||||||
if (host.id === host_id) {
|
host = scope.hosts[scope.hostsMap[host_id]];
|
||||||
scope.hosts[i].ok += (status === 'successful') ? 1 : 0;
|
host.ok += (status === 'successful') ? 1 : 0;
|
||||||
scope.hosts[i].changed += (status === 'changed') ? 1 : 0;
|
host.changed += (status === 'changed') ? 1 : 0;
|
||||||
scope.hosts[i].unreachable += (status === 'unreachable') ? 1 : 0;
|
host.unreachable += (status === 'unreachable') ? 1 : 0;
|
||||||
scope.hosts[i].failed += (status === 'failed') ? 1 : 0;
|
host.failed += (status === 'failed') ? 1 : 0;
|
||||||
host_found = true;
|
}
|
||||||
return false;
|
else {
|
||||||
}
|
|
||||||
return true;
|
|
||||||
});
|
|
||||||
|
|
||||||
if (!host_found) {
|
|
||||||
// keep totalls for the summary graph
|
// keep totalls for the summary graph
|
||||||
scope.host_summary.total += 1;
|
scope.host_summary.total += 1;
|
||||||
scope.host_summary.ok += (status === 'successful') ? 1 : 0;
|
scope.host_summary.ok += (status === 'successful') ? 1 : 0;
|
||||||
@@ -511,23 +431,29 @@ function(UpdatePlayStatus, UpdateHostStatus, UpdatePlayChild, AddHostResult, Sel
|
|||||||
unreachable: (status === 'unreachable') ? 1 : 0,
|
unreachable: (status === 'unreachable') ? 1 : 0,
|
||||||
failed: (status === 'failed') ? 1 : 0
|
failed: (status === 'failed') ? 1 : 0
|
||||||
});
|
});
|
||||||
scope.hosts.sort(function(a,b) {
|
|
||||||
if (a.name < b.name) {
|
scope.hosts.sort(function (a, b) {
|
||||||
return -1;
|
if (a.name > b.name)
|
||||||
}
|
|
||||||
if (a.name > b.name) {
|
|
||||||
return 1;
|
return 1;
|
||||||
}
|
if (a.name < b.name)
|
||||||
|
return -1;
|
||||||
|
// a must be equal to b
|
||||||
return 0;
|
return 0;
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// prune the hosts array and rebuild the map
|
||||||
if (scope.hosts.length > scope.hostSummaryTableRows) {
|
if (scope.hosts.length > scope.hostSummaryTableRows) {
|
||||||
scope.hosts.pop();
|
scope.hosts.pop();
|
||||||
}
|
}
|
||||||
|
scope.hostsMap = {};
|
||||||
|
scope.hosts.forEach(function(host, idx){
|
||||||
|
scope.hostsMap[host.id] = idx;
|
||||||
|
});
|
||||||
$('#tasks-table-detail').mCustomScrollbar("update");
|
$('#tasks-table-detail').mCustomScrollbar("update");
|
||||||
/*setTimeout( function() {
|
/*setTimeout( function() {
|
||||||
scope.auto_scroll = true;
|
scope.auto_scroll = true;
|
||||||
$('#hosts-summary-table').mCustomScrollbar("scrollTo", "bottom");
|
$('#hosts-summary-table').mCustomScrollbar("scrollTo", "bottom");
|
||||||
}, 700);*/
|
}, 700);*/
|
||||||
}
|
}
|
||||||
|
|
||||||
UpdateTaskStatus({
|
UpdateTaskStatus({
|
||||||
@@ -562,113 +488,85 @@ function(UpdatePlayStatus, UpdateHostStatus, UpdatePlayChild, AddHostResult, Sel
|
|||||||
created = params.created,
|
created = params.created,
|
||||||
name = params.name,
|
name = params.name,
|
||||||
msg = params.message,
|
msg = params.message,
|
||||||
play_id, first,
|
pId, diff, task, play_id, first;
|
||||||
tmp = [];
|
|
||||||
|
|
||||||
// Remove any rows not related to the current task
|
if (scope.activeTask === task_id && !scope.hostResultsMap[host_id]) {
|
||||||
scope.hostResults.forEach(function(result, idx) {
|
// the event applies to the currently selected task
|
||||||
if (result.task_id !== task_id) {
|
scope.hostResults.push({
|
||||||
tmp.push(idx);
|
id: event_id,
|
||||||
|
status: status,
|
||||||
|
host_id: host_id,
|
||||||
|
task_id: task_id,
|
||||||
|
name: name,
|
||||||
|
created: created,
|
||||||
|
msg: msg
|
||||||
|
});
|
||||||
|
scope.hostResults.sort(function(a,b) {
|
||||||
|
if (a.name < b.name) {
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
if (a.name > b.name) {
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
});
|
||||||
|
// Keep the list pruned to a limited # of hosts
|
||||||
|
if (scope.hostResults.length > scope.hostTableRows) {
|
||||||
|
scope.hostResults.splice(0,1);
|
||||||
}
|
}
|
||||||
});
|
|
||||||
tmp.forEach(function(row) {
|
|
||||||
scope.hostResults.splice(row, 1);
|
|
||||||
});
|
|
||||||
|
|
||||||
// Sort
|
|
||||||
scope.hostResults.sort(function(a,b) {
|
|
||||||
if (a.name < b.name) {
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
if (a.name > b.name) {
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
return 0;
|
|
||||||
});
|
|
||||||
|
|
||||||
// Keep the list pruned to a limited # of hosts
|
|
||||||
if (scope.hostResults.length === scope.hostTableRows) {
|
|
||||||
scope.hostResults.splice(0,1);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Add the new row
|
// get the correct play_id for this event. the task_id is always correct/available, but sometimes
|
||||||
scope.hostResults.push({
|
// the event does not have a play_id
|
||||||
id: event_id,
|
for (pId in scope.tasks) {
|
||||||
status: status,
|
if (scope.tasks[pId][task_id]) {
|
||||||
host_id: host_id,
|
play_id = pId;
|
||||||
task_id: task_id,
|
|
||||||
name: name,
|
|
||||||
created: created,
|
|
||||||
msg: msg
|
|
||||||
});
|
|
||||||
|
|
||||||
// Sort one more time
|
|
||||||
scope.hostResults.sort(function(a,b) {
|
|
||||||
if (a.name < b.name) {
|
|
||||||
return -1;
|
|
||||||
}
|
}
|
||||||
if (a.name > b.name) {
|
}
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
return 0;
|
|
||||||
});
|
|
||||||
scope.tasks.every(function(task) {
|
|
||||||
if (task.id === task_id) {
|
|
||||||
play_id = task.play_id;
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
return true;
|
|
||||||
});
|
|
||||||
|
|
||||||
first = FindFirstTaskofPlay({
|
first = FindFirstTaskofPlay({
|
||||||
scope: scope,
|
scope: scope,
|
||||||
play_id: play_id
|
play_id: play_id
|
||||||
});
|
});
|
||||||
|
|
||||||
scope.tasks.every(function(task, idx) {
|
task = scope.tasks[play_id][task_id];
|
||||||
var diff;
|
if (task_id === first) {
|
||||||
if (task.id === task_id) {
|
task.hostCount += 1;
|
||||||
if (task.id === first) {
|
}
|
||||||
scope.tasks[idx].hostCount += 1;
|
task.reportedHosts++;
|
||||||
}
|
task.failedCount += (status === 'failed' || status === 'unreachable') ? 1 : 0;
|
||||||
scope.tasks[idx].reportedHosts++;
|
task.changedCount += (status === 'changed') ? 1 : 0;
|
||||||
scope.tasks[idx].failedCount += (status === 'failed' || status === 'unreachable') ? 1 : 0;
|
task.successfulCount += (status === 'successful') ? 1 : 0;
|
||||||
scope.tasks[idx].changedCount += (status === 'changed') ? 1 : 0;
|
task.skippedCount += (status === 'skipped') ? 1 : 0;
|
||||||
scope.tasks[idx].successfulCount += (status === 'successful') ? 1 : 0;
|
|
||||||
scope.tasks[idx].skippedCount += (status === 'skipped') ? 1 : 0;
|
|
||||||
|
|
||||||
scope.tasks[idx].failedPct = (scope.tasks[idx].hostCount > 0) ? Math.ceil((100 * (scope.tasks[idx].failedCount / scope.tasks[idx].hostCount))) : 0;
|
task.failedPct = (task.hostCount > 0) ? Math.ceil((100 * (task.failedCount / task.hostCount))) : 0;
|
||||||
scope.tasks[idx].changedPct = (scope.tasks[idx].hostCount > 0) ? Math.ceil((100 * (scope.tasks[idx].changedCount / scope.tasks[idx].hostCount))) : 0;
|
task.changedPct = (task.hostCount > 0) ? Math.ceil((100 * (task.changedCount / task.hostCount))) : 0;
|
||||||
scope.tasks[idx].skippedPct = (scope.tasks[idx].hostCount > 0) ? Math.ceil((100 * (scope.tasks[idx].skippedCount / scope.tasks[idx].hostCount))) : 0;
|
task.skippedPct = (task.hostCount > 0) ? Math.ceil((100 * (task.skippedCount / task.hostCount))) : 0;
|
||||||
scope.tasks[idx].successfulPct = (scope.tasks[idx].hostCount > 0) ? Math.ceil((100 * (scope.tasks[idx].successfulCount / scope.tasks[idx].hostCount))) : 0;
|
task.successfulPct = (task.hostCount > 0) ? Math.ceil((100 * (task.successfulCount / task.hostCount))) : 0;
|
||||||
|
|
||||||
diff = (scope.tasks[idx].failedPct + scope.tasks[idx].changedPct + scope.tasks[idx].skippedPct + scope.tasks[idx].successfulPct) - 100;
|
diff = (task.failedPct + task.changedPct + task.skippedPct + task.successfulPct) - 100;
|
||||||
if (diff > 0) {
|
if (diff > 0) {
|
||||||
if (scope.tasks[idx].failedPct > diff) {
|
if (task.failedPct > diff) {
|
||||||
scope.tasks[idx].failedPct = scope.tasks[idx].failedPct - diff;
|
task.failedPct = task.failedPct - diff;
|
||||||
}
|
|
||||||
else if (scope.tasks[idx].changedPct > diff) {
|
|
||||||
scope.tasks[idx].changedPct = scope.tasks[idx].changedPct - diff;
|
|
||||||
}
|
|
||||||
else if (scope.tasks[idx].skippedPct > diff) {
|
|
||||||
scope.tasks[idx].skippedPct = scope.tasks[idx].skippedPct - diff;
|
|
||||||
}
|
|
||||||
else if (scope.tasks[idx].successfulPct > diff) {
|
|
||||||
scope.tasks[idx].successfulPct = scope.tasks[idx].successfulPct - diff;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
scope.tasks[idx].successfulStyle = (scope.tasks[idx].successfulPct > 0) ? { display: 'inline-block', width: scope.tasks[idx].successfulPct + '%' } : { display: 'none' };
|
|
||||||
scope.tasks[idx].changedStyle = (scope.tasks[idx].changedPct > 0) ? { display: 'inline-block', width: scope.tasks[idx].changedPct + '%' } : { display: 'none' };
|
|
||||||
scope.tasks[idx].skippedStyle = (scope.tasks[idx].skippedPct > 0) ? { display: 'inline-block', width: scope.tasks[idx].skippedPct + '%' } : { display: 'none' };
|
|
||||||
scope.tasks[idx].failedStyle = (scope.tasks[idx].failedPct > 0) ? { display: 'inline-block', width: scope.tasks[idx].failedPct + '%' } : { display: 'none' };
|
|
||||||
$('#' + task.id + '-' + task.play_id + '-' + 'successful-bar').css(scope.tasks[idx].successfulStyle);
|
|
||||||
$('#' + task.id + '-' + task.play_id + '-' + 'changed-bar').css(scope.tasks[idx].changedStyle);
|
|
||||||
$('#' + task.id + '-' + task.play_id + '-' + 'skipped-bar').css(scope.tasks[idx].skippedStyle);
|
|
||||||
$('#' + task.id + '-' + task.play_id + '-' + 'failed-bar').css(scope.tasks[idx].failedStyle);
|
|
||||||
return false;
|
|
||||||
}
|
}
|
||||||
return true;
|
else if (task.changedPct > diff) {
|
||||||
});
|
task.changedPct = task.changedPct - diff;
|
||||||
|
}
|
||||||
|
else if (task.skippedPct > diff) {
|
||||||
|
task.skippedPct = task.skippedPct - diff;
|
||||||
|
}
|
||||||
|
else if (task.successfulPct > diff) {
|
||||||
|
task.successfulPct = task.successfulPct - diff;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
task.successfulStyle = (task.successfulPct > 0) ? { display: 'inline-block', width: task.successfulPct + '%' } : { display: 'none' };
|
||||||
|
task.changedStyle = (task.changedPct > 0) ? { display: 'inline-block', width: task.changedPct + '%' } : { display: 'none' };
|
||||||
|
task.skippedStyle = (task.skippedPct > 0) ? { display: 'inline-block', width: task.skippedPct + '%' } : { display: 'none' };
|
||||||
|
task.failedStyle = (task.failedPct > 0) ? { display: 'inline-block', width: task.failedPct + '%' } : { display: 'none' };
|
||||||
|
$('#' + task.id + '-' + task.play_id + '-' + 'successful-bar').css(task.successfulStyle);
|
||||||
|
$('#' + task.id + '-' + task.play_id + '-' + 'changed-bar').css(task.changedStyle);
|
||||||
|
$('#' + task.id + '-' + task.play_id + '-' + 'skipped-bar').css(task.skippedStyle);
|
||||||
|
$('#' + task.id + '-' + task.play_id + '-' + 'failed-bar').css(task.failedStyle);
|
||||||
};
|
};
|
||||||
}])
|
}])
|
||||||
|
|
||||||
@@ -676,50 +574,38 @@ function(UpdatePlayStatus, UpdateHostStatus, UpdatePlayChild, AddHostResult, Sel
|
|||||||
return function(params) {
|
return function(params) {
|
||||||
var scope = params.scope,
|
var scope = params.scope,
|
||||||
id = params.id,
|
id = params.id,
|
||||||
max_task_id = 0;
|
taskIds;
|
||||||
scope.plays.forEach(function(play, idx) {
|
|
||||||
if (play.id === id) {
|
scope.plays[scope.activePlay].playActiveClass = '';
|
||||||
scope.plays[idx].playActiveClass = 'active';
|
scope.plays[id].playActiveClass = 'active';
|
||||||
scope.activePlay = id;
|
scope.activePlay = id;
|
||||||
scope.activePlayName = play.name;
|
scope.activePlayName = scope.plays[id].name;
|
||||||
}
|
|
||||||
else {
|
|
||||||
scope.plays[idx].playActiveClass = '';
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
// Select the last task
|
// Select the last task
|
||||||
scope.tasks.forEach(function(task) {
|
if (scope.tasks[scope.activePlay]) {
|
||||||
if (task.play_id === scope.activePlay && task.id > max_task_id) {
|
taskIds = Object.keys(scope.tasks[scope.activePlay]);
|
||||||
max_task_id = task.id;
|
SelectTask({
|
||||||
}
|
scope: scope,
|
||||||
});
|
id: taskIds[taskIds.length - 1]
|
||||||
scope.activeTask = max_task_id;
|
});
|
||||||
SelectTask({
|
}
|
||||||
scope: scope,
|
else {
|
||||||
id: max_task_id
|
// Not tasks found, clear the host list
|
||||||
});
|
scope.hostResults = [];
|
||||||
|
}
|
||||||
};
|
};
|
||||||
}])
|
}])
|
||||||
|
|
||||||
.factory('SelectTask', ['SelectHost', 'GetBasePath', '$routeParams', 'Rest', 'ProcessErrors',
|
.factory('SelectTask', ['LoadHosts', function(LoadHosts) {
|
||||||
function(SelectHost, GetBasePath, $routeParams, Rest, ProcessErrors) {
|
|
||||||
return function(params) {
|
return function(params) {
|
||||||
var scope = params.scope,
|
var scope = params.scope,
|
||||||
id = params.id,
|
id = params.id,
|
||||||
callback = params.callback,
|
callback = params.callback;
|
||||||
url;
|
|
||||||
|
|
||||||
scope.tasks.forEach(function(task, idx) {
|
scope.tasks[scope.activePlay][scope.activeTask].taskActiveClass = '';
|
||||||
if (task.id === id) {
|
scope.tasks[scope.activePlay][id].taskActiveClass = 'active';
|
||||||
scope.tasks[idx].taskActiveClass = 'active';
|
scope.activeTask = id;
|
||||||
scope.activeTask = id;
|
scope.activeTaskName = scope.tasks[scope.activePlay][id].name;
|
||||||
scope.activeTaskName = task.name;
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
scope.tasks[idx].taskActiveClass = '';
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
if (callback) {
|
if (callback) {
|
||||||
callback();
|
callback();
|
||||||
@@ -731,38 +617,42 @@ function(UpdatePlayStatus, UpdateHostStatus, UpdatePlayChild, AddHostResult, Sel
|
|||||||
$('#tasks-table-detail').mCustomScrollbar("scrollTo", "bottom");
|
$('#tasks-table-detail').mCustomScrollbar("scrollTo", "bottom");
|
||||||
}, 700);
|
}, 700);
|
||||||
|
|
||||||
// Get current list of hosts from the API
|
LoadHosts({
|
||||||
|
scope: scope
|
||||||
// is the job done? if so, only select hosts for the last task?
|
});
|
||||||
|
};
|
||||||
|
}])
|
||||||
|
|
||||||
|
.factory('LoadHosts', ['Rest', 'ProcessErrors', 'SelectHost', function(Rest, ProcessErrors, SelectHost) {
|
||||||
|
return function(params) {
|
||||||
|
var scope = params.scope,
|
||||||
|
url;
|
||||||
scope.hostResults = [];
|
scope.hostResults = [];
|
||||||
if (id > 0) {
|
// If we have a selected task, then get the list of hosts
|
||||||
// If we have a selected task, then get the list of hosts
|
url = scope.job.related.job_events + '?parent=' + scope.activeTask + '&';
|
||||||
url = GetBasePath('jobs') + $routeParams.id + '/job_events/?parent=' + id + '&';
|
url += (scope.search_all_hosts_name) ? 'host__name__icontains=' + scope.search_all_hosts_name + '&' : '';
|
||||||
url += (scope.search_all_hosts_name) ? 'host__name__icontains=' + scope.search_all_hosts_name + '&' : '';
|
url += (scope.searchAllStatus === 'failed') ? 'failed=true&' : '';
|
||||||
url += (scope.searchAllStatus === 'failed') ? 'failed=true&' : '';
|
url += 'host__isnull=false&page_size=' + scope.hostTableRows + '&order_by=host__name';
|
||||||
url += 'host__isnull=false&page_size=' + scope.hostTableRows + '&order_by=host__name';
|
Rest.setUrl(url);
|
||||||
Rest.setUrl(url);
|
Rest.get()
|
||||||
Rest.get()
|
.success(function(data) {
|
||||||
.success(function(data) {
|
data.results.forEach(function(event) {
|
||||||
data.results.forEach(function(row) {
|
scope.hostResults.push({
|
||||||
scope.hostResults.push({
|
id: event.id,
|
||||||
id: row.id,
|
status: ( (event.failed) ? 'failed' : (event.changed) ? 'changed' : 'successful' ),
|
||||||
status: ( (row.failed) ? 'failed' : (row.changed) ? 'changed' : 'successful' ),
|
host_id: event.host,
|
||||||
host_id: row.host,
|
task_id: event.parent,
|
||||||
task_id: row.parent,
|
name: event.event_data.host,
|
||||||
name: row.event_data.host,
|
created: event.created,
|
||||||
created: row.created,
|
msg: ( (event.event_data && event.event_data.res) ? event.event_data.res.msg : '' )
|
||||||
msg: ( (row.event_data && row.event_data.res) ? row.event_data.res.msg : '' )
|
|
||||||
});
|
|
||||||
});
|
});
|
||||||
SelectHost({ scope: scope });
|
|
||||||
})
|
|
||||||
.error(function(data, status) {
|
|
||||||
ProcessErrors(scope, data, status, null, { hdr: 'Error!',
|
|
||||||
msg: 'Call to ' + url + '. GET returned: ' + status });
|
|
||||||
});
|
});
|
||||||
}
|
SelectHost({ scope: scope });
|
||||||
|
})
|
||||||
|
.error(function(data, status) {
|
||||||
|
ProcessErrors(scope, data, status, null, { hdr: 'Error!',
|
||||||
|
msg: 'Call to ' + url + '. GET returned: ' + status });
|
||||||
|
});
|
||||||
};
|
};
|
||||||
}])
|
}])
|
||||||
|
|
||||||
|
|||||||
@@ -53,8 +53,10 @@
|
|||||||
</div>
|
</div>
|
||||||
<div id="plays-table-detail" aw-custom-scroll class="table-detail">
|
<div id="plays-table-detail" aw-custom-scroll class="table-detail">
|
||||||
<div class="row cursor-pointer"
|
<div class="row cursor-pointer"
|
||||||
ng-repeat="play in plays | FilterById : search_all_plays | filter:{ status : searchAllStatus}"
|
ng-repeat="(play_id, play) in plays track by play_id"
|
||||||
ng-class="play.playActiveClass" ng-click="selectPlay(play.id)">
|
ng-class="play.playActiveClass" ng-click="selectPlay(play.id)">
|
||||||
|
<!-- | FilterById : search_all_plays | filter:{ status : searchAllStatus} -->
|
||||||
|
|
||||||
<div class="col-lg-1 col-md-1 col-sm-2 hidden-xs">{{ play.created | date: 'HH:mm:ss' }}</div>
|
<div class="col-lg-1 col-md-1 col-sm-2 hidden-xs">{{ play.created | date: 'HH:mm:ss' }}</div>
|
||||||
<div class="col-lg-1 col-md-1 hidden-sm hidden-xs" aw-tool-tip="Completed at {{ play.finished | date:'HH:mm:ss' }}"
|
<div class="col-lg-1 col-md-1 hidden-sm hidden-xs" aw-tool-tip="Completed at {{ play.finished | date:'HH:mm:ss' }}"
|
||||||
data-placement="top">{{ play.elapsed }}
|
data-placement="top">{{ play.elapsed }}
|
||||||
@@ -78,8 +80,9 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div id="tasks-table-detail" aw-custom-scroll class="table-detail">
|
<div id="tasks-table-detail" aw-custom-scroll class="table-detail">
|
||||||
<div class="row cursor-pointer" ng-repeat="task in taskList = (tasks | FilterById : search_all_tasks | filter:{ status : searchAllStatus} | filter:{ play_id: activePlay })" ng-class="task.taskActiveClass"
|
<div class="row cursor-pointer" ng-repeat="(task_id, task) in taskList = (tasks[activePlay]) track by task_id" ng-class="task.taskActiveClass" ng-click="selectTask(task.id)">
|
||||||
ng-click="selectTask(task.id)">
|
|
||||||
|
<!-- = (tasks | FilterById : search_all_tasks | filter:{ status : searchAllStatus} | filter:{ play_id: activePlay }} -->
|
||||||
<div class="col-lg-1 col-md-1 col-sm-2 hidden-xs">{{ task.created | date: 'HH:mm:ss' }}</div>
|
<div class="col-lg-1 col-md-1 col-sm-2 hidden-xs">{{ task.created | date: 'HH:mm:ss' }}</div>
|
||||||
<div class="col-lg-1 col-md-1 hidden-sm hidden-xs" aw-tool-tip="Completed at {{ task.finished | date:'HH:mm:ss' }}"
|
<div class="col-lg-1 col-md-1 hidden-sm hidden-xs" aw-tool-tip="Completed at {{ task.finished | date:'HH:mm:ss' }}"
|
||||||
data-placement="top">{{ task.elapsed }}
|
data-placement="top">{{ task.elapsed }}
|
||||||
@@ -92,7 +95,7 @@
|
|||||||
<div class="status-bar"><div class="successful-hosts inner-bar" id="{{ task.id }}-{{ task.play_id }}-successful-bar" aw-tool-tip="Hosts OK" data-placement="top" ng-style="task.successfulStyle">{{ task.successfulCount }}</div><div class="changed-hosts inner-bar" id="{{ task.id }}-{{ task.play_id }}-changed-bar" aw-tool-tip="Hosts Changed" data-placement="top" ng-style="task.changedStyle">{{ task.changedCount }}</div><div class="skipped-hosts inner-bar" id="{{ task.id }}-{{ task.play_id }}-skipped-bar" aw-tool-tip="Hosts Skipped" data-placement="top" ng-style="task.skippedStyle">{{ task.skippedCount }}</div><div class="failed-hosts inner-bar" id="{{ task.id }}-{{ task.play_id }}-failed-bar" aw-tool-tip="Hosts Failed" data-placement="top" ng-style="task.failedStyle">{{ task.failedCount }}</div><div class="no-matching-hosts inner-bar" id="{{ task.id }}-{{ task.play_id }}-no-matching-hosts-bar" aw-tool-tip="No matching hosts were found" data-placement="top" style="width: 100%;" ng-show="task.status === 'no-m atching-hosts'">No matching hosts</div></div>
|
<div class="status-bar"><div class="successful-hosts inner-bar" id="{{ task.id }}-{{ task.play_id }}-successful-bar" aw-tool-tip="Hosts OK" data-placement="top" ng-style="task.successfulStyle">{{ task.successfulCount }}</div><div class="changed-hosts inner-bar" id="{{ task.id }}-{{ task.play_id }}-changed-bar" aw-tool-tip="Hosts Changed" data-placement="top" ng-style="task.changedStyle">{{ task.changedCount }}</div><div class="skipped-hosts inner-bar" id="{{ task.id }}-{{ task.play_id }}-skipped-bar" aw-tool-tip="Hosts Skipped" data-placement="top" ng-style="task.skippedStyle">{{ task.skippedCount }}</div><div class="failed-hosts inner-bar" id="{{ task.id }}-{{ task.play_id }}-failed-bar" aw-tool-tip="Hosts Failed" data-placement="top" ng-style="task.failedStyle">{{ task.failedCount }}</div><div class="no-matching-hosts inner-bar" id="{{ task.id }}-{{ task.play_id }}-no-matching-hosts-bar" aw-tool-tip="No matching hosts were found" data-placement="top" style="width: 100%;" ng-show="task.status === 'no-m atching-hosts'">No matching hosts</div></div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="row" ng-show="taskList.length === 0">
|
<div class="row" ng-show="!taskList">
|
||||||
<div class="col-lg-12">
|
<div class="col-lg-12">
|
||||||
<div class="loading-info">No tasks matched</div>
|
<div class="loading-info">No tasks matched</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -112,7 +115,8 @@
|
|||||||
<div id="hosts-table-detail" aw-custom-scroll data-on-total-scroll="HostDetailOnTotalScroll"
|
<div id="hosts-table-detail" aw-custom-scroll data-on-total-scroll="HostDetailOnTotalScroll"
|
||||||
data-on-total-scroll-back="HostDetailOnTotalScrollBack" class="table-detail">
|
data-on-total-scroll-back="HostDetailOnTotalScrollBack" class="table-detail">
|
||||||
<div id="hosts-table-detail-inner">
|
<div id="hosts-table-detail-inner">
|
||||||
<div class="row" ng-repeat="result in results = (hostResults | filter:{ status : searchAllStatus} | filter:{ task_id: activeTask })">
|
<div class="row" ng-repeat="result in results = (hostResults | filter:{ status : searchAllStatus })">
|
||||||
|
<!-- ng-repeat="result in results = (hostResults | filter:{ status : searchAllStatus} })" -->
|
||||||
<div class="col-lg-7 col-md-7 col-sm-7 col-xs-7 status-column">
|
<div class="col-lg-7 col-md-7 col-sm-7 col-xs-7 status-column">
|
||||||
<a href="" ng-click="viewEvent(result.id)" aw-tool-tip="Event Id: {{ result.id }} Status: {{ result.status }}. Click for details" data-placement="top"><i class="fa icon-job-{{ result.status }}"></i> {{ result.name }}</a>
|
<a href="" ng-click="viewEvent(result.id)" aw-tool-tip="Event Id: {{ result.id }} Status: {{ result.status }}. Click for details" data-placement="top"><i class="fa icon-job-{{ result.status }}"></i> {{ result.name }}</a>
|
||||||
</div>
|
</div>
|
||||||
@@ -120,7 +124,7 @@
|
|||||||
{{ result.msg }}
|
{{ result.msg }}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="row" ng-show="!results || results.length === 0">
|
<div class="row" ng-show="!results">
|
||||||
<div class="col-lg-12">
|
<div class="col-lg-12">
|
||||||
<div class="loading-info">No hosts matched</div>
|
<div class="loading-info">No hosts matched</div>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
Reference in New Issue
Block a user