2017-05-15 15:22:03 -05:00
|
|
|
import { forever, queue } from 'async'
|
2017-06-10 15:15:25 -05:00
|
|
|
import * as Sequelize from 'sequelize'
|
2017-05-15 15:22:03 -05:00
|
|
|
|
2017-05-22 13:58:25 -05:00
|
|
|
import { database as db } from '../../initializers/database'
|
2017-05-15 15:22:03 -05:00
|
|
|
import {
|
|
|
|
JOBS_FETCHING_INTERVAL,
|
|
|
|
JOBS_FETCH_LIMIT_PER_CYCLE,
|
|
|
|
JOB_STATES
|
|
|
|
} from '../../initializers'
|
|
|
|
import { logger } from '../../helpers'
|
2017-06-10 15:15:25 -05:00
|
|
|
import { JobInstance } from '../../models'
|
|
|
|
import { JobHandler, jobHandlers } from './handlers'
|
|
|
|
|
|
|
|
type JobQueueCallback = (err: Error) => void
|
2017-05-15 15:22:03 -05:00
|
|
|
|
|
|
|
class JobScheduler {
|
|
|
|
|
|
|
|
private static instance: JobScheduler
|
|
|
|
|
|
|
|
private constructor () { }
|
|
|
|
|
|
|
|
static get Instance () {
|
|
|
|
return this.instance || (this.instance = new this())
|
|
|
|
}
|
|
|
|
|
|
|
|
activate () {
|
|
|
|
const limit = JOBS_FETCH_LIMIT_PER_CYCLE
|
|
|
|
|
|
|
|
logger.info('Jobs scheduler activated.')
|
|
|
|
|
2017-06-10 15:15:25 -05:00
|
|
|
const jobsQueue = queue<JobInstance, JobQueueCallback>(this.processJob.bind(this))
|
2017-05-15 15:22:03 -05:00
|
|
|
|
|
|
|
// Finish processing jobs from a previous start
|
|
|
|
const state = JOB_STATES.PROCESSING
|
|
|
|
db.Job.listWithLimit(limit, state, (err, jobs) => {
|
|
|
|
this.enqueueJobs(err, jobsQueue, jobs)
|
|
|
|
|
|
|
|
forever(
|
|
|
|
next => {
|
|
|
|
if (jobsQueue.length() !== 0) {
|
|
|
|
// Finish processing the queue first
|
|
|
|
return setTimeout(next, JOBS_FETCHING_INTERVAL)
|
|
|
|
}
|
|
|
|
|
|
|
|
const state = JOB_STATES.PENDING
|
|
|
|
db.Job.listWithLimit(limit, state, (err, jobs) => {
|
|
|
|
if (err) {
|
|
|
|
logger.error('Cannot list pending jobs.', { error: err })
|
|
|
|
} else {
|
|
|
|
jobs.forEach(job => {
|
|
|
|
jobsQueue.push(job)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
// Optimization: we could use "drain" from queue object
|
|
|
|
return setTimeout(next, JOBS_FETCHING_INTERVAL)
|
|
|
|
})
|
|
|
|
},
|
|
|
|
|
|
|
|
err => { logger.error('Error in job scheduler queue.', { error: err }) }
|
|
|
|
)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2017-06-10 15:15:25 -05:00
|
|
|
createJob (transaction: Sequelize.Transaction, handlerName: string, handlerInputData: object, callback: (err: Error) => void) {
|
2017-05-15 15:22:03 -05:00
|
|
|
const createQuery = {
|
|
|
|
state: JOB_STATES.PENDING,
|
|
|
|
handlerName,
|
|
|
|
handlerInputData
|
|
|
|
}
|
|
|
|
const options = { transaction }
|
|
|
|
|
|
|
|
db.Job.create(createQuery, options).asCallback(callback)
|
|
|
|
}
|
|
|
|
|
2017-06-10 15:15:25 -05:00
|
|
|
private enqueueJobs (err: Error, jobsQueue: AsyncQueue<JobInstance>, jobs: JobInstance[]) {
|
2017-05-15 15:22:03 -05:00
|
|
|
if (err) {
|
|
|
|
logger.error('Cannot list pending jobs.', { error: err })
|
|
|
|
} else {
|
|
|
|
jobs.forEach(job => {
|
|
|
|
jobsQueue.push(job)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-06-10 15:15:25 -05:00
|
|
|
private processJob (job: JobInstance, callback: (err: Error) => void) {
|
2017-05-15 15:22:03 -05:00
|
|
|
const jobHandler = jobHandlers[job.handlerName]
|
|
|
|
|
|
|
|
logger.info('Processing job %d with handler %s.', job.id, job.handlerName)
|
|
|
|
|
|
|
|
job.state = JOB_STATES.PROCESSING
|
|
|
|
job.save().asCallback(err => {
|
|
|
|
if (err) return this.cannotSaveJobError(err, callback)
|
|
|
|
|
|
|
|
if (jobHandler === undefined) {
|
2017-06-10 15:15:25 -05:00
|
|
|
logger.error('Unknown job handler for job %s.', job.handlerName)
|
|
|
|
return callback(null)
|
2017-05-15 15:22:03 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
return jobHandler.process(job.handlerInputData, (err, result) => {
|
|
|
|
if (err) {
|
|
|
|
logger.error('Error in job handler %s.', job.handlerName, { error: err })
|
|
|
|
return this.onJobError(jobHandler, job, result, callback)
|
|
|
|
}
|
|
|
|
|
|
|
|
return this.onJobSuccess(jobHandler, job, result, callback)
|
|
|
|
})
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2017-06-10 15:15:25 -05:00
|
|
|
private onJobError (jobHandler: JobHandler<any>, job: JobInstance, jobResult: any, callback: (err: Error) => void) {
|
2017-05-15 15:22:03 -05:00
|
|
|
job.state = JOB_STATES.ERROR
|
|
|
|
|
|
|
|
job.save().asCallback(err => {
|
|
|
|
if (err) return this.cannotSaveJobError(err, callback)
|
|
|
|
|
|
|
|
return jobHandler.onError(err, job.id, jobResult, callback)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2017-06-10 15:15:25 -05:00
|
|
|
private onJobSuccess (jobHandler: JobHandler<any>, job: JobInstance, jobResult: any, callback: (err: Error) => void) {
|
2017-05-15 15:22:03 -05:00
|
|
|
job.state = JOB_STATES.SUCCESS
|
|
|
|
|
|
|
|
job.save().asCallback(err => {
|
|
|
|
if (err) return this.cannotSaveJobError(err, callback)
|
|
|
|
|
|
|
|
return jobHandler.onSuccess(err, job.id, jobResult, callback)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2017-06-10 15:15:25 -05:00
|
|
|
private cannotSaveJobError (err: Error, callback: (err: Error) => void) {
|
2017-05-15 15:22:03 -05:00
|
|
|
logger.error('Cannot save new job state.', { error: err })
|
|
|
|
return callback(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// ---------------------------------------------------------------------------
|
|
|
|
|
|
|
|
export {
|
|
|
|
JobScheduler
|
|
|
|
}
|