]> git.immae.eu Git - github/Chocobozzz/PeerTube.git/blame - server/lib/jobs/job-scheduler.ts
Server: upgrade packages
[github/Chocobozzz/PeerTube.git] / server / lib / jobs / job-scheduler.ts
CommitLineData
bcd1c9e1 1import { AsyncQueue, forever, queue } from 'async'
69818c93 2import * as Sequelize from 'sequelize'
65fcc311 3
e02643f3 4import { database as db } from '../../initializers/database'
65fcc311
C
5import {
6 JOBS_FETCHING_INTERVAL,
7 JOBS_FETCH_LIMIT_PER_CYCLE,
8 JOB_STATES
9} from '../../initializers'
10import { logger } from '../../helpers'
69818c93
C
11import { JobInstance } from '../../models'
12import { JobHandler, jobHandlers } from './handlers'
13
14type JobQueueCallback = (err: Error) => void
65fcc311
C
15
16class JobScheduler {
17
18 private static instance: JobScheduler
19
20 private constructor () { }
21
22 static get Instance () {
23 return this.instance || (this.instance = new this())
24 }
25
26 activate () {
27 const limit = JOBS_FETCH_LIMIT_PER_CYCLE
28
29 logger.info('Jobs scheduler activated.')
30
69818c93 31 const jobsQueue = queue<JobInstance, JobQueueCallback>(this.processJob.bind(this))
65fcc311
C
32
33 // Finish processing jobs from a previous start
34 const state = JOB_STATES.PROCESSING
6fcd19ba
C
35 db.Job.listWithLimit(limit, state)
36 .then(jobs => {
37 this.enqueueJobs(jobsQueue, jobs)
38
39 forever(
40 next => {
41 if (jobsQueue.length() !== 0) {
42 // Finish processing the queue first
43 return setTimeout(next, JOBS_FETCHING_INTERVAL)
65fcc311
C
44 }
45
6fcd19ba
C
46 const state = JOB_STATES.PENDING
47 db.Job.listWithLimit(limit, state)
48 .then(jobs => {
49 this.enqueueJobs(jobsQueue, jobs)
65fcc311 50
6fcd19ba
C
51 // Optimization: we could use "drain" from queue object
52 return setTimeout(next, JOBS_FETCHING_INTERVAL)
53 })
ad0997ad 54 .catch(err => logger.error('Cannot list pending jobs.', err))
6fcd19ba
C
55 },
56
ad0997ad 57 err => logger.error('Error in job scheduler queue.', err)
6fcd19ba
C
58 )
59 })
ad0997ad 60 .catch(err => logger.error('Cannot list pending jobs.', err))
65fcc311
C
61 }
62
6fcd19ba 63 createJob (transaction: Sequelize.Transaction, handlerName: string, handlerInputData: object) {
65fcc311
C
64 const createQuery = {
65 state: JOB_STATES.PENDING,
66 handlerName,
67 handlerInputData
68 }
69 const options = { transaction }
70
6fcd19ba 71 return db.Job.create(createQuery, options)
65fcc311
C
72 }
73
6fcd19ba
C
74 private enqueueJobs (jobsQueue: AsyncQueue<JobInstance>, jobs: JobInstance[]) {
75 jobs.forEach(job => jobsQueue.push(job))
65fcc311
C
76 }
77
69818c93 78 private processJob (job: JobInstance, callback: (err: Error) => void) {
65fcc311 79 const jobHandler = jobHandlers[job.handlerName]
6fcd19ba
C
80 if (jobHandler === undefined) {
81 logger.error('Unknown job handler for job %s.', job.handlerName)
82 return callback(null)
83 }
65fcc311
C
84
85 logger.info('Processing job %d with handler %s.', job.id, job.handlerName)
86
87 job.state = JOB_STATES.PROCESSING
6fcd19ba
C
88 return job.save()
89 .then(() => {
90 return jobHandler.process(job.handlerInputData)
91 })
92 .then(
93 result => {
94 return this.onJobSuccess(jobHandler, job, result)
95 },
65fcc311 96
6fcd19ba 97 err => {
ad0997ad 98 logger.error('Error in job handler %s.', job.handlerName, err)
6fcd19ba 99 return this.onJobError(jobHandler, job, err)
65fcc311 100 }
6fcd19ba
C
101 )
102 .then(() => callback(null))
103 .catch(err => {
104 this.cannotSaveJobError(err)
105 return callback(err)
65fcc311 106 })
65fcc311
C
107 }
108
6fcd19ba 109 private onJobError (jobHandler: JobHandler<any>, job: JobInstance, err: Error) {
65fcc311
C
110 job.state = JOB_STATES.ERROR
111
6fcd19ba
C
112 return job.save()
113 .then(() => jobHandler.onError(err, job.id))
114 .catch(err => this.cannotSaveJobError(err))
65fcc311
C
115 }
116
6fcd19ba 117 private onJobSuccess (jobHandler: JobHandler<any>, job: JobInstance, jobResult: any) {
65fcc311
C
118 job.state = JOB_STATES.SUCCESS
119
6fcd19ba
C
120 return job.save()
121 .then(() => jobHandler.onSuccess(job.id, jobResult))
122 .catch(err => this.cannotSaveJobError(err))
65fcc311
C
123 }
124
6fcd19ba 125 private cannotSaveJobError (err: Error) {
ad0997ad 126 logger.error('Cannot save new job state.', err)
65fcc311
C
127 }
128}
129
130// ---------------------------------------------------------------------------
131
132export {
133 JobScheduler
134}