mirror of
https://github.com/QuivrHQ/quivr.git
synced 2024-12-15 01:21:48 +03:00
feat(user): Delete User Data from frontend (#2476)
# Description Please include a summary of the changes and the related issue. Please also include relevant motivation and context. ## Checklist before requesting a review Please delete options that are not relevant. - [ ] My code follows the style guidelines of this project - [ ] I have performed a self-review of my code - [ ] I have commented hard-to-understand areas - [ ] I have ideally added tests that prove my fix is effective or that my feature works - [ ] New and existing unit tests pass locally with my changes - [ ] Any dependent changes have been merged ## Screenshots (if appropriate): --------- Co-authored-by: Stan Girard <girard.stanislas@gmail.com>
This commit is contained in:
parent
699097f6ac
commit
8d54187713
@ -79,6 +79,26 @@ def get_user_identity_route(
|
|||||||
"""
|
"""
|
||||||
return user_repository.get_user_identity(current_user.id)
|
return user_repository.get_user_identity(current_user.id)
|
||||||
|
|
||||||
|
@user_router.delete(
|
||||||
|
"/user_data",
|
||||||
|
dependencies=[Depends(AuthBearer())],
|
||||||
|
tags=["User"],
|
||||||
|
)
|
||||||
|
async def delete_user_data_route(
|
||||||
|
current_user: UserIdentity = Depends(get_current_user),
|
||||||
|
):
|
||||||
|
"""
|
||||||
|
Delete a user.
|
||||||
|
|
||||||
|
- `user_id`: The ID of the user to delete.
|
||||||
|
|
||||||
|
This endpoint deletes a user from the system.
|
||||||
|
"""
|
||||||
|
|
||||||
|
user_repository.delete_user_data(current_user.id)
|
||||||
|
|
||||||
|
return {"message": "User deleted successfully"}
|
||||||
|
|
||||||
@user_router.get(
|
@user_router.get(
|
||||||
"/user/credits",
|
"/user/credits",
|
||||||
dependencies=[Depends(AuthBearer())],
|
dependencies=[Depends(AuthBearer())],
|
||||||
|
@ -76,6 +76,30 @@ class Users(UsersInterface):
|
|||||||
).execute()
|
).execute()
|
||||||
return response.data[0]["email"]
|
return response.data[0]["email"]
|
||||||
|
|
||||||
|
def delete_user_data(self, user_id):
|
||||||
|
response = (
|
||||||
|
self.db.from_("brains_users")
|
||||||
|
.select("brain_id")
|
||||||
|
.filter("rights", "eq", "Owner")
|
||||||
|
.filter("user_id", "eq", str(user_id))
|
||||||
|
.execute()
|
||||||
|
)
|
||||||
|
brain_ids = [row["brain_id"] for row in response.data]
|
||||||
|
|
||||||
|
for brain_id in brain_ids:
|
||||||
|
self.db.table("brains").delete().filter("brain_id", "eq", brain_id).execute()
|
||||||
|
|
||||||
|
for brain_id in brain_ids:
|
||||||
|
self.db.table("brains_vectors").delete().filter("brain_id", "eq", brain_id).execute()
|
||||||
|
|
||||||
|
for brain_id in brain_ids:
|
||||||
|
self.db.table("chat_history").delete().filter("brain_id", "eq", brain_id).execute()
|
||||||
|
|
||||||
|
self.db.table("user_settings").delete().filter("user_id", "eq", str(user_id)).execute()
|
||||||
|
self.db.table("user_identity").delete().filter("user_id", "eq", str(user_id)).execute()
|
||||||
|
self.db.table("users").delete().filter("id", "eq", str(user_id)).execute()
|
||||||
|
|
||||||
|
|
||||||
def get_user_credits(self, user_id):
|
def get_user_credits(self, user_id):
|
||||||
user_usage_instance = user_usage.UserUsage(id=user_id)
|
user_usage_instance = user_usage.UserUsage(id=user_id)
|
||||||
|
|
||||||
|
@ -46,6 +46,15 @@ class UsersInterface(ABC):
|
|||||||
pass
|
pass
|
||||||
|
|
||||||
@abstractmethod
|
@abstractmethod
|
||||||
|
def delete_user_data(self, user_id: str):
|
||||||
|
"""
|
||||||
|
Delete a user.
|
||||||
|
|
||||||
|
- `user_id`: The ID of the user to delete.
|
||||||
|
|
||||||
|
This endpoint deletes a user from the system.
|
||||||
|
"""
|
||||||
|
@abstractmethod
|
||||||
def get_user_credits(self, user_id: UUID) -> int:
|
def get_user_credits(self, user_id: UUID) -> int:
|
||||||
"""
|
"""
|
||||||
Get user remaining credits
|
Get user remaining credits
|
||||||
|
@ -7,3 +7,14 @@
|
|||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
gap: Spacings.$spacing05;
|
gap: Spacings.$spacing05;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.modal_wrapper {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: Spacings.$spacing05;
|
||||||
|
|
||||||
|
.buttons {
|
||||||
|
display: flex;
|
||||||
|
justify-content: space-between;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -1,7 +1,9 @@
|
|||||||
"use client";
|
"use client";
|
||||||
|
|
||||||
|
import { useState } from "react";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
|
|
||||||
|
import { useUserApi } from "@/lib/api/user/useUserApi";
|
||||||
import PageHeader from "@/lib/components/PageHeader/PageHeader";
|
import PageHeader from "@/lib/components/PageHeader/PageHeader";
|
||||||
import { Modal } from "@/lib/components/ui/Modal/Modal";
|
import { Modal } from "@/lib/components/ui/Modal/Modal";
|
||||||
import QuivrButton from "@/lib/components/ui/QuivrButton/QuivrButton";
|
import QuivrButton from "@/lib/components/ui/QuivrButton/QuivrButton";
|
||||||
@ -18,7 +20,10 @@ import { useLogoutModal } from "../../lib/hooks/useLogoutModal";
|
|||||||
const UserPage = (): JSX.Element => {
|
const UserPage = (): JSX.Element => {
|
||||||
const { session } = useSupabase();
|
const { session } = useSupabase();
|
||||||
const { userData } = useUserData();
|
const { userData } = useUserData();
|
||||||
|
const { deleteUserData } = useUserApi();
|
||||||
const { t } = useTranslation(["translation", "logout"]);
|
const { t } = useTranslation(["translation", "logout"]);
|
||||||
|
const [deleteAccountModalOpened, setDeleteAccountModalOpened] =
|
||||||
|
useState(false);
|
||||||
const {
|
const {
|
||||||
handleLogout,
|
handleLogout,
|
||||||
isLoggingOut,
|
isLoggingOut,
|
||||||
@ -26,14 +31,24 @@ const UserPage = (): JSX.Element => {
|
|||||||
setIsLogoutModalOpened,
|
setIsLogoutModalOpened,
|
||||||
} = useLogoutModal();
|
} = useLogoutModal();
|
||||||
|
|
||||||
const button: ButtonType = {
|
const buttons: ButtonType[] = [
|
||||||
label: "Logout",
|
{
|
||||||
color: "dangerous",
|
label: "Logout",
|
||||||
onClick: () => {
|
color: "dangerous",
|
||||||
setIsLogoutModalOpened(true);
|
onClick: () => {
|
||||||
|
setIsLogoutModalOpened(true);
|
||||||
|
},
|
||||||
|
iconName: "logout",
|
||||||
},
|
},
|
||||||
iconName: "logout",
|
{
|
||||||
};
|
label: "Delete Account",
|
||||||
|
color: "dangerous",
|
||||||
|
onClick: () => {
|
||||||
|
setDeleteAccountModalOpened(true);
|
||||||
|
},
|
||||||
|
iconName: "delete",
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
if (!session || !userData) {
|
if (!session || !userData) {
|
||||||
redirectToLogin();
|
redirectToLogin();
|
||||||
@ -42,7 +57,7 @@ const UserPage = (): JSX.Element => {
|
|||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<div className={styles.page_header}>
|
<div className={styles.page_header}>
|
||||||
<PageHeader iconName="user" label="Profile" buttons={[button]} />
|
<PageHeader iconName="user" label="Profile" buttons={buttons} />
|
||||||
</div>
|
</div>
|
||||||
<div className={styles.user_page_container}>
|
<div className={styles.user_page_container}>
|
||||||
<div className={styles.content_wrapper}>
|
<div className={styles.content_wrapper}>
|
||||||
@ -55,11 +70,9 @@ const UserPage = (): JSX.Element => {
|
|||||||
size="auto"
|
size="auto"
|
||||||
CloseTrigger={<div />}
|
CloseTrigger={<div />}
|
||||||
>
|
>
|
||||||
<div className="text-center flex flex-col items-center gap-5">
|
<div className={styles.modal_wrapper}>
|
||||||
<h2 className="text-lg font-medium mb-5">
|
<h2>{t("areYouSure", { ns: "logout" })}</h2>
|
||||||
{t("areYouSure", { ns: "logout" })}
|
<div className={styles.buttons}>
|
||||||
</h2>
|
|
||||||
<div className="flex gap-5 items-center justify-center">
|
|
||||||
<QuivrButton
|
<QuivrButton
|
||||||
onClick={() => setIsLogoutModalOpened(false)}
|
onClick={() => setIsLogoutModalOpened(false)}
|
||||||
color="primary"
|
color="primary"
|
||||||
@ -76,6 +89,34 @@ const UserPage = (): JSX.Element => {
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</Modal>
|
</Modal>
|
||||||
|
<Modal
|
||||||
|
isOpen={deleteAccountModalOpened}
|
||||||
|
setOpen={setDeleteAccountModalOpened}
|
||||||
|
size="auto"
|
||||||
|
CloseTrigger={<div />}
|
||||||
|
>
|
||||||
|
<div className={styles.modal_wrapper}>
|
||||||
|
<h2>Are you sure you want to delete your account ?</h2>
|
||||||
|
<div className={styles.buttons}>
|
||||||
|
<QuivrButton
|
||||||
|
onClick={() => setDeleteAccountModalOpened(false)}
|
||||||
|
color="primary"
|
||||||
|
label={t("cancel", { ns: "logout" })}
|
||||||
|
iconName="close"
|
||||||
|
></QuivrButton>
|
||||||
|
<QuivrButton
|
||||||
|
isLoading={isLoggingOut}
|
||||||
|
color="dangerous"
|
||||||
|
onClick={() => {
|
||||||
|
void deleteUserData();
|
||||||
|
void handleLogout();
|
||||||
|
}}
|
||||||
|
label="Delete Account"
|
||||||
|
iconName="logout"
|
||||||
|
></QuivrButton>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</Modal>
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
@ -1,6 +1,7 @@
|
|||||||
import { useAxios } from "@/lib/hooks";
|
import { useAxios } from "@/lib/hooks";
|
||||||
|
|
||||||
import {
|
import {
|
||||||
|
deleteUserData,
|
||||||
getUser,
|
getUser,
|
||||||
getUserCredits,
|
getUserCredits,
|
||||||
getUserIdentity,
|
getUserIdentity,
|
||||||
@ -18,6 +19,7 @@ export const useUserApi = () => {
|
|||||||
) => updateUserIdentity(userIdentityUpdatableProperties, axiosInstance),
|
) => updateUserIdentity(userIdentityUpdatableProperties, axiosInstance),
|
||||||
getUserIdentity: async () => getUserIdentity(axiosInstance),
|
getUserIdentity: async () => getUserIdentity(axiosInstance),
|
||||||
getUser: async () => getUser(axiosInstance),
|
getUser: async () => getUser(axiosInstance),
|
||||||
|
deleteUserData: async () => deleteUserData(axiosInstance),
|
||||||
getUserCredits: async () => getUserCredits(axiosInstance),
|
getUserCredits: async () => getUserCredits(axiosInstance),
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
@ -31,7 +31,7 @@ export type UserIdentityUpdatableProperties = {
|
|||||||
};
|
};
|
||||||
|
|
||||||
export type UserIdentity = {
|
export type UserIdentity = {
|
||||||
user_id: UUID;
|
id: UUID;
|
||||||
onboarded: boolean;
|
onboarded: boolean;
|
||||||
username: string;
|
username: string;
|
||||||
};
|
};
|
||||||
@ -54,6 +54,12 @@ export const getUser = async (
|
|||||||
axiosInstance: AxiosInstance
|
axiosInstance: AxiosInstance
|
||||||
): Promise<UserStats> => (await axiosInstance.get<UserStats>("/user")).data;
|
): Promise<UserStats> => (await axiosInstance.get<UserStats>("/user")).data;
|
||||||
|
|
||||||
|
export const deleteUserData = async (
|
||||||
|
axiosInstance: AxiosInstance
|
||||||
|
): Promise<void> => {
|
||||||
|
await axiosInstance.delete(`/user_data`);
|
||||||
|
};
|
||||||
|
|
||||||
export const getUserCredits = async (
|
export const getUserCredits = async (
|
||||||
axiosInstance: AxiosInstance
|
axiosInstance: AxiosInstance
|
||||||
): Promise<number> => (await axiosInstance.get<number>("/user/credits")).data;
|
): Promise<number> => (await axiosInstance.get<number>("/user/credits")).data;
|
||||||
|
@ -30,10 +30,6 @@ export const useLogoutModal = () => {
|
|||||||
text: t("error", { errorMessage: error.message, ns: "logout" }),
|
text: t("error", { errorMessage: error.message, ns: "logout" }),
|
||||||
});
|
});
|
||||||
} else {
|
} else {
|
||||||
publish({
|
|
||||||
variant: "success",
|
|
||||||
text: t("loggedOut", { ns: "logout" }),
|
|
||||||
});
|
|
||||||
window.location.href = "/";
|
window.location.href = "/";
|
||||||
}
|
}
|
||||||
setIsLoggingOut(false);
|
setIsLoggingOut(false);
|
||||||
|
Loading…
Reference in New Issue
Block a user