mirror of
https://github.com/OneUptime/oneuptime.git
synced 2026-04-06 00:32:12 +02:00
- Simplified import statements in PhoneNumber.ts and Config.ts. - Consolidated async function calls in TwilioCallProvider.ts for clarity. - Enhanced comment formatting and structure in TwilioCallProvider.ts. - Reformatted SQL migration code for better readability in MigrationName.ts. - Updated array syntax in Index.ts for consistency. - Improved formatting and spacing in IncomingCallPolicyEscalationRuleService.ts. - Standardized JSX formatting in various components for better readability. - Enhanced conditional rendering logic in IncomingCallPolicies.tsx and Index.tsx. - Cleaned up unnecessary whitespace and improved code consistency across multiple files.
268 lines
7.2 KiB
TypeScript
268 lines
7.2 KiB
TypeScript
import DatabaseService from "./DatabaseService";
|
|
import CreateBy from "../Types/Database/CreateBy";
|
|
import { OnCreate, OnDelete, OnUpdate } from "../Types/Database/Hooks";
|
|
import DeleteBy from "../Types/Database/DeleteBy";
|
|
import UpdateBy from "../Types/Database/UpdateBy";
|
|
import Query from "../Types/Database/Query";
|
|
import QueryHelper from "../Types/Database/QueryHelper";
|
|
import BadDataException from "../../Types/Exception/BadDataException";
|
|
import PositiveNumber from "../../Types/PositiveNumber";
|
|
import ObjectID from "../../Types/ObjectID";
|
|
import SortOrder from "../../Types/BaseDatabase/SortOrder";
|
|
import LIMIT_MAX from "../../Types/Database/LimitMax";
|
|
import IncomingCallPolicyEscalationRule from "../../Models/DatabaseModels/IncomingCallPolicyEscalationRule";
|
|
|
|
export class Service extends DatabaseService<IncomingCallPolicyEscalationRule> {
|
|
public constructor() {
|
|
super(IncomingCallPolicyEscalationRule);
|
|
}
|
|
|
|
protected override async onBeforeCreate(
|
|
createBy: CreateBy<IncomingCallPolicyEscalationRule>,
|
|
): Promise<OnCreate<IncomingCallPolicyEscalationRule>> {
|
|
// Validate mutual exclusivity: either userId OR onCallDutyPolicyScheduleId must be set
|
|
const hasUser: boolean = Boolean(createBy.data.userId);
|
|
const hasSchedule: boolean = Boolean(
|
|
createBy.data.onCallDutyPolicyScheduleId,
|
|
);
|
|
|
|
if (!hasUser && !hasSchedule) {
|
|
throw new BadDataException(
|
|
"Either a User or an On-Call Schedule must be specified for the escalation rule",
|
|
);
|
|
}
|
|
|
|
if (hasUser && hasSchedule) {
|
|
throw new BadDataException(
|
|
"Only one of User or On-Call Schedule can be specified, not both",
|
|
);
|
|
}
|
|
|
|
if (!createBy.data.incomingCallPolicyId) {
|
|
throw new BadDataException("incomingCallPolicyId is required");
|
|
}
|
|
|
|
// Auto-generate order if not provided
|
|
if (!createBy.data.order) {
|
|
const query: Query<IncomingCallPolicyEscalationRule> = {
|
|
incomingCallPolicyId: createBy.data.incomingCallPolicyId,
|
|
};
|
|
|
|
const count: PositiveNumber = await this.countBy({
|
|
query: query,
|
|
props: {
|
|
isRoot: true,
|
|
},
|
|
});
|
|
|
|
createBy.data.order = count.toNumber() + 1;
|
|
}
|
|
|
|
await this.rearrangeOrder(
|
|
createBy.data.order,
|
|
createBy.data.incomingCallPolicyId,
|
|
true,
|
|
);
|
|
|
|
return {
|
|
createBy,
|
|
carryForward: null,
|
|
};
|
|
}
|
|
|
|
protected override async onBeforeDelete(
|
|
deleteBy: DeleteBy<IncomingCallPolicyEscalationRule>,
|
|
): Promise<OnDelete<IncomingCallPolicyEscalationRule>> {
|
|
if (!deleteBy.query._id && !deleteBy.props.isRoot) {
|
|
throw new BadDataException(
|
|
"_id should be present when deleting escalation rule. Please try the delete with objectId",
|
|
);
|
|
}
|
|
|
|
let resource: IncomingCallPolicyEscalationRule | null = null;
|
|
|
|
if (!deleteBy.props.isRoot) {
|
|
resource = await this.findOneBy({
|
|
query: deleteBy.query,
|
|
props: {
|
|
isRoot: true,
|
|
},
|
|
select: {
|
|
order: true,
|
|
incomingCallPolicyId: true,
|
|
projectId: true,
|
|
},
|
|
});
|
|
|
|
if (!resource) {
|
|
throw new BadDataException(
|
|
"IncomingCallPolicyEscalationRule with this id not found",
|
|
);
|
|
}
|
|
}
|
|
|
|
return {
|
|
deleteBy,
|
|
carryForward: resource,
|
|
};
|
|
}
|
|
|
|
protected override async onDeleteSuccess(
|
|
onDelete: OnDelete<IncomingCallPolicyEscalationRule>,
|
|
_itemIdsBeforeDelete: ObjectID[],
|
|
): Promise<OnDelete<IncomingCallPolicyEscalationRule>> {
|
|
const deleteBy: DeleteBy<IncomingCallPolicyEscalationRule> =
|
|
onDelete.deleteBy;
|
|
const resource: IncomingCallPolicyEscalationRule | null =
|
|
onDelete.carryForward;
|
|
|
|
if (!deleteBy.props.isRoot && resource) {
|
|
if (resource && resource.order && resource.incomingCallPolicyId) {
|
|
await this.rearrangeOrder(
|
|
resource.order,
|
|
resource.incomingCallPolicyId,
|
|
false,
|
|
);
|
|
}
|
|
}
|
|
|
|
return {
|
|
deleteBy: deleteBy,
|
|
carryForward: null,
|
|
};
|
|
}
|
|
|
|
protected override async onBeforeUpdate(
|
|
updateBy: UpdateBy<IncomingCallPolicyEscalationRule>,
|
|
): Promise<OnUpdate<IncomingCallPolicyEscalationRule>> {
|
|
if (updateBy.data.order && !updateBy.props.isRoot && updateBy.query._id) {
|
|
const resource: IncomingCallPolicyEscalationRule | null =
|
|
await this.findOneBy({
|
|
query: {
|
|
_id: updateBy.query._id!,
|
|
},
|
|
props: {
|
|
isRoot: true,
|
|
},
|
|
select: {
|
|
order: true,
|
|
incomingCallPolicyId: true,
|
|
_id: true,
|
|
},
|
|
});
|
|
|
|
const currentOrder: number = resource?.order as number;
|
|
const newOrder: number = updateBy.data.order as number;
|
|
|
|
const resources: Array<IncomingCallPolicyEscalationRule> =
|
|
await this.findBy({
|
|
query: {
|
|
incomingCallPolicyId: resource?.incomingCallPolicyId as ObjectID,
|
|
},
|
|
limit: LIMIT_MAX,
|
|
skip: 0,
|
|
props: {
|
|
isRoot: true,
|
|
},
|
|
select: {
|
|
order: true,
|
|
incomingCallPolicyId: true,
|
|
_id: true,
|
|
},
|
|
});
|
|
|
|
if (currentOrder > newOrder) {
|
|
// moving up.
|
|
for (const resource of resources) {
|
|
if (resource.order! >= newOrder && resource.order! < currentOrder) {
|
|
// increment order.
|
|
await this.updateOneBy({
|
|
query: {
|
|
_id: resource._id!,
|
|
},
|
|
data: {
|
|
order: resource.order! + 1,
|
|
},
|
|
props: {
|
|
isRoot: true,
|
|
},
|
|
});
|
|
}
|
|
}
|
|
}
|
|
|
|
if (newOrder > currentOrder) {
|
|
// moving down.
|
|
for (const resource of resources) {
|
|
if (resource.order! <= newOrder) {
|
|
// increment order.
|
|
await this.updateOneBy({
|
|
query: {
|
|
_id: resource._id!,
|
|
},
|
|
data: {
|
|
order: resource.order! - 1,
|
|
},
|
|
props: {
|
|
isRoot: true,
|
|
},
|
|
});
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return { updateBy, carryForward: null };
|
|
}
|
|
|
|
private async rearrangeOrder(
|
|
currentOrder: number,
|
|
incomingCallPolicyId: ObjectID,
|
|
increaseOrder: boolean = true,
|
|
): Promise<void> {
|
|
const resources: Array<IncomingCallPolicyEscalationRule> =
|
|
await this.findBy({
|
|
query: {
|
|
order: QueryHelper.greaterThanEqualTo(currentOrder),
|
|
incomingCallPolicyId: incomingCallPolicyId,
|
|
},
|
|
limit: LIMIT_MAX,
|
|
skip: 0,
|
|
props: {
|
|
isRoot: true,
|
|
},
|
|
select: {
|
|
_id: true,
|
|
order: true,
|
|
},
|
|
sort: {
|
|
order: SortOrder.Ascending,
|
|
},
|
|
});
|
|
|
|
let newOrder: number = currentOrder;
|
|
|
|
for (const resource of resources) {
|
|
if (increaseOrder) {
|
|
newOrder = resource.order! + 1;
|
|
} else {
|
|
newOrder = resource.order! - 1;
|
|
}
|
|
|
|
await this.updateOneBy({
|
|
query: {
|
|
_id: resource._id!,
|
|
},
|
|
data: {
|
|
order: newOrder,
|
|
},
|
|
props: {
|
|
isRoot: true,
|
|
},
|
|
});
|
|
}
|
|
}
|
|
}
|
|
|
|
export default new Service();
|