Compare commits
2 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 7c4e20099d | |||
| 3521a0ff4f |
32
scripts/verify-personal-org.mjs
Normal file
32
scripts/verify-personal-org.mjs
Normal file
@@ -0,0 +1,32 @@
|
|||||||
|
// Standalone JS port of `lib/personal-org.ts::isPersonalOrgName`
|
||||||
|
// for offline verification.
|
||||||
|
|
||||||
|
const PERSONAL_ORG_SUFFIX = " (Personal)";
|
||||||
|
|
||||||
|
function isPersonalOrgName(orgName) {
|
||||||
|
if (!orgName) return false;
|
||||||
|
return orgName.trimEnd().endsWith(PERSONAL_ORG_SUFFIX);
|
||||||
|
}
|
||||||
|
|
||||||
|
const cases = [
|
||||||
|
["Bob Müller (Personal)", true, "personal account"],
|
||||||
|
["Acme GmbH", false, "company"],
|
||||||
|
["Acme (Personal) Ltd", false, "suffix in middle does not count"],
|
||||||
|
["Bob (Personal) ", true, "trailing whitespace tolerated"],
|
||||||
|
["Bob (personal)", false, "case-sensitive — lowercase doesn't match"],
|
||||||
|
["", false, "empty"],
|
||||||
|
[null, false, "null"],
|
||||||
|
[undefined, false, "undefined"],
|
||||||
|
["Bob (Personal)x", false, "non-trailing suffix"],
|
||||||
|
[" (Personal)", true, "minimal — empty user name (degenerate but matches)"],
|
||||||
|
];
|
||||||
|
|
||||||
|
let pass = 0, fail = 0;
|
||||||
|
for (const [name, expected, note] of cases) {
|
||||||
|
const got = isPersonalOrgName(name);
|
||||||
|
const ok = got === expected;
|
||||||
|
console.log(`${ok ? "PASS" : "FAIL"} got=${got} want=${expected} [${note}] input=${JSON.stringify(name)}`);
|
||||||
|
if (ok) pass++; else fail++;
|
||||||
|
}
|
||||||
|
console.log(`\n${pass} pass, ${fail} fail`);
|
||||||
|
process.exit(fail === 0 ? 0 : 1);
|
||||||
38
scripts/verify-role-gates.mjs
Normal file
38
scripts/verify-role-gates.mjs
Normal file
@@ -0,0 +1,38 @@
|
|||||||
|
// Standalone JS port of `lib/session.ts::canMutate` and `isCustomerOwner`
|
||||||
|
// for offline verification.
|
||||||
|
//
|
||||||
|
// SessionUser shape mirrors the TypeScript interface:
|
||||||
|
// { roles: Role[], isPlatform: boolean, ... }
|
||||||
|
|
||||||
|
function canMutate(user) {
|
||||||
|
return user.isPlatform || user.roles.includes("owner");
|
||||||
|
}
|
||||||
|
|
||||||
|
function isCustomerOwner(user) {
|
||||||
|
return !user.isPlatform && user.roles.includes("owner");
|
||||||
|
}
|
||||||
|
|
||||||
|
const cases = [
|
||||||
|
// [user, fn, expected, note]
|
||||||
|
[{ isPlatform: true, roles: ["platform_admin"] }, canMutate, true, "platform admin can mutate"],
|
||||||
|
[{ isPlatform: true, roles: ["platform_operator"] }, canMutate, true, "platform operator can mutate"],
|
||||||
|
[{ isPlatform: false, roles: ["owner"] }, canMutate, true, "customer owner can mutate"],
|
||||||
|
[{ isPlatform: false, roles: ["user"] }, canMutate, false, "customer user cannot mutate"],
|
||||||
|
[{ isPlatform: false, roles: [] }, canMutate, false, "no roles cannot mutate"],
|
||||||
|
[{ isPlatform: false, roles: ["owner", "user"] }, canMutate, true, "owner+user (owner wins)"],
|
||||||
|
|
||||||
|
[{ isPlatform: true, roles: ["platform_admin", "owner"] }, isCustomerOwner, false, "platform user with owner role is NOT customerOwner"],
|
||||||
|
[{ isPlatform: false, roles: ["owner"] }, isCustomerOwner, true, "pure customer owner"],
|
||||||
|
[{ isPlatform: false, roles: ["user"] }, isCustomerOwner, false, "customer user is not customerOwner"],
|
||||||
|
[{ isPlatform: false, roles: [] }, isCustomerOwner, false, "empty roles is not customerOwner"],
|
||||||
|
];
|
||||||
|
|
||||||
|
let pass = 0, fail = 0;
|
||||||
|
for (const [user, fn, expected, note] of cases) {
|
||||||
|
const got = fn(user);
|
||||||
|
const ok = got === expected;
|
||||||
|
console.log(`${ok ? "PASS" : "FAIL"} got=${got} want=${expected} [${note}]`);
|
||||||
|
if (ok) pass++; else fail++;
|
||||||
|
}
|
||||||
|
console.log(`\n${pass} pass, ${fail} fail`);
|
||||||
|
process.exit(fail === 0 ? 0 : 1);
|
||||||
@@ -1,4 +1,4 @@
|
|||||||
import { getSessionUser } from "@/lib/session";
|
import { getSessionUser, canMutate } from "@/lib/session";
|
||||||
import { getTranslations } from "next-intl/server";
|
import { getTranslations } from "next-intl/server";
|
||||||
import { redirect } from "next/navigation";
|
import { redirect } from "next/navigation";
|
||||||
import { OnboardingFlow } from "@/components/onboarding/onboarding-flow";
|
import { OnboardingFlow } from "@/components/onboarding/onboarding-flow";
|
||||||
@@ -16,11 +16,17 @@ import Link from "next/link";
|
|||||||
*
|
*
|
||||||
* Platform admins are redirected to /dashboard — they shouldn't be
|
* Platform admins are redirected to /dashboard — they shouldn't be
|
||||||
* creating tenant instances under their own org.
|
* creating tenant instances under their own org.
|
||||||
|
*
|
||||||
|
* Slice 5: customer-side `user` role is also redirected — only owners
|
||||||
|
* may create new instances. The server-side POST handler enforces the
|
||||||
|
* same; this redirect is purely UX so /user-role members don't land on
|
||||||
|
* a wizard that will 403 on submit.
|
||||||
*/
|
*/
|
||||||
export default async function NewInstancePage() {
|
export default async function NewInstancePage() {
|
||||||
const user = await getSessionUser();
|
const user = await getSessionUser();
|
||||||
if (!user) redirect("/login");
|
if (!user) redirect("/login");
|
||||||
if (user.isPlatform) redirect("/dashboard");
|
if (user.isPlatform) redirect("/dashboard");
|
||||||
|
if (!canMutate(user)) redirect("/dashboard");
|
||||||
|
|
||||||
const t = await getTranslations("dashboard");
|
const t = await getTranslations("dashboard");
|
||||||
|
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { getSessionUser } from "@/lib/session";
|
import { getSessionUser, canMutate } from "@/lib/session";
|
||||||
import { getTranslations, getFormatter } from "next-intl/server";
|
import { getTranslations, getFormatter } from "next-intl/server";
|
||||||
import { redirect } from "next/navigation";
|
import { redirect } from "next/navigation";
|
||||||
import { listTenants } from "@/lib/k8s";
|
import { listTenants } from "@/lib/k8s";
|
||||||
@@ -149,8 +149,39 @@ export default async function DashboardPage() {
|
|||||||
(r) => !r.tenantName || !orgTenants.some((t) => t.metadata.name === r.tenantName)
|
(r) => !r.tenantName || !orgTenants.some((t) => t.metadata.name === r.tenantName)
|
||||||
);
|
);
|
||||||
|
|
||||||
|
// Slice 5: only owners (and platform users, who'd typically be using
|
||||||
|
// the admin panel anyway) see the "Create new instance" link. A
|
||||||
|
// `user`-role member sees the dashboard but not the create flow —
|
||||||
|
// they need to ask an owner.
|
||||||
|
const canCreate = canMutate(user);
|
||||||
|
|
||||||
// First-time user: empty company. Show the onboarding wizard inline.
|
// First-time user: empty company. Show the onboarding wizard inline.
|
||||||
|
// Note: the registering user is always granted `owner` on their new
|
||||||
|
// org by registerCustomer, so this branch is only reachable by an
|
||||||
|
// owner — no role check needed here. But a customer-side `user`
|
||||||
|
// promoted into a fresh empty org (Slice 7 invites) would also land
|
||||||
|
// here without permission to submit. Belt-and-braces gate.
|
||||||
if (orgTenants.length === 0 && inflightRequests.length === 0) {
|
if (orgTenants.length === 0 && inflightRequests.length === 0) {
|
||||||
|
if (!canCreate) {
|
||||||
|
return (
|
||||||
|
<div>
|
||||||
|
<div className="mb-8 animate-in">
|
||||||
|
<h1 className="font-display text-2xl font-semibold accent-rule mb-2">
|
||||||
|
{t("title")}
|
||||||
|
</h1>
|
||||||
|
<p className="text-text-secondary text-sm mt-4">
|
||||||
|
{t("welcome", { name: user.name || user.email })}
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
<Card className="animate-in animate-in-delay-1">
|
||||||
|
<p className="text-sm text-text-secondary text-center py-6">
|
||||||
|
{t("noAccessNoInstances")}
|
||||||
|
</p>
|
||||||
|
</Card>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div>
|
<div>
|
||||||
<div className="mb-8 animate-in">
|
<div className="mb-8 animate-in">
|
||||||
@@ -170,7 +201,7 @@ export default async function DashboardPage() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Returning customer: list of tenants + in-flight requests, plus
|
// Returning customer: list of tenants + in-flight requests, plus
|
||||||
// a button to add another instance.
|
// a button to add another instance (owners only).
|
||||||
return (
|
return (
|
||||||
<div>
|
<div>
|
||||||
<div className="mb-8 animate-in flex items-start justify-between gap-4">
|
<div className="mb-8 animate-in flex items-start justify-between gap-4">
|
||||||
@@ -183,12 +214,14 @@ export default async function DashboardPage() {
|
|||||||
</p>
|
</p>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<Link
|
{canCreate && (
|
||||||
href="/dashboard/new"
|
<Link
|
||||||
className="shrink-0 inline-flex items-center gap-1.5 py-2 px-4 bg-accent text-white text-xs font-medium rounded-lg hover:bg-accent-dim transition-colors"
|
href="/dashboard/new"
|
||||||
>
|
className="shrink-0 inline-flex items-center gap-1.5 py-2 px-4 bg-accent text-white text-xs font-medium rounded-lg hover:bg-accent-dim transition-colors"
|
||||||
<span>+</span> {t("createInstance")}
|
>
|
||||||
</Link>
|
<span>+</span> {t("createInstance")}
|
||||||
|
</Link>
|
||||||
|
)}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{/* In-flight (pending/approved/provisioning/rejected) requests */}
|
{/* In-flight (pending/approved/provisioning/rejected) requests */}
|
||||||
|
|||||||
@@ -7,6 +7,13 @@ import { Card } from "@/components/ui/card";
|
|||||||
|
|
||||||
type FormState = "idle" | "submitting" | "success" | "error";
|
type FormState = "idle" | "submitting" | "success" | "error";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Slice 4: a "Register as individual" toggle distinguishes personal
|
||||||
|
* accounts from company registrations. When the toggle is on:
|
||||||
|
* - the company name field is hidden (and not sent)
|
||||||
|
* - the server skips the duplicate-domain check
|
||||||
|
* - the ZITADEL org is named "{givenName} {familyName} (Personal)"
|
||||||
|
*/
|
||||||
export default function RegisterPage() {
|
export default function RegisterPage() {
|
||||||
const t = useTranslations("register");
|
const t = useTranslations("register");
|
||||||
const tCommon = useTranslations("common");
|
const tCommon = useTranslations("common");
|
||||||
@@ -18,6 +25,7 @@ export default function RegisterPage() {
|
|||||||
familyName: "",
|
familyName: "",
|
||||||
email: "",
|
email: "",
|
||||||
});
|
});
|
||||||
|
const [isPersonal, setIsPersonal] = useState(false);
|
||||||
const [state, setState] = useState<FormState>("idle");
|
const [state, setState] = useState<FormState>("idle");
|
||||||
const [error, setError] = useState("");
|
const [error, setError] = useState("");
|
||||||
|
|
||||||
@@ -31,15 +39,23 @@ export default function RegisterPage() {
|
|||||||
setState("submitting");
|
setState("submitting");
|
||||||
|
|
||||||
try {
|
try {
|
||||||
|
// Build the request body explicitly. For personals we omit
|
||||||
|
// companyName so the server knows to derive the org name from
|
||||||
|
// the user's full name. The Zod schema accepts the omission.
|
||||||
|
const body: Record<string, unknown> = {
|
||||||
|
givenName: form.givenName,
|
||||||
|
familyName: form.familyName,
|
||||||
|
email: form.email,
|
||||||
|
isPersonal,
|
||||||
|
};
|
||||||
|
if (!isPersonal) {
|
||||||
|
body.companyName = form.companyName;
|
||||||
|
}
|
||||||
|
|
||||||
const res = await fetch("/api/register", {
|
const res = await fetch("/api/register", {
|
||||||
method: "POST",
|
method: "POST",
|
||||||
headers: { "Content-Type": "application/json" },
|
headers: { "Content-Type": "application/json" },
|
||||||
body: JSON.stringify({
|
body: JSON.stringify(body),
|
||||||
companyName: form.companyName,
|
|
||||||
givenName: form.givenName,
|
|
||||||
familyName: form.familyName,
|
|
||||||
email: form.email,
|
|
||||||
}),
|
|
||||||
});
|
});
|
||||||
|
|
||||||
if (!res.ok) {
|
if (!res.ok) {
|
||||||
@@ -104,21 +120,41 @@ export default function RegisterPage() {
|
|||||||
|
|
||||||
<Card className="animate-in animate-in-delay-1">
|
<Card className="animate-in animate-in-delay-1">
|
||||||
<form onSubmit={handleSubmit} className="space-y-4">
|
<form onSubmit={handleSubmit} className="space-y-4">
|
||||||
{/* Company name */}
|
{/* Personal-account toggle */}
|
||||||
<div>
|
<label className="flex items-start gap-3 cursor-pointer select-none p-3 rounded-lg border border-border bg-surface-2 hover:border-accent/40 transition-colors">
|
||||||
<label className="block text-xs font-semibold uppercase tracking-wider text-text-muted mb-1.5">
|
|
||||||
{t("companyName")}
|
|
||||||
</label>
|
|
||||||
<input
|
<input
|
||||||
name="companyName"
|
type="checkbox"
|
||||||
type="text"
|
checked={isPersonal}
|
||||||
required
|
onChange={(e) => setIsPersonal(e.target.checked)}
|
||||||
value={form.companyName}
|
className="mt-0.5 h-4 w-4 rounded border-border bg-surface-1 text-accent focus:ring-1 focus:ring-accent focus:ring-offset-0"
|
||||||
onChange={handleChange}
|
|
||||||
placeholder={t("companyNamePlaceholder")}
|
|
||||||
className="w-full px-3 py-2 bg-surface-2 border border-border rounded-lg text-sm text-text-primary placeholder:text-text-muted focus:outline-none focus:ring-1 focus:ring-accent focus:border-accent transition-colors"
|
|
||||||
/>
|
/>
|
||||||
</div>
|
<div className="flex-1 min-w-0">
|
||||||
|
<div className="text-sm font-medium text-text-primary">
|
||||||
|
{t("individualToggle")}
|
||||||
|
</div>
|
||||||
|
<div className="text-xs text-text-muted mt-0.5">
|
||||||
|
{t("individualHint")}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</label>
|
||||||
|
|
||||||
|
{/* Company name — hidden for personal */}
|
||||||
|
{!isPersonal && (
|
||||||
|
<div>
|
||||||
|
<label className="block text-xs font-semibold uppercase tracking-wider text-text-muted mb-1.5">
|
||||||
|
{t("companyName")}
|
||||||
|
</label>
|
||||||
|
<input
|
||||||
|
name="companyName"
|
||||||
|
type="text"
|
||||||
|
required
|
||||||
|
value={form.companyName}
|
||||||
|
onChange={handleChange}
|
||||||
|
placeholder={t("companyNamePlaceholder")}
|
||||||
|
className="w-full px-3 py-2 bg-surface-2 border border-border rounded-lg text-sm text-text-primary placeholder:text-text-muted focus:outline-none focus:ring-1 focus:ring-accent focus:border-accent transition-colors"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
{/* Name row */}
|
{/* Name row */}
|
||||||
<div className="grid grid-cols-2 gap-3">
|
<div className="grid grid-cols-2 gap-3">
|
||||||
@@ -161,7 +197,7 @@ export default function RegisterPage() {
|
|||||||
required
|
required
|
||||||
value={form.email}
|
value={form.email}
|
||||||
onChange={handleChange}
|
onChange={handleChange}
|
||||||
placeholder="you@company.ch"
|
placeholder={isPersonal ? "you@example.ch" : "you@company.ch"}
|
||||||
className="w-full px-3 py-2 bg-surface-2 border border-border rounded-lg text-sm text-text-primary placeholder:text-text-muted focus:outline-none focus:ring-1 focus:ring-accent focus:border-accent transition-colors"
|
className="w-full px-3 py-2 bg-surface-2 border border-border rounded-lg text-sm text-text-primary placeholder:text-text-muted focus:outline-none focus:ring-1 focus:ring-accent focus:border-accent transition-colors"
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { getSessionUser } from "@/lib/session";
|
import { getSessionUser, canMutate } from "@/lib/session";
|
||||||
import { getTranslations, getFormatter } from "next-intl/server";
|
import { getTranslations, getFormatter } from "next-intl/server";
|
||||||
import { redirect, notFound } from "next/navigation";
|
import { redirect, notFound } from "next/navigation";
|
||||||
import { getTenant } from "@/lib/k8s";
|
import { getTenant } from "@/lib/k8s";
|
||||||
@@ -34,6 +34,11 @@ export default async function TenantDetailPage({
|
|||||||
notFound();
|
notFound();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Slice 5: editable surface gated on owner role. Platform users always
|
||||||
|
// can edit; customer-side, only `owner` may. `user`-role members see
|
||||||
|
// the same page but with edit controls hidden / fields read-only.
|
||||||
|
const canEdit = canMutate(user);
|
||||||
|
|
||||||
const enabledPackages = tenant.spec.packages || [];
|
const enabledPackages = tenant.spec.packages || [];
|
||||||
const workspaceFiles = tenant.spec.workspaceFiles || {};
|
const workspaceFiles = tenant.spec.workspaceFiles || {};
|
||||||
const enabledChannels = enabledPackages.filter((pkg) =>
|
const enabledChannels = enabledPackages.filter((pkg) =>
|
||||||
@@ -100,6 +105,7 @@ export default async function TenantDetailPage({
|
|||||||
tenantName={name}
|
tenantName={name}
|
||||||
enabledPackages={enabledPackages}
|
enabledPackages={enabledPackages}
|
||||||
conditions={tenant.status?.conditions}
|
conditions={tenant.status?.conditions}
|
||||||
|
canEdit={canEdit}
|
||||||
/>
|
/>
|
||||||
</section>
|
</section>
|
||||||
|
|
||||||
@@ -110,6 +116,7 @@ export default async function TenantDetailPage({
|
|||||||
tenantName={name}
|
tenantName={name}
|
||||||
enabledChannels={enabledChannels}
|
enabledChannels={enabledChannels}
|
||||||
initialChannelUsers={channelUsers}
|
initialChannelUsers={channelUsers}
|
||||||
|
canEdit={canEdit}
|
||||||
/>
|
/>
|
||||||
</section>
|
</section>
|
||||||
)}
|
)}
|
||||||
@@ -119,7 +126,7 @@ export default async function TenantDetailPage({
|
|||||||
<h2 className="text-xs font-semibold uppercase tracking-wider text-text-muted mb-3">
|
<h2 className="text-xs font-semibold uppercase tracking-wider text-text-muted mb-3">
|
||||||
{t("workspaceFiles")}
|
{t("workspaceFiles")}
|
||||||
</h2>
|
</h2>
|
||||||
<WorkspaceEditor tenantName={name} files={workspaceFiles} />
|
<WorkspaceEditor tenantName={name} files={workspaceFiles} canEdit={canEdit} />
|
||||||
</section>
|
</section>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -63,10 +63,10 @@ export async function POST(
|
|||||||
const isReApproval = tenantRequest.status === "rejected";
|
const isReApproval = tenantRequest.status === "rejected";
|
||||||
|
|
||||||
// Build the CR name: see `lib/tenant-naming.ts` for the format spec.
|
// Build the CR name: see `lib/tenant-naming.ts` for the format spec.
|
||||||
// For now all approvals are kind="company" — the personal branch is
|
// Slice 4: for personal accounts the slug is replaced by the literal
|
||||||
// wired but unused until Slice 4 introduces the `is_personal` column.
|
// "p-" prefix so no PII is embedded in the K8s namespace name.
|
||||||
const tenantName = deriveTenantName(
|
const tenantName = deriveTenantName(
|
||||||
"company",
|
tenantRequest.isPersonal ? "personal" : "company",
|
||||||
tenantRequest.companyName,
|
tenantRequest.companyName,
|
||||||
tenantRequest.id
|
tenantRequest.id
|
||||||
);
|
);
|
||||||
@@ -101,13 +101,17 @@ export async function POST(
|
|||||||
};
|
};
|
||||||
|
|
||||||
// Step 4: Create the PiecedTenant CR.
|
// Step 4: Create the PiecedTenant CR.
|
||||||
// displayName: prefer the customer-chosen instance name; fall back to
|
// displayName precedence:
|
||||||
// the company name. With multi-tenant per org, instanceName is what
|
// 1. customer-chosen instance name (Slice 3 multi-tenant)
|
||||||
// distinguishes "Acme Production" from "Acme Dev" on the dashboard.
|
// 2. for personal accounts, the contact name (avoids exposing the
|
||||||
|
// synthetic "{name} (Personal)" company name in the OpenClaw UI)
|
||||||
|
// 3. company name otherwise
|
||||||
const displayName =
|
const displayName =
|
||||||
tenantRequest.instanceName && tenantRequest.instanceName.trim().length > 0
|
tenantRequest.instanceName && tenantRequest.instanceName.trim().length > 0
|
||||||
? tenantRequest.instanceName.trim()
|
? tenantRequest.instanceName.trim()
|
||||||
: tenantRequest.companyName;
|
: tenantRequest.isPersonal
|
||||||
|
? tenantRequest.contactName || "Assistant"
|
||||||
|
: tenantRequest.companyName;
|
||||||
|
|
||||||
await createTenant(
|
await createTenant(
|
||||||
tenantName,
|
tenantName,
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
import { NextRequest, NextResponse } from "next/server";
|
import { NextRequest, NextResponse } from "next/server";
|
||||||
import { getSessionUser } from "@/lib/session";
|
import { getSessionUser, canMutate } from "@/lib/session";
|
||||||
import {
|
import {
|
||||||
createTenantRequest,
|
createTenantRequest,
|
||||||
getTenantRequestById,
|
getTenantRequestById,
|
||||||
@@ -10,6 +10,7 @@ import {
|
|||||||
import { getTenant, listTenants } from "@/lib/k8s";
|
import { getTenant, listTenants } from "@/lib/k8s";
|
||||||
import { sendAdminNotificationEmail } from "@/lib/email";
|
import { sendAdminNotificationEmail } from "@/lib/email";
|
||||||
import { encryptSecrets } from "@/lib/crypto";
|
import { encryptSecrets } from "@/lib/crypto";
|
||||||
|
import { isPersonalOrgName } from "@/lib/personal-org";
|
||||||
import type { OnboardingInput, PiecedTenant, TenantRequest } from "@/types";
|
import type { OnboardingInput, PiecedTenant, TenantRequest } from "@/types";
|
||||||
import { z } from "zod";
|
import { z } from "zod";
|
||||||
|
|
||||||
@@ -156,6 +157,15 @@ export async function POST(request: Request) {
|
|||||||
return NextResponse.json({ error: "Unauthorized" }, { status: 401 });
|
return NextResponse.json({ error: "Unauthorized" }, { status: 401 });
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Slice 5: only owners (or platform users) may create new instances.
|
||||||
|
// A `user`-role member of an existing org cannot self-provision.
|
||||||
|
if (!canMutate(user)) {
|
||||||
|
return NextResponse.json(
|
||||||
|
{ error: "Only the organization owner can create new instances." },
|
||||||
|
{ status: 403 }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
const body = await request.json();
|
const body = await request.json();
|
||||||
const parsed = onboardingSchema.safeParse(body);
|
const parsed = onboardingSchema.safeParse(body);
|
||||||
if (!parsed.success) {
|
if (!parsed.success) {
|
||||||
@@ -176,6 +186,16 @@ export async function POST(request: Request) {
|
|||||||
// company line in favour of the recorded company name.
|
// company line in favour of the recorded company name.
|
||||||
const prior = await getMostRecentApprovedRequestForOrg(user.orgId);
|
const prior = await getMostRecentApprovedRequestForOrg(user.orgId);
|
||||||
|
|
||||||
|
// Slice 4: detect personal-account orgs by the canonical " (Personal)"
|
||||||
|
// suffix on the ZITADEL org name. Set at registration, stable for the
|
||||||
|
// lifetime of the org. Persisted on the row so admin views and the
|
||||||
|
// approve handler don't have to re-derive it.
|
||||||
|
//
|
||||||
|
// If any prior row has is_personal set, prefer that — it's the same
|
||||||
|
// org and the value can't change. (The prior-row check is defensive;
|
||||||
|
// the org-name check should agree.)
|
||||||
|
const isPersonal = prior?.isPersonal ?? isPersonalOrgName(user.orgName);
|
||||||
|
|
||||||
// Encrypt package secrets if provided
|
// Encrypt package secrets if provided
|
||||||
let encryptedSecrets: Buffer | undefined;
|
let encryptedSecrets: Buffer | undefined;
|
||||||
if (input.packageSecrets && Object.keys(input.packageSecrets).length > 0) {
|
if (input.packageSecrets && Object.keys(input.packageSecrets).length > 0) {
|
||||||
@@ -212,6 +232,7 @@ export async function POST(request: Request) {
|
|||||||
billingAddress,
|
billingAddress,
|
||||||
billingNotes,
|
billingNotes,
|
||||||
encryptedSecrets,
|
encryptedSecrets,
|
||||||
|
isPersonal,
|
||||||
});
|
});
|
||||||
|
|
||||||
// Notify admin about the new request. For follow-up instances, include
|
// Notify admin about the new request. For follow-up instances, include
|
||||||
|
|||||||
@@ -5,18 +5,54 @@ import { checkDuplicateDomain } from "@/lib/db";
|
|||||||
import type { RegistrationInput } from "@/types";
|
import type { RegistrationInput } from "@/types";
|
||||||
import { z } from "zod";
|
import { z } from "zod";
|
||||||
|
|
||||||
const registrationSchema = z.object({
|
/**
|
||||||
companyName: z.string().min(2).max(100),
|
* Registration schema.
|
||||||
givenName: z.string().min(1).max(100),
|
*
|
||||||
familyName: z.string().min(1).max(100),
|
* Slice 4 changes
|
||||||
email: z.string().email(),
|
* ---------------
|
||||||
preferredLanguage: z.enum(["en", "de", "fr", "it"]).optional(),
|
* - `companyName` is no longer always required. It's required when
|
||||||
});
|
* `isPersonal` is false/absent, ignored when `isPersonal` is true.
|
||||||
|
* - `isPersonal` flag distinguishes personal accounts. The server
|
||||||
|
* derives the ZITADEL org name from `${givenName} ${familyName}
|
||||||
|
* (Personal)` for personals — the suffix is the canonical marker
|
||||||
|
* that downstream code (onboarding POST, admin views) uses to
|
||||||
|
* distinguish personal orgs from companies. Customers cannot rename
|
||||||
|
* their own org, so the suffix is stable.
|
||||||
|
* - Personal accounts skip the duplicate-domain check entirely. Their
|
||||||
|
* row is also excluded from future domain checks (see
|
||||||
|
* `lib/domain-check.ts::findDuplicateInDb`).
|
||||||
|
*/
|
||||||
|
const registrationSchema = z
|
||||||
|
.object({
|
||||||
|
companyName: z.string().min(2).max(100).optional(),
|
||||||
|
givenName: z.string().min(1).max(100),
|
||||||
|
familyName: z.string().min(1).max(100),
|
||||||
|
email: z.string().email(),
|
||||||
|
preferredLanguage: z.enum(["en", "de", "fr", "it"]).optional(),
|
||||||
|
isPersonal: z.boolean().optional().default(false),
|
||||||
|
})
|
||||||
|
.refine(
|
||||||
|
(data) =>
|
||||||
|
data.isPersonal || (data.companyName && data.companyName.trim().length >= 2),
|
||||||
|
{
|
||||||
|
message: "Company name is required for company registrations",
|
||||||
|
path: ["companyName"],
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
/** 3 registrations per IP per hour */
|
/** 3 registrations per IP per hour */
|
||||||
const RATE_LIMIT = 3;
|
const RATE_LIMIT = 3;
|
||||||
const RATE_WINDOW_MS = 3_600_000; // 1 hour
|
const RATE_WINDOW_MS = 3_600_000; // 1 hour
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Suffix appended to personal-account ZITADEL org names. Used here to
|
||||||
|
* build the org name and elsewhere (session.orgName check) to detect
|
||||||
|
* whether the current user is on a personal org.
|
||||||
|
*
|
||||||
|
* Keep this in sync with `isPersonalOrgName()` in `lib/personal-org.ts`.
|
||||||
|
*/
|
||||||
|
const PERSONAL_ORG_SUFFIX = " (Personal)";
|
||||||
|
|
||||||
export async function POST(request: NextRequest) {
|
export async function POST(request: NextRequest) {
|
||||||
// --- Rate limiting ---
|
// --- Rate limiting ---
|
||||||
const ip =
|
const ip =
|
||||||
@@ -53,31 +89,45 @@ export async function POST(request: NextRequest) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const input: RegistrationInput = parsed.data;
|
const input: RegistrationInput = parsed.data;
|
||||||
|
const isPersonal = input.isPersonal === true;
|
||||||
|
|
||||||
// --- Duplicate-domain check ---
|
// --- Duplicate-domain check (skipped for personal accounts) ---
|
||||||
//
|
//
|
||||||
// Block if another active tenant_request or ZITADEL org already exists
|
// Personal accounts are explicitly allowed to use any email domain
|
||||||
// for this corporate email domain. Public domains (gmail, gmx, etc.)
|
// (including corporate). Their tenant_request rows are excluded
|
||||||
// are exempted by checkDuplicateDomain.
|
// from this check by lib/domain-check.ts, so a personal account
|
||||||
//
|
// doesn't block a later real-company registration on the same
|
||||||
// We return a structured `code: "duplicate_domain"` with the matched
|
// domain.
|
||||||
// domain so the client can render the localized message via
|
if (!isPersonal) {
|
||||||
// register.duplicateDomain (with {domain} interpolation). The fallback
|
const dup = await checkDuplicateDomain(input.email);
|
||||||
// English string is included for non-i18n clients (curl, monitoring).
|
if (dup.blocked && dup.domain) {
|
||||||
const dup = await checkDuplicateDomain(input.email);
|
return NextResponse.json(
|
||||||
if (dup.blocked && dup.domain) {
|
{
|
||||||
return NextResponse.json(
|
error: `An account for the email domain ${dup.domain} is already registered. Please contact your company administrator or PieCed IT support.`,
|
||||||
{
|
code: "duplicate_domain",
|
||||||
error: `An account for the email domain ${dup.domain} is already registered. Please contact your company administrator or PieCed IT support.`,
|
domain: dup.domain,
|
||||||
code: "duplicate_domain",
|
},
|
||||||
domain: dup.domain,
|
{ status: 409 },
|
||||||
},
|
);
|
||||||
{ status: 409 },
|
}
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// --- Determine the ZITADEL org name ---
|
||||||
|
//
|
||||||
|
// For company: use the customer-supplied companyName (already
|
||||||
|
// validated to be present + ≥2 chars by the schema refinement).
|
||||||
|
// For personal: synthesise from full name + " (Personal)" suffix.
|
||||||
|
// The suffix is the canonical marker for personal orgs.
|
||||||
|
//
|
||||||
|
// ZITADEL does NOT enforce org-name uniqueness, so two "Hans Müller
|
||||||
|
// (Personal)" orgs can coexist; the org id is what matters for our
|
||||||
|
// labelling and lookups, the name is human-readable only.
|
||||||
|
const orgName = isPersonal
|
||||||
|
? `${input.givenName.trim()} ${input.familyName.trim()}${PERSONAL_ORG_SUFFIX}`
|
||||||
|
: input.companyName!.trim();
|
||||||
|
|
||||||
const result = await registerCustomer({
|
const result = await registerCustomer({
|
||||||
companyName: input.companyName,
|
companyName: orgName,
|
||||||
email: input.email,
|
email: input.email,
|
||||||
givenName: input.givenName,
|
givenName: input.givenName,
|
||||||
familyName: input.familyName,
|
familyName: input.familyName,
|
||||||
@@ -88,6 +138,7 @@ export async function POST(request: NextRequest) {
|
|||||||
{
|
{
|
||||||
orgId: result.orgId,
|
orgId: result.orgId,
|
||||||
userId: result.userId,
|
userId: result.userId,
|
||||||
|
isPersonal,
|
||||||
message:
|
message:
|
||||||
"Registration successful. You will receive an invitation email to set your password.",
|
"Registration successful. You will receive an invitation email to set your password.",
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
import { NextRequest, NextResponse } from "next/server";
|
import { NextRequest, NextResponse } from "next/server";
|
||||||
import { getSessionUser } from "@/lib/session";
|
import { getSessionUser, canMutate } from "@/lib/session";
|
||||||
import { getTenant, patchTenantSpec } from "@/lib/k8s";
|
import { getTenant, patchTenantSpec } from "@/lib/k8s";
|
||||||
import { getPackageDef } from "@/lib/packages";
|
import { getPackageDef } from "@/lib/packages";
|
||||||
import { safeError } from "@/lib/errors";
|
import { safeError } from "@/lib/errors";
|
||||||
@@ -46,7 +46,7 @@ export async function PATCH(
|
|||||||
if (!user)
|
if (!user)
|
||||||
return NextResponse.json({ error: "Unauthorized" }, { status: 401 });
|
return NextResponse.json({ error: "Unauthorized" }, { status: 401 });
|
||||||
|
|
||||||
if (!user.isPlatform && !user.roles.includes("owner")) {
|
if (!canMutate(user)) {
|
||||||
return NextResponse.json({ error: "Forbidden" }, { status: 403 });
|
return NextResponse.json({ error: "Forbidden" }, { status: 403 });
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
import { NextRequest, NextResponse } from "next/server";
|
import { NextRequest, NextResponse } from "next/server";
|
||||||
import { getSessionUser } from "@/lib/session";
|
import { getSessionUser, canMutate } from "@/lib/session";
|
||||||
import { getTenant } from "@/lib/k8s";
|
import { getTenant } from "@/lib/k8s";
|
||||||
import { writePackageSecrets } from "@/lib/openbao";
|
import { writePackageSecrets } from "@/lib/openbao";
|
||||||
import { getPackageDef } from "@/lib/packages";
|
import { getPackageDef } from "@/lib/packages";
|
||||||
@@ -12,7 +12,7 @@ export async function POST(
|
|||||||
if (!user)
|
if (!user)
|
||||||
return NextResponse.json({ error: "Unauthorized" }, { status: 401 });
|
return NextResponse.json({ error: "Unauthorized" }, { status: 401 });
|
||||||
|
|
||||||
if (!user.isPlatform && !user.roles.includes("owner")) {
|
if (!canMutate(user)) {
|
||||||
return NextResponse.json({ error: "Forbidden" }, { status: 403 });
|
return NextResponse.json({ error: "Forbidden" }, { status: 403 });
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -17,12 +17,15 @@ interface ChannelUsersProps {
|
|||||||
enabledChannels: string[];
|
enabledChannels: string[];
|
||||||
/** Current channelUsers from the PiecedTenant spec */
|
/** Current channelUsers from the PiecedTenant spec */
|
||||||
initialChannelUsers: Record<string, string[]>;
|
initialChannelUsers: Record<string, string[]>;
|
||||||
|
/** Slice 5: when false, add inputs and remove ✕ buttons are hidden. */
|
||||||
|
canEdit?: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
export function ChannelUsers({
|
export function ChannelUsers({
|
||||||
tenantName,
|
tenantName,
|
||||||
enabledChannels,
|
enabledChannels,
|
||||||
initialChannelUsers,
|
initialChannelUsers,
|
||||||
|
canEdit = true,
|
||||||
}: ChannelUsersProps) {
|
}: ChannelUsersProps) {
|
||||||
const t = useTranslations("channelUsers");
|
const t = useTranslations("channelUsers");
|
||||||
const router = useRouter();
|
const router = useRouter();
|
||||||
@@ -146,44 +149,48 @@ export function ChannelUsers({
|
|||||||
className="inline-flex items-center gap-1.5 px-2.5 py-1 text-xs font-mono bg-accent/10 text-accent border border-accent/20 rounded-full"
|
className="inline-flex items-center gap-1.5 px-2.5 py-1 text-xs font-mono bg-accent/10 text-accent border border-accent/20 rounded-full"
|
||||||
>
|
>
|
||||||
{userId}
|
{userId}
|
||||||
<button
|
{canEdit && (
|
||||||
onClick={() => handleRemove(channel, userId)}
|
<button
|
||||||
disabled={saving}
|
onClick={() => handleRemove(channel, userId)}
|
||||||
className="text-accent/60 hover:text-red-400 transition-colors disabled:opacity-50"
|
disabled={saving}
|
||||||
title={t("remove")}
|
className="text-accent/60 hover:text-red-400 transition-colors disabled:opacity-50"
|
||||||
>
|
title={t("remove")}
|
||||||
✕
|
>
|
||||||
</button>
|
✕
|
||||||
|
</button>
|
||||||
|
)}
|
||||||
</span>
|
</span>
|
||||||
))}
|
))}
|
||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
{/* Add user */}
|
{/* Add user — hidden in read-only mode */}
|
||||||
<div className="flex gap-2">
|
{canEdit && (
|
||||||
<input
|
<div className="flex gap-2">
|
||||||
type="text"
|
<input
|
||||||
value={inputValues[channel] || ""}
|
type="text"
|
||||||
onChange={(e) =>
|
value={inputValues[channel] || ""}
|
||||||
setInputValues((prev) => ({
|
onChange={(e) =>
|
||||||
...prev,
|
setInputValues((prev) => ({
|
||||||
[channel]: e.target.value,
|
...prev,
|
||||||
}))
|
[channel]: e.target.value,
|
||||||
}
|
}))
|
||||||
onKeyDown={(e) => {
|
}
|
||||||
if (e.key === "Enter") handleAdd(channel);
|
onKeyDown={(e) => {
|
||||||
}}
|
if (e.key === "Enter") handleAdd(channel);
|
||||||
placeholder={t("placeholder")}
|
}}
|
||||||
className="flex-1 px-3 py-2 bg-surface-1 border border-border rounded-lg text-sm text-text-primary font-mono placeholder:text-text-muted focus:outline-none focus:ring-1 focus:ring-accent focus:border-accent transition-colors"
|
placeholder={t("placeholder")}
|
||||||
/>
|
className="flex-1 px-3 py-2 bg-surface-1 border border-border rounded-lg text-sm text-text-primary font-mono placeholder:text-text-muted focus:outline-none focus:ring-1 focus:ring-accent focus:border-accent transition-colors"
|
||||||
<button
|
/>
|
||||||
onClick={() => handleAdd(channel)}
|
<button
|
||||||
disabled={saving || !inputValues[channel]?.trim()}
|
onClick={() => handleAdd(channel)}
|
||||||
className="px-4 py-2 text-sm font-medium bg-accent text-white rounded-lg hover:bg-accent-dim transition-colors disabled:opacity-50 disabled:cursor-not-allowed"
|
disabled={saving || !inputValues[channel]?.trim()}
|
||||||
>
|
className="px-4 py-2 text-sm font-medium bg-accent text-white rounded-lg hover:bg-accent-dim transition-colors disabled:opacity-50 disabled:cursor-not-allowed"
|
||||||
{saving ? "…" : t("add")}
|
>
|
||||||
</button>
|
{saving ? "…" : t("add")}
|
||||||
</div>
|
</button>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
})}
|
})}
|
||||||
|
|||||||
@@ -4,6 +4,7 @@ import { useState, useCallback, useEffect, useRef } from "react";
|
|||||||
import { useTranslations } from "next-intl";
|
import { useTranslations } from "next-intl";
|
||||||
import { Card } from "@/components/ui/card";
|
import { Card } from "@/components/ui/card";
|
||||||
import { PACKAGE_CATALOG, type PackageDef } from "@/lib/packages";
|
import { PACKAGE_CATALOG, type PackageDef } from "@/lib/packages";
|
||||||
|
import { isPersonalOrgName, PERSONAL_ORG_SUFFIX } from "@/lib/personal-org";
|
||||||
|
|
||||||
type Step = "welcome" | "configure" | "billing" | "confirm";
|
type Step = "welcome" | "configure" | "billing" | "confirm";
|
||||||
|
|
||||||
@@ -55,6 +56,16 @@ export function OnboardingWizard({ orgName, onComplete }: WizardProps) {
|
|||||||
const tPkg = useTranslations("packages");
|
const tPkg = useTranslations("packages");
|
||||||
const tCommon = useTranslations("common");
|
const tCommon = useTranslations("common");
|
||||||
|
|
||||||
|
// Slice 4: personal accounts have an org name of the form
|
||||||
|
// "{givenName} {familyName} (Personal)". For SOUL.md and the billing
|
||||||
|
// company line, strip the suffix so the visible string is the user's
|
||||||
|
// actual name (no stray "(Personal)" leaking onto invoices or into
|
||||||
|
// the assistant's prompt).
|
||||||
|
const isPersonal = isPersonalOrgName(orgName);
|
||||||
|
const displayOrgName = isPersonal
|
||||||
|
? orgName.slice(0, -PERSONAL_ORG_SUFFIX.length).trim()
|
||||||
|
: orgName;
|
||||||
|
|
||||||
const [step, setStep] = useState<Step>("welcome");
|
const [step, setStep] = useState<Step>("welcome");
|
||||||
const [submitting, setSubmitting] = useState(false);
|
const [submitting, setSubmitting] = useState(false);
|
||||||
const [error, setError] = useState("");
|
const [error, setError] = useState("");
|
||||||
@@ -64,11 +75,14 @@ export function OnboardingWizard({ orgName, onComplete }: WizardProps) {
|
|||||||
const [config, setConfig] = useState({
|
const [config, setConfig] = useState({
|
||||||
instanceName: "",
|
instanceName: "",
|
||||||
agentName: "Assistant",
|
agentName: "Assistant",
|
||||||
soulMd: FALLBACK_SOUL.replace("{company}", orgName),
|
soulMd: FALLBACK_SOUL.replace("{company}", displayOrgName),
|
||||||
agentsMd: FALLBACK_AGENTS,
|
agentsMd: FALLBACK_AGENTS,
|
||||||
packages: [] as string[],
|
packages: [] as string[],
|
||||||
billingAddress: {
|
billingAddress: {
|
||||||
company: orgName,
|
// For personal accounts, leave the company field empty — it'll
|
||||||
|
// appear on invoices. The user can still type something if they
|
||||||
|
// want to.
|
||||||
|
company: isPersonal ? "" : displayOrgName,
|
||||||
street: "",
|
street: "",
|
||||||
city: "",
|
city: "",
|
||||||
postalCode: "",
|
postalCode: "",
|
||||||
|
|||||||
@@ -10,9 +10,18 @@ interface Props {
|
|||||||
status?: "pending" | "active" | "error";
|
status?: "pending" | "active" | "error";
|
||||||
tenantName: string;
|
tenantName: string;
|
||||||
onToggled: () => void;
|
onToggled: () => void;
|
||||||
|
/** Slice 5: when false, the enable/disable button is hidden. */
|
||||||
|
canEdit?: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
export function PackageCard({ pkg, enabled, status, tenantName, onToggled }: Props) {
|
export function PackageCard({
|
||||||
|
pkg,
|
||||||
|
enabled,
|
||||||
|
status,
|
||||||
|
tenantName,
|
||||||
|
onToggled,
|
||||||
|
canEdit = true,
|
||||||
|
}: Props) {
|
||||||
const t = useTranslations();
|
const t = useTranslations();
|
||||||
const [showModal, setShowModal] = useState(false);
|
const [showModal, setShowModal] = useState(false);
|
||||||
const [secrets, setSecrets] = useState<Record<string, string>>({});
|
const [secrets, setSecrets] = useState<Record<string, string>>({});
|
||||||
@@ -113,17 +122,27 @@ export function PackageCard({ pkg, enabled, status, tenantName, onToggled }: Pro
|
|||||||
{pkg.requiresSecrets && (
|
{pkg.requiresSecrets && (
|
||||||
<span className="text-[10px] text-text-muted">{t("packages.requiresApiKey")}</span>
|
<span className="text-[10px] text-text-muted">{t("packages.requiresApiKey")}</span>
|
||||||
)}
|
)}
|
||||||
<button
|
{canEdit ? (
|
||||||
onClick={enabled ? () => togglePackage(false) : handleEnable}
|
<button
|
||||||
disabled={saving}
|
onClick={enabled ? () => togglePackage(false) : handleEnable}
|
||||||
className={`ml-auto rounded-lg px-3 py-1.5 text-xs font-medium transition-all cursor-pointer ${
|
disabled={saving}
|
||||||
enabled
|
className={`ml-auto rounded-lg px-3 py-1.5 text-xs font-medium transition-all cursor-pointer ${
|
||||||
? "bg-surface-3 text-text-secondary hover:text-text-primary hover:bg-surface-2"
|
enabled
|
||||||
: "bg-accent text-surface-0 hover:bg-accent-dim shadow-lg shadow-accent/20"
|
? "bg-surface-3 text-text-secondary hover:text-text-primary hover:bg-surface-2"
|
||||||
} disabled:opacity-50`}
|
: "bg-accent text-surface-0 hover:bg-accent-dim shadow-lg shadow-accent/20"
|
||||||
>
|
} disabled:opacity-50`}
|
||||||
{saving ? "…" : enabled ? t("packages.disable") : t("packages.enable")}
|
>
|
||||||
</button>
|
{saving ? "…" : enabled ? t("packages.disable") : t("packages.enable")}
|
||||||
|
</button>
|
||||||
|
) : (
|
||||||
|
// Slice 5: read-only viewers see a static badge instead of a
|
||||||
|
// toggle. The status badge above the divider already conveys
|
||||||
|
// "active/pending/error"; this just clarifies "you can't change
|
||||||
|
// it" without duplicating the status colour.
|
||||||
|
<span className="ml-auto text-[10px] text-text-muted italic">
|
||||||
|
{enabled ? t("packages.statusEnabled") : t("packages.statusDisabled")}
|
||||||
|
</span>
|
||||||
|
)}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
|||||||
@@ -10,6 +10,8 @@ interface Props {
|
|||||||
enabledPackages: string[];
|
enabledPackages: string[];
|
||||||
conditions?: Array<{ type: string; status: string; reason?: string }>;
|
conditions?: Array<{ type: string; status: string; reason?: string }>;
|
||||||
onRefresh?: () => void;
|
onRefresh?: () => void;
|
||||||
|
/** Slice 5: when false, package toggles and edit affordances are hidden. */
|
||||||
|
canEdit?: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
const CATEGORIES = [
|
const CATEGORIES = [
|
||||||
@@ -30,7 +32,13 @@ function getPackageStatus(
|
|||||||
return "error";
|
return "error";
|
||||||
}
|
}
|
||||||
|
|
||||||
export function PackageList({ tenantName, enabledPackages, conditions, onRefresh }: Props) {
|
export function PackageList({
|
||||||
|
tenantName,
|
||||||
|
enabledPackages,
|
||||||
|
conditions,
|
||||||
|
onRefresh,
|
||||||
|
canEdit = true,
|
||||||
|
}: Props) {
|
||||||
const t = useTranslations("packages");
|
const t = useTranslations("packages");
|
||||||
const router = useRouter();
|
const router = useRouter();
|
||||||
const handleRefresh = onRefresh || (() => router.refresh());
|
const handleRefresh = onRefresh || (() => router.refresh());
|
||||||
@@ -55,6 +63,7 @@ export function PackageList({ tenantName, enabledPackages, conditions, onRefresh
|
|||||||
status={getPackageStatus(pkg.id, enabledPackages.includes(pkg.id), conditions)}
|
status={getPackageStatus(pkg.id, enabledPackages.includes(pkg.id), conditions)}
|
||||||
tenantName={tenantName}
|
tenantName={tenantName}
|
||||||
onToggled={handleRefresh}
|
onToggled={handleRefresh}
|
||||||
|
canEdit={canEdit}
|
||||||
/>
|
/>
|
||||||
))}
|
))}
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -8,9 +8,11 @@ const FILE_TABS = ["SOUL.md", "AGENTS.md", "TOOLS.md"] as const;
|
|||||||
interface Props {
|
interface Props {
|
||||||
tenantName: string;
|
tenantName: string;
|
||||||
files: Record<string, string>;
|
files: Record<string, string>;
|
||||||
|
/** Slice 5: when false, save button hidden and textarea is read-only. */
|
||||||
|
canEdit?: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
export function WorkspaceEditor({ tenantName, files }: Props) {
|
export function WorkspaceEditor({ tenantName, files, canEdit = true }: Props) {
|
||||||
const t = useTranslations("workspace");
|
const t = useTranslations("workspace");
|
||||||
const [activeTab, setActiveTab] = useState<string>("SOUL.md");
|
const [activeTab, setActiveTab] = useState<string>("SOUL.md");
|
||||||
const [localFiles, setLocalFiles] = useState<Record<string, string>>(files);
|
const [localFiles, setLocalFiles] = useState<Record<string, string>>(files);
|
||||||
@@ -19,6 +21,7 @@ export function WorkspaceEditor({ tenantName, files }: Props) {
|
|||||||
const [error, setError] = useState<string | null>(null);
|
const [error, setError] = useState<string | null>(null);
|
||||||
|
|
||||||
function handleChange(content: string) {
|
function handleChange(content: string) {
|
||||||
|
if (!canEdit) return;
|
||||||
setLocalFiles((prev) => ({ ...prev, [activeTab]: content }));
|
setLocalFiles((prev) => ({ ...prev, [activeTab]: content }));
|
||||||
setDirty(true);
|
setDirty(true);
|
||||||
}
|
}
|
||||||
@@ -62,20 +65,25 @@ export function WorkspaceEditor({ tenantName, files }: Props) {
|
|||||||
</button>
|
</button>
|
||||||
))}
|
))}
|
||||||
</div>
|
</div>
|
||||||
<button
|
{canEdit && (
|
||||||
onClick={handleSave}
|
<button
|
||||||
disabled={!dirty || saving}
|
onClick={handleSave}
|
||||||
className="rounded-lg bg-accent px-3 py-1 text-xs font-medium text-surface-0 hover:bg-accent-dim disabled:opacity-40 cursor-pointer"
|
disabled={!dirty || saving}
|
||||||
>
|
className="rounded-lg bg-accent px-3 py-1 text-xs font-medium text-surface-0 hover:bg-accent-dim disabled:opacity-40 cursor-pointer"
|
||||||
{saving ? "…" : t("save")}
|
>
|
||||||
</button>
|
{saving ? "…" : t("save")}
|
||||||
|
</button>
|
||||||
|
)}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<textarea
|
<textarea
|
||||||
value={localFiles[activeTab] || ""}
|
value={localFiles[activeTab] || ""}
|
||||||
onChange={(e) => handleChange(e.target.value)}
|
onChange={(e) => handleChange(e.target.value)}
|
||||||
|
readOnly={!canEdit}
|
||||||
spellCheck={false}
|
spellCheck={false}
|
||||||
className="w-full min-h-[300px] resize-y bg-transparent p-4 font-mono text-sm text-text-secondary placeholder:text-text-muted focus:outline-none"
|
className={`w-full min-h-[300px] resize-y bg-transparent p-4 font-mono text-sm text-text-secondary placeholder:text-text-muted focus:outline-none ${
|
||||||
|
!canEdit ? "cursor-default" : ""
|
||||||
|
}`}
|
||||||
placeholder={t("placeholder", { file: activeTab })}
|
placeholder={t("placeholder", { file: activeTab })}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
|
|||||||
@@ -1,14 +1,25 @@
|
|||||||
import NextAuth from "next-auth";
|
import NextAuth from "next-auth";
|
||||||
import type { NextAuthConfig } from "next-auth";
|
import type { NextAuthConfig } from "next-auth";
|
||||||
import type { PlatformRole, SessionUser, ZitadelClaims } from "@/types";
|
import type { PlatformRole, Role, SessionUser, ZitadelClaims } from "@/types";
|
||||||
|
|
||||||
const PLATFORM_ROLES: PlatformRole[] = ["platform_admin", "platform_operator"];
|
const PLATFORM_ROLES: PlatformRole[] = ["platform_admin", "platform_operator"];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Pull the role keys from the ZITADEL `urn:zitadel:iam:org:project:roles`
|
||||||
|
* claim. The claim is shaped as { roleKey: { orgId: orgName } } — we only
|
||||||
|
* need the keys.
|
||||||
|
*
|
||||||
|
* Slice 5: returns Role[] (the union) rather than PlatformRole[]. The
|
||||||
|
* keys can be either platform or customer roles depending on what the
|
||||||
|
* project authorization granted; the SessionUser carries them all and
|
||||||
|
* downstream helpers (canMutate, isCustomerOwner, requirePlatformRole)
|
||||||
|
* decide what each subset means.
|
||||||
|
*/
|
||||||
function extractRoles(
|
function extractRoles(
|
||||||
rolesObj?: Record<string, Record<string, string>>
|
rolesObj?: Record<string, Record<string, string>>
|
||||||
): PlatformRole[] {
|
): Role[] {
|
||||||
if (!rolesObj) return [];
|
if (!rolesObj) return [];
|
||||||
return Object.keys(rolesObj) as PlatformRole[];
|
return Object.keys(rolesObj) as Role[];
|
||||||
}
|
}
|
||||||
|
|
||||||
export const authConfig: NextAuthConfig = {
|
export const authConfig: NextAuthConfig = {
|
||||||
@@ -50,7 +61,7 @@ export const authConfig: NextAuthConfig = {
|
|||||||
return token;
|
return token;
|
||||||
},
|
},
|
||||||
async session({ session, token }) {
|
async session({ session, token }) {
|
||||||
const roles = (token.roles as PlatformRole[]) ?? [];
|
const roles = (token.roles as Role[]) ?? [];
|
||||||
const sessionUser: SessionUser = {
|
const sessionUser: SessionUser = {
|
||||||
id: token.sub!,
|
id: token.sub!,
|
||||||
name: session.user?.name ?? "",
|
name: session.user?.name ?? "",
|
||||||
@@ -58,7 +69,9 @@ export const authConfig: NextAuthConfig = {
|
|||||||
orgId: token.orgId as string,
|
orgId: token.orgId as string,
|
||||||
orgName: token.orgName as string,
|
orgName: token.orgName as string,
|
||||||
roles,
|
roles,
|
||||||
isPlatform: roles.some((r) => PLATFORM_ROLES.includes(r)),
|
isPlatform: roles.some((r) =>
|
||||||
|
PLATFORM_ROLES.includes(r as PlatformRole)
|
||||||
|
),
|
||||||
};
|
};
|
||||||
(session as any).platformUser = sessionUser;
|
(session as any).platformUser = sessionUser;
|
||||||
return session;
|
return session;
|
||||||
|
|||||||
@@ -55,6 +55,7 @@ const MIGRATION_SQL = `
|
|||||||
admin_notes TEXT,
|
admin_notes TEXT,
|
||||||
tenant_name TEXT,
|
tenant_name TEXT,
|
||||||
encrypted_secrets BYTEA,
|
encrypted_secrets BYTEA,
|
||||||
|
is_personal BOOLEAN NOT NULL DEFAULT FALSE,
|
||||||
created_at TIMESTAMPTZ NOT NULL DEFAULT now(),
|
created_at TIMESTAMPTZ NOT NULL DEFAULT now(),
|
||||||
updated_at TIMESTAMPTZ NOT NULL DEFAULT now()
|
updated_at TIMESTAMPTZ NOT NULL DEFAULT now()
|
||||||
);
|
);
|
||||||
@@ -70,6 +71,7 @@ const MIGRATION_SQL = `
|
|||||||
ALTER TABLE tenant_requests ADD COLUMN IF NOT EXISTS encrypted_secrets BYTEA;
|
ALTER TABLE tenant_requests ADD COLUMN IF NOT EXISTS encrypted_secrets BYTEA;
|
||||||
ALTER TABLE tenant_requests ADD COLUMN IF NOT EXISTS agents_md TEXT;
|
ALTER TABLE tenant_requests ADD COLUMN IF NOT EXISTS agents_md TEXT;
|
||||||
ALTER TABLE tenant_requests ADD COLUMN IF NOT EXISTS instance_name TEXT;
|
ALTER TABLE tenant_requests ADD COLUMN IF NOT EXISTS instance_name TEXT;
|
||||||
|
ALTER TABLE tenant_requests ADD COLUMN IF NOT EXISTS is_personal BOOLEAN NOT NULL DEFAULT FALSE;
|
||||||
|
|
||||||
-- Slice 3: drop the legacy 1-org-1-request constraint if it exists
|
-- Slice 3: drop the legacy 1-org-1-request constraint if it exists
|
||||||
ALTER TABLE tenant_requests DROP CONSTRAINT IF EXISTS tenant_requests_zitadel_org_id_key;
|
ALTER TABLE tenant_requests DROP CONSTRAINT IF EXISTS tenant_requests_zitadel_org_id_key;
|
||||||
@@ -156,8 +158,9 @@ export async function createTenantRequest(
|
|||||||
`INSERT INTO tenant_requests
|
`INSERT INTO tenant_requests
|
||||||
(zitadel_org_id, zitadel_user_id, company_name, instance_name,
|
(zitadel_org_id, zitadel_user_id, company_name, instance_name,
|
||||||
contact_name, contact_email, agent_name, soul_md, agents_md,
|
contact_name, contact_email, agent_name, soul_md, agents_md,
|
||||||
packages, billing_address, billing_notes, encrypted_secrets)
|
packages, billing_address, billing_notes, encrypted_secrets,
|
||||||
VALUES ($1, $2, $3, $4, $5, $6, $7, $8, $9, $10, $11, $12, $13)
|
is_personal)
|
||||||
|
VALUES ($1, $2, $3, $4, $5, $6, $7, $8, $9, $10, $11, $12, $13, $14)
|
||||||
RETURNING *`,
|
RETURNING *`,
|
||||||
[
|
[
|
||||||
params.zitadelOrgId,
|
params.zitadelOrgId,
|
||||||
@@ -173,6 +176,7 @@ export async function createTenantRequest(
|
|||||||
JSON.stringify(params.billingAddress),
|
JSON.stringify(params.billingAddress),
|
||||||
params.billingNotes,
|
params.billingNotes,
|
||||||
params.encryptedSecrets ?? null,
|
params.encryptedSecrets ?? null,
|
||||||
|
params.isPersonal ?? false,
|
||||||
]
|
]
|
||||||
);
|
);
|
||||||
return mapRow(result.rows[0]);
|
return mapRow(result.rows[0]);
|
||||||
@@ -408,6 +412,7 @@ function mapRow(row: any): TenantRequest {
|
|||||||
adminNotes: row.admin_notes,
|
adminNotes: row.admin_notes,
|
||||||
tenantName: row.tenant_name,
|
tenantName: row.tenant_name,
|
||||||
encryptedSecrets: row.encrypted_secrets ?? null,
|
encryptedSecrets: row.encrypted_secrets ?? null,
|
||||||
|
isPersonal: row.is_personal ?? false,
|
||||||
createdAt: row.created_at?.toISOString?.() ?? row.created_at,
|
createdAt: row.created_at?.toISOString?.() ?? row.created_at,
|
||||||
updatedAt: row.updated_at?.toISOString?.() ?? row.updated_at,
|
updatedAt: row.updated_at?.toISOString?.() ?? row.updated_at,
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -140,6 +140,12 @@ export function isPublicEmailDomain(domain: string): boolean {
|
|||||||
* Look up active tenant_requests whose contact_email shares the given domain.
|
* Look up active tenant_requests whose contact_email shares the given domain.
|
||||||
* Active = status NOT IN ('rejected', 'deleted').
|
* Active = status NOT IN ('rejected', 'deleted').
|
||||||
*
|
*
|
||||||
|
* Slice 4: personal-account rows (is_personal = TRUE) are excluded. A
|
||||||
|
* person's personal account doesn't claim the domain on behalf of a
|
||||||
|
* company — alice@acme.ch registering as a personal account must not
|
||||||
|
* block the actual Acme GmbH from registering later. The personal flag
|
||||||
|
* lives on the row itself, set by /api/register at creation time.
|
||||||
|
*
|
||||||
* Uses LOWER() on both sides to handle any historical case inconsistency in
|
* Uses LOWER() on both sides to handle any historical case inconsistency in
|
||||||
* stored emails. The pattern '%@<domain>' is anchored so 'acme.ch' does not
|
* stored emails. The pattern '%@<domain>' is anchored so 'acme.ch' does not
|
||||||
* match 'notacme.ch' or 'acme.ch.evil.com'.
|
* match 'notacme.ch' or 'acme.ch.evil.com'.
|
||||||
@@ -151,7 +157,8 @@ async function findDuplicateInDb(
|
|||||||
const result = await pool.query<{ count: string }>(
|
const result = await pool.query<{ count: string }>(
|
||||||
`SELECT COUNT(*) AS count FROM tenant_requests
|
`SELECT COUNT(*) AS count FROM tenant_requests
|
||||||
WHERE LOWER(contact_email) LIKE $1
|
WHERE LOWER(contact_email) LIKE $1
|
||||||
AND status NOT IN ('rejected', 'deleted')`,
|
AND status NOT IN ('rejected', 'deleted')
|
||||||
|
AND is_personal = FALSE`,
|
||||||
[`%@${domain.toLowerCase()}`]
|
[`%@${domain.toLowerCase()}`]
|
||||||
);
|
);
|
||||||
return Number(result.rows[0]?.count ?? 0) > 0;
|
return Number(result.rows[0]?.count ?? 0) > 0;
|
||||||
|
|||||||
40
src/lib/personal-org.ts
Normal file
40
src/lib/personal-org.ts
Normal file
@@ -0,0 +1,40 @@
|
|||||||
|
/**
|
||||||
|
* Personal-account helpers.
|
||||||
|
*
|
||||||
|
* Slice 4 establishes the convention that ZITADEL org names for personal
|
||||||
|
* accounts end with the literal " (Personal)" suffix. This file
|
||||||
|
* centralises the suffix and the predicate so both registration (which
|
||||||
|
* sets the suffix) and onboarding (which reads it from the session) use
|
||||||
|
* the same canonical form.
|
||||||
|
*
|
||||||
|
* Why a name suffix and not ZITADEL org metadata?
|
||||||
|
* -----------------------------------------------
|
||||||
|
* 1. The suffix is visible in ZITADEL Console, admin tools, JWT claims,
|
||||||
|
* etc. — useful debugging signal at zero cost.
|
||||||
|
* 2. Customers cannot rename their own org (requires IAM_OWNER, which
|
||||||
|
* only the SA holds), so the suffix is stable for the lifetime of
|
||||||
|
* the org.
|
||||||
|
* 3. No extra ZITADEL API calls at onboarding time to fetch metadata.
|
||||||
|
* 4. No extra portal DB tables.
|
||||||
|
*
|
||||||
|
* The trade-off: an admin who manually renames a personal org via
|
||||||
|
* ZITADEL Console could remove the suffix, after which onboarding
|
||||||
|
* would treat that org as a company. That's a deliberate destructive
|
||||||
|
* action and the worst outcome is a misnamed K8s CR; nothing breaks.
|
||||||
|
*/
|
||||||
|
|
||||||
|
export const PERSONAL_ORG_SUFFIX = " (Personal)";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns true when the given ZITADEL org name marks a personal account.
|
||||||
|
*
|
||||||
|
* The check is exact-suffix match (after trimming). Whitespace inside
|
||||||
|
* the suffix is significant — `" (personal)"` lowercase or `"(Personal)"`
|
||||||
|
* without the leading space are not matches and not personal orgs.
|
||||||
|
*
|
||||||
|
* Pass `session.orgName` from the SessionUser at the call site.
|
||||||
|
*/
|
||||||
|
export function isPersonalOrgName(orgName: string | null | undefined): boolean {
|
||||||
|
if (!orgName) return false;
|
||||||
|
return orgName.trimEnd().endsWith(PERSONAL_ORG_SUFFIX);
|
||||||
|
}
|
||||||
@@ -1,19 +1,87 @@
|
|||||||
import { auth } from "@/lib/auth";
|
import { auth } from "@/lib/auth";
|
||||||
import type { SessionUser } from "@/types";
|
import type { SessionUser } from "@/types";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Read-only session lookup. Returns the SessionUser stashed on the
|
||||||
|
* NextAuth session by `auth.ts::callbacks.session`, or null if there
|
||||||
|
* is no authenticated session.
|
||||||
|
*/
|
||||||
export async function getSessionUser(): Promise<SessionUser | null> {
|
export async function getSessionUser(): Promise<SessionUser | null> {
|
||||||
const session = await auth();
|
const session = await auth();
|
||||||
return (session as any)?.platformUser ?? null;
|
return (session as any)?.platformUser ?? null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Throws if there is no authenticated session. Otherwise returns the
|
||||||
|
* SessionUser. Use at the top of any handler that requires a logged-in
|
||||||
|
* user regardless of role.
|
||||||
|
*/
|
||||||
export async function requireSession(): Promise<SessionUser> {
|
export async function requireSession(): Promise<SessionUser> {
|
||||||
const user = await getSessionUser();
|
const user = await getSessionUser();
|
||||||
if (!user) throw new Error("Unauthorized");
|
if (!user) throw new Error("Unauthorized");
|
||||||
return user;
|
return user;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Throws unless the caller has a platform-level role
|
||||||
|
* (platform_admin or platform_operator). Use to gate /api/admin/*
|
||||||
|
* routes — these handle ANY customer's org and must not be accessible
|
||||||
|
* to customer-role users.
|
||||||
|
*/
|
||||||
export async function requirePlatformRole(): Promise<SessionUser> {
|
export async function requirePlatformRole(): Promise<SessionUser> {
|
||||||
const user = await requireSession();
|
const user = await requireSession();
|
||||||
if (!user.isPlatform) throw new Error("Forbidden: platform role required");
|
if (!user.isPlatform) throw new Error("Forbidden: platform role required");
|
||||||
return user;
|
return user;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// ---------------------------------------------------------------------------
|
||||||
|
// Slice 5: role predicates and gates
|
||||||
|
// ---------------------------------------------------------------------------
|
||||||
|
//
|
||||||
|
// Naming convention: `is*` are pure predicates over a SessionUser,
|
||||||
|
// safe to call inline in JSX/server components. `require*` throw on
|
||||||
|
// failure and are meant for the top of route handlers.
|
||||||
|
|
||||||
|
/**
|
||||||
|
* True when the user is a platform admin/operator OR holds the
|
||||||
|
* `owner` customer role on their org.
|
||||||
|
*
|
||||||
|
* This is the single check for "can mutate". Platform users always
|
||||||
|
* win because they administer all orgs cross-cut. Customer-side, only
|
||||||
|
* `owner` may mutate; `user` (and any future read-only customer role)
|
||||||
|
* cannot.
|
||||||
|
*/
|
||||||
|
export function canMutate(user: SessionUser): boolean {
|
||||||
|
return user.isPlatform || user.roles.includes("owner");
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* True when the user holds the customer `owner` role on their org.
|
||||||
|
* Excludes platform users — use {@link canMutate} when both should
|
||||||
|
* be allowed.
|
||||||
|
*
|
||||||
|
* Useful for permissions that are specifically about "this customer's
|
||||||
|
* own owner", e.g. "owner can invite users into their own org" — a
|
||||||
|
* platform user shouldn't be casually inviting users into a customer
|
||||||
|
* org, that's an admin-console action and goes through different
|
||||||
|
* tooling.
|
||||||
|
*/
|
||||||
|
export function isCustomerOwner(user: SessionUser): boolean {
|
||||||
|
return !user.isPlatform && user.roles.includes("owner");
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Throws unless `canMutate(user) === true`. Use at the top of any
|
||||||
|
* mutating customer-side handler.
|
||||||
|
*
|
||||||
|
* The thrown error message is intentionally generic — handlers
|
||||||
|
* should catch and translate to a 403 JSON response so the client
|
||||||
|
* doesn't see a stack trace.
|
||||||
|
*/
|
||||||
|
export async function requireOwnerRole(): Promise<SessionUser> {
|
||||||
|
const user = await requireSession();
|
||||||
|
if (!canMutate(user)) {
|
||||||
|
throw new Error("Forbidden: owner role required");
|
||||||
|
}
|
||||||
|
return user;
|
||||||
|
}
|
||||||
|
|||||||
@@ -35,7 +35,9 @@
|
|||||||
"successTitle": "Registrierung eingegangen",
|
"successTitle": "Registrierung eingegangen",
|
||||||
"successDescription": "Sie erhalten eine Einladungs-E-Mail mit einem Link, um Ihr Passwort festzulegen und Ihre E-Mail-Adresse zu bestätigen. Danach können Sie sich anmelden und Ihren KI-Assistenten einrichten.",
|
"successDescription": "Sie erhalten eine Einladungs-E-Mail mit einem Link, um Ihr Passwort festzulegen und Ihre E-Mail-Adresse zu bestätigen. Danach können Sie sich anmelden und Ihren KI-Assistenten einrichten.",
|
||||||
"goToLogin": "Zur Anmeldung",
|
"goToLogin": "Zur Anmeldung",
|
||||||
"duplicateDomain": "Für die E-Mail-Domain {domain} ist bereits ein Konto registriert. Bitte wenden Sie sich an Ihren Firmenadministrator, um eingeladen zu werden, oder kontaktieren Sie den PieCed-IT-Support, falls dies ein Fehler ist."
|
"duplicateDomain": "Für die E-Mail-Domain {domain} ist bereits ein Konto registriert. Bitte wenden Sie sich an Ihren Firmenadministrator, um eingeladen zu werden, oder kontaktieren Sie den PieCed-IT-Support, falls dies ein Fehler ist.",
|
||||||
|
"individualToggle": "Als Privatperson registrieren",
|
||||||
|
"individualHint": "Aktivieren Sie diese Option, wenn Sie sich nicht im Namen eines Unternehmens registrieren. Ihr Konto wird als persönlicher Arbeitsbereich eingerichtet."
|
||||||
},
|
},
|
||||||
"onboarding": {
|
"onboarding": {
|
||||||
"loading": "Status wird geladen…",
|
"loading": "Status wird geladen…",
|
||||||
@@ -101,7 +103,8 @@
|
|||||||
"instances": "Ihre Instanzen",
|
"instances": "Ihre Instanzen",
|
||||||
"inflightRequests": "Laufende Anfragen",
|
"inflightRequests": "Laufende Anfragen",
|
||||||
"createInstance": "Neue Instanz erstellen",
|
"createInstance": "Neue Instanz erstellen",
|
||||||
"createInstanceDescription": "Eine weitere KI-Assistent-Instanz für Ihre Organisation bereitstellen. Die Anfrage wird von einem Administrator geprüft, bevor die Instanz erstellt wird."
|
"createInstanceDescription": "Eine weitere KI-Assistent-Instanz für Ihre Organisation bereitstellen. Die Anfrage wird von einem Administrator geprüft, bevor die Instanz erstellt wird.",
|
||||||
|
"noAccessNoInstances": "Ihre Organisation hat noch keine Instanzen. Bitte bitten Sie den Eigentümer der Organisation, eine einzurichten."
|
||||||
},
|
},
|
||||||
"tenantDetail": {
|
"tenantDetail": {
|
||||||
"agent": "Agent",
|
"agent": "Agent",
|
||||||
@@ -177,7 +180,9 @@
|
|||||||
},
|
},
|
||||||
"documentProcessing": {
|
"documentProcessing": {
|
||||||
"description": "Aktivieren Sie Dokumentenverarbeitung, Zusammenfassung und Extraktion."
|
"description": "Aktivieren Sie Dokumentenverarbeitung, Zusammenfassung und Extraktion."
|
||||||
}
|
},
|
||||||
|
"statusEnabled": "aktiviert",
|
||||||
|
"statusDisabled": "deaktiviert"
|
||||||
},
|
},
|
||||||
"admin": {
|
"admin": {
|
||||||
"title": "Plattform-Admin",
|
"title": "Plattform-Admin",
|
||||||
|
|||||||
@@ -35,7 +35,9 @@
|
|||||||
"successTitle": "Registration received",
|
"successTitle": "Registration received",
|
||||||
"successDescription": "You will receive an invitation email with a link to set your password and verify your email address. Once completed, you can sign in to set up your AI assistant.",
|
"successDescription": "You will receive an invitation email with a link to set your password and verify your email address. Once completed, you can sign in to set up your AI assistant.",
|
||||||
"goToLogin": "Go to Sign In",
|
"goToLogin": "Go to Sign In",
|
||||||
"duplicateDomain": "An account for the email domain {domain} is already registered. Please contact your company administrator to be invited, or reach out to PieCed IT support if you believe this is in error."
|
"duplicateDomain": "An account for the email domain {domain} is already registered. Please contact your company administrator to be invited, or reach out to PieCed IT support if you believe this is in error.",
|
||||||
|
"individualToggle": "Register as an individual",
|
||||||
|
"individualHint": "Tick this if you're not registering on behalf of a company. Your account will be set up as a personal workspace."
|
||||||
},
|
},
|
||||||
"onboarding": {
|
"onboarding": {
|
||||||
"loading": "Loading status…",
|
"loading": "Loading status…",
|
||||||
@@ -101,7 +103,8 @@
|
|||||||
"instances": "Your instances",
|
"instances": "Your instances",
|
||||||
"inflightRequests": "In-flight requests",
|
"inflightRequests": "In-flight requests",
|
||||||
"createInstance": "Create new instance",
|
"createInstance": "Create new instance",
|
||||||
"createInstanceDescription": "Provision an additional AI assistant instance for your organization. The request will be reviewed by an administrator before the instance is created."
|
"createInstanceDescription": "Provision an additional AI assistant instance for your organization. The request will be reviewed by an administrator before the instance is created.",
|
||||||
|
"noAccessNoInstances": "Your organization doesn't have any instances yet. Please ask the organization owner to set one up."
|
||||||
},
|
},
|
||||||
"tenantDetail": {
|
"tenantDetail": {
|
||||||
"agent": "Agent",
|
"agent": "Agent",
|
||||||
@@ -177,7 +180,9 @@
|
|||||||
},
|
},
|
||||||
"documentProcessing": {
|
"documentProcessing": {
|
||||||
"description": "Enable document parsing, summarization, and extraction."
|
"description": "Enable document parsing, summarization, and extraction."
|
||||||
}
|
},
|
||||||
|
"statusEnabled": "enabled",
|
||||||
|
"statusDisabled": "disabled"
|
||||||
},
|
},
|
||||||
"admin": {
|
"admin": {
|
||||||
"title": "Platform Admin",
|
"title": "Platform Admin",
|
||||||
|
|||||||
@@ -35,7 +35,9 @@
|
|||||||
"successTitle": "Inscription reçue",
|
"successTitle": "Inscription reçue",
|
||||||
"successDescription": "Vous recevrez un e-mail d'invitation avec un lien pour définir votre mot de passe et vérifier votre adresse e-mail. Ensuite, vous pourrez vous connecter et configurer votre assistant IA.",
|
"successDescription": "Vous recevrez un e-mail d'invitation avec un lien pour définir votre mot de passe et vérifier votre adresse e-mail. Ensuite, vous pourrez vous connecter et configurer votre assistant IA.",
|
||||||
"goToLogin": "Aller à la connexion",
|
"goToLogin": "Aller à la connexion",
|
||||||
"duplicateDomain": "Un compte pour le domaine de courriel {domain} est déjà enregistré. Veuillez contacter l'administrateur de votre entreprise pour être invité, ou contactez le support PieCed IT si vous pensez qu'il s'agit d'une erreur."
|
"duplicateDomain": "Un compte pour le domaine de courriel {domain} est déjà enregistré. Veuillez contacter l'administrateur de votre entreprise pour être invité, ou contactez le support PieCed IT si vous pensez qu'il s'agit d'une erreur.",
|
||||||
|
"individualToggle": "S'inscrire en tant que particulier",
|
||||||
|
"individualHint": "Cochez cette case si vous ne vous inscrivez pas au nom d'une entreprise. Votre compte sera configuré comme espace de travail personnel."
|
||||||
},
|
},
|
||||||
"onboarding": {
|
"onboarding": {
|
||||||
"loading": "Chargement du statut…",
|
"loading": "Chargement du statut…",
|
||||||
@@ -101,7 +103,8 @@
|
|||||||
"instances": "Vos instances",
|
"instances": "Vos instances",
|
||||||
"inflightRequests": "Demandes en cours",
|
"inflightRequests": "Demandes en cours",
|
||||||
"createInstance": "Créer une nouvelle instance",
|
"createInstance": "Créer une nouvelle instance",
|
||||||
"createInstanceDescription": "Provisionner une instance supplémentaire d'assistant IA pour votre organisation. La demande sera examinée par un administrateur avant la création de l'instance."
|
"createInstanceDescription": "Provisionner une instance supplémentaire d'assistant IA pour votre organisation. La demande sera examinée par un administrateur avant la création de l'instance.",
|
||||||
|
"noAccessNoInstances": "Votre organisation n'a pas encore d'instances. Demandez au propriétaire de l'organisation d'en configurer une."
|
||||||
},
|
},
|
||||||
"tenantDetail": {
|
"tenantDetail": {
|
||||||
"agent": "Agent",
|
"agent": "Agent",
|
||||||
@@ -177,7 +180,9 @@
|
|||||||
},
|
},
|
||||||
"documentProcessing": {
|
"documentProcessing": {
|
||||||
"description": "Activez l'analyse, le résumé et l'extraction de documents."
|
"description": "Activez l'analyse, le résumé et l'extraction de documents."
|
||||||
}
|
},
|
||||||
|
"statusEnabled": "activé",
|
||||||
|
"statusDisabled": "désactivé"
|
||||||
},
|
},
|
||||||
"admin": {
|
"admin": {
|
||||||
"title": "Admin plateforme",
|
"title": "Admin plateforme",
|
||||||
|
|||||||
@@ -35,7 +35,9 @@
|
|||||||
"successTitle": "Registrazione ricevuta",
|
"successTitle": "Registrazione ricevuta",
|
||||||
"successDescription": "Riceverai un'e-mail di invito con un link per impostare la password e verificare il tuo indirizzo e-mail. Dopodiché potrai accedere e configurare il tuo assistente IA.",
|
"successDescription": "Riceverai un'e-mail di invito con un link per impostare la password e verificare il tuo indirizzo e-mail. Dopodiché potrai accedere e configurare il tuo assistente IA.",
|
||||||
"goToLogin": "Vai all'accesso",
|
"goToLogin": "Vai all'accesso",
|
||||||
"duplicateDomain": "Un account per il dominio e-mail {domain} è già registrato. Contatta l'amministratore della tua azienda per essere invitato, oppure contatta il supporto PieCed IT se ritieni che si tratti di un errore."
|
"duplicateDomain": "Un account per il dominio e-mail {domain} è già registrato. Contatta l'amministratore della tua azienda per essere invitato, oppure contatta il supporto PieCed IT se ritieni che si tratti di un errore.",
|
||||||
|
"individualToggle": "Registrati come privato",
|
||||||
|
"individualHint": "Seleziona questa opzione se non ti stai registrando per conto di un'azienda. Il tuo account sarà configurato come area di lavoro personale."
|
||||||
},
|
},
|
||||||
"onboarding": {
|
"onboarding": {
|
||||||
"loading": "Caricamento stato…",
|
"loading": "Caricamento stato…",
|
||||||
@@ -101,7 +103,8 @@
|
|||||||
"instances": "Le tue istanze",
|
"instances": "Le tue istanze",
|
||||||
"inflightRequests": "Richieste in corso",
|
"inflightRequests": "Richieste in corso",
|
||||||
"createInstance": "Crea nuova istanza",
|
"createInstance": "Crea nuova istanza",
|
||||||
"createInstanceDescription": "Effettua il provisioning di un'ulteriore istanza dell'assistente IA per la tua organizzazione. La richiesta sarà esaminata da un amministratore prima della creazione dell'istanza."
|
"createInstanceDescription": "Effettua il provisioning di un'ulteriore istanza dell'assistente IA per la tua organizzazione. La richiesta sarà esaminata da un amministratore prima della creazione dell'istanza.",
|
||||||
|
"noAccessNoInstances": "La tua organizzazione non ha ancora istanze. Chiedi al proprietario dell'organizzazione di configurarne una."
|
||||||
},
|
},
|
||||||
"tenantDetail": {
|
"tenantDetail": {
|
||||||
"agent": "Agente",
|
"agent": "Agente",
|
||||||
@@ -177,7 +180,9 @@
|
|||||||
},
|
},
|
||||||
"documentProcessing": {
|
"documentProcessing": {
|
||||||
"description": "Attiva l'analisi, il riassunto e l'estrazione di documenti."
|
"description": "Attiva l'analisi, il riassunto e l'estrazione di documenti."
|
||||||
}
|
},
|
||||||
|
"statusEnabled": "abilitato",
|
||||||
|
"statusDisabled": "disabilitato"
|
||||||
},
|
},
|
||||||
"admin": {
|
"admin": {
|
||||||
"title": "Admin piattaforma",
|
"title": "Admin piattaforma",
|
||||||
|
|||||||
@@ -5,12 +5,39 @@ export interface ZitadelClaims {
|
|||||||
"urn:zitadel:iam:org:project:roles"?: Record<string, Record<string, string>>;
|
"urn:zitadel:iam:org:project:roles"?: Record<string, Record<string, string>>;
|
||||||
}
|
}
|
||||||
|
|
||||||
export type PlatformRole =
|
/**
|
||||||
| "platform_admin"
|
* Platform-level roles, granted to PieCed staff only. Hold the IAM-level
|
||||||
| "platform_operator"
|
* authority to administer the entire installation regardless of which
|
||||||
| "owner"
|
* customer org a request lands on.
|
||||||
| "user"
|
*/
|
||||||
| "viewer";
|
export type PlatformRole = "platform_admin" | "platform_operator";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Customer-level roles, granted by ZITADEL project authorizations on
|
||||||
|
* each customer org's "OpenClaw Platform" project grant.
|
||||||
|
*
|
||||||
|
* Slice 5 dropped the previously-defined `viewer` role. With the portal
|
||||||
|
* acting purely as a control plane (the assistant itself runs at
|
||||||
|
* separate URLs with their own auth), `user` and `viewer` collapsed
|
||||||
|
* to the same surface — read-only access to instance state and usage.
|
||||||
|
*
|
||||||
|
* - `owner` can mutate (packages, workspace files, channel users,
|
||||||
|
* instance creation, member invites in Slice 7).
|
||||||
|
* - `user` is read-only in the portal. From Slice 6 onwards `user`
|
||||||
|
* visibility is also narrowed to assigned tenants only.
|
||||||
|
*/
|
||||||
|
export type CustomerRole = "owner" | "user";
|
||||||
|
|
||||||
|
/** Union of all roles a JWT can carry. */
|
||||||
|
export type Role = PlatformRole | CustomerRole;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @deprecated Use {@link Role} for the union, or {@link PlatformRole}
|
||||||
|
* / {@link CustomerRole} when you mean a specific subset.
|
||||||
|
* Kept as a re-export only so existing imports don't
|
||||||
|
* explode in mid-migration commits.
|
||||||
|
*/
|
||||||
|
export type LegacyPlatformRole = Role;
|
||||||
|
|
||||||
export interface SessionUser {
|
export interface SessionUser {
|
||||||
id: string;
|
id: string;
|
||||||
@@ -18,7 +45,7 @@ export interface SessionUser {
|
|||||||
email: string;
|
email: string;
|
||||||
orgId: string;
|
orgId: string;
|
||||||
orgName: string;
|
orgName: string;
|
||||||
roles: PlatformRole[];
|
roles: Role[];
|
||||||
isPlatform: boolean;
|
isPlatform: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -83,11 +110,23 @@ export interface UsageSummary {
|
|||||||
|
|
||||||
// Registration
|
// Registration
|
||||||
export interface RegistrationInput {
|
export interface RegistrationInput {
|
||||||
companyName: string;
|
/**
|
||||||
|
* Required for company registrations. Ignored when `isPersonal` is true —
|
||||||
|
* the server then derives the ZITADEL org name from the user's full name
|
||||||
|
* with a "(Personal)" suffix.
|
||||||
|
*/
|
||||||
|
companyName?: string;
|
||||||
givenName: string;
|
givenName: string;
|
||||||
familyName: string;
|
familyName: string;
|
||||||
email: string;
|
email: string;
|
||||||
preferredLanguage?: string;
|
preferredLanguage?: string;
|
||||||
|
/**
|
||||||
|
* Slice 4: when true, registration creates a personal account (one
|
||||||
|
* person, no company). Domain-uniqueness check is skipped, ZITADEL org
|
||||||
|
* is named "{givenName} {familyName} (Personal)", subsequent tenants
|
||||||
|
* are named with the `p-{requestId[:8]}` convention.
|
||||||
|
*/
|
||||||
|
isPersonal?: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Billing address
|
// Billing address
|
||||||
@@ -131,6 +170,13 @@ export interface TenantRequest {
|
|||||||
adminNotes?: string;
|
adminNotes?: string;
|
||||||
tenantName?: string;
|
tenantName?: string;
|
||||||
encryptedSecrets?: Buffer | null;
|
encryptedSecrets?: Buffer | null;
|
||||||
|
/**
|
||||||
|
* Slice 4: true for personal accounts. Drives CR-naming (`p-{suffix}`
|
||||||
|
* vs `{slug}-{suffix}` in `lib/tenant-naming.ts`), display-name
|
||||||
|
* fallback (contact name vs company name), and exclusion from the
|
||||||
|
* domain-uniqueness check on subsequent registrations.
|
||||||
|
*/
|
||||||
|
isPersonal?: boolean;
|
||||||
createdAt: string;
|
createdAt: string;
|
||||||
updatedAt: string;
|
updatedAt: string;
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user