1 import { AsyncQueue, forever, queue } from 'async'
2 import * as Sequelize from 'sequelize'
3 import { JobCategory } from '../../../shared'
4 import { logger } from '../../helpers'
5 import { database as db, JOB_STATES, JOBS_FETCH_LIMIT_PER_CYCLE, JOBS_FETCHING_INTERVAL } from '../../initializers'
6 import { JobInstance } from '../../models'
7 import { error } from 'util'
9 export interface JobHandler<P, T> {
10 process (data: object, jobId: number): Promise<T>
11 onError (err: Error, jobId: number)
12 onSuccess (jobId: number, jobResult: T, jobScheduler: JobScheduler<P, T>)
14 type JobQueueCallback = (err: Error) => void
16 class JobScheduler<P, T> {
19 private jobCategory: JobCategory,
20 private jobHandlers: { [ id: string ]: JobHandler<P, T> }
24 const limit = JOBS_FETCH_LIMIT_PER_CYCLE[this.jobCategory]
26 logger.info('Jobs scheduler %s activated.', this.jobCategory)
28 const jobsQueue = queue<JobInstance, JobQueueCallback>(this.processJob.bind(this))
30 // Finish processing jobs from a previous start
31 const state = JOB_STATES.PROCESSING
33 const jobs = await db.Job.listWithLimitByCategory(limit, state, this.jobCategory)
35 this.enqueueJobs(jobsQueue, jobs)
37 logger.error('Cannot list pending jobs.', err)
42 if (jobsQueue.length() !== 0) {
43 // Finish processing the queue first
44 return setTimeout(next, JOBS_FETCHING_INTERVAL)
47 const state = JOB_STATES.PENDING
49 const jobs = await db.Job.listWithLimitByCategory(limit, state, this.jobCategory)
51 this.enqueueJobs(jobsQueue, jobs)
53 logger.error('Cannot list pending jobs.', err)
56 // Optimization: we could use "drain" from queue object
57 return setTimeout(next, JOBS_FETCHING_INTERVAL)
60 err => logger.error('Error in job scheduler queue.', err)
64 createJob (transaction: Sequelize.Transaction, handlerName: string, handlerInputData: P) {
66 state: JOB_STATES.PENDING,
67 category: this.jobCategory,
72 const options = { transaction }
74 return db.Job.create(createQuery, options)
77 private enqueueJobs (jobsQueue: AsyncQueue<JobInstance>, jobs: JobInstance[]) {
78 jobs.forEach(job => jobsQueue.push(job))
81 private async processJob (job: JobInstance, callback: (err: Error) => void) {
82 const jobHandler = this.jobHandlers[job.handlerName]
83 if (jobHandler === undefined) {
84 const errorString = 'Unknown job handler ' + job.handlerName + ' for job ' + job.id
85 logger.error(errorString)
87 const error = new Error(errorString)
88 await this.onJobError(jobHandler, job, error)
89 return callback(error)
92 logger.info('Processing job %d with handler %s.', job.id, job.handlerName)
94 job.state = JOB_STATES.PROCESSING
98 const result: T = await jobHandler.process(job.handlerInputData, job.id)
99 await this.onJobSuccess(jobHandler, job, result)
101 logger.error('Error in job handler %s.', job.handlerName, err)
104 await this.onJobError(jobHandler, job, err)
106 this.cannotSaveJobError(innerErr)
107 return callback(innerErr)
111 return callback(null)
114 private async onJobError (jobHandler: JobHandler<P, T>, job: JobInstance, err: Error) {
115 job.state = JOB_STATES.ERROR
119 if (jobHandler) await jobHandler.onError(err, job.id)
121 this.cannotSaveJobError(err)
125 private async onJobSuccess (jobHandler: JobHandler<P, T>, job: JobInstance, jobResult: T) {
126 job.state = JOB_STATES.SUCCESS
130 jobHandler.onSuccess(job.id, jobResult, this)
132 this.cannotSaveJobError(err)
136 private cannotSaveJobError (err: Error) {
137 logger.error('Cannot save new job state.', err)
141 // ---------------------------------------------------------------------------