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
						
					
					
				| '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
 | |
| }
 |