implement update user, update password and delete account
This commit is contained in:
parent
12983316f5
commit
c9b657e44c
@ -1,24 +0,0 @@
|
|||||||
import { NotFoundError, SecurePassword, resolver } from "blitz";
|
|
||||||
|
|
||||||
import db from "../../../db";
|
|
||||||
import { authenticateUser } from "./login";
|
|
||||||
import { ChangePassword } from "../validations";
|
|
||||||
|
|
||||||
export default resolver.pipe(
|
|
||||||
resolver.zod(ChangePassword),
|
|
||||||
resolver.authorize(),
|
|
||||||
async ({ currentPassword, newPassword }, ctx) => {
|
|
||||||
const user = await db.user.findFirst({ where: { id: ctx.session.userId! } });
|
|
||||||
if (!user) throw new NotFoundError();
|
|
||||||
|
|
||||||
await authenticateUser(user.email, currentPassword);
|
|
||||||
|
|
||||||
const hashedPassword = await SecurePassword.hash(newPassword.trim());
|
|
||||||
await db.user.update({
|
|
||||||
where: { id: user.id },
|
|
||||||
data: { hashedPassword },
|
|
||||||
});
|
|
||||||
|
|
||||||
return true;
|
|
||||||
},
|
|
||||||
);
|
|
@ -1,4 +1,4 @@
|
|||||||
import { Ctx } from "blitz";
|
import type { Ctx } from "blitz";
|
||||||
|
|
||||||
export default async function logout(_ = null, ctx: Ctx) {
|
export default async function logout(_ = null, ctx: Ctx) {
|
||||||
return await ctx.session.$revoke();
|
return await ctx.session.$revoke();
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
import { z } from "zod";
|
import { z } from "zod";
|
||||||
|
|
||||||
const password = z.string().min(10).max(100);
|
export const password = z.string().min(10).max(100);
|
||||||
|
|
||||||
export const Signup = z.object({
|
export const Signup = z.object({
|
||||||
email: z.string().email(),
|
email: z.string().email(),
|
||||||
@ -26,8 +26,3 @@ export const ResetPassword = z
|
|||||||
message: "Passwords don't match",
|
message: "Passwords don't match",
|
||||||
path: ["passwordConfirmation"], // set the path of the error
|
path: ["passwordConfirmation"], // set the path of the error
|
||||||
});
|
});
|
||||||
|
|
||||||
export const ChangePassword = z.object({
|
|
||||||
currentPassword: z.string(),
|
|
||||||
newPassword: password,
|
|
||||||
});
|
|
||||||
|
72
app/settings/api/queue/delete-user-data.ts
Normal file
72
app/settings/api/queue/delete-user-data.ts
Normal file
@ -0,0 +1,72 @@
|
|||||||
|
import { Queue } from "quirrel/blitz";
|
||||||
|
|
||||||
|
import db, { MembershipRole } from "../../../../db";
|
||||||
|
import appLogger from "../../../../integrations/logger";
|
||||||
|
|
||||||
|
const logger = appLogger.child({ queue: "delete-user-data" });
|
||||||
|
|
||||||
|
type Payload = {
|
||||||
|
userId: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
const deleteUserData = Queue<Payload>("api/queue/delete-user-data", async ({ userId }) => {
|
||||||
|
const user = await db.user.findFirst({
|
||||||
|
where: { id: userId },
|
||||||
|
include: {
|
||||||
|
memberships: {
|
||||||
|
include: {
|
||||||
|
organization: {
|
||||||
|
include: { memberships: { include: { user: true } } },
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
});
|
||||||
|
if (!user) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
switch (user.memberships[0]!.role) {
|
||||||
|
case MembershipRole.OWNER: {
|
||||||
|
const organization = user.memberships[0]!.organization;
|
||||||
|
const where = { organizationId: organization.id };
|
||||||
|
await Promise.all<unknown>([
|
||||||
|
db.notificationSubscription.deleteMany({ where }),
|
||||||
|
db.phoneCall.deleteMany({ where }),
|
||||||
|
db.message.deleteMany({ where }),
|
||||||
|
db.processingPhoneNumber.deleteMany({ where }),
|
||||||
|
]);
|
||||||
|
await db.phoneNumber.deleteMany({ where });
|
||||||
|
|
||||||
|
const orgMembers = organization.memberships
|
||||||
|
.map((membership) => membership.user!)
|
||||||
|
.filter((user) => user !== null);
|
||||||
|
await Promise.all(
|
||||||
|
orgMembers.map((member) =>
|
||||||
|
Promise.all([
|
||||||
|
db.token.deleteMany({ where: { userId: member.id } }),
|
||||||
|
db.session.deleteMany({ where: { userId: member.id } }),
|
||||||
|
db.membership.deleteMany({ where: { userId: member.id } }),
|
||||||
|
db.user.delete({ where: { id: member.id } }),
|
||||||
|
]),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
await db.organization.delete({ where: { id: organization.id } });
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case MembershipRole.USER: {
|
||||||
|
await Promise.all([
|
||||||
|
db.token.deleteMany({ where: { userId: user.id } }),
|
||||||
|
db.session.deleteMany({ where: { userId: user.id } }),
|
||||||
|
db.user.delete({ where: { id: user.id } }),
|
||||||
|
db.membership.deleteMany({ where: { userId: user.id } }),
|
||||||
|
]);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case MembershipRole.ADMIN:
|
||||||
|
// nothing to do here?
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
export default deleteUserData;
|
28
app/settings/api/queue/notify-email-change.ts
Normal file
28
app/settings/api/queue/notify-email-change.ts
Normal file
@ -0,0 +1,28 @@
|
|||||||
|
import { Queue } from "quirrel/blitz";
|
||||||
|
|
||||||
|
import appLogger from "../../../../integrations/logger";
|
||||||
|
import { sendEmail } from "../../../../integrations/ses";
|
||||||
|
|
||||||
|
const logger = appLogger.child({ queue: "notify-email-change" });
|
||||||
|
|
||||||
|
type Payload = {
|
||||||
|
oldEmail: string;
|
||||||
|
newEmail: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
const notifyEmailChangeQueue = Queue<Payload>("api/queue/notify-email-change", async ({ oldEmail, newEmail }) => {
|
||||||
|
await Promise.all([
|
||||||
|
sendEmail({
|
||||||
|
recipients: [oldEmail],
|
||||||
|
subject: "",
|
||||||
|
body: "",
|
||||||
|
}),
|
||||||
|
sendEmail({
|
||||||
|
recipients: [newEmail],
|
||||||
|
subject: "",
|
||||||
|
body: "",
|
||||||
|
}),
|
||||||
|
]);
|
||||||
|
});
|
||||||
|
|
||||||
|
export default notifyEmailChangeQueue;
|
@ -1,11 +1,14 @@
|
|||||||
import { useRef, useState } from "react";
|
import { useRef, useState } from "react";
|
||||||
|
import { useMutation } from "blitz";
|
||||||
import clsx from "clsx";
|
import clsx from "clsx";
|
||||||
|
|
||||||
import Button from "./button";
|
import Button from "./button";
|
||||||
import SettingsSection from "./settings-section";
|
import SettingsSection from "./settings-section";
|
||||||
import Modal, { ModalTitle } from "./modal";
|
import Modal, { ModalTitle } from "./modal";
|
||||||
|
import deleteUser from "../mutations/delete-user";
|
||||||
|
|
||||||
export default function DangerZone() {
|
export default function DangerZone() {
|
||||||
|
const deleteUserMutation = useMutation(deleteUser)[0];
|
||||||
const [isDeletingUser, setIsDeletingUser] = useState(false);
|
const [isDeletingUser, setIsDeletingUser] = useState(false);
|
||||||
const [isConfirmationModalOpen, setIsConfirmationModalOpen] = useState(false);
|
const [isConfirmationModalOpen, setIsConfirmationModalOpen] = useState(false);
|
||||||
const modalCancelButtonRef = useRef<HTMLButtonElement>(null);
|
const modalCancelButtonRef = useRef<HTMLButtonElement>(null);
|
||||||
@ -19,14 +22,17 @@ export default function DangerZone() {
|
|||||||
};
|
};
|
||||||
const onConfirm = () => {
|
const onConfirm = () => {
|
||||||
setIsDeletingUser(true);
|
setIsDeletingUser(true);
|
||||||
// user.deleteUser();
|
return deleteUserMutation();
|
||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SettingsSection title="Danger Zone" description="Highway to the Danger Zone 𝅘𝅥𝅮">
|
<SettingsSection title="Danger Zone" description="Highway to the Danger Zone 𝅘𝅥𝅮">
|
||||||
<div className="shadow border border-red-300 sm:rounded-md sm:overflow-hidden">
|
<div className="shadow border border-red-300 sm:rounded-md sm:overflow-hidden">
|
||||||
<div className="flex justify-between items-center flex-row px-4 py-5 bg-white sm:p-6">
|
<div className="flex justify-between items-center flex-row px-4 py-5 space-x-2 bg-white sm:p-6">
|
||||||
<p>Once you delete your account, all of its data will be permanently deleted.</p>
|
<p>
|
||||||
|
Once you delete your account, all of its data will be permanently deleted and any ongoing
|
||||||
|
subscription will be cancelled.
|
||||||
|
</p>
|
||||||
|
|
||||||
<span className="text-base font-medium">
|
<span className="text-base font-medium">
|
||||||
<Button variant="error" type="button" onClick={() => setIsConfirmationModalOpen(true)}>
|
<Button variant="error" type="button" onClick={() => setIsConfirmationModalOpen(true)}>
|
||||||
|
@ -1,8 +1,9 @@
|
|||||||
import type { FunctionComponent } from "react";
|
import type { FunctionComponent } from "react";
|
||||||
import { useEffect, useState } from "react";
|
import { useEffect, useState } from "react";
|
||||||
import { useRouter } from "blitz";
|
import { useMutation } from "blitz";
|
||||||
import { useForm } from "react-hook-form";
|
import { useForm } from "react-hook-form";
|
||||||
|
|
||||||
|
import updateUser from "../mutations/update-user";
|
||||||
import Alert from "./alert";
|
import Alert from "./alert";
|
||||||
import Button from "./button";
|
import Button from "./button";
|
||||||
import SettingsSection from "./settings-section";
|
import SettingsSection from "./settings-section";
|
||||||
@ -19,7 +20,7 @@ const logger = appLogger.child({ module: "profile-settings" });
|
|||||||
|
|
||||||
const ProfileInformations: FunctionComponent = () => {
|
const ProfileInformations: FunctionComponent = () => {
|
||||||
const { user } = useCurrentUser();
|
const { user } = useCurrentUser();
|
||||||
const router = useRouter();
|
const updateUserMutation = useMutation(updateUser)[0];
|
||||||
const {
|
const {
|
||||||
register,
|
register,
|
||||||
handleSubmit,
|
handleSubmit,
|
||||||
@ -39,16 +40,9 @@ const ProfileInformations: FunctionComponent = () => {
|
|||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
// TODO
|
await updateUserMutation({ email, name });
|
||||||
// await updateUser({ email, data: { name } });
|
|
||||||
} catch (error: any) {
|
} catch (error: any) {
|
||||||
logger.error(error.response, "error updating user infos");
|
logger.error(error.response, "error updating user infos");
|
||||||
|
|
||||||
if (error.response.status === 401) {
|
|
||||||
logger.error("session expired, redirecting to sign in page");
|
|
||||||
return router.push("/auth/sign-in");
|
|
||||||
}
|
|
||||||
|
|
||||||
setErrorMessage(error.response.data.errorMessage);
|
setErrorMessage(error.response.data.errorMessage);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
import type { FunctionComponent } from "react";
|
import type { FunctionComponent } from "react";
|
||||||
import { useState } from "react";
|
import { useState } from "react";
|
||||||
import { useRouter } from "blitz";
|
import { useMutation } from "blitz";
|
||||||
import { useForm } from "react-hook-form";
|
import { useForm } from "react-hook-form";
|
||||||
|
|
||||||
import Alert from "./alert";
|
import Alert from "./alert";
|
||||||
@ -8,16 +8,17 @@ import Button from "./button";
|
|||||||
import SettingsSection from "./settings-section";
|
import SettingsSection from "./settings-section";
|
||||||
|
|
||||||
import appLogger from "../../../integrations/logger";
|
import appLogger from "../../../integrations/logger";
|
||||||
|
import changePassword from "../mutations/change-password";
|
||||||
|
|
||||||
const logger = appLogger.child({ module: "update-password" });
|
const logger = appLogger.child({ module: "update-password" });
|
||||||
|
|
||||||
type Form = {
|
type Form = {
|
||||||
|
currentPassword: string;
|
||||||
newPassword: string;
|
newPassword: string;
|
||||||
newPasswordConfirmation: string;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
const UpdatePassword: FunctionComponent = () => {
|
const UpdatePassword: FunctionComponent = () => {
|
||||||
const router = useRouter();
|
const changePasswordMutation = useMutation(changePassword)[0];
|
||||||
const {
|
const {
|
||||||
register,
|
register,
|
||||||
handleSubmit,
|
handleSubmit,
|
||||||
@ -25,28 +26,18 @@ const UpdatePassword: FunctionComponent = () => {
|
|||||||
} = useForm<Form>();
|
} = useForm<Form>();
|
||||||
const [errorMessage, setErrorMessage] = useState("");
|
const [errorMessage, setErrorMessage] = useState("");
|
||||||
|
|
||||||
const onSubmit = handleSubmit(async ({ newPassword, newPasswordConfirmation }) => {
|
const onSubmit = handleSubmit(async ({ currentPassword, newPassword }) => {
|
||||||
if (isSubmitting) {
|
if (isSubmitting) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (newPassword !== newPasswordConfirmation) {
|
setErrorMessage("");
|
||||||
setErrorMessage("New passwords don't match");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
try {
|
try {
|
||||||
// TODO
|
await changePasswordMutation({ currentPassword, newPassword });
|
||||||
// await customer.updateUser({ password: newPassword });
|
|
||||||
} catch (error: any) {
|
} catch (error: any) {
|
||||||
logger.error(error.response, "error updating user infos");
|
logger.error(error, "error updating user infos");
|
||||||
|
setErrorMessage(error.message);
|
||||||
if (error.response.status === 401) {
|
|
||||||
logger.error("session expired, redirecting to sign in page");
|
|
||||||
return router.push("/auth/sign-in");
|
|
||||||
}
|
|
||||||
|
|
||||||
setErrorMessage(error.response.data.errorMessage);
|
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -62,7 +53,7 @@ const UpdatePassword: FunctionComponent = () => {
|
|||||||
</div>
|
</div>
|
||||||
) : null}
|
) : null}
|
||||||
|
|
||||||
{isSubmitSuccessful ? (
|
{!isSubmitting && isSubmitSuccessful && !errorMessage ? (
|
||||||
<div className="mb-8">
|
<div className="mb-8">
|
||||||
<Alert title="Saved successfully" message="Your changes have been saved." variant="success" />
|
<Alert title="Saved successfully" message="Your changes have been saved." variant="success" />
|
||||||
</div>
|
</div>
|
||||||
@ -70,6 +61,25 @@ const UpdatePassword: FunctionComponent = () => {
|
|||||||
|
|
||||||
<div className="shadow sm:rounded-md sm:overflow-hidden">
|
<div className="shadow sm:rounded-md sm:overflow-hidden">
|
||||||
<div className="px-4 py-5 bg-white space-y-6 sm:p-6">
|
<div className="px-4 py-5 bg-white space-y-6 sm:p-6">
|
||||||
|
<div>
|
||||||
|
<label
|
||||||
|
htmlFor="currentPassword"
|
||||||
|
className="flex justify-between text-sm font-medium leading-5 text-gray-700"
|
||||||
|
>
|
||||||
|
<div>Current password</div>
|
||||||
|
</label>
|
||||||
|
<div className="mt-1 rounded-md shadow-sm">
|
||||||
|
<input
|
||||||
|
id="currentPassword"
|
||||||
|
type="password"
|
||||||
|
tabIndex={3}
|
||||||
|
className="appearance-none block w-full px-3 py-2 border border-gray-300 rounded-md placeholder-gray-400 focus:outline-none focus:shadow-outline-primary focus:border-primary-300 transition duration-150 ease-in-out sm:text-sm sm:leading-5"
|
||||||
|
{...register("currentPassword")}
|
||||||
|
required
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
<div>
|
<div>
|
||||||
<label
|
<label
|
||||||
htmlFor="newPassword"
|
htmlFor="newPassword"
|
||||||
@ -81,28 +91,9 @@ const UpdatePassword: FunctionComponent = () => {
|
|||||||
<input
|
<input
|
||||||
id="newPassword"
|
id="newPassword"
|
||||||
type="password"
|
type="password"
|
||||||
tabIndex={3}
|
|
||||||
className="appearance-none block w-full px-3 py-2 border border-gray-300 rounded-md placeholder-gray-400 focus:outline-none focus:shadow-outline-primary focus:border-primary-300 transition duration-150 ease-in-out sm:text-sm sm:leading-5"
|
|
||||||
{...register("newPassword")}
|
|
||||||
required
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div>
|
|
||||||
<label
|
|
||||||
htmlFor="newPasswordConfirmation"
|
|
||||||
className="flex justify-between text-sm font-medium leading-5 text-gray-700"
|
|
||||||
>
|
|
||||||
<div>Confirm new password</div>
|
|
||||||
</label>
|
|
||||||
<div className="mt-1 rounded-md shadow-sm">
|
|
||||||
<input
|
|
||||||
id="newPasswordConfirmation"
|
|
||||||
type="password"
|
|
||||||
tabIndex={4}
|
tabIndex={4}
|
||||||
className="appearance-none block w-full px-3 py-2 border border-gray-300 rounded-md placeholder-gray-400 focus:outline-none focus:shadow-outline-primary focus:border-primary-300 transition duration-150 ease-in-out sm:text-sm sm:leading-5"
|
className="appearance-none block w-full px-3 py-2 border border-gray-300 rounded-md placeholder-gray-400 focus:outline-none focus:shadow-outline-primary focus:border-primary-300 transition duration-150 ease-in-out sm:text-sm sm:leading-5"
|
||||||
{...register("newPasswordConfirmation")}
|
{...register("newPassword")}
|
||||||
required
|
required
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
|
36
app/settings/mutations/change-password.ts
Normal file
36
app/settings/mutations/change-password.ts
Normal file
@ -0,0 +1,36 @@
|
|||||||
|
import { AuthenticationError, NotFoundError, resolver, SecurePassword } from "blitz";
|
||||||
|
import { z } from "zod";
|
||||||
|
|
||||||
|
import db from "../../../db";
|
||||||
|
import { authenticateUser } from "../../auth/mutations/login";
|
||||||
|
import { password } from "../../auth/validations";
|
||||||
|
|
||||||
|
const Body = z.object({
|
||||||
|
currentPassword: z.string(),
|
||||||
|
newPassword: password,
|
||||||
|
});
|
||||||
|
|
||||||
|
export default resolver.pipe(
|
||||||
|
resolver.zod(Body),
|
||||||
|
resolver.authorize(),
|
||||||
|
async ({ currentPassword, newPassword }, ctx) => {
|
||||||
|
const user = await db.user.findFirst({ where: { id: ctx.session.userId! } });
|
||||||
|
if (!user) throw new NotFoundError();
|
||||||
|
|
||||||
|
try {
|
||||||
|
await authenticateUser(user.email, currentPassword);
|
||||||
|
} catch (error) {
|
||||||
|
if (error instanceof AuthenticationError) {
|
||||||
|
throw new Error("Current password is incorrect");
|
||||||
|
}
|
||||||
|
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
|
||||||
|
const hashedPassword = await SecurePassword.hash(newPassword.trim());
|
||||||
|
await db.user.update({
|
||||||
|
where: { id: user.id },
|
||||||
|
data: { hashedPassword },
|
||||||
|
});
|
||||||
|
},
|
||||||
|
);
|
14
app/settings/mutations/delete-user.ts
Normal file
14
app/settings/mutations/delete-user.ts
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
import { NotFoundError, resolver } from "blitz";
|
||||||
|
|
||||||
|
import db from "../../../db";
|
||||||
|
import logout from "../../auth/mutations/logout";
|
||||||
|
import deleteUserData from "../api/queue/delete-user-data";
|
||||||
|
|
||||||
|
export default resolver.pipe(resolver.authorize(), async (_ = null, ctx) => {
|
||||||
|
const user = await db.user.findFirst({ where: { id: ctx.session.userId! } });
|
||||||
|
if (!user) throw new NotFoundError();
|
||||||
|
|
||||||
|
await db.user.update({ where: { id: user.id }, data: { hashedPassword: "pending deletion" } });
|
||||||
|
await deleteUserData.enqueue({ userId: user.id });
|
||||||
|
await logout(null, ctx);
|
||||||
|
});
|
25
app/settings/mutations/update-user.ts
Normal file
25
app/settings/mutations/update-user.ts
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
import { NotFoundError, resolver } from "blitz";
|
||||||
|
import { z } from "zod";
|
||||||
|
|
||||||
|
import db from "../../../db";
|
||||||
|
import notifyEmailChangeQueue from "../api/queue/notify-email-change";
|
||||||
|
|
||||||
|
const Body = z.object({
|
||||||
|
email: z.string().email(),
|
||||||
|
name: z.string(),
|
||||||
|
});
|
||||||
|
|
||||||
|
export default resolver.pipe(resolver.zod(Body), resolver.authorize(), async ({ email, name }, ctx) => {
|
||||||
|
const user = await db.user.findFirst({ where: { id: ctx.session.userId! } });
|
||||||
|
if (!user) throw new NotFoundError();
|
||||||
|
|
||||||
|
const oldEmail = user.email;
|
||||||
|
await db.user.update({
|
||||||
|
where: { id: user.id },
|
||||||
|
data: { email, name },
|
||||||
|
});
|
||||||
|
|
||||||
|
if (oldEmail !== email) {
|
||||||
|
// await notifyEmailChangeQueue.enqueue({ newEmail: email, oldEmail: user.email });
|
||||||
|
}
|
||||||
|
});
|
Loading…
Reference in New Issue
Block a user