Merge pull request #311 from CartoDB/309-skipped-status
Fixes #309, added skipped status to fallback-jobs
This commit is contained in:
commit
58055080c9
@ -6,6 +6,7 @@ var JOB_STATUS_ENUM = {
|
|||||||
DONE: 'done',
|
DONE: 'done',
|
||||||
CANCELLED: 'cancelled',
|
CANCELLED: 'cancelled',
|
||||||
FAILED: 'failed',
|
FAILED: 'failed',
|
||||||
|
SKIPPED: 'skipped',
|
||||||
UNKNOWN: 'unknown'
|
UNKNOWN: 'unknown'
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -1,18 +1,9 @@
|
|||||||
'use strict';
|
'use strict';
|
||||||
|
|
||||||
var assert = require('assert');
|
var util = require('util');
|
||||||
var uuid = require('node-uuid');
|
var uuid = require('node-uuid');
|
||||||
|
var JobStateMachine = require('./job_state_machine');
|
||||||
var jobStatus = require('../job_status');
|
var jobStatus = require('../job_status');
|
||||||
var validStatusTransitions = [
|
|
||||||
[jobStatus.PENDING, jobStatus.RUNNING],
|
|
||||||
[jobStatus.PENDING, jobStatus.CANCELLED],
|
|
||||||
[jobStatus.PENDING, jobStatus.UNKNOWN],
|
|
||||||
[jobStatus.RUNNING, jobStatus.DONE],
|
|
||||||
[jobStatus.RUNNING, jobStatus.FAILED],
|
|
||||||
[jobStatus.RUNNING, jobStatus.CANCELLED],
|
|
||||||
[jobStatus.RUNNING, jobStatus.PENDING],
|
|
||||||
[jobStatus.RUNNING, jobStatus.UNKNOWN]
|
|
||||||
];
|
|
||||||
var mandatoryProperties = [
|
var mandatoryProperties = [
|
||||||
'job_id',
|
'job_id',
|
||||||
'status',
|
'status',
|
||||||
@ -24,6 +15,8 @@ var mandatoryProperties = [
|
|||||||
];
|
];
|
||||||
|
|
||||||
function JobBase(data) {
|
function JobBase(data) {
|
||||||
|
JobStateMachine.call(this);
|
||||||
|
|
||||||
var now = new Date().toISOString();
|
var now = new Date().toISOString();
|
||||||
|
|
||||||
this.data = data;
|
this.data = data;
|
||||||
@ -40,24 +33,10 @@ function JobBase(data) {
|
|||||||
this.data.updated_at = now;
|
this.data.updated_at = now;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
util.inherits(JobBase, JobStateMachine);
|
||||||
|
|
||||||
module.exports = JobBase;
|
module.exports = JobBase;
|
||||||
|
|
||||||
JobBase.prototype.isValidStatusTransition = function (initialStatus, finalStatus) {
|
|
||||||
var transition = [ initialStatus, finalStatus ];
|
|
||||||
|
|
||||||
for (var i = 0; i < validStatusTransitions.length; i++) {
|
|
||||||
try {
|
|
||||||
assert.deepEqual(transition, validStatusTransitions[i]);
|
|
||||||
return true;
|
|
||||||
} catch (e) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return false;
|
|
||||||
};
|
|
||||||
|
|
||||||
// should be implemented by childs
|
// should be implemented by childs
|
||||||
JobBase.prototype.getNextQuery = function () {
|
JobBase.prototype.getNextQuery = function () {
|
||||||
throw new Error('Unimplemented method');
|
throw new Error('Unimplemented method');
|
||||||
@ -105,7 +84,7 @@ JobBase.prototype.setQuery = function (query) {
|
|||||||
JobBase.prototype.setStatus = function (finalStatus, errorMesssage) {
|
JobBase.prototype.setStatus = function (finalStatus, errorMesssage) {
|
||||||
var now = new Date().toISOString();
|
var now = new Date().toISOString();
|
||||||
var initialStatus = this.data.status;
|
var initialStatus = this.data.status;
|
||||||
var isValid = this.isValidStatusTransition(initialStatus, finalStatus);
|
var isValid = this.isValidTransition(initialStatus, finalStatus);
|
||||||
|
|
||||||
if (!isValid) {
|
if (!isValid) {
|
||||||
throw new Error('Cannot set status from ' + initialStatus + ' to ' + finalStatus);
|
throw new Error('Cannot set status from ' + initialStatus + ' to ' + finalStatus);
|
||||||
|
@ -3,34 +3,29 @@
|
|||||||
var util = require('util');
|
var util = require('util');
|
||||||
var JobBase = require('./job_base');
|
var JobBase = require('./job_base');
|
||||||
var jobStatus = require('../job_status');
|
var jobStatus = require('../job_status');
|
||||||
var breakStatus = [
|
var QueryFallback = require('./query/query_fallback');
|
||||||
jobStatus.CANCELLED,
|
var MainFallback = require('./query/main_fallback');
|
||||||
jobStatus.FAILED,
|
var QueryFactory = require('./query/query_factory');
|
||||||
jobStatus.UNKNOWN
|
|
||||||
];
|
|
||||||
function isBreakStatus(status) {
|
|
||||||
return breakStatus.indexOf(status) !== -1;
|
|
||||||
}
|
|
||||||
var finalStatus = [
|
|
||||||
jobStatus.CANCELLED,
|
|
||||||
jobStatus.DONE,
|
|
||||||
jobStatus.FAILED,
|
|
||||||
jobStatus.UNKNOWN
|
|
||||||
];
|
|
||||||
function isFinalStatus(status) {
|
|
||||||
return finalStatus.indexOf(status) !== -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
function JobFallback(jobDefinition) {
|
function JobFallback(jobDefinition) {
|
||||||
JobBase.call(this, jobDefinition);
|
JobBase.call(this, jobDefinition);
|
||||||
|
|
||||||
this.init();
|
this.init();
|
||||||
|
|
||||||
|
this.queries = [];
|
||||||
|
for (var i = 0; i < this.data.query.query.length; i++) {
|
||||||
|
this.queries[i] = QueryFactory.create(this.data, i);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (MainFallback.is(this.data)) {
|
||||||
|
this.fallback = new MainFallback();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
util.inherits(JobFallback, JobBase);
|
util.inherits(JobFallback, JobBase);
|
||||||
|
|
||||||
module.exports = JobFallback;
|
module.exports = JobFallback;
|
||||||
|
|
||||||
// from user: {
|
// 1. from user: {
|
||||||
// query: {
|
// query: {
|
||||||
// query: [{
|
// query: [{
|
||||||
// query: 'select ...',
|
// query: 'select ...',
|
||||||
@ -39,7 +34,8 @@ module.exports = JobFallback;
|
|||||||
// onerror: 'select ...'
|
// onerror: 'select ...'
|
||||||
// }
|
// }
|
||||||
// }
|
// }
|
||||||
// from redis: {
|
//
|
||||||
|
// 2. from redis: {
|
||||||
// status: 'pending',
|
// status: 'pending',
|
||||||
// fallback_status: 'pending'
|
// fallback_status: 'pending'
|
||||||
// query: {
|
// query: {
|
||||||
@ -63,11 +59,7 @@ JobFallback.is = function (query) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
for (var i = 0; i < query.query.length; i++) {
|
for (var i = 0; i < query.query.length; i++) {
|
||||||
if (!query.query[i].query) {
|
if (!QueryFallback.is(query.query[i])) {
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (typeof query.query[i].query !== 'string') {
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -76,98 +68,65 @@ JobFallback.is = function (query) {
|
|||||||
};
|
};
|
||||||
|
|
||||||
JobFallback.prototype.init = function () {
|
JobFallback.prototype.init = function () {
|
||||||
// jshint maxcomplexity: 8
|
|
||||||
for (var i = 0; i < this.data.query.query.length; i++) {
|
for (var i = 0; i < this.data.query.query.length; i++) {
|
||||||
if ((this.data.query.query[i].onsuccess || this.data.query.query[i].onerror) &&
|
if (shouldInitStatus(this.data.query.query[i])){
|
||||||
!this.data.query.query[i].status) {
|
|
||||||
this.data.query.query[i].status = jobStatus.PENDING;
|
this.data.query.query[i].status = jobStatus.PENDING;
|
||||||
|
}
|
||||||
|
if (shouldInitQueryFallbackStatus(this.data.query.query[i])) {
|
||||||
this.data.query.query[i].fallback_status = jobStatus.PENDING;
|
this.data.query.query[i].fallback_status = jobStatus.PENDING;
|
||||||
} else if (!this.data.query.query[i].status){
|
|
||||||
this.data.query.query[i].status = jobStatus.PENDING;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if ((this.data.query.onsuccess || this.data.query.onerror) && !this.data.status) {
|
if (shouldInitStatus(this.data)) {
|
||||||
this.data.status = jobStatus.PENDING;
|
this.data.status = jobStatus.PENDING;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (shouldInitFallbackStatus(this.data)) {
|
||||||
this.data.fallback_status = jobStatus.PENDING;
|
this.data.fallback_status = jobStatus.PENDING;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
} else if (!this.data.status) {
|
function shouldInitStatus(jobOrQuery) {
|
||||||
this.data.status = jobStatus.PENDING;
|
return !jobOrQuery.status;
|
||||||
|
}
|
||||||
|
|
||||||
|
function shouldInitQueryFallbackStatus(query) {
|
||||||
|
return (query.onsuccess || query.onerror) && !query.fallback_status;
|
||||||
|
}
|
||||||
|
|
||||||
|
function shouldInitFallbackStatus(job) {
|
||||||
|
return (job.query.onsuccess || job.query.onerror) && !job.fallback_status;
|
||||||
|
}
|
||||||
|
|
||||||
|
JobFallback.prototype.getNextQueryFromQueries = function () {
|
||||||
|
for (var i = 0; i < this.queries.length; i++) {
|
||||||
|
if (this.queries[i].hasNextQuery(this.data)) {
|
||||||
|
return this.queries[i].getNextQuery(this.data);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
JobFallback.prototype.hasNextQueryFromQueries = function () {
|
||||||
|
return !!this.getNextQueryFromQueries();
|
||||||
|
};
|
||||||
|
|
||||||
|
JobFallback.prototype.getNextQueryFromFallback = function () {
|
||||||
|
if (this.fallback && this.fallback.hasNextQuery(this.data)) {
|
||||||
|
|
||||||
|
return this.fallback.getNextQuery(this.data);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
JobFallback.prototype.getNextQuery = function () {
|
JobFallback.prototype.getNextQuery = function () {
|
||||||
var query = this._getNextQueryFromQuery();
|
var query = this.getNextQueryFromQueries();
|
||||||
|
|
||||||
if (!query) {
|
if (!query) {
|
||||||
query = this._getNextQueryFromJobFallback();
|
query = this.getNextQueryFromFallback();
|
||||||
}
|
}
|
||||||
|
|
||||||
return query;
|
return query;
|
||||||
};
|
};
|
||||||
|
|
||||||
JobFallback.prototype._hasNextQueryFromQuery = function () {
|
|
||||||
return !!this._getNextQueryFromQuery();
|
|
||||||
};
|
|
||||||
|
|
||||||
JobFallback.prototype._getNextQueryFromQuery = function () {
|
|
||||||
// jshint maxcomplexity: 8
|
|
||||||
for (var i = 0; i < this.data.query.query.length; i++) {
|
|
||||||
|
|
||||||
if (this.data.query.query[i].fallback_status) {
|
|
||||||
if (this._isNextQuery(i)) {
|
|
||||||
return this.data.query.query[i].query;
|
|
||||||
} else if (this._isNextQueryOnSuccess(i)) {
|
|
||||||
return this.data.query.query[i].onsuccess;
|
|
||||||
} else if (this._isNextQueryOnError(i)) {
|
|
||||||
return this.data.query.query[i].onerror;
|
|
||||||
} else if (isBreakStatus(this.data.query.query[i].status)) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
} else if (this.data.query.query[i].status === jobStatus.PENDING) {
|
|
||||||
return this.data.query.query[i].query;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
JobFallback.prototype._getNextQueryFromJobFallback = function () {
|
|
||||||
if (this.data.fallback_status) {
|
|
||||||
if (this._isNextQueryOnSuccessJob()) {
|
|
||||||
return this.data.query.onsuccess;
|
|
||||||
} else if (this._isNextQueryOnErrorJob()) {
|
|
||||||
return this.data.query.onerror;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
JobFallback.prototype._isNextQuery = function (index) {
|
|
||||||
return this.data.query.query[index].status === jobStatus.PENDING;
|
|
||||||
};
|
|
||||||
|
|
||||||
JobFallback.prototype._isNextQueryOnSuccess = function (index) {
|
|
||||||
return this.data.query.query[index].status === jobStatus.DONE &&
|
|
||||||
this.data.query.query[index].onsuccess &&
|
|
||||||
this.data.query.query[index].fallback_status === jobStatus.PENDING;
|
|
||||||
};
|
|
||||||
|
|
||||||
JobFallback.prototype._isNextQueryOnError = function (index) {
|
|
||||||
return this.data.query.query[index].status === jobStatus.FAILED &&
|
|
||||||
this.data.query.query[index].onerror &&
|
|
||||||
this.data.query.query[index].fallback_status === jobStatus.PENDING;
|
|
||||||
};
|
|
||||||
|
|
||||||
JobFallback.prototype._isNextQueryOnSuccessJob = function () {
|
|
||||||
return this.data.status === jobStatus.DONE &&
|
|
||||||
this.data.query.onsuccess &&
|
|
||||||
this.data.fallback_status === jobStatus.PENDING;
|
|
||||||
};
|
|
||||||
|
|
||||||
JobFallback.prototype._isNextQueryOnErrorJob = function () {
|
|
||||||
return this.data.status === jobStatus.FAILED &&
|
|
||||||
this.data.query.onerror &&
|
|
||||||
this.data.fallback_status === jobStatus.PENDING;
|
|
||||||
};
|
|
||||||
|
|
||||||
JobFallback.prototype.setQuery = function (query) {
|
JobFallback.prototype.setQuery = function (query) {
|
||||||
if (!JobFallback.is(query)) {
|
if (!JobFallback.is(query)) {
|
||||||
throw new Error('You must indicate a valid SQL');
|
throw new Error('You must indicate a valid SQL');
|
||||||
@ -178,126 +137,72 @@ JobFallback.prototype.setQuery = function (query) {
|
|||||||
|
|
||||||
JobFallback.prototype.setStatus = function (status, errorMesssage) {
|
JobFallback.prototype.setStatus = function (status, errorMesssage) {
|
||||||
var now = new Date().toISOString();
|
var now = new Date().toISOString();
|
||||||
var resultFromQuery = this._setQueryStatus(status, errorMesssage);
|
|
||||||
var resultFromJob = this._setJobStatus(status, resultFromQuery.isChangeAppliedToQueryFallback, errorMesssage);
|
|
||||||
|
|
||||||
if (!resultFromJob.isValid && !resultFromQuery.isValid) {
|
var hasChanged = this.setQueryStatus(status, this.data, errorMesssage);
|
||||||
throw new Error('Cannot set status from ' + this.data.status + ' to ' + status);
|
hasChanged = this.setJobStatus(status, this.data, hasChanged, errorMesssage);
|
||||||
|
hasChanged = this.setFallbackStatus(status, this.data, hasChanged);
|
||||||
|
|
||||||
|
if (!hasChanged.isValid) {
|
||||||
|
throw new Error('Cannot set status to ' + status);
|
||||||
}
|
}
|
||||||
|
|
||||||
this.data.updated_at = now;
|
this.data.updated_at = now;
|
||||||
};
|
};
|
||||||
|
|
||||||
JobFallback.prototype._getLastStatusFromFinishedQuery = function () {
|
JobFallback.prototype.setQueryStatus = function (status, job, errorMesssage) {
|
||||||
var lastStatus = jobStatus.DONE;
|
return this.queries.reduce(function (hasChanged, query) {
|
||||||
|
var result = query.setStatus(status, this.data, hasChanged, errorMesssage);
|
||||||
for (var i = 0; i < this.data.query.query.length; i++) {
|
return result.isValid ? result : hasChanged;
|
||||||
if (this.data.query.query[i].fallback_status) {
|
}.bind(this), { isValid: false, appliedToFallback: false });
|
||||||
if (isFinalStatus(this.data.query.query[i].status)) {
|
|
||||||
lastStatus = this.data.query.query[i].status;
|
|
||||||
} else {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
if (isFinalStatus(this.data.query.query[i].status)) {
|
|
||||||
lastStatus = this.data.query.query[i].status;
|
|
||||||
} else {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return lastStatus;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
JobFallback.prototype._setJobStatus = function (status, isChangeAppliedToQueryFallback, errorMesssage) {
|
JobFallback.prototype.setJobStatus = function (status, job, hasChanged, errorMesssage) {
|
||||||
var isValid = false;
|
var result = {
|
||||||
|
isValid: false,
|
||||||
status = this._shiftJobStatus(status, isChangeAppliedToQueryFallback);
|
appliedToFallback: false
|
||||||
|
|
||||||
isValid = this.isValidStatusTransition(this.data.status, status);
|
|
||||||
|
|
||||||
if (isValid) {
|
|
||||||
this.data.status = status;
|
|
||||||
} else if (this.data.fallback_status) {
|
|
||||||
|
|
||||||
isValid = this.isValidStatusTransition(this.data.fallback_status, status);
|
|
||||||
|
|
||||||
if (isValid) {
|
|
||||||
this.data.fallback_status = status;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (status === jobStatus.FAILED && errorMesssage && !isChangeAppliedToQueryFallback) {
|
|
||||||
this.data.failed_reason = errorMesssage;
|
|
||||||
}
|
|
||||||
|
|
||||||
return {
|
|
||||||
isValid: isValid
|
|
||||||
};
|
|
||||||
};
|
};
|
||||||
|
|
||||||
JobFallback.prototype._shiftJobStatus = function (status, isChangeAppliedToQueryFallback) {
|
status = this.shiftStatus(status, hasChanged);
|
||||||
|
|
||||||
|
result.isValid = this.isValidTransition(job.status, status);
|
||||||
|
if (result.isValid) {
|
||||||
|
job.status = status;
|
||||||
|
if (status === jobStatus.FAILED && errorMesssage && !hasChanged.appliedToFallback) {
|
||||||
|
job.failed_reason = errorMesssage;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return result.isValid ? result : hasChanged;
|
||||||
|
};
|
||||||
|
|
||||||
|
JobFallback.prototype.setFallbackStatus = function (status, job, hasChanged) {
|
||||||
|
var result = hasChanged;
|
||||||
|
|
||||||
|
if (this.fallback && !this.hasNextQueryFromQueries()) {
|
||||||
|
result = this.fallback.setStatus(status, job, hasChanged);
|
||||||
|
}
|
||||||
|
|
||||||
|
return result.isValid ? result : hasChanged;
|
||||||
|
};
|
||||||
|
|
||||||
|
JobFallback.prototype.shiftStatus = function (status, hasChanged) {
|
||||||
// jshint maxcomplexity: 7
|
// jshint maxcomplexity: 7
|
||||||
|
if (hasChanged.appliedToFallback) {
|
||||||
// In some scenarios we have to change the normal flow in order to keep consistency
|
if (!this.hasNextQueryFromQueries() && (status === jobStatus.DONE || status === jobStatus.FAILED)) {
|
||||||
// between query's status and job's status.
|
status = this.getLastFinishedStatus();
|
||||||
|
|
||||||
if (isChangeAppliedToQueryFallback) {
|
|
||||||
if (!this._hasNextQueryFromQuery() && (status === jobStatus.DONE || status === jobStatus.FAILED)) {
|
|
||||||
status = this._getLastStatusFromFinishedQuery();
|
|
||||||
} else if (status === jobStatus.DONE || status === jobStatus.FAILED){
|
} else if (status === jobStatus.DONE || status === jobStatus.FAILED){
|
||||||
status = jobStatus.PENDING;
|
status = jobStatus.PENDING;
|
||||||
}
|
}
|
||||||
} else if (this._hasNextQueryFromQuery() && status !== jobStatus.RUNNING) {
|
} else if (this.hasNextQueryFromQueries() && status !== jobStatus.RUNNING) {
|
||||||
status = jobStatus.PENDING;
|
status = jobStatus.PENDING;
|
||||||
}
|
}
|
||||||
|
|
||||||
return status;
|
return status;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
JobFallback.prototype.getLastFinishedStatus = function () {
|
||||||
JobFallback.prototype._shouldTryToApplyStatusTransitionToQueryFallback = function (index) {
|
return this.queries.reduce(function (lastFinished, query) {
|
||||||
return (this.data.query.query[index].status === jobStatus.DONE && this.data.query.query[index].onsuccess) ||
|
var status = query.getStatus(this.data);
|
||||||
(this.data.query.query[index].status === jobStatus.FAILED && this.data.query.query[index].onerror);
|
return this.isFinalStatus(status) ? status : lastFinished;
|
||||||
};
|
}.bind(this), jobStatus.DONE);
|
||||||
|
|
||||||
JobFallback.prototype._setQueryStatus = function (status, errorMesssage) {
|
|
||||||
// jshint maxcomplexity: 7
|
|
||||||
var isValid = false;
|
|
||||||
var isChangeAppliedToQueryFallback = false;
|
|
||||||
|
|
||||||
for (var i = 0; i < this.data.query.query.length; i++) {
|
|
||||||
isValid = this.isValidStatusTransition(this.data.query.query[i].status, status);
|
|
||||||
|
|
||||||
if (isValid) {
|
|
||||||
this.data.query.query[i].status = status;
|
|
||||||
|
|
||||||
if (status === jobStatus.FAILED && errorMesssage) {
|
|
||||||
this.data.query.query[i].failed_reason = errorMesssage;
|
|
||||||
}
|
|
||||||
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (this._shouldTryToApplyStatusTransitionToQueryFallback(i)) {
|
|
||||||
isValid = this.isValidStatusTransition(this.data.query.query[i].fallback_status, status);
|
|
||||||
|
|
||||||
if (isValid) {
|
|
||||||
this.data.query.query[i].fallback_status = status;
|
|
||||||
|
|
||||||
if (status === jobStatus.FAILED && errorMesssage) {
|
|
||||||
this.data.query.query[i].failed_reason = errorMesssage;
|
|
||||||
}
|
|
||||||
|
|
||||||
isChangeAppliedToQueryFallback = true;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return {
|
|
||||||
isValid: isValid,
|
|
||||||
isChangeAppliedToQueryFallback: isChangeAppliedToQueryFallback
|
|
||||||
};
|
|
||||||
};
|
};
|
||||||
|
@ -76,7 +76,7 @@ JobMultiple.prototype.setStatus = function (finalStatus, errorMesssage) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
for (var i = 0; i < this.data.query.length; i++) {
|
for (var i = 0; i < this.data.query.length; i++) {
|
||||||
var isValid = JobMultiple.super_.prototype.isValidStatusTransition(this.data.query[i].status, finalStatus);
|
var isValid = JobMultiple.super_.prototype.isValidTransition(this.data.query[i].status, finalStatus);
|
||||||
|
|
||||||
if (isValid) {
|
if (isValid) {
|
||||||
this.data.query[i].status = finalStatus;
|
this.data.query[i].status = finalStatus;
|
||||||
|
46
batch/models/job_state_machine.js
Normal file
46
batch/models/job_state_machine.js
Normal file
@ -0,0 +1,46 @@
|
|||||||
|
'use strict';
|
||||||
|
|
||||||
|
var assert = require('assert');
|
||||||
|
var jobStatus = require('../job_status');
|
||||||
|
var finalStatus = [
|
||||||
|
jobStatus.CANCELLED,
|
||||||
|
jobStatus.DONE,
|
||||||
|
jobStatus.FAILED,
|
||||||
|
jobStatus.UNKNOWN
|
||||||
|
];
|
||||||
|
|
||||||
|
var validStatusTransitions = [
|
||||||
|
[jobStatus.PENDING, jobStatus.RUNNING],
|
||||||
|
[jobStatus.PENDING, jobStatus.CANCELLED],
|
||||||
|
[jobStatus.PENDING, jobStatus.UNKNOWN],
|
||||||
|
[jobStatus.PENDING, jobStatus.SKIPPED],
|
||||||
|
[jobStatus.RUNNING, jobStatus.DONE],
|
||||||
|
[jobStatus.RUNNING, jobStatus.FAILED],
|
||||||
|
[jobStatus.RUNNING, jobStatus.CANCELLED],
|
||||||
|
[jobStatus.RUNNING, jobStatus.PENDING],
|
||||||
|
[jobStatus.RUNNING, jobStatus.UNKNOWN]
|
||||||
|
];
|
||||||
|
|
||||||
|
function JobStateMachine () {
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = JobStateMachine;
|
||||||
|
|
||||||
|
JobStateMachine.prototype.isValidTransition = function (initialStatus, finalStatus) {
|
||||||
|
var transition = [ initialStatus, finalStatus ];
|
||||||
|
|
||||||
|
for (var i = 0; i < validStatusTransitions.length; i++) {
|
||||||
|
try {
|
||||||
|
assert.deepEqual(transition, validStatusTransitions[i]);
|
||||||
|
return true;
|
||||||
|
} catch (e) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
};
|
||||||
|
|
||||||
|
JobStateMachine.prototype.isFinalStatus = function (status) {
|
||||||
|
return finalStatus.indexOf(status) !== -1;
|
||||||
|
};
|
69
batch/models/query/fallback.js
Normal file
69
batch/models/query/fallback.js
Normal file
@ -0,0 +1,69 @@
|
|||||||
|
'use strict';
|
||||||
|
|
||||||
|
var util = require('util');
|
||||||
|
var QueryBase = require('./query_base');
|
||||||
|
var jobStatus = require('../../job_status');
|
||||||
|
|
||||||
|
function Fallback(index) {
|
||||||
|
QueryBase.call(this, index);
|
||||||
|
}
|
||||||
|
util.inherits(Fallback, QueryBase);
|
||||||
|
|
||||||
|
module.exports = Fallback;
|
||||||
|
|
||||||
|
Fallback.is = function (query) {
|
||||||
|
if (query.onsuccess || query.onerror) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
};
|
||||||
|
|
||||||
|
Fallback.prototype.getNextQuery = function (job) {
|
||||||
|
if (this.hasOnSuccess(job)) {
|
||||||
|
return this.getOnSuccess(job);
|
||||||
|
}
|
||||||
|
if (this.hasOnError(job)) {
|
||||||
|
return this.getOnError(job);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
Fallback.prototype.getOnSuccess = function (job) {
|
||||||
|
if (job.query.query[this.index].status === jobStatus.DONE &&
|
||||||
|
job.query.query[this.index].fallback_status === jobStatus.PENDING) {
|
||||||
|
return job.query.query[this.index].onsuccess;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
Fallback.prototype.hasOnSuccess = function (job) {
|
||||||
|
return !!this.getOnSuccess(job);
|
||||||
|
};
|
||||||
|
|
||||||
|
Fallback.prototype.getOnError = function (job) {
|
||||||
|
if (job.query.query[this.index].status === jobStatus.FAILED &&
|
||||||
|
job.query.query[this.index].fallback_status === jobStatus.PENDING) {
|
||||||
|
return job.query.query[this.index].onerror;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
Fallback.prototype.hasOnError = function (job) {
|
||||||
|
return !!this.getOnError(job);
|
||||||
|
};
|
||||||
|
|
||||||
|
Fallback.prototype.setStatus = function (status, job, errorMessage) {
|
||||||
|
var isValid = false;
|
||||||
|
|
||||||
|
isValid = this.isValidTransition(job.query.query[this.index].fallback_status, status);
|
||||||
|
|
||||||
|
if (isValid) {
|
||||||
|
job.query.query[this.index].fallback_status = status;
|
||||||
|
if (status === jobStatus.FAILED && errorMessage) {
|
||||||
|
job.query.query[this.index].failed_reason = errorMessage;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return isValid;
|
||||||
|
};
|
||||||
|
|
||||||
|
Fallback.prototype.getStatus = function (job) {
|
||||||
|
return job.query.query[this.index].fallback_status;
|
||||||
|
};
|
74
batch/models/query/main_fallback.js
Normal file
74
batch/models/query/main_fallback.js
Normal file
@ -0,0 +1,74 @@
|
|||||||
|
'use strict';
|
||||||
|
|
||||||
|
var util = require('util');
|
||||||
|
var QueryBase = require('./query_base');
|
||||||
|
var jobStatus = require('../../job_status');
|
||||||
|
|
||||||
|
function MainFallback() {
|
||||||
|
QueryBase.call(this);
|
||||||
|
}
|
||||||
|
util.inherits(MainFallback, QueryBase);
|
||||||
|
|
||||||
|
module.exports = MainFallback;
|
||||||
|
|
||||||
|
MainFallback.is = function (job) {
|
||||||
|
if (job.query.onsuccess || job.query.onerror) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
};
|
||||||
|
|
||||||
|
MainFallback.prototype.getNextQuery = function (job) {
|
||||||
|
if (this.hasOnSuccess(job)) {
|
||||||
|
return this.getOnSuccess(job);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.hasOnError(job)) {
|
||||||
|
return this.getOnError(job);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
MainFallback.prototype.getOnSuccess = function (job) {
|
||||||
|
if (job.status === jobStatus.DONE && job.fallback_status === jobStatus.PENDING) {
|
||||||
|
return job.query.onsuccess;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
MainFallback.prototype.hasOnSuccess = function (job) {
|
||||||
|
return !!this.getOnSuccess(job);
|
||||||
|
};
|
||||||
|
|
||||||
|
MainFallback.prototype.getOnError = function (job) {
|
||||||
|
if (job.status === jobStatus.FAILED && job.fallback_status === jobStatus.PENDING) {
|
||||||
|
return job.query.onerror;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
MainFallback.prototype.hasOnError = function (job) {
|
||||||
|
return !!this.getOnError(job);
|
||||||
|
};
|
||||||
|
|
||||||
|
MainFallback.prototype.setStatus = function (status, job, previous) {
|
||||||
|
var isValid = false;
|
||||||
|
var appliedToFallback = false;
|
||||||
|
|
||||||
|
if (previous.isValid && !previous.appliedToFallback) {
|
||||||
|
if (this.isFinalStatus(status) && !this.hasNextQuery(job)) {
|
||||||
|
isValid = this.isValidTransition(job.fallback_status, jobStatus.SKIPPED);
|
||||||
|
|
||||||
|
if (isValid) {
|
||||||
|
job.fallback_status = jobStatus.SKIPPED;
|
||||||
|
appliedToFallback = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else if (!previous.isValid) {
|
||||||
|
isValid = this.isValidTransition(job.fallback_status, status);
|
||||||
|
|
||||||
|
if (isValid) {
|
||||||
|
job.fallback_status = status;
|
||||||
|
appliedToFallback = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return { isValid: isValid, appliedToFallback: appliedToFallback };
|
||||||
|
};
|
45
batch/models/query/query.js
Normal file
45
batch/models/query/query.js
Normal file
@ -0,0 +1,45 @@
|
|||||||
|
'use strict';
|
||||||
|
|
||||||
|
var util = require('util');
|
||||||
|
var QueryBase = require('./query_base');
|
||||||
|
var jobStatus = require('../../job_status');
|
||||||
|
|
||||||
|
function Query(index) {
|
||||||
|
QueryBase.call(this, index);
|
||||||
|
}
|
||||||
|
util.inherits(Query, QueryBase);
|
||||||
|
|
||||||
|
module.exports = Query;
|
||||||
|
|
||||||
|
Query.is = function (query) {
|
||||||
|
if (query.query && typeof query.query === 'string') {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
};
|
||||||
|
|
||||||
|
Query.prototype.getNextQuery = function (job) {
|
||||||
|
if (job.query.query[this.index].status === jobStatus.PENDING) {
|
||||||
|
return job.query.query[this.index].query;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
Query.prototype.setStatus = function (status, job, errorMesssage) {
|
||||||
|
var isValid = false;
|
||||||
|
|
||||||
|
isValid = this.isValidTransition(job.query.query[this.index].status, status);
|
||||||
|
|
||||||
|
if (isValid) {
|
||||||
|
job.query.query[this.index].status = status;
|
||||||
|
if (status === jobStatus.FAILED && errorMesssage) {
|
||||||
|
job.query.query[this.index].failed_reason = errorMesssage;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return isValid;
|
||||||
|
};
|
||||||
|
|
||||||
|
Query.prototype.getStatus = function (job) {
|
||||||
|
return job.query.query[this.index].status;
|
||||||
|
};
|
31
batch/models/query/query_base.js
Normal file
31
batch/models/query/query_base.js
Normal file
@ -0,0 +1,31 @@
|
|||||||
|
'use strict';
|
||||||
|
|
||||||
|
var util = require('util');
|
||||||
|
var JobStateMachine = require('../job_state_machine');
|
||||||
|
|
||||||
|
function QueryBase(index) {
|
||||||
|
JobStateMachine.call(this);
|
||||||
|
|
||||||
|
this.index = index;
|
||||||
|
}
|
||||||
|
util.inherits(QueryBase, JobStateMachine);
|
||||||
|
|
||||||
|
module.exports = QueryBase;
|
||||||
|
|
||||||
|
// should be implemented
|
||||||
|
QueryBase.prototype.setStatus = function () {
|
||||||
|
throw new Error('Unimplemented method');
|
||||||
|
};
|
||||||
|
|
||||||
|
// should be implemented
|
||||||
|
QueryBase.prototype.getNextQuery = function () {
|
||||||
|
throw new Error('Unimplemented method');
|
||||||
|
};
|
||||||
|
|
||||||
|
QueryBase.prototype.hasNextQuery = function (job) {
|
||||||
|
return !!this.getNextQuery(job);
|
||||||
|
};
|
||||||
|
|
||||||
|
QueryBase.prototype.getStatus = function () {
|
||||||
|
throw new Error('Unimplemented method');
|
||||||
|
};
|
16
batch/models/query/query_factory.js
Normal file
16
batch/models/query/query_factory.js
Normal file
@ -0,0 +1,16 @@
|
|||||||
|
'use strict';
|
||||||
|
|
||||||
|
var QueryFallback = require('./query_fallback');
|
||||||
|
|
||||||
|
function QueryFactory() {
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = QueryFactory;
|
||||||
|
|
||||||
|
QueryFactory.create = function (job, index) {
|
||||||
|
if (QueryFallback.is(job.query.query[index])) {
|
||||||
|
return new QueryFallback(job, index);
|
||||||
|
}
|
||||||
|
|
||||||
|
throw new Error('there is no query class for the provided query');
|
||||||
|
};
|
75
batch/models/query/query_fallback.js
Normal file
75
batch/models/query/query_fallback.js
Normal file
@ -0,0 +1,75 @@
|
|||||||
|
'use strict';
|
||||||
|
|
||||||
|
var util = require('util');
|
||||||
|
var QueryBase = require('./query_base');
|
||||||
|
var Query = require('./query');
|
||||||
|
var Fallback = require('./fallback');
|
||||||
|
var jobStatus = require('../../job_status');
|
||||||
|
|
||||||
|
function QueryFallback(job, index) {
|
||||||
|
QueryBase.call(this, index);
|
||||||
|
|
||||||
|
this.init(job, index);
|
||||||
|
}
|
||||||
|
|
||||||
|
util.inherits(QueryFallback, QueryBase);
|
||||||
|
|
||||||
|
QueryFallback.is = function (query) {
|
||||||
|
if (Query.is(query)) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
};
|
||||||
|
|
||||||
|
QueryFallback.prototype.init = function (job, index) {
|
||||||
|
this.query = new Query(index);
|
||||||
|
|
||||||
|
if (Fallback.is(job.query.query[index])) {
|
||||||
|
this.fallback = new Fallback(index);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
QueryFallback.prototype.getNextQuery = function (job) {
|
||||||
|
if (this.query.hasNextQuery(job)) {
|
||||||
|
return this.query.getNextQuery(job);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.fallback && this.fallback.hasNextQuery(job)) {
|
||||||
|
return this.fallback.getNextQuery(job);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
QueryFallback.prototype.setStatus = function (status, job, previous, errorMesssage) {
|
||||||
|
// jshint maxcomplexity: 9
|
||||||
|
var isValid = false;
|
||||||
|
var appliedToFallback = false;
|
||||||
|
|
||||||
|
if (previous.isValid && !previous.appliedToFallback) {
|
||||||
|
if (status === jobStatus.FAILED || status === jobStatus.CANCELLED) {
|
||||||
|
this.query.setStatus(jobStatus.SKIPPED, job, errorMesssage);
|
||||||
|
|
||||||
|
if (this.fallback) {
|
||||||
|
this.fallback.setStatus(jobStatus.SKIPPED, job);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else if (!previous.isValid) {
|
||||||
|
isValid = this.query.setStatus(status, job, errorMesssage);
|
||||||
|
|
||||||
|
if (this.fallback) {
|
||||||
|
if (!isValid) {
|
||||||
|
isValid = this.fallback.setStatus(status, job, errorMesssage);
|
||||||
|
appliedToFallback = true;
|
||||||
|
} else if (isValid && this.isFinalStatus(status) && !this.fallback.hasNextQuery(job)) {
|
||||||
|
this.fallback.setStatus(jobStatus.SKIPPED, job);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return { isValid: isValid, appliedToFallback: appliedToFallback };
|
||||||
|
};
|
||||||
|
|
||||||
|
QueryFallback.prototype.getStatus = function (job) {
|
||||||
|
return this.query.getStatus(job);
|
||||||
|
};
|
||||||
|
|
||||||
|
module.exports = QueryFallback;
|
@ -31,7 +31,6 @@ describe('Batch API fallback job', function () {
|
|||||||
describe('"onsuccess" on first query should be triggered', function () {
|
describe('"onsuccess" on first query should be triggered', function () {
|
||||||
var fallbackJob = {};
|
var fallbackJob = {};
|
||||||
|
|
||||||
|
|
||||||
it('should create a job', function (done) {
|
it('should create a job', function (done) {
|
||||||
assert.response(app, {
|
assert.response(app, {
|
||||||
url: '/api/v2/sql/job?api_key=1234',
|
url: '/api/v2/sql/job?api_key=1234',
|
||||||
@ -133,7 +132,7 @@ describe('Batch API fallback job', function () {
|
|||||||
"query": "SELECT * FROM untitle_table_4",
|
"query": "SELECT * FROM untitle_table_4",
|
||||||
"onerror": "SELECT * FROM untitle_table_4 limit 1",
|
"onerror": "SELECT * FROM untitle_table_4 limit 1",
|
||||||
"status": "done",
|
"status": "done",
|
||||||
"fallback_status": "pending"
|
"fallback_status": "skipped"
|
||||||
}]
|
}]
|
||||||
};
|
};
|
||||||
var interval = setInterval(function () {
|
var interval = setInterval(function () {
|
||||||
@ -268,7 +267,7 @@ describe('Batch API fallback job', function () {
|
|||||||
query: 'SELECT * FROM nonexistent_table /* query should fail */',
|
query: 'SELECT * FROM nonexistent_table /* query should fail */',
|
||||||
onsuccess: 'SELECT * FROM untitle_table_4 limit 1',
|
onsuccess: 'SELECT * FROM untitle_table_4 limit 1',
|
||||||
status: 'failed',
|
status: 'failed',
|
||||||
fallback_status: 'pending',
|
fallback_status: 'skipped',
|
||||||
failed_reason: 'relation "nonexistent_table" does not exist'
|
failed_reason: 'relation "nonexistent_table" does not exist'
|
||||||
}]
|
}]
|
||||||
};
|
};
|
||||||
@ -424,7 +423,7 @@ describe('Batch API fallback job', function () {
|
|||||||
return done(err);
|
return done(err);
|
||||||
}
|
}
|
||||||
var job = JSON.parse(res.body);
|
var job = JSON.parse(res.body);
|
||||||
if (job.status === jobStatus.FAILED && job.fallback_status === jobStatus.PENDING) {
|
if (job.status === jobStatus.FAILED && job.fallback_status === jobStatus.SKIPPED) {
|
||||||
clearInterval(interval);
|
clearInterval(interval);
|
||||||
assert.deepEqual(job.query, expectedQuery);
|
assert.deepEqual(job.query, expectedQuery);
|
||||||
done();
|
done();
|
||||||
@ -560,7 +559,7 @@ describe('Batch API fallback job', function () {
|
|||||||
return done(err);
|
return done(err);
|
||||||
}
|
}
|
||||||
var job = JSON.parse(res.body);
|
var job = JSON.parse(res.body);
|
||||||
if (job.status === jobStatus.DONE && job.fallback_status === jobStatus.PENDING) {
|
if (job.status === jobStatus.DONE && job.fallback_status === jobStatus.SKIPPED) {
|
||||||
clearInterval(interval);
|
clearInterval(interval);
|
||||||
assert.deepEqual(job.query, expectedQuery);
|
assert.deepEqual(job.query, expectedQuery);
|
||||||
done();
|
done();
|
||||||
@ -759,13 +758,13 @@ describe('Batch API fallback job', function () {
|
|||||||
"query": "SELECT * FROM nonexistent_table /* should fail */",
|
"query": "SELECT * FROM nonexistent_table /* should fail */",
|
||||||
"onsuccess": "SELECT * FROM untitle_table_4 limit 1",
|
"onsuccess": "SELECT * FROM untitle_table_4 limit 1",
|
||||||
"status": "failed",
|
"status": "failed",
|
||||||
"fallback_status": "pending",
|
"fallback_status": "skipped",
|
||||||
"failed_reason": 'relation "nonexistent_table" does not exist'
|
"failed_reason": 'relation "nonexistent_table" does not exist'
|
||||||
}, {
|
}, {
|
||||||
"query": "SELECT * FROM untitle_table_4 limit 2",
|
"query": "SELECT * FROM untitle_table_4 limit 2",
|
||||||
"onsuccess": "SELECT * FROM untitle_table_4 limit 3",
|
"onsuccess": "SELECT * FROM untitle_table_4 limit 3",
|
||||||
"status": "pending",
|
"status": "skipped",
|
||||||
"fallback_status": "pending"
|
"fallback_status": "skipped"
|
||||||
}]
|
}]
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -842,7 +841,7 @@ describe('Batch API fallback job', function () {
|
|||||||
"query": "SELECT * FROM nonexistent_table /* should fail */",
|
"query": "SELECT * FROM nonexistent_table /* should fail */",
|
||||||
"onsuccess": "SELECT * FROM untitle_table_4 limit 3",
|
"onsuccess": "SELECT * FROM untitle_table_4 limit 3",
|
||||||
"status": "failed",
|
"status": "failed",
|
||||||
"fallback_status": "pending",
|
"fallback_status": "skipped",
|
||||||
"failed_reason": 'relation "nonexistent_table" does not exist'
|
"failed_reason": 'relation "nonexistent_table" does not exist'
|
||||||
}]
|
}]
|
||||||
};
|
};
|
||||||
@ -875,7 +874,7 @@ describe('Batch API fallback job', function () {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('"onerror" should not be triggered for any query', function () {
|
describe('"onerror" should not be triggered for any query and "skipped"', function () {
|
||||||
var fallbackJob = {};
|
var fallbackJob = {};
|
||||||
|
|
||||||
it('should create a job', function (done) {
|
it('should create a job', function (done) {
|
||||||
@ -914,12 +913,12 @@ describe('Batch API fallback job', function () {
|
|||||||
query: 'SELECT * FROM untitle_table_4 limit 1',
|
query: 'SELECT * FROM untitle_table_4 limit 1',
|
||||||
onerror: 'SELECT * FROM untitle_table_4 limit 2',
|
onerror: 'SELECT * FROM untitle_table_4 limit 2',
|
||||||
status: 'done',
|
status: 'done',
|
||||||
fallback_status: 'pending'
|
fallback_status: 'skipped'
|
||||||
}, {
|
}, {
|
||||||
query: 'SELECT * FROM untitle_table_4 limit 3',
|
query: 'SELECT * FROM untitle_table_4 limit 3',
|
||||||
onerror: 'SELECT * FROM untitle_table_4 limit 4',
|
onerror: 'SELECT * FROM untitle_table_4 limit 4',
|
||||||
status: 'done',
|
status: 'done',
|
||||||
fallback_status: 'pending'
|
fallback_status: 'skipped'
|
||||||
}]
|
}]
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -943,6 +942,144 @@ describe('Batch API fallback job', function () {
|
|||||||
assert.deepEqual(job.query, expectedQuery);
|
assert.deepEqual(job.query, expectedQuery);
|
||||||
done();
|
done();
|
||||||
} else if (job.status === jobStatus.FAILED || job.status === jobStatus.CANCELLED) {
|
} else if (job.status === jobStatus.FAILED || job.status === jobStatus.CANCELLED) {
|
||||||
|
clearInterval(interval);
|
||||||
|
done(new Error('Job ' + job.job_id + ' is ' + job.status + ', expected to be done'));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}, 50);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('"onsuccess" should be "skipped"', function () {
|
||||||
|
var fallbackJob = {};
|
||||||
|
|
||||||
|
it('should create a job', function (done) {
|
||||||
|
assert.response(app, {
|
||||||
|
url: '/api/v2/sql/job?api_key=1234',
|
||||||
|
headers: {
|
||||||
|
'Content-Type': 'application/x-www-form-urlencoded',
|
||||||
|
'host': 'vizzuality.cartodb.com'
|
||||||
|
},
|
||||||
|
method: 'POST',
|
||||||
|
data: querystring.stringify({
|
||||||
|
query: {
|
||||||
|
query: [{
|
||||||
|
query: "SELECT * FROM untitle_table_4 limit 1, /* should fail */",
|
||||||
|
onsuccess: "SELECT * FROM untitle_table_4 limit 2"
|
||||||
|
}]
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}, {
|
||||||
|
status: 201
|
||||||
|
}, function (res, err) {
|
||||||
|
if (err) {
|
||||||
|
return done(err);
|
||||||
|
}
|
||||||
|
fallbackJob = JSON.parse(res.body);
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('job should be failed', function (done) {
|
||||||
|
var expectedQuery = {
|
||||||
|
query: [{
|
||||||
|
query: 'SELECT * FROM untitle_table_4 limit 1, /* should fail */',
|
||||||
|
onsuccess: 'SELECT * FROM untitle_table_4 limit 2',
|
||||||
|
status: 'failed',
|
||||||
|
fallback_status: 'skipped',
|
||||||
|
failed_reason: 'syntax error at end of input'
|
||||||
|
}]
|
||||||
|
};
|
||||||
|
|
||||||
|
var interval = setInterval(function () {
|
||||||
|
assert.response(app, {
|
||||||
|
url: '/api/v2/sql/job/' + fallbackJob.job_id + '?api_key=1234&',
|
||||||
|
headers: {
|
||||||
|
'Content-Type': 'application/x-www-form-urlencoded',
|
||||||
|
'host': 'vizzuality.cartodb.com'
|
||||||
|
},
|
||||||
|
method: 'GET'
|
||||||
|
}, {
|
||||||
|
status: 200
|
||||||
|
}, function (res, err) {
|
||||||
|
if (err) {
|
||||||
|
return done(err);
|
||||||
|
}
|
||||||
|
var job = JSON.parse(res.body);
|
||||||
|
if (job.status === jobStatus.FAILED) {
|
||||||
|
clearInterval(interval);
|
||||||
|
assert.deepEqual(job.query, expectedQuery);
|
||||||
|
done();
|
||||||
|
} else if (job.status === jobStatus.DONE || job.status === jobStatus.CANCELLED) {
|
||||||
|
clearInterval(interval);
|
||||||
|
done(new Error('Job ' + job.job_id + ' is ' + job.status + ', expected to be failed'));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}, 50);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
|
||||||
|
describe('"onsuccess" should not be triggered and "skipped"', function () {
|
||||||
|
var fallbackJob = {};
|
||||||
|
|
||||||
|
it('should create a job', function (done) {
|
||||||
|
assert.response(app, {
|
||||||
|
url: '/api/v2/sql/job?api_key=1234',
|
||||||
|
headers: {
|
||||||
|
'Content-Type': 'application/x-www-form-urlencoded',
|
||||||
|
'host': 'vizzuality.cartodb.com'
|
||||||
|
},
|
||||||
|
method: 'POST',
|
||||||
|
data: querystring.stringify({
|
||||||
|
query: {
|
||||||
|
query: [{
|
||||||
|
query: "SELECT * FROM untitle_table_4 limit 1, /* should fail */",
|
||||||
|
}],
|
||||||
|
onsuccess: "SELECT * FROM untitle_table_4 limit 2"
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}, {
|
||||||
|
status: 201
|
||||||
|
}, function (res, err) {
|
||||||
|
if (err) {
|
||||||
|
return done(err);
|
||||||
|
}
|
||||||
|
fallbackJob = JSON.parse(res.body);
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('job should be failed', function (done) {
|
||||||
|
var expectedQuery = {
|
||||||
|
query: [{
|
||||||
|
query: 'SELECT * FROM untitle_table_4 limit 1, /* should fail */',
|
||||||
|
status: 'failed',
|
||||||
|
failed_reason: 'syntax error at end of input'
|
||||||
|
}],
|
||||||
|
onsuccess: 'SELECT * FROM untitle_table_4 limit 2'
|
||||||
|
};
|
||||||
|
|
||||||
|
var interval = setInterval(function () {
|
||||||
|
assert.response(app, {
|
||||||
|
url: '/api/v2/sql/job/' + fallbackJob.job_id + '?api_key=1234&',
|
||||||
|
headers: {
|
||||||
|
'Content-Type': 'application/x-www-form-urlencoded',
|
||||||
|
'host': 'vizzuality.cartodb.com'
|
||||||
|
},
|
||||||
|
method: 'GET'
|
||||||
|
}, {
|
||||||
|
status: 200
|
||||||
|
}, function (res, err) {
|
||||||
|
if (err) {
|
||||||
|
return done(err);
|
||||||
|
}
|
||||||
|
var job = JSON.parse(res.body);
|
||||||
|
if (job.status === jobStatus.FAILED && job.fallback_status === jobStatus.SKIPPED) {
|
||||||
|
clearInterval(interval);
|
||||||
|
assert.deepEqual(job.query, expectedQuery);
|
||||||
|
done();
|
||||||
|
} else if (job.status === jobStatus.DONE || job.status === jobStatus.CANCELLED) {
|
||||||
clearInterval(interval);
|
clearInterval(interval);
|
||||||
done(new Error('Job ' + job.job_id + ' is ' + job.status + ', expected to be failed'));
|
done(new Error('Job ' + job.job_id + ' is ' + job.status + ', expected to be failed'));
|
||||||
}
|
}
|
||||||
@ -1329,7 +1466,7 @@ describe('Batch API fallback job', function () {
|
|||||||
job.status === jobStatus.FAILED ||
|
job.status === jobStatus.FAILED ||
|
||||||
job.status === jobStatus.CANCELLED) {
|
job.status === jobStatus.CANCELLED) {
|
||||||
clearInterval(interval);
|
clearInterval(interval);
|
||||||
done(new Error('Job ' + job.job_id + ' is ' + job.status + ', expected to be done'));
|
done(new Error('Job ' + job.job_id + ' is ' + job.status + ', expected to be running'));
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}, 50);
|
}, 50);
|
||||||
@ -1341,7 +1478,7 @@ describe('Batch API fallback job', function () {
|
|||||||
"query": "SELECT pg_sleep(3)",
|
"query": "SELECT pg_sleep(3)",
|
||||||
"onsuccess": "SELECT pg_sleep(0)",
|
"onsuccess": "SELECT pg_sleep(0)",
|
||||||
"status": "cancelled",
|
"status": "cancelled",
|
||||||
"fallback_status": "pending"
|
"fallback_status": "skipped"
|
||||||
}],
|
}],
|
||||||
"onsuccess": "SELECT pg_sleep(0)"
|
"onsuccess": "SELECT pg_sleep(0)"
|
||||||
};
|
};
|
||||||
@ -1360,7 +1497,7 @@ describe('Batch API fallback job', function () {
|
|||||||
return done(err);
|
return done(err);
|
||||||
}
|
}
|
||||||
var job = JSON.parse(res.body);
|
var job = JSON.parse(res.body);
|
||||||
if (job.status === jobStatus.CANCELLED && job.fallback_status === jobStatus.PENDING) {
|
if (job.status === jobStatus.CANCELLED && job.fallback_status === jobStatus.SKIPPED) {
|
||||||
assert.deepEqual(job.query, expectedQuery);
|
assert.deepEqual(job.query, expectedQuery);
|
||||||
done();
|
done();
|
||||||
} else if (job.status === jobStatus.DONE || job.status === jobStatus.FAILED) {
|
} else if (job.status === jobStatus.DONE || job.status === jobStatus.FAILED) {
|
||||||
@ -1469,7 +1606,7 @@ describe('Batch API fallback job', function () {
|
|||||||
return done(err);
|
return done(err);
|
||||||
}
|
}
|
||||||
var job = JSON.parse(res.body);
|
var job = JSON.parse(res.body);
|
||||||
if (job.status === jobStatus.CANCELLED && job.fallback_status === jobStatus.PENDING) {
|
if (job.status === jobStatus.CANCELLED && job.fallback_status === jobStatus.SKIPPED) {
|
||||||
assert.deepEqual(job.query, expectedQuery);
|
assert.deepEqual(job.query, expectedQuery);
|
||||||
done();
|
done();
|
||||||
} else if (job.status === jobStatus.DONE || job.status === jobStatus.FAILED) {
|
} else if (job.status === jobStatus.DONE || job.status === jobStatus.FAILED) {
|
||||||
@ -1478,4 +1615,166 @@ describe('Batch API fallback job', function () {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
describe('should run first "onerror" and job "onerror" and skip the other ones', function () {
|
||||||
|
var fallbackJob = {};
|
||||||
|
|
||||||
|
it('should create a job', function (done) {
|
||||||
|
assert.response(app, {
|
||||||
|
url: '/api/v2/sql/job?api_key=1234',
|
||||||
|
headers: {
|
||||||
|
'Content-Type': 'application/x-www-form-urlencoded',
|
||||||
|
'host': 'vizzuality.cartodb.com'
|
||||||
|
},
|
||||||
|
method: 'POST',
|
||||||
|
data: querystring.stringify({
|
||||||
|
"query": {
|
||||||
|
"query": [{
|
||||||
|
"query": "SELECT * FROM untitle_table_4 limit 1, should fail",
|
||||||
|
"onerror": "SELECT * FROM untitle_table_4 limit 2"
|
||||||
|
}, {
|
||||||
|
"query": "SELECT * FROM untitle_table_4 limit 3",
|
||||||
|
"onerror": "SELECT * FROM untitle_table_4 limit 4"
|
||||||
|
}],
|
||||||
|
"onerror": "SELECT * FROM untitle_table_4 limit 5"
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}, {
|
||||||
|
status: 201
|
||||||
|
}, function (res, err) {
|
||||||
|
if (err) {
|
||||||
|
return done(err);
|
||||||
|
}
|
||||||
|
fallbackJob = JSON.parse(res.body);
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('job should fail', function (done) {
|
||||||
|
var expectedQuery = {
|
||||||
|
"query": [
|
||||||
|
{
|
||||||
|
"query": "SELECT * FROM untitle_table_4 limit 1, should fail",
|
||||||
|
"onerror": "SELECT * FROM untitle_table_4 limit 2",
|
||||||
|
"status": "failed",
|
||||||
|
"fallback_status": "done",
|
||||||
|
"failed_reason": "LIMIT #,# syntax is not supported"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"query": "SELECT * FROM untitle_table_4 limit 3",
|
||||||
|
"onerror": "SELECT * FROM untitle_table_4 limit 4",
|
||||||
|
"status": "skipped",
|
||||||
|
"fallback_status": "skipped"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"onerror": "SELECT * FROM untitle_table_4 limit 5"
|
||||||
|
};
|
||||||
|
|
||||||
|
var interval = setInterval(function () {
|
||||||
|
assert.response(app, {
|
||||||
|
url: '/api/v2/sql/job/' + fallbackJob.job_id + '?api_key=1234&',
|
||||||
|
headers: {
|
||||||
|
'Content-Type': 'application/x-www-form-urlencoded',
|
||||||
|
'host': 'vizzuality.cartodb.com'
|
||||||
|
},
|
||||||
|
method: 'GET'
|
||||||
|
}, {
|
||||||
|
status: 200
|
||||||
|
}, function (res, err) {
|
||||||
|
if (err) {
|
||||||
|
return done(err);
|
||||||
|
}
|
||||||
|
var job = JSON.parse(res.body);
|
||||||
|
if (job.status === jobStatus.FAILED && job.fallback_status === jobStatus.DONE) {
|
||||||
|
clearInterval(interval);
|
||||||
|
assert.deepEqual(job.query, expectedQuery);
|
||||||
|
done();
|
||||||
|
} else if (job.status === jobStatus.DONE || job.status === jobStatus.CANCELLED) {
|
||||||
|
clearInterval(interval);
|
||||||
|
done(new Error('Job ' + job.job_id + ' is ' + job.status + ', expected to be failed'));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}, 50);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
|
||||||
|
describe('should fail first "onerror" and job "onerror" and skip the other ones', function () {
|
||||||
|
var fallbackJob = {};
|
||||||
|
|
||||||
|
it('should create a job', function (done) {
|
||||||
|
assert.response(app, {
|
||||||
|
url: '/api/v2/sql/job?api_key=1234',
|
||||||
|
headers: {
|
||||||
|
'Content-Type': 'application/x-www-form-urlencoded',
|
||||||
|
'host': 'vizzuality.cartodb.com'
|
||||||
|
},
|
||||||
|
method: 'POST',
|
||||||
|
data: querystring.stringify({
|
||||||
|
"query": {
|
||||||
|
"query": [{
|
||||||
|
"query": "SELECT * FROM atm_madrid limit 1, should fail",
|
||||||
|
"onerror": "SELECT * FROM atm_madrid limit 2"
|
||||||
|
}, {
|
||||||
|
"query": "SELECT * FROM atm_madrid limit 3",
|
||||||
|
"onerror": "SELECT * FROM atm_madrid limit 4"
|
||||||
|
}],
|
||||||
|
"onerror": "SELECT * FROM atm_madrid limit 5"
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}, {
|
||||||
|
status: 201
|
||||||
|
}, function (res, err) {
|
||||||
|
if (err) {
|
||||||
|
return done(err);
|
||||||
|
}
|
||||||
|
fallbackJob = JSON.parse(res.body);
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('job should fail', function (done) {
|
||||||
|
var expectedQuery = {
|
||||||
|
query: [{
|
||||||
|
query: 'SELECT * FROM atm_madrid limit 1, should fail',
|
||||||
|
onerror: 'SELECT * FROM atm_madrid limit 2',
|
||||||
|
status: 'failed',
|
||||||
|
fallback_status: 'failed',
|
||||||
|
failed_reason: 'relation "atm_madrid" does not exist'
|
||||||
|
}, {
|
||||||
|
query: 'SELECT * FROM atm_madrid limit 3',
|
||||||
|
onerror: 'SELECT * FROM atm_madrid limit 4',
|
||||||
|
status: 'skipped',
|
||||||
|
fallback_status: 'skipped'
|
||||||
|
}],
|
||||||
|
onerror: 'SELECT * FROM atm_madrid limit 5'
|
||||||
|
};
|
||||||
|
|
||||||
|
var interval = setInterval(function () {
|
||||||
|
assert.response(app, {
|
||||||
|
url: '/api/v2/sql/job/' + fallbackJob.job_id + '?api_key=1234&',
|
||||||
|
headers: {
|
||||||
|
'Content-Type': 'application/x-www-form-urlencoded',
|
||||||
|
'host': 'vizzuality.cartodb.com'
|
||||||
|
},
|
||||||
|
method: 'GET'
|
||||||
|
}, {
|
||||||
|
status: 200
|
||||||
|
}, function (res, err) {
|
||||||
|
if (err) {
|
||||||
|
return done(err);
|
||||||
|
}
|
||||||
|
var job = JSON.parse(res.body);
|
||||||
|
if (job.status === jobStatus.FAILED && job.fallback_status === jobStatus.FAILED) {
|
||||||
|
clearInterval(interval);
|
||||||
|
assert.deepEqual(job.query, expectedQuery);
|
||||||
|
done();
|
||||||
|
} else if (job.status === jobStatus.DONE || job.status === jobStatus.CANCELLED) {
|
||||||
|
clearInterval(interval);
|
||||||
|
done(new Error('Job ' + job.job_id + ' is ' + job.status + ', expected to be failed'));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}, 50);
|
||||||
|
});
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
Loading…
Reference in New Issue
Block a user