mirror of
https://github.com/documenso/documenso.git
synced 2025-11-13 08:13:56 +10:00
feat: admin ui for disabling users (#1547)
This commit is contained in:
@ -30,8 +30,8 @@ export type DeleteUserDialogProps = {
|
|||||||
};
|
};
|
||||||
|
|
||||||
export const DeleteUserDialog = ({ className, user }: DeleteUserDialogProps) => {
|
export const DeleteUserDialog = ({ className, user }: DeleteUserDialogProps) => {
|
||||||
const { toast } = useToast();
|
|
||||||
const { _ } = useLingui();
|
const { _ } = useLingui();
|
||||||
|
const { toast } = useToast();
|
||||||
|
|
||||||
const router = useRouter();
|
const router = useRouter();
|
||||||
|
|
||||||
@ -44,7 +44,6 @@ export const DeleteUserDialog = ({ className, user }: DeleteUserDialogProps) =>
|
|||||||
try {
|
try {
|
||||||
await deleteUser({
|
await deleteUser({
|
||||||
id: user.id,
|
id: user.id,
|
||||||
email,
|
|
||||||
});
|
});
|
||||||
|
|
||||||
toast({
|
toast({
|
||||||
@ -78,7 +77,7 @@ export const DeleteUserDialog = ({ className, user }: DeleteUserDialogProps) =>
|
|||||||
return (
|
return (
|
||||||
<div className={className}>
|
<div className={className}>
|
||||||
<Alert
|
<Alert
|
||||||
className="flex flex-col items-center justify-between gap-4 p-6 md:flex-row "
|
className="flex flex-col items-center justify-between gap-4 p-6 md:flex-row"
|
||||||
variant="neutral"
|
variant="neutral"
|
||||||
>
|
>
|
||||||
<div>
|
<div>
|
||||||
|
|||||||
@ -0,0 +1,141 @@
|
|||||||
|
'use client';
|
||||||
|
|
||||||
|
import { useState } from 'react';
|
||||||
|
|
||||||
|
import { Trans, msg } from '@lingui/macro';
|
||||||
|
import { useLingui } from '@lingui/react';
|
||||||
|
import { match } from 'ts-pattern';
|
||||||
|
|
||||||
|
import { AppError, AppErrorCode } from '@documenso/lib/errors/app-error';
|
||||||
|
import type { User } from '@documenso/prisma/client';
|
||||||
|
import { trpc } from '@documenso/trpc/react';
|
||||||
|
import { Alert, AlertDescription, AlertTitle } from '@documenso/ui/primitives/alert';
|
||||||
|
import { Button } from '@documenso/ui/primitives/button';
|
||||||
|
import {
|
||||||
|
Dialog,
|
||||||
|
DialogContent,
|
||||||
|
DialogDescription,
|
||||||
|
DialogFooter,
|
||||||
|
DialogHeader,
|
||||||
|
DialogTitle,
|
||||||
|
DialogTrigger,
|
||||||
|
} from '@documenso/ui/primitives/dialog';
|
||||||
|
import { Input } from '@documenso/ui/primitives/input';
|
||||||
|
import { useToast } from '@documenso/ui/primitives/use-toast';
|
||||||
|
|
||||||
|
export type DisableUserDialogProps = {
|
||||||
|
className?: string;
|
||||||
|
userToDisable: User;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const DisableUserDialog = ({ className, userToDisable }: DisableUserDialogProps) => {
|
||||||
|
const { _ } = useLingui();
|
||||||
|
const { toast } = useToast();
|
||||||
|
|
||||||
|
const [email, setEmail] = useState('');
|
||||||
|
|
||||||
|
const { mutateAsync: disableUser, isLoading: isDisablingUser } =
|
||||||
|
trpc.admin.disableUser.useMutation();
|
||||||
|
|
||||||
|
const onDisableAccount = async () => {
|
||||||
|
try {
|
||||||
|
await disableUser({
|
||||||
|
id: userToDisable.id,
|
||||||
|
});
|
||||||
|
|
||||||
|
toast({
|
||||||
|
title: _(msg`Account disabled`),
|
||||||
|
description: _(msg`The account has been disabled successfully.`),
|
||||||
|
duration: 5000,
|
||||||
|
});
|
||||||
|
} catch (err) {
|
||||||
|
const error = AppError.parseError(err);
|
||||||
|
|
||||||
|
const errorMessage = match(error.code)
|
||||||
|
.with(AppErrorCode.NOT_FOUND, () => msg`User not found.`)
|
||||||
|
.with(AppErrorCode.UNAUTHORIZED, () => msg`You are not authorized to disable this user.`)
|
||||||
|
.otherwise(() => msg`An error occurred while disabling the user.`);
|
||||||
|
|
||||||
|
toast({
|
||||||
|
title: _(msg`Error`),
|
||||||
|
description: _(errorMessage),
|
||||||
|
variant: 'destructive',
|
||||||
|
duration: 7500,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className={className}>
|
||||||
|
<Alert
|
||||||
|
className="flex flex-col items-center justify-between gap-4 p-6 md:flex-row"
|
||||||
|
variant="neutral"
|
||||||
|
>
|
||||||
|
<div>
|
||||||
|
<AlertTitle>Disable Account</AlertTitle>
|
||||||
|
<AlertDescription className="mr-2">
|
||||||
|
<Trans>
|
||||||
|
Disabling the user results in the user not being able to use the account. It also
|
||||||
|
disables all the related contents such as subscription, webhooks, teams, and API keys.
|
||||||
|
</Trans>
|
||||||
|
</AlertDescription>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="flex-shrink-0">
|
||||||
|
<Dialog>
|
||||||
|
<DialogTrigger asChild>
|
||||||
|
<Button variant="destructive">
|
||||||
|
<Trans>Disable Account</Trans>
|
||||||
|
</Button>
|
||||||
|
</DialogTrigger>
|
||||||
|
|
||||||
|
<DialogContent>
|
||||||
|
<DialogHeader className="space-y-4">
|
||||||
|
<DialogTitle>
|
||||||
|
<Trans>Disable Account</Trans>
|
||||||
|
</DialogTitle>
|
||||||
|
|
||||||
|
<Alert variant="destructive">
|
||||||
|
<AlertDescription className="selection:bg-red-100">
|
||||||
|
<Trans>
|
||||||
|
This action is reversible, but please be careful as the account may be
|
||||||
|
affected permanently (e.g. their settings and contents not being restored
|
||||||
|
properly).
|
||||||
|
</Trans>
|
||||||
|
</AlertDescription>
|
||||||
|
</Alert>
|
||||||
|
</DialogHeader>
|
||||||
|
|
||||||
|
<div>
|
||||||
|
<DialogDescription>
|
||||||
|
<Trans>
|
||||||
|
To confirm, please enter the accounts email address <br />({userToDisable.email}
|
||||||
|
).
|
||||||
|
</Trans>
|
||||||
|
</DialogDescription>
|
||||||
|
|
||||||
|
<Input
|
||||||
|
className="mt-2"
|
||||||
|
type="email"
|
||||||
|
value={email}
|
||||||
|
onChange={(e) => setEmail(e.target.value)}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<DialogFooter>
|
||||||
|
<Button
|
||||||
|
onClick={onDisableAccount}
|
||||||
|
loading={isDisablingUser}
|
||||||
|
variant="destructive"
|
||||||
|
disabled={email !== userToDisable.email}
|
||||||
|
>
|
||||||
|
<Trans>Disable account</Trans>
|
||||||
|
</Button>
|
||||||
|
</DialogFooter>
|
||||||
|
</DialogContent>
|
||||||
|
</Dialog>
|
||||||
|
</div>
|
||||||
|
</Alert>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
@ -0,0 +1,130 @@
|
|||||||
|
'use client';
|
||||||
|
|
||||||
|
import { useState } from 'react';
|
||||||
|
|
||||||
|
import { Trans, msg } from '@lingui/macro';
|
||||||
|
import { useLingui } from '@lingui/react';
|
||||||
|
import { match } from 'ts-pattern';
|
||||||
|
|
||||||
|
import { AppError, AppErrorCode } from '@documenso/lib/errors/app-error';
|
||||||
|
import type { User } from '@documenso/prisma/client';
|
||||||
|
import { trpc } from '@documenso/trpc/react';
|
||||||
|
import { Alert, AlertDescription, AlertTitle } from '@documenso/ui/primitives/alert';
|
||||||
|
import { Button } from '@documenso/ui/primitives/button';
|
||||||
|
import {
|
||||||
|
Dialog,
|
||||||
|
DialogContent,
|
||||||
|
DialogDescription,
|
||||||
|
DialogFooter,
|
||||||
|
DialogHeader,
|
||||||
|
DialogTitle,
|
||||||
|
DialogTrigger,
|
||||||
|
} from '@documenso/ui/primitives/dialog';
|
||||||
|
import { Input } from '@documenso/ui/primitives/input';
|
||||||
|
import { useToast } from '@documenso/ui/primitives/use-toast';
|
||||||
|
|
||||||
|
export type EnableUserDialogProps = {
|
||||||
|
className?: string;
|
||||||
|
userToEnable: User;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const EnableUserDialog = ({ className, userToEnable }: EnableUserDialogProps) => {
|
||||||
|
const { toast } = useToast();
|
||||||
|
const { _ } = useLingui();
|
||||||
|
|
||||||
|
const [email, setEmail] = useState('');
|
||||||
|
|
||||||
|
const { mutateAsync: enableUser, isLoading: isEnablingUser } =
|
||||||
|
trpc.admin.enableUser.useMutation();
|
||||||
|
|
||||||
|
const onEnableAccount = async () => {
|
||||||
|
try {
|
||||||
|
await enableUser({
|
||||||
|
id: userToEnable.id,
|
||||||
|
});
|
||||||
|
|
||||||
|
toast({
|
||||||
|
title: _(msg`Account enabled`),
|
||||||
|
description: _(msg`The account has been enabled successfully.`),
|
||||||
|
duration: 5000,
|
||||||
|
});
|
||||||
|
} catch (err) {
|
||||||
|
const error = AppError.parseError(err);
|
||||||
|
|
||||||
|
const errorMessage = match(error.code)
|
||||||
|
.with(AppErrorCode.NOT_FOUND, () => msg`User not found.`)
|
||||||
|
.with(AppErrorCode.UNAUTHORIZED, () => msg`You are not authorized to enable this user.`)
|
||||||
|
.otherwise(() => msg`An error occurred while enabling the user.`);
|
||||||
|
|
||||||
|
toast({
|
||||||
|
title: _(msg`Error`),
|
||||||
|
description: _(errorMessage),
|
||||||
|
variant: 'destructive',
|
||||||
|
duration: 7500,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className={className}>
|
||||||
|
<Alert
|
||||||
|
className="flex flex-col items-center justify-between gap-4 p-6 md:flex-row"
|
||||||
|
variant="neutral"
|
||||||
|
>
|
||||||
|
<div>
|
||||||
|
<AlertTitle>Enable Account</AlertTitle>
|
||||||
|
<AlertDescription className="mr-2">
|
||||||
|
<Trans>
|
||||||
|
Enabling the account results in the user being able to use the account again, and all
|
||||||
|
the related features such as webhooks, teams, and API keys for example.
|
||||||
|
</Trans>
|
||||||
|
</AlertDescription>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="flex-shrink-0">
|
||||||
|
<Dialog>
|
||||||
|
<DialogTrigger asChild>
|
||||||
|
<Button>
|
||||||
|
<Trans>Enable Account</Trans>
|
||||||
|
</Button>
|
||||||
|
</DialogTrigger>
|
||||||
|
|
||||||
|
<DialogContent>
|
||||||
|
<DialogHeader className="space-y-4">
|
||||||
|
<DialogTitle>
|
||||||
|
<Trans>Enable Account</Trans>
|
||||||
|
</DialogTitle>
|
||||||
|
</DialogHeader>
|
||||||
|
|
||||||
|
<div>
|
||||||
|
<DialogDescription>
|
||||||
|
<Trans>
|
||||||
|
To confirm, please enter the accounts email address <br />({userToEnable.email}
|
||||||
|
).
|
||||||
|
</Trans>
|
||||||
|
</DialogDescription>
|
||||||
|
|
||||||
|
<Input
|
||||||
|
className="mt-2"
|
||||||
|
type="email"
|
||||||
|
value={email}
|
||||||
|
onChange={(e) => setEmail(e.target.value)}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<DialogFooter>
|
||||||
|
<Button
|
||||||
|
onClick={onEnableAccount}
|
||||||
|
loading={isEnablingUser}
|
||||||
|
disabled={email !== userToEnable.email}
|
||||||
|
>
|
||||||
|
<Trans>Enable account</Trans>
|
||||||
|
</Button>
|
||||||
|
</DialogFooter>
|
||||||
|
</DialogContent>
|
||||||
|
</Dialog>
|
||||||
|
</div>
|
||||||
|
</Alert>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
@ -23,6 +23,8 @@ import { Input } from '@documenso/ui/primitives/input';
|
|||||||
import { useToast } from '@documenso/ui/primitives/use-toast';
|
import { useToast } from '@documenso/ui/primitives/use-toast';
|
||||||
|
|
||||||
import { DeleteUserDialog } from './delete-user-dialog';
|
import { DeleteUserDialog } from './delete-user-dialog';
|
||||||
|
import { DisableUserDialog } from './disable-user-dialog';
|
||||||
|
import { EnableUserDialog } from './enable-user-dialog';
|
||||||
import { MultiSelectRoleCombobox } from './multiselect-role-combobox';
|
import { MultiSelectRoleCombobox } from './multiselect-role-combobox';
|
||||||
|
|
||||||
const ZUserFormSchema = ZAdminUpdateProfileMutationSchema.omit({ id: true });
|
const ZUserFormSchema = ZAdminUpdateProfileMutationSchema.omit({ id: true });
|
||||||
@ -153,7 +155,11 @@ export default function UserPage({ params }: { params: { id: number } }) {
|
|||||||
|
|
||||||
<hr className="my-4" />
|
<hr className="my-4" />
|
||||||
|
|
||||||
{user && <DeleteUserDialog user={user} />}
|
<div className="flex flex-col items-center gap-4">
|
||||||
|
{user && <DeleteUserDialog user={user} />}
|
||||||
|
{user && user.disabled && <EnableUserDialog userToEnable={user} />}
|
||||||
|
{user && !user.disabled && <DisableUserDialog userToDisable={user} />}
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@ -1,5 +1,6 @@
|
|||||||
import type { NextApiRequest } from 'next';
|
import type { NextApiRequest } from 'next';
|
||||||
|
|
||||||
|
import { AppError, AppErrorCode } from '@documenso/lib/errors/app-error';
|
||||||
import { getApiTokenByToken } from '@documenso/lib/server-only/public-api/get-api-token-by-token';
|
import { getApiTokenByToken } from '@documenso/lib/server-only/public-api/get-api-token-by-token';
|
||||||
import type { Team, User } from '@documenso/prisma/client';
|
import type { Team, User } from '@documenso/prisma/client';
|
||||||
|
|
||||||
@ -22,18 +23,33 @@ export const authenticatedMiddleware = <
|
|||||||
const [token] = (authorization || '').split('Bearer ').filter((s) => s.length > 0);
|
const [token] = (authorization || '').split('Bearer ').filter((s) => s.length > 0);
|
||||||
|
|
||||||
if (!token) {
|
if (!token) {
|
||||||
throw new Error('Token was not provided for authenticated middleware');
|
throw new AppError(AppErrorCode.UNAUTHORIZED, {
|
||||||
|
message: 'API token was not provided',
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
const apiToken = await getApiTokenByToken({ token });
|
const apiToken = await getApiTokenByToken({ token });
|
||||||
|
|
||||||
|
if (apiToken.user.disabled) {
|
||||||
|
throw new AppError(AppErrorCode.UNAUTHORIZED, {
|
||||||
|
message: 'User is disabled',
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
return await handler(args, apiToken.user, apiToken.team);
|
return await handler(args, apiToken.user, apiToken.team);
|
||||||
} catch (_err) {
|
} catch (err) {
|
||||||
console.log({ _err });
|
console.log({ err: err });
|
||||||
|
|
||||||
|
let message = 'Unauthorized';
|
||||||
|
|
||||||
|
if (err instanceof AppError) {
|
||||||
|
message = err.message;
|
||||||
|
}
|
||||||
|
|
||||||
return {
|
return {
|
||||||
status: 401,
|
status: 401,
|
||||||
body: {
|
body: {
|
||||||
message: 'Unauthorized',
|
message,
|
||||||
},
|
},
|
||||||
} as const;
|
} as const;
|
||||||
}
|
}
|
||||||
|
|||||||
@ -21,6 +21,10 @@ export const getServerComponentSession = cache(async () => {
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
|
if (user.disabled) {
|
||||||
|
return { user: null, session: null };
|
||||||
|
}
|
||||||
|
|
||||||
return { user, session };
|
return { user, session };
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|||||||
69
packages/lib/server-only/user/disable-user.ts
Normal file
69
packages/lib/server-only/user/disable-user.ts
Normal file
@ -0,0 +1,69 @@
|
|||||||
|
import { AppError } from '@documenso/lib/errors/app-error';
|
||||||
|
import { prisma } from '@documenso/prisma';
|
||||||
|
|
||||||
|
export type DisableUserOptions = {
|
||||||
|
id: number;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const disableUser = async ({ id }: DisableUserOptions) => {
|
||||||
|
const user = await prisma.user.findFirst({
|
||||||
|
where: {
|
||||||
|
id,
|
||||||
|
},
|
||||||
|
include: {
|
||||||
|
ApiToken: true,
|
||||||
|
Webhooks: true,
|
||||||
|
passkeys: true,
|
||||||
|
VerificationToken: true,
|
||||||
|
PasswordResetToken: true,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!user) {
|
||||||
|
throw new AppError('There was an error disabling the user');
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
await prisma.$transaction(async (tx) => {
|
||||||
|
await tx.user.update({
|
||||||
|
where: { id },
|
||||||
|
data: { disabled: true },
|
||||||
|
});
|
||||||
|
|
||||||
|
await tx.apiToken.updateMany({
|
||||||
|
where: { userId: id },
|
||||||
|
data: {
|
||||||
|
expires: new Date(),
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
await tx.webhook.updateMany({
|
||||||
|
where: { userId: id },
|
||||||
|
data: {
|
||||||
|
enabled: false,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
await tx.verificationToken.updateMany({
|
||||||
|
where: { userId: id },
|
||||||
|
data: {
|
||||||
|
expires: new Date(),
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
await tx.passwordResetToken.updateMany({
|
||||||
|
where: { userId: id },
|
||||||
|
data: {
|
||||||
|
expiry: new Date(),
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
await tx.passkey.deleteMany({
|
||||||
|
where: { userId: id },
|
||||||
|
});
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Error disabling user', error);
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
};
|
||||||
27
packages/lib/server-only/user/enable-user.ts
Normal file
27
packages/lib/server-only/user/enable-user.ts
Normal file
@ -0,0 +1,27 @@
|
|||||||
|
import { AppError } from '@documenso/lib/errors/app-error';
|
||||||
|
import { prisma } from '@documenso/prisma';
|
||||||
|
|
||||||
|
export type EnableUserOptions = {
|
||||||
|
id: number;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const enableUser = async ({ id }: EnableUserOptions) => {
|
||||||
|
const user = await prisma.user.findFirst({
|
||||||
|
where: {
|
||||||
|
id,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!user) {
|
||||||
|
throw new AppError('There was an error enabling the user');
|
||||||
|
}
|
||||||
|
|
||||||
|
await prisma.user.update({
|
||||||
|
where: {
|
||||||
|
id,
|
||||||
|
},
|
||||||
|
data: {
|
||||||
|
disabled: false,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
};
|
||||||
@ -1,3 +1,4 @@
|
|||||||
|
import { AppError, AppErrorCode } from '@documenso/lib/errors/app-error';
|
||||||
import { findDocuments } from '@documenso/lib/server-only/admin/get-all-documents';
|
import { findDocuments } from '@documenso/lib/server-only/admin/get-all-documents';
|
||||||
import { getEntireDocument } from '@documenso/lib/server-only/admin/get-entire-document';
|
import { getEntireDocument } from '@documenso/lib/server-only/admin/get-entire-document';
|
||||||
import { updateRecipient } from '@documenso/lib/server-only/admin/update-recipient';
|
import { updateRecipient } from '@documenso/lib/server-only/admin/update-recipient';
|
||||||
@ -7,6 +8,8 @@ import { sendDeleteEmail } from '@documenso/lib/server-only/document/send-delete
|
|||||||
import { superDeleteDocument } from '@documenso/lib/server-only/document/super-delete-document';
|
import { superDeleteDocument } from '@documenso/lib/server-only/document/super-delete-document';
|
||||||
import { upsertSiteSetting } from '@documenso/lib/server-only/site-settings/upsert-site-setting';
|
import { upsertSiteSetting } from '@documenso/lib/server-only/site-settings/upsert-site-setting';
|
||||||
import { deleteUser } from '@documenso/lib/server-only/user/delete-user';
|
import { deleteUser } from '@documenso/lib/server-only/user/delete-user';
|
||||||
|
import { disableUser } from '@documenso/lib/server-only/user/disable-user';
|
||||||
|
import { enableUser } from '@documenso/lib/server-only/user/enable-user';
|
||||||
import { getUserById } from '@documenso/lib/server-only/user/get-user-by-id';
|
import { getUserById } from '@documenso/lib/server-only/user/get-user-by-id';
|
||||||
import { extractNextApiRequestMetadata } from '@documenso/lib/universal/extract-request-metadata';
|
import { extractNextApiRequestMetadata } from '@documenso/lib/universal/extract-request-metadata';
|
||||||
import { DocumentStatus } from '@documenso/prisma/client';
|
import { DocumentStatus } from '@documenso/prisma/client';
|
||||||
@ -15,6 +18,8 @@ import { adminProcedure, router } from '../trpc';
|
|||||||
import {
|
import {
|
||||||
ZAdminDeleteDocumentMutationSchema,
|
ZAdminDeleteDocumentMutationSchema,
|
||||||
ZAdminDeleteUserMutationSchema,
|
ZAdminDeleteUserMutationSchema,
|
||||||
|
ZAdminDisableUserMutationSchema,
|
||||||
|
ZAdminEnableUserMutationSchema,
|
||||||
ZAdminFindDocumentsQuerySchema,
|
ZAdminFindDocumentsQuerySchema,
|
||||||
ZAdminResealDocumentMutationSchema,
|
ZAdminResealDocumentMutationSchema,
|
||||||
ZAdminUpdateProfileMutationSchema,
|
ZAdminUpdateProfileMutationSchema,
|
||||||
@ -70,13 +75,43 @@ export const adminRouter = router({
|
|||||||
return await sealDocument({ documentId: id, isResealing });
|
return await sealDocument({ documentId: id, isResealing });
|
||||||
}),
|
}),
|
||||||
|
|
||||||
|
enableUser: adminProcedure.input(ZAdminEnableUserMutationSchema).mutation(async ({ input }) => {
|
||||||
|
const { id } = input;
|
||||||
|
|
||||||
|
const user = await getUserById({ id }).catch(() => null);
|
||||||
|
|
||||||
|
if (!user) {
|
||||||
|
throw new AppError(AppErrorCode.NOT_FOUND, {
|
||||||
|
message: 'User not found',
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
return await enableUser({ id });
|
||||||
|
}),
|
||||||
|
|
||||||
|
disableUser: adminProcedure.input(ZAdminDisableUserMutationSchema).mutation(async ({ input }) => {
|
||||||
|
const { id } = input;
|
||||||
|
|
||||||
|
const user = await getUserById({ id }).catch(() => null);
|
||||||
|
|
||||||
|
if (!user) {
|
||||||
|
throw new AppError(AppErrorCode.NOT_FOUND, {
|
||||||
|
message: 'User not found',
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
return await disableUser({ id });
|
||||||
|
}),
|
||||||
|
|
||||||
deleteUser: adminProcedure.input(ZAdminDeleteUserMutationSchema).mutation(async ({ input }) => {
|
deleteUser: adminProcedure.input(ZAdminDeleteUserMutationSchema).mutation(async ({ input }) => {
|
||||||
const { id, email } = input;
|
const { id } = input;
|
||||||
|
|
||||||
const user = await getUserById({ id });
|
const user = await getUserById({ id }).catch(() => null);
|
||||||
|
|
||||||
if (user.email !== email) {
|
if (!user) {
|
||||||
throw new Error('Email does not match');
|
throw new AppError(AppErrorCode.NOT_FOUND, {
|
||||||
|
message: 'User not found',
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
return await deleteUser({ id });
|
return await deleteUser({ id });
|
||||||
|
|||||||
@ -43,11 +43,22 @@ export type TAdminResealDocumentMutationSchema = z.infer<typeof ZAdminResealDocu
|
|||||||
|
|
||||||
export const ZAdminDeleteUserMutationSchema = z.object({
|
export const ZAdminDeleteUserMutationSchema = z.object({
|
||||||
id: z.number().min(1),
|
id: z.number().min(1),
|
||||||
email: z.string().email(),
|
|
||||||
});
|
});
|
||||||
|
|
||||||
export type TAdminDeleteUserMutationSchema = z.infer<typeof ZAdminDeleteUserMutationSchema>;
|
export type TAdminDeleteUserMutationSchema = z.infer<typeof ZAdminDeleteUserMutationSchema>;
|
||||||
|
|
||||||
|
export const ZAdminEnableUserMutationSchema = z.object({
|
||||||
|
id: z.number().min(1),
|
||||||
|
});
|
||||||
|
|
||||||
|
export type TAdminEnableUserMutationSchema = z.infer<typeof ZAdminEnableUserMutationSchema>;
|
||||||
|
|
||||||
|
export const ZAdminDisableUserMutationSchema = z.object({
|
||||||
|
id: z.number().min(1),
|
||||||
|
});
|
||||||
|
|
||||||
|
export type TAdminDisableUserMutationSchema = z.infer<typeof ZAdminDisableUserMutationSchema>;
|
||||||
|
|
||||||
export const ZAdminDeleteDocumentMutationSchema = z.object({
|
export const ZAdminDeleteDocumentMutationSchema = z.object({
|
||||||
id: z.number().min(1),
|
id: z.number().min(1),
|
||||||
reason: z.string(),
|
reason: z.string(),
|
||||||
|
|||||||
Reference in New Issue
Block a user