feat: 添加注销账号功能
- 在个人资料页面添加注销账号按钮 - 需要输入用户名确认才能删除 - 删除所有用户数据:牌组、卡片、笔记、关注等 - 添加 8 种语言翻译
This commit is contained in:
@@ -422,6 +422,21 @@
|
|||||||
"notSet": "Nicht festgelegt",
|
"notSet": "Nicht festgelegt",
|
||||||
"memberSince": "Mitglied seit",
|
"memberSince": "Mitglied seit",
|
||||||
"logout": "Abmelden",
|
"logout": "Abmelden",
|
||||||
|
"deleteAccount": {
|
||||||
|
"button": "Konto löschen",
|
||||||
|
"title": "Konto löschen",
|
||||||
|
"warning": "Diese Aktion ist unwiderruflich. Alle Ihre Daten werden dauerhaft gelöscht.",
|
||||||
|
"warningDecks": "Alle Ihre Decks und Karten",
|
||||||
|
"warningCards": "All Ihr Lernfortschritt",
|
||||||
|
"warningHistory": "All Ihr Übersetzungs- und Wörterbuchverlauf",
|
||||||
|
"warningPermanent": "Diese Aktion kann nicht rückgängig gemacht werden",
|
||||||
|
"confirmLabel": "Geben Sie Ihren Benutzernamen zur Bestätigung ein:",
|
||||||
|
"usernameMismatch": "Benutzername stimmt nicht überein",
|
||||||
|
"cancel": "Abbrechen",
|
||||||
|
"confirm": "Mein Konto löschen",
|
||||||
|
"success": "Konto erfolgreich gelöscht",
|
||||||
|
"failed": "Konto konnte nicht gelöscht werden"
|
||||||
|
},
|
||||||
"folders": {
|
"folders": {
|
||||||
"title": "Decks",
|
"title": "Decks",
|
||||||
"noFolders": "Noch keine Decks",
|
"noFolders": "Noch keine Decks",
|
||||||
|
|||||||
@@ -428,6 +428,21 @@
|
|||||||
"memberSince": "Member Since",
|
"memberSince": "Member Since",
|
||||||
"joined": "Joined",
|
"joined": "Joined",
|
||||||
"logout": "Logout",
|
"logout": "Logout",
|
||||||
|
"deleteAccount": {
|
||||||
|
"button": "Delete Account",
|
||||||
|
"title": "Delete Account",
|
||||||
|
"warning": "This action is irreversible. All your data will be permanently deleted.",
|
||||||
|
"warningDecks": "All your decks and cards",
|
||||||
|
"warningCards": "All your learning progress",
|
||||||
|
"warningHistory": "All your translation and dictionary history",
|
||||||
|
"warningPermanent": "This action cannot be undone",
|
||||||
|
"confirmLabel": "Type your username to confirm:",
|
||||||
|
"usernameMismatch": "Username does not match",
|
||||||
|
"cancel": "Cancel",
|
||||||
|
"confirm": "Delete My Account",
|
||||||
|
"success": "Account deleted successfully",
|
||||||
|
"failed": "Failed to delete account"
|
||||||
|
},
|
||||||
"decks": {
|
"decks": {
|
||||||
"title": "Decks",
|
"title": "Decks",
|
||||||
"noDecks": "No decks yet",
|
"noDecks": "No decks yet",
|
||||||
|
|||||||
@@ -422,6 +422,21 @@
|
|||||||
"notSet": "Non défini",
|
"notSet": "Non défini",
|
||||||
"memberSince": "Membre depuis",
|
"memberSince": "Membre depuis",
|
||||||
"logout": "Déconnexion",
|
"logout": "Déconnexion",
|
||||||
|
"deleteAccount": {
|
||||||
|
"button": "Supprimer le compte",
|
||||||
|
"title": "Supprimer le compte",
|
||||||
|
"warning": "Cette action est irréversible. Toutes vos données seront définitivement supprimées.",
|
||||||
|
"warningDecks": "Tous vos decks et cartes",
|
||||||
|
"warningCards": "Tout votre progression d'apprentissage",
|
||||||
|
"warningHistory": "Tout votre historique de traduction et de dictionnaire",
|
||||||
|
"warningPermanent": "Cette action ne peut pas être annulée",
|
||||||
|
"confirmLabel": "Tapez votre nom d'utilisateur pour confirmer :",
|
||||||
|
"usernameMismatch": "Le nom d'utilisateur ne correspond pas",
|
||||||
|
"cancel": "Annuler",
|
||||||
|
"confirm": "Supprimer mon compte",
|
||||||
|
"success": "Compte supprimé avec succès",
|
||||||
|
"failed": "Échec de la suppression du compte"
|
||||||
|
},
|
||||||
"decks": {
|
"decks": {
|
||||||
"title": "Decks",
|
"title": "Decks",
|
||||||
"noDecks": "Pas encore de decks",
|
"noDecks": "Pas encore de decks",
|
||||||
|
|||||||
@@ -422,6 +422,21 @@
|
|||||||
"notSet": "Non Impostato",
|
"notSet": "Non Impostato",
|
||||||
"memberSince": "Membro Dal",
|
"memberSince": "Membro Dal",
|
||||||
"logout": "Esci",
|
"logout": "Esci",
|
||||||
|
"deleteAccount": {
|
||||||
|
"button": "Elimina Account",
|
||||||
|
"title": "Elimina Account",
|
||||||
|
"warning": "Questa azione è irreversibile. Tutti i tuoi dati saranno eliminati definitivamente.",
|
||||||
|
"warningDecks": "Tutti i tuoi mazzi e le tue carte",
|
||||||
|
"warningCards": "Tutto il tuo progresso di apprendimento",
|
||||||
|
"warningHistory": "Tutto il tuo cronologia di traduzione e dizionario",
|
||||||
|
"warningPermanent": "Questa azione non può essere annullata",
|
||||||
|
"confirmLabel": "Digita il tuo nome utente per confermare:",
|
||||||
|
"usernameMismatch": "Il nome utente non corrisponde",
|
||||||
|
"cancel": "Annulla",
|
||||||
|
"confirm": "Elimina il mio account",
|
||||||
|
"success": "Account eliminato con successo",
|
||||||
|
"failed": "Impossibile eliminare l'account"
|
||||||
|
},
|
||||||
"decks": {
|
"decks": {
|
||||||
"title": "Mazzi",
|
"title": "Mazzi",
|
||||||
"noDecks": "Nessun mazzo ancora",
|
"noDecks": "Nessun mazzo ancora",
|
||||||
|
|||||||
@@ -413,6 +413,21 @@
|
|||||||
"notSet": "未設定",
|
"notSet": "未設定",
|
||||||
"memberSince": "登録日",
|
"memberSince": "登録日",
|
||||||
"logout": "ログアウト",
|
"logout": "ログアウト",
|
||||||
|
"deleteAccount": {
|
||||||
|
"button": "アカウント削除",
|
||||||
|
"title": "アカウント削除",
|
||||||
|
"warning": "この操作は取り消せません。すべてのデータが完全に削除されます。",
|
||||||
|
"warningDecks": "すべてのデッキとカード",
|
||||||
|
"warningCards": "すべての学習履歴",
|
||||||
|
"warningHistory": "すべての翻訳と辞書の履歴",
|
||||||
|
"warningPermanent": "この操作は取り消せません",
|
||||||
|
"confirmLabel": "確認のためユーザー名を入力してください:",
|
||||||
|
"usernameMismatch": "ユーザー名が一致しません",
|
||||||
|
"cancel": "キャンセル",
|
||||||
|
"confirm": "アカウントを削除する",
|
||||||
|
"success": "アカウントが正常に削除されました",
|
||||||
|
"failed": "アカウントの削除に失敗しました"
|
||||||
|
},
|
||||||
"decks": {
|
"decks": {
|
||||||
"title": "デッキ",
|
"title": "デッキ",
|
||||||
"noDecks": "まだデッキがありません",
|
"noDecks": "まだデッキがありません",
|
||||||
|
|||||||
@@ -422,6 +422,21 @@
|
|||||||
"notSet": "설정되지 않음",
|
"notSet": "설정되지 않음",
|
||||||
"memberSince": "가입일",
|
"memberSince": "가입일",
|
||||||
"logout": "로그아웃",
|
"logout": "로그아웃",
|
||||||
|
"deleteAccount": {
|
||||||
|
"button": "계정 삭제",
|
||||||
|
"title": "계정 삭제",
|
||||||
|
"warning": "이 작업은 되돌릴 수 없습니다. 모든 데이터가 영구적으로 삭제됩니다.",
|
||||||
|
"warningDecks": "모든 덱과 카드",
|
||||||
|
"warningCards": "모든 학습 진행 상황",
|
||||||
|
"warningHistory": "모든 번역 및 사전 기록",
|
||||||
|
"warningPermanent": "이 작업은 취소할 수 없습니다",
|
||||||
|
"confirmLabel": "확인을 위해 사용자명을 입력하세요:",
|
||||||
|
"usernameMismatch": "사용자명이 일치하지 않습니다",
|
||||||
|
"cancel": "취소",
|
||||||
|
"confirm": "내 계정 삭제",
|
||||||
|
"success": "계정이 성공적으로 삭제되었습니다",
|
||||||
|
"failed": "계정 삭제에 실패했습니다"
|
||||||
|
},
|
||||||
"folders": {
|
"folders": {
|
||||||
"title": "덱",
|
"title": "덱",
|
||||||
"noFolders": "아직 덱이 없습니다",
|
"noFolders": "아직 덱이 없습니다",
|
||||||
|
|||||||
@@ -422,6 +422,21 @@
|
|||||||
"notSet": "تەڭشەلمىگەن",
|
"notSet": "تەڭشەلمىگەن",
|
||||||
"memberSince": "ئەزا بولغاندىن بېرى",
|
"memberSince": "ئەزا بولغاندىن بېرى",
|
||||||
"logout": "چىكىنىش",
|
"logout": "چىكىنىش",
|
||||||
|
"deleteAccount": {
|
||||||
|
"button": "ھېساباتنى ئۆچۈرۈش",
|
||||||
|
"title": "ھېساباتنى ئۆچۈرۈش",
|
||||||
|
"warning": "بۇ مەشغۇلاتنى ئەسلىگە قايتۇرغىلى بولمايدۇ. بارلىق سانلىق مەلۇماتلىرىڭىز مەڭگۈلۈك ئۆچۈرۈلىدۇ.",
|
||||||
|
"warningDecks": "بارلىق دېك ۋە كارتلىرىڭىز",
|
||||||
|
"warningCards": "بارلىق ئۆگىنىش ئىلگىرىلەشلىرىڭىز",
|
||||||
|
"warningHistory": "بارلىق تەرجىمە ۋە لۇغەت تارىخىڭىز",
|
||||||
|
"warningPermanent": "بۇ مەشغۇلاتنى بىكار قىلغىلى بولمايدۇ",
|
||||||
|
"confirmLabel": "جەزىملەش ئۈچۈن ئىشلەتكۈچى ئاتىڭىزنى كىرگۈزۈڭ:",
|
||||||
|
"usernameMismatch": "ئىشلەتكۈچى ئاتى ماس كەلمەيدۇ",
|
||||||
|
"cancel": "بىكار قىلىش",
|
||||||
|
"confirm": "ھېساباتىمنى ئۆچۈرۈش",
|
||||||
|
"success": "ھېسابات مۇۋەپپەقىيەتلىك ئۆچۈرۈلدى",
|
||||||
|
"failed": "ھېساباتنى ئۆچۈرۈش مەغلۇپ بولدى"
|
||||||
|
},
|
||||||
"decks": {
|
"decks": {
|
||||||
"title": "دېكلار",
|
"title": "دېكلار",
|
||||||
"noDecks": "تېخى دېك يوق",
|
"noDecks": "تېخى دېك يوق",
|
||||||
|
|||||||
@@ -428,6 +428,21 @@
|
|||||||
"memberSince": "注册时间",
|
"memberSince": "注册时间",
|
||||||
"joined": "加入于",
|
"joined": "加入于",
|
||||||
"logout": "登出",
|
"logout": "登出",
|
||||||
|
"deleteAccount": {
|
||||||
|
"button": "注销账号",
|
||||||
|
"title": "注销账号",
|
||||||
|
"warning": "此操作不可逆,您的所有数据将被永久删除。",
|
||||||
|
"warningDecks": "您的所有牌组和卡片",
|
||||||
|
"warningCards": "您的所有学习进度",
|
||||||
|
"warningHistory": "您的所有翻译和词典历史",
|
||||||
|
"warningPermanent": "此操作无法撤销",
|
||||||
|
"confirmLabel": "输入您的用户名以确认:",
|
||||||
|
"usernameMismatch": "用户名不匹配",
|
||||||
|
"cancel": "取消",
|
||||||
|
"confirm": "注销我的账号",
|
||||||
|
"success": "账号已成功注销",
|
||||||
|
"failed": "注销账号失败"
|
||||||
|
},
|
||||||
"decks": {
|
"decks": {
|
||||||
"title": "牌组",
|
"title": "牌组",
|
||||||
"noDecks": "还没有牌组",
|
"noDecks": "还没有牌组",
|
||||||
|
|||||||
100
src/app/(auth)/users/[username]/DeleteAccountButton.tsx
Normal file
100
src/app/(auth)/users/[username]/DeleteAccountButton.tsx
Normal file
@@ -0,0 +1,100 @@
|
|||||||
|
"use client";
|
||||||
|
|
||||||
|
import { useState } from "react";
|
||||||
|
import { useRouter } from "next/navigation";
|
||||||
|
import { useTranslations } from "next-intl";
|
||||||
|
import { toast } from "sonner";
|
||||||
|
import { Button } from "@/design-system/base/button";
|
||||||
|
import { Modal } from "@/design-system/overlay/modal";
|
||||||
|
import { actionDeleteAccount } from "@/modules/auth/auth-action";
|
||||||
|
|
||||||
|
interface DeleteAccountButtonProps {
|
||||||
|
username: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function DeleteAccountButton({ username }: DeleteAccountButtonProps) {
|
||||||
|
const t = useTranslations("user_profile");
|
||||||
|
const router = useRouter();
|
||||||
|
const [showModal, setShowModal] = useState(false);
|
||||||
|
const [confirmUsername, setConfirmUsername] = useState("");
|
||||||
|
const [loading, setLoading] = useState(false);
|
||||||
|
|
||||||
|
const handleDelete = async () => {
|
||||||
|
if (confirmUsername !== username) {
|
||||||
|
toast.error(t("deleteAccount.usernameMismatch"));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
setLoading(true);
|
||||||
|
try {
|
||||||
|
const result = await actionDeleteAccount();
|
||||||
|
if (result.success) {
|
||||||
|
toast.success(t("deleteAccount.success"));
|
||||||
|
router.push("/");
|
||||||
|
} else {
|
||||||
|
toast.error(result.message || t("deleteAccount.failed"));
|
||||||
|
}
|
||||||
|
} catch {
|
||||||
|
toast.error(t("deleteAccount.failed"));
|
||||||
|
} finally {
|
||||||
|
setLoading(false);
|
||||||
|
setShowModal(false);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<Button variant="error" onClick={() => setShowModal(true)}>
|
||||||
|
{t("deleteAccount.button")}
|
||||||
|
</Button>
|
||||||
|
|
||||||
|
<Modal open={showModal} onClose={() => setShowModal(false)}>
|
||||||
|
<div className="p-6">
|
||||||
|
<h2 className="text-xl font-bold text-red-600 mb-4">
|
||||||
|
{t("deleteAccount.title")}
|
||||||
|
</h2>
|
||||||
|
|
||||||
|
<div className="space-y-4">
|
||||||
|
<p className="text-gray-700">
|
||||||
|
{t("deleteAccount.warning")}
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<ul className="list-disc list-inside text-gray-600 text-sm space-y-1">
|
||||||
|
<li>{t("deleteAccount.warningDecks")}</li>
|
||||||
|
<li>{t("deleteAccount.warningCards")}</li>
|
||||||
|
<li>{t("deleteAccount.warningHistory")}</li>
|
||||||
|
<li>{t("deleteAccount.warningPermanent")}</li>
|
||||||
|
</ul>
|
||||||
|
|
||||||
|
<div className="mt-4">
|
||||||
|
<label className="block text-sm font-medium text-gray-700 mb-2">
|
||||||
|
{t("deleteAccount.confirmLabel")} <span className="font-mono font-bold">{username}</span>
|
||||||
|
</label>
|
||||||
|
<input
|
||||||
|
type="text"
|
||||||
|
value={confirmUsername}
|
||||||
|
onChange={(e) => setConfirmUsername(e.target.value)}
|
||||||
|
className="w-full px-3 py-2 border border-gray-300 rounded-md focus:outline-none focus:ring-2 focus:ring-red-500"
|
||||||
|
placeholder={username}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="flex justify-end gap-3 mt-6">
|
||||||
|
<Button variant="secondary" onClick={() => setShowModal(false)}>
|
||||||
|
{t("deleteAccount.cancel")}
|
||||||
|
</Button>
|
||||||
|
<Button
|
||||||
|
variant="error"
|
||||||
|
onClick={handleDelete}
|
||||||
|
loading={loading}
|
||||||
|
disabled={confirmUsername !== username}
|
||||||
|
>
|
||||||
|
{t("deleteAccount.confirm")}
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</Modal>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -10,6 +10,7 @@ import { getTranslations } from "next-intl/server";
|
|||||||
import { auth } from "@/auth";
|
import { auth } from "@/auth";
|
||||||
import { headers } from "next/headers";
|
import { headers } from "next/headers";
|
||||||
import { FollowStats } from "@/components/follow/FollowStats";
|
import { FollowStats } from "@/components/follow/FollowStats";
|
||||||
|
import { DeleteAccountButton } from "./DeleteAccountButton";
|
||||||
|
|
||||||
interface UserPageProps {
|
interface UserPageProps {
|
||||||
params: Promise<{ username: string; }>;
|
params: Promise<{ username: string; }>;
|
||||||
@@ -45,7 +46,14 @@ export default async function UserPage({ params }: UserPageProps) {
|
|||||||
<div className="bg-white rounded-lg shadow-md p-6 mb-6">
|
<div className="bg-white rounded-lg shadow-md p-6 mb-6">
|
||||||
<div className="flex items-center justify-between mb-4">
|
<div className="flex items-center justify-between mb-4">
|
||||||
<div></div>
|
<div></div>
|
||||||
{isOwnProfile && <LinkButton href="/logout">{t("logout")}</LinkButton>}
|
<div className="flex items-center gap-3">
|
||||||
|
{isOwnProfile && (
|
||||||
|
<>
|
||||||
|
<LinkButton href="/logout">{t("logout")}</LinkButton>
|
||||||
|
<DeleteAccountButton username={username} />
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div className="flex flex-col sm:flex-row items-start sm:items-center space-y-4 sm:space-y-0 sm:space-x-6">
|
<div className="flex flex-col sm:flex-row items-start sm:items-center space-y-4 sm:space-y-0 sm:space-x-6">
|
||||||
{user.image ? (
|
{user.image ? (
|
||||||
|
|||||||
@@ -61,3 +61,8 @@ export type ActionOutputUserProfile = {
|
|||||||
updatedAt: Date;
|
updatedAt: Date;
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export type ActionOutputDeleteAccount = {
|
||||||
|
success: boolean;
|
||||||
|
message: string;
|
||||||
|
};
|
||||||
|
|||||||
@@ -5,11 +5,13 @@ import { headers } from "next/headers";
|
|||||||
import { redirect } from "next/navigation";
|
import { redirect } from "next/navigation";
|
||||||
import { ValidateError } from "@/lib/errors";
|
import { ValidateError } from "@/lib/errors";
|
||||||
import { createLogger } from "@/lib/logger";
|
import { createLogger } from "@/lib/logger";
|
||||||
|
import { prisma } from "@/lib/db";
|
||||||
import {
|
import {
|
||||||
ActionInputGetUserProfileByUsername,
|
ActionInputGetUserProfileByUsername,
|
||||||
ActionInputSignIn,
|
ActionInputSignIn,
|
||||||
ActionInputSignUp,
|
ActionInputSignUp,
|
||||||
ActionOutputAuth,
|
ActionOutputAuth,
|
||||||
|
ActionOutputDeleteAccount,
|
||||||
ActionOutputUserProfile,
|
ActionOutputUserProfile,
|
||||||
validateActionInputGetUserProfileByUsername,
|
validateActionInputGetUserProfileByUsername,
|
||||||
validateActionInputSignIn,
|
validateActionInputSignIn,
|
||||||
@@ -180,3 +182,91 @@ export async function actionGetUserProfileByUsername(dto: ActionInputGetUserProf
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Delete account action
|
||||||
|
* Permanently deletes the current user and all associated data
|
||||||
|
*/
|
||||||
|
export async function actionDeleteAccount(): Promise<ActionOutputDeleteAccount> {
|
||||||
|
try {
|
||||||
|
const session = await auth.api.getSession({ headers: await headers() });
|
||||||
|
if (!session?.user?.id) {
|
||||||
|
return { success: false, message: "Unauthorized" };
|
||||||
|
}
|
||||||
|
|
||||||
|
const userId = session.user.id;
|
||||||
|
|
||||||
|
await prisma.$transaction(async (tx) => {
|
||||||
|
// Delete in correct order to avoid foreign key constraints
|
||||||
|
// 1. Revlogs (depend on cards)
|
||||||
|
await tx.revlog.deleteMany({
|
||||||
|
where: { card: { note: { userId } } }
|
||||||
|
});
|
||||||
|
|
||||||
|
// 2. Cards (depend on notes and decks)
|
||||||
|
await tx.card.deleteMany({
|
||||||
|
where: { note: { userId } }
|
||||||
|
});
|
||||||
|
|
||||||
|
// 3. Notes (depend on note types)
|
||||||
|
await tx.note.deleteMany({
|
||||||
|
where: { userId }
|
||||||
|
});
|
||||||
|
|
||||||
|
// 4. Note types
|
||||||
|
await tx.noteType.deleteMany({
|
||||||
|
where: { userId }
|
||||||
|
});
|
||||||
|
|
||||||
|
// 5. Deck favorites
|
||||||
|
await tx.deckFavorite.deleteMany({
|
||||||
|
where: { userId }
|
||||||
|
});
|
||||||
|
|
||||||
|
// 6. Decks
|
||||||
|
await tx.deck.deleteMany({
|
||||||
|
where: { userId }
|
||||||
|
});
|
||||||
|
|
||||||
|
// 7. Follows (both as follower and following)
|
||||||
|
await tx.follow.deleteMany({
|
||||||
|
where: {
|
||||||
|
OR: [
|
||||||
|
{ followerId: userId },
|
||||||
|
{ followingId: userId }
|
||||||
|
]
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// 8. Dictionary lookups
|
||||||
|
await tx.dictionaryLookUp.deleteMany({
|
||||||
|
where: { userId }
|
||||||
|
});
|
||||||
|
|
||||||
|
// 9. Translation history
|
||||||
|
await tx.translationHistory.deleteMany({
|
||||||
|
where: { userId }
|
||||||
|
});
|
||||||
|
|
||||||
|
// 10. Sessions
|
||||||
|
await tx.session.deleteMany({
|
||||||
|
where: { userId }
|
||||||
|
});
|
||||||
|
|
||||||
|
// 11. Accounts
|
||||||
|
await tx.account.deleteMany({
|
||||||
|
where: { userId }
|
||||||
|
});
|
||||||
|
|
||||||
|
// 12. Finally, delete the user
|
||||||
|
await tx.user.delete({
|
||||||
|
where: { id: userId }
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
return { success: true, message: "Account deleted successfully" };
|
||||||
|
} catch (e) {
|
||||||
|
log.error("Delete account failed", { error: e });
|
||||||
|
return { success: false, message: "Failed to delete account" };
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user