import { model, Schema, Model } from 'mongoose'; import createEventSchema, { EventDocument } from './schema'; import cron from 'cron'; interface Event extends EventDocument { log(message: string): void; start(): void; complete(): void; fail(error: Error): void; computeNextRunAt(): Date; } export interface EventModel extends Model> { findNextEvents(): Event[]; findMissedEvents(): Event[]; } const CronJob = cron.CronJob; const createEventModel = (name: string, contextSchema: Context): EventModel => { const schema = createEventSchema(contextSchema); // Schema methods schema.method('log', function(message: string) { // TODO: Actually create logs console.log(message); }); schema.method('start', function() { this.log('Event started') this.lastRunAt = new Date(); this.status = 'running'; return this.save(); }); schema.method('complete', function() { this.log('Event complete') this.status = 'complete'; return this.save(); }); schema.method('fail', function(error: Error) { this.log(error); this.error = error; this.status = 'failed'; return this.save(); }); schema.method('computeNextRunAt', function() { const job = new CronJob(this.schedule); const nextRunAt = job.nextDates(); return nextRunAt.toDate(); }); // Statics schema.static('findMissedEvents', async function () { return this.find({ nextRunAt: { // TODO: skip single-fire events $lt: new Date() }, }); }); schema.static('findNextEvents', function(limit = 10) { return this.find( { nextRunAt: { $gt: new Date() }, }, null, { sort: { nextRunAt: 1 }, limit } ) }); // Hooks schema.pre>('save', async function() { this.nextRunAt = this.computeNextRunAt(); }); return model, EventModel>(name, schema); }; export default createEventModel;