1 import { AsyncQueue, forever, queue } from 'async'
2 import * as Sequelize from 'sequelize'
4 import { database as db } from '../../initializers/database'
6 JOBS_FETCHING_INTERVAL,
7 JOBS_FETCH_LIMIT_PER_CYCLE,
9 } from '../../initializers'
10 import { logger } from '../../helpers'
11 import { JobInstance } from '../../models'
12 import { JobHandler, jobHandlers } from './handlers'
14 type JobQueueCallback = (err: Error) => void
18 private static instance: JobScheduler
20 private constructor () { }
22 static get Instance () {
23 return this.instance || (this.instance = new this())
27 const limit = JOBS_FETCH_LIMIT_PER_CYCLE
29 logger.info('Jobs scheduler activated.')
31 const jobsQueue = queue<JobInstance, JobQueueCallback>(this.processJob.bind(this))
33 // Finish processing jobs from a previous start
34 const state = JOB_STATES.PROCESSING
35 db.Job.listWithLimit(limit, state)
37 this.enqueueJobs(jobsQueue, jobs)
41 if (jobsQueue.length() !== 0) {
42 // Finish processing the queue first
43 return setTimeout(next, JOBS_FETCHING_INTERVAL)
46 const state = JOB_STATES.PENDING
47 db.Job.listWithLimit(limit, state)
49 this.enqueueJobs(jobsQueue, jobs)
51 // Optimization: we could use "drain" from queue object
52 return setTimeout(next, JOBS_FETCHING_INTERVAL)
54 .catch(err => logger.error('Cannot list pending jobs.', err))
57 err => logger.error('Error in job scheduler queue.', err)
60 .catch(err => logger.error('Cannot list pending jobs.', err))
63 createJob (transaction: Sequelize.Transaction, handlerName: string, handlerInputData: object) {
65 state: JOB_STATES.PENDING,
69 const options = { transaction }
71 return db.Job.create(createQuery, options)
74 private enqueueJobs (jobsQueue: AsyncQueue<JobInstance>, jobs: JobInstance[]) {
75 jobs.forEach(job => jobsQueue.push(job))
78 private processJob (job: JobInstance, callback: (err: Error) => void) {
79 const jobHandler = jobHandlers[job.handlerName]
80 if (jobHandler === undefined) {
81 logger.error('Unknown job handler for job %s.', job.handlerName)
85 logger.info('Processing job %d with handler %s.', job.id, job.handlerName)
87 job.state = JOB_STATES.PROCESSING
90 return jobHandler.process(job.handlerInputData)
94 return this.onJobSuccess(jobHandler, job, result)
98 logger.error('Error in job handler %s.', job.handlerName, err)
99 return this.onJobError(jobHandler, job, err)
102 .then(() => callback(null))
104 this.cannotSaveJobError(err)
109 private onJobError (jobHandler: JobHandler<any>, job: JobInstance, err: Error) {
110 job.state = JOB_STATES.ERROR
113 .then(() => jobHandler.onError(err, job.id))
114 .catch(err => this.cannotSaveJobError(err))
117 private onJobSuccess (jobHandler: JobHandler<any>, job: JobInstance, jobResult: any) {
118 job.state = JOB_STATES.SUCCESS
121 .then(() => jobHandler.onSuccess(job.id, jobResult))
122 .catch(err => this.cannotSaveJobError(err))
125 private cannotSaveJobError (err: Error) {
126 logger.error('Cannot save new job state.', err)
130 // ---------------------------------------------------------------------------