mirror of
https://github.com/formbricks/formbricks.git
synced 2026-01-06 05:40:02 -06:00
Co-authored-by: Johannes <johannes@formbricks.com> Co-authored-by: Matthias Nannt <mail@matthiasnannt.com>
617 lines
16 KiB
TypeScript
617 lines
16 KiB
TypeScript
import "server-only";
|
|
|
|
import { prisma } from "@formbricks/database";
|
|
import { ZOptionalNumber, ZString } from "@formbricks/types/common";
|
|
import { ZId } from "@formbricks/types/environment";
|
|
import { DatabaseError, ResourceNotFoundError } from "@formbricks/types/errors";
|
|
import { TPerson } from "@formbricks/types/people";
|
|
import {
|
|
TResponse,
|
|
TResponseInput,
|
|
TResponseLegacyInput,
|
|
TResponseUpdateInput,
|
|
ZResponseInput,
|
|
ZResponseLegacyInput,
|
|
ZResponseUpdateInput,
|
|
} from "@formbricks/types/responses";
|
|
import { TTag } from "@formbricks/types/tags";
|
|
import { Prisma } from "@prisma/client";
|
|
import { unstable_cache } from "next/cache";
|
|
import { ITEMS_PER_PAGE, SERVICES_REVALIDATION_INTERVAL } from "../constants";
|
|
import { deleteDisplayByResponseId } from "../display/service";
|
|
import { createPerson, getPerson, getPersonByUserId, transformPrismaPerson } from "../person/service";
|
|
import { calculateTtcTotal, formatResponseDateFields } from "../response/util";
|
|
import { responseNoteCache } from "../responseNote/cache";
|
|
import { getResponseNotes } from "../responseNote/service";
|
|
import { captureTelemetry } from "../telemetry";
|
|
import { validateInputs } from "../utils/validate";
|
|
import { responseCache } from "./cache";
|
|
|
|
const responseSelection = {
|
|
id: true,
|
|
createdAt: true,
|
|
updatedAt: true,
|
|
surveyId: true,
|
|
finished: true,
|
|
data: true,
|
|
meta: true,
|
|
ttc: true,
|
|
personAttributes: true,
|
|
singleUseId: true,
|
|
person: {
|
|
select: {
|
|
id: true,
|
|
userId: true,
|
|
createdAt: true,
|
|
updatedAt: true,
|
|
environmentId: true,
|
|
attributes: {
|
|
select: {
|
|
value: true,
|
|
attributeClass: {
|
|
select: {
|
|
name: true,
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
tags: {
|
|
select: {
|
|
tag: {
|
|
select: {
|
|
id: true,
|
|
createdAt: true,
|
|
updatedAt: true,
|
|
name: true,
|
|
environmentId: true,
|
|
},
|
|
},
|
|
},
|
|
},
|
|
notes: {
|
|
select: {
|
|
id: true,
|
|
createdAt: true,
|
|
updatedAt: true,
|
|
text: true,
|
|
user: {
|
|
select: {
|
|
id: true,
|
|
name: true,
|
|
},
|
|
},
|
|
isResolved: true,
|
|
isEdited: true,
|
|
},
|
|
},
|
|
};
|
|
|
|
export const getResponsesByPersonId = async (
|
|
personId: string,
|
|
page?: number
|
|
): Promise<Array<TResponse> | null> => {
|
|
const responses = await unstable_cache(
|
|
async () => {
|
|
validateInputs([personId, ZId], [page, ZOptionalNumber]);
|
|
|
|
try {
|
|
const responsePrisma = await prisma.response.findMany({
|
|
where: {
|
|
personId,
|
|
},
|
|
select: responseSelection,
|
|
take: page ? ITEMS_PER_PAGE : undefined,
|
|
skip: page ? ITEMS_PER_PAGE * (page - 1) : undefined,
|
|
});
|
|
|
|
if (!responsePrisma) {
|
|
throw new ResourceNotFoundError("Response from PersonId", personId);
|
|
}
|
|
|
|
let responses: Array<TResponse> = [];
|
|
|
|
responsePrisma.forEach((response) => {
|
|
responses.push({
|
|
...response,
|
|
person: response.person ? transformPrismaPerson(response.person) : null,
|
|
tags: response.tags.map((tagPrisma: { tag: TTag }) => tagPrisma.tag),
|
|
});
|
|
});
|
|
|
|
return responses;
|
|
} catch (error) {
|
|
if (error instanceof Prisma.PrismaClientKnownRequestError) {
|
|
throw new DatabaseError(error.message);
|
|
}
|
|
|
|
throw error;
|
|
}
|
|
},
|
|
[`getResponsesByPersonId-${personId}-${page}`],
|
|
{
|
|
tags: [responseCache.tag.byPersonId(personId)],
|
|
revalidate: SERVICES_REVALIDATION_INTERVAL,
|
|
}
|
|
)();
|
|
|
|
return responses.map((response) => ({
|
|
...response,
|
|
...formatResponseDateFields(response),
|
|
}));
|
|
};
|
|
|
|
export const getResponseBySingleUseId = async (
|
|
surveyId: string,
|
|
singleUseId: string
|
|
): Promise<TResponse | null> => {
|
|
const response = await unstable_cache(
|
|
async () => {
|
|
validateInputs([surveyId, ZId], [singleUseId, ZString]);
|
|
|
|
try {
|
|
const responsePrisma = await prisma.response.findUnique({
|
|
where: {
|
|
surveyId_singleUseId: { surveyId, singleUseId },
|
|
},
|
|
select: responseSelection,
|
|
});
|
|
|
|
if (!responsePrisma) {
|
|
return null;
|
|
}
|
|
|
|
const response: TResponse = {
|
|
...responsePrisma,
|
|
person: responsePrisma.person ? transformPrismaPerson(responsePrisma.person) : null,
|
|
tags: responsePrisma.tags.map((tagPrisma: { tag: TTag }) => tagPrisma.tag),
|
|
};
|
|
|
|
return response;
|
|
} catch (error) {
|
|
if (error instanceof Prisma.PrismaClientKnownRequestError) {
|
|
throw new DatabaseError(error.message);
|
|
}
|
|
|
|
throw error;
|
|
}
|
|
},
|
|
[`getResponseBySingleUseId-${surveyId}-${singleUseId}`],
|
|
{
|
|
tags: [responseCache.tag.bySingleUseId(surveyId, singleUseId)],
|
|
revalidate: SERVICES_REVALIDATION_INTERVAL,
|
|
}
|
|
)();
|
|
|
|
if (!response) {
|
|
return null;
|
|
}
|
|
|
|
return {
|
|
...response,
|
|
...formatResponseDateFields(response),
|
|
};
|
|
};
|
|
|
|
export const createResponse = async (responseInput: TResponseInput): Promise<TResponse> => {
|
|
validateInputs([responseInput, ZResponseInput]);
|
|
captureTelemetry("response created");
|
|
|
|
const { environmentId, userId, surveyId, finished, data, meta, singleUseId } = responseInput;
|
|
|
|
try {
|
|
let person: TPerson | null = null;
|
|
|
|
if (userId) {
|
|
person = await getPersonByUserId(environmentId, userId);
|
|
if (!person) {
|
|
// create person if it does not exist
|
|
person = await createPerson(environmentId, userId);
|
|
}
|
|
}
|
|
|
|
const responsePrisma = await prisma.response.create({
|
|
data: {
|
|
survey: {
|
|
connect: {
|
|
id: surveyId,
|
|
},
|
|
},
|
|
finished: finished,
|
|
data: data,
|
|
...(person?.id && {
|
|
person: {
|
|
connect: {
|
|
id: person.id,
|
|
},
|
|
},
|
|
personAttributes: person?.attributes,
|
|
}),
|
|
...(meta && ({ meta } as Prisma.JsonObject)),
|
|
singleUseId,
|
|
},
|
|
select: responseSelection,
|
|
});
|
|
|
|
const response: TResponse = {
|
|
...responsePrisma,
|
|
person: responsePrisma.person ? transformPrismaPerson(responsePrisma.person) : null,
|
|
tags: responsePrisma.tags.map((tagPrisma: { tag: TTag }) => tagPrisma.tag),
|
|
};
|
|
|
|
responseCache.revalidate({
|
|
id: response.id,
|
|
personId: response.person?.id,
|
|
surveyId: response.surveyId,
|
|
});
|
|
|
|
responseNoteCache.revalidate({
|
|
responseId: response.id,
|
|
});
|
|
|
|
return response;
|
|
} catch (error) {
|
|
if (error instanceof Prisma.PrismaClientKnownRequestError) {
|
|
throw new DatabaseError(error.message);
|
|
}
|
|
|
|
throw error;
|
|
}
|
|
};
|
|
|
|
export const createResponseLegacy = async (responseInput: TResponseLegacyInput): Promise<TResponse> => {
|
|
validateInputs([responseInput, ZResponseLegacyInput]);
|
|
captureTelemetry("response created");
|
|
|
|
try {
|
|
let person: TPerson | null = null;
|
|
|
|
if (responseInput.personId) {
|
|
person = await getPerson(responseInput.personId);
|
|
}
|
|
const ttcTemp = responseInput.ttc;
|
|
const questionId = Object.keys(ttcTemp)[0];
|
|
const ttc = responseInput.finished
|
|
? {
|
|
...ttcTemp,
|
|
_total: ttcTemp[questionId], // Add _total property with the same value
|
|
}
|
|
: ttcTemp;
|
|
const responsePrisma = await prisma.response.create({
|
|
data: {
|
|
survey: {
|
|
connect: {
|
|
id: responseInput.surveyId,
|
|
},
|
|
},
|
|
finished: responseInput.finished,
|
|
data: responseInput.data,
|
|
ttc,
|
|
...(responseInput.personId && {
|
|
person: {
|
|
connect: {
|
|
id: responseInput.personId,
|
|
},
|
|
},
|
|
personAttributes: person?.attributes,
|
|
}),
|
|
|
|
...(responseInput.meta && ({ meta: responseInput?.meta } as Prisma.JsonObject)),
|
|
singleUseId: responseInput.singleUseId,
|
|
},
|
|
select: responseSelection,
|
|
});
|
|
|
|
const response: TResponse = {
|
|
...responsePrisma,
|
|
person: responsePrisma.person ? transformPrismaPerson(responsePrisma.person) : null,
|
|
tags: responsePrisma.tags.map((tagPrisma: { tag: TTag }) => tagPrisma.tag),
|
|
};
|
|
|
|
responseCache.revalidate({
|
|
id: response.id,
|
|
personId: response.person?.id,
|
|
surveyId: response.surveyId,
|
|
});
|
|
|
|
responseNoteCache.revalidate({
|
|
responseId: response.id,
|
|
});
|
|
|
|
return response;
|
|
} catch (error) {
|
|
if (error instanceof Prisma.PrismaClientKnownRequestError) {
|
|
throw new DatabaseError(error.message);
|
|
}
|
|
|
|
throw error;
|
|
}
|
|
};
|
|
|
|
export const getResponse = async (responseId: string): Promise<TResponse | null> => {
|
|
const response = await unstable_cache(
|
|
async () => {
|
|
validateInputs([responseId, ZId]);
|
|
|
|
try {
|
|
const responsePrisma = await prisma.response.findUnique({
|
|
where: {
|
|
id: responseId,
|
|
},
|
|
select: responseSelection,
|
|
});
|
|
|
|
if (!responsePrisma) {
|
|
throw new ResourceNotFoundError("Response", responseId);
|
|
}
|
|
|
|
const response: TResponse = {
|
|
...responsePrisma,
|
|
person: responsePrisma.person ? transformPrismaPerson(responsePrisma.person) : null,
|
|
tags: responsePrisma.tags.map((tagPrisma: { tag: TTag }) => tagPrisma.tag),
|
|
};
|
|
|
|
return response;
|
|
} catch (error) {
|
|
if (error instanceof Prisma.PrismaClientKnownRequestError) {
|
|
throw new DatabaseError(error.message);
|
|
}
|
|
|
|
throw error;
|
|
}
|
|
},
|
|
[`getResponse-${responseId}`],
|
|
{
|
|
tags: [responseCache.tag.byId(responseId), responseNoteCache.tag.byResponseId(responseId)],
|
|
revalidate: SERVICES_REVALIDATION_INTERVAL,
|
|
}
|
|
)();
|
|
|
|
if (!response) {
|
|
return null;
|
|
}
|
|
|
|
return {
|
|
...response,
|
|
...formatResponseDateFields(response),
|
|
} as TResponse;
|
|
};
|
|
|
|
export const getResponses = async (surveyId: string, page?: number): Promise<TResponse[]> => {
|
|
const responses = await unstable_cache(
|
|
async () => {
|
|
validateInputs([surveyId, ZId], [page, ZOptionalNumber]);
|
|
|
|
try {
|
|
const responses = await prisma.response.findMany({
|
|
where: {
|
|
surveyId,
|
|
},
|
|
select: responseSelection,
|
|
orderBy: [
|
|
{
|
|
createdAt: "desc",
|
|
},
|
|
],
|
|
take: page ? ITEMS_PER_PAGE : undefined,
|
|
skip: page ? ITEMS_PER_PAGE * (page - 1) : undefined,
|
|
});
|
|
|
|
const transformedResponses: TResponse[] = await Promise.all(
|
|
responses.map(async (responsePrisma) => {
|
|
return {
|
|
...responsePrisma,
|
|
person: responsePrisma.person ? transformPrismaPerson(responsePrisma.person) : null,
|
|
tags: responsePrisma.tags.map((tagPrisma: { tag: TTag }) => tagPrisma.tag),
|
|
};
|
|
})
|
|
);
|
|
|
|
return transformedResponses;
|
|
} catch (error) {
|
|
if (error instanceof Prisma.PrismaClientKnownRequestError) {
|
|
throw new DatabaseError(error.message);
|
|
}
|
|
|
|
throw error;
|
|
}
|
|
},
|
|
[`getResponses-${surveyId}`],
|
|
{
|
|
tags: [responseCache.tag.bySurveyId(surveyId)],
|
|
revalidate: SERVICES_REVALIDATION_INTERVAL,
|
|
}
|
|
)();
|
|
|
|
return responses.map((response) => ({
|
|
...response,
|
|
...formatResponseDateFields(response),
|
|
}));
|
|
};
|
|
|
|
export const getResponsesByEnvironmentId = async (
|
|
environmentId: string,
|
|
page?: number
|
|
): Promise<TResponse[]> => {
|
|
const responses = await unstable_cache(
|
|
async () => {
|
|
validateInputs([environmentId, ZId], [page, ZOptionalNumber]);
|
|
|
|
try {
|
|
const responses = await prisma.response.findMany({
|
|
where: {
|
|
survey: {
|
|
environmentId,
|
|
},
|
|
},
|
|
select: responseSelection,
|
|
orderBy: [
|
|
{
|
|
createdAt: "desc",
|
|
},
|
|
],
|
|
take: page ? ITEMS_PER_PAGE : undefined,
|
|
skip: page ? ITEMS_PER_PAGE * (page - 1) : undefined,
|
|
});
|
|
|
|
const transformedResponses: TResponse[] = await Promise.all(
|
|
responses.map(async (responsePrisma) => {
|
|
return {
|
|
...responsePrisma,
|
|
person: responsePrisma.person ? transformPrismaPerson(responsePrisma.person) : null,
|
|
tags: responsePrisma.tags.map((tagPrisma: { tag: TTag }) => tagPrisma.tag),
|
|
};
|
|
})
|
|
);
|
|
|
|
return transformedResponses;
|
|
} catch (error) {
|
|
if (error instanceof Prisma.PrismaClientKnownRequestError) {
|
|
throw new DatabaseError(error.message);
|
|
}
|
|
|
|
throw error;
|
|
}
|
|
},
|
|
[`getResponsesByEnvironmentId-${environmentId}`],
|
|
{
|
|
tags: [responseCache.tag.byEnvironmentId(environmentId)],
|
|
revalidate: SERVICES_REVALIDATION_INTERVAL,
|
|
}
|
|
)();
|
|
|
|
return responses.map((response) => ({
|
|
...response,
|
|
...formatResponseDateFields(response),
|
|
}));
|
|
};
|
|
|
|
export const updateResponse = async (
|
|
responseId: string,
|
|
responseInput: TResponseUpdateInput
|
|
): Promise<TResponse> => {
|
|
validateInputs([responseId, ZId], [responseInput, ZResponseUpdateInput]);
|
|
try {
|
|
// const currentResponse = await getResponse(responseId);
|
|
|
|
// use direct prisma call to avoid cache issues
|
|
const currentResponse = await prisma.response.findUnique({
|
|
where: {
|
|
id: responseId,
|
|
},
|
|
select: responseSelection,
|
|
});
|
|
|
|
if (!currentResponse) {
|
|
throw new ResourceNotFoundError("Response", responseId);
|
|
}
|
|
|
|
// merge data object
|
|
const data = {
|
|
...currentResponse.data,
|
|
...responseInput.data,
|
|
};
|
|
const ttc = responseInput.finished ? calculateTtcTotal(responseInput.ttc) : responseInput.ttc;
|
|
|
|
const responsePrisma = await prisma.response.update({
|
|
where: {
|
|
id: responseId,
|
|
},
|
|
data: {
|
|
finished: responseInput.finished,
|
|
data,
|
|
ttc,
|
|
},
|
|
select: responseSelection,
|
|
});
|
|
|
|
const response: TResponse = {
|
|
...responsePrisma,
|
|
person: responsePrisma.person ? transformPrismaPerson(responsePrisma.person) : null,
|
|
tags: responsePrisma.tags.map((tagPrisma: { tag: TTag }) => tagPrisma.tag),
|
|
};
|
|
|
|
responseCache.revalidate({
|
|
id: response.id,
|
|
personId: response.person?.id,
|
|
surveyId: response.surveyId,
|
|
});
|
|
|
|
responseNoteCache.revalidate({
|
|
responseId: response.id,
|
|
});
|
|
|
|
return response;
|
|
} catch (error) {
|
|
if (error instanceof Prisma.PrismaClientKnownRequestError) {
|
|
throw new DatabaseError(error.message);
|
|
}
|
|
|
|
throw error;
|
|
}
|
|
};
|
|
|
|
export const deleteResponse = async (responseId: string): Promise<TResponse> => {
|
|
validateInputs([responseId, ZId]);
|
|
try {
|
|
const responsePrisma = await prisma.response.delete({
|
|
where: {
|
|
id: responseId,
|
|
},
|
|
select: responseSelection,
|
|
});
|
|
|
|
const responseNotes = await getResponseNotes(responsePrisma.id);
|
|
const response: TResponse = {
|
|
...responsePrisma,
|
|
notes: responseNotes,
|
|
person: responsePrisma.person ? transformPrismaPerson(responsePrisma.person) : null,
|
|
tags: responsePrisma.tags.map((tagPrisma: { tag: TTag }) => tagPrisma.tag),
|
|
};
|
|
|
|
deleteDisplayByResponseId(responseId, response.surveyId);
|
|
|
|
responseCache.revalidate({
|
|
id: response.id,
|
|
personId: response.person?.id,
|
|
surveyId: response.surveyId,
|
|
});
|
|
|
|
responseNoteCache.revalidate({
|
|
responseId: response.id,
|
|
});
|
|
|
|
return response;
|
|
} catch (error) {
|
|
if (error instanceof Prisma.PrismaClientKnownRequestError) {
|
|
throw new DatabaseError(error.message);
|
|
}
|
|
|
|
throw error;
|
|
}
|
|
};
|
|
|
|
export const getResponseCountBySurveyId = async (surveyId: string): Promise<number> =>
|
|
unstable_cache(
|
|
async () => {
|
|
validateInputs([surveyId, ZId]);
|
|
|
|
try {
|
|
const responseCount = await prisma.response.count({
|
|
where: {
|
|
surveyId: surveyId,
|
|
},
|
|
});
|
|
return responseCount;
|
|
} catch (error) {
|
|
throw error;
|
|
}
|
|
},
|
|
[`getResponseCountBySurveyId-${surveyId}`],
|
|
{
|
|
tags: [responseCache.tag.bySurveyId(surveyId)],
|
|
revalidate: SERVICES_REVALIDATION_INTERVAL,
|
|
}
|
|
)();
|