You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					349 lines
				
				8.7 KiB
			
		
		
			
		
	
	
					349 lines
				
				8.7 KiB
			| 
											3 years ago
										 | 'use strict'; | ||
|  | 
 | ||
|  | const lt = require('long-timeout') | ||
|  | const CronDate = require('cron-parser/lib/date') | ||
|  | const sorted = require('sorted-array-functions') | ||
|  | 
 | ||
|  | const invocations = []; | ||
|  | let currentInvocation = null; | ||
|  | 
 | ||
|  | /* DoesntRecur rule */ | ||
|  | const DoesntRecur = new RecurrenceRule(); | ||
|  | DoesntRecur.recurs = false; | ||
|  | 
 | ||
|  | /* Invocation object */ | ||
|  | function Invocation(job, fireDate, recurrenceRule, endDate) { | ||
|  |   this.job = job; | ||
|  |   this.fireDate = fireDate; | ||
|  |   this.endDate = endDate; | ||
|  |   this.recurrenceRule = recurrenceRule || DoesntRecur; | ||
|  | 
 | ||
|  |   this.timerID = null; | ||
|  | } | ||
|  | 
 | ||
|  | function sorter(a, b) { | ||
|  |   return (a.fireDate.getTime() - b.fireDate.getTime()); | ||
|  | } | ||
|  | 
 | ||
|  | /* Range object */ | ||
|  | function Range(start, end, step) { | ||
|  |   this.start = start || 0; | ||
|  |   this.end = end || 60; | ||
|  |   this.step = step || 1; | ||
|  | } | ||
|  | 
 | ||
|  | Range.prototype.contains = function(val) { | ||
|  |   if (this.step === null || this.step === 1) { | ||
|  |     return (val >= this.start && val <= this.end); | ||
|  |   } else { | ||
|  |     for (let i = this.start; i < this.end; i += this.step) { | ||
|  |       if (i === val) { | ||
|  |         return true; | ||
|  |       } | ||
|  |     } | ||
|  | 
 | ||
|  |     return false; | ||
|  |   } | ||
|  | }; | ||
|  | 
 | ||
|  | /* RecurrenceRule object */ | ||
|  | /* | ||
|  |   Interpreting each property: | ||
|  |   null - any value is valid | ||
|  |   number - fixed value | ||
|  |   Range - value must fall in range | ||
|  |   array - value must validate against any item in list | ||
|  | 
 | ||
|  |   NOTE: Cron months are 1-based, but RecurrenceRule months are 0-based. | ||
|  | */ | ||
|  | function RecurrenceRule(year, month, date, dayOfWeek, hour, minute, second) { | ||
|  |   this.recurs = true; | ||
|  | 
 | ||
|  |   this.year = (year == null) ? null : year; | ||
|  |   this.month = (month == null) ? null : month; | ||
|  |   this.date = (date == null) ? null : date; | ||
|  |   this.dayOfWeek = (dayOfWeek == null) ? null : dayOfWeek; | ||
|  |   this.hour = (hour == null) ? null : hour; | ||
|  |   this.minute = (minute == null) ? null : minute; | ||
|  |   this.second = (second == null) ? 0 : second; | ||
|  | } | ||
|  | 
 | ||
|  | RecurrenceRule.prototype.isValid = function() { | ||
|  |   function isValidType(num) { | ||
|  |     if (Array.isArray(num) || (num instanceof Array)) { | ||
|  |       return num.every(function(e) { | ||
|  |         return isValidType(e); | ||
|  |       }); | ||
|  |     } | ||
|  |     return !(Number.isNaN(Number(num)) && !(num instanceof Range)); | ||
|  |   } | ||
|  |   if (this.month !== null && (this.month < 0 || this.month > 11 || !isValidType(this.month))) { | ||
|  |     return false; | ||
|  |   } | ||
|  |   if (this.dayOfWeek !== null && (this.dayOfWeek < 0 || this.dayOfWeek > 6 || !isValidType(this.dayOfWeek))) { | ||
|  |     return false; | ||
|  |   } | ||
|  |   if (this.hour !== null && (this.hour < 0 || this.hour > 23 || !isValidType(this.hour))) { | ||
|  |     return false; | ||
|  |   } | ||
|  |   if (this.minute !== null && (this.minute < 0 || this.minute > 59 || !isValidType(this.minute))) { | ||
|  |     return false; | ||
|  |   } | ||
|  |   if (this.second !== null && (this.second < 0 || this.second > 59 || !isValidType(this.second))) { | ||
|  |     return false; | ||
|  |   } | ||
|  |   if (this.date !== null) { | ||
|  |     if(!isValidType(this.date)) { | ||
|  |       return false; | ||
|  |     } | ||
|  |     switch (this.month) { | ||
|  |       case 3: | ||
|  |       case 5: | ||
|  |       case 8: | ||
|  |       case 10: | ||
|  |         if (this.date < 1 || this. date > 30) { | ||
|  |           return false; | ||
|  |         } | ||
|  |         break; | ||
|  |       case 1: | ||
|  |         if (this.date < 1 || this. date > 29) { | ||
|  |           return false; | ||
|  |         } | ||
|  |         break; | ||
|  |       default: | ||
|  |         if (this.date < 1 || this. date > 31) { | ||
|  |           return false; | ||
|  |         } | ||
|  |     } | ||
|  |   } | ||
|  |   return true; | ||
|  | }; | ||
|  | 
 | ||
