mirror of
https://github.com/formbricks/formbricks.git
synced 2026-02-05 14:24:33 -06:00
240 lines
7.0 KiB
TypeScript
240 lines
7.0 KiB
TypeScript
"use server";
|
|
|
|
import "server-only";
|
|
import { ActionClass, Prisma } from "@prisma/client";
|
|
import { cache as reactCache } from "react";
|
|
import { prisma } from "@formbricks/database";
|
|
import { PrismaErrorType } from "@formbricks/database/types/error";
|
|
import { TActionClass, TActionClassInput, ZActionClassInput } from "@formbricks/types/action-classes";
|
|
import { ZId, ZOptionalNumber, ZString } from "@formbricks/types/common";
|
|
import { DatabaseError, ResourceNotFoundError } from "@formbricks/types/errors";
|
|
import { cache } from "../cache";
|
|
import { ITEMS_PER_PAGE } from "../constants";
|
|
import { surveyCache } from "../survey/cache";
|
|
import { validateInputs } from "../utils/validate";
|
|
import { actionClassCache } from "./cache";
|
|
|
|
const selectActionClass = {
|
|
id: true,
|
|
createdAt: true,
|
|
updatedAt: true,
|
|
name: true,
|
|
description: true,
|
|
type: true,
|
|
key: true,
|
|
noCodeConfig: true,
|
|
environmentId: true,
|
|
} satisfies Prisma.ActionClassSelect;
|
|
|
|
export const getActionClasses = reactCache(
|
|
async (environmentId: string, page?: number): Promise<TActionClass[]> =>
|
|
cache(
|
|
async () => {
|
|
validateInputs([environmentId, ZId], [page, ZOptionalNumber]);
|
|
|
|
try {
|
|
return await prisma.actionClass.findMany({
|
|
where: {
|
|
environmentId: environmentId,
|
|
},
|
|
select: selectActionClass,
|
|
take: page ? ITEMS_PER_PAGE : undefined,
|
|
skip: page ? ITEMS_PER_PAGE * (page - 1) : undefined,
|
|
orderBy: {
|
|
createdAt: "asc",
|
|
},
|
|
});
|
|
} catch (error) {
|
|
throw new DatabaseError(`Database error when fetching actions for environment ${environmentId}`);
|
|
}
|
|
},
|
|
[`getActionClasses-${environmentId}-${page}`],
|
|
{
|
|
tags: [actionClassCache.tag.byEnvironmentId(environmentId)],
|
|
}
|
|
)()
|
|
);
|
|
|
|
// This function is used to get an action by its name and environmentId(it can return private actions as well)
|
|
export const getActionClassByEnvironmentIdAndName = reactCache(
|
|
async (environmentId: string, name: string): Promise<TActionClass | null> =>
|
|
cache(
|
|
async () => {
|
|
validateInputs([environmentId, ZId], [name, ZString]);
|
|
|
|
try {
|
|
const actionClass = await prisma.actionClass.findFirst({
|
|
where: {
|
|
name,
|
|
environmentId,
|
|
},
|
|
select: selectActionClass,
|
|
});
|
|
|
|
return actionClass;
|
|
} catch (error) {
|
|
throw new DatabaseError(`Database error when fetching action`);
|
|
}
|
|
},
|
|
[`getActionClassByEnvironmentIdAndName-${environmentId}-${name}`],
|
|
{
|
|
tags: [actionClassCache.tag.byNameAndEnvironmentId(name, environmentId)],
|
|
}
|
|
)()
|
|
);
|
|
|
|
export const getActionClass = reactCache(
|
|
async (actionClassId: string): Promise<TActionClass | null> =>
|
|
cache(
|
|
async () => {
|
|
validateInputs([actionClassId, ZId]);
|
|
|
|
try {
|
|
const actionClass = await prisma.actionClass.findUnique({
|
|
where: {
|
|
id: actionClassId,
|
|
},
|
|
select: selectActionClass,
|
|
});
|
|
|
|
return actionClass;
|
|
} catch (error) {
|
|
throw new DatabaseError(`Database error when fetching action`);
|
|
}
|
|
},
|
|
[`getActionClass-${actionClassId}`],
|
|
{
|
|
tags: [actionClassCache.tag.byId(actionClassId)],
|
|
}
|
|
)()
|
|
);
|
|
|
|
export const deleteActionClass = async (actionClassId: string): Promise<TActionClass> => {
|
|
validateInputs([actionClassId, ZId]);
|
|
|
|
try {
|
|
const actionClass = await prisma.actionClass.delete({
|
|
where: {
|
|
id: actionClassId,
|
|
},
|
|
select: selectActionClass,
|
|
});
|
|
if (actionClass === null) throw new ResourceNotFoundError("Action", actionClassId);
|
|
|
|
actionClassCache.revalidate({
|
|
environmentId: actionClass.environmentId,
|
|
id: actionClassId,
|
|
name: actionClass.name,
|
|
});
|
|
|
|
return actionClass;
|
|
} catch (error) {
|
|
if (error instanceof Prisma.PrismaClientKnownRequestError) {
|
|
throw new DatabaseError(error.message);
|
|
}
|
|
throw error;
|
|
}
|
|
};
|
|
|
|
export const createActionClass = async (
|
|
environmentId: string,
|
|
actionClass: TActionClassInput
|
|
): Promise<ActionClass> => {
|
|
validateInputs([environmentId, ZId], [actionClass, ZActionClassInput]);
|
|
|
|
const { environmentId: _, ...actionClassInput } = actionClass;
|
|
|
|
try {
|
|
const actionClassPrisma = await prisma.actionClass.create({
|
|
data: {
|
|
...actionClassInput,
|
|
environment: { connect: { id: environmentId } },
|
|
key: actionClassInput.type === "code" ? actionClassInput.key : undefined,
|
|
noCodeConfig: actionClassInput.type === "noCode" ? actionClassInput.noCodeConfig || {} : undefined,
|
|
},
|
|
select: selectActionClass,
|
|
});
|
|
|
|
actionClassCache.revalidate({
|
|
name: actionClassPrisma.name,
|
|
environmentId: actionClassPrisma.environmentId,
|
|
id: actionClassPrisma.id,
|
|
});
|
|
|
|
return actionClassPrisma;
|
|
} catch (error) {
|
|
if (
|
|
error instanceof Prisma.PrismaClientKnownRequestError &&
|
|
error.code === PrismaErrorType.UniqueConstraintViolation
|
|
) {
|
|
throw new DatabaseError(
|
|
`Action with ${error.meta?.target?.[0]} ${actionClass[error.meta?.target?.[0]]} already exists`
|
|
);
|
|
}
|
|
|
|
throw new DatabaseError(`Database error when creating an action for environment ${environmentId}`);
|
|
}
|
|
};
|
|
|
|
export const updateActionClass = async (
|
|
environmentId: string,
|
|
actionClassId: string,
|
|
inputActionClass: Partial<TActionClassInput>
|
|
): Promise<TActionClass> => {
|
|
validateInputs([environmentId, ZId], [actionClassId, ZId], [inputActionClass, ZActionClassInput]);
|
|
|
|
const { environmentId: _, ...actionClassInput } = inputActionClass;
|
|
try {
|
|
const result = await prisma.actionClass.update({
|
|
where: {
|
|
id: actionClassId,
|
|
},
|
|
data: {
|
|
...actionClassInput,
|
|
environment: { connect: { id: environmentId } },
|
|
key: actionClassInput.type === "code" ? actionClassInput.key : undefined,
|
|
noCodeConfig: actionClassInput.type === "noCode" ? actionClassInput.noCodeConfig || {} : undefined,
|
|
},
|
|
select: {
|
|
...selectActionClass,
|
|
surveyTriggers: {
|
|
select: {
|
|
surveyId: true,
|
|
},
|
|
},
|
|
},
|
|
});
|
|
|
|
// revalidate cache
|
|
actionClassCache.revalidate({
|
|
environmentId: result.environmentId,
|
|
name: result.name,
|
|
id: result.id,
|
|
});
|
|
|
|
// @ts-expect-error
|
|
const surveyIds = result.surveyTriggers.map((survey) => survey.surveyId);
|
|
for (const surveyId of surveyIds) {
|
|
surveyCache.revalidate({
|
|
id: surveyId,
|
|
});
|
|
}
|
|
|
|
return result;
|
|
} catch (error) {
|
|
if (
|
|
error instanceof Prisma.PrismaClientKnownRequestError &&
|
|
error.code === PrismaErrorType.UniqueConstraintViolation
|
|
) {
|
|
throw new DatabaseError(
|
|
`Action with ${error.meta?.target?.[0]} ${inputActionClass[error.meta?.target?.[0]]} already exists`
|
|
);
|
|
}
|
|
|
|
if (error instanceof Prisma.PrismaClientKnownRequestError) {
|
|
throw new DatabaseError(error.message);
|
|
}
|
|
throw error;
|
|
}
|
|
};
|