oneuptime/Worker/Jobs/ScheduledMaintenance/ScheduleRecurringEvents.ts

231 lines
9.2 KiB
TypeScript
Raw Permalink Normal View History

import RunCron from "../../Utils/Cron";
import LIMIT_MAX from "Common/Types/Database/LimitMax";
import OneUptimeDate from "Common/Types/Date";
import { EVERY_MINUTE } from "Common/Utils/CronTime";
import ScheduledMaintenanceService from "Common/Server/Services/ScheduledMaintenanceService";
import QueryHelper from "Common/Server/Types/Database/QueryHelper";
import ScheduledMaintenanceTemplate from "Common/Models/DatabaseModels/ScheduledMaintenanceTemplate";
import ScheduledMaintenanceTemplateService from "Common/Server/Services/ScheduledMaintenanceTemplateService";
import ScheduledMaintenanceTemplateOwnerUserService from "Common/Server/Services/ScheduledMaintenanceTemplateOwnerUserService";
import ScheduledMaintenanceOwnerUser from "Common/Models/DatabaseModels/ScheduledMaintenanceOwnerUser";
import ScheduledMaintenanceTemplateOwnerUser from "Common/Models/DatabaseModels/ScheduledMaintenanceTemplateOwnerUser";
import ScheduledMaintenanceOwnerTeamService from "Common/Server/Services/ScheduledMaintenanceOwnerTeamService";
import ScheduledMaintenanceTemplateOwnerTeamService from "Common/Server/Services/ScheduledMaintenanceTemplateOwnerTeamService";
import ScheduledMaintenance from "Common/Models/DatabaseModels/ScheduledMaintenance";
import ScheduledMaintenanceOwnerUserService from "Common/Server/Services/ScheduledMaintenanceOwnerUserService";
import ScheduledMaintenanceOwnerTeam from "Common/Models/DatabaseModels/ScheduledMaintenanceOwnerTeam";
import logger from "Common/Server/Utils/Logger";
import Recurring from "Common/Types/Events/Recurring";
RunCron(
"ScheduledMaintenance:ScheduleRecurringEvents",
{ schedule: EVERY_MINUTE, runOnStartup: false },
async () => {
logger.debug("ScheduledMaintenance:ScheduleRecurringEvents Running");
// get all scheduled events of all the projects.
const recurringTemplates: Array<ScheduledMaintenanceTemplate> =
await ScheduledMaintenanceTemplateService.findBy({
query: {
isRecurringEvent: true,
scheduleNextEventAt: QueryHelper.lessThanEqualTo(
OneUptimeDate.getCurrentDate(),
),
},
props: {
isRoot: true,
},
limit: LIMIT_MAX,
skip: 0,
select: {
_id: true,
projectId: true,
changeMonitorStatusToId: true,
shouldStatusPageSubscribersBeNotifiedWhenEventChangedToEnded: true,
shouldStatusPageSubscribersBeNotifiedOnEventCreated: true,
shouldStatusPageSubscribersBeNotifiedWhenEventChangedToOngoing: true,
monitors: true,
statusPages: true,
scheduleNextEventAt: true,
firstEventStartsAt: true,
firstEventEndsAt: true,
firstEventScheduledAt: true,
title: true,
description: true,
labels: true,
isRecurringEvent: true,
recurringInterval: true,
sendSubscriberNotificationsOnBeforeTheEvent: true,
},
});
// change their state to Ongoing.
for (const recurringTemplate of recurringTemplates) {
try {
logger.debug(
`ScheduledMaintenance:ScheduleRecurringEvents: Updating event: ${recurringTemplate.id}`,
);
if (recurringTemplate.recurringInterval === undefined) {
continue;
}
// update the next scheduled time for this event.
const recurringInterval: Recurring =
recurringTemplate.recurringInterval!;
const nextScheduledTime: Date =
ScheduledMaintenanceTemplateService.getNextEventTime({
dateAndTime: recurringTemplate.scheduleNextEventAt!,
recurringInterval,
});
await ScheduledMaintenanceTemplateService.updateOneById({
id: recurringTemplate.id!,
data: {
scheduleNextEventAt: nextScheduledTime,
},
props: {
isRoot: true,
},
});
// get owner users for this template.
const ownerUsers: Array<ScheduledMaintenanceTemplateOwnerUser> =
await ScheduledMaintenanceTemplateOwnerUserService.findBy({
query: {
scheduledMaintenanceTemplateId: recurringTemplate.id!,
},
props: {
isRoot: true,
},
limit: LIMIT_MAX,
skip: 0,
select: {
userId: true,
},
});
// owner teams.
const ownerTeams: Array<ScheduledMaintenanceOwnerTeam> =
await ScheduledMaintenanceTemplateOwnerTeamService.findBy({
query: {
scheduledMaintenanceTemplateId: recurringTemplate.id!,
},
props: {
isRoot: true,
},
limit: LIMIT_MAX,
skip: 0,
select: {
teamId: true,
},
});
// now create a new scheduled maintenance event for this template.
let scheduledMaintenanceEvent: ScheduledMaintenance =
new ScheduledMaintenance();
scheduledMaintenanceEvent.projectId = recurringTemplate.projectId!;
scheduledMaintenanceEvent.changeMonitorStatusToId =
recurringTemplate.changeMonitorStatusToId!;
scheduledMaintenanceEvent.shouldStatusPageSubscribersBeNotifiedWhenEventChangedToEnded =
recurringTemplate.shouldStatusPageSubscribersBeNotifiedWhenEventChangedToEnded!;
scheduledMaintenanceEvent.shouldStatusPageSubscribersBeNotifiedOnEventCreated =
recurringTemplate.shouldStatusPageSubscribersBeNotifiedOnEventCreated!;
scheduledMaintenanceEvent.shouldStatusPageSubscribersBeNotifiedWhenEventChangedToOngoing =
recurringTemplate.shouldStatusPageSubscribersBeNotifiedWhenEventChangedToOngoing!;
scheduledMaintenanceEvent.monitors = recurringTemplate.monitors!;
scheduledMaintenanceEvent.statusPages = recurringTemplate.statusPages!;
scheduledMaintenanceEvent.title = recurringTemplate.title!;
scheduledMaintenanceEvent.description = recurringTemplate.description!;
scheduledMaintenanceEvent.labels = recurringTemplate.labels!;
scheduledMaintenanceEvent.sendSubscriberNotificationsOnBeforeTheEvent =
recurringTemplate.sendSubscriberNotificationsOnBeforeTheEvent!;
const eventscheduledTime: Date = recurringTemplate.scheduleNextEventAt!;
const firstScheduledTime: Date =
recurringTemplate.firstEventScheduledAt!;
const firstStartTime: Date = recurringTemplate.firstEventStartsAt!;
const firstEndTime: Date = recurringTemplate.firstEventEndsAt!;
const minutesBetwenScheduledAndStartTime: number =
OneUptimeDate.getMinutesBetweenTwoDates(
eventscheduledTime,
firstStartTime,
);
const minutesBetweenScheduledAndEndTime: number =
OneUptimeDate.getMinutesBetweenTwoDates(
eventscheduledTime,
firstEndTime,
);
// set the scheduled time for this event.
scheduledMaintenanceEvent.createdAt = eventscheduledTime!;
scheduledMaintenanceEvent.startsAt = OneUptimeDate.addRemoveMinutes(
firstScheduledTime,
minutesBetwenScheduledAndStartTime,
);
scheduledMaintenanceEvent.endsAt = OneUptimeDate.addRemoveMinutes(
firstScheduledTime,
minutesBetweenScheduledAndEndTime,
);
// now create this event.
scheduledMaintenanceEvent = await ScheduledMaintenanceService.create({
data: scheduledMaintenanceEvent,
props: {
isRoot: true,
},
});
// now add owners and teams to this event.
for (const ownerUser of ownerUsers) {
const scheduledMaintenanceOwnerUser: ScheduledMaintenanceOwnerUser =
new ScheduledMaintenanceOwnerUser();
scheduledMaintenanceOwnerUser.scheduledMaintenanceId =
scheduledMaintenanceEvent.id!;
scheduledMaintenanceOwnerUser.projectId =
scheduledMaintenanceEvent.projectId!;
scheduledMaintenanceOwnerUser.userId = ownerUser.userId!;
await ScheduledMaintenanceOwnerUserService.create({
data: scheduledMaintenanceOwnerUser,
props: {
isRoot: true,
},
});
}
// now do the same for owner teams.
for (const ownerTeam of ownerTeams) {
const scheduledMaintenanceOwnerTeam: ScheduledMaintenanceOwnerTeam =
new ScheduledMaintenanceOwnerTeam();
scheduledMaintenanceOwnerTeam.scheduledMaintenanceId =
scheduledMaintenanceEvent.id!;
scheduledMaintenanceOwnerTeam.projectId =
scheduledMaintenanceEvent.projectId!;
scheduledMaintenanceOwnerTeam.teamId = ownerTeam.teamId!;
await ScheduledMaintenanceOwnerTeamService.create({
data: scheduledMaintenanceOwnerTeam,
props: {
isRoot: true,
},
});
}
logger.debug(
`ScheduledMaintenance:ScheduleRecurringEvents: Created event: ${scheduledMaintenanceEvent.id}`,
);
} catch (e) {
logger.error(
`ScheduledMaintenance:ScheduleRecurringEvents: Error creating event for template: ${recurringTemplate.id}`,
);
logger.error(e);
}
}
},
);