|  | RecurrenceRule.prototype.nextInvocationDate = function(base) { | ||
|  |   const next = this._nextInvocationDate(base); | ||
|  |   return next ? next.toDate() : null; | ||
|  | }; | ||
|  | 
 | ||
|  | RecurrenceRule.prototype._nextInvocationDate = function(base) { | ||
|  |   base = ((base instanceof CronDate) || (base instanceof Date)) ? base : (new Date()); | ||
|  |   if (!this.recurs) { | ||
|  |     return null; | ||
|  |   } | ||
|  | 
 | ||
|  |   if(!this.isValid()) { | ||
|  |     return null; | ||
|  |   } | ||
|  | 
 | ||
|  |   const now = new CronDate(Date.now(), this.tz); | ||
|  |   let fullYear = now.getFullYear(); | ||
|  |   if ((this.year !== null) && | ||
|  |     (typeof this.year == 'number') && | ||
|  |     (this.year < fullYear)) { | ||
|  |     return null; | ||
|  |   } | ||
|  | 
 | ||
|  |   let next = new CronDate(base.getTime(), this.tz); | ||
|  |   next.addSecond(); | ||
|  | 
 | ||
|  |   while (true) { | ||
|  |     if (this.year !== null) { | ||
|  |       fullYear = next.getFullYear(); | ||
|  |       if ((typeof this.year == 'number') && (this.year < fullYear)) { | ||
|  |         next = null; | ||
|  |         break; | ||
|  |       } | ||
|  | 
 | ||
|  |       if (!recurMatch(fullYear, this.year)) { | ||
|  |         next.addYear(); | ||
|  |         next.setMonth(0); | ||
|  |         next.setDate(1); | ||
|  |         next.setHours(0); | ||
|  |         next.setMinutes(0); | ||
|  |         next.setSeconds(0); | ||
|  |         continue; | ||
|  |       } | ||
|  |     } | ||
|  |     if (this.month != null && !recurMatch(next.getMonth(), this.month)) { | ||
|  |       next.addMonth(); | ||
|  |       continue; | ||
|  |     } | ||
|  |     if (this.date != null && !recurMatch(next.getDate(), this.date)) { | ||
|  |       next.addDay(); | ||
|  |       continue; | ||
|  |     } | ||
|  |     if (this.dayOfWeek != null && !recurMatch(next.getDay(), this.dayOfWeek)) { | ||
|  |       next.addDay(); | ||
|  |       continue; | ||
|  |     } | ||
|  |     if (this.hour != null && !recurMatch(next.getHours(), this.hour)) { | ||
|  |       next.addHour(); | ||
|  |       continue; | ||
|  |     } | ||
|  |     if (this.minute != null && !recurMatch(next.getMinutes(), this.minute)) { | ||
|  |       next.addMinute(); | ||
|  |       continue; | ||
|  |     } | ||
|  |     if (this.second != null && !recurMatch(next.getSeconds(), this.second)) { | ||
|  |       next.addSecond(); | ||
|  |       continue; | ||
|  |     } | ||
|  | 
 | ||
|  |     break; | ||
|  |   } | ||
|  | 
 | ||
|  |   return next; | ||
|  | }; | ||
|  | 
 | ||
