2022-11-03 14:40:03 +00:00
|
|
|
import { App_RoutingForms_Form, Prisma, User, WebhookTriggerEvents } from "@prisma/client";
|
2022-07-14 12:40:53 +00:00
|
|
|
import { z } from "zod";
|
|
|
|
|
2022-10-12 13:04:51 +00:00
|
|
|
import getWebhooks from "@calcom/features/webhooks/lib/getWebhooks";
|
|
|
|
import { sendGenericWebhookPayload } from "@calcom/features/webhooks/lib/sendPayload";
|
2022-11-03 14:40:03 +00:00
|
|
|
import logger from "@calcom/lib/logger";
|
|
|
|
import { RoutingFormSettings } from "@calcom/prisma/zod-utils";
|
2022-07-22 17:27:06 +00:00
|
|
|
import { TRPCError } from "@calcom/trpc/server";
|
2022-11-10 23:40:01 +00:00
|
|
|
import { authedProcedure, publicProcedure, router } from "@calcom/trpc/server/trpc";
|
2022-11-03 14:40:03 +00:00
|
|
|
import { Ensure } from "@calcom/types/utils";
|
2022-07-22 17:27:06 +00:00
|
|
|
|
2022-11-03 14:40:03 +00:00
|
|
|
import ResponseEmail from "./emails/templates/response-email";
|
2022-11-11 09:57:44 +00:00
|
|
|
import { jsonLogicToPrisma } from "./jsonLogicToPrisma";
|
2023-01-04 13:30:46 +00:00
|
|
|
import { createFallbackRoute } from "./lib/createFallbackRoute";
|
|
|
|
import getConnectedForms from "./lib/getConnectedForms";
|
2022-10-06 06:49:03 +00:00
|
|
|
import { getSerializableForm } from "./lib/getSerializableForm";
|
2023-01-04 13:30:46 +00:00
|
|
|
import { isFallbackRoute } from "./lib/isFallbackRoute";
|
|
|
|
import { isFormEditAllowed } from "./lib/isFormEditAllowed";
|
|
|
|
import isRouter from "./lib/isRouter";
|
|
|
|
import isRouterLinkedField from "./lib/isRouterLinkedField";
|
2022-11-03 14:40:03 +00:00
|
|
|
import { Response, SerializableForm } from "./types/types";
|
2023-01-04 13:30:46 +00:00
|
|
|
import { zodFields, zodRouterRoute, zodRoutes } from "./zod";
|
2022-07-14 12:40:53 +00:00
|
|
|
|
2022-11-03 14:40:03 +00:00
|
|
|
async function onFormSubmission(
|
|
|
|
form: Ensure<SerializableForm<App_RoutingForms_Form> & { user: User }, "fields">,
|
|
|
|
response: Response
|
|
|
|
) {
|
|
|
|
const fieldResponsesByName: Record<string, typeof response[keyof typeof response]["value"]> = {};
|
|
|
|
|
|
|
|
for (const [fieldId, fieldResponse] of Object.entries(response)) {
|
|
|
|
// Use the label lowercased as the key to identify a field.
|
|
|
|
const key =
|
|
|
|
form.fields.find((f) => f.id === fieldId)?.identifier ||
|
|
|
|
(fieldResponse.label as keyof typeof fieldResponsesByName);
|
|
|
|
fieldResponsesByName[key] = fieldResponse.value;
|
|
|
|
}
|
|
|
|
|
|
|
|
const subscriberOptions = {
|
|
|
|
userId: form.user.id,
|
|
|
|
// It isn't an eventType webhook
|
|
|
|
eventTypeId: -1,
|
|
|
|
triggerEvent: WebhookTriggerEvents.FORM_SUBMITTED,
|
|
|
|
};
|
|
|
|
|
|
|
|
const webhooks = await getWebhooks(subscriberOptions);
|
|
|
|
const promises = webhooks.map((webhook) => {
|
|
|
|
sendGenericWebhookPayload(
|
|
|
|
webhook.secret,
|
|
|
|
"FORM_SUBMITTED",
|
|
|
|
new Date().toISOString(),
|
|
|
|
webhook,
|
|
|
|
fieldResponsesByName
|
|
|
|
).catch((e) => {
|
|
|
|
console.error(`Error executing routing form webhook`, webhook, e);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
await Promise.all(promises);
|
|
|
|
if (form.settings?.emailOwnerOnSubmission) {
|
|
|
|
logger.debug(
|
|
|
|
`Preparing to send Form Response email for Form:${form.id} to form owner: ${form.user.email}`
|
|
|
|
);
|
|
|
|
await sendResponseEmail(form, response, form.user.email);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const sendResponseEmail = async (
|
|
|
|
form: Pick<App_RoutingForms_Form, "id" | "name">,
|
|
|
|
response: Response,
|
|
|
|
ownerEmail: string
|
|
|
|
) => {
|
|
|
|
try {
|
|
|
|
const email = new ResponseEmail({ form: form, toAddresses: [ownerEmail], response: response });
|
|
|
|
await email.sendEmail();
|
|
|
|
} catch (e) {
|
|
|
|
logger.error("Error sending response email", e);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2022-11-10 23:40:01 +00:00
|
|
|
const appRoutingForms = router({
|
|
|
|
public: router({
|
|
|
|
response: publicProcedure
|
|
|
|
.input(
|
|
|
|
z.object({
|
|
|
|
formId: z.string(),
|
|
|
|
formFillerId: z.string(),
|
|
|
|
response: z.record(
|
|
|
|
z.object({
|
|
|
|
label: z.string(),
|
|
|
|
value: z.union([z.string(), z.array(z.string())]),
|
|
|
|
})
|
|
|
|
),
|
|
|
|
})
|
|
|
|
)
|
|
|
|
.mutation(async ({ ctx, input }) => {
|
|
|
|
const { prisma } = ctx;
|
2022-07-19 18:36:39 +00:00
|
|
|
try {
|
|
|
|
const { response, formId } = input;
|
|
|
|
const form = await prisma.app_RoutingForms_Form.findFirst({
|
|
|
|
where: {
|
|
|
|
id: formId,
|
2022-07-14 12:40:53 +00:00
|
|
|
},
|
2022-07-20 18:30:57 +00:00
|
|
|
include: {
|
|
|
|
user: true,
|
|
|
|
},
|
2022-07-14 12:40:53 +00:00
|
|
|
});
|
2022-07-19 18:36:39 +00:00
|
|
|
if (!form) {
|
|
|
|
throw new TRPCError({
|
|
|
|
code: "NOT_FOUND",
|
|
|
|
});
|
|
|
|
}
|
2022-07-14 12:40:53 +00:00
|
|
|
|
2023-01-04 13:30:46 +00:00
|
|
|
const serializableForm = await getSerializableForm(form);
|
2022-11-03 14:40:03 +00:00
|
|
|
if (!serializableForm.fields) {
|
2022-07-19 18:36:39 +00:00
|
|
|
// There is no point in submitting a form that doesn't have fields defined
|
|
|
|
throw new TRPCError({
|
|
|
|
code: "BAD_REQUEST",
|
|
|
|
});
|
|
|
|
}
|
2022-07-14 12:40:53 +00:00
|
|
|
|
2022-11-03 14:40:03 +00:00
|
|
|
const serializableFormWithFields = {
|
|
|
|
...serializableForm,
|
|
|
|
fields: serializableForm.fields,
|
|
|
|
};
|
|
|
|
|
|
|
|
const missingFields = serializableFormWithFields.fields
|
2022-07-19 18:36:39 +00:00
|
|
|
.filter((field) => !(field.required ? response[field.id]?.value : true))
|
|
|
|
.map((f) => f.label);
|
2022-07-14 12:40:53 +00:00
|
|
|
|
2022-07-19 18:36:39 +00:00
|
|
|
if (missingFields.length) {
|
|
|
|
throw new TRPCError({
|
|
|
|
code: "BAD_REQUEST",
|
|
|
|
message: `Missing required fields ${missingFields.join(", ")}`,
|
|
|
|
});
|
|
|
|
}
|
2022-11-03 14:40:03 +00:00
|
|
|
const invalidFields = serializableFormWithFields.fields
|
2022-07-19 18:36:39 +00:00
|
|
|
.filter((field) => {
|
|
|
|
const fieldValue = response[field.id]?.value;
|
|
|
|
// The field isn't required at this point. Validate only if it's set
|
|
|
|
if (!fieldValue) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
let schema;
|
|
|
|
if (field.type === "email") {
|
|
|
|
schema = z.string().email();
|
|
|
|
} else if (field.type === "phone") {
|
|
|
|
schema = z.any();
|
|
|
|
} else {
|
|
|
|
schema = z.any();
|
|
|
|
}
|
|
|
|
return !schema.safeParse(fieldValue).success;
|
|
|
|
})
|
|
|
|
.map((f) => ({ label: f.label, type: f.type }));
|
|
|
|
|
|
|
|
if (invalidFields.length) {
|
|
|
|
throw new TRPCError({
|
|
|
|
code: "BAD_REQUEST",
|
|
|
|
message: `Invalid fields ${invalidFields.map((f) => `${f.label}: ${f.type}`)}`,
|
|
|
|
});
|
|
|
|
}
|
2022-07-14 12:40:53 +00:00
|
|
|
|
2022-11-03 14:40:03 +00:00
|
|
|
const dbFormResponse = await prisma.app_RoutingForms_FormResponse.create({
|
2022-07-19 18:36:39 +00:00
|
|
|
data: input,
|
2022-07-14 12:40:53 +00:00
|
|
|
});
|
2022-11-03 14:40:03 +00:00
|
|
|
|
|
|
|
await onFormSubmission(serializableFormWithFields, dbFormResponse.response as Response);
|
|
|
|
return dbFormResponse;
|
2022-07-19 18:36:39 +00:00
|
|
|
} catch (e) {
|
|
|
|
if (e instanceof Prisma.PrismaClientKnownRequestError) {
|
|
|
|
if (e.code === "P2002") {
|
|
|
|
throw new TRPCError({
|
|
|
|
code: "CONFLICT",
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
throw e;
|
2022-07-14 12:40:53 +00:00
|
|
|
}
|
2022-11-10 23:40:01 +00:00
|
|
|
}),
|
|
|
|
}),
|
|
|
|
forms: authedProcedure.query(async ({ ctx }) => {
|
|
|
|
const { prisma, user } = ctx;
|
|
|
|
const forms = await prisma.app_RoutingForms_Form.findMany({
|
|
|
|
where: {
|
|
|
|
userId: user.id,
|
2022-07-19 18:36:39 +00:00
|
|
|
},
|
2022-11-10 23:40:01 +00:00
|
|
|
orderBy: {
|
|
|
|
createdAt: "desc",
|
|
|
|
},
|
|
|
|
include: {
|
|
|
|
_count: {
|
|
|
|
select: {
|
|
|
|
responses: true,
|
|
|
|
},
|
2022-07-19 18:36:39 +00:00
|
|
|
},
|
2022-11-10 23:40:01 +00:00
|
|
|
},
|
|
|
|
});
|
|
|
|
|
2023-01-04 13:30:46 +00:00
|
|
|
const serializableForms = [];
|
|
|
|
for (let i = 0; i < forms.length; i++) {
|
|
|
|
serializableForms.push(await getSerializableForm(forms[i]));
|
|
|
|
}
|
2022-11-10 23:40:01 +00:00
|
|
|
return serializableForms;
|
|
|
|
}),
|
|
|
|
formQuery: authedProcedure
|
|
|
|
.input(
|
|
|
|
z.object({
|
|
|
|
id: z.string(),
|
2022-07-19 18:36:39 +00:00
|
|
|
})
|
2022-11-10 23:40:01 +00:00
|
|
|
)
|
|
|
|
.query(async ({ ctx, input }) => {
|
|
|
|
const { prisma, user } = ctx;
|
|
|
|
const form = await prisma.app_RoutingForms_Form.findFirst({
|
|
|
|
where: {
|
|
|
|
userId: user.id,
|
|
|
|
id: input.id,
|
|
|
|
},
|
|
|
|
include: {
|
|
|
|
_count: {
|
|
|
|
select: {
|
|
|
|
responses: true,
|
2022-10-11 15:52:18 +00:00
|
|
|
},
|
2022-11-10 23:40:01 +00:00
|
|
|
},
|
2022-07-19 18:36:39 +00:00
|
|
|
},
|
2022-11-10 23:40:01 +00:00
|
|
|
});
|
2022-08-13 11:04:57 +00:00
|
|
|
|
2022-11-10 23:40:01 +00:00
|
|
|
if (!form) {
|
|
|
|
return null;
|
|
|
|
}
|
2022-08-13 11:04:57 +00:00
|
|
|
|
2023-01-04 13:30:46 +00:00
|
|
|
return await getSerializableForm(form);
|
2022-11-10 23:40:01 +00:00
|
|
|
}),
|
|
|
|
formMutation: authedProcedure
|
|
|
|
.input(
|
|
|
|
z.object({
|
|
|
|
id: z.string(),
|
|
|
|
name: z.string(),
|
|
|
|
description: z.string().nullable().optional(),
|
|
|
|
disabled: z.boolean().optional(),
|
|
|
|
fields: zodFields,
|
|
|
|
routes: zodRoutes,
|
|
|
|
addFallback: z.boolean().optional(),
|
|
|
|
duplicateFrom: z.string().nullable().optional(),
|
2023-01-04 13:30:46 +00:00
|
|
|
shouldConnect: z.boolean().optional(),
|
2022-11-10 23:40:01 +00:00
|
|
|
settings: RoutingFormSettings.optional(),
|
|
|
|
})
|
|
|
|
)
|
|
|
|
.mutation(async ({ ctx, input }) => {
|
|
|
|
const { user, prisma } = ctx;
|
2023-01-04 13:30:46 +00:00
|
|
|
const { name, id, description, settings, disabled, addFallback, duplicateFrom, shouldConnect } = input;
|
|
|
|
if (!(await isFormEditAllowed({ userId: user.id, formId: id }))) {
|
2022-11-10 23:40:01 +00:00
|
|
|
throw new TRPCError({
|
|
|
|
code: "FORBIDDEN",
|
|
|
|
});
|
|
|
|
}
|
2023-01-04 13:30:46 +00:00
|
|
|
let { routes: inputRoutes } = input;
|
|
|
|
let { fields: inputFields } = input;
|
|
|
|
inputFields = inputFields || [];
|
|
|
|
inputRoutes = inputRoutes || [];
|
|
|
|
type InputFields = typeof inputFields;
|
|
|
|
type InputRoutes = typeof inputRoutes;
|
|
|
|
let routes: InputRoutes;
|
|
|
|
let fields: InputFields;
|
|
|
|
type DuplicateFrom = NonNullable<typeof duplicateFrom>;
|
2022-07-19 18:36:39 +00:00
|
|
|
|
2023-01-04 13:30:46 +00:00
|
|
|
const dbForm = await prisma.app_RoutingForms_Form.findUnique({
|
2022-11-10 23:40:01 +00:00
|
|
|
where: {
|
|
|
|
id: id,
|
|
|
|
},
|
|
|
|
select: {
|
|
|
|
id: true,
|
|
|
|
user: true,
|
|
|
|
name: true,
|
|
|
|
description: true,
|
|
|
|
userId: true,
|
|
|
|
disabled: true,
|
|
|
|
createdAt: true,
|
|
|
|
updatedAt: true,
|
|
|
|
routes: true,
|
|
|
|
fields: true,
|
|
|
|
settings: true,
|
|
|
|
},
|
|
|
|
});
|
2022-11-08 14:21:53 +00:00
|
|
|
|
2023-01-04 13:30:46 +00:00
|
|
|
const dbSerializedForm = dbForm ? await getSerializableForm(dbForm, true) : null;
|
2022-07-19 18:36:39 +00:00
|
|
|
|
2023-01-04 13:30:46 +00:00
|
|
|
if (duplicateFrom) {
|
|
|
|
({ routes, fields } = await getRoutesAndFieldsForDuplication(duplicateFrom));
|
|
|
|
} else {
|
|
|
|
[fields, routes] = [inputFields, inputRoutes];
|
|
|
|
if (dbSerializedForm) {
|
|
|
|
fields = markMissingFieldsDeleted(dbSerializedForm, fields);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (dbSerializedForm) {
|
|
|
|
// If it's an existing form being mutated, update fields in the connected forms(if any).
|
|
|
|
await updateFieldsInConnectedForms(dbSerializedForm, inputFields);
|
2022-11-10 23:40:01 +00:00
|
|
|
}
|
|
|
|
|
2023-01-04 13:30:46 +00:00
|
|
|
fields = await getUpdatedRouterLinkedFields(fields, routes);
|
|
|
|
|
2022-11-10 23:40:01 +00:00
|
|
|
if (addFallback) {
|
|
|
|
// Add a fallback route if there is none
|
2023-01-04 13:30:46 +00:00
|
|
|
if (!routes.find(isFallbackRoute)) {
|
|
|
|
routes.push(createFallbackRoute());
|
2022-11-10 23:40:01 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return await prisma.app_RoutingForms_Form.upsert({
|
|
|
|
where: {
|
|
|
|
id: id,
|
2022-07-19 18:36:39 +00:00
|
|
|
},
|
2022-11-10 23:40:01 +00:00
|
|
|
create: {
|
|
|
|
user: {
|
|
|
|
connect: {
|
|
|
|
id: user.id,
|
2022-07-19 18:36:39 +00:00
|
|
|
},
|
2022-11-10 23:40:01 +00:00
|
|
|
},
|
2023-01-04 13:30:46 +00:00
|
|
|
fields,
|
2022-11-10 23:40:01 +00:00
|
|
|
name: name,
|
|
|
|
description,
|
|
|
|
// Prisma doesn't allow setting null value directly for JSON. It recommends using JsonNull for that case.
|
|
|
|
routes: routes === null ? Prisma.JsonNull : routes,
|
|
|
|
id: id,
|
2022-07-19 18:36:39 +00:00
|
|
|
},
|
2022-11-10 23:40:01 +00:00
|
|
|
update: {
|
|
|
|
disabled: disabled,
|
2023-01-04 13:30:46 +00:00
|
|
|
fields,
|
2022-11-10 23:40:01 +00:00
|
|
|
name: name,
|
|
|
|
description,
|
|
|
|
settings: settings === null ? Prisma.JsonNull : settings,
|
|
|
|
routes: routes === null ? Prisma.JsonNull : routes,
|
|
|
|
},
|
|
|
|
});
|
2023-01-04 13:30:46 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* If Form has Router Linked fields, enrich them with the latest info from the Router
|
|
|
|
* If Form doesn't have Router fields but there is a Router used in routes, add all the fields from the Router
|
|
|
|
*/
|
|
|
|
async function getUpdatedRouterLinkedFields(fields: InputFields, routes: InputRoutes) {
|
|
|
|
const routerLinkedFields: Record<string, boolean> = {};
|
|
|
|
for (const [, field] of Object.entries(fields)) {
|
|
|
|
if (!isRouterLinkedField(field)) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
routerLinkedFields[field.routerId] = true;
|
|
|
|
|
|
|
|
if (!routes.some((route) => route.id === field.routerId)) {
|
|
|
|
// If the field is from a router that is not available anymore, mark it as deleted
|
|
|
|
field.deleted = true;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
// Get back deleted field as now the Router is there for it.
|
|
|
|
if (field.deleted) field.deleted = false;
|
|
|
|
const router = await prisma.app_RoutingForms_Form.findFirst({
|
|
|
|
where: {
|
|
|
|
id: field.routerId,
|
|
|
|
userId: user.id,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
if (router) {
|
|
|
|
assertIfInvalidRouter(router);
|
|
|
|
const parsedRouterFields = zodFields.parse(router.fields);
|
|
|
|
|
|
|
|
// There is a field from some router available, make sure that the field has up-to-date info from the router
|
|
|
|
const routerField = parsedRouterFields?.find((f) => f.id === field.id);
|
|
|
|
// Update local field(cache) with router field on every mutation
|
|
|
|
Object.assign(field, routerField);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (const [, route] of Object.entries(routes)) {
|
|
|
|
if (!isRouter(route)) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If there is a field that belongs to router, then all fields must be there already. So, need to add Router fields
|
|
|
|
if (routerLinkedFields[route.id]) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
const router = await prisma.app_RoutingForms_Form.findFirst({
|
|
|
|
where: {
|
|
|
|
id: route.id,
|
|
|
|
userId: user.id,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
if (router) {
|
|
|
|
assertIfInvalidRouter(router);
|
|
|
|
const parsedRouterFields = zodFields.parse(router.fields);
|
|
|
|
const fieldsFromRouter = parsedRouterFields
|
|
|
|
?.filter((f) => !f.deleted)
|
|
|
|
.map((f) => {
|
|
|
|
return {
|
|
|
|
...f,
|
|
|
|
routerId: route.id,
|
|
|
|
};
|
|
|
|
});
|
|
|
|
|
|
|
|
if (fieldsFromRouter) {
|
|
|
|
fields = fields.concat(fieldsFromRouter);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return fields;
|
|
|
|
}
|
|
|
|
|
|
|
|
function findFieldWithId(id: string, fields: InputFields) {
|
|
|
|
return fields.find((field) => field.id === id);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Update fields in connected forms as per the inputFields
|
|
|
|
*/
|
|
|
|
async function updateFieldsInConnectedForms(
|
|
|
|
serializedForm: SerializableForm<App_RoutingForms_Form>,
|
|
|
|
inputFields: InputFields
|
|
|
|
) {
|
|
|
|
for (const [, connectedForm] of Object.entries(serializedForm.connectedForms)) {
|
|
|
|
const connectedFormDb = await prisma.app_RoutingForms_Form.findFirst({
|
|
|
|
where: {
|
|
|
|
id: connectedForm.id,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
if (!connectedFormDb) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
const connectedFormFields = zodFields.parse(connectedFormDb.fields);
|
|
|
|
|
|
|
|
const fieldsThatAreNotInConnectedForm = (
|
|
|
|
inputFields?.filter((f) => !findFieldWithId(f.id, connectedFormFields || [])) || []
|
|
|
|
).map((f) => ({
|
|
|
|
...f,
|
|
|
|
routerId: serializedForm.id,
|
|
|
|
}));
|
|
|
|
|
|
|
|
const updatedConnectedFormFields = connectedFormFields
|
|
|
|
// Update fields that are already in connected form
|
|
|
|
?.map((field) => {
|
|
|
|
if (isRouterLinkedField(field) && field.routerId === serializedForm.id) {
|
|
|
|
return {
|
|
|
|
...field,
|
|
|
|
...findFieldWithId(field.id, inputFields || []),
|
|
|
|
};
|
|
|
|
}
|
|
|
|
return field;
|
|
|
|
})
|
|
|
|
// Add fields that are not there
|
|
|
|
.concat(fieldsThatAreNotInConnectedForm);
|
|
|
|
|
|
|
|
await prisma.app_RoutingForms_Form.update({
|
|
|
|
where: {
|
|
|
|
id: connectedForm.id,
|
|
|
|
},
|
|
|
|
data: {
|
|
|
|
fields: updatedConnectedFormFields,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
async function getRoutesAndFieldsForDuplication(duplicateFrom: DuplicateFrom) {
|
|
|
|
const sourceForm = await prisma.app_RoutingForms_Form.findFirst({
|
|
|
|
where: {
|
|
|
|
userId: user.id,
|
|
|
|
id: duplicateFrom,
|
|
|
|
},
|
|
|
|
select: {
|
|
|
|
id: true,
|
|
|
|
fields: true,
|
|
|
|
routes: true,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
if (!sourceForm) {
|
|
|
|
throw new TRPCError({
|
|
|
|
code: "BAD_REQUEST",
|
|
|
|
message: `Form to duplicate: ${duplicateFrom} not found`,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
//TODO: Instead of parsing separately, use getSerializableForm. That would automatically remove deleted fields as well.
|
|
|
|
const fieldsParsed = zodFields.safeParse(sourceForm.fields);
|
|
|
|
const routesParsed = zodRoutes.safeParse(sourceForm.routes);
|
|
|
|
if (!fieldsParsed.success || !routesParsed.success) {
|
|
|
|
throw new TRPCError({
|
|
|
|
code: "INTERNAL_SERVER_ERROR",
|
|
|
|
message: "Could not parse source form's fields or routes",
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
let fields, routes;
|
|
|
|
if (shouldConnect) {
|
|
|
|
routes = [
|
|
|
|
// This connected route would automatically link the fields
|
|
|
|
zodRouterRoute.parse({
|
|
|
|
id: sourceForm.id,
|
|
|
|
isRouter: true,
|
|
|
|
}),
|
|
|
|
];
|
|
|
|
fields =
|
|
|
|
fieldsParsed.data
|
|
|
|
// Deleted fields in the form shouldn't be added to the new form
|
|
|
|
?.filter((f) => !f.deleted)
|
|
|
|
.map((f) => {
|
|
|
|
return {
|
|
|
|
id: f.id,
|
|
|
|
routerId: sourceForm.id,
|
|
|
|
label: "",
|
|
|
|
type: "",
|
|
|
|
};
|
|
|
|
}) || [];
|
|
|
|
} else {
|
|
|
|
// Duplicate just routes and fields
|
|
|
|
// We don't want name, description and responses to be copied
|
|
|
|
routes = routesParsed.data || [];
|
|
|
|
// FIXME: Deleted fields shouldn't come in duplicate
|
|
|
|
fields = fieldsParsed.data || [];
|
|
|
|
}
|
|
|
|
return { routes, fields };
|
|
|
|
}
|
|
|
|
|
|
|
|
function markMissingFieldsDeleted(
|
|
|
|
serializedForm: SerializableForm<App_RoutingForms_Form>,
|
|
|
|
fields: InputFields
|
|
|
|
) {
|
|
|
|
// Find all fields that are in DB(including deleted) but not in the mutation
|
|
|
|
// e.g. inputFields is [A,B,C]. DB is [A,B,C,D,E,F]. It means D,E,F got deleted
|
|
|
|
const deletedFields =
|
|
|
|
serializedForm.fields?.filter((f) => !fields.find((field) => field.id === f.id)) || [];
|
|
|
|
|
|
|
|
// Add back deleted fields in the end and mark them deleted.
|
|
|
|
// Fields mustn't be deleted, to make sure columns never decrease which hugely simplifies CSV generation
|
|
|
|
fields = fields.concat(
|
|
|
|
deletedFields.map((f) => {
|
|
|
|
f.deleted = true;
|
|
|
|
return f;
|
|
|
|
})
|
|
|
|
);
|
|
|
|
return fields;
|
|
|
|
}
|
|
|
|
function assertIfInvalidRouter(router: App_RoutingForms_Form) {
|
|
|
|
const routesOfRouter = zodRoutes.parse(router.routes);
|
|
|
|
if (routesOfRouter) {
|
|
|
|
if (routesOfRouter.find(isRouter)) {
|
|
|
|
throw new TRPCError({
|
|
|
|
code: "BAD_REQUEST",
|
|
|
|
message:
|
|
|
|
"A form being used as a Router must be a Origin form. It must not be using any other Router.",
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2022-11-10 23:40:01 +00:00
|
|
|
}),
|
|
|
|
deleteForm: authedProcedure
|
|
|
|
.input(
|
|
|
|
z.object({
|
|
|
|
id: z.string(),
|
2022-07-19 18:36:39 +00:00
|
|
|
})
|
2022-11-10 23:40:01 +00:00
|
|
|
)
|
|
|
|
.mutation(async ({ ctx, input }) => {
|
|
|
|
const { user, prisma } = ctx;
|
2023-01-04 13:30:46 +00:00
|
|
|
if (!(await isFormEditAllowed({ userId: user.id, formId: input.id }))) {
|
2022-11-10 23:40:01 +00:00
|
|
|
throw new TRPCError({
|
|
|
|
code: "FORBIDDEN",
|
|
|
|
});
|
|
|
|
}
|
2023-01-04 13:30:46 +00:00
|
|
|
|
|
|
|
const areFormsUsingIt = (
|
|
|
|
await getConnectedForms(prisma, {
|
|
|
|
id: input.id,
|
|
|
|
userId: user.id,
|
|
|
|
})
|
|
|
|
).length;
|
|
|
|
if (areFormsUsingIt) {
|
|
|
|
throw new TRPCError({
|
|
|
|
code: "BAD_REQUEST",
|
|
|
|
message: "This form is being used by other forms. Please remove it's usage from there first.",
|
|
|
|
});
|
|
|
|
}
|
2022-11-10 23:40:01 +00:00
|
|
|
return await prisma.app_RoutingForms_Form.deleteMany({
|
|
|
|
where: {
|
|
|
|
id: input.id,
|
|
|
|
userId: user.id,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
}),
|
2023-01-04 13:30:46 +00:00
|
|
|
|
2022-11-11 09:57:44 +00:00
|
|
|
report: authedProcedure
|
|
|
|
.input(
|
|
|
|
z.object({
|
|
|
|
formId: z.string(),
|
|
|
|
jsonLogicQuery: z.object({
|
|
|
|
logic: z.union([z.record(z.any()), z.null()]),
|
|
|
|
}),
|
|
|
|
cursor: z.number().nullish(), // <-- "cursor" needs to exist when using useInfiniteQuery, but can be any type
|
|
|
|
})
|
|
|
|
)
|
|
|
|
.query(async ({ ctx: { prisma }, input }) => {
|
|
|
|
// Can be any prisma `where` clause
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
|
|
const prismaWhere: Record<string, any> = input.jsonLogicQuery
|
|
|
|
? jsonLogicToPrisma(input.jsonLogicQuery)
|
|
|
|
: {};
|
|
|
|
const skip = input.cursor ?? 0;
|
|
|
|
const take = 50;
|
|
|
|
logger.debug(
|
|
|
|
`Built Prisma where ${JSON.stringify(prismaWhere)} from jsonLogicQuery ${JSON.stringify(
|
|
|
|
input.jsonLogicQuery
|
|
|
|
)}`
|
|
|
|
);
|
|
|
|
const form = await prisma.app_RoutingForms_Form.findUnique({
|
|
|
|
where: {
|
|
|
|
id: input.formId,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
|
|
|
if (!form) {
|
|
|
|
throw new TRPCError({
|
|
|
|
code: "NOT_FOUND",
|
|
|
|
message: "Form not found",
|
|
|
|
});
|
|
|
|
}
|
|
|
|
// TODO: Second argument is required to return deleted operators.
|
2023-01-04 13:30:46 +00:00
|
|
|
const serializedForm = await getSerializableForm(form, true);
|
2022-11-11 09:57:44 +00:00
|
|
|
|
|
|
|
const rows = await prisma.app_RoutingForms_FormResponse.findMany({
|
|
|
|
where: {
|
|
|
|
formId: input.formId,
|
|
|
|
...prismaWhere,
|
|
|
|
},
|
|
|
|
take,
|
|
|
|
skip,
|
|
|
|
});
|
|
|
|
const fields = serializedForm?.fields || [];
|
|
|
|
const headers = fields.map((f) => f.label + (f.deleted ? "(Deleted)" : ""));
|
|
|
|
const responses: string[][] = [];
|
|
|
|
rows.forEach((r) => {
|
|
|
|
const rowResponses: string[] = [];
|
|
|
|
responses.push(rowResponses);
|
|
|
|
fields.forEach((field) => {
|
|
|
|
if (!r.response) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const response = r.response as Response;
|
|
|
|
const value = response[field.id]?.value || "";
|
|
|
|
let stringValue = "";
|
|
|
|
if (value instanceof Array) {
|
|
|
|
stringValue = value.join(", ");
|
|
|
|
} else {
|
|
|
|
stringValue = value;
|
|
|
|
}
|
|
|
|
rowResponses.push(stringValue);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
const areThereNoResultsOrLessThanAskedFor = !rows.length || rows.length < take;
|
|
|
|
return {
|
|
|
|
headers,
|
|
|
|
responses,
|
|
|
|
nextCursor: areThereNoResultsOrLessThanAskedFor ? null : skip + rows.length,
|
|
|
|
};
|
|
|
|
}),
|
2022-11-10 23:40:01 +00:00
|
|
|
});
|
2022-07-14 12:40:53 +00:00
|
|
|
|
2022-11-10 23:40:01 +00:00
|
|
|
export default appRoutingForms;
|