nothingworry's picture
document deletion and improve tenant ID management
345b8ff
raw
history blame
9.85 kB
"use client";
import { useCallback, useMemo, useState } from "react";
import Link from "next/link";
import { AdminRulesPanel } from "@/components/admin-rules-panel";
import { Footer } from "@/components/footer";
import { useTenant } from "@/contexts/TenantContext";
import { TenantSelector } from "@/components/tenant-selector";
const BACKEND_BASE_URL = process.env.NEXT_PUBLIC_BACKEND_BASE_URL ?? "http://localhost:8000";
type StatusState = { tone: "info" | "success" | "error"; message: string } | null;
export default function AdminRulesPage() {
const { tenantId } = useTenant();
const [rulesInput, setRulesInput] = useState("");
const [deleteInput, setDeleteInput] = useState("");
const [rules, setRules] = useState<string[]>([]);
const [loading, setLoading] = useState(false);
const [status, setStatus] = useState<StatusState>(null);
const headers = useMemo(() => {
if (!tenantId.trim()) return undefined;
return {
"Content-Type": "application/json",
"x-tenant-id": tenantId.trim(),
};
}, [tenantId]);
const requireTenant = useCallback(() => {
if (!tenantId.trim()) {
setStatus({ tone: "error", message: "Enter a tenant ID in the navbar first." });
return false;
}
return true;
}, [tenantId]);
const handleRefresh = useCallback(async () => {
if (!requireTenant()) return;
try {
setLoading(true);
setStatus({ tone: "info", message: "Loading rules..." });
const response = await fetch(`${BACKEND_BASE_URL}/admin/rules`, {
method: "GET",
headers,
});
if (!response.ok) {
throw new Error(`Backend error ${response.status}`);
}
const data = await response.json();
setRules(data.rules ?? []);
setStatus({ tone: "success", message: "Rules synced." });
} catch (error: any) {
setStatus({ tone: "error", message: error.message || "Failed to fetch rules" });
} finally {
setLoading(false);
}
}, [headers, requireTenant]);
const handleUpload = useCallback(async () => {
if (!requireTenant()) return;
const lines = rulesInput
.split("\n")
.map((line) => line.trim())
.filter(Boolean);
if (!lines.length) {
setStatus({ tone: "error", message: "Add at least one rule to upload." });
return;
}
try {
setLoading(true);
setStatus({ tone: "info", message: "Uploading rules..." });
const response = await fetch(`${BACKEND_BASE_URL}/admin/rules/bulk`, {
method: "POST",
headers,
body: JSON.stringify({ rules: lines }),
});
if (!response.ok) {
const details = await response.text();
throw new Error(details || `Backend error ${response.status}`);
}
await handleRefresh();
setRulesInput("");
setStatus({ tone: "success", message: "Rules uploaded successfully." });
} catch (error: any) {
setStatus({ tone: "error", message: error.message || "Failed to upload rules" });
} finally {
setLoading(false);
}
}, [handleRefresh, headers, requireTenant, rulesInput]);
const handleDelete = useCallback(async () => {
if (!requireTenant()) return;
if (!deleteInput.trim()) {
setStatus({ tone: "error", message: "Enter the rule text you want to delete." });
return;
}
try {
setLoading(true);
setStatus({ tone: "info", message: "Deleting rule..." });
const response = await fetch(
`${BACKEND_BASE_URL}/admin/rules/${encodeURIComponent(deleteInput.trim())}`,
{
method: "DELETE",
headers,
}
);
if (!response.ok) {
const details = await response.text();
throw new Error(details || `Backend error ${response.status}`);
}
await handleRefresh();
setDeleteInput("");
setStatus({ tone: "success", message: "Rule deleted." });
} catch (error: any) {
setStatus({ tone: "error", message: error.message || "Failed to delete rule" });
} finally {
setLoading(false);
}
}, [deleteInput, handleRefresh, headers, requireTenant]);
return (
<main className="mx-auto flex min-h-screen max-w-5xl flex-col gap-10 px-4 pb-16 pt-12 sm:px-6 lg:px-8">
<header className="flex flex-col gap-4 rounded-2xl border border-white/10 bg-white/5 px-6 py-6 text-slate-100 shadow-lg shadow-slate-950/40">
<div className="flex items-center justify-between gap-3">
<div className="flex items-center gap-3 text-base font-semibold">
<span className="inline-flex h-10 w-10 items-center justify-center rounded-2xl bg-gradient-to-br from-sky-400 to-cyan-500 text-slate-950">
IC
</span>
IntegraChat · Admin Rule Ingestion
</div>
<div className="flex items-center gap-4">
<TenantSelector />
<Link href="/" className="text-xs font-semibold uppercase tracking-[0.3em] text-cyan-300 hover:text-white">
← Back Home
</Link>
</div>
</div>
<p className="text-sm text-slate-300">
Push governance policies, compliance workflows, and red-flag patterns to the backend&apos;s persistent rules
store using the MCP admin services.
</p>
</header>
<AdminRulesPanel />
<section className="rounded-3xl border border-white/10 bg-white/5 p-6 shadow-2xl shadow-slate-950/40">
<div className="flex flex-col gap-6">
<div className="flex items-center justify-end gap-3">
<button
onClick={handleRefresh}
disabled={loading}
className="flex-1 rounded-full bg-gradient-to-r from-cyan-400 to-blue-500 px-6 py-3 text-sm font-semibold text-slate-950 shadow-lg shadow-cyan-500/30 disabled:opacity-60"
>
Refresh Rules
</button>
</div>
<div className="grid gap-4 lg:grid-cols-2">
<label className="flex flex-col gap-2 text-sm font-semibold text-slate-200">
Bulk Upload Rules (one per line)
<textarea
value={rulesInput}
onChange={(e) => setRulesInput(e.target.value)}
placeholder="Disallow sharing salaries\nBlock deleting production data"
rows={6}
className="rounded-2xl border border-white/10 bg-slate-900/50 px-4 py-3 text-base text-white outline-none ring-0 focus:border-cyan-400"
/>
</label>
<label className="flex flex-col gap-2 text-sm font-semibold text-slate-200">
Delete Rule (exact text match)
<textarea
value={deleteInput}
onChange={(e) => setDeleteInput(e.target.value)}
placeholder="Enter the exact rule text to remove..."
rows={6}
className="rounded-2xl border border-white/10 bg-slate-900/50 px-4 py-3 text-base text-white outline-none ring-0 focus:border-cyan-400"
/>
</label>
</div>
<div className="flex flex-wrap gap-3">
<button
onClick={handleUpload}
disabled={loading}
className="rounded-full bg-gradient-to-r from-emerald-400 to-lime-400 px-6 py-3 text-sm font-semibold text-slate-900 shadow-lg shadow-emerald-500/30 disabled:opacity-60"
>
Upload / Append Rules
</button>
<button
onClick={handleDelete}
disabled={loading}
className="rounded-full border border-rose-500 px-6 py-3 text-sm font-semibold text-rose-300 transition hover:bg-rose-500/10 disabled:opacity-60"
>
Delete Rule
</button>
</div>
{status && (
<div
className={`rounded-2xl border px-4 py-3 text-sm ${
status.tone === "error"
? "border-rose-500/40 bg-rose-500/10 text-rose-200"
: status.tone === "success"
? "border-emerald-500/40 bg-emerald-500/10 text-emerald-200"
: "border-cyan-500/40 bg-cyan-500/10 text-cyan-200"
}`}
>
{status.message}
</div>
)}
<div className="rounded-2xl border border-white/10 bg-slate-950/40">
<div className="flex items-center justify-between border-b border-white/5 px-5 py-3 text-xs uppercase tracking-[0.3em] text-slate-400">
<span>Rule Set</span>
<span>{rules.length} entries</span>
</div>
<div className="overflow-x-auto">
<table className="w-full text-left text-sm text-slate-200">
<thead className="bg-white/5 text-xs uppercase tracking-[0.2em] text-slate-400">
<tr>
<th className="px-4 py-3">#</th>
<th className="px-4 py-3">Rule</th>
</tr>
</thead>
<tbody>
{rules.length === 0 && (
<tr>
<td colSpan={2} className="px-4 py-6 text-center text-slate-500">
No rules loaded. Use the refresh button above.
</td>
</tr>
)}
{rules.map((rule, idx) => (
<tr key={`${rule}-${idx}`} className="border-t border-white/5">
<td className="px-4 py-3 text-slate-400">{idx + 1}</td>
<td className="px-4 py-3">{rule}</td>
</tr>
))}
</tbody>
</table>
</div>
</div>
</div>
</section>
<Footer />
</main>
);
}