mirror of
https://github.com/documenso/documenso.git
synced 2025-11-13 08:13:56 +10:00
feat: add recipient creation
This commit is contained in:
@ -1,5 +1,17 @@
|
|||||||
|
import type { NextApiRequest, NextApiResponse } from 'next';
|
||||||
|
|
||||||
import { createNextRouter } from '@documenso/api/next';
|
import { createNextRouter } from '@documenso/api/next';
|
||||||
import { ApiContractV1 } from '@documenso/api/v1/contract';
|
import { ApiContractV1 } from '@documenso/api/v1/contract';
|
||||||
import { ApiContractV1Implementation } from '@documenso/api/v1/implementation';
|
import { ApiContractV1Implementation } from '@documenso/api/v1/implementation';
|
||||||
|
|
||||||
export default createNextRouter(ApiContractV1, ApiContractV1Implementation);
|
const nextRouteHandler = createNextRouter(ApiContractV1, ApiContractV1Implementation, {
|
||||||
|
responseValidation: false,
|
||||||
|
});
|
||||||
|
|
||||||
|
export default async function handler(req: NextApiRequest, res: NextApiResponse) {
|
||||||
|
// TODO: Dirty hack to make ts-rest handler work with next.js in a more intuitive way.
|
||||||
|
req.query['ts-rest'] = Array.isArray(req.query['ts-rest']) ? req.query['ts-rest'] : []; // Make `ts-rest` an array.
|
||||||
|
req.query['ts-rest'].unshift('api', 'v1'); // Prepend our base path to the array.
|
||||||
|
|
||||||
|
return await nextRouteHandler(req, res);
|
||||||
|
}
|
||||||
|
|||||||
19
package-lock.json
generated
19
package-lock.json
generated
@ -6718,8 +6718,7 @@
|
|||||||
"node_modules/@types/prop-types": {
|
"node_modules/@types/prop-types": {
|
||||||
"version": "15.7.11",
|
"version": "15.7.11",
|
||||||
"resolved": "https://registry.npmjs.org/@types/prop-types/-/prop-types-15.7.11.tgz",
|
"resolved": "https://registry.npmjs.org/@types/prop-types/-/prop-types-15.7.11.tgz",
|
||||||
"integrity": "sha512-ga8y9v9uyeiLdpKddhxYQkxNDrfvuPrlFb0N1qnZZByvcElJaXthF1UhvCh9TLWJBEHeNtdnbysW7Y6Uq8CVng==",
|
"integrity": "sha512-ga8y9v9uyeiLdpKddhxYQkxNDrfvuPrlFb0N1qnZZByvcElJaXthF1UhvCh9TLWJBEHeNtdnbysW7Y6Uq8CVng=="
|
||||||
"devOptional": true
|
|
||||||
},
|
},
|
||||||
"node_modules/@types/ramda": {
|
"node_modules/@types/ramda": {
|
||||||
"version": "0.29.9",
|
"version": "0.29.9",
|
||||||
@ -6733,7 +6732,6 @@
|
|||||||
"version": "18.2.18",
|
"version": "18.2.18",
|
||||||
"resolved": "https://registry.npmjs.org/@types/react/-/react-18.2.18.tgz",
|
"resolved": "https://registry.npmjs.org/@types/react/-/react-18.2.18.tgz",
|
||||||
"integrity": "sha512-da4NTSeBv/P34xoZPhtcLkmZuJ+oYaCxHmyHzwaDQo9RQPBeXV+06gEk2FpqEcsX9XrnNLvRpVh6bdavDSjtiQ==",
|
"integrity": "sha512-da4NTSeBv/P34xoZPhtcLkmZuJ+oYaCxHmyHzwaDQo9RQPBeXV+06gEk2FpqEcsX9XrnNLvRpVh6bdavDSjtiQ==",
|
||||||
"devOptional": true,
|
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@types/prop-types": "*",
|
"@types/prop-types": "*",
|
||||||
"@types/scheduler": "*",
|
"@types/scheduler": "*",
|
||||||
@ -6757,14 +6755,21 @@
|
|||||||
"node_modules/@types/scheduler": {
|
"node_modules/@types/scheduler": {
|
||||||
"version": "0.16.8",
|
"version": "0.16.8",
|
||||||
"resolved": "https://registry.npmjs.org/@types/scheduler/-/scheduler-0.16.8.tgz",
|
"resolved": "https://registry.npmjs.org/@types/scheduler/-/scheduler-0.16.8.tgz",
|
||||||
"integrity": "sha512-WZLiwShhwLRmeV6zH+GkbOFT6Z6VklCItrDioxUnv+u4Ll+8vKeFySoFyK/0ctcRpOmwAicELfmys1sDc/Rw+A==",
|
"integrity": "sha512-WZLiwShhwLRmeV6zH+GkbOFT6Z6VklCItrDioxUnv+u4Ll+8vKeFySoFyK/0ctcRpOmwAicELfmys1sDc/Rw+A=="
|
||||||
"devOptional": true
|
|
||||||
},
|
},
|
||||||
"node_modules/@types/semver": {
|
"node_modules/@types/semver": {
|
||||||
"version": "7.5.6",
|
"version": "7.5.6",
|
||||||
"resolved": "https://registry.npmjs.org/@types/semver/-/semver-7.5.6.tgz",
|
"resolved": "https://registry.npmjs.org/@types/semver/-/semver-7.5.6.tgz",
|
||||||
"integrity": "sha512-dn1l8LaMea/IjDoHNd9J52uBbInB796CDffS6VdIxvqYCPSG0V0DzHp76GpaWnlhg88uYyPbXCDIowa86ybd5A=="
|
"integrity": "sha512-dn1l8LaMea/IjDoHNd9J52uBbInB796CDffS6VdIxvqYCPSG0V0DzHp76GpaWnlhg88uYyPbXCDIowa86ybd5A=="
|
||||||
},
|
},
|
||||||
|
"node_modules/@types/swagger-ui-react": {
|
||||||
|
"version": "4.18.3",
|
||||||
|
"resolved": "https://registry.npmjs.org/@types/swagger-ui-react/-/swagger-ui-react-4.18.3.tgz",
|
||||||
|
"integrity": "sha512-Mo/R7IjDVwtiFPs84pWvh5pI9iyNGBjmfielxqbOh2Jv+8WVSDVe8Nu25kb5BOuV2xmGS3o33jr6nwDJMBcX+Q==",
|
||||||
|
"dependencies": {
|
||||||
|
"@types/react": "*"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/@types/unist": {
|
"node_modules/@types/unist": {
|
||||||
"version": "2.0.10",
|
"version": "2.0.10",
|
||||||
"resolved": "https://registry.npmjs.org/@types/unist/-/unist-2.0.10.tgz",
|
"resolved": "https://registry.npmjs.org/@types/unist/-/unist-2.0.10.tgz",
|
||||||
@ -8729,8 +8734,7 @@
|
|||||||
"node_modules/csstype": {
|
"node_modules/csstype": {
|
||||||
"version": "3.1.2",
|
"version": "3.1.2",
|
||||||
"resolved": "https://registry.npmjs.org/csstype/-/csstype-3.1.2.tgz",
|
"resolved": "https://registry.npmjs.org/csstype/-/csstype-3.1.2.tgz",
|
||||||
"integrity": "sha512-I7K1Uu0MBPzaFKg4nI5Q7Vs2t+3gWWW648spaF+Rg7pI9ds18Ugn+lvg4SHczUdKlHI5LWBXyqfS8+DufyBsgQ==",
|
"integrity": "sha512-I7K1Uu0MBPzaFKg4nI5Q7Vs2t+3gWWW648spaF+Rg7pI9ds18Ugn+lvg4SHczUdKlHI5LWBXyqfS8+DufyBsgQ=="
|
||||||
"devOptional": true
|
|
||||||
},
|
},
|
||||||
"node_modules/d3-array": {
|
"node_modules/d3-array": {
|
||||||
"version": "3.2.4",
|
"version": "3.2.4",
|
||||||
@ -20739,6 +20743,7 @@
|
|||||||
"@ts-rest/core": "^3.30.5",
|
"@ts-rest/core": "^3.30.5",
|
||||||
"@ts-rest/next": "^3.30.5",
|
"@ts-rest/next": "^3.30.5",
|
||||||
"@ts-rest/open-api": "^3.33.0",
|
"@ts-rest/open-api": "^3.33.0",
|
||||||
|
"@types/swagger-ui-react": "^4.18.3",
|
||||||
"luxon": "^3.4.0",
|
"luxon": "^3.4.0",
|
||||||
"superjson": "^1.13.1",
|
"superjson": "^1.13.1",
|
||||||
"swagger-ui-react": "^5.11.0",
|
"swagger-ui-react": "^5.11.0",
|
||||||
|
|||||||
@ -20,6 +20,7 @@
|
|||||||
"@ts-rest/core": "^3.30.5",
|
"@ts-rest/core": "^3.30.5",
|
||||||
"@ts-rest/next": "^3.30.5",
|
"@ts-rest/next": "^3.30.5",
|
||||||
"@ts-rest/open-api": "^3.33.0",
|
"@ts-rest/open-api": "^3.33.0",
|
||||||
|
"@types/swagger-ui-react": "^4.18.3",
|
||||||
"luxon": "^3.4.0",
|
"luxon": "^3.4.0",
|
||||||
"superjson": "^1.13.1",
|
"superjson": "^1.13.1",
|
||||||
"swagger-ui-react": "^5.11.0",
|
"swagger-ui-react": "^5.11.0",
|
||||||
|
|||||||
@ -1,3 +1,5 @@
|
|||||||
|
'use client';
|
||||||
|
|
||||||
import SwaggerUI from 'swagger-ui-react';
|
import SwaggerUI from 'swagger-ui-react';
|
||||||
import 'swagger-ui-react/swagger-ui.css';
|
import 'swagger-ui-react/swagger-ui.css';
|
||||||
|
|
||||||
|
|||||||
@ -4,9 +4,11 @@ import {
|
|||||||
ZSendDocumentForSigningMutationSchema as SendDocumentMutationSchema,
|
ZSendDocumentForSigningMutationSchema as SendDocumentMutationSchema,
|
||||||
ZAuthorizationHeadersSchema,
|
ZAuthorizationHeadersSchema,
|
||||||
ZCreateDocumentMutationSchema,
|
ZCreateDocumentMutationSchema,
|
||||||
|
ZCreateRecipientMutationSchema,
|
||||||
ZDeleteDocumentMutationSchema,
|
ZDeleteDocumentMutationSchema,
|
||||||
ZGetDocumentsQuerySchema,
|
ZGetDocumentsQuerySchema,
|
||||||
ZSuccessfulDocumentResponseSchema,
|
ZSuccessfulDocumentResponseSchema,
|
||||||
|
ZSuccessfulRecipientResponseSchema,
|
||||||
ZSuccessfulResponseSchema,
|
ZSuccessfulResponseSchema,
|
||||||
ZSuccessfulSigningResponseSchema,
|
ZSuccessfulSigningResponseSchema,
|
||||||
ZUnsuccessfulResponseSchema,
|
ZUnsuccessfulResponseSchema,
|
||||||
@ -19,7 +21,7 @@ export const ApiContractV1 = c.router(
|
|||||||
{
|
{
|
||||||
getDocuments: {
|
getDocuments: {
|
||||||
method: 'GET',
|
method: 'GET',
|
||||||
path: '/documents',
|
path: '/api/v1/documents',
|
||||||
query: ZGetDocumentsQuerySchema,
|
query: ZGetDocumentsQuerySchema,
|
||||||
responses: {
|
responses: {
|
||||||
200: ZSuccessfulResponseSchema,
|
200: ZSuccessfulResponseSchema,
|
||||||
@ -31,7 +33,7 @@ export const ApiContractV1 = c.router(
|
|||||||
|
|
||||||
getDocument: {
|
getDocument: {
|
||||||
method: 'GET',
|
method: 'GET',
|
||||||
path: `/documents/:id`,
|
path: '/api/v1/documents/:id',
|
||||||
responses: {
|
responses: {
|
||||||
200: ZSuccessfulDocumentResponseSchema,
|
200: ZSuccessfulDocumentResponseSchema,
|
||||||
401: ZUnsuccessfulResponseSchema,
|
401: ZUnsuccessfulResponseSchema,
|
||||||
@ -42,7 +44,7 @@ export const ApiContractV1 = c.router(
|
|||||||
|
|
||||||
createDocument: {
|
createDocument: {
|
||||||
method: 'POST',
|
method: 'POST',
|
||||||
path: '/documents',
|
path: '/api/v1/documents',
|
||||||
body: ZCreateDocumentMutationSchema,
|
body: ZCreateDocumentMutationSchema,
|
||||||
responses: {
|
responses: {
|
||||||
200: ZUploadDocumentSuccessfulSchema,
|
200: ZUploadDocumentSuccessfulSchema,
|
||||||
@ -53,8 +55,8 @@ export const ApiContractV1 = c.router(
|
|||||||
},
|
},
|
||||||
|
|
||||||
sendDocument: {
|
sendDocument: {
|
||||||
method: 'PATCH',
|
method: 'POST',
|
||||||
path: '/documents/:id/send',
|
path: '/api/v1/documents/:id/send',
|
||||||
body: SendDocumentMutationSchema,
|
body: SendDocumentMutationSchema,
|
||||||
responses: {
|
responses: {
|
||||||
200: ZSuccessfulSigningResponseSchema,
|
200: ZSuccessfulSigningResponseSchema,
|
||||||
@ -68,7 +70,7 @@ export const ApiContractV1 = c.router(
|
|||||||
|
|
||||||
deleteDocument: {
|
deleteDocument: {
|
||||||
method: 'DELETE',
|
method: 'DELETE',
|
||||||
path: `/documents/:id`,
|
path: '/api/v1/documents/:id',
|
||||||
body: ZDeleteDocumentMutationSchema,
|
body: ZDeleteDocumentMutationSchema,
|
||||||
responses: {
|
responses: {
|
||||||
200: ZSuccessfulDocumentResponseSchema,
|
200: ZSuccessfulDocumentResponseSchema,
|
||||||
@ -77,6 +79,20 @@ export const ApiContractV1 = c.router(
|
|||||||
},
|
},
|
||||||
summary: 'Delete a document',
|
summary: 'Delete a document',
|
||||||
},
|
},
|
||||||
|
|
||||||
|
createRecipient: {
|
||||||
|
method: 'POST',
|
||||||
|
path: '/api/v1/documents/:id/recipients',
|
||||||
|
body: ZCreateRecipientMutationSchema,
|
||||||
|
responses: {
|
||||||
|
200: ZSuccessfulRecipientResponseSchema,
|
||||||
|
400: ZUnsuccessfulResponseSchema,
|
||||||
|
401: ZUnsuccessfulResponseSchema,
|
||||||
|
404: ZUnsuccessfulResponseSchema,
|
||||||
|
500: ZUnsuccessfulResponseSchema,
|
||||||
|
},
|
||||||
|
summary: 'Create a recipient for a document',
|
||||||
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
baseHeaders: ZAuthorizationHeadersSchema,
|
baseHeaders: ZAuthorizationHeadersSchema,
|
||||||
|
|||||||
@ -1,13 +1,13 @@
|
|||||||
import { createNextRoute } from '@ts-rest/next';
|
import { createNextRoute } from '@ts-rest/next';
|
||||||
|
|
||||||
import { upsertDocumentMeta } from '@documenso/lib/server-only/document-meta/upsert-document-meta';
|
|
||||||
import { deleteDocument } from '@documenso/lib/server-only/document/delete-document';
|
import { deleteDocument } from '@documenso/lib/server-only/document/delete-document';
|
||||||
import { findDocuments } from '@documenso/lib/server-only/document/find-documents';
|
import { findDocuments } from '@documenso/lib/server-only/document/find-documents';
|
||||||
import { getDocumentById } from '@documenso/lib/server-only/document/get-document-by-id';
|
import { getDocumentById } from '@documenso/lib/server-only/document/get-document-by-id';
|
||||||
import { sendDocument } from '@documenso/lib/server-only/document/send-document';
|
import { sendDocument } from '@documenso/lib/server-only/document/send-document';
|
||||||
import { setFieldsForDocument } from '@documenso/lib/server-only/field/set-fields-for-document';
|
import { getRecipientsForDocument } from '@documenso/lib/server-only/recipient/get-recipients-for-document';
|
||||||
import { setRecipientsForDocument } from '@documenso/lib/server-only/recipient/set-recipients-for-document';
|
import { setRecipientsForDocument } from '@documenso/lib/server-only/recipient/set-recipients-for-document';
|
||||||
import { getPresignPostUrl } from '@documenso/lib/universal/upload/server-actions';
|
import { getPresignPostUrl } from '@documenso/lib/universal/upload/server-actions';
|
||||||
|
import { DocumentStatus } from '@documenso/prisma/client';
|
||||||
|
|
||||||
import { ApiContractV1 } from './contract';
|
import { ApiContractV1 } from './contract';
|
||||||
import { authenticatedMiddleware } from './middleware/authenticated';
|
import { authenticatedMiddleware } from './middleware/authenticated';
|
||||||
@ -99,7 +99,6 @@ export const ApiContractV1Implementation = createNextRoute(ApiContractV1, {
|
|||||||
|
|
||||||
sendDocument: authenticatedMiddleware(async (args, user) => {
|
sendDocument: authenticatedMiddleware(async (args, user) => {
|
||||||
const { id } = args.params;
|
const { id } = args.params;
|
||||||
const { body } = args;
|
|
||||||
|
|
||||||
const document = await getDocumentById({ id: Number(id), userId: user.id });
|
const document = await getDocumentById({ id: Number(id), userId: user.id });
|
||||||
|
|
||||||
@ -122,38 +121,38 @@ export const ApiContractV1Implementation = createNextRoute(ApiContractV1, {
|
|||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
await setRecipientsForDocument({
|
// await setRecipientsForDocument({
|
||||||
userId: user.id,
|
// userId: user.id,
|
||||||
documentId: Number(id),
|
// documentId: Number(id),
|
||||||
recipients: [
|
// recipients: [
|
||||||
{
|
// {
|
||||||
email: body.signerEmail,
|
// email: body.signerEmail,
|
||||||
name: body.signerName ?? '',
|
// name: body.signerName ?? '',
|
||||||
},
|
// },
|
||||||
],
|
// ],
|
||||||
});
|
// });
|
||||||
|
|
||||||
await setFieldsForDocument({
|
// await setFieldsForDocument({
|
||||||
documentId: Number(id),
|
// documentId: Number(id),
|
||||||
userId: user.id,
|
// userId: user.id,
|
||||||
fields: body.fields.map((field) => ({
|
// fields: body.fields.map((field) => ({
|
||||||
signerEmail: body.signerEmail,
|
// signerEmail: body.signerEmail,
|
||||||
type: field.fieldType,
|
// type: field.fieldType,
|
||||||
pageNumber: field.pageNumber,
|
// pageNumber: field.pageNumber,
|
||||||
pageX: field.pageX,
|
// pageX: field.pageX,
|
||||||
pageY: field.pageY,
|
// pageY: field.pageY,
|
||||||
pageWidth: field.pageWidth,
|
// pageWidth: field.pageWidth,
|
||||||
pageHeight: field.pageHeight,
|
// pageHeight: field.pageHeight,
|
||||||
})),
|
// })),
|
||||||
});
|
// });
|
||||||
|
|
||||||
if (body.emailBody || body.emailSubject) {
|
// if (body.emailBody || body.emailSubject) {
|
||||||
await upsertDocumentMeta({
|
// await upsertDocumentMeta({
|
||||||
documentId: Number(id),
|
// documentId: Number(id),
|
||||||
subject: body.emailSubject ?? '',
|
// subject: body.emailSubject ?? '',
|
||||||
message: body.emailBody ?? '',
|
// message: body.emailBody ?? '',
|
||||||
});
|
// });
|
||||||
}
|
// }
|
||||||
|
|
||||||
await sendDocument({
|
await sendDocument({
|
||||||
documentId: Number(id),
|
documentId: Number(id),
|
||||||
@ -175,4 +174,80 @@ export const ApiContractV1Implementation = createNextRoute(ApiContractV1, {
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
}),
|
}),
|
||||||
|
|
||||||
|
createRecipient: authenticatedMiddleware(async (args, user) => {
|
||||||
|
const { id: documentId } = args.params;
|
||||||
|
const { name, email } = args.body;
|
||||||
|
|
||||||
|
const document = await getDocumentById({
|
||||||
|
id: Number(documentId),
|
||||||
|
userId: user.id,
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!document) {
|
||||||
|
return {
|
||||||
|
status: 404,
|
||||||
|
body: {
|
||||||
|
message: 'Document not found',
|
||||||
|
},
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
if (document.status === DocumentStatus.COMPLETED) {
|
||||||
|
return {
|
||||||
|
status: 400,
|
||||||
|
body: {
|
||||||
|
message: 'Document is already completed',
|
||||||
|
},
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
const recipients = await getRecipientsForDocument({
|
||||||
|
documentId: Number(documentId),
|
||||||
|
userId: user.id,
|
||||||
|
});
|
||||||
|
|
||||||
|
const recipientAlreadyExists = recipients.some((recipient) => recipient.email === email);
|
||||||
|
|
||||||
|
if (recipientAlreadyExists) {
|
||||||
|
return {
|
||||||
|
status: 400,
|
||||||
|
body: {
|
||||||
|
message: 'Recipient already exists',
|
||||||
|
},
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
const newRecipients = await setRecipientsForDocument({
|
||||||
|
documentId: Number(documentId),
|
||||||
|
userId: user.id,
|
||||||
|
recipients: [
|
||||||
|
...recipients,
|
||||||
|
{
|
||||||
|
email,
|
||||||
|
name,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
});
|
||||||
|
|
||||||
|
const newRecipient = newRecipients.find((recipient) => recipient.email === email);
|
||||||
|
|
||||||
|
if (!newRecipient) {
|
||||||
|
throw new Error('Recipient not found');
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
status: 200,
|
||||||
|
body: newRecipient,
|
||||||
|
};
|
||||||
|
} catch (err) {
|
||||||
|
return {
|
||||||
|
status: 500,
|
||||||
|
body: {
|
||||||
|
message: 'An error has occured while creating the recipient',
|
||||||
|
},
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}),
|
||||||
});
|
});
|
||||||
|
|||||||
@ -1,6 +1,6 @@
|
|||||||
import { z } from 'zod';
|
import { z } from 'zod';
|
||||||
|
|
||||||
import { FieldType } from '@documenso/prisma/client';
|
import { ReadStatus, SendStatus, SigningStatus } from '@documenso/prisma/client';
|
||||||
|
|
||||||
export const ZGetDocumentsQuerySchema = z.object({
|
export const ZGetDocumentsQuerySchema = z.object({
|
||||||
page: z.string().optional(),
|
page: z.string().optional(),
|
||||||
@ -9,9 +9,9 @@ export const ZGetDocumentsQuerySchema = z.object({
|
|||||||
|
|
||||||
export type TGetDocumentsQuerySchema = z.infer<typeof ZGetDocumentsQuerySchema>;
|
export type TGetDocumentsQuerySchema = z.infer<typeof ZGetDocumentsQuerySchema>;
|
||||||
|
|
||||||
export const ZDeleteDocumentMutationSchema = z.string();
|
export const ZDeleteDocumentMutationSchema = null;
|
||||||
|
|
||||||
export type TDeleteDocumentMutationSchema = z.infer<typeof ZDeleteDocumentMutationSchema>;
|
export type TDeleteDocumentMutationSchema = typeof ZDeleteDocumentMutationSchema;
|
||||||
|
|
||||||
export const ZSuccessfulDocumentResponseSchema = z.object({
|
export const ZSuccessfulDocumentResponseSchema = z.object({
|
||||||
id: z.number(),
|
id: z.number(),
|
||||||
@ -26,26 +26,9 @@ export const ZSuccessfulDocumentResponseSchema = z.object({
|
|||||||
|
|
||||||
export type TSuccessfulDocumentResponseSchema = z.infer<typeof ZSuccessfulDocumentResponseSchema>;
|
export type TSuccessfulDocumentResponseSchema = z.infer<typeof ZSuccessfulDocumentResponseSchema>;
|
||||||
|
|
||||||
export const ZSendDocumentForSigningMutationSchema = z.object({
|
export const ZSendDocumentForSigningMutationSchema = null;
|
||||||
signerEmail: z.string(),
|
|
||||||
signerName: z.string().optional(),
|
|
||||||
emailSubject: z.string().optional(),
|
|
||||||
emailBody: z.string().optional(),
|
|
||||||
fields: z.array(
|
|
||||||
z.object({
|
|
||||||
fieldType: z.nativeEnum(FieldType),
|
|
||||||
pageNumber: z.number(),
|
|
||||||
pageX: z.number(),
|
|
||||||
pageY: z.number(),
|
|
||||||
pageWidth: z.number(),
|
|
||||||
pageHeight: z.number(),
|
|
||||||
}),
|
|
||||||
),
|
|
||||||
});
|
|
||||||
|
|
||||||
export type TSendDocumentForSigningMutationSchema = z.infer<
|
export type TSendDocumentForSigningMutationSchema = typeof ZSendDocumentForSigningMutationSchema;
|
||||||
typeof ZSendDocumentForSigningMutationSchema
|
|
||||||
>;
|
|
||||||
|
|
||||||
export const ZUploadDocumentSuccessfulSchema = z.object({
|
export const ZUploadDocumentSuccessfulSchema = z.object({
|
||||||
url: z.string(),
|
url: z.string(),
|
||||||
@ -61,6 +44,29 @@ export const ZCreateDocumentMutationSchema = z.object({
|
|||||||
|
|
||||||
export type TCreateDocumentMutationSchema = z.infer<typeof ZCreateDocumentMutationSchema>;
|
export type TCreateDocumentMutationSchema = z.infer<typeof ZCreateDocumentMutationSchema>;
|
||||||
|
|
||||||
|
export const ZCreateRecipientMutationSchema = z.object({
|
||||||
|
name: z.string().min(1),
|
||||||
|
email: z.string().email().min(1),
|
||||||
|
});
|
||||||
|
|
||||||
|
export type TCreateRecipientMutationSchema = z.infer<typeof ZCreateRecipientMutationSchema>;
|
||||||
|
|
||||||
|
export const ZSuccessfulRecipientResponseSchema = z.object({
|
||||||
|
id: z.number(),
|
||||||
|
documentId: z.number(),
|
||||||
|
email: z.string().email().min(1),
|
||||||
|
name: z.string(),
|
||||||
|
token: z.string(),
|
||||||
|
// !: Not used for now
|
||||||
|
// expired: z.string(),
|
||||||
|
signedAt: z.date().nullable(),
|
||||||
|
readStatus: z.nativeEnum(ReadStatus),
|
||||||
|
signingStatus: z.nativeEnum(SigningStatus),
|
||||||
|
sendStatus: z.nativeEnum(SendStatus),
|
||||||
|
});
|
||||||
|
|
||||||
|
export type TSuccessfulRecipientResponseSchema = z.infer<typeof ZSuccessfulRecipientResponseSchema>;
|
||||||
|
|
||||||
export const ZSuccessfulResponseSchema = z.object({
|
export const ZSuccessfulResponseSchema = z.object({
|
||||||
documents: ZSuccessfulDocumentResponseSchema.array(),
|
documents: ZSuccessfulDocumentResponseSchema.array(),
|
||||||
totalPages: z.number(),
|
totalPages: z.number(),
|
||||||
|
|||||||
@ -6,7 +6,7 @@ export type GetApiTokenByIdOptions = {
|
|||||||
};
|
};
|
||||||
|
|
||||||
export const getApiTokenById = async ({ id, userId }: GetApiTokenByIdOptions) => {
|
export const getApiTokenById = async ({ id, userId }: GetApiTokenByIdOptions) => {
|
||||||
return prisma.apiToken.findFirstOrThrow({
|
return await prisma.apiToken.findFirstOrThrow({
|
||||||
where: {
|
where: {
|
||||||
id,
|
id,
|
||||||
userId,
|
userId,
|
||||||
|
|||||||
@ -1,80 +0,0 @@
|
|||||||
import { initContract } from '@ts-rest/core';
|
|
||||||
|
|
||||||
import {
|
|
||||||
AuthorizationHeadersSchema,
|
|
||||||
CreateDocumentMutationSchema,
|
|
||||||
DeleteDocumentMutationSchema,
|
|
||||||
GetDocumentsQuerySchema,
|
|
||||||
SendDocumentForSigningMutationSchema,
|
|
||||||
SuccessfulDocumentResponseSchema,
|
|
||||||
SuccessfulResponseSchema,
|
|
||||||
SuccessfulSigningResponseSchema,
|
|
||||||
UnsuccessfulResponseSchema,
|
|
||||||
UploadDocumentSuccessfulSchema,
|
|
||||||
} from './schema';
|
|
||||||
|
|
||||||
const c = initContract();
|
|
||||||
|
|
||||||
export const contract = c.router(
|
|
||||||
{
|
|
||||||
getDocuments: {
|
|
||||||
method: 'GET',
|
|
||||||
path: '/documents',
|
|
||||||
query: GetDocumentsQuerySchema,
|
|
||||||
responses: {
|
|
||||||
200: SuccessfulResponseSchema,
|
|
||||||
401: UnsuccessfulResponseSchema,
|
|
||||||
404: UnsuccessfulResponseSchema,
|
|
||||||
},
|
|
||||||
summary: 'Get all documents',
|
|
||||||
},
|
|
||||||
getDocument: {
|
|
||||||
method: 'GET',
|
|
||||||
path: `/documents/:id`,
|
|
||||||
responses: {
|
|
||||||
200: SuccessfulDocumentResponseSchema,
|
|
||||||
401: UnsuccessfulResponseSchema,
|
|
||||||
404: UnsuccessfulResponseSchema,
|
|
||||||
},
|
|
||||||
summary: 'Get a single document',
|
|
||||||
},
|
|
||||||
createDocument: {
|
|
||||||
method: 'POST',
|
|
||||||
path: '/documents',
|
|
||||||
body: CreateDocumentMutationSchema,
|
|
||||||
responses: {
|
|
||||||
200: UploadDocumentSuccessfulSchema,
|
|
||||||
401: UnsuccessfulResponseSchema,
|
|
||||||
404: UnsuccessfulResponseSchema,
|
|
||||||
},
|
|
||||||
summary: 'Upload a new document and get a presigned URL',
|
|
||||||
},
|
|
||||||
sendDocumentForSigning: {
|
|
||||||
method: 'PATCH',
|
|
||||||
path: '/documents/:id/send',
|
|
||||||
body: SendDocumentForSigningMutationSchema,
|
|
||||||
responses: {
|
|
||||||
200: SuccessfulSigningResponseSchema,
|
|
||||||
400: UnsuccessfulResponseSchema,
|
|
||||||
401: UnsuccessfulResponseSchema,
|
|
||||||
404: UnsuccessfulResponseSchema,
|
|
||||||
500: UnsuccessfulResponseSchema,
|
|
||||||
},
|
|
||||||
summary: 'Send a document for signing',
|
|
||||||
},
|
|
||||||
deleteDocument: {
|
|
||||||
method: 'DELETE',
|
|
||||||
path: `/documents/:id`,
|
|
||||||
body: DeleteDocumentMutationSchema,
|
|
||||||
responses: {
|
|
||||||
200: SuccessfulDocumentResponseSchema,
|
|
||||||
401: UnsuccessfulResponseSchema,
|
|
||||||
404: UnsuccessfulResponseSchema,
|
|
||||||
},
|
|
||||||
summary: 'Delete a document',
|
|
||||||
},
|
|
||||||
},
|
|
||||||
{
|
|
||||||
baseHeaders: AuthorizationHeadersSchema,
|
|
||||||
},
|
|
||||||
);
|
|
||||||
@ -1,65 +0,0 @@
|
|||||||
import { z } from 'zod';
|
|
||||||
|
|
||||||
import { FieldType } from '@documenso/prisma/client';
|
|
||||||
|
|
||||||
export const GetDocumentsQuerySchema = z.object({
|
|
||||||
page: z.string().optional(),
|
|
||||||
perPage: z.string().optional(),
|
|
||||||
});
|
|
||||||
|
|
||||||
export const DeleteDocumentMutationSchema = z.string();
|
|
||||||
|
|
||||||
export const SuccessfulDocumentResponseSchema = z.object({
|
|
||||||
id: z.number(),
|
|
||||||
userId: z.number(),
|
|
||||||
title: z.string(),
|
|
||||||
status: z.string(),
|
|
||||||
documentDataId: z.string(),
|
|
||||||
createdAt: z.date(),
|
|
||||||
updatedAt: z.date(),
|
|
||||||
completedAt: z.date().nullable(),
|
|
||||||
});
|
|
||||||
|
|
||||||
export const SendDocumentForSigningMutationSchema = z.object({
|
|
||||||
signerEmail: z.string(),
|
|
||||||
signerName: z.string().optional(),
|
|
||||||
emailSubject: z.string().optional(),
|
|
||||||
emailBody: z.string().optional(),
|
|
||||||
fields: z.array(
|
|
||||||
z.object({
|
|
||||||
fieldType: z.nativeEnum(FieldType),
|
|
||||||
pageNumber: z.number(),
|
|
||||||
pageX: z.number(),
|
|
||||||
pageY: z.number(),
|
|
||||||
pageWidth: z.number(),
|
|
||||||
pageHeight: z.number(),
|
|
||||||
}),
|
|
||||||
),
|
|
||||||
});
|
|
||||||
|
|
||||||
export const UploadDocumentSuccessfulSchema = z.object({
|
|
||||||
url: z.string(),
|
|
||||||
key: z.string(),
|
|
||||||
});
|
|
||||||
|
|
||||||
export const CreateDocumentMutationSchema = z.object({
|
|
||||||
fileName: z.string(),
|
|
||||||
contentType: z.string().default('PDF'),
|
|
||||||
});
|
|
||||||
|
|
||||||
export const SuccessfulResponseSchema = z.object({
|
|
||||||
documents: SuccessfulDocumentResponseSchema.array(),
|
|
||||||
totalPages: z.number(),
|
|
||||||
});
|
|
||||||
|
|
||||||
export const SuccessfulSigningResponseSchema = z.object({
|
|
||||||
message: z.string(),
|
|
||||||
});
|
|
||||||
|
|
||||||
export const UnsuccessfulResponseSchema = z.object({
|
|
||||||
message: z.string(),
|
|
||||||
});
|
|
||||||
|
|
||||||
export const AuthorizationHeadersSchema = z.object({
|
|
||||||
authorization: z.string(),
|
|
||||||
});
|
|
||||||
@ -1,3 +0,0 @@
|
|||||||
import { createNextRoute, createNextRouter } from '@ts-rest/next';
|
|
||||||
|
|
||||||
export { createNextRoute, createNextRouter };
|
|
||||||
Reference in New Issue
Block a user