|  | function recurMatch(val, matcher) { | ||
|  |   if (matcher == null) { | ||
|  |     return true; | ||
|  |   } | ||
|  | 
 | ||
|  |   if (typeof matcher === 'number') { | ||
|  |     return (val === matcher); | ||
|  |   } else if(typeof matcher === 'string') { | ||
|  |     return (val === Number(matcher)); | ||
|  |   } else if (matcher instanceof Range) { | ||
|  |     return matcher.contains(val); | ||
|  |   } else if (Array.isArray(matcher) || (matcher instanceof Array)) { | ||
|  |     for (let i = 0; i < matcher.length; i++) { | ||
|  |       if (recurMatch(val, matcher[i])) { | ||
|  |         return true; | ||
|  |       } | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  |   return false; | ||
|  | } | ||
|  | 
 | ||
|  | /* Date-based scheduler */ | ||
|  | function runOnDate(date, job) { | ||
|  |   const now = Date.now(); | ||
|  |   const then = date.getTime(); | ||
|  | 
 | ||
|  |   return lt.setTimeout(function() { | ||
|  |     if (then > Date.now()) | ||
|  |       runOnDate(date, job); | ||
|  |     else | ||
|  |       job(); | ||
|  |   }, (then < now ? 0 : then - now)); | ||
|  | } | ||
|  | 
 | ||
|  | function scheduleInvocation(invocation) { | ||
|  |   sorted.add(invocations, invocation, sorter); | ||
|  |   prepareNextInvocation(); | ||
|  |   const date = invocation.fireDate instanceof CronDate ? invocation.fireDate.toDate() : invocation.fireDate; | ||
|  |   invocation.job.emit('scheduled', date); | ||
|  | } | ||
|  | 
 | ||
|  | function prepareNextInvocation() { | ||
|  |   if (invocations.length > 0 && currentInvocation !== invocations[0]) { | ||
|  |     if (currentInvocation !== null) { | ||
|  |       lt.clearTimeout(currentInvocation.timerID); | ||
|  |       currentInvocation.timerID = null; | ||
|  |       currentInvocation = null; | ||
|  |     } | ||
|  | 
 | ||
|  |     currentInvocation = invocations[0]; | ||
|  | 
 | ||
|  |     const job = currentInvocation.job; | ||
|  |     const cinv = currentInvocation; | ||
|  |     currentInvocation.timerID = runOnDate(currentInvocation.fireDate, function() { | ||
|  |       currentInvocationFinished(); | ||
|  | 
 | ||
|  |       if (job.callback) { | ||
|  |         job.callback(); | ||
|  |       } | ||
|  | 
 | ||
|  |       if (cinv.recurrenceRule.recurs || cinv.recurrenceRule._endDate === null) { | ||
|  |         const inv = scheduleNextRecurrence(cinv.recurrenceRule, cinv.job, cinv.fireDate, cinv.endDate); | ||
|  |         if (inv !== null) { | ||
|  |           inv.job.trackInvocation(inv); | ||
|  |         } | ||
|  |       } | ||
|  | 
 | ||
|  |       job.stopTrackingInvocation(cinv); | ||
|  | 
 | ||
|  |       try { | ||
|  |         const result = job.invoke(cinv.fireDate instanceof CronDate ? cinv.fireDate.toDate() : cinv.fireDate); | ||
|  |         job.emit('run'); | ||
|  |         job.running += 1; | ||
|  | 
 | ||
|  |         if (result instanceof Promise) { | ||
|  |           result.then(function (value) { | ||
|  |             job.emit('success', value); | ||
|  |             job.running -= 1; | ||
|  |           }).catch(function (err) { | ||
|  |             job.emit('error', err); | ||
|  |             job.running -= 1; | ||
|  |           }); | ||
|  |         } else { | ||
|  |           job.emit('success', result); | ||
|  |           job.running -= 1; | ||
|  |         } | ||
|  |       } catch (err) { | ||
|  |         job.emit('error', err); | ||
|  |         job.running -= 1; | ||
|  |       } | ||
|  | 
 | ||
|  |       if (job.isOneTimeJob) { | ||
|  |         job.deleteFromSchedule(); | ||
|  |       } | ||
|  |     }); | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | function currentInvocationFinished() { | ||
|  |   invocations.shift(); | ||
|  |   currentInvocation = null; | ||
|  |   prepareNextInvocation(); | ||
|  | } | ||
|  | 
 | ||
|  | function cancelInvocation(invocation) { | ||
|  |   const idx = invocations.indexOf(invocation); | ||
|  |   if (idx > -1) { | ||
|  |     invocations.splice(idx, 1); | ||
|  |     if (invocation.timerID !== null) { | ||
|  |       lt.clearTimeout(invocation.timerID); | ||
|  |     } | ||
|  | 
 | ||
|  |     if (currentInvocation === invocation) { | ||
|  |       currentInvocation = null; | ||
|  |     } | ||
|  | 
 | ||
|  |     invocation.job.emit('canceled', invocation.fireDate); | ||
|  |     prepareNextInvocation(); | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | /* Recurrence scheduler */ | ||
|  | function scheduleNextRecurrence(rule, job, prevDate, endDate) { | ||
|  | 
 | ||
|  |   prevDate = (prevDate instanceof CronDate) ? prevDate : new CronDate(); | ||
|  | 
 | ||
|  |   const date = (rule instanceof RecurrenceRule) ? rule._nextInvocationDate(prevDate) : rule.next(); | ||
|  |   if (date === null) { | ||
|  |     return null; | ||
|  |   } | ||
|  | 
 | ||
|  |   if ((endDate instanceof CronDate) && date.getTime() > endDate.getTime()) { | ||
|  |     return null; | ||
|  |   } | ||
|  | 
 | ||
|  |   const inv = new Invocation(job, date, rule, endDate); | ||
|  |   scheduleInvocation(inv); | ||
|  | 
 | ||
|  |   return inv; | ||
|  | } | ||
|  | 
 | ||
|  | module.exports = { | ||
|  |   Range, | ||
|  |   RecurrenceRule, | ||
|  |   Invocation, | ||
|  |   cancelInvocation, | ||
|  |   scheduleInvocation, | ||
|  |   scheduleNextRecurrence, | ||
|  |   sorter, | ||
|  |   _invocations: invocations | ||
|  | } |