From 4186fb51f66043b221237c7ba91ac5a95a4f32b0 Mon Sep 17 00:00:00 2001 From: eug-vs Date: Wed, 2 Dec 2020 02:22:38 +0300 Subject: feat: add log model --- lib/event.model.ts | 100 +++++++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 100 insertions(+) create mode 100644 lib/event.model.ts (limited to 'lib/event.model.ts') diff --git a/lib/event.model.ts b/lib/event.model.ts new file mode 100644 index 0000000..a430674 --- /dev/null +++ b/lib/event.model.ts @@ -0,0 +1,100 @@ +import { model, Schema, Model } from 'mongoose'; +import cron from 'cron'; +import createEventSchema, { EventDocument } from './event.schema'; +import { LogDocument } from './log.schema'; +import LogModel from './log.model'; + +interface Event extends EventDocument { + log(message: string): void; + start(): void; + complete(): void; + fail(error: Error): void; + computeNextRunAt(): Date; + getLogs(): Promise; +} + +export interface EventModel extends Model> { + findNextEvents(): Event[]; + findMissedEvents(): Event[]; +} + +const CronJob = cron.CronJob; + +const createEventModel = (name: string, contextSchema: Schema): EventModel => { + const schema = createEventSchema(contextSchema); + + // Schema methods + schema.method('log', function(message: string) { + const timestamp = new Date().toLocaleString('en'); + console.log(`[${timestamp}] ${this.name}: ${message}`); + return LogModel.create({ eventId: this._id, 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(); + }); + + schema.method('getLogs', function() { + return LogModel.find({ eventId: this._id }); + }); + + // 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; + -- cgit v1.2.3