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