Files
GibSend/apps/web/src/server/service/domain-service.ts
2025-08-09 20:37:29 +10:00

205 lines
4.8 KiB
TypeScript

import dns from "dns";
import util from "util";
import * as tldts from "tldts";
import * as ses from "~/server/aws/ses";
import { db } from "~/server/db";
import { SesSettingsService } from "./ses-settings-service";
import { UnsendApiError } from "../public-api/api-error";
import { logger } from "../logger/log";
const dnsResolveTxt = util.promisify(dns.resolveTxt);
export async function validateDomainFromEmail(email: string, teamId: number) {
// Extract email from format like 'Name <email@domain>' this will allow entries such as "Someone @ something <some@domain.com>" to parse correctly as well.
const match = email.match(/<([^>]+)>/);
let fromDomain: string | undefined;
if (match && match[1]) {
const parts = match[1].split("@");
fromDomain = parts.length > 1 ? parts[1] : undefined;
} else {
const parts = email.split("@");
fromDomain = parts.length > 1 ? parts[1] : undefined;
}
if (fromDomain?.endsWith(">")) {
fromDomain = fromDomain.slice(0, -1);
}
if (!fromDomain) {
throw new UnsendApiError({
code: "BAD_REQUEST",
message: "From email is invalid",
});
}
const domain = await db.domain.findUnique({
where: { name: fromDomain, teamId },
});
if (!domain) {
throw new UnsendApiError({
code: "BAD_REQUEST",
message: `Domain: ${fromDomain} of from email is wrong. Use the domain verified by unsend`,
});
}
if (domain.status !== "SUCCESS") {
throw new UnsendApiError({
code: "BAD_REQUEST",
message: `Domain: ${fromDomain} is not verified`,
});
}
return domain;
}
export async function createDomain(
teamId: number,
name: string,
region: string,
sesTenantId?: string
) {
const domainStr = tldts.getDomain(name);
logger.info({ domainStr, name, region }, "Creating domain");
if (!domainStr) {
throw new Error("Invalid domain");
}
const setting = await SesSettingsService.getSetting(region);
if (!setting) {
throw new Error("Ses setting not found");
}
const subdomain = tldts.getSubdomain(name);
const publicKey = await ses.addDomain(name, region, sesTenantId);
const domain = await db.domain.create({
data: {
name,
publicKey,
teamId,
subdomain,
region,
sesTenantId,
},
});
return domain;
}
export async function getDomain(id: number) {
let domain = await db.domain.findUnique({
where: {
id,
},
});
if (!domain) {
throw new Error("Domain not found");
}
if (domain.isVerifying) {
const domainIdentity = await ses.getDomainIdentity(
domain.name,
domain.region
);
const dkimStatus = domainIdentity.DkimAttributes?.Status;
const spfDetails = domainIdentity.MailFromAttributes?.MailFromDomainStatus;
const verificationError = domainIdentity.VerificationInfo?.ErrorType;
const verificationStatus = domainIdentity.VerificationStatus;
const lastCheckedTime =
domainIdentity.VerificationInfo?.LastCheckedTimestamp;
const _dmarcRecord = await getDmarcRecord(domain.name);
const dmarcRecord = _dmarcRecord?.[0]?.[0];
domain = await db.domain.update({
where: {
id,
},
data: {
dkimStatus,
spfDetails,
status: verificationStatus ?? "NOT_STARTED",
dmarcAdded: dmarcRecord ? true : false,
isVerifying:
verificationStatus === "SUCCESS" &&
dkimStatus === "SUCCESS" &&
spfDetails === "SUCCESS"
? false
: true,
},
});
return {
...domain,
dkimStatus: dkimStatus?.toString() ?? null,
spfDetails: spfDetails?.toString() ?? null,
verificationError: verificationError?.toString() ?? null,
lastCheckedTime,
dmarcAdded: dmarcRecord ? true : false,
};
}
return domain;
}
export async function updateDomain(
id: number,
data: { clickTracking?: boolean; openTracking?: boolean }
) {
return db.domain.update({
where: { id },
data,
});
}
export async function deleteDomain(id: number) {
const domain = await db.domain.findUnique({
where: { id },
});
if (!domain) {
throw new Error("Domain not found");
}
const deleted = await ses.deleteDomain(
domain.name,
domain.region,
domain.sesTenantId ?? undefined
);
if (!deleted) {
throw new Error("Error in deleting domain");
}
return db.domain.delete({
where: { id },
});
}
export async function getDomains(teamId: number) {
return db.domain.findMany({
where: {
teamId,
},
orderBy: {
createdAt: "desc",
},
});
}
async function getDmarcRecord(domain: string) {
try {
const dmarcRecord = await dnsResolveTxt(`_dmarc.${domain}`);
return dmarcRecord;
} catch (error) {
logger.error({ err: error, domain }, "Error fetching DMARC record");
return null; // or handle error as appropriate
}
}