mirror of
https://github.com/formbricks/formbricks.git
synced 2025-12-30 18:30:32 -06:00
Co-authored-by: Dhruwang <dhruwangjariwala18@gmail.com> Co-authored-by: Matthias Nannt <mail@matthiasnannt.com> Co-authored-by: pandeymangg <anshuman.pandey9999@gmail.com>
101 lines
2.8 KiB
TypeScript
101 lines
2.8 KiB
TypeScript
import { getProjectPermissionByUserId, getTeamRoleByTeamIdUserId } from "@/modules/ee/teams/lib/roles";
|
|
import { TTeamPermission } from "@/modules/ee/teams/project-teams/types/teams";
|
|
import { TTeamRole } from "@/modules/ee/teams/team-list/types/teams";
|
|
import { returnValidationErrors } from "next-safe-action";
|
|
import { ZodIssue, z } from "zod";
|
|
import { getMembershipRole } from "@formbricks/lib/membership/hooks/actions";
|
|
import { AuthorizationError } from "@formbricks/types/errors";
|
|
import { TOrganizationRole } from "@formbricks/types/memberships";
|
|
|
|
const formatErrors = (issues: ZodIssue[]): Record<string, { _errors: string[] }> => {
|
|
return {
|
|
...issues.reduce((acc, issue) => {
|
|
acc[issue.path.join(".")] = {
|
|
_errors: [issue.message],
|
|
};
|
|
return acc;
|
|
}, {}),
|
|
};
|
|
};
|
|
|
|
export type TAccess<T extends z.ZodRawShape> =
|
|
| {
|
|
type: "organization";
|
|
schema?: z.ZodObject<T>;
|
|
data?: z.ZodObject<T>["_output"];
|
|
roles: TOrganizationRole[];
|
|
}
|
|
| {
|
|
type: "projectTeam";
|
|
minPermission?: TTeamPermission;
|
|
projectId: string;
|
|
}
|
|
| {
|
|
type: "team";
|
|
minPermission?: TTeamRole;
|
|
teamId: string;
|
|
};
|
|
|
|
const teamPermissionWeight = {
|
|
read: 1,
|
|
readWrite: 2,
|
|
manage: 3,
|
|
};
|
|
|
|
const teamRoleWeight = {
|
|
contributor: 1,
|
|
admin: 2,
|
|
};
|
|
|
|
export const checkAuthorizationUpdated = async <T extends z.ZodRawShape>({
|
|
userId,
|
|
organizationId,
|
|
access,
|
|
}: {
|
|
userId: string;
|
|
organizationId: string;
|
|
access: TAccess<T>[];
|
|
}) => {
|
|
const role = await getMembershipRole(userId, organizationId);
|
|
|
|
for (let accessItem of access) {
|
|
if (accessItem.type === "organization") {
|
|
if (accessItem.schema) {
|
|
const resultSchema = accessItem.schema.strict();
|
|
const parsedResult = resultSchema.safeParse(accessItem.data);
|
|
if (!parsedResult.success) {
|
|
// @ts-expect-error -- TODO: match dynamic next-safe-action types
|
|
return returnValidationErrors(resultSchema, formatErrors(parsedResult.error.issues));
|
|
}
|
|
}
|
|
|
|
if (accessItem.roles.includes(role)) {
|
|
return true;
|
|
}
|
|
} else {
|
|
if (accessItem.type === "projectTeam") {
|
|
const projectPermission = await getProjectPermissionByUserId(userId, accessItem.projectId);
|
|
if (
|
|
!projectPermission ||
|
|
(accessItem.minPermission !== undefined &&
|
|
teamPermissionWeight[projectPermission] < teamPermissionWeight[accessItem.minPermission])
|
|
) {
|
|
continue;
|
|
}
|
|
} else {
|
|
const teamRole = await getTeamRoleByTeamIdUserId(accessItem.teamId, userId);
|
|
if (
|
|
!teamRole ||
|
|
(accessItem.minPermission !== undefined &&
|
|
teamRoleWeight[teamRole] < teamRoleWeight[accessItem.minPermission])
|
|
) {
|
|
continue;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
}
|
|
|
|
throw new AuthorizationError("Not authorized");
|
|
};
|