107 lines
3.1 KiB
TypeScript
107 lines
3.1 KiB
TypeScript
import { NotFoundError, resolver } from "blitz";
|
|
import { z } from "zod";
|
|
import twilio from "twilio";
|
|
import RestException from "twilio/lib/base/RestException";
|
|
|
|
import db from "db";
|
|
import getCurrentUser from "app/users/queries/get-current-user";
|
|
import setTwilioWebhooks from "../api/queue/set-twilio-webhooks";
|
|
import fetchMessagesQueue from "app/messages/api/queue/fetch-messages";
|
|
import fetchCallsQueue from "app/phone-calls/api/queue/fetch-calls";
|
|
|
|
const Body = z.object({
|
|
phoneNumberSid: z.string(),
|
|
});
|
|
|
|
export default resolver.pipe(resolver.zod(Body), resolver.authorize(), async ({ phoneNumberSid }, context) => {
|
|
const user = await getCurrentUser(null, context);
|
|
const organization = user?.memberships[0]!.organization;
|
|
if (!user || !organization || !organization.twilioAccountSid || !organization.twilioAuthToken) {
|
|
return;
|
|
}
|
|
|
|
const phoneNumbers = await twilio(
|
|
organization.twilioAccountSid,
|
|
organization.twilioAuthToken,
|
|
).incomingPhoneNumbers.list();
|
|
const twilioPhoneNumber = phoneNumbers.find((phoneNumber) => phoneNumber.sid === phoneNumberSid);
|
|
if (!twilioPhoneNumber) {
|
|
throw new NotFoundError();
|
|
}
|
|
|
|
const organizationId = organization.id;
|
|
const orgCurrentlyActivePhoneNumber = await db.phoneNumber.findFirst({ where: { organizationId } });
|
|
if (orgCurrentlyActivePhoneNumber) {
|
|
// TODO: delete this and allow switching phone numbers easily
|
|
await db.phoneNumber.delete({
|
|
where: {
|
|
organizationId_id: {
|
|
organizationId,
|
|
id: orgCurrentlyActivePhoneNumber.id,
|
|
},
|
|
},
|
|
});
|
|
}
|
|
|
|
const phoneNumber = await db.phoneNumber.findFirst({ where: { id: phoneNumberSid } });
|
|
if (!phoneNumber) {
|
|
await db.phoneNumber.create({
|
|
data: {
|
|
organizationId,
|
|
id: phoneNumberSid,
|
|
number: twilioPhoneNumber.phoneNumber,
|
|
},
|
|
});
|
|
}
|
|
|
|
let newApiKey;
|
|
const mainTwilioClient = twilio(organization.twilioAccountSid, organization.twilioAuthToken);
|
|
if (!organization.twilioApiKey) {
|
|
newApiKey = await mainTwilioClient.newKeys.create({ friendlyName: "Shellphone API key" });
|
|
} else {
|
|
try {
|
|
await mainTwilioClient.keys.get(organization.twilioApiKey);
|
|
} catch (error) {
|
|
if (!(error instanceof RestException) || error.code !== 20404) {
|
|
throw error;
|
|
}
|
|
|
|
// API key was not found, create a new one
|
|
newApiKey = await mainTwilioClient.newKeys.create({ friendlyName: "Shellphone API key" });
|
|
}
|
|
}
|
|
if (newApiKey) {
|
|
await db.organization.update({
|
|
where: { id: organizationId },
|
|
data: {
|
|
twilioApiKey: newApiKey.sid,
|
|
twilioApiSecret: newApiKey.secret,
|
|
},
|
|
});
|
|
}
|
|
|
|
const phoneNumberId = phoneNumberSid;
|
|
await Promise.all([
|
|
setTwilioWebhooks.enqueue(
|
|
{ organizationId, phoneNumberId },
|
|
{ id: `set-twilio-webhooks-${organizationId}-${phoneNumberId}` },
|
|
),
|
|
db.processingPhoneNumber.create({
|
|
data: {
|
|
organizationId,
|
|
phoneNumberId,
|
|
hasFetchedMessages: false,
|
|
hasFetchedCalls: false,
|
|
},
|
|
}),
|
|
fetchMessagesQueue.enqueue(
|
|
{ organizationId, phoneNumberId },
|
|
{ id: `fetch-messages-${organizationId}-${phoneNumberId}` },
|
|
),
|
|
fetchCallsQueue.enqueue(
|
|
{ organizationId, phoneNumberId },
|
|
{ id: `fetch-messages-${organizationId}-${phoneNumberId}` },
|
|
),
|
|
]);
|
|
});
|