68f82d79c2
KVKK 'veri taşınabilirliği' ve 'unutulma hakkı' için iki uçtan yeni
akış: dışa aktar ve kalıcı sil. İkisi de /settings/workspace altındaki
yeni 'Tehlikeli Bölge' kartına eklendi, sadece owner görür.
Export — GET /api/account/export
- requireTenant guard.
- Sırayla tenant-owned tablolar (settings, profiles, connections,
patients, clinic_pricing, jobs, job_files, history, prosthetics,
finance_entries, payments, notifications, audit_logs) gezilir.
Çift sahipli tablolar (jobs, connections, payments, clinic_pricing,
job_files, history) için iki kez sorgu atılıp $id ile dedupe edilir.
- { exportedAt, tenantId, tenantKind, requestedBy, data } JSON olarak
Content-Disposition: attachment ile döner. ~ek dep yok.
Delete — deleteWorkspaceAction (owner-only, server action)
- Onay: kullanıcının companyName'i birebir yazması gerekir.
- 1) Storage: tenant logosu + arşivlenmemiş job_files objelerini sil.
- 2) DB: hard-delete tüm tenant verisi — notifications, audit_logs,
payments, finance_entries, history, job_files, jobs, clinic_pricing,
prosthetics, patients, connections, profiles, tenant_settings.
Her tablo için Query.equal + sayfa sayfa 500'er silme.
- 3) teams.delete(tenantId) — son aşama, takım yok artık.
- 4) active-tenant cookie temizle; session bırak (kullanıcı başka
workspace'e geçebilir veya çıkış yapabilir).
- redirect('/onboarding').
UI — DangerZone (client)
- 'JSON indir' butonu: /api/account/export'a fetch, blob download.
- 'Sil' Dialog: çalışma alanı adı confirm input ile gated; eşit
olana kadar 'Kalıcı olarak sil' disabled.
142 lines
5.2 KiB
TypeScript
142 lines
5.2 KiB
TypeScript
"use client";
|
||
|
||
import { useActionState, useEffect, useState, useTransition } from "react";
|
||
import { AlertTriangle, Download, Loader2, Trash2 } from "lucide-react";
|
||
import { toast } from "sonner";
|
||
|
||
import { Button } from "@/components/ui/button";
|
||
import { Card, CardContent, CardDescription, CardHeader, CardTitle } from "@/components/ui/card";
|
||
import {
|
||
Dialog,
|
||
DialogClose,
|
||
DialogContent,
|
||
DialogDescription,
|
||
DialogFooter,
|
||
DialogHeader,
|
||
DialogTitle,
|
||
} from "@/components/ui/dialog";
|
||
import { Input } from "@/components/ui/input";
|
||
import { Label } from "@/components/ui/label";
|
||
import {
|
||
deleteWorkspaceAction,
|
||
initialDeleteWorkspaceState,
|
||
} from "@/lib/appwrite/account-delete-actions";
|
||
|
||
export function DangerZone({ companyName }: { companyName: string }) {
|
||
const [downloadBusy, startDownload] = useTransition();
|
||
const [open, setOpen] = useState(false);
|
||
const [confirm, setConfirm] = useState("");
|
||
const [state, action, pending] = useActionState(
|
||
deleteWorkspaceAction,
|
||
initialDeleteWorkspaceState,
|
||
);
|
||
|
||
useEffect(() => {
|
||
if (state.error) toast.error(state.error);
|
||
}, [state]);
|
||
|
||
function downloadExport() {
|
||
startDownload(async () => {
|
||
try {
|
||
const res = await fetch("/api/account/export");
|
||
if (!res.ok) throw new Error(`HTTP ${res.status}`);
|
||
const blob = await res.blob();
|
||
const url = URL.createObjectURL(blob);
|
||
const a = document.createElement("a");
|
||
a.href = url;
|
||
a.download = `dls-export-${Date.now()}.json`;
|
||
document.body.appendChild(a);
|
||
a.click();
|
||
a.remove();
|
||
URL.revokeObjectURL(url);
|
||
toast.success("Veri dışa aktarıldı.");
|
||
} catch (e) {
|
||
toast.error(e instanceof Error ? e.message : "İndirilemedi.");
|
||
}
|
||
});
|
||
}
|
||
|
||
return (
|
||
<Card className="border-destructive/40">
|
||
<CardHeader>
|
||
<CardTitle className="flex items-center gap-2 text-base">
|
||
<AlertTriangle className="text-destructive size-4" />
|
||
Tehlikeli Bölge
|
||
</CardTitle>
|
||
<CardDescription>
|
||
Verinizi dışa aktarın veya çalışma alanını kalıcı olarak silin.
|
||
</CardDescription>
|
||
</CardHeader>
|
||
<CardContent className="grid gap-3">
|
||
<div className="flex flex-wrap items-center justify-between gap-3 rounded-md border p-3">
|
||
<div className="min-w-0">
|
||
<p className="font-medium">Verilerimi indir</p>
|
||
<p className="text-muted-foreground text-xs">
|
||
Çalışma alanınızdaki tüm veriler (hastalar, işler, ödemeler, geçmiş)
|
||
JSON formatında dışa aktarılır. Silmeden önce yedek almanız önerilir.
|
||
</p>
|
||
</div>
|
||
<Button variant="outline" onClick={downloadExport} disabled={downloadBusy}>
|
||
{downloadBusy ? <Loader2 className="size-4 animate-spin" /> : <Download className="size-4" />}
|
||
JSON indir
|
||
</Button>
|
||
</div>
|
||
|
||
<div className="border-destructive/40 flex flex-wrap items-center justify-between gap-3 rounded-md border p-3">
|
||
<div className="min-w-0">
|
||
<p className="font-medium">Çalışma alanını sil</p>
|
||
<p className="text-muted-foreground text-xs">
|
||
Tüm hastalar, işler, ödemeler, dosyalar ve geçmiş kalıcı olarak silinir.
|
||
Bu işlem geri alınamaz.
|
||
</p>
|
||
</div>
|
||
<Dialog open={open} onOpenChange={setOpen}>
|
||
<Button variant="destructive" onClick={() => setOpen(true)}>
|
||
<Trash2 className="size-4" />
|
||
Sil
|
||
</Button>
|
||
<DialogContent>
|
||
<DialogHeader>
|
||
<DialogTitle>Çalışma alanını kalıcı sil</DialogTitle>
|
||
<DialogDescription>
|
||
Onaylamak için aşağıya <strong>{companyName}</strong> yazın. Bu
|
||
işlem hastalar, işler, ödemeler, dosyalar ve tüm geçmişi içerir
|
||
ve geri alınamaz.
|
||
</DialogDescription>
|
||
</DialogHeader>
|
||
<form action={action} className="grid gap-3">
|
||
<div className="grid gap-2">
|
||
<Label htmlFor="confirm">Çalışma alanı adı</Label>
|
||
<Input
|
||
id="confirm"
|
||
name="confirm"
|
||
autoComplete="off"
|
||
value={confirm}
|
||
onChange={(e) => setConfirm(e.target.value)}
|
||
placeholder={companyName}
|
||
/>
|
||
</div>
|
||
<DialogFooter>
|
||
<DialogClose asChild>
|
||
<Button type="button" variant="outline">
|
||
Vazgeç
|
||
</Button>
|
||
</DialogClose>
|
||
<Button
|
||
type="submit"
|
||
variant="destructive"
|
||
disabled={pending || confirm.trim() !== companyName}
|
||
>
|
||
{pending ? <Loader2 className="size-4 animate-spin" /> : <Trash2 className="size-4" />}
|
||
Kalıcı olarak sil
|
||
</Button>
|
||
</DialogFooter>
|
||
</form>
|
||
</DialogContent>
|
||
</Dialog>
|
||
</div>
|
||
</CardContent>
|
||
</Card>
|
||
);
|
||
}
|