mirror of
https://github.com/documenso/documenso.git
synced 2025-11-14 00:32:43 +10:00
feat: trigger webhook functionality
This commit is contained in:
@ -2,7 +2,9 @@
|
|||||||
|
|
||||||
import { prisma } from '@documenso/prisma';
|
import { prisma } from '@documenso/prisma';
|
||||||
import { DocumentStatus, SigningStatus } from '@documenso/prisma/client';
|
import { DocumentStatus, SigningStatus } from '@documenso/prisma/client';
|
||||||
|
import { WebhookTriggerEvents } from '@documenso/prisma/client';
|
||||||
|
|
||||||
|
import { triggerWebhook } from '../../universal/trigger-webhook';
|
||||||
import { sealDocument } from './seal-document';
|
import { sealDocument } from './seal-document';
|
||||||
import { sendPendingEmail } from './send-pending-email';
|
import { sendPendingEmail } from './send-pending-email';
|
||||||
|
|
||||||
@ -11,13 +13,8 @@ export type CompleteDocumentWithTokenOptions = {
|
|||||||
documentId: number;
|
documentId: number;
|
||||||
};
|
};
|
||||||
|
|
||||||
export const completeDocumentWithToken = async ({
|
const getDocument = async ({ token, documentId }: CompleteDocumentWithTokenOptions) => {
|
||||||
token,
|
return await prisma.document.findFirstOrThrow({
|
||||||
documentId,
|
|
||||||
}: CompleteDocumentWithTokenOptions) => {
|
|
||||||
'use server';
|
|
||||||
|
|
||||||
const document = await prisma.document.findFirstOrThrow({
|
|
||||||
where: {
|
where: {
|
||||||
id: documentId,
|
id: documentId,
|
||||||
Recipient: {
|
Recipient: {
|
||||||
@ -34,6 +31,15 @@ export const completeDocumentWithToken = async ({
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
export const completeDocumentWithToken = async ({
|
||||||
|
token,
|
||||||
|
documentId,
|
||||||
|
}: CompleteDocumentWithTokenOptions) => {
|
||||||
|
'use server';
|
||||||
|
|
||||||
|
const document = await getDocument({ token, documentId });
|
||||||
|
|
||||||
if (document.status === DocumentStatus.COMPLETED) {
|
if (document.status === DocumentStatus.COMPLETED) {
|
||||||
throw new Error(`Document ${document.id} has already been completed`);
|
throw new Error(`Document ${document.id} has already been completed`);
|
||||||
@ -101,4 +107,11 @@ export const completeDocumentWithToken = async ({
|
|||||||
if (documents.count > 0) {
|
if (documents.count > 0) {
|
||||||
await sealDocument({ documentId: document.id });
|
await sealDocument({ documentId: document.id });
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const updatedDocument = await getDocument({ token, documentId });
|
||||||
|
|
||||||
|
await triggerWebhook({
|
||||||
|
eventTrigger: WebhookTriggerEvents.DOCUMENT_SIGNED,
|
||||||
|
documentData: updatedDocument,
|
||||||
|
});
|
||||||
};
|
};
|
||||||
|
|||||||
@ -1,6 +1,9 @@
|
|||||||
'use server';
|
'use server';
|
||||||
|
|
||||||
import { prisma } from '@documenso/prisma';
|
import { prisma } from '@documenso/prisma';
|
||||||
|
import { WebhookTriggerEvents } from '@documenso/prisma/client';
|
||||||
|
|
||||||
|
import { triggerWebhook } from '../../universal/trigger-webhook';
|
||||||
|
|
||||||
export type CreateDocumentOptions = {
|
export type CreateDocumentOptions = {
|
||||||
title: string;
|
title: string;
|
||||||
@ -29,7 +32,7 @@ export const createDocument = async ({
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
return await tx.document.create({
|
const createdDocument = await tx.document.create({
|
||||||
data: {
|
data: {
|
||||||
title,
|
title,
|
||||||
documentDataId,
|
documentDataId,
|
||||||
@ -37,5 +40,12 @@ export const createDocument = async ({
|
|||||||
teamId,
|
teamId,
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
|
await triggerWebhook({
|
||||||
|
eventTrigger: WebhookTriggerEvents.DOCUMENT_CREATED,
|
||||||
|
documentData: createdDocument,
|
||||||
|
});
|
||||||
|
|
||||||
|
return createdDocument;
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|||||||
17
packages/lib/server-only/webhooks/get-all-webhooks.ts
Normal file
17
packages/lib/server-only/webhooks/get-all-webhooks.ts
Normal file
@ -0,0 +1,17 @@
|
|||||||
|
import { prisma } from '@documenso/prisma';
|
||||||
|
import type { WebhookTriggerEvents } from '@documenso/prisma/client';
|
||||||
|
|
||||||
|
export type GetAllWebhooksOptions = {
|
||||||
|
eventTrigger: WebhookTriggerEvents;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const getAllWebhooks = async ({ eventTrigger }: GetAllWebhooksOptions) => {
|
||||||
|
return prisma.webhook.findMany({
|
||||||
|
where: {
|
||||||
|
eventTriggers: {
|
||||||
|
has: eventTrigger,
|
||||||
|
},
|
||||||
|
enabled: true,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
};
|
||||||
39
packages/lib/universal/post-webhook-payload.ts
Normal file
39
packages/lib/universal/post-webhook-payload.ts
Normal file
@ -0,0 +1,39 @@
|
|||||||
|
import type { Document, Webhook } from '@documenso/prisma/client';
|
||||||
|
|
||||||
|
export type PostWebhookPayloadOptions = {
|
||||||
|
webhookData: Pick<Webhook, 'webhookUrl' | 'secret' | 'eventTriggers'>;
|
||||||
|
documentData: Document;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const postWebhookPayload = async ({
|
||||||
|
webhookData,
|
||||||
|
documentData,
|
||||||
|
}: PostWebhookPayloadOptions) => {
|
||||||
|
const { webhookUrl, secret } = webhookData;
|
||||||
|
|
||||||
|
const payload = {
|
||||||
|
event: webhookData.eventTriggers.toString(),
|
||||||
|
createdAt: new Date().toISOString(),
|
||||||
|
webhookEndpoint: webhookUrl,
|
||||||
|
payload: documentData,
|
||||||
|
};
|
||||||
|
|
||||||
|
const response = await fetch(webhookUrl, {
|
||||||
|
method: 'POST',
|
||||||
|
body: JSON.stringify(payload),
|
||||||
|
headers: {
|
||||||
|
'Content-Type': 'application/json',
|
||||||
|
'X-Documenso-Secret': secret ?? '',
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!response.ok) {
|
||||||
|
throw new Error(`Webhook failed with the status code ${response.status}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
status: response.status,
|
||||||
|
statusText: response.statusText,
|
||||||
|
message: 'Webhook sent successfully',
|
||||||
|
};
|
||||||
|
};
|
||||||
30
packages/lib/universal/trigger-webhook.ts
Normal file
30
packages/lib/universal/trigger-webhook.ts
Normal file
@ -0,0 +1,30 @@
|
|||||||
|
import type { Document, WebhookTriggerEvents } from '@documenso/prisma/client';
|
||||||
|
|
||||||
|
import { getAllWebhooks } from '../server-only/webhooks/get-all-webhooks';
|
||||||
|
import { postWebhookPayload } from './post-webhook-payload';
|
||||||
|
|
||||||
|
export type TriggerWebhookOptions = {
|
||||||
|
eventTrigger: WebhookTriggerEvents;
|
||||||
|
documentData: Document;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const triggerWebhook = async ({ eventTrigger, documentData }: TriggerWebhookOptions) => {
|
||||||
|
try {
|
||||||
|
const allWebhooks = await getAllWebhooks({ eventTrigger });
|
||||||
|
|
||||||
|
const webhookPromises = allWebhooks.map((webhook) => {
|
||||||
|
const { webhookUrl, secret } = webhook;
|
||||||
|
|
||||||
|
postWebhookPayload({
|
||||||
|
webhookData: { webhookUrl, secret, eventTriggers: [eventTrigger] },
|
||||||
|
documentData,
|
||||||
|
}).catch((_err) => {
|
||||||
|
throw new Error(`Failed to send webhook to ${webhookUrl}`);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
return Promise.all(webhookPromises);
|
||||||
|
} catch (err) {
|
||||||
|
throw new Error(`Failed to trigger webhook`);
|
||||||
|
}
|
||||||
|
};
|
||||||
Reference in New Issue
Block a user