mirror of
https://github.com/OneUptime/oneuptime.git
synced 2026-04-06 00:32:12 +02:00
204 lines
5.4 KiB
TypeScript
204 lines
5.4 KiB
TypeScript
import { IsBillingEnabled } from "../EnvironmentConfig";
|
|
import CreateBy from "../Types/Database/CreateBy";
|
|
import DeleteBy from "../Types/Database/DeleteBy";
|
|
import { OnCreate, OnDelete } from "../Types/Database/Hooks";
|
|
import logger from "../Utils/Logger";
|
|
import DatabaseService from "./DatabaseService";
|
|
import ProjectService from "./ProjectService";
|
|
import UserNotificationRuleService from "./UserNotificationRuleService";
|
|
import WhatsAppService from "./WhatsAppService";
|
|
import LIMIT_MAX from "../../Types/Database/LimitMax";
|
|
import BadDataException from "../../Types/Exception/BadDataException";
|
|
import ObjectID from "../../Types/ObjectID";
|
|
import Text from "../../Types/Text";
|
|
import Project from "../../Models/DatabaseModels/Project";
|
|
import Model from "../../Models/DatabaseModels/UserWhatsApp";
|
|
import CaptureSpan from "../Utils/Telemetry/CaptureSpan";
|
|
import WhatsAppMessage from "../../Types/WhatsApp/WhatsAppMessage";
|
|
import {
|
|
WhatsAppTemplateIds,
|
|
WhatsAppTemplateLanguage,
|
|
WhatsAppTemplateId,
|
|
} from "../../Types/WhatsApp/WhatsAppTemplates";
|
|
import HTTPErrorResponse from "../../Types/API/HTTPErrorResponse";
|
|
import HTTPResponse from "../../Types/API/HTTPResponse";
|
|
import { JSONObject } from "../../Types/JSON";
|
|
|
|
export class Service extends DatabaseService<Model> {
|
|
public constructor() {
|
|
super(Model);
|
|
}
|
|
|
|
@CaptureSpan()
|
|
protected override async onBeforeDelete(
|
|
deleteBy: DeleteBy<Model>,
|
|
): Promise<OnDelete<Model>> {
|
|
const itemsToDelete: Array<Model> = await this.findBy({
|
|
query: deleteBy.query,
|
|
select: {
|
|
_id: true,
|
|
projectId: true,
|
|
},
|
|
skip: 0,
|
|
limit: LIMIT_MAX,
|
|
props: {
|
|
isRoot: true,
|
|
},
|
|
});
|
|
|
|
for (const item of itemsToDelete) {
|
|
await UserNotificationRuleService.deleteBy({
|
|
query: {
|
|
userWhatsAppId: item.id!,
|
|
projectId: item.projectId!,
|
|
},
|
|
limit: LIMIT_MAX,
|
|
skip: 0,
|
|
props: {
|
|
isRoot: true,
|
|
},
|
|
});
|
|
}
|
|
|
|
return {
|
|
deleteBy,
|
|
carryForward: null,
|
|
};
|
|
}
|
|
|
|
@CaptureSpan()
|
|
protected override async onBeforeCreate(
|
|
createBy: CreateBy<Model>,
|
|
): Promise<OnCreate<Model>> {
|
|
if (!createBy.props.isRoot && createBy.data.isVerified) {
|
|
throw new BadDataException("isVerified cannot be set to true");
|
|
}
|
|
|
|
const project: Project | null = await ProjectService.findOneById({
|
|
id: createBy.data.projectId!,
|
|
props: {
|
|
isRoot: true,
|
|
},
|
|
select: {
|
|
enableWhatsAppNotifications: true,
|
|
smsOrCallCurrentBalanceInUSDCents: true,
|
|
},
|
|
});
|
|
|
|
if (!project) {
|
|
throw new BadDataException("Project not found");
|
|
}
|
|
|
|
if (!project.enableWhatsAppNotifications) {
|
|
throw new BadDataException(
|
|
"WhatsApp notifications are disabled for this project. Please enable them in Project Settings > Notification Settings.",
|
|
);
|
|
}
|
|
|
|
if (
|
|
(project.smsOrCallCurrentBalanceInUSDCents as number) <= 100 &&
|
|
IsBillingEnabled
|
|
) {
|
|
throw new BadDataException(
|
|
"Your WhatsApp balance is low. Please recharge your balance in Project Settings > Notification Settings.",
|
|
);
|
|
}
|
|
|
|
return {
|
|
createBy,
|
|
carryForward: null,
|
|
};
|
|
}
|
|
|
|
@CaptureSpan()
|
|
protected override async onCreateSuccess(
|
|
_onCreate: OnCreate<Model>,
|
|
createdItem: Model,
|
|
): Promise<Model> {
|
|
if (!createdItem.isVerified) {
|
|
this.sendVerificationCode(createdItem).catch((error: Error) => {
|
|
logger.error(error);
|
|
});
|
|
}
|
|
|
|
return createdItem;
|
|
}
|
|
|
|
@CaptureSpan()
|
|
public async resendVerificationCode(itemId: ObjectID): Promise<void> {
|
|
const item: Model | null = await this.findOneById({
|
|
id: itemId,
|
|
props: {
|
|
isRoot: true,
|
|
},
|
|
select: {
|
|
phone: true,
|
|
verificationCode: true,
|
|
isVerified: true,
|
|
projectId: true,
|
|
userId: true,
|
|
},
|
|
});
|
|
|
|
if (!item) {
|
|
throw new BadDataException(
|
|
"Item with ID " + itemId.toString() + " not found",
|
|
);
|
|
}
|
|
|
|
if (item.isVerified) {
|
|
throw new BadDataException("WhatsApp number already verified");
|
|
}
|
|
|
|
item.verificationCode = Text.generateRandomNumber(6);
|
|
|
|
await this.updateOneById({
|
|
id: item.id!,
|
|
props: {
|
|
isRoot: true,
|
|
},
|
|
data: {
|
|
verificationCode: item.verificationCode,
|
|
},
|
|
});
|
|
|
|
await this.sendVerificationCode(item);
|
|
}
|
|
|
|
public async sendVerificationCode(item: Model): Promise<void> {
|
|
if (!item.projectId || !item.userId || !item.phone) {
|
|
logger.warn("Cannot send WhatsApp verification code. Missing data.");
|
|
throw new BadDataException(
|
|
"Unable to send WhatsApp verification code. Please remove this number and add it again.",
|
|
);
|
|
}
|
|
|
|
const templateKey: WhatsAppTemplateId =
|
|
WhatsAppTemplateIds.VerificationCode;
|
|
const templateVariables: Record<string, string> = {
|
|
"1": item.verificationCode || "",
|
|
};
|
|
|
|
const whatsAppMessage: WhatsAppMessage = {
|
|
to: item.phone,
|
|
body: "",
|
|
templateKey,
|
|
templateVariables,
|
|
templateLanguageCode: WhatsAppTemplateLanguage[templateKey],
|
|
};
|
|
|
|
const response: HTTPResponse<JSONObject> =
|
|
await WhatsAppService.sendWhatsAppMessage(whatsAppMessage, {
|
|
projectId: item.projectId,
|
|
isSensitive: true,
|
|
userId: item.userId,
|
|
});
|
|
|
|
if (response instanceof HTTPErrorResponse) {
|
|
throw response;
|
|
}
|
|
}
|
|
}
|
|
|
|
export default new Service();
|