mirror of
https://github.com/documenso/documenso.git
synced 2025-11-13 00:03:33 +10:00
Compare commits
1 Commits
release
...
a66a56042c
| Author | SHA1 | Date | |
|---|---|---|---|
| a66a56042c |
@ -20,12 +20,12 @@ import {
|
|||||||
getEnvelopeWhereInput,
|
getEnvelopeWhereInput,
|
||||||
} from '@documenso/lib/server-only/envelope/get-envelope-by-id';
|
} from '@documenso/lib/server-only/envelope/get-envelope-by-id';
|
||||||
import { deleteDocumentField } from '@documenso/lib/server-only/field/delete-document-field';
|
import { deleteDocumentField } from '@documenso/lib/server-only/field/delete-document-field';
|
||||||
import { updateDocumentFields } from '@documenso/lib/server-only/field/update-document-fields';
|
import { updateEnvelopeFields } from '@documenso/lib/server-only/field/update-envelope-fields';
|
||||||
import { insertFormValuesInPdf } from '@documenso/lib/server-only/pdf/insert-form-values-in-pdf';
|
import { insertFormValuesInPdf } from '@documenso/lib/server-only/pdf/insert-form-values-in-pdf';
|
||||||
import { deleteDocumentRecipient } from '@documenso/lib/server-only/recipient/delete-document-recipient';
|
import { deleteEnvelopeRecipient } from '@documenso/lib/server-only/recipient/delete-envelope-recipient';
|
||||||
import { getRecipientsForDocument } from '@documenso/lib/server-only/recipient/get-recipients-for-document';
|
import { getRecipientsForDocument } from '@documenso/lib/server-only/recipient/get-recipients-for-document';
|
||||||
import { setDocumentRecipients } from '@documenso/lib/server-only/recipient/set-document-recipients';
|
import { setDocumentRecipients } from '@documenso/lib/server-only/recipient/set-document-recipients';
|
||||||
import { updateDocumentRecipients } from '@documenso/lib/server-only/recipient/update-document-recipients';
|
import { updateEnvelopeRecipients } from '@documenso/lib/server-only/recipient/update-envelope-recipients';
|
||||||
import { createDocumentFromTemplate } from '@documenso/lib/server-only/template/create-document-from-template';
|
import { createDocumentFromTemplate } from '@documenso/lib/server-only/template/create-document-from-template';
|
||||||
import { deleteTemplate } from '@documenso/lib/server-only/template/delete-template';
|
import { deleteTemplate } from '@documenso/lib/server-only/template/delete-template';
|
||||||
import { findTemplates } from '@documenso/lib/server-only/template/find-templates';
|
import { findTemplates } from '@documenso/lib/server-only/template/find-templates';
|
||||||
@ -1285,7 +1285,7 @@ export const ApiContractV1Implementation = tsr.router(ApiContractV1, {
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
const updatedRecipient = await updateDocumentRecipients({
|
const updatedRecipient = await updateEnvelopeRecipients({
|
||||||
userId: user.id,
|
userId: user.id,
|
||||||
teamId: team.id,
|
teamId: team.id,
|
||||||
id: {
|
id: {
|
||||||
@ -1336,7 +1336,7 @@ export const ApiContractV1Implementation = tsr.router(ApiContractV1, {
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
const deletedRecipient = await deleteDocumentRecipient({
|
const deletedRecipient = await deleteEnvelopeRecipient({
|
||||||
userId: user.id,
|
userId: user.id,
|
||||||
teamId: team.id,
|
teamId: team.id,
|
||||||
recipientId: Number(recipientId),
|
recipientId: Number(recipientId),
|
||||||
@ -1634,10 +1634,13 @@ export const ApiContractV1Implementation = tsr.router(ApiContractV1, {
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
const { fields } = await updateDocumentFields({
|
const { fields } = await updateEnvelopeFields({
|
||||||
userId: user.id,
|
userId: user.id,
|
||||||
teamId: team.id,
|
teamId: team.id,
|
||||||
documentId: legacyDocumentId,
|
id: {
|
||||||
|
type: 'documentId',
|
||||||
|
id: legacyDocumentId,
|
||||||
|
},
|
||||||
fields: [
|
fields: [
|
||||||
{
|
{
|
||||||
id: Number(fieldId),
|
id: Number(fieldId),
|
||||||
|
|||||||
@ -11,7 +11,7 @@ export type GetFieldByIdOptions = {
|
|||||||
userId: number;
|
userId: number;
|
||||||
teamId: number;
|
teamId: number;
|
||||||
fieldId: number;
|
fieldId: number;
|
||||||
envelopeType: EnvelopeType;
|
envelopeType?: EnvelopeType;
|
||||||
};
|
};
|
||||||
|
|
||||||
export const getFieldById = async ({
|
export const getFieldById = async ({
|
||||||
@ -41,7 +41,7 @@ export const getFieldById = async ({
|
|||||||
type: 'envelopeId',
|
type: 'envelopeId',
|
||||||
id: field.envelopeId,
|
id: field.envelopeId,
|
||||||
},
|
},
|
||||||
type: envelopeType,
|
type: envelopeType ?? null,
|
||||||
userId,
|
userId,
|
||||||
teamId,
|
teamId,
|
||||||
});
|
});
|
||||||
|
|||||||
@ -10,18 +10,21 @@ import {
|
|||||||
import { prisma } from '@documenso/prisma';
|
import { prisma } from '@documenso/prisma';
|
||||||
|
|
||||||
import { AppError, AppErrorCode } from '../../errors/app-error';
|
import { AppError, AppErrorCode } from '../../errors/app-error';
|
||||||
|
import { type EnvelopeIdOptions } from '../../utils/envelope';
|
||||||
import { mapFieldToLegacyField } from '../../utils/fields';
|
import { mapFieldToLegacyField } from '../../utils/fields';
|
||||||
import { canRecipientFieldsBeModified } from '../../utils/recipients';
|
import { canRecipientFieldsBeModified } from '../../utils/recipients';
|
||||||
import { getEnvelopeWhereInput } from '../envelope/get-envelope-by-id';
|
import { getEnvelopeWhereInput } from '../envelope/get-envelope-by-id';
|
||||||
|
|
||||||
export interface UpdateDocumentFieldsOptions {
|
export interface UpdateEnvelopeFieldsOptions {
|
||||||
userId: number;
|
userId: number;
|
||||||
teamId: number;
|
teamId: number;
|
||||||
documentId: number;
|
id: EnvelopeIdOptions;
|
||||||
|
type?: EnvelopeType | null; // Only used to enforce the type.
|
||||||
fields: {
|
fields: {
|
||||||
id: number;
|
id: number;
|
||||||
type?: FieldType;
|
type?: FieldType;
|
||||||
pageNumber?: number;
|
pageNumber?: number;
|
||||||
|
envelopeItemId?: string;
|
||||||
pageX?: number;
|
pageX?: number;
|
||||||
pageY?: number;
|
pageY?: number;
|
||||||
width?: number;
|
width?: number;
|
||||||
@ -31,19 +34,17 @@ export interface UpdateDocumentFieldsOptions {
|
|||||||
requestMetadata: ApiRequestMetadata;
|
requestMetadata: ApiRequestMetadata;
|
||||||
}
|
}
|
||||||
|
|
||||||
export const updateDocumentFields = async ({
|
export const updateEnvelopeFields = async ({
|
||||||
userId,
|
userId,
|
||||||
teamId,
|
teamId,
|
||||||
documentId,
|
id,
|
||||||
|
type = null,
|
||||||
fields,
|
fields,
|
||||||
requestMetadata,
|
requestMetadata,
|
||||||
}: UpdateDocumentFieldsOptions) => {
|
}: UpdateEnvelopeFieldsOptions) => {
|
||||||
const { envelopeWhereInput } = await getEnvelopeWhereInput({
|
const { envelopeWhereInput } = await getEnvelopeWhereInput({
|
||||||
id: {
|
id,
|
||||||
type: 'documentId',
|
type,
|
||||||
id: documentId,
|
|
||||||
},
|
|
||||||
type: EnvelopeType.DOCUMENT,
|
|
||||||
userId,
|
userId,
|
||||||
teamId,
|
teamId,
|
||||||
});
|
});
|
||||||
@ -53,18 +54,19 @@ export const updateDocumentFields = async ({
|
|||||||
include: {
|
include: {
|
||||||
recipients: true,
|
recipients: true,
|
||||||
fields: true,
|
fields: true,
|
||||||
|
envelopeItems: true,
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
if (!envelope) {
|
if (!envelope) {
|
||||||
throw new AppError(AppErrorCode.NOT_FOUND, {
|
throw new AppError(AppErrorCode.NOT_FOUND, {
|
||||||
message: 'Document not found',
|
message: 'Envelope not found',
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
if (envelope.completedAt) {
|
if (envelope.completedAt) {
|
||||||
throw new AppError(AppErrorCode.INVALID_REQUEST, {
|
throw new AppError(AppErrorCode.INVALID_REQUEST, {
|
||||||
message: 'Document already complete',
|
message: 'Envelope already complete',
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -96,6 +98,29 @@ export const updateDocumentFields = async ({
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const fieldType = field.type || originalField.type;
|
||||||
|
const fieldMetaType = field.fieldMeta?.type || originalField.fieldMeta?.type;
|
||||||
|
|
||||||
|
// Not going to mess with V1 envelopes.
|
||||||
|
if (
|
||||||
|
envelope.internalVersion === 2 &&
|
||||||
|
fieldMetaType &&
|
||||||
|
fieldMetaType.toLowerCase() !== fieldType.toLowerCase()
|
||||||
|
) {
|
||||||
|
throw new AppError(AppErrorCode.INVALID_REQUEST, {
|
||||||
|
message: 'Field meta type does not match the field type',
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
if (
|
||||||
|
field.envelopeItemId &&
|
||||||
|
!envelope.envelopeItems.some((item) => item.id === field.envelopeItemId)
|
||||||
|
) {
|
||||||
|
throw new AppError(AppErrorCode.INVALID_REQUEST, {
|
||||||
|
message: 'Envelope item not found',
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
return {
|
return {
|
||||||
originalField,
|
originalField,
|
||||||
updateData: field,
|
updateData: field,
|
||||||
@ -118,12 +143,14 @@ export const updateDocumentFields = async ({
|
|||||||
width: updateData.width,
|
width: updateData.width,
|
||||||
height: updateData.height,
|
height: updateData.height,
|
||||||
fieldMeta: updateData.fieldMeta,
|
fieldMeta: updateData.fieldMeta,
|
||||||
|
envelopeItemId: updateData.envelopeItemId,
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// Handle field updated audit log.
|
||||||
|
if (envelope.type === EnvelopeType.DOCUMENT) {
|
||||||
const changes = diffFieldChanges(originalField, updatedField);
|
const changes = diffFieldChanges(originalField, updatedField);
|
||||||
|
|
||||||
// Handle field updated audit log.
|
|
||||||
if (changes.length > 0) {
|
if (changes.length > 0) {
|
||||||
await tx.documentAuditLog.create({
|
await tx.documentAuditLog.create({
|
||||||
data: createDocumentAuditLogData({
|
data: createDocumentAuditLogData({
|
||||||
@ -140,6 +167,7 @@ export const updateDocumentFields = async ({
|
|||||||
}),
|
}),
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return updatedField;
|
return updatedField;
|
||||||
}),
|
}),
|
||||||
@ -1,116 +0,0 @@
|
|||||||
import { EnvelopeType, type FieldType } from '@prisma/client';
|
|
||||||
|
|
||||||
import type { TFieldMetaSchema } from '@documenso/lib/types/field-meta';
|
|
||||||
import { prisma } from '@documenso/prisma';
|
|
||||||
|
|
||||||
import { AppError, AppErrorCode } from '../../errors/app-error';
|
|
||||||
import { mapFieldToLegacyField } from '../../utils/fields';
|
|
||||||
import { canRecipientFieldsBeModified } from '../../utils/recipients';
|
|
||||||
import { getEnvelopeWhereInput } from '../envelope/get-envelope-by-id';
|
|
||||||
|
|
||||||
export interface UpdateTemplateFieldsOptions {
|
|
||||||
userId: number;
|
|
||||||
teamId: number;
|
|
||||||
templateId: number;
|
|
||||||
fields: {
|
|
||||||
id: number;
|
|
||||||
type?: FieldType;
|
|
||||||
pageNumber?: number;
|
|
||||||
pageX?: number;
|
|
||||||
pageY?: number;
|
|
||||||
width?: number;
|
|
||||||
height?: number;
|
|
||||||
fieldMeta?: TFieldMetaSchema;
|
|
||||||
}[];
|
|
||||||
}
|
|
||||||
|
|
||||||
export const updateTemplateFields = async ({
|
|
||||||
userId,
|
|
||||||
teamId,
|
|
||||||
templateId,
|
|
||||||
fields,
|
|
||||||
}: UpdateTemplateFieldsOptions) => {
|
|
||||||
const { envelopeWhereInput } = await getEnvelopeWhereInput({
|
|
||||||
id: {
|
|
||||||
type: 'templateId',
|
|
||||||
id: templateId,
|
|
||||||
},
|
|
||||||
type: EnvelopeType.TEMPLATE,
|
|
||||||
userId,
|
|
||||||
teamId,
|
|
||||||
});
|
|
||||||
|
|
||||||
const envelope = await prisma.envelope.findFirst({
|
|
||||||
where: envelopeWhereInput,
|
|
||||||
include: {
|
|
||||||
recipients: true,
|
|
||||||
fields: true,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
if (!envelope) {
|
|
||||||
throw new AppError(AppErrorCode.NOT_FOUND, {
|
|
||||||
message: 'Document not found',
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
const fieldsToUpdate = fields.map((field) => {
|
|
||||||
const originalField = envelope.fields.find((existingField) => existingField.id === field.id);
|
|
||||||
|
|
||||||
if (!originalField) {
|
|
||||||
throw new AppError(AppErrorCode.NOT_FOUND, {
|
|
||||||
message: `Field with id ${field.id} not found`,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
const recipient = envelope.recipients.find(
|
|
||||||
(recipient) => recipient.id === originalField.recipientId,
|
|
||||||
);
|
|
||||||
|
|
||||||
// Each field MUST have a recipient associated with it.
|
|
||||||
if (!recipient) {
|
|
||||||
throw new AppError(AppErrorCode.INVALID_REQUEST, {
|
|
||||||
message: `Recipient attached to field ${field.id} not found`,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
// Check whether the recipient associated with the field can be modified.
|
|
||||||
if (!canRecipientFieldsBeModified(recipient, envelope.fields)) {
|
|
||||||
throw new AppError(AppErrorCode.INVALID_REQUEST, {
|
|
||||||
message:
|
|
||||||
'Cannot modify a field where the recipient has already interacted with the document',
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
return {
|
|
||||||
updateData: field,
|
|
||||||
};
|
|
||||||
});
|
|
||||||
|
|
||||||
const updatedFields = await prisma.$transaction(async (tx) => {
|
|
||||||
return await Promise.all(
|
|
||||||
fieldsToUpdate.map(async ({ updateData }) => {
|
|
||||||
const updatedField = await tx.field.update({
|
|
||||||
where: {
|
|
||||||
id: updateData.id,
|
|
||||||
},
|
|
||||||
data: {
|
|
||||||
type: updateData.type,
|
|
||||||
page: updateData.pageNumber,
|
|
||||||
positionX: updateData.pageX,
|
|
||||||
positionY: updateData.pageY,
|
|
||||||
width: updateData.width,
|
|
||||||
height: updateData.height,
|
|
||||||
fieldMeta: updateData.fieldMeta,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
return updatedField;
|
|
||||||
}),
|
|
||||||
);
|
|
||||||
});
|
|
||||||
|
|
||||||
return {
|
|
||||||
fields: updatedFields.map((field) => mapFieldToLegacyField(field, envelope)),
|
|
||||||
};
|
|
||||||
};
|
|
||||||
@ -15,7 +15,7 @@ import type { EnvelopeIdOptions } from '../../utils/envelope';
|
|||||||
import { mapRecipientToLegacyRecipient } from '../../utils/recipients';
|
import { mapRecipientToLegacyRecipient } from '../../utils/recipients';
|
||||||
import { getEnvelopeWhereInput } from '../envelope/get-envelope-by-id';
|
import { getEnvelopeWhereInput } from '../envelope/get-envelope-by-id';
|
||||||
|
|
||||||
export interface CreateDocumentRecipientsOptions {
|
export interface CreateEnvelopeRecipientsOptions {
|
||||||
userId: number;
|
userId: number;
|
||||||
teamId: number;
|
teamId: number;
|
||||||
id: EnvelopeIdOptions;
|
id: EnvelopeIdOptions;
|
||||||
@ -30,16 +30,16 @@ export interface CreateDocumentRecipientsOptions {
|
|||||||
requestMetadata: ApiRequestMetadata;
|
requestMetadata: ApiRequestMetadata;
|
||||||
}
|
}
|
||||||
|
|
||||||
export const createDocumentRecipients = async ({
|
export const createEnvelopeRecipients = async ({
|
||||||
userId,
|
userId,
|
||||||
teamId,
|
teamId,
|
||||||
id,
|
id,
|
||||||
recipients: recipientsToCreate,
|
recipients: recipientsToCreate,
|
||||||
requestMetadata,
|
requestMetadata,
|
||||||
}: CreateDocumentRecipientsOptions) => {
|
}: CreateEnvelopeRecipientsOptions) => {
|
||||||
const { envelopeWhereInput } = await getEnvelopeWhereInput({
|
const { envelopeWhereInput } = await getEnvelopeWhereInput({
|
||||||
id,
|
id,
|
||||||
type: EnvelopeType.DOCUMENT,
|
type: null,
|
||||||
userId,
|
userId,
|
||||||
teamId,
|
teamId,
|
||||||
});
|
});
|
||||||
@ -62,13 +62,13 @@ export const createDocumentRecipients = async ({
|
|||||||
|
|
||||||
if (!envelope) {
|
if (!envelope) {
|
||||||
throw new AppError(AppErrorCode.NOT_FOUND, {
|
throw new AppError(AppErrorCode.NOT_FOUND, {
|
||||||
message: 'Document not found',
|
message: 'Envelope not found',
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
if (envelope.completedAt) {
|
if (envelope.completedAt) {
|
||||||
throw new AppError(AppErrorCode.INVALID_REQUEST, {
|
throw new AppError(AppErrorCode.INVALID_REQUEST, {
|
||||||
message: 'Document already complete',
|
message: 'Envelope already complete',
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -112,6 +112,7 @@ export const createDocumentRecipients = async ({
|
|||||||
});
|
});
|
||||||
|
|
||||||
// Handle recipient created audit log.
|
// Handle recipient created audit log.
|
||||||
|
if (envelope.type === EnvelopeType.DOCUMENT) {
|
||||||
await tx.documentAuditLog.create({
|
await tx.documentAuditLog.create({
|
||||||
data: createDocumentAuditLogData({
|
data: createDocumentAuditLogData({
|
||||||
type: DOCUMENT_AUDIT_LOG_TYPE.RECIPIENT_CREATED,
|
type: DOCUMENT_AUDIT_LOG_TYPE.RECIPIENT_CREATED,
|
||||||
@ -127,6 +128,7 @@ export const createDocumentRecipients = async ({
|
|||||||
},
|
},
|
||||||
}),
|
}),
|
||||||
});
|
});
|
||||||
|
}
|
||||||
|
|
||||||
return createdRecipient;
|
return createdRecipient;
|
||||||
}),
|
}),
|
||||||
@ -1,115 +0,0 @@
|
|||||||
import { EnvelopeType, RecipientRole } from '@prisma/client';
|
|
||||||
import { SendStatus, SigningStatus } from '@prisma/client';
|
|
||||||
|
|
||||||
import type { TRecipientAccessAuthTypes } from '@documenso/lib/types/document-auth';
|
|
||||||
import { type TRecipientActionAuthTypes } from '@documenso/lib/types/document-auth';
|
|
||||||
import { nanoid } from '@documenso/lib/universal/id';
|
|
||||||
import { createRecipientAuthOptions } from '@documenso/lib/utils/document-auth';
|
|
||||||
import { prisma } from '@documenso/prisma';
|
|
||||||
|
|
||||||
import { AppError, AppErrorCode } from '../../errors/app-error';
|
|
||||||
import { mapRecipientToLegacyRecipient } from '../../utils/recipients';
|
|
||||||
import { getEnvelopeWhereInput } from '../envelope/get-envelope-by-id';
|
|
||||||
|
|
||||||
export interface CreateTemplateRecipientsOptions {
|
|
||||||
userId: number;
|
|
||||||
teamId: number;
|
|
||||||
templateId: number;
|
|
||||||
recipients: {
|
|
||||||
email: string;
|
|
||||||
name: string;
|
|
||||||
role: RecipientRole;
|
|
||||||
signingOrder?: number | null;
|
|
||||||
accessAuth?: TRecipientAccessAuthTypes[];
|
|
||||||
actionAuth?: TRecipientActionAuthTypes[];
|
|
||||||
}[];
|
|
||||||
}
|
|
||||||
|
|
||||||
export const createTemplateRecipients = async ({
|
|
||||||
userId,
|
|
||||||
teamId,
|
|
||||||
templateId,
|
|
||||||
recipients: recipientsToCreate,
|
|
||||||
}: CreateTemplateRecipientsOptions) => {
|
|
||||||
const { envelopeWhereInput } = await getEnvelopeWhereInput({
|
|
||||||
id: {
|
|
||||||
type: 'templateId',
|
|
||||||
id: templateId,
|
|
||||||
},
|
|
||||||
type: EnvelopeType.TEMPLATE,
|
|
||||||
userId,
|
|
||||||
teamId,
|
|
||||||
});
|
|
||||||
|
|
||||||
const template = await prisma.envelope.findFirst({
|
|
||||||
where: envelopeWhereInput,
|
|
||||||
include: {
|
|
||||||
recipients: true,
|
|
||||||
team: {
|
|
||||||
select: {
|
|
||||||
organisation: {
|
|
||||||
select: {
|
|
||||||
organisationClaim: true,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
if (!template) {
|
|
||||||
throw new AppError(AppErrorCode.NOT_FOUND, {
|
|
||||||
message: 'Template not found',
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
const recipientsHaveActionAuth = recipientsToCreate.some(
|
|
||||||
(recipient) => recipient.actionAuth && recipient.actionAuth.length > 0,
|
|
||||||
);
|
|
||||||
|
|
||||||
// Check if user has permission to set the global action auth.
|
|
||||||
if (recipientsHaveActionAuth && !template.team.organisation.organisationClaim.flags.cfr21) {
|
|
||||||
throw new AppError(AppErrorCode.UNAUTHORIZED, {
|
|
||||||
message: 'You do not have permission to set the action auth',
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
const normalizedRecipients = recipientsToCreate.map((recipient) => ({
|
|
||||||
...recipient,
|
|
||||||
email: recipient.email.toLowerCase(),
|
|
||||||
}));
|
|
||||||
|
|
||||||
const createdRecipients = await prisma.$transaction(async (tx) => {
|
|
||||||
return await Promise.all(
|
|
||||||
normalizedRecipients.map(async (recipient) => {
|
|
||||||
const authOptions = createRecipientAuthOptions({
|
|
||||||
accessAuth: recipient.accessAuth ?? [],
|
|
||||||
actionAuth: recipient.actionAuth ?? [],
|
|
||||||
});
|
|
||||||
|
|
||||||
const createdRecipient = await tx.recipient.create({
|
|
||||||
data: {
|
|
||||||
envelopeId: template.id,
|
|
||||||
name: recipient.name,
|
|
||||||
email: recipient.email,
|
|
||||||
role: recipient.role,
|
|
||||||
signingOrder: recipient.signingOrder,
|
|
||||||
token: nanoid(),
|
|
||||||
sendStatus: recipient.role === RecipientRole.CC ? SendStatus.SENT : SendStatus.NOT_SENT,
|
|
||||||
signingStatus:
|
|
||||||
recipient.role === RecipientRole.CC ? SigningStatus.SIGNED : SigningStatus.NOT_SIGNED,
|
|
||||||
authOptions,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
return createdRecipient;
|
|
||||||
}),
|
|
||||||
);
|
|
||||||
});
|
|
||||||
|
|
||||||
return {
|
|
||||||
recipients: createdRecipients.map((recipient) =>
|
|
||||||
mapRecipientToLegacyRecipient(recipient, template),
|
|
||||||
),
|
|
||||||
};
|
|
||||||
};
|
|
||||||
@ -14,26 +14,27 @@ import { NEXT_PUBLIC_WEBAPP_URL } from '../../constants/app';
|
|||||||
import { AppError, AppErrorCode } from '../../errors/app-error';
|
import { AppError, AppErrorCode } from '../../errors/app-error';
|
||||||
import { extractDerivedDocumentEmailSettings } from '../../types/document-email';
|
import { extractDerivedDocumentEmailSettings } from '../../types/document-email';
|
||||||
import { createDocumentAuditLogData } from '../../utils/document-audit-logs';
|
import { createDocumentAuditLogData } from '../../utils/document-audit-logs';
|
||||||
|
import { canRecipientBeModified } from '../../utils/recipients';
|
||||||
import { renderEmailWithI18N } from '../../utils/render-email-with-i18n';
|
import { renderEmailWithI18N } from '../../utils/render-email-with-i18n';
|
||||||
import { buildTeamWhereQuery } from '../../utils/teams';
|
import { buildTeamWhereQuery } from '../../utils/teams';
|
||||||
import { getEmailContext } from '../email/get-email-context';
|
import { getEmailContext } from '../email/get-email-context';
|
||||||
|
import { getEnvelopeWhereInput } from '../envelope/get-envelope-by-id';
|
||||||
|
|
||||||
export interface DeleteDocumentRecipientOptions {
|
export interface DeleteEnvelopeRecipientOptions {
|
||||||
userId: number;
|
userId: number;
|
||||||
teamId: number;
|
teamId: number;
|
||||||
recipientId: number;
|
recipientId: number;
|
||||||
requestMetadata: ApiRequestMetadata;
|
requestMetadata: ApiRequestMetadata;
|
||||||
}
|
}
|
||||||
|
|
||||||
export const deleteDocumentRecipient = async ({
|
export const deleteEnvelopeRecipient = async ({
|
||||||
userId,
|
userId,
|
||||||
teamId,
|
teamId,
|
||||||
recipientId,
|
recipientId,
|
||||||
requestMetadata,
|
requestMetadata,
|
||||||
}: DeleteDocumentRecipientOptions) => {
|
}: DeleteEnvelopeRecipientOptions) => {
|
||||||
const envelope = await prisma.envelope.findFirst({
|
const envelope = await prisma.envelope.findFirst({
|
||||||
where: {
|
where: {
|
||||||
type: EnvelopeType.DOCUMENT,
|
|
||||||
recipients: {
|
recipients: {
|
||||||
some: {
|
some: {
|
||||||
id: recipientId,
|
id: recipientId,
|
||||||
@ -48,6 +49,9 @@ export const deleteDocumentRecipient = async ({
|
|||||||
where: {
|
where: {
|
||||||
id: recipientId,
|
id: recipientId,
|
||||||
},
|
},
|
||||||
|
include: {
|
||||||
|
fields: true,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
@ -89,7 +93,24 @@ export const deleteDocumentRecipient = async ({
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!canRecipientBeModified(recipientToDelete, recipientToDelete.fields)) {
|
||||||
|
throw new AppError(AppErrorCode.INVALID_REQUEST, {
|
||||||
|
message: 'Recipient has already interacted with the document.',
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
const { envelopeWhereInput } = await getEnvelopeWhereInput({
|
||||||
|
id: {
|
||||||
|
type: 'envelopeId',
|
||||||
|
id: envelope.id,
|
||||||
|
},
|
||||||
|
type: null,
|
||||||
|
userId,
|
||||||
|
teamId,
|
||||||
|
});
|
||||||
|
|
||||||
const deletedRecipient = await prisma.$transaction(async (tx) => {
|
const deletedRecipient = await prisma.$transaction(async (tx) => {
|
||||||
|
if (envelope.type === EnvelopeType.DOCUMENT) {
|
||||||
await tx.documentAuditLog.create({
|
await tx.documentAuditLog.create({
|
||||||
data: createDocumentAuditLogData({
|
data: createDocumentAuditLogData({
|
||||||
type: DOCUMENT_AUDIT_LOG_TYPE.RECIPIENT_DELETED,
|
type: DOCUMENT_AUDIT_LOG_TYPE.RECIPIENT_DELETED,
|
||||||
@ -103,10 +124,12 @@ export const deleteDocumentRecipient = async ({
|
|||||||
},
|
},
|
||||||
}),
|
}),
|
||||||
});
|
});
|
||||||
|
}
|
||||||
|
|
||||||
return await tx.recipient.delete({
|
return await tx.recipient.delete({
|
||||||
where: {
|
where: {
|
||||||
id: recipientId,
|
id: recipientId,
|
||||||
|
envelope: envelopeWhereInput,
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
@ -116,7 +139,11 @@ export const deleteDocumentRecipient = async ({
|
|||||||
).recipientRemoved;
|
).recipientRemoved;
|
||||||
|
|
||||||
// Send email to deleted recipient.
|
// Send email to deleted recipient.
|
||||||
if (recipientToDelete.sendStatus === SendStatus.SENT && isRecipientRemovedEmailEnabled) {
|
if (
|
||||||
|
recipientToDelete.sendStatus === SendStatus.SENT &&
|
||||||
|
isRecipientRemovedEmailEnabled &&
|
||||||
|
envelope.type === EnvelopeType.DOCUMENT
|
||||||
|
) {
|
||||||
const assetBaseUrl = NEXT_PUBLIC_WEBAPP_URL() || 'http://localhost:3000';
|
const assetBaseUrl = NEXT_PUBLIC_WEBAPP_URL() || 'http://localhost:3000';
|
||||||
|
|
||||||
const template = createElement(RecipientRemovedFromDocumentTemplate, {
|
const template = createElement(RecipientRemovedFromDocumentTemplate, {
|
||||||
@ -1,58 +0,0 @@
|
|||||||
import { EnvelopeType } from '@prisma/client';
|
|
||||||
|
|
||||||
import { prisma } from '@documenso/prisma';
|
|
||||||
|
|
||||||
import { AppError, AppErrorCode } from '../../errors/app-error';
|
|
||||||
import { buildTeamWhereQuery } from '../../utils/teams';
|
|
||||||
import { getEnvelopeWhereInput } from '../envelope/get-envelope-by-id';
|
|
||||||
|
|
||||||
export interface DeleteTemplateRecipientOptions {
|
|
||||||
userId: number;
|
|
||||||
teamId: number;
|
|
||||||
recipientId: number;
|
|
||||||
}
|
|
||||||
|
|
||||||
export const deleteTemplateRecipient = async ({
|
|
||||||
userId,
|
|
||||||
teamId,
|
|
||||||
recipientId,
|
|
||||||
}: DeleteTemplateRecipientOptions): Promise<void> => {
|
|
||||||
const recipientToDelete = await prisma.recipient.findFirst({
|
|
||||||
where: {
|
|
||||||
id: recipientId,
|
|
||||||
envelope: {
|
|
||||||
type: EnvelopeType.TEMPLATE,
|
|
||||||
team: buildTeamWhereQuery({ teamId, userId }),
|
|
||||||
},
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
if (!recipientToDelete) {
|
|
||||||
throw new AppError(AppErrorCode.NOT_FOUND, {
|
|
||||||
message: 'Recipient not found',
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
const { envelopeWhereInput } = await getEnvelopeWhereInput({
|
|
||||||
id: {
|
|
||||||
type: 'envelopeId',
|
|
||||||
id: recipientToDelete.envelopeId,
|
|
||||||
},
|
|
||||||
type: EnvelopeType.TEMPLATE,
|
|
||||||
userId,
|
|
||||||
teamId,
|
|
||||||
});
|
|
||||||
|
|
||||||
if (!recipientToDelete || recipientToDelete.id !== recipientId) {
|
|
||||||
throw new AppError(AppErrorCode.NOT_FOUND, {
|
|
||||||
message: 'Recipient not found',
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
await prisma.recipient.delete({
|
|
||||||
where: {
|
|
||||||
id: recipientId,
|
|
||||||
envelope: envelopeWhereInput,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
};
|
|
||||||
@ -1,5 +1,4 @@
|
|||||||
import { EnvelopeType, RecipientRole } from '@prisma/client';
|
import { EnvelopeType, RecipientRole, SendStatus, SigningStatus } from '@prisma/client';
|
||||||
import { SendStatus, SigningStatus } from '@prisma/client';
|
|
||||||
|
|
||||||
import { DOCUMENT_AUDIT_LOG_TYPE } from '@documenso/lib/types/document-audit-logs';
|
import { DOCUMENT_AUDIT_LOG_TYPE } from '@documenso/lib/types/document-audit-logs';
|
||||||
import type { TRecipientAccessAuthTypes } from '@documenso/lib/types/document-auth';
|
import type { TRecipientAccessAuthTypes } from '@documenso/lib/types/document-auth';
|
||||||
@ -16,29 +15,38 @@ import { createRecipientAuthOptions } from '@documenso/lib/utils/document-auth';
|
|||||||
import { prisma } from '@documenso/prisma';
|
import { prisma } from '@documenso/prisma';
|
||||||
|
|
||||||
import { AppError, AppErrorCode } from '../../errors/app-error';
|
import { AppError, AppErrorCode } from '../../errors/app-error';
|
||||||
import { type EnvelopeIdOptions, mapSecondaryIdToDocumentId } from '../../utils/envelope';
|
import { extractLegacyIds } from '../../universal/id';
|
||||||
|
import { type EnvelopeIdOptions } from '../../utils/envelope';
|
||||||
import { mapFieldToLegacyField } from '../../utils/fields';
|
import { mapFieldToLegacyField } from '../../utils/fields';
|
||||||
import { canRecipientBeModified } from '../../utils/recipients';
|
import { canRecipientBeModified } from '../../utils/recipients';
|
||||||
import { getEnvelopeWhereInput } from '../envelope/get-envelope-by-id';
|
import { getEnvelopeWhereInput } from '../envelope/get-envelope-by-id';
|
||||||
|
|
||||||
export interface UpdateDocumentRecipientsOptions {
|
export interface UpdateEnvelopeRecipientsOptions {
|
||||||
userId: number;
|
userId: number;
|
||||||
teamId: number;
|
teamId: number;
|
||||||
id: EnvelopeIdOptions;
|
id: EnvelopeIdOptions;
|
||||||
recipients: RecipientData[];
|
recipients: {
|
||||||
|
id: number;
|
||||||
|
email?: string;
|
||||||
|
name?: string;
|
||||||
|
role?: RecipientRole;
|
||||||
|
signingOrder?: number | null;
|
||||||
|
accessAuth?: TRecipientAccessAuthTypes[];
|
||||||
|
actionAuth?: TRecipientActionAuthTypes[];
|
||||||
|
}[];
|
||||||
requestMetadata: ApiRequestMetadata;
|
requestMetadata: ApiRequestMetadata;
|
||||||
}
|
}
|
||||||
|
|
||||||
export const updateDocumentRecipients = async ({
|
export const updateEnvelopeRecipients = async ({
|
||||||
userId,
|
userId,
|
||||||
teamId,
|
teamId,
|
||||||
id,
|
id,
|
||||||
recipients,
|
recipients,
|
||||||
requestMetadata,
|
requestMetadata,
|
||||||
}: UpdateDocumentRecipientsOptions) => {
|
}: UpdateEnvelopeRecipientsOptions) => {
|
||||||
const { envelopeWhereInput } = await getEnvelopeWhereInput({
|
const { envelopeWhereInput } = await getEnvelopeWhereInput({
|
||||||
id,
|
id,
|
||||||
type: EnvelopeType.DOCUMENT,
|
type: null,
|
||||||
userId,
|
userId,
|
||||||
teamId,
|
teamId,
|
||||||
});
|
});
|
||||||
@ -62,13 +70,13 @@ export const updateDocumentRecipients = async ({
|
|||||||
|
|
||||||
if (!envelope) {
|
if (!envelope) {
|
||||||
throw new AppError(AppErrorCode.NOT_FOUND, {
|
throw new AppError(AppErrorCode.NOT_FOUND, {
|
||||||
message: 'Document not found',
|
message: 'Envelope not found',
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
if (envelope.completedAt) {
|
if (envelope.completedAt) {
|
||||||
throw new AppError(AppErrorCode.INVALID_REQUEST, {
|
throw new AppError(AppErrorCode.INVALID_REQUEST, {
|
||||||
message: 'Document already complete',
|
message: 'Envelope already complete',
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -160,9 +168,10 @@ export const updateDocumentRecipients = async ({
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Handle recipient updated audit log.
|
||||||
|
if (envelope.type === EnvelopeType.DOCUMENT) {
|
||||||
const changes = diffRecipientChanges(originalRecipient, updatedRecipient);
|
const changes = diffRecipientChanges(originalRecipient, updatedRecipient);
|
||||||
|
|
||||||
// Handle recipient updated audit log.
|
|
||||||
if (changes.length > 0) {
|
if (changes.length > 0) {
|
||||||
await tx.documentAuditLog.create({
|
await tx.documentAuditLog.create({
|
||||||
data: createDocumentAuditLogData({
|
data: createDocumentAuditLogData({
|
||||||
@ -179,6 +188,7 @@ export const updateDocumentRecipients = async ({
|
|||||||
}),
|
}),
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return updatedRecipient;
|
return updatedRecipient;
|
||||||
}),
|
}),
|
||||||
@ -188,19 +198,8 @@ export const updateDocumentRecipients = async ({
|
|||||||
return {
|
return {
|
||||||
recipients: updatedRecipients.map((recipient) => ({
|
recipients: updatedRecipients.map((recipient) => ({
|
||||||
...recipient,
|
...recipient,
|
||||||
documentId: mapSecondaryIdToDocumentId(envelope.secondaryId),
|
...extractLegacyIds(envelope),
|
||||||
templateId: null,
|
|
||||||
fields: recipient.fields.map((field) => mapFieldToLegacyField(field, envelope)),
|
fields: recipient.fields.map((field) => mapFieldToLegacyField(field, envelope)),
|
||||||
})),
|
})),
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
type RecipientData = {
|
|
||||||
id: number;
|
|
||||||
email?: string;
|
|
||||||
name?: string;
|
|
||||||
role?: RecipientRole;
|
|
||||||
signingOrder?: number | null;
|
|
||||||
accessAuth?: TRecipientAccessAuthTypes[];
|
|
||||||
actionAuth?: TRecipientActionAuthTypes[];
|
|
||||||
};
|
|
||||||
@ -1,168 +0,0 @@
|
|||||||
import { EnvelopeType, RecipientRole } from '@prisma/client';
|
|
||||||
import { SendStatus, SigningStatus } from '@prisma/client';
|
|
||||||
|
|
||||||
import type { TRecipientAccessAuthTypes } from '@documenso/lib/types/document-auth';
|
|
||||||
import {
|
|
||||||
type TRecipientActionAuthTypes,
|
|
||||||
ZRecipientAuthOptionsSchema,
|
|
||||||
} from '@documenso/lib/types/document-auth';
|
|
||||||
import { createRecipientAuthOptions } from '@documenso/lib/utils/document-auth';
|
|
||||||
import { prisma } from '@documenso/prisma';
|
|
||||||
|
|
||||||
import { AppError, AppErrorCode } from '../../errors/app-error';
|
|
||||||
import { mapSecondaryIdToTemplateId } from '../../utils/envelope';
|
|
||||||
import { mapFieldToLegacyField } from '../../utils/fields';
|
|
||||||
import { getEnvelopeWhereInput } from '../envelope/get-envelope-by-id';
|
|
||||||
|
|
||||||
export interface UpdateTemplateRecipientsOptions {
|
|
||||||
userId: number;
|
|
||||||
teamId: number;
|
|
||||||
templateId: number;
|
|
||||||
recipients: {
|
|
||||||
id: number;
|
|
||||||
email?: string;
|
|
||||||
name?: string;
|
|
||||||
role?: RecipientRole;
|
|
||||||
signingOrder?: number | null;
|
|
||||||
accessAuth?: TRecipientAccessAuthTypes[];
|
|
||||||
actionAuth?: TRecipientActionAuthTypes[];
|
|
||||||
}[];
|
|
||||||
}
|
|
||||||
|
|
||||||
export const updateTemplateRecipients = async ({
|
|
||||||
userId,
|
|
||||||
teamId,
|
|
||||||
templateId,
|
|
||||||
recipients,
|
|
||||||
}: UpdateTemplateRecipientsOptions) => {
|
|
||||||
const { envelopeWhereInput } = await getEnvelopeWhereInput({
|
|
||||||
id: {
|
|
||||||
type: 'templateId',
|
|
||||||
id: templateId,
|
|
||||||
},
|
|
||||||
type: EnvelopeType.TEMPLATE,
|
|
||||||
userId,
|
|
||||||
teamId,
|
|
||||||
});
|
|
||||||
|
|
||||||
const envelope = await prisma.envelope.findFirst({
|
|
||||||
where: envelopeWhereInput,
|
|
||||||
include: {
|
|
||||||
recipients: true,
|
|
||||||
team: {
|
|
||||||
select: {
|
|
||||||
organisation: {
|
|
||||||
select: {
|
|
||||||
organisationClaim: true,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
if (!envelope) {
|
|
||||||
throw new AppError(AppErrorCode.NOT_FOUND, {
|
|
||||||
message: 'Template not found',
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
const recipientsHaveActionAuth = recipients.some(
|
|
||||||
(recipient) => recipient.actionAuth && recipient.actionAuth.length > 0,
|
|
||||||
);
|
|
||||||
|
|
||||||
// Check if user has permission to set the global action auth.
|
|
||||||
if (recipientsHaveActionAuth && !envelope.team.organisation.organisationClaim.flags.cfr21) {
|
|
||||||
throw new AppError(AppErrorCode.UNAUTHORIZED, {
|
|
||||||
message: 'You do not have permission to set the action auth',
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
const recipientsToUpdate = recipients.map((recipient) => {
|
|
||||||
const originalRecipient = envelope.recipients.find(
|
|
||||||
(existingRecipient) => existingRecipient.id === recipient.id,
|
|
||||||
);
|
|
||||||
|
|
||||||
if (!originalRecipient) {
|
|
||||||
throw new AppError(AppErrorCode.NOT_FOUND, {
|
|
||||||
message: `Recipient with id ${recipient.id} not found`,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
return {
|
|
||||||
originalRecipient,
|
|
||||||
recipientUpdateData: recipient,
|
|
||||||
};
|
|
||||||
});
|
|
||||||
|
|
||||||
const updatedRecipients = await prisma.$transaction(async (tx) => {
|
|
||||||
return await Promise.all(
|
|
||||||
recipientsToUpdate.map(async ({ originalRecipient, recipientUpdateData }) => {
|
|
||||||
let authOptions = ZRecipientAuthOptionsSchema.parse(originalRecipient.authOptions);
|
|
||||||
|
|
||||||
if (
|
|
||||||
recipientUpdateData.actionAuth !== undefined ||
|
|
||||||
recipientUpdateData.accessAuth !== undefined
|
|
||||||
) {
|
|
||||||
authOptions = createRecipientAuthOptions({
|
|
||||||
accessAuth: recipientUpdateData.accessAuth || authOptions.accessAuth,
|
|
||||||
actionAuth: recipientUpdateData.actionAuth || authOptions.actionAuth,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
const mergedRecipient = {
|
|
||||||
...originalRecipient,
|
|
||||||
...recipientUpdateData,
|
|
||||||
};
|
|
||||||
|
|
||||||
const updatedRecipient = await tx.recipient.update({
|
|
||||||
where: {
|
|
||||||
id: originalRecipient.id,
|
|
||||||
envelopeId: envelope.id,
|
|
||||||
},
|
|
||||||
data: {
|
|
||||||
name: mergedRecipient.name,
|
|
||||||
email: mergedRecipient.email,
|
|
||||||
role: mergedRecipient.role,
|
|
||||||
signingOrder: mergedRecipient.signingOrder,
|
|
||||||
envelopeId: envelope.id,
|
|
||||||
sendStatus:
|
|
||||||
mergedRecipient.role === RecipientRole.CC ? SendStatus.SENT : SendStatus.NOT_SENT,
|
|
||||||
signingStatus:
|
|
||||||
mergedRecipient.role === RecipientRole.CC
|
|
||||||
? SigningStatus.SIGNED
|
|
||||||
: SigningStatus.NOT_SIGNED,
|
|
||||||
authOptions,
|
|
||||||
},
|
|
||||||
include: {
|
|
||||||
fields: true,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
// Clear all fields if the recipient role is changed to a type that cannot have fields.
|
|
||||||
if (
|
|
||||||
originalRecipient.role !== updatedRecipient.role &&
|
|
||||||
(updatedRecipient.role === RecipientRole.CC ||
|
|
||||||
updatedRecipient.role === RecipientRole.VIEWER)
|
|
||||||
) {
|
|
||||||
await tx.field.deleteMany({
|
|
||||||
where: {
|
|
||||||
recipientId: updatedRecipient.id,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
return updatedRecipient;
|
|
||||||
}),
|
|
||||||
);
|
|
||||||
});
|
|
||||||
|
|
||||||
return {
|
|
||||||
recipients: updatedRecipients.map((recipient) => ({
|
|
||||||
...recipient,
|
|
||||||
documentId: null,
|
|
||||||
templateId: mapSecondaryIdToTemplateId(envelope.secondaryId),
|
|
||||||
fields: recipient.fields.map((field) => mapFieldToLegacyField(field, envelope)),
|
|
||||||
})),
|
|
||||||
};
|
|
||||||
};
|
|
||||||
@ -37,11 +37,8 @@ export const ZEnvelopeSchema = EnvelopeSchema.pick({
|
|||||||
userId: true,
|
userId: true,
|
||||||
teamId: true,
|
teamId: true,
|
||||||
folderId: true,
|
folderId: true,
|
||||||
|
templateId: true,
|
||||||
}).extend({
|
}).extend({
|
||||||
templateId: z
|
|
||||||
.number()
|
|
||||||
.nullish()
|
|
||||||
.describe('The ID of the template that the document was created from, if any.'),
|
|
||||||
documentMeta: DocumentMetaSchema.pick({
|
documentMeta: DocumentMetaSchema.pick({
|
||||||
signingOrder: true,
|
signingOrder: true,
|
||||||
distributionMethod: true,
|
distributionMethod: true,
|
||||||
|
|||||||
@ -50,6 +50,11 @@ export const ZFieldSchema = FieldSchema.pick({
|
|||||||
templateId: z.number().nullish(),
|
templateId: z.number().nullish(),
|
||||||
});
|
});
|
||||||
|
|
||||||
|
export const ZEnvelopeFieldSchema = ZFieldSchema.omit({
|
||||||
|
documentId: true,
|
||||||
|
templateId: true,
|
||||||
|
});
|
||||||
|
|
||||||
export const ZFieldPageNumberSchema = z
|
export const ZFieldPageNumberSchema = z
|
||||||
.number()
|
.number()
|
||||||
.min(1)
|
.min(1)
|
||||||
@ -69,6 +74,30 @@ export const ZFieldWidthSchema = z.number().min(1).describe('The width of the fi
|
|||||||
|
|
||||||
export const ZFieldHeightSchema = z.number().min(1).describe('The height of the field.');
|
export const ZFieldHeightSchema = z.number().min(1).describe('The height of the field.');
|
||||||
|
|
||||||
|
export const ZClampedFieldPageXSchema = z
|
||||||
|
.number()
|
||||||
|
.min(0)
|
||||||
|
.max(100)
|
||||||
|
.describe('The percentage based X coordinate where the field will be placed.');
|
||||||
|
|
||||||
|
export const ZClampedFieldPageYSchema = z
|
||||||
|
.number()
|
||||||
|
.min(0)
|
||||||
|
.max(100)
|
||||||
|
.describe('The percentage based Y coordinate where the field will be placed.');
|
||||||
|
|
||||||
|
export const ZClampedFieldWidthSchema = z
|
||||||
|
.number()
|
||||||
|
.min(0)
|
||||||
|
.max(100)
|
||||||
|
.describe('The percentage based width of the field on the page.');
|
||||||
|
|
||||||
|
export const ZClampedFieldHeightSchema = z
|
||||||
|
.number()
|
||||||
|
.min(0)
|
||||||
|
.max(100)
|
||||||
|
.describe('The percentage based height of the field on the page.');
|
||||||
|
|
||||||
// ---------------------------------------------
|
// ---------------------------------------------
|
||||||
|
|
||||||
const PrismaDecimalSchema = z.preprocess(
|
const PrismaDecimalSchema = z.preprocess(
|
||||||
|
|||||||
@ -95,3 +95,18 @@ export const ZRecipientManySchema = RecipientSchema.pick({
|
|||||||
documentId: z.number().nullish(),
|
documentId: z.number().nullish(),
|
||||||
templateId: z.number().nullish(),
|
templateId: z.number().nullish(),
|
||||||
});
|
});
|
||||||
|
|
||||||
|
export const ZEnvelopeRecipientSchema = ZRecipientSchema.omit({
|
||||||
|
documentId: true,
|
||||||
|
templateId: true,
|
||||||
|
});
|
||||||
|
|
||||||
|
export const ZEnvelopeRecipientLiteSchema = ZRecipientLiteSchema.omit({
|
||||||
|
documentId: true,
|
||||||
|
templateId: true,
|
||||||
|
});
|
||||||
|
|
||||||
|
export const ZEnvelopeRecipientManySchema = ZRecipientManySchema.omit({
|
||||||
|
documentId: true,
|
||||||
|
templateId: true,
|
||||||
|
});
|
||||||
|
|||||||
@ -13,11 +13,21 @@ import {
|
|||||||
} from './create-envelope-items.types';
|
} from './create-envelope-items.types';
|
||||||
|
|
||||||
export const createEnvelopeItemsRoute = authenticatedProcedure
|
export const createEnvelopeItemsRoute = authenticatedProcedure
|
||||||
|
// Todo: Envelopes - Pending direct uploads
|
||||||
|
// .meta({
|
||||||
|
// openapi: {
|
||||||
|
// method: 'POST',
|
||||||
|
// path: '/envelope/item/create-many',
|
||||||
|
// summary: 'Create envelope items',
|
||||||
|
// description: 'Create multiple envelope items for an envelope',
|
||||||
|
// tags: ['Envelope Item'],
|
||||||
|
// },
|
||||||
|
// })
|
||||||
.input(ZCreateEnvelopeItemsRequestSchema)
|
.input(ZCreateEnvelopeItemsRequestSchema)
|
||||||
.output(ZCreateEnvelopeItemsResponseSchema)
|
.output(ZCreateEnvelopeItemsResponseSchema)
|
||||||
.mutation(async ({ input, ctx }) => {
|
.mutation(async ({ input, ctx }) => {
|
||||||
const { user, teamId, metadata } = ctx;
|
const { user, teamId, metadata } = ctx;
|
||||||
const { envelopeId, items } = input;
|
const { envelopeId, data: items } = input;
|
||||||
|
|
||||||
ctx.logger.info({
|
ctx.logger.info({
|
||||||
input: {
|
input: {
|
||||||
|
|||||||
@ -7,7 +7,7 @@ import { ZDocumentTitleSchema } from '../document-router/schema';
|
|||||||
|
|
||||||
export const ZCreateEnvelopeItemsRequestSchema = z.object({
|
export const ZCreateEnvelopeItemsRequestSchema = z.object({
|
||||||
envelopeId: z.string(),
|
envelopeId: z.string(),
|
||||||
items: z
|
data: z
|
||||||
.object({
|
.object({
|
||||||
title: ZDocumentTitleSchema,
|
title: ZDocumentTitleSchema,
|
||||||
documentDataId: z.string(),
|
documentDataId: z.string(),
|
||||||
|
|||||||
@ -9,6 +9,15 @@ import {
|
|||||||
} from './create-envelope.types';
|
} from './create-envelope.types';
|
||||||
|
|
||||||
export const createEnvelopeRoute = authenticatedProcedure
|
export const createEnvelopeRoute = authenticatedProcedure
|
||||||
|
// Todo: Envelopes - Pending direct uploads
|
||||||
|
// .meta({
|
||||||
|
// openapi: {
|
||||||
|
// method: 'POST',
|
||||||
|
// path: '/envelope/create',
|
||||||
|
// summary: 'Create envelope',
|
||||||
|
// tags: ['Envelope'],
|
||||||
|
// },
|
||||||
|
// })
|
||||||
.input(ZCreateEnvelopeRequestSchema)
|
.input(ZCreateEnvelopeRequestSchema)
|
||||||
.output(ZCreateEnvelopeResponseSchema)
|
.output(ZCreateEnvelopeResponseSchema)
|
||||||
.mutation(async ({ input, ctx }) => {
|
.mutation(async ({ input, ctx }) => {
|
||||||
|
|||||||
@ -24,16 +24,6 @@ import {
|
|||||||
} from '../document-router/schema';
|
} from '../document-router/schema';
|
||||||
import { ZCreateRecipientSchema } from '../recipient-router/schema';
|
import { ZCreateRecipientSchema } from '../recipient-router/schema';
|
||||||
|
|
||||||
// Currently not in use until we allow passthrough documents on create.
|
|
||||||
// export const createEnvelopeMeta: TrpcRouteMeta = {
|
|
||||||
// openapi: {
|
|
||||||
// method: 'POST',
|
|
||||||
// path: '/envelope/create',
|
|
||||||
// summary: 'Create envelope',
|
|
||||||
// tags: ['Envelope'],
|
|
||||||
// },
|
|
||||||
// };
|
|
||||||
|
|
||||||
export const ZCreateEnvelopeRequestSchema = z.object({
|
export const ZCreateEnvelopeRequestSchema = z.object({
|
||||||
title: ZDocumentTitleSchema,
|
title: ZDocumentTitleSchema,
|
||||||
type: z.nativeEnum(EnvelopeType),
|
type: z.nativeEnum(EnvelopeType),
|
||||||
|
|||||||
@ -12,6 +12,15 @@ import {
|
|||||||
} from './delete-envelope-item.types';
|
} from './delete-envelope-item.types';
|
||||||
|
|
||||||
export const deleteEnvelopeItemRoute = authenticatedProcedure
|
export const deleteEnvelopeItemRoute = authenticatedProcedure
|
||||||
|
.meta({
|
||||||
|
openapi: {
|
||||||
|
method: 'POST',
|
||||||
|
path: '/envelope/item/delete',
|
||||||
|
summary: 'Delete envelope item',
|
||||||
|
description: 'Delete an envelope item from an envelope',
|
||||||
|
tags: ['Envelope Item'],
|
||||||
|
},
|
||||||
|
})
|
||||||
.input(ZDeleteEnvelopeItemRequestSchema)
|
.input(ZDeleteEnvelopeItemRequestSchema)
|
||||||
.output(ZDeleteEnvelopeItemResponseSchema)
|
.output(ZDeleteEnvelopeItemResponseSchema)
|
||||||
.mutation(async ({ input, ctx }) => {
|
.mutation(async ({ input, ctx }) => {
|
||||||
|
|||||||
@ -1,8 +1,10 @@
|
|||||||
import { EnvelopeType } from '@prisma/client';
|
import { EnvelopeType } from '@prisma/client';
|
||||||
import { match } from 'ts-pattern';
|
import { match } from 'ts-pattern';
|
||||||
|
|
||||||
|
import { AppError, AppErrorCode } from '@documenso/lib/errors/app-error';
|
||||||
import { deleteDocument } from '@documenso/lib/server-only/document/delete-document';
|
import { deleteDocument } from '@documenso/lib/server-only/document/delete-document';
|
||||||
import { deleteTemplate } from '@documenso/lib/server-only/template/delete-template';
|
import { deleteTemplate } from '@documenso/lib/server-only/template/delete-template';
|
||||||
|
import { prisma } from '@documenso/prisma';
|
||||||
|
|
||||||
import { authenticatedProcedure } from '../trpc';
|
import { authenticatedProcedure } from '../trpc';
|
||||||
import {
|
import {
|
||||||
@ -11,12 +13,19 @@ import {
|
|||||||
} from './delete-envelope.types';
|
} from './delete-envelope.types';
|
||||||
|
|
||||||
export const deleteEnvelopeRoute = authenticatedProcedure
|
export const deleteEnvelopeRoute = authenticatedProcedure
|
||||||
// .meta(deleteEnvelopeMeta)
|
.meta({
|
||||||
|
openapi: {
|
||||||
|
method: 'POST',
|
||||||
|
path: '/envelope/delete',
|
||||||
|
summary: 'Delete envelope',
|
||||||
|
tags: ['Envelope'],
|
||||||
|
},
|
||||||
|
})
|
||||||
.input(ZDeleteEnvelopeRequestSchema)
|
.input(ZDeleteEnvelopeRequestSchema)
|
||||||
.output(ZDeleteEnvelopeResponseSchema)
|
.output(ZDeleteEnvelopeResponseSchema)
|
||||||
.mutation(async ({ input, ctx }) => {
|
.mutation(async ({ input, ctx }) => {
|
||||||
const { teamId } = ctx;
|
const { teamId } = ctx;
|
||||||
const { envelopeId, envelopeType } = input;
|
const { envelopeId } = input;
|
||||||
|
|
||||||
ctx.logger.info({
|
ctx.logger.info({
|
||||||
input: {
|
input: {
|
||||||
@ -24,7 +33,22 @@ export const deleteEnvelopeRoute = authenticatedProcedure
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
await match(envelopeType)
|
const unsafeEnvelope = await prisma.envelope.findUnique({
|
||||||
|
where: {
|
||||||
|
id: envelopeId,
|
||||||
|
},
|
||||||
|
select: {
|
||||||
|
type: true,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!unsafeEnvelope) {
|
||||||
|
throw new AppError(AppErrorCode.NOT_FOUND, {
|
||||||
|
message: 'Envelope not found',
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
await match(unsafeEnvelope.type)
|
||||||
.with(EnvelopeType.DOCUMENT, async () =>
|
.with(EnvelopeType.DOCUMENT, async () =>
|
||||||
deleteDocument({
|
deleteDocument({
|
||||||
userId: ctx.user.id,
|
userId: ctx.user.id,
|
||||||
|
|||||||
@ -1,18 +1,7 @@
|
|||||||
import { EnvelopeType } from '@prisma/client';
|
|
||||||
import { z } from 'zod';
|
import { z } from 'zod';
|
||||||
|
|
||||||
// export const deleteEnvelopeMeta: TrpcRouteMeta = {
|
|
||||||
// openapi: {
|
|
||||||
// method: 'POST',
|
|
||||||
// path: '/envelope/delete',
|
|
||||||
// summary: 'Delete envelope',
|
|
||||||
// tags: ['Envelope'],
|
|
||||||
// },
|
|
||||||
// };
|
|
||||||
|
|
||||||
export const ZDeleteEnvelopeRequestSchema = z.object({
|
export const ZDeleteEnvelopeRequestSchema = z.object({
|
||||||
envelopeId: z.string(),
|
envelopeId: z.string(),
|
||||||
envelopeType: z.nativeEnum(EnvelopeType),
|
|
||||||
});
|
});
|
||||||
|
|
||||||
export const ZDeleteEnvelopeResponseSchema = z.void();
|
export const ZDeleteEnvelopeResponseSchema = z.void();
|
||||||
|
|||||||
@ -8,7 +8,15 @@ import {
|
|||||||
} from './distribute-envelope.types';
|
} from './distribute-envelope.types';
|
||||||
|
|
||||||
export const distributeEnvelopeRoute = authenticatedProcedure
|
export const distributeEnvelopeRoute = authenticatedProcedure
|
||||||
// .meta(distributeEnvelopeMeta)
|
.meta({
|
||||||
|
openapi: {
|
||||||
|
method: 'POST',
|
||||||
|
path: '/envelope/distribute',
|
||||||
|
summary: 'Distribute envelope',
|
||||||
|
description: 'Send the envelope to recipients based on your distribution method',
|
||||||
|
tags: ['Envelope'],
|
||||||
|
},
|
||||||
|
})
|
||||||
.input(ZDistributeEnvelopeRequestSchema)
|
.input(ZDistributeEnvelopeRequestSchema)
|
||||||
.output(ZDistributeEnvelopeResponseSchema)
|
.output(ZDistributeEnvelopeResponseSchema)
|
||||||
.mutation(async ({ input, ctx }) => {
|
.mutation(async ({ input, ctx }) => {
|
||||||
|
|||||||
@ -2,16 +2,6 @@ import { z } from 'zod';
|
|||||||
|
|
||||||
import { ZDocumentMetaUpdateSchema } from '@documenso/lib/types/document-meta';
|
import { ZDocumentMetaUpdateSchema } from '@documenso/lib/types/document-meta';
|
||||||
|
|
||||||
// export const distributeEnvelopeMeta: TrpcRouteMeta = {
|
|
||||||
// openapi: {
|
|
||||||
// method: 'POST',
|
|
||||||
// path: '/envelope/distribute',
|
|
||||||
// summary: 'Distribute envelope',
|
|
||||||
// description: 'Send the document out to recipients based on your distribution method',
|
|
||||||
// tags: ['Envelope'],
|
|
||||||
// },
|
|
||||||
// };
|
|
||||||
|
|
||||||
export const ZDistributeEnvelopeRequestSchema = z.object({
|
export const ZDistributeEnvelopeRequestSchema = z.object({
|
||||||
envelopeId: z.string().describe('The ID of the envelope to send.'),
|
envelopeId: z.string().describe('The ID of the envelope to send.'),
|
||||||
meta: ZDocumentMetaUpdateSchema.pick({
|
meta: ZDocumentMetaUpdateSchema.pick({
|
||||||
|
|||||||
@ -7,6 +7,15 @@ import {
|
|||||||
} from './duplicate-envelope.types';
|
} from './duplicate-envelope.types';
|
||||||
|
|
||||||
export const duplicateEnvelopeRoute = authenticatedProcedure
|
export const duplicateEnvelopeRoute = authenticatedProcedure
|
||||||
|
.meta({
|
||||||
|
openapi: {
|
||||||
|
method: 'POST',
|
||||||
|
path: '/envelope/duplicate',
|
||||||
|
summary: 'Duplicate envelope',
|
||||||
|
description: 'Duplicate an envelope with all its settings',
|
||||||
|
tags: ['Envelope'],
|
||||||
|
},
|
||||||
|
})
|
||||||
.input(ZDuplicateEnvelopeRequestSchema)
|
.input(ZDuplicateEnvelopeRequestSchema)
|
||||||
.output(ZDuplicateEnvelopeResponseSchema)
|
.output(ZDuplicateEnvelopeResponseSchema)
|
||||||
.mutation(async ({ input, ctx }) => {
|
.mutation(async ({ input, ctx }) => {
|
||||||
|
|||||||
@ -0,0 +1,41 @@
|
|||||||
|
import { createEnvelopeFields } from '@documenso/lib/server-only/field/create-envelope-fields';
|
||||||
|
|
||||||
|
import { authenticatedProcedure } from '../../trpc';
|
||||||
|
import {
|
||||||
|
ZCreateEnvelopeFieldsRequestSchema,
|
||||||
|
ZCreateEnvelopeFieldsResponseSchema,
|
||||||
|
} from './create-envelope-fields.types';
|
||||||
|
|
||||||
|
export const createEnvelopeFieldsRoute = authenticatedProcedure
|
||||||
|
.meta({
|
||||||
|
openapi: {
|
||||||
|
method: 'POST',
|
||||||
|
path: '/envelope/field/create-many',
|
||||||
|
summary: 'Create envelope fields',
|
||||||
|
description: 'Create multiple fields for an envelope',
|
||||||
|
tags: ['Envelope Fields'],
|
||||||
|
},
|
||||||
|
})
|
||||||
|
.input(ZCreateEnvelopeFieldsRequestSchema)
|
||||||
|
.output(ZCreateEnvelopeFieldsResponseSchema)
|
||||||
|
.mutation(async ({ input, ctx }) => {
|
||||||
|
const { user, teamId, metadata } = ctx;
|
||||||
|
const { envelopeId, data: fields } = input;
|
||||||
|
|
||||||
|
ctx.logger.info({
|
||||||
|
input: {
|
||||||
|
envelopeId,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
return await createEnvelopeFields({
|
||||||
|
userId: user.id,
|
||||||
|
teamId,
|
||||||
|
id: {
|
||||||
|
type: 'envelopeId',
|
||||||
|
id: envelopeId,
|
||||||
|
},
|
||||||
|
fields,
|
||||||
|
requestMetadata: metadata,
|
||||||
|
});
|
||||||
|
});
|
||||||
@ -0,0 +1,40 @@
|
|||||||
|
import { z } from 'zod';
|
||||||
|
|
||||||
|
import {
|
||||||
|
ZClampedFieldHeightSchema,
|
||||||
|
ZClampedFieldPageXSchema,
|
||||||
|
ZClampedFieldPageYSchema,
|
||||||
|
ZClampedFieldWidthSchema,
|
||||||
|
ZFieldPageNumberSchema,
|
||||||
|
ZFieldSchema,
|
||||||
|
} from '@documenso/lib/types/field';
|
||||||
|
import { ZFieldAndMetaSchema } from '@documenso/lib/types/field-meta';
|
||||||
|
|
||||||
|
const ZCreateFieldSchema = ZFieldAndMetaSchema.and(
|
||||||
|
z.object({
|
||||||
|
recipientId: z.number().describe('The ID of the recipient to create the field for'),
|
||||||
|
envelopeItemId: z
|
||||||
|
.string()
|
||||||
|
.optional()
|
||||||
|
.describe(
|
||||||
|
'The ID of the envelope item to put the field on. If not provided, field will be placed on the first item.',
|
||||||
|
),
|
||||||
|
pageNumber: ZFieldPageNumberSchema,
|
||||||
|
pageX: ZClampedFieldPageXSchema,
|
||||||
|
pageY: ZClampedFieldPageYSchema,
|
||||||
|
width: ZClampedFieldWidthSchema,
|
||||||
|
height: ZClampedFieldHeightSchema,
|
||||||
|
}),
|
||||||
|
);
|
||||||
|
|
||||||
|
export const ZCreateEnvelopeFieldsRequestSchema = z.object({
|
||||||
|
envelopeId: z.string(),
|
||||||
|
data: ZCreateFieldSchema.array(),
|
||||||
|
});
|
||||||
|
|
||||||
|
export const ZCreateEnvelopeFieldsResponseSchema = z.object({
|
||||||
|
fields: z.array(ZFieldSchema),
|
||||||
|
});
|
||||||
|
|
||||||
|
export type TCreateEnvelopeFieldsRequest = z.infer<typeof ZCreateEnvelopeFieldsRequestSchema>;
|
||||||
|
export type TCreateEnvelopeFieldsResponse = z.infer<typeof ZCreateEnvelopeFieldsResponseSchema>;
|
||||||
@ -0,0 +1,125 @@
|
|||||||
|
import { EnvelopeType } from '@prisma/client';
|
||||||
|
|
||||||
|
import { AppError, AppErrorCode } from '@documenso/lib/errors/app-error';
|
||||||
|
import { getEnvelopeWhereInput } from '@documenso/lib/server-only/envelope/get-envelope-by-id';
|
||||||
|
import { DOCUMENT_AUDIT_LOG_TYPE } from '@documenso/lib/types/document-audit-logs';
|
||||||
|
import { createDocumentAuditLogData } from '@documenso/lib/utils/document-audit-logs';
|
||||||
|
import { canRecipientFieldsBeModified } from '@documenso/lib/utils/recipients';
|
||||||
|
import { prisma } from '@documenso/prisma';
|
||||||
|
|
||||||
|
import { authenticatedProcedure } from '../../trpc';
|
||||||
|
import {
|
||||||
|
ZDeleteEnvelopeFieldRequestSchema,
|
||||||
|
ZDeleteEnvelopeFieldResponseSchema,
|
||||||
|
} from './delete-envelope-field.types';
|
||||||
|
|
||||||
|
export const deleteEnvelopeFieldRoute = authenticatedProcedure
|
||||||
|
.meta({
|
||||||
|
openapi: {
|
||||||
|
method: 'POST',
|
||||||
|
path: '/envelope/field/delete',
|
||||||
|
summary: 'Delete envelope field',
|
||||||
|
description: 'Delete an envelope field',
|
||||||
|
tags: ['Envelope Field'],
|
||||||
|
},
|
||||||
|
})
|
||||||
|
.input(ZDeleteEnvelopeFieldRequestSchema)
|
||||||
|
.output(ZDeleteEnvelopeFieldResponseSchema)
|
||||||
|
.mutation(async ({ input, ctx }) => {
|
||||||
|
const { user, teamId, metadata } = ctx;
|
||||||
|
const { fieldId } = input;
|
||||||
|
|
||||||
|
ctx.logger.info({
|
||||||
|
input: {
|
||||||
|
fieldId,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
const unsafeField = await prisma.field.findUnique({
|
||||||
|
where: {
|
||||||
|
id: fieldId,
|
||||||
|
},
|
||||||
|
select: {
|
||||||
|
envelopeId: true,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!unsafeField) {
|
||||||
|
throw new AppError(AppErrorCode.NOT_FOUND, {
|
||||||
|
message: 'Field not found',
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
const { envelopeWhereInput } = await getEnvelopeWhereInput({
|
||||||
|
id: {
|
||||||
|
type: 'envelopeId',
|
||||||
|
id: unsafeField.envelopeId,
|
||||||
|
},
|
||||||
|
type: null,
|
||||||
|
userId: user.id,
|
||||||
|
teamId,
|
||||||
|
});
|
||||||
|
|
||||||
|
const envelope = await prisma.envelope.findUnique({
|
||||||
|
where: envelopeWhereInput,
|
||||||
|
include: {
|
||||||
|
recipients: {
|
||||||
|
include: {
|
||||||
|
fields: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
const recipientWithFields = envelope?.recipients.find((recipient) =>
|
||||||
|
recipient.fields.some((field) => field.id === fieldId),
|
||||||
|
);
|
||||||
|
const fieldToDelete = recipientWithFields?.fields.find((field) => field.id === fieldId);
|
||||||
|
|
||||||
|
if (!envelope || !recipientWithFields || !fieldToDelete) {
|
||||||
|
throw new AppError(AppErrorCode.NOT_FOUND, {
|
||||||
|
message: 'Field not found',
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
if (envelope.completedAt) {
|
||||||
|
throw new AppError(AppErrorCode.INVALID_REQUEST, {
|
||||||
|
message: 'Envelope already complete',
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check whether the recipient associated with the field can have new fields created.
|
||||||
|
if (!canRecipientFieldsBeModified(recipientWithFields, recipientWithFields.fields)) {
|
||||||
|
throw new AppError(AppErrorCode.INVALID_REQUEST, {
|
||||||
|
message: 'Recipient has already interacted with the document.',
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
await prisma.$transaction(async (tx) => {
|
||||||
|
const deletedField = await tx.field.delete({
|
||||||
|
where: {
|
||||||
|
id: fieldToDelete.id,
|
||||||
|
envelopeId: envelope.id,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
// Handle field deleted audit log.
|
||||||
|
if (envelope.type === EnvelopeType.DOCUMENT) {
|
||||||
|
await tx.documentAuditLog.create({
|
||||||
|
data: createDocumentAuditLogData({
|
||||||
|
type: DOCUMENT_AUDIT_LOG_TYPE.FIELD_DELETED,
|
||||||
|
envelopeId: envelope.id,
|
||||||
|
metadata,
|
||||||
|
data: {
|
||||||
|
fieldId: deletedField.secondaryId,
|
||||||
|
fieldRecipientEmail: recipientWithFields.email,
|
||||||
|
fieldRecipientId: deletedField.recipientId,
|
||||||
|
fieldType: deletedField.type,
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
return deletedField;
|
||||||
|
});
|
||||||
|
});
|
||||||
@ -0,0 +1,10 @@
|
|||||||
|
import { z } from 'zod';
|
||||||
|
|
||||||
|
export const ZDeleteEnvelopeFieldRequestSchema = z.object({
|
||||||
|
fieldId: z.number(),
|
||||||
|
});
|
||||||
|
|
||||||
|
export const ZDeleteEnvelopeFieldResponseSchema = z.void();
|
||||||
|
|
||||||
|
export type TDeleteEnvelopeFieldRequest = z.infer<typeof ZDeleteEnvelopeFieldRequestSchema>;
|
||||||
|
export type TDeleteEnvelopeFieldResponse = z.infer<typeof ZDeleteEnvelopeFieldResponseSchema>;
|
||||||
@ -0,0 +1,36 @@
|
|||||||
|
import { getFieldById } from '@documenso/lib/server-only/field/get-field-by-id';
|
||||||
|
|
||||||
|
import { authenticatedProcedure } from '../../trpc';
|
||||||
|
import {
|
||||||
|
ZGetEnvelopeFieldRequestSchema,
|
||||||
|
ZGetEnvelopeFieldResponseSchema,
|
||||||
|
} from './get-envelope-field.types';
|
||||||
|
|
||||||
|
export const getEnvelopeFieldRoute = authenticatedProcedure
|
||||||
|
.meta({
|
||||||
|
openapi: {
|
||||||
|
method: 'GET',
|
||||||
|
path: '/envelope/field/{fieldId}',
|
||||||
|
summary: 'Get envelope field',
|
||||||
|
description: 'Returns an envelope field given an ID',
|
||||||
|
tags: ['Envelope Field'],
|
||||||
|
},
|
||||||
|
})
|
||||||
|
.input(ZGetEnvelopeFieldRequestSchema)
|
||||||
|
.output(ZGetEnvelopeFieldResponseSchema)
|
||||||
|
.query(async ({ input, ctx }) => {
|
||||||
|
const { teamId, user } = ctx;
|
||||||
|
const { fieldId } = input;
|
||||||
|
|
||||||
|
ctx.logger.info({
|
||||||
|
input: {
|
||||||
|
fieldId,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
return await getFieldById({
|
||||||
|
userId: user.id,
|
||||||
|
teamId,
|
||||||
|
fieldId,
|
||||||
|
});
|
||||||
|
});
|
||||||
@ -0,0 +1,12 @@
|
|||||||
|
import { z } from 'zod';
|
||||||
|
|
||||||
|
import { ZEnvelopeFieldSchema } from '@documenso/lib/types/field';
|
||||||
|
|
||||||
|
export const ZGetEnvelopeFieldRequestSchema = z.object({
|
||||||
|
fieldId: z.number(),
|
||||||
|
});
|
||||||
|
|
||||||
|
export const ZGetEnvelopeFieldResponseSchema = ZEnvelopeFieldSchema;
|
||||||
|
|
||||||
|
export type TGetEnvelopeFieldRequest = z.infer<typeof ZGetEnvelopeFieldRequestSchema>;
|
||||||
|
export type TGetEnvelopeFieldResponse = z.infer<typeof ZGetEnvelopeFieldResponseSchema>;
|
||||||
@ -0,0 +1,42 @@
|
|||||||
|
import { updateEnvelopeFields } from '@documenso/lib/server-only/field/update-envelope-fields';
|
||||||
|
|
||||||
|
import { authenticatedProcedure } from '../../trpc';
|
||||||
|
import {
|
||||||
|
ZUpdateEnvelopeFieldsRequestSchema,
|
||||||
|
ZUpdateEnvelopeFieldsResponseSchema,
|
||||||
|
} from './update-envelope-fields.types';
|
||||||
|
|
||||||
|
export const updateEnvelopeFieldsRoute = authenticatedProcedure
|
||||||
|
.meta({
|
||||||
|
openapi: {
|
||||||
|
method: 'POST',
|
||||||
|
path: '/envelope/field/update-many',
|
||||||
|
summary: 'Update envelope fields',
|
||||||
|
description: 'Update multiple envelope fields for an envelope',
|
||||||
|
tags: ['Envelope Field'],
|
||||||
|
},
|
||||||
|
})
|
||||||
|
.input(ZUpdateEnvelopeFieldsRequestSchema)
|
||||||
|
.output(ZUpdateEnvelopeFieldsResponseSchema)
|
||||||
|
.mutation(async ({ input, ctx }) => {
|
||||||
|
const { user, teamId } = ctx;
|
||||||
|
const { envelopeId, data: fields } = input;
|
||||||
|
|
||||||
|
ctx.logger.info({
|
||||||
|
input: {
|
||||||
|
envelopeId,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
return await updateEnvelopeFields({
|
||||||
|
userId: user.id,
|
||||||
|
teamId,
|
||||||
|
id: {
|
||||||
|
type: 'envelopeId',
|
||||||
|
id: envelopeId,
|
||||||
|
},
|
||||||
|
type: null,
|
||||||
|
fields,
|
||||||
|
requestMetadata: ctx.metadata,
|
||||||
|
});
|
||||||
|
});
|
||||||
@ -0,0 +1,40 @@
|
|||||||
|
import { z } from 'zod';
|
||||||
|
|
||||||
|
import {
|
||||||
|
ZClampedFieldHeightSchema,
|
||||||
|
ZClampedFieldPageXSchema,
|
||||||
|
ZClampedFieldPageYSchema,
|
||||||
|
ZClampedFieldWidthSchema,
|
||||||
|
ZFieldPageNumberSchema,
|
||||||
|
ZFieldSchema,
|
||||||
|
} from '@documenso/lib/types/field';
|
||||||
|
import { ZFieldAndMetaSchema } from '@documenso/lib/types/field-meta';
|
||||||
|
|
||||||
|
const ZUpdateFieldSchema = ZFieldAndMetaSchema.and(
|
||||||
|
z.object({
|
||||||
|
id: z.number().describe('The ID of the field to update.'),
|
||||||
|
envelopeItemId: z
|
||||||
|
.string()
|
||||||
|
.optional()
|
||||||
|
.describe(
|
||||||
|
'The ID of the envelope item to put the field on. If not provided, field will be placed on the first item.',
|
||||||
|
),
|
||||||
|
pageNumber: ZFieldPageNumberSchema.optional(),
|
||||||
|
pageX: ZClampedFieldPageXSchema.optional(),
|
||||||
|
pageY: ZClampedFieldPageYSchema.optional(),
|
||||||
|
width: ZClampedFieldWidthSchema.optional(),
|
||||||
|
height: ZClampedFieldHeightSchema.optional(),
|
||||||
|
}),
|
||||||
|
);
|
||||||
|
|
||||||
|
export const ZUpdateEnvelopeFieldsRequestSchema = z.object({
|
||||||
|
envelopeId: z.string(),
|
||||||
|
data: ZUpdateFieldSchema.array(),
|
||||||
|
});
|
||||||
|
|
||||||
|
export const ZUpdateEnvelopeFieldsResponseSchema = z.object({
|
||||||
|
fields: z.array(ZFieldSchema),
|
||||||
|
});
|
||||||
|
|
||||||
|
export type TUpdateEnvelopeFieldsRequest = z.infer<typeof ZUpdateEnvelopeFieldsRequestSchema>;
|
||||||
|
export type TUpdateEnvelopeFieldsResponse = z.infer<typeof ZUpdateEnvelopeFieldsResponseSchema>;
|
||||||
@ -0,0 +1,41 @@
|
|||||||
|
import { createEnvelopeRecipients } from '@documenso/lib/server-only/recipient/create-envelope-recipients';
|
||||||
|
|
||||||
|
import { authenticatedProcedure } from '../../trpc';
|
||||||
|
import {
|
||||||
|
ZCreateEnvelopeRecipientsRequestSchema,
|
||||||
|
ZCreateEnvelopeRecipientsResponseSchema,
|
||||||
|
} from './create-envelope-recipients.types';
|
||||||
|
|
||||||
|
export const createEnvelopeRecipientsRoute = authenticatedProcedure
|
||||||
|
.meta({
|
||||||
|
openapi: {
|
||||||
|
method: 'POST',
|
||||||
|
path: '/envelope/recipient/create-many',
|
||||||
|
summary: 'Create envelope recipients',
|
||||||
|
description: 'Create multiple recipients for an envelope',
|
||||||
|
tags: ['Envelope Recipients'],
|
||||||
|
},
|
||||||
|
})
|
||||||
|
.input(ZCreateEnvelopeRecipientsRequestSchema)
|
||||||
|
.output(ZCreateEnvelopeRecipientsResponseSchema)
|
||||||
|
.mutation(async ({ input, ctx }) => {
|
||||||
|
const { user, teamId, metadata } = ctx;
|
||||||
|
const { envelopeId, data: recipients } = input;
|
||||||
|
|
||||||
|
ctx.logger.info({
|
||||||
|
input: {
|
||||||
|
envelopeId,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
return await createEnvelopeRecipients({
|
||||||
|
userId: user.id,
|
||||||
|
teamId,
|
||||||
|
id: {
|
||||||
|
type: 'envelopeId',
|
||||||
|
id: envelopeId,
|
||||||
|
},
|
||||||
|
recipients,
|
||||||
|
requestMetadata: metadata,
|
||||||
|
});
|
||||||
|
});
|
||||||
@ -0,0 +1,21 @@
|
|||||||
|
import { z } from 'zod';
|
||||||
|
|
||||||
|
import { ZEnvelopeRecipientLiteSchema } from '@documenso/lib/types/recipient';
|
||||||
|
|
||||||
|
import { ZCreateRecipientSchema } from '../../recipient-router/schema';
|
||||||
|
|
||||||
|
export const ZCreateEnvelopeRecipientsRequestSchema = z.object({
|
||||||
|
envelopeId: z.string(),
|
||||||
|
data: ZCreateRecipientSchema.array(),
|
||||||
|
});
|
||||||
|
|
||||||
|
export const ZCreateEnvelopeRecipientsResponseSchema = z.object({
|
||||||
|
recipients: ZEnvelopeRecipientLiteSchema.array(),
|
||||||
|
});
|
||||||
|
|
||||||
|
export type TCreateEnvelopeRecipientsRequest = z.infer<
|
||||||
|
typeof ZCreateEnvelopeRecipientsRequestSchema
|
||||||
|
>;
|
||||||
|
export type TCreateEnvelopeRecipientsResponse = z.infer<
|
||||||
|
typeof ZCreateEnvelopeRecipientsResponseSchema
|
||||||
|
>;
|
||||||
@ -0,0 +1,37 @@
|
|||||||
|
import { deleteEnvelopeRecipient } from '@documenso/lib/server-only/recipient/delete-envelope-recipient';
|
||||||
|
|
||||||
|
import { authenticatedProcedure } from '../../trpc';
|
||||||
|
import {
|
||||||
|
ZDeleteEnvelopeRecipientRequestSchema,
|
||||||
|
ZDeleteEnvelopeRecipientResponseSchema,
|
||||||
|
} from './delete-envelope-recipient.types';
|
||||||
|
|
||||||
|
export const deleteEnvelopeRecipientRoute = authenticatedProcedure
|
||||||
|
.meta({
|
||||||
|
openapi: {
|
||||||
|
method: 'POST',
|
||||||
|
path: '/envelope/recipient/delete',
|
||||||
|
summary: 'Delete envelope recipient',
|
||||||
|
description: 'Delete an envelope recipient',
|
||||||
|
tags: ['Envelope Recipient'],
|
||||||
|
},
|
||||||
|
})
|
||||||
|
.input(ZDeleteEnvelopeRecipientRequestSchema)
|
||||||
|
.output(ZDeleteEnvelopeRecipientResponseSchema)
|
||||||
|
.mutation(async ({ input, ctx }) => {
|
||||||
|
const { user, teamId, metadata } = ctx;
|
||||||
|
const { recipientId } = input;
|
||||||
|
|
||||||
|
ctx.logger.info({
|
||||||
|
input: {
|
||||||
|
recipientId,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
await deleteEnvelopeRecipient({
|
||||||
|
userId: user.id,
|
||||||
|
teamId,
|
||||||
|
recipientId,
|
||||||
|
requestMetadata: metadata,
|
||||||
|
});
|
||||||
|
});
|
||||||
@ -0,0 +1,12 @@
|
|||||||
|
import { z } from 'zod';
|
||||||
|
|
||||||
|
export const ZDeleteEnvelopeRecipientRequestSchema = z.object({
|
||||||
|
recipientId: z.number(),
|
||||||
|
});
|
||||||
|
|
||||||
|
export const ZDeleteEnvelopeRecipientResponseSchema = z.void();
|
||||||
|
|
||||||
|
export type TDeleteEnvelopeRecipientRequest = z.infer<typeof ZDeleteEnvelopeRecipientRequestSchema>;
|
||||||
|
export type TDeleteEnvelopeRecipientResponse = z.infer<
|
||||||
|
typeof ZDeleteEnvelopeRecipientResponseSchema
|
||||||
|
>;
|
||||||
@ -0,0 +1,52 @@
|
|||||||
|
import { AppError, AppErrorCode } from '@documenso/lib/errors/app-error';
|
||||||
|
import { buildTeamWhereQuery } from '@documenso/lib/utils/teams';
|
||||||
|
import { prisma } from '@documenso/prisma';
|
||||||
|
|
||||||
|
import { authenticatedProcedure } from '../../trpc';
|
||||||
|
import {
|
||||||
|
ZGetEnvelopeRecipientRequestSchema,
|
||||||
|
ZGetEnvelopeRecipientResponseSchema,
|
||||||
|
} from './get-envelope-recipient.types';
|
||||||
|
|
||||||
|
export const getEnvelopeRecipientRoute = authenticatedProcedure
|
||||||
|
.meta({
|
||||||
|
openapi: {
|
||||||
|
method: 'GET',
|
||||||
|
path: '/envelope/recipient/{recipientId}',
|
||||||
|
summary: 'Get envelope recipient',
|
||||||
|
description: 'Returns an envelope recipient given an ID',
|
||||||
|
tags: ['Envelope Recipient'],
|
||||||
|
},
|
||||||
|
})
|
||||||
|
.input(ZGetEnvelopeRecipientRequestSchema)
|
||||||
|
.output(ZGetEnvelopeRecipientResponseSchema)
|
||||||
|
.query(async ({ input, ctx }) => {
|
||||||
|
const { teamId, user } = ctx;
|
||||||
|
const { recipientId } = input;
|
||||||
|
|
||||||
|
ctx.logger.info({
|
||||||
|
input: {
|
||||||
|
recipientId,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
const recipient = await prisma.recipient.findFirst({
|
||||||
|
where: {
|
||||||
|
id: recipientId,
|
||||||
|
envelope: {
|
||||||
|
team: buildTeamWhereQuery({ teamId, userId: user.id }),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
include: {
|
||||||
|
fields: true,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!recipient) {
|
||||||
|
throw new AppError(AppErrorCode.NOT_FOUND, {
|
||||||
|
message: 'Recipient not found',
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
return recipient;
|
||||||
|
});
|
||||||
@ -0,0 +1,12 @@
|
|||||||
|
import { z } from 'zod';
|
||||||
|
|
||||||
|
import { ZEnvelopeRecipientSchema } from '@documenso/lib/types/recipient';
|
||||||
|
|
||||||
|
export const ZGetEnvelopeRecipientRequestSchema = z.object({
|
||||||
|
recipientId: z.number(),
|
||||||
|
});
|
||||||
|
|
||||||
|
export const ZGetEnvelopeRecipientResponseSchema = ZEnvelopeRecipientSchema;
|
||||||
|
|
||||||
|
export type TGetEnvelopeRecipientRequest = z.infer<typeof ZGetEnvelopeRecipientRequestSchema>;
|
||||||
|
export type TGetEnvelopeRecipientResponse = z.infer<typeof ZGetEnvelopeRecipientResponseSchema>;
|
||||||
@ -0,0 +1,41 @@
|
|||||||
|
import { updateEnvelopeRecipients } from '@documenso/lib/server-only/recipient/update-envelope-recipients';
|
||||||
|
|
||||||
|
import { authenticatedProcedure } from '../../trpc';
|
||||||
|
import {
|
||||||
|
ZUpdateEnvelopeRecipientsRequestSchema,
|
||||||
|
ZUpdateEnvelopeRecipientsResponseSchema,
|
||||||
|
} from './update-envelope-recipients.types';
|
||||||
|
|
||||||
|
export const updateEnvelopeRecipientsRoute = authenticatedProcedure
|
||||||
|
.meta({
|
||||||
|
openapi: {
|
||||||
|
method: 'POST',
|
||||||
|
path: '/envelope/recipient/update-many',
|
||||||
|
summary: 'Update envelope recipients',
|
||||||
|
description: 'Update multiple recipients for an envelope',
|
||||||
|
tags: ['Envelope Recipient'],
|
||||||
|
},
|
||||||
|
})
|
||||||
|
.input(ZUpdateEnvelopeRecipientsRequestSchema)
|
||||||
|
.output(ZUpdateEnvelopeRecipientsResponseSchema)
|
||||||
|
.mutation(async ({ input, ctx }) => {
|
||||||
|
const { user, teamId } = ctx;
|
||||||
|
const { envelopeId, data: recipients } = input;
|
||||||
|
|
||||||
|
ctx.logger.info({
|
||||||
|
input: {
|
||||||
|
envelopeId,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
return await updateEnvelopeRecipients({
|
||||||
|
userId: user.id,
|
||||||
|
teamId,
|
||||||
|
id: {
|
||||||
|
type: 'envelopeId',
|
||||||
|
id: envelopeId,
|
||||||
|
},
|
||||||
|
recipients,
|
||||||
|
requestMetadata: ctx.metadata,
|
||||||
|
});
|
||||||
|
});
|
||||||
@ -0,0 +1,21 @@
|
|||||||
|
import { z } from 'zod';
|
||||||
|
|
||||||
|
import { ZRecipientLiteSchema } from '@documenso/lib/types/recipient';
|
||||||
|
|
||||||
|
import { ZUpdateRecipientSchema } from '../../recipient-router/schema';
|
||||||
|
|
||||||
|
export const ZUpdateEnvelopeRecipientsRequestSchema = z.object({
|
||||||
|
envelopeId: z.string(),
|
||||||
|
data: ZUpdateRecipientSchema.array(),
|
||||||
|
});
|
||||||
|
|
||||||
|
export const ZUpdateEnvelopeRecipientsResponseSchema = z.object({
|
||||||
|
recipients: ZRecipientLiteSchema.array(),
|
||||||
|
});
|
||||||
|
|
||||||
|
export type TUpdateEnvelopeRecipientsRequest = z.infer<
|
||||||
|
typeof ZUpdateEnvelopeRecipientsRequestSchema
|
||||||
|
>;
|
||||||
|
export type TUpdateEnvelopeRecipientsResponse = z.infer<
|
||||||
|
typeof ZUpdateEnvelopeRecipientsResponseSchema
|
||||||
|
>;
|
||||||
@ -4,7 +4,15 @@ import { authenticatedProcedure } from '../trpc';
|
|||||||
import { ZGetEnvelopeRequestSchema, ZGetEnvelopeResponseSchema } from './get-envelope.types';
|
import { ZGetEnvelopeRequestSchema, ZGetEnvelopeResponseSchema } from './get-envelope.types';
|
||||||
|
|
||||||
export const getEnvelopeRoute = authenticatedProcedure
|
export const getEnvelopeRoute = authenticatedProcedure
|
||||||
// .meta(getEnvelopeMeta)
|
.meta({
|
||||||
|
openapi: {
|
||||||
|
method: 'GET',
|
||||||
|
path: '/envelope/{envelopeId}',
|
||||||
|
summary: 'Get envelope',
|
||||||
|
description: 'Returns an envelope given an ID',
|
||||||
|
tags: ['Envelope'],
|
||||||
|
},
|
||||||
|
})
|
||||||
.input(ZGetEnvelopeRequestSchema)
|
.input(ZGetEnvelopeRequestSchema)
|
||||||
.output(ZGetEnvelopeResponseSchema)
|
.output(ZGetEnvelopeResponseSchema)
|
||||||
.query(async ({ input, ctx }) => {
|
.query(async ({ input, ctx }) => {
|
||||||
|
|||||||
@ -2,16 +2,6 @@ import { z } from 'zod';
|
|||||||
|
|
||||||
import { ZEnvelopeSchema } from '@documenso/lib/types/envelope';
|
import { ZEnvelopeSchema } from '@documenso/lib/types/envelope';
|
||||||
|
|
||||||
// export const getEnvelopeMeta: TrpcRouteMeta = {
|
|
||||||
// openapi: {
|
|
||||||
// method: 'GET',
|
|
||||||
// path: '/envelope/{envelopeId}',
|
|
||||||
// summary: 'Get envelope',
|
|
||||||
// description: 'Returns a envelope given an ID',
|
|
||||||
// tags: ['Envelope'],
|
|
||||||
// },
|
|
||||||
// };
|
|
||||||
|
|
||||||
export const ZGetEnvelopeRequestSchema = z.object({
|
export const ZGetEnvelopeRequestSchema = z.object({
|
||||||
envelopeId: z.string(),
|
envelopeId: z.string(),
|
||||||
});
|
});
|
||||||
|
|||||||
@ -7,7 +7,16 @@ import {
|
|||||||
} from './redistribute-envelope.types';
|
} from './redistribute-envelope.types';
|
||||||
|
|
||||||
export const redistributeEnvelopeRoute = authenticatedProcedure
|
export const redistributeEnvelopeRoute = authenticatedProcedure
|
||||||
// .meta(redistributeEnvelopeMeta)
|
.meta({
|
||||||
|
openapi: {
|
||||||
|
method: 'POST',
|
||||||
|
path: '/envelope/redistribute',
|
||||||
|
summary: 'Redistribute envelope',
|
||||||
|
description:
|
||||||
|
'Redistribute the envelope to the provided recipients who have not actioned the envelope. Will use the distribution method set in the envelope',
|
||||||
|
tags: ['Envelope'],
|
||||||
|
},
|
||||||
|
})
|
||||||
.input(ZRedistributeEnvelopeRequestSchema)
|
.input(ZRedistributeEnvelopeRequestSchema)
|
||||||
.output(ZRedistributeEnvelopeResponseSchema)
|
.output(ZRedistributeEnvelopeResponseSchema)
|
||||||
.mutation(async ({ input, ctx }) => {
|
.mutation(async ({ input, ctx }) => {
|
||||||
|
|||||||
@ -1,16 +1,5 @@
|
|||||||
import { z } from 'zod';
|
import { z } from 'zod';
|
||||||
|
|
||||||
// export const redistributeEnvelopeMeta: TrpcRouteMeta = {
|
|
||||||
// openapi: {
|
|
||||||
// method: 'POST',
|
|
||||||
// path: '/envelope/redistribute',
|
|
||||||
// summary: 'Redistribute document',
|
|
||||||
// description:
|
|
||||||
// 'Redistribute the document to the provided recipients who have not actioned the document. Will use the distribution method set in the document',
|
|
||||||
// tags: ['Envelope'],
|
|
||||||
// },
|
|
||||||
// };
|
|
||||||
|
|
||||||
export const ZRedistributeEnvelopeRequestSchema = z.object({
|
export const ZRedistributeEnvelopeRequestSchema = z.object({
|
||||||
envelopeId: z.string(),
|
envelopeId: z.string(),
|
||||||
recipients: z
|
recipients: z
|
||||||
|
|||||||
@ -9,6 +9,14 @@ import { deleteEnvelopeRoute } from './delete-envelope';
|
|||||||
import { deleteEnvelopeItemRoute } from './delete-envelope-item';
|
import { deleteEnvelopeItemRoute } from './delete-envelope-item';
|
||||||
import { distributeEnvelopeRoute } from './distribute-envelope';
|
import { distributeEnvelopeRoute } from './distribute-envelope';
|
||||||
import { duplicateEnvelopeRoute } from './duplicate-envelope';
|
import { duplicateEnvelopeRoute } from './duplicate-envelope';
|
||||||
|
import { createEnvelopeFieldsRoute } from './envelope-fields/create-envelope-fields';
|
||||||
|
import { deleteEnvelopeFieldRoute } from './envelope-fields/delete-envelope-field';
|
||||||
|
import { getEnvelopeFieldRoute } from './envelope-fields/get-envelope-field';
|
||||||
|
import { updateEnvelopeFieldsRoute } from './envelope-fields/update-envelope-fields';
|
||||||
|
import { createEnvelopeRecipientsRoute } from './envelope-recipients/create-envelope-recipients';
|
||||||
|
import { deleteEnvelopeRecipientRoute } from './envelope-recipients/delete-envelope-recipient';
|
||||||
|
import { getEnvelopeRecipientRoute } from './envelope-recipients/get-envelope-recipient';
|
||||||
|
import { updateEnvelopeRecipientsRoute } from './envelope-recipients/update-envelope-recipients';
|
||||||
import { getEnvelopeRoute } from './get-envelope';
|
import { getEnvelopeRoute } from './get-envelope';
|
||||||
import { getEnvelopeItemsRoute } from './get-envelope-items';
|
import { getEnvelopeItemsRoute } from './get-envelope-items';
|
||||||
import { getEnvelopeItemsByTokenRoute } from './get-envelope-items-by-token';
|
import { getEnvelopeItemsByTokenRoute } from './get-envelope-items-by-token';
|
||||||
@ -27,8 +35,6 @@ export const envelopeRouter = router({
|
|||||||
duplicate: duplicateEnvelopeRoute,
|
duplicate: duplicateEnvelopeRoute,
|
||||||
distribute: distributeEnvelopeRoute,
|
distribute: distributeEnvelopeRoute,
|
||||||
redistribute: redistributeEnvelopeRoute,
|
redistribute: redistributeEnvelopeRoute,
|
||||||
// share: shareEnvelopeRoute,
|
|
||||||
|
|
||||||
item: {
|
item: {
|
||||||
getMany: getEnvelopeItemsRoute,
|
getMany: getEnvelopeItemsRoute,
|
||||||
getManyByToken: getEnvelopeItemsByTokenRoute,
|
getManyByToken: getEnvelopeItemsByTokenRoute,
|
||||||
@ -37,9 +43,17 @@ export const envelopeRouter = router({
|
|||||||
delete: deleteEnvelopeItemRoute,
|
delete: deleteEnvelopeItemRoute,
|
||||||
},
|
},
|
||||||
recipient: {
|
recipient: {
|
||||||
|
get: getEnvelopeRecipientRoute,
|
||||||
|
createMany: createEnvelopeRecipientsRoute,
|
||||||
|
updateMany: updateEnvelopeRecipientsRoute,
|
||||||
|
delete: deleteEnvelopeRecipientRoute,
|
||||||
set: setEnvelopeRecipientsRoute,
|
set: setEnvelopeRecipientsRoute,
|
||||||
},
|
},
|
||||||
field: {
|
field: {
|
||||||
|
get: getEnvelopeFieldRoute,
|
||||||
|
createMany: createEnvelopeFieldsRoute,
|
||||||
|
updateMany: updateEnvelopeFieldsRoute,
|
||||||
|
delete: deleteEnvelopeFieldRoute,
|
||||||
set: setEnvelopeFieldsRoute,
|
set: setEnvelopeFieldsRoute,
|
||||||
sign: signEnvelopeFieldRoute,
|
sign: signEnvelopeFieldRoute,
|
||||||
},
|
},
|
||||||
|
|||||||
@ -1,6 +1,12 @@
|
|||||||
import { EnvelopeType, FieldType } from '@prisma/client';
|
import { EnvelopeType, FieldType } from '@prisma/client';
|
||||||
import { z } from 'zod';
|
import { z } from 'zod';
|
||||||
|
|
||||||
|
import {
|
||||||
|
ZClampedFieldHeightSchema,
|
||||||
|
ZClampedFieldPageXSchema,
|
||||||
|
ZClampedFieldPageYSchema,
|
||||||
|
ZClampedFieldWidthSchema,
|
||||||
|
} from '@documenso/lib/types/field';
|
||||||
import { ZFieldMetaSchema } from '@documenso/lib/types/field-meta';
|
import { ZFieldMetaSchema } from '@documenso/lib/types/field-meta';
|
||||||
|
|
||||||
export const ZSetEnvelopeFieldsRequestSchema = z.object({
|
export const ZSetEnvelopeFieldsRequestSchema = z.object({
|
||||||
@ -20,28 +26,11 @@ export const ZSetEnvelopeFieldsRequestSchema = z.object({
|
|||||||
.number()
|
.number()
|
||||||
.min(1)
|
.min(1)
|
||||||
.describe('The page number of the field on the envelope. Starts from 1.'),
|
.describe('The page number of the field on the envelope. Starts from 1.'),
|
||||||
// Todo: Envelopes - Extract these 0-100 schemas with better descriptions.
|
positionX: ZClampedFieldPageXSchema,
|
||||||
positionX: z
|
positionY: ZClampedFieldPageYSchema,
|
||||||
.number()
|
width: ZClampedFieldWidthSchema,
|
||||||
.min(0)
|
height: ZClampedFieldHeightSchema,
|
||||||
.max(100)
|
fieldMeta: ZFieldMetaSchema,
|
||||||
.describe('The percentage based X position of the field on the envelope.'),
|
|
||||||
positionY: z
|
|
||||||
.number()
|
|
||||||
.min(0)
|
|
||||||
.max(100)
|
|
||||||
.describe('The percentage based Y position of the field on the envelope.'),
|
|
||||||
width: z
|
|
||||||
.number()
|
|
||||||
.min(0)
|
|
||||||
.max(100)
|
|
||||||
.describe('The percentage based width of the field on the envelope.'),
|
|
||||||
height: z
|
|
||||||
.number()
|
|
||||||
.min(0)
|
|
||||||
.max(100)
|
|
||||||
.describe('The percentage based height of the field on the envelope.'),
|
|
||||||
fieldMeta: ZFieldMetaSchema, // Todo: Envelopes - Use a more strict form?
|
|
||||||
}),
|
}),
|
||||||
),
|
),
|
||||||
});
|
});
|
||||||
|
|||||||
@ -10,6 +10,15 @@ import {
|
|||||||
} from './update-envelope-items.types';
|
} from './update-envelope-items.types';
|
||||||
|
|
||||||
export const updateEnvelopeItemsRoute = authenticatedProcedure
|
export const updateEnvelopeItemsRoute = authenticatedProcedure
|
||||||
|
.meta({
|
||||||
|
openapi: {
|
||||||
|
method: 'POST',
|
||||||
|
path: '/envelope/item/update-many',
|
||||||
|
summary: 'Update envelope items',
|
||||||
|
description: 'Update multiple envelope items for an envelope',
|
||||||
|
tags: ['Envelope Item'],
|
||||||
|
},
|
||||||
|
})
|
||||||
.input(ZUpdateEnvelopeItemsRequestSchema)
|
.input(ZUpdateEnvelopeItemsRequestSchema)
|
||||||
.output(ZUpdateEnvelopeItemsResponseSchema)
|
.output(ZUpdateEnvelopeItemsResponseSchema)
|
||||||
.mutation(async ({ input, ctx }) => {
|
.mutation(async ({ input, ctx }) => {
|
||||||
|
|||||||
@ -7,7 +7,14 @@ import {
|
|||||||
} from './update-envelope.types';
|
} from './update-envelope.types';
|
||||||
|
|
||||||
export const updateEnvelopeRoute = authenticatedProcedure
|
export const updateEnvelopeRoute = authenticatedProcedure
|
||||||
// .meta(updateEnvelopeTrpcMeta)
|
.meta({
|
||||||
|
openapi: {
|
||||||
|
method: 'POST',
|
||||||
|
path: '/envelope/update',
|
||||||
|
summary: 'Update envelope',
|
||||||
|
tags: ['Envelope'],
|
||||||
|
},
|
||||||
|
})
|
||||||
.input(ZUpdateEnvelopeRequestSchema)
|
.input(ZUpdateEnvelopeRequestSchema)
|
||||||
.output(ZUpdateEnvelopeResponseSchema)
|
.output(ZUpdateEnvelopeResponseSchema)
|
||||||
.mutation(async ({ input, ctx }) => {
|
.mutation(async ({ input, ctx }) => {
|
||||||
|
|||||||
@ -15,15 +15,6 @@ import {
|
|||||||
ZDocumentVisibilitySchema,
|
ZDocumentVisibilitySchema,
|
||||||
} from '../document-router/schema';
|
} from '../document-router/schema';
|
||||||
|
|
||||||
// export const updateEnvelopeMeta: TrpcRouteMeta = {
|
|
||||||
// openapi: {
|
|
||||||
// method: 'POST',
|
|
||||||
// path: '/envelope/update',
|
|
||||||
// summary: 'Update envelope',
|
|
||||||
// tags: ['Envelope'],
|
|
||||||
// },
|
|
||||||
// };
|
|
||||||
|
|
||||||
export const ZUpdateEnvelopeRequestSchema = z.object({
|
export const ZUpdateEnvelopeRequestSchema = z.object({
|
||||||
envelopeId: z.string(),
|
envelopeId: z.string(),
|
||||||
envelopeType: z.nativeEnum(EnvelopeType),
|
envelopeType: z.nativeEnum(EnvelopeType),
|
||||||
|
|||||||
@ -8,8 +8,7 @@ import { removeSignedFieldWithToken } from '@documenso/lib/server-only/field/rem
|
|||||||
import { setFieldsForDocument } from '@documenso/lib/server-only/field/set-fields-for-document';
|
import { setFieldsForDocument } from '@documenso/lib/server-only/field/set-fields-for-document';
|
||||||
import { setFieldsForTemplate } from '@documenso/lib/server-only/field/set-fields-for-template';
|
import { setFieldsForTemplate } from '@documenso/lib/server-only/field/set-fields-for-template';
|
||||||
import { signFieldWithToken } from '@documenso/lib/server-only/field/sign-field-with-token';
|
import { signFieldWithToken } from '@documenso/lib/server-only/field/sign-field-with-token';
|
||||||
import { updateDocumentFields } from '@documenso/lib/server-only/field/update-document-fields';
|
import { updateEnvelopeFields } from '@documenso/lib/server-only/field/update-envelope-fields';
|
||||||
import { updateTemplateFields } from '@documenso/lib/server-only/field/update-template-fields';
|
|
||||||
|
|
||||||
import { ZGenericSuccessResponse, ZSuccessResponseSchema } from '../document-router/schema';
|
import { ZGenericSuccessResponse, ZSuccessResponseSchema } from '../document-router/schema';
|
||||||
import { authenticatedProcedure, procedure, router } from '../trpc';
|
import { authenticatedProcedure, procedure, router } from '../trpc';
|
||||||
@ -178,10 +177,14 @@ export const fieldRouter = router({
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
const updatedFields = await updateDocumentFields({
|
const updatedFields = await updateEnvelopeFields({
|
||||||
userId: ctx.user.id,
|
userId: ctx.user.id,
|
||||||
teamId,
|
teamId,
|
||||||
documentId,
|
id: {
|
||||||
|
type: 'documentId',
|
||||||
|
id: documentId,
|
||||||
|
},
|
||||||
|
type: EnvelopeType.DOCUMENT,
|
||||||
fields: [field],
|
fields: [field],
|
||||||
requestMetadata: ctx.metadata,
|
requestMetadata: ctx.metadata,
|
||||||
});
|
});
|
||||||
@ -214,10 +217,14 @@ export const fieldRouter = router({
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
return await updateDocumentFields({
|
return await updateEnvelopeFields({
|
||||||
userId: ctx.user.id,
|
userId: ctx.user.id,
|
||||||
teamId,
|
teamId,
|
||||||
documentId,
|
id: {
|
||||||
|
type: 'documentId',
|
||||||
|
id: documentId,
|
||||||
|
},
|
||||||
|
type: EnvelopeType.DOCUMENT,
|
||||||
fields,
|
fields,
|
||||||
requestMetadata: ctx.metadata,
|
requestMetadata: ctx.metadata,
|
||||||
});
|
});
|
||||||
@ -431,11 +438,16 @@ export const fieldRouter = router({
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
const updatedFields = await updateTemplateFields({
|
const updatedFields = await updateEnvelopeFields({
|
||||||
userId: ctx.user.id,
|
userId: ctx.user.id,
|
||||||
teamId,
|
teamId,
|
||||||
templateId,
|
id: {
|
||||||
|
type: 'templateId',
|
||||||
|
id: templateId,
|
||||||
|
},
|
||||||
|
type: EnvelopeType.TEMPLATE,
|
||||||
fields: [field],
|
fields: [field],
|
||||||
|
requestMetadata: ctx.metadata,
|
||||||
});
|
});
|
||||||
|
|
||||||
return updatedFields.fields[0];
|
return updatedFields.fields[0];
|
||||||
@ -466,11 +478,16 @@ export const fieldRouter = router({
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
return await updateTemplateFields({
|
return await updateEnvelopeFields({
|
||||||
userId: ctx.user.id,
|
userId: ctx.user.id,
|
||||||
teamId,
|
teamId,
|
||||||
templateId,
|
id: {
|
||||||
|
type: 'templateId',
|
||||||
|
id: templateId,
|
||||||
|
},
|
||||||
|
type: EnvelopeType.TEMPLATE,
|
||||||
fields,
|
fields,
|
||||||
|
requestMetadata: ctx.metadata,
|
||||||
});
|
});
|
||||||
}),
|
}),
|
||||||
|
|
||||||
|
|||||||
@ -2,15 +2,12 @@ import { EnvelopeType } from '@prisma/client';
|
|||||||
|
|
||||||
import { completeDocumentWithToken } from '@documenso/lib/server-only/document/complete-document-with-token';
|
import { completeDocumentWithToken } from '@documenso/lib/server-only/document/complete-document-with-token';
|
||||||
import { rejectDocumentWithToken } from '@documenso/lib/server-only/document/reject-document-with-token';
|
import { rejectDocumentWithToken } from '@documenso/lib/server-only/document/reject-document-with-token';
|
||||||
import { createDocumentRecipients } from '@documenso/lib/server-only/recipient/create-document-recipients';
|
import { createEnvelopeRecipients } from '@documenso/lib/server-only/recipient/create-envelope-recipients';
|
||||||
import { createTemplateRecipients } from '@documenso/lib/server-only/recipient/create-template-recipients';
|
import { deleteEnvelopeRecipient } from '@documenso/lib/server-only/recipient/delete-envelope-recipient';
|
||||||
import { deleteDocumentRecipient } from '@documenso/lib/server-only/recipient/delete-document-recipient';
|
|
||||||
import { deleteTemplateRecipient } from '@documenso/lib/server-only/recipient/delete-template-recipient';
|
|
||||||
import { getRecipientById } from '@documenso/lib/server-only/recipient/get-recipient-by-id';
|
import { getRecipientById } from '@documenso/lib/server-only/recipient/get-recipient-by-id';
|
||||||
import { setDocumentRecipients } from '@documenso/lib/server-only/recipient/set-document-recipients';
|
import { setDocumentRecipients } from '@documenso/lib/server-only/recipient/set-document-recipients';
|
||||||
import { setTemplateRecipients } from '@documenso/lib/server-only/recipient/set-template-recipients';
|
import { setTemplateRecipients } from '@documenso/lib/server-only/recipient/set-template-recipients';
|
||||||
import { updateDocumentRecipients } from '@documenso/lib/server-only/recipient/update-document-recipients';
|
import { updateEnvelopeRecipients } from '@documenso/lib/server-only/recipient/update-envelope-recipients';
|
||||||
import { updateTemplateRecipients } from '@documenso/lib/server-only/recipient/update-template-recipients';
|
|
||||||
|
|
||||||
import { ZGenericSuccessResponse, ZSuccessResponseSchema } from '../document-router/schema';
|
import { ZGenericSuccessResponse, ZSuccessResponseSchema } from '../document-router/schema';
|
||||||
import { authenticatedProcedure, procedure, router } from '../trpc';
|
import { authenticatedProcedure, procedure, router } from '../trpc';
|
||||||
@ -108,7 +105,7 @@ export const recipientRouter = router({
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
const createdRecipients = await createDocumentRecipients({
|
const createdRecipients = await createEnvelopeRecipients({
|
||||||
userId: ctx.user.id,
|
userId: ctx.user.id,
|
||||||
teamId,
|
teamId,
|
||||||
id: {
|
id: {
|
||||||
@ -147,7 +144,7 @@ export const recipientRouter = router({
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
return await createDocumentRecipients({
|
return await createEnvelopeRecipients({
|
||||||
userId: ctx.user.id,
|
userId: ctx.user.id,
|
||||||
teamId,
|
teamId,
|
||||||
id: {
|
id: {
|
||||||
@ -184,7 +181,7 @@ export const recipientRouter = router({
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
const updatedRecipients = await updateDocumentRecipients({
|
const updatedRecipients = await updateEnvelopeRecipients({
|
||||||
userId: ctx.user.id,
|
userId: ctx.user.id,
|
||||||
teamId,
|
teamId,
|
||||||
id: {
|
id: {
|
||||||
@ -223,7 +220,7 @@ export const recipientRouter = router({
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
return await updateDocumentRecipients({
|
return await updateEnvelopeRecipients({
|
||||||
userId: ctx.user.id,
|
userId: ctx.user.id,
|
||||||
teamId,
|
teamId,
|
||||||
id: {
|
id: {
|
||||||
@ -259,7 +256,7 @@ export const recipientRouter = router({
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
await deleteDocumentRecipient({
|
await deleteEnvelopeRecipient({
|
||||||
userId: ctx.user.id,
|
userId: ctx.user.id,
|
||||||
teamId,
|
teamId,
|
||||||
recipientId,
|
recipientId,
|
||||||
@ -363,11 +360,15 @@ export const recipientRouter = router({
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
const createdRecipients = await createTemplateRecipients({
|
const createdRecipients = await createEnvelopeRecipients({
|
||||||
userId: ctx.user.id,
|
userId: ctx.user.id,
|
||||||
teamId,
|
teamId,
|
||||||
templateId,
|
id: {
|
||||||
|
id: templateId,
|
||||||
|
type: 'templateId',
|
||||||
|
},
|
||||||
recipients: [recipient],
|
recipients: [recipient],
|
||||||
|
requestMetadata: ctx.metadata,
|
||||||
});
|
});
|
||||||
|
|
||||||
return createdRecipients.recipients[0];
|
return createdRecipients.recipients[0];
|
||||||
@ -398,11 +399,15 @@ export const recipientRouter = router({
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
return await createTemplateRecipients({
|
return await createEnvelopeRecipients({
|
||||||
userId: ctx.user.id,
|
userId: ctx.user.id,
|
||||||
teamId,
|
teamId,
|
||||||
templateId,
|
id: {
|
||||||
|
id: templateId,
|
||||||
|
type: 'templateId',
|
||||||
|
},
|
||||||
recipients,
|
recipients,
|
||||||
|
requestMetadata: ctx.metadata,
|
||||||
});
|
});
|
||||||
}),
|
}),
|
||||||
|
|
||||||
@ -431,11 +436,15 @@ export const recipientRouter = router({
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
const updatedRecipients = await updateTemplateRecipients({
|
const updatedRecipients = await updateEnvelopeRecipients({
|
||||||
userId: ctx.user.id,
|
userId: ctx.user.id,
|
||||||
teamId,
|
teamId,
|
||||||
templateId,
|
id: {
|
||||||
|
type: 'templateId',
|
||||||
|
id: templateId,
|
||||||
|
},
|
||||||
recipients: [recipient],
|
recipients: [recipient],
|
||||||
|
requestMetadata: ctx.metadata,
|
||||||
});
|
});
|
||||||
|
|
||||||
return updatedRecipients.recipients[0];
|
return updatedRecipients.recipients[0];
|
||||||
@ -466,11 +475,15 @@ export const recipientRouter = router({
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
return await updateTemplateRecipients({
|
return await updateEnvelopeRecipients({
|
||||||
userId: ctx.user.id,
|
userId: ctx.user.id,
|
||||||
teamId,
|
teamId,
|
||||||
templateId,
|
id: {
|
||||||
|
type: 'templateId',
|
||||||
|
id: templateId,
|
||||||
|
},
|
||||||
recipients,
|
recipients,
|
||||||
|
requestMetadata: ctx.metadata,
|
||||||
});
|
});
|
||||||
}),
|
}),
|
||||||
|
|
||||||
@ -498,10 +511,11 @@ export const recipientRouter = router({
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
await deleteTemplateRecipient({
|
await deleteEnvelopeRecipient({
|
||||||
recipientId,
|
recipientId,
|
||||||
userId: ctx.user.id,
|
userId: ctx.user.id,
|
||||||
teamId,
|
teamId,
|
||||||
|
requestMetadata: ctx.metadata,
|
||||||
});
|
});
|
||||||
|
|
||||||
return ZGenericSuccessResponse;
|
return ZGenericSuccessResponse;
|
||||||
|
|||||||
Reference in New Issue
Block a user