mirror of
https://github.com/formbricks/formbricks.git
synced 2025-12-29 09:50:10 -06:00
197 lines
5.8 KiB
TypeScript
197 lines
5.8 KiB
TypeScript
"use server";
|
|
|
|
import "server-only";
|
|
import { cache as reactCache } from "react";
|
|
import { prisma } from "@formbricks/database";
|
|
import { ActionClass, Prisma } from "@formbricks/database/generated/client";
|
|
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 { ITEMS_PER_PAGE } from "../constants";
|
|
import { validateInputs } from "../utils/validate";
|
|
|
|
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[]> => {
|
|
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}`);
|
|
}
|
|
}
|
|
);
|
|
|
|
// 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> => {
|
|
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`);
|
|
}
|
|
}
|
|
);
|
|
|
|
export const getActionClass = reactCache(async (actionClassId: string): Promise<TActionClass | null> => {
|
|
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`);
|
|
}
|
|
});
|
|
|
|
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);
|
|
|
|
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 === null
|
|
? undefined
|
|
: actionClassInput.noCodeConfig
|
|
: undefined,
|
|
},
|
|
select: selectActionClass,
|
|
});
|
|
|
|
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 === null
|
|
? undefined
|
|
: actionClassInput.noCodeConfig
|
|
: undefined,
|
|
},
|
|
select: {
|
|
...selectActionClass,
|
|
surveyTriggers: {
|
|
select: {
|
|
surveyId: true,
|
|
},
|
|
},
|
|
},
|
|
});
|
|
|
|
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;
|
|
}
|
|
};
|