mirror of
https://github.com/formbricks/formbricks.git
synced 2026-01-06 05:40:02 -06:00
268 lines
6.4 KiB
TypeScript
268 lines
6.4 KiB
TypeScript
import "server-only";
|
|
|
|
import { Prisma } from "@prisma/client";
|
|
import { unstable_cache } from "next/cache";
|
|
import { z } from "zod";
|
|
|
|
import { prisma } from "@formbricks/database";
|
|
import { ZId } from "@formbricks/types/environment";
|
|
import { DatabaseError, ResourceNotFoundError } from "@formbricks/types/errors";
|
|
import { TMembership } from "@formbricks/types/memberships";
|
|
import { TUser, TUserCreateInput, TUserUpdateInput, ZUser, ZUserUpdateInput } from "@formbricks/types/user";
|
|
|
|
import { SERVICES_REVALIDATION_INTERVAL } from "../constants";
|
|
import { createCustomerIoCustomer } from "../customerio";
|
|
import { updateMembership } from "../membership/service";
|
|
import { deleteTeam } from "../team/service";
|
|
import { formatDateFields } from "../utils/datetime";
|
|
import { validateInputs } from "../utils/validate";
|
|
import { userCache } from "./cache";
|
|
|
|
const responseSelection = {
|
|
id: true,
|
|
name: true,
|
|
email: true,
|
|
emailVerified: true,
|
|
imageUrl: true,
|
|
createdAt: true,
|
|
updatedAt: true,
|
|
onboardingCompleted: true,
|
|
twoFactorEnabled: true,
|
|
identityProvider: true,
|
|
objective: true,
|
|
notificationSettings: true,
|
|
};
|
|
|
|
// function to retrive basic information about a user's user
|
|
export const getUser = async (id: string): Promise<TUser | null> => {
|
|
const user = await unstable_cache(
|
|
async () => {
|
|
validateInputs([id, ZId]);
|
|
|
|
try {
|
|
const user = await prisma.user.findUnique({
|
|
where: {
|
|
id,
|
|
},
|
|
select: responseSelection,
|
|
});
|
|
|
|
if (!user) {
|
|
return null;
|
|
}
|
|
return user;
|
|
} catch (error) {
|
|
if (error instanceof Prisma.PrismaClientKnownRequestError) {
|
|
throw new DatabaseError(error.message);
|
|
}
|
|
|
|
throw error;
|
|
}
|
|
},
|
|
[`getUser-${id}`],
|
|
{
|
|
tags: [userCache.tag.byId(id)],
|
|
revalidate: SERVICES_REVALIDATION_INTERVAL,
|
|
}
|
|
)();
|
|
|
|
return user
|
|
? {
|
|
...user,
|
|
...formatDateFields(user, ZUser),
|
|
}
|
|
: null;
|
|
};
|
|
|
|
export const getUserByEmail = async (email: string): Promise<TUser | null> => {
|
|
const user = await unstable_cache(
|
|
async () => {
|
|
validateInputs([email, z.string().email()]);
|
|
|
|
try {
|
|
const user = await prisma.user.findFirst({
|
|
where: {
|
|
email,
|
|
},
|
|
select: responseSelection,
|
|
});
|
|
|
|
return user;
|
|
} catch (error) {
|
|
if (error instanceof Prisma.PrismaClientKnownRequestError) {
|
|
throw new DatabaseError(error.message);
|
|
}
|
|
|
|
throw error;
|
|
}
|
|
},
|
|
[`getUserByEmail-${email}`],
|
|
{
|
|
tags: [userCache.tag.byEmail(email)],
|
|
revalidate: SERVICES_REVALIDATION_INTERVAL,
|
|
}
|
|
)();
|
|
|
|
return user
|
|
? {
|
|
...user,
|
|
...formatDateFields(user, ZUser),
|
|
}
|
|
: null;
|
|
};
|
|
|
|
const getAdminMemberships = (memberships: TMembership[]): TMembership[] =>
|
|
memberships.filter((membership) => membership.role === "admin");
|
|
|
|
// function to update a user's user
|
|
export const updateUser = async (personId: string, data: TUserUpdateInput): Promise<TUser> => {
|
|
validateInputs([personId, ZId], [data, ZUserUpdateInput.partial()]);
|
|
|
|
try {
|
|
const updatedUser = await prisma.user.update({
|
|
where: {
|
|
id: personId,
|
|
},
|
|
data: data,
|
|
select: responseSelection,
|
|
});
|
|
|
|
userCache.revalidate({
|
|
email: updatedUser.email,
|
|
id: updatedUser.id,
|
|
});
|
|
|
|
return updatedUser;
|
|
} catch (error) {
|
|
if (error instanceof Prisma.PrismaClientKnownRequestError && error.code === "P2016") {
|
|
throw new ResourceNotFoundError("User", personId);
|
|
} else {
|
|
throw error; // Re-throw any other errors
|
|
}
|
|
}
|
|
};
|
|
|
|
const deleteUserById = async (id: string): Promise<TUser> => {
|
|
validateInputs([id, ZId]);
|
|
|
|
const user = await prisma.user.delete({
|
|
where: {
|
|
id,
|
|
},
|
|
select: responseSelection,
|
|
});
|
|
|
|
userCache.revalidate({
|
|
email: user.email,
|
|
id,
|
|
});
|
|
|
|
return user;
|
|
};
|
|
|
|
export const createUser = async (data: TUserCreateInput): Promise<TUser> => {
|
|
validateInputs([data, ZUserUpdateInput]);
|
|
|
|
const user = await prisma.user.create({
|
|
data: data,
|
|
select: responseSelection,
|
|
});
|
|
|
|
userCache.revalidate({
|
|
email: user.email,
|
|
id: user.id,
|
|
});
|
|
|
|
// send new user customer.io to customer.io
|
|
createCustomerIoCustomer(user);
|
|
|
|
return user;
|
|
};
|
|
|
|
// function to delete a user's user including teams
|
|
export const deleteUser = async (id: string): Promise<TUser> => {
|
|
validateInputs([id, ZId]);
|
|
|
|
try {
|
|
const currentUserMemberships = await prisma.membership.findMany({
|
|
where: {
|
|
userId: id,
|
|
},
|
|
include: {
|
|
team: {
|
|
select: {
|
|
id: true,
|
|
name: true,
|
|
memberships: true,
|
|
},
|
|
},
|
|
},
|
|
});
|
|
|
|
for (const currentUserMembership of currentUserMemberships) {
|
|
const teamMemberships = currentUserMembership.team.memberships;
|
|
const role = currentUserMembership.role;
|
|
const teamId = currentUserMembership.teamId;
|
|
|
|
const teamAdminMemberships = getAdminMemberships(teamMemberships);
|
|
const teamHasAtLeastOneAdmin = teamAdminMemberships.length > 0;
|
|
const teamHasOnlyOneMember = teamMemberships.length === 1;
|
|
const currentUserIsTeamOwner = role === "owner";
|
|
|
|
if (teamHasOnlyOneMember) {
|
|
await deleteTeam(teamId);
|
|
} else if (currentUserIsTeamOwner && teamHasAtLeastOneAdmin) {
|
|
const firstAdmin = teamAdminMemberships[0];
|
|
await updateMembership(firstAdmin.userId, teamId, { role: "owner" });
|
|
} else if (currentUserIsTeamOwner) {
|
|
await deleteTeam(teamId);
|
|
}
|
|
}
|
|
|
|
const deletedUser = await deleteUserById(id);
|
|
|
|
return deletedUser;
|
|
} catch (error) {
|
|
if (error instanceof Prisma.PrismaClientKnownRequestError) {
|
|
throw new DatabaseError(error.message);
|
|
}
|
|
|
|
throw error;
|
|
}
|
|
};
|
|
|
|
export const getUsersWithTeam = async (teamId: string): Promise<TUser[]> => {
|
|
validateInputs([teamId, ZId]);
|
|
|
|
const users = await prisma.user.findMany({
|
|
where: {
|
|
memberships: {
|
|
some: {
|
|
teamId,
|
|
},
|
|
},
|
|
},
|
|
select: responseSelection,
|
|
});
|
|
|
|
return users;
|
|
};
|
|
|
|
export const userIdRelatedToApiKey = async (apiKey: string) => {
|
|
const userId = await prisma.apiKey.findUnique({
|
|
where: { id: apiKey },
|
|
select: {
|
|
environment: {
|
|
select: {
|
|
people: {
|
|
select: {
|
|
userId: true,
|
|
},
|
|
},
|
|
},
|
|
},
|
|
},
|
|
});
|
|
return userId;
|
|
};
|