import { DocumentSource, WebhookTriggerEvents } from '@prisma/client'; import type { DocumentVisibility } from '@prisma/client'; import { normalizePdf as makeNormalizedPdf } from '@documenso/lib/server-only/pdf/normalize-pdf'; import { DOCUMENT_AUDIT_LOG_TYPE } from '@documenso/lib/types/document-audit-logs'; import type { ApiRequestMetadata } from '@documenso/lib/universal/extract-request-metadata'; import { createDocumentAuditLogData } from '@documenso/lib/utils/document-audit-logs'; import { prisma } from '@documenso/prisma'; import { AppError, AppErrorCode } from '../../errors/app-error'; import { ZWebhookDocumentSchema, mapDocumentToWebhookDocumentPayload, } from '../../types/webhook-payload'; import { prefixedId } from '../../universal/id'; import { getFileServerSide } from '../../universal/upload/get-file.server'; import { putPdfFileServerSide } from '../../universal/upload/put-file.server'; import { determineDocumentVisibility } from '../../utils/document-visibility'; import { getTeamById } from '../team/get-team'; import { getTeamSettings } from '../team/get-team-settings'; import { triggerWebhook } from '../webhooks/trigger/trigger-webhook'; export type CreateDocumentOptions = { title: string; externalId?: string | null; userId: number; teamId: number; documentDataId: string; formValues?: Record; normalizePdf?: boolean; timezone?: string; requestMetadata: ApiRequestMetadata; folderId?: string; }; export const createDocument = async ({ userId, title, externalId, documentDataId, teamId, normalizePdf, formValues, requestMetadata, timezone, folderId, }: CreateDocumentOptions) => { const team = await getTeamById({ userId, teamId }); const settings = await getTeamSettings({ userId, teamId, }); let folderVisibility: DocumentVisibility | undefined; if (folderId) { const folder = await prisma.folder.findFirst({ where: { id: folderId, userId, teamId, }, select: { visibility: true, }, }); if (!folder) { throw new AppError(AppErrorCode.NOT_FOUND, { message: 'Folder not found', }); } folderVisibility = folder.visibility; } if (normalizePdf) { const documentData = await prisma.documentData.findFirst({ where: { id: documentDataId, }, }); if (documentData) { const buffer = await getFileServerSide(documentData); const normalizedPdf = await makeNormalizedPdf(Buffer.from(buffer)); const newDocumentData = await putPdfFileServerSide({ name: title.endsWith('.pdf') ? title : `${title}.pdf`, type: 'application/pdf', arrayBuffer: async () => Promise.resolve(normalizedPdf), }); // eslint-disable-next-line require-atomic-updates documentDataId = newDocumentData.id; } } return await prisma.$transaction(async (tx) => { const document = await tx.document.create({ data: { title, qrToken: prefixedId('qr'), externalId, documentDataId, userId, teamId, folderId, visibility: folderVisibility ?? determineDocumentVisibility(settings.documentVisibility, team.currentTeamRole), formValues, source: DocumentSource.DOCUMENT, documentMeta: { create: { language: settings.documentLanguage, timezone: timezone, typedSignatureEnabled: settings.typedSignatureEnabled, uploadSignatureEnabled: settings.uploadSignatureEnabled, drawSignatureEnabled: settings.drawSignatureEnabled, }, }, }, }); await tx.documentAuditLog.create({ data: createDocumentAuditLogData({ type: DOCUMENT_AUDIT_LOG_TYPE.DOCUMENT_CREATED, documentId: document.id, metadata: requestMetadata, data: { title, source: { type: DocumentSource.DOCUMENT, }, }, }), }); const createdDocument = await tx.document.findFirst({ where: { id: document.id, }, include: { documentMeta: true, recipients: true, }, }); if (!createdDocument) { throw new Error('Document not found'); } await triggerWebhook({ event: WebhookTriggerEvents.DOCUMENT_CREATED, data: ZWebhookDocumentSchema.parse(mapDocumentToWebhookDocumentPayload(createdDocument)), userId, teamId, }); return createdDocument; }); };