Compare commits
8 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| c0ff22394c | |||
| 395d2f43cc | |||
| 6f42b56ad5 | |||
| 85c4302f7a | |||
| 726151d90b | |||
| a13af83655 | |||
| b58bdadad4 | |||
| d375a099f0 |
@@ -17,3 +17,7 @@ LITELLM_MASTER_KEY=
|
||||
|
||||
# Portal Database (CloudNativePG)
|
||||
DATABASE_URL=postgresql://portal:${PORTAL_DB_PASSWORD}@portal-db-rw.pieced-system.svc:5432/portal
|
||||
|
||||
# Threema relay (central pieced-threema-gateway)
|
||||
THREEMA_RELAY_URL=http://pieced-threema-gateway.threema-gateway.svc:8080
|
||||
THREEMA_RELAY_ADMIN_TOKEN=__from_openbao__
|
||||
|
||||
134
README.md
134
README.md
@@ -1,100 +1,66 @@
|
||||
# PieCed Portal
|
||||
# Threema UX v2 — QR available on demand
|
||||
|
||||
Customer self-service portal for the PieCed IT multi-tenant OpenClaw platform.
|
||||
Replaces the previous attempt that had the QR rendering inline above
|
||||
the channel help text. That placement didn't work for you in practice
|
||||
because it wasn't visible when you actually wanted it.
|
||||
|
||||
## Stack
|
||||
## What this does instead
|
||||
|
||||
| Layer | Choice |
|
||||
|-------|--------|
|
||||
| Framework | Next.js 15 LTS (App Router, standalone output, Turbopack) |
|
||||
| Auth | NextAuth v5 + ZITADEL OIDC (CODE flow) |
|
||||
| Tenant mgmt | Direct K8s API → `PiecedTenant` CRs (Option A) |
|
||||
| Usage data | LiteLLM `/team/info` + `/global/spend/logs` |
|
||||
| i18n | next-intl 4.x (en/de) |
|
||||
| Styling | Tailwind CSS 4 |
|
||||
| Deployment | Container in `pieced-system`, exposed at `app.pieced.ch` |
|
||||
1. **"Show QR" link** next to the channel title in the threema card —
|
||||
visible at all times, clickable any time you want the QR.
|
||||
|
||||
## Setup
|
||||
2. **Auto-opens the modal** the first time you focus the add-ID input
|
||||
on the page (so a new user adding their first ID sees it without
|
||||
needing to click "Show QR" themselves). Doesn't re-pop after
|
||||
dismissal — the link covers re-opens.
|
||||
|
||||
### 1. ZITADEL Application
|
||||
3. **Modal** with QR + 3-step instructions + "AIAGENT" label. Plain
|
||||
`<img>` (no next/image), closes on ESC / overlay click / × button.
|
||||
|
||||
In ZITADEL console (`auth.pieced.ch`), project "OpenClaw Platform":
|
||||
The earlier `threema-setup.tsx` component file is removed — replaced by
|
||||
`threema-qr-modal.tsx`.
|
||||
|
||||
1. Create Application → **PieCed Portal** → Web → Authentication Method: **CODE**
|
||||
2. Redirect URI: `https://app.pieced.ch/api/auth/callback/zitadel`
|
||||
3. Post-logout URI: `https://app.pieced.ch/login`
|
||||
4. Note Client ID and Client Secret
|
||||
## Files
|
||||
|
||||
### 2. OpenBao Secrets
|
||||
```
|
||||
src/lib/threema-gateway-config.ts # unchanged from before — central gateway constants
|
||||
src/components/channel-users/threema-qr-modal.tsx # NEW — the modal
|
||||
src/components/channel-users/channel-users.tsx # MODIFIED — Show QR button + focus auto-open + modal mount
|
||||
deploy/patch-i18n-threema.mjs # adds threemaSetup.showQr label in 4 langs
|
||||
public/threema/qr_code_AIAGENT.png # unchanged
|
||||
```
|
||||
|
||||
## Apply
|
||||
|
||||
```bash
|
||||
bao kv put pieced/portal/oidc \
|
||||
client_id="<from step 1>" \
|
||||
client_secret="<from step 1>" \
|
||||
nextauth_secret="$(openssl rand -base64 32)"
|
||||
cd /path/to/pieced-portal
|
||||
|
||||
# Remove the old inline component if you applied the previous archive
|
||||
rm -f src/components/channel-users/threema-setup.tsx
|
||||
|
||||
# Drop new + modified files
|
||||
unzip -o /path/to/threema-ux-v2.zip
|
||||
|
||||
# Update messages
|
||||
node deploy/patch-i18n-threema.mjs
|
||||
|
||||
# TS check
|
||||
npx tsc --noEmit
|
||||
|
||||
git add -A
|
||||
git commit -m "Threema QR: on-demand modal + auto-open on first add"
|
||||
git push
|
||||
```
|
||||
|
||||
### 3. Build & Push
|
||||
|
||||
```bash
|
||||
docker build -t registry.c5ai.ch/pieced/pieced-portal:0.1.0 .
|
||||
docker push registry.c5ai.ch/pieced/pieced-portal:0.1.0
|
||||
```
|
||||
|
||||
Update image tag in `pieced-gitops/apps/portal/deployment.yaml`, push, ArgoCD syncs.
|
||||
|
||||
### 4. DNS
|
||||
|
||||
Ensure `app.pieced.ch` A record → MetalLB ingress IP (or ExternalDNS handles it).
|
||||
|
||||
## Local Development
|
||||
|
||||
```bash
|
||||
cp .env.example .env.local
|
||||
# Fill in values — K8s client uses ~/.kube/config locally
|
||||
npm install
|
||||
npm run dev
|
||||
```
|
||||
|
||||
## Project Structure
|
||||
After redeploy, the threema card under Authorized Users shows:
|
||||
|
||||
```
|
||||
src/
|
||||
├── app/
|
||||
│ ├── api/
|
||||
│ │ ├── auth/[...nextauth]/route.ts # NextAuth handler
|
||||
│ │ ├── tenants/route.ts # Tenant CRUD (K8s API)
|
||||
│ │ └── usage/route.ts # Usage stub
|
||||
│ ├── [locale]/
|
||||
│ │ ├── layout.tsx # Locale layout + NavShell
|
||||
│ │ ├── page.tsx # Redirect → /dashboard
|
||||
│ │ ├── login/page.tsx # ZITADEL sign-in
|
||||
│ │ ├── dashboard/page.tsx # Customer dashboard
|
||||
│ │ └── admin/page.tsx # Platform admin tenant list
|
||||
│ ├── layout.tsx # Root layout
|
||||
│ └── globals.css # Tailwind 4 theme
|
||||
├── components/
|
||||
│ ├── layout/nav-shell.tsx # Header + navigation
|
||||
│ └── ui/ # Reusable UI components
|
||||
├── i18n/
|
||||
│ ├── routing.ts # next-intl 4.x routing config
|
||||
│ ├── navigation.ts # Localized Link, redirect, etc.
|
||||
│ └── request.ts # Server-side i18n config
|
||||
├── lib/
|
||||
│ ├── auth.ts # NextAuth v5 + ZITADEL config
|
||||
│ ├── k8s.ts # K8s client for PiecedTenant CRs
|
||||
│ ├── litellm.ts # LiteLLM API client
|
||||
│ └── session.ts # Session helpers
|
||||
├── messages/
|
||||
│ ├── en.json
|
||||
│ └── de.json
|
||||
└── types/index.ts # Shared TypeScript types
|
||||
threema [Show QR] 0 users
|
||||
────────────────────
|
||||
<help text: 'Enter your own Threema ID...'>
|
||||
────────────────────
|
||||
[ input: A8K2P3X7 ] [ Add ]
|
||||
^^^ focusing this opens the QR modal the first time
|
||||
```
|
||||
|
||||
## Session Roadmap
|
||||
|
||||
- **6.1** ← This session: scaffold, auth, basic pages
|
||||
- **6.2**: Instance management, package config, usage display
|
||||
- **6.3**: Onboarding flow (create ZITADEL org → PiecedTenant CR)
|
||||
- **6.4**: Workspace editor (SOUL.md, AGENTS.md, TOOLS.md)
|
||||
- **6.5**: Admin panel (tenant lifecycle, billing overview)
|
||||
Clicking "Show QR" or focusing the input → modal with QR + steps.
|
||||
|
||||
70
deploy/README-threema.md
Normal file
70
deploy/README-threema.md
Normal file
@@ -0,0 +1,70 @@
|
||||
# Wiring Threema relay into the portal
|
||||
|
||||
Drop-in files in this archive:
|
||||
|
||||
```
|
||||
src/lib/packages.ts # add 'threema' to catalog + customProvisioning flag
|
||||
src/lib/threema-relay.ts # new — admin API client
|
||||
src/app/api/tenants/[name]/threema/route.ts # new — POST provision / DELETE deprovision
|
||||
src/app/api/tenants/[name]/threema/routes/route.ts # new — atomic add/remove of a single Threema ID
|
||||
src/components/channel-users/channel-users.tsx # branch threema through relay-managed endpoint
|
||||
src/components/packages/package-card.tsx # handle customProvisioning enable/disable
|
||||
deploy/patch-i18n-threema.mjs # idempotent i18n key injection
|
||||
```
|
||||
|
||||
## Manual steps after dropping in
|
||||
|
||||
1. `.env` (and `.env.example`) — add:
|
||||
```
|
||||
THREEMA_RELAY_URL=http://pieced-threema-gateway.threema-gateway.svc:8080
|
||||
THREEMA_RELAY_ADMIN_TOKEN=__from_openbao__
|
||||
```
|
||||
The portal pod's OpenBao client should also read `secret/data/threema-gateway/admin` and surface `token` as this env var (existing ESO pattern in the portal's Helm chart).
|
||||
|
||||
2. Patch the message files (one-time):
|
||||
```bash
|
||||
node deploy/patch-i18n-threema.mjs
|
||||
```
|
||||
|
||||
3. Re-export `CHANNEL_PACKAGE_IDS` is unchanged in source; verify
|
||||
`tenants/[name]/page.tsx` still derives the enabled-channels list
|
||||
from it — it should now include `threema` automatically once a
|
||||
tenant has it in `spec.packages`.
|
||||
|
||||
4. Type-check:
|
||||
```bash
|
||||
npx tsc --noEmit
|
||||
```
|
||||
|
||||
## Flow summary
|
||||
|
||||
### Enabling Threema for a tenant
|
||||
1. Customer toggles the threema package card.
|
||||
2. PackageCard sees `customProvisioning: true` → POSTs `/api/tenants/<name>/threema`.
|
||||
3. Handler calls relay `POST /admin/tokens` → gets `{token, hmacSecret}`.
|
||||
4. Handler writes them to OpenBao at `secret/data/tenants/tenant-<name>/threema-relay`.
|
||||
5. PackageCard then PATCHes `tenant.spec.packages` to include `threema`.
|
||||
6. Operator reconciles: ExternalSecret syncs OpenBao → Secret; OpenClaw pod restarts with `THREEMA_RELAY_*` env vars; plugin registers `threema` channel.
|
||||
|
||||
### Customer adds a Threema ID
|
||||
1. UI calls `POST /api/tenants/<name>/threema/routes` with `{threemaId}`.
|
||||
2. Handler calls relay `POST /admin/routes` (uniqueness enforced at PK).
|
||||
3. On 201 or 409-from-same-tenant: handler patches K8s `spec.channelUsers.threema`.
|
||||
4. On 409-from-other-tenant: 409 to client with explanation.
|
||||
5. On K8s patch failure after relay success: handler compensates by `DELETE /admin/routes/...` at the relay.
|
||||
|
||||
### Customer removes a Threema ID
|
||||
1. UI calls `DELETE /api/tenants/<name>/threema/routes?threemaId=...`.
|
||||
2. Handler patches K8s `spec.channelUsers.threema` to drop the ID.
|
||||
3. Handler calls relay `DELETE /admin/routes/...` (404 = idempotent OK).
|
||||
4. If relay drop fails: K8s already updated, surface warning but treat as success — relay deletes are idempotent on retry.
|
||||
|
||||
### Disabling Threema for a tenant
|
||||
1. Customer disables the threema card.
|
||||
2. PackageCard DELETEs `/api/tenants/<name>/threema`.
|
||||
3. Handler calls relay `DELETE /admin/tokens/<name>` (cascades to all routes for this tenant).
|
||||
4. Handler deletes OpenBao secret at `secret/data/tenants/tenant-<name>/threema-relay`.
|
||||
5. PackageCard then PATCHes `tenant.spec.packages` to drop `threema`.
|
||||
6. Operator reconciles: ExternalSecret targets a missing OpenBao path → Secret deleted → OpenClaw pod restarts without `threema` channel.
|
||||
|
||||
There's a small window (between step 4 and the operator's reconcile) where the pod still thinks it has a relay token but the relay has revoked it. Outbound during that window returns 401 from the relay; inbound is blackholed at the relay because routes are gone. Both are graceful failures.
|
||||
122
deploy/patch-i18n-threema.mjs
Normal file
122
deploy/patch-i18n-threema.mjs
Normal file
@@ -0,0 +1,122 @@
|
||||
#!/usr/bin/env node
|
||||
/**
|
||||
* Run: node deploy/patch-i18n-threema.mjs
|
||||
*
|
||||
* Idempotently injects (or overwrites) customer-facing Threema texts:
|
||||
* - packages.threema.{description, instructions, disclaimer}
|
||||
* - channelUsers.threemaIdHelp
|
||||
* - channelUsers.threemaSetup.{title, step1, step2, step3, qrAlt}
|
||||
*
|
||||
* Replaces the earlier version of this script entirely. The new texts:
|
||||
* - Drop "Gateway account" jargon (customers don't know it)
|
||||
* - Drop asterisk-prefix references (customers don't see / type it)
|
||||
* - Tell the customer to add their OWN Threema ID, not someone else's
|
||||
* - Disclose that Threema charges per message via the gateway
|
||||
* - Walk through the QR-scan + add-your-ID flow explicitly
|
||||
*
|
||||
* Re-running is safe — keys are set, not merged, so this is the
|
||||
* source of truth for the values it touches.
|
||||
*/
|
||||
import { readFileSync, writeFileSync } from "fs";
|
||||
|
||||
const i18n = {
|
||||
en: {
|
||||
pkg: {
|
||||
description:
|
||||
"Send and receive messages through Threema. Each inbound and outbound message uses the shared PieCed messaging service and incurs a per-message charge from Threema — a third-party cost, separate from your PieCed subscription.",
|
||||
instructions:
|
||||
"1. Enable this package.\n2. Open Threema on your phone, scan the QR code shown under Authorized Users → threema, and accept the contact.\n3. Add your own Threema ID under Authorized Users → threema so the assistant recognises your messages.\n4. Send a message from Threema to start chatting with the assistant.",
|
||||
disclaimer:
|
||||
"Messages between Threema and PieCed are end-to-end encrypted up to PieCed's messaging service, where they are decrypted to be routed to your assistant. Each message sent or received is counted toward Threema's per-message billing — see your plan for current rates.",
|
||||
},
|
||||
channelHelp:
|
||||
"Enter your own Threema ID — the 8 characters shown in your Threema app under Settings → My Threema ID. Once added, you'll be able to chat with the assistant directly from Threema.",
|
||||
setup: {
|
||||
title: "Add the assistant to your Threema",
|
||||
step1: "Open Threema on your phone.",
|
||||
step2: "Tap the scan icon and scan this QR code to add the assistant as a contact.",
|
||||
step3: "Then add your own Threema ID below.",
|
||||
qrAlt: "QR code to add {gateway} as a Threema contact",
|
||||
showQr: "Show QR",
|
||||
},
|
||||
},
|
||||
de: {
|
||||
pkg: {
|
||||
description:
|
||||
"Senden und empfangen Sie Nachrichten über Threema. Jede eingehende und ausgehende Nachricht läuft über den gemeinsamen PieCed-Messaging-Dienst und verursacht eine Gebühr pro Nachricht bei Threema — eine Drittanbieter-Kostenposition, unabhängig von Ihrem PieCed-Abonnement.",
|
||||
instructions:
|
||||
"1. Aktivieren Sie dieses Paket.\n2. Öffnen Sie Threema auf Ihrem Telefon, scannen Sie den QR-Code unter Autorisierte Benutzer → threema und akzeptieren Sie den Kontakt.\n3. Tragen Sie Ihre eigene Threema-ID unter Autorisierte Benutzer → threema ein, damit der Assistent Ihre Nachrichten erkennt.\n4. Schreiben Sie eine Nachricht aus Threema, um das Gespräch zu beginnen.",
|
||||
disclaimer:
|
||||
"Nachrichten zwischen Threema und PieCed werden Ende-zu-Ende verschlüsselt bis zum PieCed-Messaging-Dienst, wo sie entschlüsselt und an Ihren Assistenten weitergeleitet werden. Jede gesendete oder empfangene Nachricht wird gemäss Threema-Tarif pro Nachricht abgerechnet — die aktuellen Preise finden Sie in Ihrem Plan.",
|
||||
},
|
||||
channelHelp:
|
||||
"Geben Sie Ihre eigene Threema-ID ein — die 8 Zeichen, die in Ihrer Threema-App unter Einstellungen → Meine Threema-ID angezeigt werden. Anschliessend können Sie direkt aus Threema mit dem Assistenten chatten.",
|
||||
setup: {
|
||||
title: "Assistenten zu Threema hinzufügen",
|
||||
step1: "Öffnen Sie Threema auf Ihrem Telefon.",
|
||||
step2: "Tippen Sie auf das Scan-Symbol und scannen Sie diesen QR-Code, um den Assistenten als Kontakt hinzuzufügen.",
|
||||
step3: "Fügen Sie anschliessend unten Ihre eigene Threema-ID hinzu.",
|
||||
qrAlt: "QR-Code, um {gateway} als Threema-Kontakt hinzuzufügen",
|
||||
showQr: "QR anzeigen",
|
||||
},
|
||||
},
|
||||
fr: {
|
||||
pkg: {
|
||||
description:
|
||||
"Envoyez et recevez des messages via Threema. Chaque message entrant ou sortant transite par le service de messagerie PieCed partagé et entraîne des frais par message facturés par Threema — un coût tiers, distinct de votre abonnement PieCed.",
|
||||
instructions:
|
||||
"1. Activez ce package.\n2. Ouvrez Threema sur votre téléphone, scannez le QR code affiché dans Utilisateurs autorisés → threema, puis acceptez le contact.\n3. Ajoutez votre propre identifiant Threema sous Utilisateurs autorisés → threema afin que l'assistant reconnaisse vos messages.\n4. Envoyez un message depuis Threema pour commencer la conversation.",
|
||||
disclaimer:
|
||||
"Les messages entre Threema et PieCed sont chiffrés de bout en bout jusqu'au service de messagerie PieCed, où ils sont déchiffrés pour être acheminés vers votre assistant. Chaque message envoyé ou reçu est facturé par Threema selon son tarif par message — consultez votre plan pour les tarifs en vigueur.",
|
||||
},
|
||||
channelHelp:
|
||||
"Saisissez votre propre identifiant Threema — les 8 caractères affichés dans votre application Threema sous Réglages → Mon identifiant Threema. Une fois ajouté, vous pourrez discuter directement avec l'assistant depuis Threema.",
|
||||
setup: {
|
||||
title: "Ajouter l'assistant à Threema",
|
||||
step1: "Ouvrez Threema sur votre téléphone.",
|
||||
step2: "Appuyez sur l'icône de scan et scannez ce QR code pour ajouter l'assistant comme contact.",
|
||||
step3: "Puis ajoutez votre propre identifiant Threema ci-dessous.",
|
||||
qrAlt: "QR code pour ajouter {gateway} comme contact Threema",
|
||||
showQr: "Afficher le QR",
|
||||
},
|
||||
},
|
||||
it: {
|
||||
pkg: {
|
||||
description:
|
||||
"Invia e ricevi messaggi tramite Threema. Ogni messaggio in entrata e in uscita passa attraverso il servizio di messaggistica condiviso di PieCed e comporta un addebito per messaggio da parte di Threema — un costo di terzi, separato dall'abbonamento PieCed.",
|
||||
instructions:
|
||||
"1. Attiva questo pacchetto.\n2. Apri Threema sul tuo telefono, scansiona il QR code mostrato in Utenti autorizzati → threema e accetta il contatto.\n3. Aggiungi il tuo ID Threema sotto Utenti autorizzati → threema affinché l'assistente riconosca i tuoi messaggi.\n4. Invia un messaggio da Threema per iniziare la conversazione.",
|
||||
disclaimer:
|
||||
"I messaggi tra Threema e PieCed sono cifrati end-to-end fino al servizio di messaggistica PieCed, dove vengono decifrati per essere inoltrati al tuo assistente. Ogni messaggio inviato o ricevuto viene addebitato da Threema secondo la sua tariffa per messaggio — consulta il tuo piano per i prezzi attuali.",
|
||||
},
|
||||
channelHelp:
|
||||
"Inserisci il tuo ID Threema — gli 8 caratteri mostrati nella tua app Threema sotto Impostazioni → Il mio ID Threema. Una volta aggiunto, potrai conversare con l'assistente direttamente da Threema.",
|
||||
setup: {
|
||||
title: "Aggiungi l'assistente a Threema",
|
||||
step1: "Apri Threema sul tuo telefono.",
|
||||
step2: "Tocca l'icona di scansione e scansiona questo QR code per aggiungere l'assistente ai contatti.",
|
||||
step3: "Quindi aggiungi il tuo ID Threema qui sotto.",
|
||||
qrAlt: "QR code per aggiungere {gateway} come contatto Threema",
|
||||
showQr: "Mostra QR",
|
||||
},
|
||||
},
|
||||
};
|
||||
|
||||
for (const [lang, entries] of Object.entries(i18n)) {
|
||||
const path = `src/messages/${lang}.json`;
|
||||
const json = JSON.parse(readFileSync(path, "utf8"));
|
||||
|
||||
json.packages = json.packages ?? {};
|
||||
json.packages.threema = {
|
||||
description: entries.pkg.description,
|
||||
instructions: entries.pkg.instructions,
|
||||
disclaimer: entries.pkg.disclaimer,
|
||||
};
|
||||
|
||||
json.channelUsers = json.channelUsers ?? {};
|
||||
json.channelUsers.threemaIdHelp = entries.channelHelp;
|
||||
json.channelUsers.threemaSetup = entries.setup;
|
||||
|
||||
writeFileSync(path, JSON.stringify(json, null, 2) + "\n");
|
||||
console.log(`Patched ${path}`);
|
||||
}
|
||||
BIN
public/threema/qr_code_AIAGENT.png
Normal file
BIN
public/threema/qr_code_AIAGENT.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 56 KiB |
71
src/app/[locale]/admin/openclaw/page.tsx
Normal file
71
src/app/[locale]/admin/openclaw/page.tsx
Normal file
@@ -0,0 +1,71 @@
|
||||
import { redirect } from "next/navigation";
|
||||
import { getTranslations } from "next-intl/server";
|
||||
import { getSessionUser } from "@/lib/session";
|
||||
import { listTenants, getOpenClawDefaults } from "@/lib/k8s";
|
||||
import { OpenClawAdminPanel } from "@/components/admin/openclaw-admin-panel";
|
||||
|
||||
/**
|
||||
* /admin/openclaw — platform-default OpenClaw image + per-tenant
|
||||
* overrides table.
|
||||
*
|
||||
* Two sections:
|
||||
* 1. Default — readable from `pieced-openclaw-config` ConfigMap.
|
||||
* Editable via the same form. Empty fields show as "(unset)"
|
||||
* and the operator falls back to its built-in default in that
|
||||
* case (intentionally invisible to the portal — the binary's
|
||||
* baked version moves with releases and we don't want the UI
|
||||
* to claim a misleading "current default").
|
||||
* 2. Tenant table — every tenant in the cluster with its current
|
||||
* override (or "follows default"). Clicking a row opens a small
|
||||
* inline editor.
|
||||
*
|
||||
* Authorization is gated server-side: `user.isPlatform` only. Any
|
||||
* other user gets redirected to /dashboard.
|
||||
*/
|
||||
export default async function OpenClawAdminPage() {
|
||||
const user = await getSessionUser();
|
||||
if (!user) redirect("/login");
|
||||
if (!user.isPlatform) redirect("/dashboard");
|
||||
const t = await getTranslations("openclawAdmin");
|
||||
|
||||
// Parallel fetch — defaults and tenants are independent.
|
||||
const [defaults, tenants] = await Promise.all([
|
||||
getOpenClawDefaults(),
|
||||
listTenants(),
|
||||
]);
|
||||
|
||||
// Sort tenants: overridden first (more interesting to review),
|
||||
// then alphabetically by display name. Helps the admin spot which
|
||||
// tenants are off the platform default at a glance.
|
||||
const sorted = [...tenants].sort((a, b) => {
|
||||
const aOverride = a.spec.openClawImage ? 1 : 0;
|
||||
const bOverride = b.spec.openClawImage ? 1 : 0;
|
||||
if (aOverride !== bOverride) return bOverride - aOverride;
|
||||
return (a.spec.displayName || a.metadata.name).localeCompare(
|
||||
b.spec.displayName || b.metadata.name
|
||||
);
|
||||
});
|
||||
|
||||
return (
|
||||
<main className="max-w-5xl mx-auto px-6 py-8">
|
||||
<div className="mb-8 animate-in">
|
||||
<h1 className="font-display text-2xl font-semibold accent-rule">
|
||||
{t("title")}
|
||||
</h1>
|
||||
<p className="text-sm text-text-secondary mt-3">{t("subtitle")}</p>
|
||||
</div>
|
||||
|
||||
<OpenClawAdminPanel
|
||||
initialDefaults={defaults}
|
||||
tenants={sorted.map((tn) => ({
|
||||
name: tn.metadata.name,
|
||||
displayName: tn.spec.displayName || tn.metadata.name,
|
||||
phase: tn.status?.phase ?? "Unknown",
|
||||
override: tn.spec.openClawImage?.tag
|
||||
? { tag: tn.spec.openClawImage.tag }
|
||||
: null,
|
||||
}))}
|
||||
/>
|
||||
</main>
|
||||
);
|
||||
}
|
||||
@@ -22,11 +22,22 @@ export default async function AdminPage() {
|
||||
|
||||
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("subtitle")}</p>
|
||||
<div className="mb-8 animate-in flex items-end justify-between gap-4 flex-wrap">
|
||||
<div>
|
||||
<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("subtitle")}</p>
|
||||
</div>
|
||||
{/* Sub-tools: links to other admin pages. Plain links rather
|
||||
than nav-shell entries — these are platform-team utilities,
|
||||
not main navigation. */}
|
||||
<a
|
||||
href="/admin/openclaw"
|
||||
className="text-sm px-4 py-2 rounded-lg border border-border text-text-secondary hover:text-text-primary hover:border-text-secondary transition-colors"
|
||||
>
|
||||
{t("openclawTool")}
|
||||
</a>
|
||||
</div>
|
||||
|
||||
<div className="animate-in animate-in-delay-1">
|
||||
|
||||
@@ -13,8 +13,15 @@ import { ChannelUsers } from "@/components/channel-users/channel-users";
|
||||
import { AssignedUsersPanel } from "@/components/tenants/assigned-users-panel";
|
||||
import { SubscriptionToggle } from "@/components/tenants/subscription-toggle";
|
||||
import { formatDateTime, formatRelative } from "@/lib/format";
|
||||
import { CHANNEL_PACKAGE_IDS } from "@/lib/packages";
|
||||
|
||||
const CHANNEL_PACKAGES = ["telegram", "discord", "email"];
|
||||
// CHANNEL_PACKAGES used to be a hardcoded literal here
|
||||
// (`["telegram", "discord", "email"]`). It now derives from the
|
||||
// portal-side catalog so adding a new channel anywhere only requires
|
||||
// editing src/lib/packages.ts. The `email` channel was dropped as
|
||||
// part of the Phase A package-model rework — IMAP/SMTP is now the
|
||||
// `mail` skill instead.
|
||||
const CHANNEL_PACKAGES = CHANNEL_PACKAGE_IDS;
|
||||
|
||||
export default async function TenantDetailPage({
|
||||
params,
|
||||
|
||||
75
src/app/api/admin/openclaw/route.ts
Normal file
75
src/app/api/admin/openclaw/route.ts
Normal file
@@ -0,0 +1,75 @@
|
||||
import { NextRequest, NextResponse } from "next/server";
|
||||
import { z } from "zod";
|
||||
import { getSessionUser } from "@/lib/session";
|
||||
import { getOpenClawDefaults, setOpenClawDefaults } from "@/lib/k8s";
|
||||
import { safeError } from "@/lib/errors";
|
||||
|
||||
/**
|
||||
* Platform-wide default OpenClaw image tag (admin-only).
|
||||
*
|
||||
* GET — read the current default tag from the
|
||||
* `pieced-openclaw-config` ConfigMap. Can be empty string if no
|
||||
* default is configured; the operator uses its built-in fallback
|
||||
* in that case.
|
||||
*
|
||||
* PATCH — update the tag. Send "" to clear. The operator watches
|
||||
* this ConfigMap and re-enqueues all tenants without a per-tenant
|
||||
* override on change, so existing tenants roll forward to the new
|
||||
* default automatically. Tenants WITH an override are unaffected.
|
||||
*
|
||||
* Tag-only by design — see operator notes.
|
||||
*/
|
||||
|
||||
const patchSchema = z.object({
|
||||
defaultTag: z.string().trim().max(256),
|
||||
});
|
||||
|
||||
export async function GET() {
|
||||
const user = await getSessionUser();
|
||||
if (!user) {
|
||||
return NextResponse.json({ error: "Unauthorized" }, { status: 401 });
|
||||
}
|
||||
if (!user.isPlatform) {
|
||||
return NextResponse.json({ error: "Forbidden" }, { status: 403 });
|
||||
}
|
||||
try {
|
||||
return NextResponse.json(await getOpenClawDefaults());
|
||||
} catch (e: any) {
|
||||
console.error("Failed to read openclaw defaults:", e);
|
||||
return NextResponse.json(
|
||||
{ error: safeError(e, "Failed to read defaults") },
|
||||
{ status: 500 }
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
export async function PATCH(req: NextRequest) {
|
||||
const user = await getSessionUser();
|
||||
if (!user) {
|
||||
return NextResponse.json({ error: "Unauthorized" }, { status: 401 });
|
||||
}
|
||||
if (!user.isPlatform) {
|
||||
return NextResponse.json({ error: "Forbidden" }, { status: 403 });
|
||||
}
|
||||
const body = await req.json().catch(() => null);
|
||||
const parsed = patchSchema.safeParse(body);
|
||||
if (!parsed.success) {
|
||||
return NextResponse.json(
|
||||
{ error: "Invalid input", details: parsed.error.flatten() },
|
||||
{ status: 400 }
|
||||
);
|
||||
}
|
||||
|
||||
try {
|
||||
const next = await setOpenClawDefaults({
|
||||
defaultTag: parsed.data.defaultTag,
|
||||
});
|
||||
return NextResponse.json(next);
|
||||
} catch (e: any) {
|
||||
console.error("Failed to update openclaw defaults:", e);
|
||||
return NextResponse.json(
|
||||
{ error: safeError(e, "Failed to update defaults") },
|
||||
{ status: 500 }
|
||||
);
|
||||
}
|
||||
}
|
||||
78
src/app/api/admin/tenants/[name]/openclaw-image/route.ts
Normal file
78
src/app/api/admin/tenants/[name]/openclaw-image/route.ts
Normal file
@@ -0,0 +1,78 @@
|
||||
import { NextRequest, NextResponse } from "next/server";
|
||||
import { z } from "zod";
|
||||
import { getSessionUser } from "@/lib/session";
|
||||
import { getTenant, patchTenantSpec } from "@/lib/k8s";
|
||||
import { safeError } from "@/lib/errors";
|
||||
|
||||
/**
|
||||
* Per-tenant OpenClaw image override (admin-only).
|
||||
*
|
||||
* Why admin-only: customers cannot pick OpenClaw versions. This
|
||||
* exists so the platform team can A/B-test new releases on specific
|
||||
* tenants without rolling them out fleet-wide. The endpoint enforces
|
||||
* `user.isPlatform`; even owners of the tenant's org cannot use it.
|
||||
*
|
||||
* PATCH body shapes:
|
||||
* - { tag: "2026.4.22" } → use this tag
|
||||
* - { tag: "" } or empty body → clear override (revert to platform
|
||||
* default)
|
||||
*
|
||||
* Tag-only by design — see operator notes for rationale.
|
||||
*/
|
||||
|
||||
const patchSchema = z.object({
|
||||
tag: z.string().trim().max(256).optional(),
|
||||
});
|
||||
|
||||
export async function PATCH(
|
||||
req: NextRequest,
|
||||
{ params }: { params: Promise<{ name: string }> }
|
||||
) {
|
||||
const user = await getSessionUser();
|
||||
if (!user) {
|
||||
return NextResponse.json({ error: "Unauthorized" }, { status: 401 });
|
||||
}
|
||||
if (!user.isPlatform) {
|
||||
return NextResponse.json({ error: "Forbidden" }, { status: 403 });
|
||||
}
|
||||
|
||||
const { name } = await params;
|
||||
const tenant = await getTenant(name);
|
||||
if (!tenant) {
|
||||
return NextResponse.json({ error: "Not found" }, { status: 404 });
|
||||
}
|
||||
|
||||
const body = await req.json().catch(() => null);
|
||||
const parsed = patchSchema.safeParse(body ?? {});
|
||||
if (!parsed.success) {
|
||||
return NextResponse.json(
|
||||
{ error: "Invalid input", details: parsed.error.flatten() },
|
||||
{ status: 400 }
|
||||
);
|
||||
}
|
||||
|
||||
const tag = parsed.data.tag ?? "";
|
||||
const isClearing = tag === "";
|
||||
|
||||
// Merge-patch semantics: openClawImage: null removes the field
|
||||
// from the spec; openClawImage: { tag } sets it.
|
||||
const spec: any = isClearing
|
||||
? { openClawImage: null }
|
||||
: { openClawImage: { tag } };
|
||||
|
||||
try {
|
||||
const updated = await patchTenantSpec(name, spec);
|
||||
return NextResponse.json({
|
||||
message: isClearing
|
||||
? "Override cleared; tenant follows platform default."
|
||||
: "Override set.",
|
||||
openClawImage: updated.spec.openClawImage ?? null,
|
||||
});
|
||||
} catch (e: any) {
|
||||
console.error("Failed to set tenant openclaw image:", e);
|
||||
return NextResponse.json(
|
||||
{ error: safeError(e, "Failed to update tenant image") },
|
||||
{ status: 500 }
|
||||
);
|
||||
}
|
||||
}
|
||||
126
src/app/api/tenants/[name]/budget/route.ts
Normal file
126
src/app/api/tenants/[name]/budget/route.ts
Normal file
@@ -0,0 +1,126 @@
|
||||
import { NextRequest, NextResponse } from "next/server";
|
||||
import { z } from "zod";
|
||||
import { getSessionUser, canMutate } from "@/lib/session";
|
||||
import { getTenant } from "@/lib/k8s";
|
||||
import { canUserSeeTenant } from "@/lib/visibility";
|
||||
import { findKeyByAlias, updateKeyBudget } from "@/lib/litellm";
|
||||
import { safeError } from "@/lib/errors";
|
||||
|
||||
/**
|
||||
* Update the per-tenant budget — operates on the LiteLLM virtual
|
||||
* key, NOT on the team.
|
||||
*
|
||||
* Why per-key
|
||||
* -----------
|
||||
* Each tenant in an org has its own virtual key
|
||||
* (`key_alias = tenant.metadata.name`); the team that owns those
|
||||
* keys is org-scoped and shared across all the org's tenants. A
|
||||
* budget on the team would cap the whole org; a budget on the key
|
||||
* caps just this one tenant. Customers landing on the tenant detail
|
||||
* page reasonably expect "edit budget" to mean "the budget of THIS
|
||||
* tenant" — so we put it on the key.
|
||||
*
|
||||
* The team-level (org-wide) budget is a separate control that lives
|
||||
* in /settings (not yet implemented) — the two coexist: LiteLLM
|
||||
* applies whichever cap is hit first.
|
||||
*
|
||||
* Schema:
|
||||
* - maxBudget: number > 0 (set a cap), or null (remove the cap).
|
||||
* - budgetDuration: one of "30d", "1mo", "1y", or null (lifetime).
|
||||
*
|
||||
* Authorization: owners and platform admins.
|
||||
*/
|
||||
|
||||
const patchSchema = z.object({
|
||||
// > 0 because LiteLLM rejects 0 and a zero cap would lock the key
|
||||
// out instantly. Upper bound 1M as a typo guard.
|
||||
maxBudget: z.number().positive().max(1_000_000).nullable(),
|
||||
budgetDuration: z.enum(["30d", "1mo", "1y"]).nullable(),
|
||||
});
|
||||
|
||||
export async function PATCH(
|
||||
req: NextRequest,
|
||||
{ params }: { params: Promise<{ name: string }> }
|
||||
) {
|
||||
const user = await getSessionUser();
|
||||
if (!user) {
|
||||
return NextResponse.json({ error: "Unauthorized" }, { status: 401 });
|
||||
}
|
||||
if (!canMutate(user)) {
|
||||
return NextResponse.json({ error: "Forbidden" }, { status: 403 });
|
||||
}
|
||||
|
||||
const { name } = await params;
|
||||
const tenant = await getTenant(name);
|
||||
if (!tenant) {
|
||||
return NextResponse.json({ error: "Not found" }, { status: 404 });
|
||||
}
|
||||
if (!(await canUserSeeTenant(user, tenant))) {
|
||||
// Don't leak existence — same 404 a non-visible tenant gets.
|
||||
return NextResponse.json({ error: "Not found" }, { status: 404 });
|
||||
}
|
||||
|
||||
const teamId = tenant.status?.litellmTeamId;
|
||||
if (!teamId) {
|
||||
return NextResponse.json(
|
||||
{
|
||||
error:
|
||||
"Tenant has no LiteLLM team yet. Please wait until provisioning completes.",
|
||||
},
|
||||
{ status: 409 }
|
||||
);
|
||||
}
|
||||
|
||||
const body = await req.json().catch(() => null);
|
||||
const parsed = patchSchema.safeParse(body);
|
||||
if (!parsed.success) {
|
||||
return NextResponse.json(
|
||||
{ error: "Invalid input", details: parsed.error.flatten() },
|
||||
{ status: 400 }
|
||||
);
|
||||
}
|
||||
|
||||
// Defensive: removing the cap should null out the duration too —
|
||||
// a reset cadence on an unlimited budget is meaningless and would
|
||||
// confuse LiteLLM's bookkeeping.
|
||||
const maxBudget = parsed.data.maxBudget;
|
||||
const budgetDuration =
|
||||
maxBudget === null ? null : parsed.data.budgetDuration;
|
||||
|
||||
// Look up the key by alias (= tenant name). The token returned is
|
||||
// what /key/update wants in the `key` field.
|
||||
let keyInfo;
|
||||
try {
|
||||
keyInfo = await findKeyByAlias(teamId, name);
|
||||
} catch (e: any) {
|
||||
console.error("Failed to look up tenant key:", e);
|
||||
return NextResponse.json(
|
||||
{ error: safeError(e, "Failed to look up tenant key") },
|
||||
{ status: 500 }
|
||||
);
|
||||
}
|
||||
if (!keyInfo) {
|
||||
return NextResponse.json(
|
||||
{
|
||||
error:
|
||||
"Tenant has no virtual key yet. Please wait until provisioning completes.",
|
||||
},
|
||||
{ status: 409 }
|
||||
);
|
||||
}
|
||||
|
||||
try {
|
||||
await updateKeyBudget(keyInfo.token, { maxBudget, budgetDuration });
|
||||
return NextResponse.json({
|
||||
message: maxBudget === null ? "Budget removed." : "Budget updated.",
|
||||
maxBudget,
|
||||
budgetDuration,
|
||||
});
|
||||
} catch (e: any) {
|
||||
console.error("Failed to update key budget:", e);
|
||||
return NextResponse.json(
|
||||
{ error: safeError(e, "Failed to update budget") },
|
||||
{ status: 500 }
|
||||
);
|
||||
}
|
||||
}
|
||||
168
src/app/api/tenants/[name]/threema/route.ts
Normal file
168
src/app/api/tenants/[name]/threema/route.ts
Normal file
@@ -0,0 +1,168 @@
|
||||
import { NextRequest, NextResponse } from "next/server";
|
||||
import { getSessionUser, canMutate } from "@/lib/session";
|
||||
import { getTenant } from "@/lib/k8s";
|
||||
import {
|
||||
writePackageSecrets,
|
||||
deletePackageSecrets,
|
||||
} from "@/lib/openbao";
|
||||
import { mintToken, revokeToken } from "@/lib/threema-relay";
|
||||
import { safeError } from "@/lib/errors";
|
||||
|
||||
/**
|
||||
* Threema package provisioning — special-cased because the credentials
|
||||
* are platform-issued (relay mints them), not customer-supplied.
|
||||
*
|
||||
* POST /api/tenants/:name/threema
|
||||
* - Mints a per-tenant bearer + HMAC secret from the central relay.
|
||||
* - Writes both to OpenBao under
|
||||
* secret/data/tenants/<tenant-{name}>/threema-relay so the
|
||||
* operator's ExternalSecret can sync them into the tenant
|
||||
* namespace alongside other channel secrets.
|
||||
* - Returns 200 on success. The caller (PackageCard) then PATCHes
|
||||
* tenant.spec.packages to add "threema".
|
||||
*
|
||||
* DELETE /api/tenants/:name/threema
|
||||
* - Revokes the per-tenant token at the relay (cascades to all
|
||||
* routes — the relay's tokens.deleteToken also deletes routes).
|
||||
* - Deletes the OpenBao secret so the ExternalSecret/operator can
|
||||
* converge cleanly.
|
||||
* - Returns 200 on success even if no token existed (idempotent).
|
||||
*
|
||||
* Failure semantics
|
||||
* -----------------
|
||||
* On POST: if minting succeeds but the OpenBao write fails, we attempt
|
||||
* to revoke the just-minted token before returning the error. That way
|
||||
* the relay doesn't keep an orphan token row that nothing can use.
|
||||
* Best-effort cleanup; if the revoke also fails, the relay admin can
|
||||
* use DELETE /admin/tokens/<name> manually.
|
||||
*
|
||||
* On DELETE: we revoke FIRST, then delete OpenBao. If revoke fails we
|
||||
* return the error and stop — leaving OpenBao alone means the pod's
|
||||
* still-mounted secret keeps working in the brief window between
|
||||
* "customer hits disable" and "operator reconciles spec without threema",
|
||||
* which is more graceful than yanking the secret out from under a
|
||||
* running pod.
|
||||
*/
|
||||
|
||||
const VAULT_SUFFIX = "threema-relay";
|
||||
|
||||
export async function POST(
|
||||
_req: NextRequest,
|
||||
{ params }: { params: Promise<{ name: string }> },
|
||||
) {
|
||||
const user = await getSessionUser();
|
||||
if (!user) return NextResponse.json({ error: "Unauthorized" }, { status: 401 });
|
||||
if (!canMutate(user))
|
||||
return NextResponse.json({ error: "Forbidden" }, { status: 403 });
|
||||
|
||||
const { name } = await params;
|
||||
|
||||
try {
|
||||
const tenant = await getTenant(name);
|
||||
if (!tenant)
|
||||
return NextResponse.json({ error: "Not found" }, { status: 404 });
|
||||
if (
|
||||
!user.isPlatform &&
|
||||
tenant.metadata.labels?.["pieced.ch/zitadel-org-id"] !== user.orgId
|
||||
) {
|
||||
return NextResponse.json({ error: "Forbidden" }, { status: 403 });
|
||||
}
|
||||
|
||||
const minted = await mintToken(name);
|
||||
if (!minted.ok) {
|
||||
return NextResponse.json(
|
||||
{ error: `Relay mint failed: ${minted.message}` },
|
||||
{ status: minted.kind === "http" ? 502 : 503 },
|
||||
);
|
||||
}
|
||||
|
||||
try {
|
||||
await writePackageSecrets(`tenant-${name}`, VAULT_SUFFIX, {
|
||||
token: minted.token,
|
||||
"hmac-secret": minted.hmacSecret,
|
||||
});
|
||||
} catch (e) {
|
||||
// Compensate: revoke the just-minted token so the relay doesn't
|
||||
// hold an orphan. Best-effort — log and continue surfacing the
|
||||
// original error.
|
||||
const revoke = await revokeToken(name);
|
||||
if (!revoke.ok) {
|
||||
console.error(
|
||||
`[threema/provision] Compensating revoke failed for ${name}: ${revoke.message}`,
|
||||
);
|
||||
}
|
||||
return NextResponse.json(
|
||||
{ error: `OpenBao write failed: ${safeError(e, "secret store unavailable")}` },
|
||||
{ status: 503 },
|
||||
);
|
||||
}
|
||||
|
||||
return NextResponse.json({ ok: true });
|
||||
} catch (e) {
|
||||
console.error("[threema/provision]", e);
|
||||
return NextResponse.json(
|
||||
{ error: safeError(e, "Provisioning failed") },
|
||||
{ status: 500 },
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
export async function DELETE(
|
||||
_req: NextRequest,
|
||||
{ params }: { params: Promise<{ name: string }> },
|
||||
) {
|
||||
const user = await getSessionUser();
|
||||
if (!user) return NextResponse.json({ error: "Unauthorized" }, { status: 401 });
|
||||
if (!canMutate(user))
|
||||
return NextResponse.json({ error: "Forbidden" }, { status: 403 });
|
||||
|
||||
const { name } = await params;
|
||||
|
||||
try {
|
||||
const tenant = await getTenant(name);
|
||||
if (!tenant)
|
||||
return NextResponse.json({ error: "Not found" }, { status: 404 });
|
||||
if (
|
||||
!user.isPlatform &&
|
||||
tenant.metadata.labels?.["pieced.ch/zitadel-org-id"] !== user.orgId
|
||||
) {
|
||||
return NextResponse.json({ error: "Forbidden" }, { status: 403 });
|
||||
}
|
||||
|
||||
const revoke = await revokeToken(name);
|
||||
// 404 from relay = nothing to revoke = idempotent success.
|
||||
if (!revoke.ok && !(revoke.kind === "http" && revoke.status === 404)) {
|
||||
return NextResponse.json(
|
||||
{ error: `Relay revoke failed: ${revoke.message}` },
|
||||
{ status: revoke.kind === "http" ? 502 : 503 },
|
||||
);
|
||||
}
|
||||
|
||||
// Delete the OpenBao secret. Idempotent — deletePackageSecrets
|
||||
// tolerates 404.
|
||||
try {
|
||||
await deletePackageSecrets(`tenant-${name}`, VAULT_SUFFIX);
|
||||
} catch (e) {
|
||||
// Already revoked at the relay — surface the openbao failure
|
||||
// but keep the partial-success state visible.
|
||||
return NextResponse.json(
|
||||
{
|
||||
error: `Token revoked, but OpenBao delete failed: ${safeError(e, "secret store unavailable")}`,
|
||||
partial: true,
|
||||
},
|
||||
{ status: 503 },
|
||||
);
|
||||
}
|
||||
|
||||
return NextResponse.json({
|
||||
ok: true,
|
||||
deletedRoutes: revoke.ok ? revoke.deletedRoutes : 0,
|
||||
});
|
||||
} catch (e) {
|
||||
console.error("[threema/deprovision]", e);
|
||||
return NextResponse.json(
|
||||
{ error: safeError(e, "Deprovisioning failed") },
|
||||
{ status: 500 },
|
||||
);
|
||||
}
|
||||
}
|
||||
217
src/app/api/tenants/[name]/threema/routes/route.ts
Normal file
217
src/app/api/tenants/[name]/threema/routes/route.ts
Normal file
@@ -0,0 +1,217 @@
|
||||
import { NextRequest, NextResponse } from "next/server";
|
||||
import { z } from "zod";
|
||||
import { getSessionUser, canMutate } from "@/lib/session";
|
||||
import { getTenant, patchTenantSpec } from "@/lib/k8s";
|
||||
import {
|
||||
createRoute,
|
||||
deleteRoute,
|
||||
isRouteConflictForOtherTenant,
|
||||
isRouteConflictForSameTenant,
|
||||
listRoutes,
|
||||
} from "@/lib/threema-relay";
|
||||
import { safeError } from "@/lib/errors";
|
||||
|
||||
/**
|
||||
* Threema route management — keeps three places in sync:
|
||||
*
|
||||
* 1. Relay DB (`routes` table) — source of truth for uniqueness
|
||||
* 2. K8s spec.channelUsers.threema — what the operator sees
|
||||
* 3. Customer UI — derived from (2)
|
||||
*
|
||||
* Add (POST) order: relay first (to claim uniqueness atomically), then
|
||||
* K8s. On K8s failure we compensate by deleting the relay route.
|
||||
*
|
||||
* Remove (DELETE) order: K8s first (UI shows it gone immediately, which
|
||||
* is the customer-facing semantic that matters), then relay. On relay
|
||||
* failure we DO NOT rollback K8s — the customer wanted it gone, and
|
||||
* the relay's stale route will be cleaned up on the next retry (deletes
|
||||
* are idempotent at the relay).
|
||||
*
|
||||
* Read-modify-write race: patchTenantSpec uses K8s merge-patch on
|
||||
* spec.channelUsers.threema, which REPLACES the entire array. We GET
|
||||
* the latest array, mutate it, then PATCH. Two concurrent adds from the
|
||||
* same customer's tabs can lose one of them. Acceptable at pilot scale
|
||||
* (single-digit customers, low concurrency); revisit with SSA + field
|
||||
* managers if it ever bites.
|
||||
*/
|
||||
|
||||
const ROUTE_BODY = z.object({
|
||||
threemaId: z
|
||||
.string()
|
||||
.regex(/^[A-Z0-9]{8}$/, "Threema ID must be 8 uppercase alphanumeric chars (no asterisk)"),
|
||||
});
|
||||
|
||||
// ---- helpers --------------------------------------------------------------
|
||||
|
||||
async function loadTenantOrError(name: string, user: Awaited<ReturnType<typeof getSessionUser>>) {
|
||||
if (!user) return { error: NextResponse.json({ error: "Unauthorized" }, { status: 401 }) };
|
||||
if (!canMutate(user))
|
||||
return { error: NextResponse.json({ error: "Forbidden" }, { status: 403 }) };
|
||||
|
||||
const tenant = await getTenant(name);
|
||||
if (!tenant)
|
||||
return { error: NextResponse.json({ error: "Not found" }, { status: 404 }) };
|
||||
if (
|
||||
!user.isPlatform &&
|
||||
tenant.metadata.labels?.["pieced.ch/zitadel-org-id"] !== user.orgId
|
||||
) {
|
||||
return { error: NextResponse.json({ error: "Forbidden" }, { status: 403 }) };
|
||||
}
|
||||
return { tenant };
|
||||
}
|
||||
|
||||
function currentThreemaIds(tenantSpec: any): string[] {
|
||||
const cu = tenantSpec?.channelUsers ?? {};
|
||||
const ids = cu.threema;
|
||||
return Array.isArray(ids) ? ids.filter((x) => typeof x === "string") : [];
|
||||
}
|
||||
|
||||
// ---- GET ------------------------------------------------------------------
|
||||
|
||||
export async function GET(
|
||||
_req: NextRequest,
|
||||
{ params }: { params: Promise<{ name: string }> },
|
||||
) {
|
||||
const user = await getSessionUser();
|
||||
const { name } = await params;
|
||||
const loaded = await loadTenantOrError(name, user);
|
||||
if (loaded.error) return loaded.error;
|
||||
|
||||
const res = await listRoutes(name);
|
||||
if (!res.ok) {
|
||||
return NextResponse.json(
|
||||
{ error: `Relay list failed: ${res.message}` },
|
||||
{ status: res.kind === "http" ? 502 : 503 },
|
||||
);
|
||||
}
|
||||
return NextResponse.json({ routes: res.routes });
|
||||
}
|
||||
|
||||
// ---- POST -----------------------------------------------------------------
|
||||
|
||||
export async function POST(
|
||||
req: NextRequest,
|
||||
{ params }: { params: Promise<{ name: string }> },
|
||||
) {
|
||||
const user = await getSessionUser();
|
||||
const { name } = await params;
|
||||
const loaded = await loadTenantOrError(name, user);
|
||||
if (loaded.error) return loaded.error;
|
||||
const tenant = loaded.tenant;
|
||||
|
||||
const body = await req.json().catch(() => null);
|
||||
const parsed = ROUTE_BODY.safeParse(body);
|
||||
if (!parsed.success) {
|
||||
return NextResponse.json(
|
||||
{ error: "Invalid threemaId", details: parsed.error.flatten() },
|
||||
{ status: 400 },
|
||||
);
|
||||
}
|
||||
const threemaId = parsed.data.threemaId;
|
||||
|
||||
// Step 1: claim at the relay. Uniqueness is enforced here.
|
||||
const claim = await createRoute(name, threemaId);
|
||||
if (!claim.ok) {
|
||||
if (isRouteConflictForOtherTenant(claim, name)) {
|
||||
return NextResponse.json(
|
||||
{ error: "This Threema ID is already registered to another tenant" },
|
||||
{ status: 409 },
|
||||
);
|
||||
}
|
||||
if (!isRouteConflictForSameTenant(claim, name)) {
|
||||
// Genuine non-409 failure
|
||||
return NextResponse.json(
|
||||
{ error: `Relay create failed: ${claim.message}` },
|
||||
{ status: claim.kind === "http" ? claim.status : 503 },
|
||||
);
|
||||
}
|
||||
// Idempotent self-claim — continue to step 2 to ensure K8s mirrors it.
|
||||
}
|
||||
|
||||
// Step 2: add to K8s spec.channelUsers.threema (idempotent).
|
||||
const existing = currentThreemaIds(tenant!.spec);
|
||||
if (!existing.includes(threemaId)) {
|
||||
const next = [...existing, threemaId];
|
||||
try {
|
||||
await patchTenantSpec(name, {
|
||||
channelUsers: {
|
||||
...(tenant!.spec?.channelUsers ?? {}),
|
||||
threema: next,
|
||||
} as Record<string, string[]>,
|
||||
});
|
||||
} catch (e) {
|
||||
// Compensate: drop the relay route so we don't leave an orphan.
|
||||
const compensate = await deleteRoute(name, threemaId);
|
||||
if (!compensate.ok) {
|
||||
console.error(
|
||||
`[threema/routes] Compensating route delete failed for ${name}/${threemaId}: ${compensate.message}`,
|
||||
);
|
||||
}
|
||||
return NextResponse.json(
|
||||
{ error: `K8s patch failed: ${safeError(e, "patch failed")}` },
|
||||
{ status: 500 },
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
return NextResponse.json({ ok: true, threemaId });
|
||||
}
|
||||
|
||||
// ---- DELETE ---------------------------------------------------------------
|
||||
|
||||
export async function DELETE(
|
||||
req: NextRequest,
|
||||
{ params }: { params: Promise<{ name: string }> },
|
||||
) {
|
||||
const user = await getSessionUser();
|
||||
const { name } = await params;
|
||||
const loaded = await loadTenantOrError(name, user);
|
||||
if (loaded.error) return loaded.error;
|
||||
const tenant = loaded.tenant;
|
||||
|
||||
// threemaId arrives as ?threemaId=... since DELETE bodies are uneven across clients.
|
||||
const threemaId = new URL(req.url).searchParams.get("threemaId") ?? "";
|
||||
const parsed = ROUTE_BODY.safeParse({ threemaId });
|
||||
if (!parsed.success) {
|
||||
return NextResponse.json(
|
||||
{ error: "Invalid threemaId", details: parsed.error.flatten() },
|
||||
{ status: 400 },
|
||||
);
|
||||
}
|
||||
|
||||
// Step 1: drop from K8s (idempotent).
|
||||
const existing = currentThreemaIds(tenant!.spec);
|
||||
if (existing.includes(parsed.data.threemaId)) {
|
||||
const next = existing.filter((id) => id !== parsed.data.threemaId);
|
||||
try {
|
||||
await patchTenantSpec(name, {
|
||||
channelUsers: {
|
||||
...(tenant!.spec?.channelUsers ?? {}),
|
||||
threema: next,
|
||||
} as Record<string, string[]>,
|
||||
});
|
||||
} catch (e) {
|
||||
return NextResponse.json(
|
||||
{ error: `K8s patch failed: ${safeError(e, "patch failed")}` },
|
||||
{ status: 500 },
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
// Step 2: drop at relay (also idempotent — 404 is fine).
|
||||
const dropped = await deleteRoute(name, parsed.data.threemaId);
|
||||
if (!dropped.ok && !(dropped.kind === "http" && dropped.status === 404)) {
|
||||
// K8s is already updated; surface but don't rollback. Next time the
|
||||
// user toggles, both will converge.
|
||||
return NextResponse.json(
|
||||
{
|
||||
ok: true,
|
||||
threemaId: parsed.data.threemaId,
|
||||
warning: `Removed from K8s but relay drop failed: ${dropped.message}`,
|
||||
},
|
||||
{ status: 200 },
|
||||
);
|
||||
}
|
||||
|
||||
return NextResponse.json({ ok: true, threemaId: parsed.data.threemaId });
|
||||
}
|
||||
@@ -2,7 +2,11 @@ import { NextRequest, NextResponse } from "next/server";
|
||||
import { getSessionUser } from "@/lib/session";
|
||||
import { listTenants } from "@/lib/k8s";
|
||||
import { listVisibleTenants } from "@/lib/visibility";
|
||||
import { getTeamInfo, getTeamSpendLogsV2 } from "@/lib/litellm";
|
||||
import {
|
||||
getTeamInfo,
|
||||
getTeamSpendLogsV2,
|
||||
findKeyByAlias,
|
||||
} from "@/lib/litellm";
|
||||
import { safeError } from "@/lib/errors";
|
||||
|
||||
/**
|
||||
@@ -126,6 +130,16 @@ export async function GET(req: NextRequest) {
|
||||
try {
|
||||
const teamInfo = await getTeamInfo(teamId);
|
||||
|
||||
// Per-tenant budget lives on the virtual key, not the team
|
||||
// (Feature 7 fix). When the request is scoped to a specific
|
||||
// tenant (keyAlias provided), look up the key so we can return
|
||||
// the per-tenant cap. Tolerate failure — older LiteLLM builds
|
||||
// or short-lived race conditions during provisioning shouldn't
|
||||
// 500 the whole usage page; we degrade to "no key info".
|
||||
const keyInfo = keyAlias
|
||||
? await findKeyByAlias(teamId, keyAlias).catch(() => null)
|
||||
: null;
|
||||
|
||||
// Page through results — server-side filtered by key_alias when
|
||||
// provided. Pagination still needed because LiteLLM caps
|
||||
// page_size at 100, and a busy tenant can easily exceed that in
|
||||
@@ -191,17 +205,38 @@ export async function GET(req: NextRequest) {
|
||||
totalSpend,
|
||||
requestCount: allRequests.length,
|
||||
},
|
||||
// Budget is always team-level (= company budget). Spend reported
|
||||
// here is the team total, not the per-key total — the customer
|
||||
// wants to see "how much of our company budget is left", not
|
||||
// just "how much has this one tenant cost".
|
||||
budget: {
|
||||
maxBudget: teamInfo?.team_info?.max_budget ?? null,
|
||||
spend: teamInfo?.team_info?.spend ?? 0,
|
||||
remaining: teamInfo?.team_info?.max_budget
|
||||
? teamInfo.team_info.max_budget - (teamInfo.team_info.spend ?? 0)
|
||||
: null,
|
||||
},
|
||||
// Budget reporting (Feature 7).
|
||||
//
|
||||
// When the caller scopes to a specific tenant (keyAlias set),
|
||||
// we report THAT tenant's per-key budget — that's what the
|
||||
// tenant detail page renders, and what the customer expects
|
||||
// when they see "Budget" on a tenant's page.
|
||||
//
|
||||
// When unscoped (admin / org-wide view), we fall back to the
|
||||
// team budget — that's the org-wide cap, conceptually different
|
||||
// but the only thing meaningful at that scope.
|
||||
//
|
||||
// The two cases display the same way; the editor button gates
|
||||
// on whether we know which tenant we're on (= keyAlias set).
|
||||
budget: keyAlias && keyInfo
|
||||
? {
|
||||
maxBudget: keyInfo.maxBudget,
|
||||
spend: keyInfo.spend,
|
||||
remaining:
|
||||
keyInfo.maxBudget !== null
|
||||
? keyInfo.maxBudget - keyInfo.spend
|
||||
: null,
|
||||
budgetDuration: keyInfo.budgetDuration,
|
||||
}
|
||||
: {
|
||||
maxBudget: teamInfo?.team_info?.max_budget ?? null,
|
||||
spend: teamInfo?.team_info?.spend ?? 0,
|
||||
remaining: teamInfo?.team_info?.max_budget
|
||||
? teamInfo.team_info.max_budget -
|
||||
(teamInfo.team_info.spend ?? 0)
|
||||
: null,
|
||||
budgetDuration: teamInfo?.team_info?.budget_duration ?? null,
|
||||
},
|
||||
rateLimits: {
|
||||
rpm: teamInfo?.team_info?.rpm_limit ?? null,
|
||||
tpm: teamInfo?.team_info?.tpm_limit ?? null,
|
||||
|
||||
277
src/components/admin/openclaw-admin-panel.tsx
Normal file
277
src/components/admin/openclaw-admin-panel.tsx
Normal file
@@ -0,0 +1,277 @@
|
||||
"use client";
|
||||
|
||||
import { useState } from "react";
|
||||
import { useRouter } from "next/navigation";
|
||||
import { useTranslations } from "next-intl";
|
||||
import { Card } from "@/components/ui/card";
|
||||
import type { OpenClawDefaults } from "@/lib/k8s";
|
||||
|
||||
interface TenantRow {
|
||||
name: string;
|
||||
displayName: string;
|
||||
phase: string;
|
||||
override: { tag: string } | null;
|
||||
}
|
||||
|
||||
interface Props {
|
||||
initialDefaults: OpenClawDefaults;
|
||||
tenants: TenantRow[];
|
||||
}
|
||||
|
||||
/**
|
||||
* Two-section admin UI:
|
||||
* - Default editor card at the top — single input for the tag.
|
||||
* - Tenant table below — each row has an inline edit/clear control.
|
||||
*
|
||||
* No optimistic updates: every save round-trips to the API and we
|
||||
* router.refresh() to re-render the server-side state. Keeps the UI
|
||||
* honest about what's actually applied (controller-runtime watch
|
||||
* latency can be a couple of seconds).
|
||||
*
|
||||
* Tag-only by design — see operator notes for rationale.
|
||||
*/
|
||||
export function OpenClawAdminPanel({ initialDefaults, tenants }: Props) {
|
||||
const t = useTranslations("openclawAdmin");
|
||||
const tCommon = useTranslations("common");
|
||||
const router = useRouter();
|
||||
|
||||
const [defaults, setDefaults] = useState(initialDefaults);
|
||||
const [defaultTag, setDefaultTag] = useState(initialDefaults.defaultTag);
|
||||
const [savingDefault, setSavingDefault] = useState(false);
|
||||
const [defaultError, setDefaultError] = useState("");
|
||||
const [defaultSaved, setDefaultSaved] = useState(false);
|
||||
|
||||
const onSaveDefault = async (e: React.FormEvent) => {
|
||||
e.preventDefault();
|
||||
setSavingDefault(true);
|
||||
setDefaultError("");
|
||||
setDefaultSaved(false);
|
||||
try {
|
||||
const res = await fetch("/api/admin/openclaw", {
|
||||
method: "PATCH",
|
||||
headers: { "Content-Type": "application/json" },
|
||||
body: JSON.stringify({ defaultTag: defaultTag.trim() }),
|
||||
});
|
||||
if (!res.ok) {
|
||||
const data = await res.json().catch(() => ({}));
|
||||
throw new Error(data.error || t("saveFailed"));
|
||||
}
|
||||
const next = await res.json();
|
||||
setDefaults(next);
|
||||
setDefaultSaved(true);
|
||||
} catch (e: any) {
|
||||
setDefaultError(e.message);
|
||||
} finally {
|
||||
setSavingDefault(false);
|
||||
}
|
||||
};
|
||||
|
||||
return (
|
||||
<div className="space-y-8">
|
||||
{/* Default editor */}
|
||||
<section className="animate-in animate-in-delay-1">
|
||||
<h2 className="text-xs font-semibold uppercase tracking-wider text-text-muted mb-3">
|
||||
{t("defaultSection")}
|
||||
</h2>
|
||||
<Card>
|
||||
<p className="text-sm text-text-secondary mb-4">
|
||||
{t("defaultDescription")}
|
||||
</p>
|
||||
<form onSubmit={onSaveDefault} className="space-y-4">
|
||||
<div>
|
||||
<label className="block text-xs uppercase tracking-wider text-text-muted mb-1">
|
||||
{t("fieldTag")}
|
||||
</label>
|
||||
<input
|
||||
type="text"
|
||||
value={defaultTag}
|
||||
onChange={(e) => setDefaultTag(e.target.value)}
|
||||
placeholder="2026.4.22"
|
||||
className="w-full px-3 py-2 rounded-lg border border-border bg-surface-2 text-text-primary text-sm font-mono focus:outline-none focus:border-text-secondary"
|
||||
/>
|
||||
<p className="text-xs text-text-muted mt-1">{t("emptyHint")}</p>
|
||||
</div>
|
||||
|
||||
{defaultError && (
|
||||
<div className="text-xs text-red-400 bg-red-400/10 border border-red-400/20 rounded-lg px-3 py-2">
|
||||
{defaultError}
|
||||
</div>
|
||||
)}
|
||||
{defaultSaved && !defaultError && (
|
||||
<div className="text-xs text-success bg-success/10 border border-success/20 rounded-lg px-3 py-2">
|
||||
{t("defaultSaved")}
|
||||
</div>
|
||||
)}
|
||||
|
||||
<div className="flex justify-end">
|
||||
<button
|
||||
type="submit"
|
||||
disabled={savingDefault}
|
||||
className="text-sm font-medium px-4 py-2 rounded-lg bg-accent text-white hover:bg-accent/90 transition-colors disabled:opacity-50"
|
||||
>
|
||||
{savingDefault ? tCommon("loading") : t("saveDefault")}
|
||||
</button>
|
||||
</div>
|
||||
</form>
|
||||
</Card>
|
||||
</section>
|
||||
|
||||
{/* Tenant overrides */}
|
||||
<section className="animate-in animate-in-delay-2">
|
||||
<h2 className="text-xs font-semibold uppercase tracking-wider text-text-muted mb-3">
|
||||
{t("overridesSection")}
|
||||
</h2>
|
||||
<Card>
|
||||
{tenants.length === 0 ? (
|
||||
<p className="text-sm text-text-secondary text-center py-6">
|
||||
{t("noTenants")}
|
||||
</p>
|
||||
) : (
|
||||
<div className="space-y-2">
|
||||
{tenants.map((tn) => (
|
||||
<TenantOverrideRow
|
||||
key={tn.name}
|
||||
tenant={tn}
|
||||
platformDefault={defaults}
|
||||
onChanged={() => router.refresh()}
|
||||
/>
|
||||
))}
|
||||
</div>
|
||||
)}
|
||||
</Card>
|
||||
</section>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* Single row in the tenants table. Collapsed by default; click to
|
||||
* expand the inline editor.
|
||||
*/
|
||||
function TenantOverrideRow({
|
||||
tenant,
|
||||
platformDefault,
|
||||
onChanged,
|
||||
}: {
|
||||
tenant: TenantRow;
|
||||
platformDefault: OpenClawDefaults;
|
||||
onChanged: () => void;
|
||||
}) {
|
||||
const t = useTranslations("openclawAdmin");
|
||||
const tCommon = useTranslations("common");
|
||||
|
||||
const [expanded, setExpanded] = useState(false);
|
||||
const [tag, setTag] = useState(tenant.override?.tag ?? "");
|
||||
const [saving, setSaving] = useState(false);
|
||||
const [error, setError] = useState("");
|
||||
|
||||
const submit = async (clear = false) => {
|
||||
setSaving(true);
|
||||
setError("");
|
||||
try {
|
||||
const res = await fetch(
|
||||
`/api/admin/tenants/${encodeURIComponent(tenant.name)}/openclaw-image`,
|
||||
{
|
||||
method: "PATCH",
|
||||
headers: { "Content-Type": "application/json" },
|
||||
body: JSON.stringify(clear ? {} : { tag: tag.trim() }),
|
||||
}
|
||||
);
|
||||
if (!res.ok) {
|
||||
const data = await res.json().catch(() => ({}));
|
||||
throw new Error(data.error || t("saveFailed"));
|
||||
}
|
||||
setExpanded(false);
|
||||
onChanged();
|
||||
} catch (e: any) {
|
||||
setError(e.message);
|
||||
} finally {
|
||||
setSaving(false);
|
||||
}
|
||||
};
|
||||
|
||||
const effective = tenant.override?.tag
|
||||
? tenant.override.tag
|
||||
: platformDefault.defaultTag || t("builtinFallback");
|
||||
|
||||
return (
|
||||
<div className="rounded-lg border border-border bg-surface-2 overflow-hidden">
|
||||
<button
|
||||
type="button"
|
||||
onClick={() => setExpanded((v) => !v)}
|
||||
className="w-full flex items-center justify-between px-4 py-3 text-left hover:bg-surface-1 transition-colors"
|
||||
>
|
||||
<div className="min-w-0 flex-1">
|
||||
<div className="font-medium text-text-primary truncate">
|
||||
{tenant.displayName}
|
||||
</div>
|
||||
<div className="text-xs text-text-muted font-mono truncate mt-0.5">
|
||||
{tenant.name}
|
||||
</div>
|
||||
</div>
|
||||
<div className="text-right ml-4 min-w-0">
|
||||
{tenant.override ? (
|
||||
<span className="inline-flex items-center px-2 py-0.5 text-xs font-medium rounded-full bg-amber-400/15 text-amber-400 border border-amber-400/20">
|
||||
{t("statusOverridden")}
|
||||
</span>
|
||||
) : (
|
||||
<span className="inline-flex items-center px-2 py-0.5 text-xs font-medium rounded-full bg-blue-400/15 text-blue-400 border border-blue-400/20">
|
||||
{t("statusFollowsDefault")}
|
||||
</span>
|
||||
)}
|
||||
<div className="text-xs text-text-muted font-mono truncate max-w-[260px] mt-1">
|
||||
{effective}
|
||||
</div>
|
||||
</div>
|
||||
</button>
|
||||
|
||||
{expanded && (
|
||||
<div className="px-4 pb-4 pt-1 border-t border-border bg-surface-1">
|
||||
<div className="mb-3">
|
||||
<label className="block text-xs uppercase tracking-wider text-text-muted mb-1">
|
||||
{t("fieldTag")}
|
||||
</label>
|
||||
<input
|
||||
type="text"
|
||||
value={tag}
|
||||
onChange={(e) => setTag(e.target.value)}
|
||||
placeholder={
|
||||
platformDefault.defaultTag
|
||||
? `${t("defaultPrefix")} ${platformDefault.defaultTag}`
|
||||
: ""
|
||||
}
|
||||
className="w-full px-3 py-2 rounded-lg border border-border bg-surface-2 text-text-primary text-sm font-mono focus:outline-none focus:border-text-secondary"
|
||||
/>
|
||||
</div>
|
||||
|
||||
{error && (
|
||||
<div className="text-xs text-red-400 bg-red-400/10 border border-red-400/20 rounded-lg px-3 py-2 mb-3">
|
||||
{error}
|
||||
</div>
|
||||
)}
|
||||
|
||||
<div className="flex flex-wrap gap-2 justify-end">
|
||||
{tenant.override && (
|
||||
<button
|
||||
type="button"
|
||||
onClick={() => submit(true)}
|
||||
disabled={saving}
|
||||
className="text-xs px-3 py-1.5 rounded-lg border border-border text-text-secondary hover:text-text-primary transition-colors disabled:opacity-50"
|
||||
>
|
||||
{saving ? tCommon("loading") : t("clearOverride")}
|
||||
</button>
|
||||
)}
|
||||
<button
|
||||
type="button"
|
||||
onClick={() => submit(false)}
|
||||
disabled={saving || !tag.trim()}
|
||||
className="text-xs px-3 py-1.5 rounded-lg bg-accent text-white hover:bg-accent/90 transition-colors disabled:opacity-50"
|
||||
>
|
||||
{saving ? tCommon("loading") : t("saveOverride")}
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
);
|
||||
}
|
||||
@@ -3,14 +3,32 @@
|
||||
import { useState, useCallback } from "react";
|
||||
import { useTranslations } from "next-intl";
|
||||
import { useRouter } from "next/navigation";
|
||||
import { ThreemaQrModal } from "./threema-qr-modal";
|
||||
|
||||
/** Maps channel IDs to the instructions for finding the user ID. */
|
||||
const CHANNEL_ID_HELP: Record<string, string> = {
|
||||
telegram: "telegramIdHelp",
|
||||
discord: "discordIdHelp",
|
||||
email: "emailIdHelp",
|
||||
threema: "threemaIdHelp",
|
||||
// email entry dropped in the Phase A rework — IMAP/SMTP is handled by
|
||||
// the `mail` skill (category=skill, not channel), so it never appears
|
||||
// in `enabledChannels`. If a future channel is added to the catalog,
|
||||
// give it an entry here so the help blurb renders.
|
||||
};
|
||||
|
||||
/**
|
||||
* Channels whose user list is managed through a dedicated endpoint
|
||||
* instead of the generic PATCH /api/tenants/:name flow.
|
||||
*
|
||||
* Threema is the only one today — adding/removing a Threema ID has to
|
||||
* synchronise with the central pieced-threema-gateway relay's `routes`
|
||||
* table (uniqueness enforced there, not in K8s). The
|
||||
* /api/tenants/:name/threema/routes endpoint owns that two-step
|
||||
* coordination (relay first, then K8s, with compensation on K8s
|
||||
* failure). Other channels just patch the K8s spec directly.
|
||||
*/
|
||||
const RELAY_MANAGED_CHANNELS = new Set(["threema"]);
|
||||
|
||||
interface ChannelUsersProps {
|
||||
tenantName: string;
|
||||
/** Currently enabled channel packages (e.g. ["telegram", "discord"]) */
|
||||
@@ -34,6 +52,14 @@ export function ChannelUsers({
|
||||
const [inputValues, setInputValues] = useState<Record<string, string>>({});
|
||||
const [channelUsers, setChannelUsers] =
|
||||
useState<Record<string, string[]>>(initialChannelUsers);
|
||||
/** Which channel's QR helper modal is open, if any. */
|
||||
const [showQrFor, setShowQrFor] = useState<string | null>(null);
|
||||
/**
|
||||
* Tracks channels for which we've already auto-opened the helper
|
||||
* modal on this page load. Prevents the modal from re-popping every
|
||||
* time the user refocuses the input after dismissing it.
|
||||
*/
|
||||
const [autoOpened, setAutoOpened] = useState<Set<string>>(() => new Set());
|
||||
|
||||
const updateChannelUsers = useCallback(
|
||||
async (updated: Record<string, string[]>) => {
|
||||
@@ -60,6 +86,70 @@ export function ChannelUsers({
|
||||
[tenantName, router]
|
||||
);
|
||||
|
||||
/**
|
||||
* Threema (and any future relay-managed channel) uses a dedicated
|
||||
* endpoint that synchronises the central relay's routes table with
|
||||
* the K8s spec atomically. We call it per-ID rather than sending the
|
||||
* whole array because uniqueness is enforced ID-by-ID at the relay,
|
||||
* and the error UX of "this ID is taken" is per-add.
|
||||
*/
|
||||
const addToRelayChannel = useCallback(
|
||||
async (channel: string, threemaId: string) => {
|
||||
setSaving(true);
|
||||
setError("");
|
||||
try {
|
||||
const res = await fetch(
|
||||
`/api/tenants/${tenantName}/${channel}/routes`,
|
||||
{
|
||||
method: "POST",
|
||||
headers: { "Content-Type": "application/json" },
|
||||
body: JSON.stringify({ threemaId }),
|
||||
}
|
||||
);
|
||||
if (!res.ok) {
|
||||
const data = await res.json().catch(() => ({}));
|
||||
throw new Error(data.error || `Add failed (HTTP ${res.status})`);
|
||||
}
|
||||
setChannelUsers((prev) => {
|
||||
const current = prev[channel] ?? [];
|
||||
if (current.includes(threemaId)) return prev;
|
||||
return { ...prev, [channel]: [...current, threemaId] };
|
||||
});
|
||||
router.refresh();
|
||||
} catch (e: any) {
|
||||
setError(e.message);
|
||||
} finally {
|
||||
setSaving(false);
|
||||
}
|
||||
},
|
||||
[tenantName, router]
|
||||
);
|
||||
|
||||
const removeFromRelayChannel = useCallback(
|
||||
async (channel: string, threemaId: string) => {
|
||||
setSaving(true);
|
||||
setError("");
|
||||
try {
|
||||
const url = `/api/tenants/${tenantName}/${channel}/routes?threemaId=${encodeURIComponent(threemaId)}`;
|
||||
const res = await fetch(url, { method: "DELETE" });
|
||||
if (!res.ok) {
|
||||
const data = await res.json().catch(() => ({}));
|
||||
throw new Error(data.error || `Remove failed (HTTP ${res.status})`);
|
||||
}
|
||||
setChannelUsers((prev) => {
|
||||
const current = prev[channel] ?? [];
|
||||
return { ...prev, [channel]: current.filter((id) => id !== threemaId) };
|
||||
});
|
||||
router.refresh();
|
||||
} catch (e: any) {
|
||||
setError(e.message);
|
||||
} finally {
|
||||
setSaving(false);
|
||||
}
|
||||
},
|
||||
[tenantName, router]
|
||||
);
|
||||
|
||||
const handleAdd = useCallback(
|
||||
(channel: string) => {
|
||||
const userId = inputValues[channel]?.trim();
|
||||
@@ -71,18 +161,27 @@ export function ChannelUsers({
|
||||
return;
|
||||
}
|
||||
|
||||
const updated = {
|
||||
...channelUsers,
|
||||
[channel]: [...current, userId],
|
||||
};
|
||||
setInputValues((prev) => ({ ...prev, [channel]: "" }));
|
||||
updateChannelUsers(updated);
|
||||
|
||||
if (RELAY_MANAGED_CHANNELS.has(channel)) {
|
||||
void addToRelayChannel(channel, userId);
|
||||
} else {
|
||||
const updated = {
|
||||
...channelUsers,
|
||||
[channel]: [...current, userId],
|
||||
};
|
||||
updateChannelUsers(updated);
|
||||
}
|
||||
},
|
||||
[channelUsers, inputValues, updateChannelUsers, t]
|
||||
[channelUsers, inputValues, updateChannelUsers, addToRelayChannel, t]
|
||||
);
|
||||
|
||||
const handleRemove = useCallback(
|
||||
(channel: string, userId: string) => {
|
||||
if (RELAY_MANAGED_CHANNELS.has(channel)) {
|
||||
void removeFromRelayChannel(channel, userId);
|
||||
return;
|
||||
}
|
||||
const current = channelUsers[channel] || [];
|
||||
const updated = {
|
||||
...channelUsers,
|
||||
@@ -90,7 +189,7 @@ export function ChannelUsers({
|
||||
};
|
||||
updateChannelUsers(updated);
|
||||
},
|
||||
[channelUsers, updateChannelUsers]
|
||||
[channelUsers, updateChannelUsers, removeFromRelayChannel]
|
||||
);
|
||||
|
||||
if (enabledChannels.length === 0) return null;
|
||||
@@ -126,9 +225,19 @@ export function ChannelUsers({
|
||||
className="bg-surface-2 border border-border rounded-lg p-4"
|
||||
>
|
||||
<div className="flex items-center justify-between mb-3">
|
||||
<h4 className="text-sm font-medium text-text-primary capitalize">
|
||||
{channel}
|
||||
</h4>
|
||||
<div className="flex items-center gap-2">
|
||||
<h4 className="text-sm font-medium text-text-primary capitalize">
|
||||
{channel}
|
||||
</h4>
|
||||
{channel === "threema" && (
|
||||
<button
|
||||
onClick={() => setShowQrFor("threema")}
|
||||
className="text-xs font-medium text-accent hover:text-accent-dim cursor-pointer underline underline-offset-2"
|
||||
>
|
||||
{t("threemaSetup.showQr")}
|
||||
</button>
|
||||
)}
|
||||
</div>
|
||||
<span className="text-xs text-text-muted tabular-nums">
|
||||
{users.length} {t("users")}
|
||||
</span>
|
||||
@@ -176,6 +285,17 @@ export function ChannelUsers({
|
||||
[channel]: e.target.value,
|
||||
}))
|
||||
}
|
||||
onFocus={() => {
|
||||
// For threema specifically, open the QR helper the
|
||||
// first time the user clicks into the input on this
|
||||
// page load. We don't repeat after dismiss — the
|
||||
// "Show QR" button next to the channel name covers
|
||||
// re-opens on demand.
|
||||
if (channel === "threema" && !autoOpened.has("threema")) {
|
||||
setShowQrFor("threema");
|
||||
setAutoOpened((prev) => new Set(prev).add("threema"));
|
||||
}
|
||||
}}
|
||||
onKeyDown={(e) => {
|
||||
if (e.key === "Enter") handleAdd(channel);
|
||||
}}
|
||||
@@ -194,6 +314,11 @@ export function ChannelUsers({
|
||||
</div>
|
||||
);
|
||||
})}
|
||||
|
||||
<ThreemaQrModal
|
||||
open={showQrFor === "threema"}
|
||||
onClose={() => setShowQrFor(null)}
|
||||
/>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
82
src/components/channel-users/threema-qr-modal.tsx
Normal file
82
src/components/channel-users/threema-qr-modal.tsx
Normal file
@@ -0,0 +1,82 @@
|
||||
"use client";
|
||||
|
||||
import { useTranslations } from "next-intl";
|
||||
import { useEffect } from "react";
|
||||
import { THREEMA_GATEWAY } from "@/lib/threema-gateway-config";
|
||||
|
||||
interface ThreemaQrModalProps {
|
||||
open: boolean;
|
||||
onClose: () => void;
|
||||
}
|
||||
|
||||
/**
|
||||
* On-demand modal showing the QR for adding the assistant on Threema.
|
||||
* Triggered by the "Show QR" button in the threema channel card and
|
||||
* closes on overlay click, ESC, or the close button.
|
||||
*
|
||||
* Uses a plain <img> not next/image — image optimization adds nothing
|
||||
* for a 57KB static PNG and removes a potential source of rendering
|
||||
* bugs in the Next.js standalone build.
|
||||
*/
|
||||
export function ThreemaQrModal({ open, onClose }: ThreemaQrModalProps) {
|
||||
const t = useTranslations("channelUsers.threemaSetup");
|
||||
|
||||
useEffect(() => {
|
||||
if (!open) return;
|
||||
function onKey(e: KeyboardEvent) {
|
||||
if (e.key === "Escape") onClose();
|
||||
}
|
||||
window.addEventListener("keydown", onKey);
|
||||
return () => window.removeEventListener("keydown", onKey);
|
||||
}, [open, onClose]);
|
||||
|
||||
if (!open) return null;
|
||||
|
||||
return (
|
||||
<div
|
||||
onClick={onClose}
|
||||
className="fixed inset-0 z-50 flex items-center justify-center bg-black/60 backdrop-blur-sm p-4"
|
||||
>
|
||||
<div
|
||||
onClick={(e) => e.stopPropagation()}
|
||||
className="w-full max-w-md bg-surface-1 border border-border rounded-2xl p-6 shadow-2xl shadow-black/40 space-y-4"
|
||||
>
|
||||
<div className="flex items-start justify-between">
|
||||
<h3 className="text-base font-semibold text-text-primary">
|
||||
{t("title")}
|
||||
</h3>
|
||||
<button
|
||||
onClick={onClose}
|
||||
className="text-text-muted hover:text-text-primary text-xl leading-none"
|
||||
aria-label="Close"
|
||||
>
|
||||
×
|
||||
</button>
|
||||
</div>
|
||||
|
||||
<div className="flex justify-center">
|
||||
<div className="bg-white p-3 rounded-md">
|
||||
{/* eslint-disable-next-line @next/next/no-img-element */}
|
||||
<img
|
||||
src={THREEMA_GATEWAY.qrCodePath}
|
||||
alt={t("qrAlt", { gateway: THREEMA_GATEWAY.displayName })}
|
||||
width={220}
|
||||
height={220}
|
||||
style={{ display: "block" }}
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<p className="text-center text-xs text-text-muted font-mono">
|
||||
{THREEMA_GATEWAY.displayName}
|
||||
</p>
|
||||
|
||||
<ol className="list-decimal list-inside text-xs text-text-secondary space-y-1.5">
|
||||
<li>{t("step1")}</li>
|
||||
<li>{t("step2")}</li>
|
||||
<li>{t("step3")}</li>
|
||||
</ol>
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
@@ -162,12 +162,9 @@ export function BudgetEditableCard({
|
||||
<h3 className="font-display text-lg font-semibold text-text-primary mb-2">
|
||||
{t("budgetEditTitle")}
|
||||
</h3>
|
||||
<p className="text-sm text-text-secondary mb-4">
|
||||
<p className="text-sm text-text-secondary mb-5">
|
||||
{t("budgetEditDescription")}
|
||||
</p>
|
||||
<div className="text-xs text-amber-400 bg-amber-400/10 border border-amber-400/20 rounded-lg px-3 py-2 mb-5">
|
||||
{t("budgetOrgScopeWarning")}
|
||||
</div>
|
||||
|
||||
<form onSubmit={onSubmit} className="space-y-4">
|
||||
{/* Mode toggle: unlimited vs capped. Two radios are
|
||||
|
||||
@@ -3,7 +3,7 @@
|
||||
import { useState, useCallback, useEffect, useRef } from "react";
|
||||
import { useTranslations } from "next-intl";
|
||||
import { Card } from "@/components/ui/card";
|
||||
import { PACKAGE_CATALOG, type PackageDef } from "@/lib/packages";
|
||||
import { PACKAGE_CATALOG, DEFAULT_PACKAGE_IDS, type PackageDef } from "@/lib/packages";
|
||||
import { isPersonalOrgName, displayOrgNameFor } from "@/lib/personal-org";
|
||||
import {
|
||||
configureStepSchema,
|
||||
@@ -69,6 +69,7 @@ translation, and general question answering.
|
||||
`;
|
||||
|
||||
const CATEGORIES = [
|
||||
{ key: "core" as const, labelKey: "categories.core" },
|
||||
{ key: "channel" as const, labelKey: "categories.channels" },
|
||||
{ key: "skill" as const, labelKey: "categories.skills" },
|
||||
] as const;
|
||||
@@ -198,7 +199,13 @@ export function OnboardingWizard({
|
||||
agentName: "Assistant",
|
||||
soulMd: FALLBACK_SOUL.replace("{company}", displayOrgName),
|
||||
agentsMd: FALLBACK_AGENTS,
|
||||
packages: [] as string[],
|
||||
// CORE defaults: heartbeat + cron + active-memory pre-selected so
|
||||
// the assistant can be proactive, run scheduled tasks, and recall
|
||||
// stable context out of the box. Customers can untoggle any of
|
||||
// them before submitting. core-voice is fully wired (Phase B)
|
||||
// but stays unselected — opt-in keeps audio spend predictable
|
||||
// for tenants who don't intend to use voice channels.
|
||||
packages: [...DEFAULT_PACKAGE_IDS] as string[],
|
||||
billingAddress: {
|
||||
// For personal accounts, leave the company field empty — it'll
|
||||
// appear on invoices. The user can still type something if they
|
||||
@@ -691,7 +698,7 @@ export function OnboardingWizard({
|
||||
<button
|
||||
type="button"
|
||||
onClick={() => togglePackage(pkg.id)}
|
||||
className="w-full flex items-center justify-between px-3 py-2.5 cursor-pointer hover:bg-surface-3/30 transition-colors"
|
||||
className="w-full flex items-center justify-between px-3 py-2.5 transition-colors cursor-pointer hover:bg-surface-3/30"
|
||||
>
|
||||
<div className="text-left">
|
||||
<span
|
||||
|
||||
@@ -30,6 +30,26 @@ export function PackageCard({
|
||||
const [error, setError] = useState<string | null>(null);
|
||||
|
||||
async function handleEnable() {
|
||||
if (pkg.customProvisioning) {
|
||||
// Platform-side provisioning, then add to packages list.
|
||||
setSaving(true);
|
||||
setError(null);
|
||||
try {
|
||||
const provRes = await fetch(`/api/tenants/${tenantName}/${pkg.id}`, {
|
||||
method: "POST",
|
||||
});
|
||||
if (!provRes.ok) {
|
||||
const err = await provRes.json().catch(() => ({}));
|
||||
throw new Error(err.error || `Provisioning failed (HTTP ${provRes.status})`);
|
||||
}
|
||||
await togglePackage(true);
|
||||
} catch (e: any) {
|
||||
setError(e.message);
|
||||
} finally {
|
||||
setSaving(false);
|
||||
}
|
||||
return;
|
||||
}
|
||||
if (pkg.requiresSecrets) {
|
||||
setShowModal(true);
|
||||
setSecrets({});
|
||||
@@ -40,6 +60,34 @@ export function PackageCard({
|
||||
await togglePackage(true);
|
||||
}
|
||||
|
||||
async function handleDisable() {
|
||||
setSaving(true);
|
||||
setError(null);
|
||||
try {
|
||||
if (pkg.customProvisioning) {
|
||||
// Revoke platform-side credentials FIRST so the relay drops
|
||||
// routes before the operator removes the channel from the
|
||||
// OpenClaw config. Partial-success (token revoked, OpenBao
|
||||
// delete failed) returns 503 with partial=true and we surface
|
||||
// the error rather than continuing — the secret may still be
|
||||
// valid in OpenBao and rolling back the relay revoke isn't
|
||||
// possible (it cascaded to routes).
|
||||
const deprovRes = await fetch(`/api/tenants/${tenantName}/${pkg.id}`, {
|
||||
method: "DELETE",
|
||||
});
|
||||
if (!deprovRes.ok) {
|
||||
const err = await deprovRes.json().catch(() => ({}));
|
||||
throw new Error(err.error || `Deprovisioning failed (HTTP ${deprovRes.status})`);
|
||||
}
|
||||
}
|
||||
await togglePackage(false);
|
||||
} catch (e: any) {
|
||||
setError(e.message);
|
||||
} finally {
|
||||
setSaving(false);
|
||||
}
|
||||
}
|
||||
|
||||
async function togglePackage(enable: boolean) {
|
||||
setSaving(true);
|
||||
try {
|
||||
@@ -124,7 +172,7 @@ export function PackageCard({
|
||||
)}
|
||||
{canEdit ? (
|
||||
<button
|
||||
onClick={enabled ? () => togglePackage(false) : handleEnable}
|
||||
onClick={enabled ? handleDisable : handleEnable}
|
||||
disabled={saving}
|
||||
className={`ml-auto rounded-lg px-3 py-1.5 text-xs font-medium transition-all cursor-pointer ${
|
||||
enabled
|
||||
|
||||
@@ -15,6 +15,7 @@ interface Props {
|
||||
}
|
||||
|
||||
const CATEGORIES = [
|
||||
{ key: "core" as const, labelKey: "categories.core" },
|
||||
{ key: "channel" as const, labelKey: "categories.channels" },
|
||||
{ key: "skill" as const, labelKey: "categories.skills" },
|
||||
] as const;
|
||||
|
||||
112
src/lib/k8s.ts
112
src/lib/k8s.ts
@@ -173,3 +173,115 @@ export async function setTenantAnnotation(
|
||||
}
|
||||
return res.json() as Promise<PiecedTenant>;
|
||||
}
|
||||
|
||||
// ---------------------------------------------------------------------------
|
||||
// OpenClaw config ConfigMap helpers (admin-only feature: per-tenant version
|
||||
// override + platform default).
|
||||
//
|
||||
// The ConfigMap lives in the operator's namespace (`pieced-system`). The
|
||||
// portal's ServiceAccount needs `get/patch` on configmaps in that namespace
|
||||
// — rules added in the gitops repo.
|
||||
//
|
||||
// Tag-only by design — see operator notes for rationale.
|
||||
// ---------------------------------------------------------------------------
|
||||
|
||||
const OPENCLAW_CONFIGMAP_NAME = "pieced-openclaw-config";
|
||||
|
||||
/**
|
||||
* Operator namespace. Reads the env var so the portal can be deployed in
|
||||
* non-default namespaces without code changes; defaults to "pieced-system"
|
||||
* matching the operator's chart default.
|
||||
*/
|
||||
function getOperatorNamespace(): string {
|
||||
return process.env.OPERATOR_NAMESPACE ?? "pieced-system";
|
||||
}
|
||||
|
||||
export interface OpenClawDefaults {
|
||||
/** Image tag (e.g. "2026.4.22"). Empty string means unset. */
|
||||
defaultTag: string;
|
||||
}
|
||||
|
||||
/**
|
||||
* Read the platform-default OpenClaw image tag. Returns empty string
|
||||
* if unset, and `{ defaultTag: "" }` if the ConfigMap doesn't exist yet
|
||||
* — the operator's built-in fallback is invisible to the portal by
|
||||
* design (we don't want the UI to claim "current default: 2026.x" when
|
||||
* it's actually the operator binary's baked-in version; that would be
|
||||
* misleading once the binary updates).
|
||||
*/
|
||||
export async function getOpenClawDefaults(): Promise<OpenClawDefaults> {
|
||||
const ns = getOperatorNamespace();
|
||||
const url = `${getBaseUrl()}/api/v1/namespaces/${ns}/configmaps/${OPENCLAW_CONFIGMAP_NAME}`;
|
||||
const res = await fetch(url, {
|
||||
headers: { Accept: "application/json", ...getAuthHeaders() },
|
||||
});
|
||||
if (res.status === 404) {
|
||||
return { defaultTag: "" };
|
||||
}
|
||||
if (!res.ok) {
|
||||
const text = await res.text();
|
||||
const err = new Error(
|
||||
`K8s GET configmap ${OPENCLAW_CONFIGMAP_NAME}: ${res.status} ${text}`
|
||||
);
|
||||
(err as any).statusCode = res.status;
|
||||
throw err;
|
||||
}
|
||||
const cm = (await res.json()) as { data?: Record<string, string> };
|
||||
return { defaultTag: cm.data?.defaultTag ?? "" };
|
||||
}
|
||||
|
||||
/**
|
||||
* Update the platform-default OpenClaw image tag. Empty string clears
|
||||
* the field (operator falls back to its built-in default).
|
||||
*
|
||||
* Creates the ConfigMap if it doesn't exist (PATCH on missing resource
|
||||
* 404s; we retry as POST). Keeps the admin UI usable on a fresh install
|
||||
* where the helm-shipped CM was deleted or never created.
|
||||
*/
|
||||
export async function setOpenClawDefaults(
|
||||
defaults: OpenClawDefaults
|
||||
): Promise<OpenClawDefaults> {
|
||||
const ns = getOperatorNamespace();
|
||||
const url = `${getBaseUrl()}/api/v1/namespaces/${ns}/configmaps/${OPENCLAW_CONFIGMAP_NAME}`;
|
||||
const patch = { data: { defaultTag: defaults.defaultTag } };
|
||||
const res = await fetch(url, {
|
||||
method: "PATCH",
|
||||
headers: {
|
||||
Accept: "application/json",
|
||||
"Content-Type": "application/merge-patch+json",
|
||||
...getAuthHeaders(),
|
||||
},
|
||||
body: JSON.stringify(patch),
|
||||
});
|
||||
if (res.status === 404) {
|
||||
const createUrl = `${getBaseUrl()}/api/v1/namespaces/${ns}/configmaps`;
|
||||
const createRes = await fetch(createUrl, {
|
||||
method: "POST",
|
||||
headers: {
|
||||
Accept: "application/json",
|
||||
"Content-Type": "application/json",
|
||||
...getAuthHeaders(),
|
||||
},
|
||||
body: JSON.stringify({
|
||||
apiVersion: "v1",
|
||||
kind: "ConfigMap",
|
||||
metadata: { name: OPENCLAW_CONFIGMAP_NAME, namespace: ns },
|
||||
data: patch.data,
|
||||
}),
|
||||
});
|
||||
if (!createRes.ok) {
|
||||
const text = await createRes.text();
|
||||
throw new Error(
|
||||
`K8s POST configmap ${OPENCLAW_CONFIGMAP_NAME}: ${createRes.status} ${text}`
|
||||
);
|
||||
}
|
||||
return defaults;
|
||||
}
|
||||
if (!res.ok) {
|
||||
const text = await res.text();
|
||||
throw new Error(
|
||||
`K8s PATCH configmap ${OPENCLAW_CONFIGMAP_NAME}: ${res.status} ${text}`
|
||||
);
|
||||
}
|
||||
return defaults;
|
||||
}
|
||||
|
||||
@@ -93,6 +93,94 @@ export async function listTeams(): Promise<any[]> {
|
||||
return Array.isArray(data) ? data : data?.data ?? data?.teams ?? [];
|
||||
}
|
||||
|
||||
/**
|
||||
* Find a virtual key on a team by its alias and return its current
|
||||
* state (token, spend, budget cap, reset cadence). Returns null if
|
||||
* the alias doesn't match any key on the team.
|
||||
*
|
||||
* Why we need this
|
||||
* ----------------
|
||||
* Per-tenant budgets live on the virtual key, not the team. The
|
||||
* portal needs to:
|
||||
* 1. Display the current key's `max_budget` / `budget_duration` /
|
||||
* `spend` on the tenant detail page.
|
||||
* 2. Pass the key's `token` to `/key/update` when the customer
|
||||
* changes the budget.
|
||||
*
|
||||
* The token is opaque to the customer; the operator's
|
||||
* `FindKeyByAlias` does the same lookup for stale-key cleanup. We
|
||||
* mirror its API call here.
|
||||
*/
|
||||
export async function findKeyByAlias(
|
||||
teamId: string,
|
||||
keyAlias: string
|
||||
): Promise<{
|
||||
token: string;
|
||||
spend: number;
|
||||
maxBudget: number | null;
|
||||
budgetDuration: string | null;
|
||||
} | null> {
|
||||
const data = await litellmFetch(
|
||||
`/key/list?team_id=${encodeURIComponent(teamId)}&return_full_object=true&include_team_keys=true`
|
||||
);
|
||||
const keys: any[] = Array.isArray(data?.keys)
|
||||
? data.keys
|
||||
: Array.isArray(data?.data)
|
||||
? data.data
|
||||
: Array.isArray(data)
|
||||
? data
|
||||
: [];
|
||||
for (const k of keys) {
|
||||
if (typeof k !== "object" || k === null) continue;
|
||||
const alias = k.key_alias ?? k.keyAlias;
|
||||
if (alias !== keyAlias) continue;
|
||||
if (typeof k.token !== "string" || !k.token) continue;
|
||||
return {
|
||||
token: k.token,
|
||||
spend: typeof k.spend === "number" ? k.spend : Number(k.spend) || 0,
|
||||
maxBudget:
|
||||
typeof k.max_budget === "number"
|
||||
? k.max_budget
|
||||
: k.max_budget == null
|
||||
? null
|
||||
: Number(k.max_budget) || null,
|
||||
budgetDuration:
|
||||
typeof k.budget_duration === "string" ? k.budget_duration : null,
|
||||
};
|
||||
}
|
||||
return null;
|
||||
}
|
||||
|
||||
/**
|
||||
* Update a virtual key's budget cap and reset duration.
|
||||
*
|
||||
* Pass `maxBudget: null` to remove the cap. Pass `budgetDuration:
|
||||
* null` to make the budget never reset (lifetime cap).
|
||||
*
|
||||
* Identified by `key` parameter — accepts either the raw `sk-...`
|
||||
* token or its hash (LiteLLM accepts both shapes on /key/update).
|
||||
* The portal flow uses the hash returned by `findKeyByAlias`.
|
||||
*/
|
||||
export async function updateKeyBudget(
|
||||
key: string,
|
||||
changes: {
|
||||
maxBudget?: number | null;
|
||||
budgetDuration?: string | null;
|
||||
}
|
||||
): Promise<void> {
|
||||
const body: Record<string, any> = { key };
|
||||
if (changes.maxBudget !== undefined) {
|
||||
body.max_budget = changes.maxBudget;
|
||||
}
|
||||
if (changes.budgetDuration !== undefined) {
|
||||
body.budget_duration = changes.budgetDuration;
|
||||
}
|
||||
await litellmFetch("/key/update", {
|
||||
method: "POST",
|
||||
body: JSON.stringify(body),
|
||||
});
|
||||
}
|
||||
|
||||
/**
|
||||
* Get LiteLLM health status.
|
||||
*/
|
||||
|
||||
@@ -1,9 +1,45 @@
|
||||
/**
|
||||
* Portal-side package catalog. Hardcoded mirror of the operator-side
|
||||
* catalog ConfigMap (deploy/helm/pieced-operator/templates/catalog-cm.yaml).
|
||||
*
|
||||
* The two have to stay in sync:
|
||||
* - `id` here must match the catalog key in the ConfigMap.
|
||||
* - `secrets[].key` here must match the catalog's env_vars[].secret_key
|
||||
* so that POST /api/tenants/:name/secrets writes to the same OpenBao
|
||||
* path the operator's ExternalSecret reads from.
|
||||
* - `requiresSecrets` is true when the catalog declares any env_var
|
||||
* that is a secret (vault_path_suffix set, no default value).
|
||||
*
|
||||
* Category model (Phase A rework):
|
||||
* - core — platform-behaviour toggles (heartbeat, cron,
|
||||
* active-memory, voice). Mostly no secrets. core-voice is
|
||||
* fully wired (Phase B): toggling installs the STT / TTS /
|
||||
* Talk surface via the operator's config_patch, routed
|
||||
* through LiteLLM (pieced-stt, pieced-tts-inbound,
|
||||
* pieced-tts-talk).
|
||||
* - channel — messaging integration.
|
||||
* - skill — ClawHub skill install.
|
||||
*
|
||||
* Custom provisioning (Threema):
|
||||
* The `threema` channel sets `requiresSecrets: false` because its
|
||||
* credentials are platform-issued, not customer-entered. Enabling
|
||||
* threema goes through a dedicated endpoint
|
||||
* (/api/tenants/:name/threema) that mints token + HMAC secret from
|
||||
* the central pieced-threema-gateway relay and writes them to OpenBao
|
||||
* at secret/data/tenants/<name>/threema-relay before the package is
|
||||
* added to spec.packages. Disabling reverses both steps. The
|
||||
* `customProvisioning` flag here tells the package-card UI to use
|
||||
* that endpoint instead of the standard /secrets+PATCH dance.
|
||||
*/
|
||||
|
||||
export interface PackageSecretField {
|
||||
key: string;
|
||||
labelKey: string;
|
||||
placeholderKey: string;
|
||||
}
|
||||
|
||||
export type PackageCategory = "core" | "channel" | "skill";
|
||||
|
||||
export interface PackageDef {
|
||||
id: string;
|
||||
name: string;
|
||||
@@ -12,10 +48,53 @@ export interface PackageDef {
|
||||
secrets?: PackageSecretField[];
|
||||
instructionsKey?: string;
|
||||
disclaimerKey?: string;
|
||||
category: "channel" | "skill";
|
||||
category: PackageCategory;
|
||||
/**
|
||||
* When true, enabling/disabling this package goes through
|
||||
* /api/tenants/:name/<id> (POST/DELETE) instead of the generic
|
||||
* /secrets+PATCH flow. The handler at that path does platform-side
|
||||
* provisioning (mint credentials, register with sibling services, etc.)
|
||||
* that the customer is not aware of.
|
||||
*/
|
||||
customProvisioning?: boolean;
|
||||
}
|
||||
|
||||
export const PACKAGE_CATALOG: PackageDef[] = [
|
||||
// -------------------------------------------------------------------------
|
||||
// CORE
|
||||
// -------------------------------------------------------------------------
|
||||
{
|
||||
id: "core-heartbeat",
|
||||
name: "Heartbeat (Proactive Checks)",
|
||||
descriptionKey: "packages.coreHeartbeat.description",
|
||||
requiresSecrets: false,
|
||||
category: "core",
|
||||
},
|
||||
{
|
||||
id: "core-cron",
|
||||
name: "Scheduled Tasks (Cron)",
|
||||
descriptionKey: "packages.coreCron.description",
|
||||
requiresSecrets: false,
|
||||
category: "core",
|
||||
},
|
||||
{
|
||||
id: "core-active-memory",
|
||||
name: "Active Memory",
|
||||
descriptionKey: "packages.coreActiveMemory.description",
|
||||
requiresSecrets: false,
|
||||
category: "core",
|
||||
},
|
||||
{
|
||||
id: "core-voice",
|
||||
name: "Voice Interaction",
|
||||
descriptionKey: "packages.coreVoice.description",
|
||||
requiresSecrets: false,
|
||||
category: "core",
|
||||
},
|
||||
|
||||
// -------------------------------------------------------------------------
|
||||
// CHANNELS
|
||||
// -------------------------------------------------------------------------
|
||||
{
|
||||
id: "telegram",
|
||||
name: "Telegram",
|
||||
@@ -43,42 +122,201 @@ export const PACKAGE_CATALOG: PackageDef[] = [
|
||||
labelKey: "packages.discord.botTokenLabel",
|
||||
placeholderKey: "packages.discord.botTokenPlaceholder",
|
||||
},
|
||||
// app-id was missing from the portal catalog historically while the
|
||||
// operator catalog declared DISCORD_APP_ID as a required env var.
|
||||
// Tenants who enabled Discord ended up with the env var blank
|
||||
// because the secrets POST never wrote an `app-id` key to OpenBao
|
||||
// and the operator's ExternalSecret couldn't populate it. Added
|
||||
// here as part of the Phase A rework to close the alignment gap;
|
||||
// not strictly secret (the application ID is visible in the bot's
|
||||
// profile URL) but stored alongside the bot token for convenience.
|
||||
{
|
||||
key: "app-id",
|
||||
labelKey: "packages.discord.appIdLabel",
|
||||
placeholderKey: "packages.discord.appIdPlaceholder",
|
||||
},
|
||||
],
|
||||
instructionsKey: "packages.discord.instructions",
|
||||
disclaimerKey: "packages.discord.disclaimer",
|
||||
category: "channel",
|
||||
},
|
||||
{
|
||||
id: "email",
|
||||
name: "Email",
|
||||
descriptionKey: "packages.email.description",
|
||||
requiresSecrets: true,
|
||||
secrets: [
|
||||
{ key: "smtp-host", labelKey: "packages.email.smtpHostLabel", placeholderKey: "packages.email.smtpHostPlaceholder" },
|
||||
{ key: "smtp-user", labelKey: "packages.email.smtpUserLabel", placeholderKey: "packages.email.smtpUserPlaceholder" },
|
||||
{ key: "smtp-password", labelKey: "packages.email.smtpPasswordLabel", placeholderKey: "packages.email.smtpPasswordPlaceholder" },
|
||||
{ key: "imap-host", labelKey: "packages.email.imapHostLabel", placeholderKey: "packages.email.imapHostPlaceholder" },
|
||||
],
|
||||
instructionsKey: "packages.email.instructions",
|
||||
disclaimerKey: "packages.email.disclaimer",
|
||||
id: "threema",
|
||||
name: "Threema",
|
||||
descriptionKey: "packages.threema.description",
|
||||
// No customer-entered secrets. The token + hmac secret are minted
|
||||
// server-side by the relay's /admin/tokens endpoint when the
|
||||
// package is enabled, and stored in OpenBao by the portal. The
|
||||
// `customProvisioning` flag steers the PackageCard UI through the
|
||||
// dedicated /api/tenants/:name/threema endpoint instead.
|
||||
requiresSecrets: false,
|
||||
customProvisioning: true,
|
||||
instructionsKey: "packages.threema.instructions",
|
||||
disclaimerKey: "packages.threema.disclaimer",
|
||||
category: "channel",
|
||||
},
|
||||
|
||||
// -------------------------------------------------------------------------
|
||||
// SKILLS
|
||||
// -------------------------------------------------------------------------
|
||||
{
|
||||
id: "web-search",
|
||||
name: "Web Search",
|
||||
descriptionKey: "packages.webSearch.description",
|
||||
id: "git-cli",
|
||||
name: "Git CLI",
|
||||
descriptionKey: "packages.gitCli.description",
|
||||
requiresSecrets: false,
|
||||
category: "skill",
|
||||
},
|
||||
{
|
||||
id: "document-processing",
|
||||
name: "Document Processing",
|
||||
descriptionKey: "packages.documentProcessing.description",
|
||||
id: "github",
|
||||
name: "GitHub (gh CLI)",
|
||||
descriptionKey: "packages.github.description",
|
||||
requiresSecrets: true,
|
||||
secrets: [
|
||||
{
|
||||
key: "token",
|
||||
labelKey: "packages.github.tokenLabel",
|
||||
placeholderKey: "packages.github.tokenPlaceholder",
|
||||
},
|
||||
],
|
||||
instructionsKey: "packages.github.instructions",
|
||||
category: "skill",
|
||||
},
|
||||
{
|
||||
id: "gitea",
|
||||
name: "Gitea",
|
||||
descriptionKey: "packages.gitea.description",
|
||||
requiresSecrets: true,
|
||||
secrets: [
|
||||
{
|
||||
key: "token",
|
||||
labelKey: "packages.gitea.tokenLabel",
|
||||
placeholderKey: "packages.gitea.tokenPlaceholder",
|
||||
},
|
||||
],
|
||||
instructionsKey: "packages.gitea.instructions",
|
||||
category: "skill",
|
||||
},
|
||||
{
|
||||
id: "whisper-self-hosted",
|
||||
name: "Whisper (Self-Hosted Transcription)",
|
||||
descriptionKey: "packages.whisperSelfHosted.description",
|
||||
requiresSecrets: false,
|
||||
category: "skill",
|
||||
},
|
||||
{
|
||||
id: "searxng-local-search",
|
||||
name: "Web Search (SearXNG)",
|
||||
descriptionKey: "packages.searxngLocalSearch.description",
|
||||
requiresSecrets: false,
|
||||
category: "skill",
|
||||
},
|
||||
{
|
||||
id: "gog",
|
||||
name: "Google Workspace (Gog)",
|
||||
descriptionKey: "packages.gog.description",
|
||||
requiresSecrets: true,
|
||||
secrets: [
|
||||
{
|
||||
key: "client-id",
|
||||
labelKey: "packages.gog.clientIdLabel",
|
||||
placeholderKey: "packages.gog.clientIdPlaceholder",
|
||||
},
|
||||
{
|
||||
key: "client-secret",
|
||||
labelKey: "packages.gog.clientSecretLabel",
|
||||
placeholderKey: "packages.gog.clientSecretPlaceholder",
|
||||
},
|
||||
{
|
||||
key: "refresh-token",
|
||||
labelKey: "packages.gog.refreshTokenLabel",
|
||||
placeholderKey: "packages.gog.refreshTokenPlaceholder",
|
||||
},
|
||||
],
|
||||
instructionsKey: "packages.gog.instructions",
|
||||
disclaimerKey: "packages.gog.disclaimer",
|
||||
category: "skill",
|
||||
},
|
||||
{
|
||||
id: "mail",
|
||||
name: "Email (IMAP / SMTP)",
|
||||
descriptionKey: "packages.mail.description",
|
||||
requiresSecrets: true,
|
||||
secrets: [
|
||||
{
|
||||
key: "imap-host",
|
||||
labelKey: "packages.mail.imapHostLabel",
|
||||
placeholderKey: "packages.mail.imapHostPlaceholder",
|
||||
},
|
||||
{
|
||||
key: "imap-user",
|
||||
labelKey: "packages.mail.imapUserLabel",
|
||||
placeholderKey: "packages.mail.imapUserPlaceholder",
|
||||
},
|
||||
{
|
||||
key: "imap-pass",
|
||||
labelKey: "packages.mail.imapPassLabel",
|
||||
placeholderKey: "packages.mail.imapPassPlaceholder",
|
||||
},
|
||||
{
|
||||
key: "smtp-host",
|
||||
labelKey: "packages.mail.smtpHostLabel",
|
||||
placeholderKey: "packages.mail.smtpHostPlaceholder",
|
||||
},
|
||||
{
|
||||
key: "smtp-user",
|
||||
labelKey: "packages.mail.smtpUserLabel",
|
||||
placeholderKey: "packages.mail.smtpUserPlaceholder",
|
||||
},
|
||||
{
|
||||
key: "smtp-pass",
|
||||
labelKey: "packages.mail.smtpPassLabel",
|
||||
placeholderKey: "packages.mail.smtpPassPlaceholder",
|
||||
},
|
||||
],
|
||||
instructionsKey: "packages.mail.instructions",
|
||||
disclaimerKey: "packages.mail.disclaimer",
|
||||
category: "skill",
|
||||
},
|
||||
];
|
||||
|
||||
export function getPackageDef(id: string): PackageDef | undefined {
|
||||
return PACKAGE_CATALOG.find((p) => p.id === id);
|
||||
}
|
||||
|
||||
/**
|
||||
* IDs of channel-category packages. Derived from the catalog so it
|
||||
* cannot drift from the source of truth (previously hardcoded as
|
||||
* `["telegram", "discord", "email"]` in tenants/[name]/page.tsx —
|
||||
* removed as part of the Phase A package-model rework).
|
||||
*
|
||||
* Consumers: tenant detail page (filter spec.packages to channel set
|
||||
* before rendering the channel-users panel).
|
||||
*/
|
||||
export const CHANNEL_PACKAGE_IDS: string[] = PACKAGE_CATALOG
|
||||
.filter((p) => p.category === "channel")
|
||||
.map((p) => p.id);
|
||||
|
||||
/**
|
||||
* Default packages selected when the wizard opens a fresh onboarding
|
||||
* request. The three CORE behaviours that make the assistant feel
|
||||
* "smart out of the box":
|
||||
* - heartbeat: proactive checks (otherwise the assistant is purely
|
||||
* reactive).
|
||||
* - cron: scheduled tasks (daily briefings, reminders).
|
||||
* - active-memory: long-term recall of stable preferences and habits.
|
||||
*
|
||||
* Each adds some token cost — active-memory the most (one extra
|
||||
* sub-agent turn per inbound message) — so customers can untoggle any
|
||||
* of them before submitting.
|
||||
*
|
||||
* core-voice is intentionally NOT a default. It is fully wired (Phase B)
|
||||
* and customers can enable it from the wizard, but it incurs separate
|
||||
* audio spend on every inbound voice note (Whisper STT) and every
|
||||
* outbound reply (kani-tts / kokoro-fastapi via LiteLLM). Opt-in keeps
|
||||
* cost predictable for tenants who don't intend to use voice channels.
|
||||
*/
|
||||
export const DEFAULT_PACKAGE_IDS: string[] = [
|
||||
"core-heartbeat",
|
||||
"core-cron",
|
||||
"core-active-memory",
|
||||
];
|
||||
|
||||
33
src/lib/threema-gateway-config.ts
Normal file
33
src/lib/threema-gateway-config.ts
Normal file
@@ -0,0 +1,33 @@
|
||||
/**
|
||||
* Threema central gateway info shown to customers.
|
||||
*
|
||||
* Today PieCed runs exactly one Threema Gateway account (*AIAGENT) and
|
||||
* every tenant talks to that. The constants below are hardcoded for
|
||||
* that account. The values are intentionally kept here (and not split
|
||||
* across i18n / env / runtime config) so when we move to multiple
|
||||
* gateway accounts there's a single file to refactor.
|
||||
*
|
||||
* To go dynamic (future):
|
||||
* 1. Replace `THREEMA_GATEWAY` constant with a runtime lookup —
|
||||
* either per-tenant from the relay's admin API, or from an
|
||||
* env var that lists the active account.
|
||||
* 2. Move the QR PNG into a server-rendered route that takes a
|
||||
* gateway ID query param.
|
||||
* 3. Update consumers (today only ThreemaSetup) to accept the
|
||||
* gateway info as a prop and pass it from a server component.
|
||||
*
|
||||
* In display contexts we strip the leading asterisk from the Threema
|
||||
* ID — customers don't understand the `*X` prefix convention used for
|
||||
* Gateway accounts, and the QR code carries the real value anyway. We
|
||||
* keep the asterisk only for places where the technical value matters
|
||||
* (server-side message routing, debug logs).
|
||||
*/
|
||||
|
||||
export const THREEMA_GATEWAY = {
|
||||
/** Technical Threema Gateway ID, with leading asterisk. */
|
||||
id: "*AIAGENT",
|
||||
/** Display name shown to customers (no asterisk). */
|
||||
displayName: "AIAGENT",
|
||||
/** Public path to the QR code PNG served from `public/`. */
|
||||
qrCodePath: "/threema/qr_code_AIAGENT.png",
|
||||
} as const;
|
||||
202
src/lib/threema-relay.ts
Normal file
202
src/lib/threema-relay.ts
Normal file
@@ -0,0 +1,202 @@
|
||||
/**
|
||||
* Admin client for the central pieced-threema-gateway relay.
|
||||
*
|
||||
* All calls authenticate with a static admin bearer (THREEMA_RELAY_ADMIN_TOKEN
|
||||
* from env, sourced from OpenBao at secret/data/threema-gateway/admin). The
|
||||
* portal is the only caller; never expose these functions through a public
|
||||
* HTTP surface.
|
||||
*
|
||||
* Resilience: every method returns a strongly-typed result rather than
|
||||
* throwing. Network errors surface as `{ ok: false, kind: 'network' }`,
|
||||
* 4xx/5xx as `{ ok: false, kind: 'http', status, ... }`. Callers that need
|
||||
* to compensate (e.g. delete a route after a K8s patch failure) can
|
||||
* inspect `kind` to decide whether retry is sensible.
|
||||
*/
|
||||
|
||||
const RELAY_URL = (process.env.THREEMA_RELAY_URL ?? "").replace(/\/+$/, "");
|
||||
const ADMIN_TOKEN = process.env.THREEMA_RELAY_ADMIN_TOKEN ?? "";
|
||||
|
||||
function assertConfigured(): void {
|
||||
if (!RELAY_URL) throw new Error("THREEMA_RELAY_URL not set");
|
||||
if (!ADMIN_TOKEN) throw new Error("THREEMA_RELAY_ADMIN_TOKEN not set");
|
||||
}
|
||||
|
||||
type RelayError =
|
||||
| { ok: false; kind: "network"; message: string }
|
||||
| { ok: false; kind: "http"; status: number; message: string; body?: unknown };
|
||||
|
||||
export type RelayResult<T> = ({ ok: true } & T) | RelayError;
|
||||
|
||||
async function call<T>(
|
||||
path: string,
|
||||
init: RequestInit,
|
||||
): Promise<RelayResult<T>> {
|
||||
assertConfigured();
|
||||
try {
|
||||
const res = await fetch(`${RELAY_URL}${path}`, {
|
||||
...init,
|
||||
headers: {
|
||||
...(init.headers ?? {}),
|
||||
Authorization: `Bearer ${ADMIN_TOKEN}`,
|
||||
"Content-Type": "application/json",
|
||||
},
|
||||
});
|
||||
let body: unknown = null;
|
||||
const text = await res.text().catch(() => "");
|
||||
if (text) {
|
||||
try {
|
||||
body = JSON.parse(text);
|
||||
} catch {
|
||||
body = text;
|
||||
}
|
||||
}
|
||||
if (!res.ok) {
|
||||
const message =
|
||||
(body && typeof body === "object" && "error" in body && typeof (body as Record<string, unknown>).error === "string"
|
||||
? ((body as Record<string, unknown>).error as string)
|
||||
: `HTTP ${res.status}`);
|
||||
return { ok: false, kind: "http", status: res.status, message, body };
|
||||
}
|
||||
return { ok: true, ...(body as Record<string, unknown>) } as RelayResult<T>;
|
||||
} catch (e) {
|
||||
return { ok: false, kind: "network", message: (e as Error).message };
|
||||
}
|
||||
}
|
||||
|
||||
// ---------------------------------------------------------------------------
|
||||
// Tokens
|
||||
// ---------------------------------------------------------------------------
|
||||
|
||||
/** Mint (or rotate) per-tenant bearer + HMAC secret. */
|
||||
export function mintToken(
|
||||
tenantName: string,
|
||||
): Promise<RelayResult<{ token: string; hmacSecret: string }>> {
|
||||
return call("/admin/tokens", {
|
||||
method: "POST",
|
||||
body: JSON.stringify({ tenantName }),
|
||||
});
|
||||
}
|
||||
|
||||
/**
|
||||
* Revoke per-tenant token. Cascades — relay also deletes all routes
|
||||
* owned by this tenant.
|
||||
*/
|
||||
export function revokeToken(
|
||||
tenantName: string,
|
||||
): Promise<RelayResult<{ deletedRoutes: number }>> {
|
||||
return call(`/admin/tokens/${encodeURIComponent(tenantName)}`, {
|
||||
method: "DELETE",
|
||||
});
|
||||
}
|
||||
|
||||
export function tokenExists(
|
||||
tenantName: string,
|
||||
): Promise<RelayResult<{ exists: true }>> {
|
||||
return call(`/admin/tokens/${encodeURIComponent(tenantName)}`, {
|
||||
method: "GET",
|
||||
});
|
||||
}
|
||||
|
||||
// ---------------------------------------------------------------------------
|
||||
// Routes
|
||||
// ---------------------------------------------------------------------------
|
||||
|
||||
export interface RouteRow {
|
||||
threema_id: string;
|
||||
tenant_name: string;
|
||||
created_at: string;
|
||||
created_by: string;
|
||||
}
|
||||
|
||||
export function listRoutes(
|
||||
tenantName: string,
|
||||
): Promise<RelayResult<{ routes: RouteRow[] }>> {
|
||||
return call(`/admin/routes?tenant=${encodeURIComponent(tenantName)}`, {
|
||||
method: "GET",
|
||||
});
|
||||
}
|
||||
|
||||
/**
|
||||
* Create a route. The relay enforces uniqueness — on conflict returns
|
||||
* status 409 with body `{ ownedBy: <tenantName> | null }`. Callers
|
||||
* should inspect `kind === 'http'` + status 409 + body.ownedBy to
|
||||
* distinguish idempotent self-claim from real conflict.
|
||||
*/
|
||||
export function createRoute(
|
||||
tenantName: string,
|
||||
threemaId: string,
|
||||
): Promise<RelayResult<{ ok: true }>> {
|
||||
return call("/admin/routes", {
|
||||
method: "POST",
|
||||
body: JSON.stringify({ tenantName, threemaId }),
|
||||
});
|
||||
}
|
||||
|
||||
export function deleteRoute(
|
||||
tenantName: string,
|
||||
threemaId: string,
|
||||
): Promise<RelayResult<{ ok: true }>> {
|
||||
return call(
|
||||
`/admin/routes/${encodeURIComponent(threemaId)}?tenant=${encodeURIComponent(tenantName)}`,
|
||||
{ method: "DELETE" },
|
||||
);
|
||||
}
|
||||
|
||||
// ---------------------------------------------------------------------------
|
||||
// Usage
|
||||
// ---------------------------------------------------------------------------
|
||||
|
||||
export interface UsageBreakdown {
|
||||
tenant: string;
|
||||
from: string;
|
||||
to: string;
|
||||
totals: { in: number; out: number };
|
||||
daily: Array<{ day: string; direction: "in" | "out"; count: number }>;
|
||||
}
|
||||
|
||||
export function getUsage(
|
||||
tenantName: string,
|
||||
from?: Date,
|
||||
to?: Date,
|
||||
): Promise<RelayResult<UsageBreakdown>> {
|
||||
const qs = new URLSearchParams({ tenant: tenantName });
|
||||
if (from) qs.set("from", from.toISOString());
|
||||
if (to) qs.set("to", to.toISOString());
|
||||
return call(`/admin/usage?${qs.toString()}`, { method: "GET" });
|
||||
}
|
||||
|
||||
// ---------------------------------------------------------------------------
|
||||
// Health
|
||||
// ---------------------------------------------------------------------------
|
||||
|
||||
export function health(): Promise<RelayResult<{ credits: number }>> {
|
||||
return call("/admin/health", { method: "GET" });
|
||||
}
|
||||
|
||||
// ---------------------------------------------------------------------------
|
||||
// Helpers for caller code
|
||||
// ---------------------------------------------------------------------------
|
||||
|
||||
/**
|
||||
* Type guard: did this route-create fail because the ID is owned by
|
||||
* someone else? Distinguishes from "owned by SAME tenant" (idempotent).
|
||||
*/
|
||||
export function isRouteConflictForOtherTenant(
|
||||
result: RelayResult<{ ok: true }>,
|
||||
tenantName: string,
|
||||
): boolean {
|
||||
if (result.ok) return false;
|
||||
if (result.kind !== "http" || result.status !== 409) return false;
|
||||
const body = result.body as { ownedBy?: string | null } | undefined;
|
||||
return !!body?.ownedBy && body.ownedBy !== tenantName;
|
||||
}
|
||||
|
||||
export function isRouteConflictForSameTenant(
|
||||
result: RelayResult<{ ok: true }>,
|
||||
tenantName: string,
|
||||
): boolean {
|
||||
if (result.ok) return false;
|
||||
if (result.kind !== "http" || result.status !== 409) return false;
|
||||
const body = result.body as { ownedBy?: string | null } | undefined;
|
||||
return body?.ownedBy === tenantName;
|
||||
}
|
||||
@@ -192,8 +192,7 @@
|
||||
"requests": "Anfragen",
|
||||
"budgetEdit": "Bearbeiten",
|
||||
"budgetEditTitle": "Budget festlegen",
|
||||
"budgetEditDescription": "Begrenzen Sie, wie viel Ihre Assistenten ausgeben können, bevor Anfragen abgelehnt werden.",
|
||||
"budgetOrgScopeWarning": "Dieses Budget gilt für alle Tenants Ihrer Organisation, nicht nur für diesen. Bei mehreren Tenants teilen sich diese das Limit.",
|
||||
"budgetEditDescription": "Begrenzen Sie, wie viel die Assistenten dieses Tenants ausgeben können, bevor Anfragen abgelehnt werden.",
|
||||
"budgetModeUnlimited": "Kein Limit",
|
||||
"budgetModeUnlimitedDescription": "Beliebige Ausgaben, kein Limit.",
|
||||
"budgetModeCapped": "Limit festlegen",
|
||||
@@ -215,7 +214,8 @@
|
||||
"packages": {
|
||||
"categories": {
|
||||
"channels": "Kanäle",
|
||||
"skills": "Fähigkeiten"
|
||||
"skills": "Fähigkeiten",
|
||||
"core": "Kern"
|
||||
},
|
||||
"enable": "Aktivieren",
|
||||
"disable": "Deaktivieren",
|
||||
@@ -240,29 +240,78 @@
|
||||
"botTokenLabel": "Discord Bot Token",
|
||||
"botTokenPlaceholder": "MTAxNjQ0OTk2NjAz...",
|
||||
"instructions": "1. Gehen Sie zu discord.com/developers/applications\n2. Erstellen Sie eine neue Anwendung und fügen Sie einen Bot hinzu\n3. Kopieren Sie den Bot-Token",
|
||||
"disclaimer": "Ich bestätige, dass ich diesen Discord-Bot besitze und PieCed IT autorisiere, ihn mit meinem KI-Assistenten zu verbinden."
|
||||
},
|
||||
"email": {
|
||||
"description": "Ermöglichen Sie Ihrem KI-Assistenten, E-Mails zu senden und zu empfangen.",
|
||||
"smtpHostLabel": "SMTP Host",
|
||||
"smtpHostPlaceholder": "smtp.example.com",
|
||||
"smtpUserLabel": "SMTP Benutzername",
|
||||
"smtpUserPlaceholder": "user@example.com",
|
||||
"smtpPasswordLabel": "SMTP Passwort",
|
||||
"smtpPasswordPlaceholder": "••••••••",
|
||||
"imapHostLabel": "IMAP Host",
|
||||
"imapHostPlaceholder": "imap.example.com",
|
||||
"instructions": "Geben Sie SMTP- und IMAP-Zugangsdaten an. Der Assistent nutzt diese zum Senden und Empfangen von Nachrichten.",
|
||||
"disclaimer": "Ich bestätige, dass ich berechtigt bin, diese E-Mail-Zugangsdaten zu verwenden und dass PieCed IT auf dieses Postfach zugreifen darf."
|
||||
},
|
||||
"webSearch": {
|
||||
"description": "Geben Sie Ihrem KI-Assistenten die Möglichkeit, im Web zu suchen."
|
||||
},
|
||||
"documentProcessing": {
|
||||
"description": "Aktivieren Sie Dokumentenverarbeitung, Zusammenfassung und Extraktion."
|
||||
"disclaimer": "Ich bestätige, dass ich diesen Discord-Bot besitze und PieCed IT autorisiere, ihn mit meinem KI-Assistenten zu verbinden.",
|
||||
"appIdLabel": "Discord-Anwendungs-ID",
|
||||
"appIdPlaceholder": "18–19-stellige numerische ID aus dem Developer Portal"
|
||||
},
|
||||
"statusEnabled": "aktiviert",
|
||||
"statusDisabled": "deaktiviert"
|
||||
"statusDisabled": "deaktiviert",
|
||||
"coreHeartbeat": {
|
||||
"description": "Periodischer Agentenlauf alle 30 Minuten, der es dem Assistenten erlaubt, Posteingang, Kalender und andere konfigurierte Quellen zu prüfen und proaktiv Bescheid zu geben, wenn etwas Aufmerksamkeit braucht. Ohne diese Option reagiert der Assistent nur, wenn Sie ihn ansprechen."
|
||||
},
|
||||
"coreCron": {
|
||||
"description": "Erlaubt dem Assistenten, geplante Aufgaben auszuführen (tägliche Briefings, wiederkehrende Erinnerungen, periodische Berichte). Standardmässig deaktiviert. Bei Deaktivierung bleibt das Cron-Werkzeug verfügbar, aber keine geplante Aufgabe wird ausgeführt."
|
||||
},
|
||||
"coreActiveMemory": {
|
||||
"description": "Erlaubt dem Assistenten, stabile Präferenzen, wiederkehrende Gewohnheiten und langfristigen Kontext aus früheren Gesprächen abzurufen. Nutzt einen zusätzlichen Sub-Agent-Lauf pro eingehender Nachricht, um den Memory-Store abzufragen. Nur Direktnachrichten. Kleiner Mehraufwand an Tokens im Tausch gegen Kontinuität und Personalisierung."
|
||||
},
|
||||
"coreVoice": {
|
||||
"description": "Spracherkennung für eingehende Sprachnachrichten und Sprachsynthese für Antworten, über das PieCed-LiteLLM-Gateway, damit Audiokosten pro Mandant erfasst werden. Die Laufzeit-Integration kommt im nächsten Plattform-Release; das Umschalten speichert die Auswahl für diese Auslieferung."
|
||||
},
|
||||
"gitCli": {
|
||||
"description": "Eigenständige Git-Kommandozeilenoperationen (clone, commit, branch, diff, log, status). Für private Repositories konfigurieren Sie die Zugangsdaten in Ihrem Workspace."
|
||||
},
|
||||
"github": {
|
||||
"description": "Interaktion mit GitHub-Repositories über die gh-CLI — Issues, Pull Requests, CI-Läufe, Releases, Gists. Erfordert ein persönliches Zugriffstoken.",
|
||||
"tokenLabel": "GitHub Persönliches Zugriffstoken",
|
||||
"tokenPlaceholder": "ghp_… oder github_pat_…",
|
||||
"instructions": "1. Öffnen Sie https://github.com/settings/tokens\n2. Erstellen Sie ein fein abgestimmtes persönliches Zugriffstoken mit den gewünschten Repo-Berechtigungen\n3. Kopieren Sie das Token (es wird nur einmal angezeigt)"
|
||||
},
|
||||
"gitea": {
|
||||
"description": "Interaktion mit einer Gitea-Instanz — Repositories, Issues, Pull Requests, Releases. Standardmässig die PieCed-Plattform-Gitea unter git.c5ai.ch.",
|
||||
"tokenLabel": "Gitea-Zugriffstoken",
|
||||
"tokenPlaceholder": "Erstellt unter Einstellungen → Anwendungen",
|
||||
"instructions": "1. Melden Sie sich bei Ihrer Gitea-Instanz an (Standard https://git.c5ai.ch)\n2. Gehen Sie zu Einstellungen → Anwendungen → Neues Token erstellen\n3. Vergeben Sie die gewünschten Berechtigungen (repo, issue, user)\n4. Kopieren Sie das Token"
|
||||
},
|
||||
"whisperSelfHosted": {
|
||||
"description": "Transkribieren Sie Audiodateien über die plattformeigene Whisper-Instanz. Nützlich für Ad-hoc-Transkriptionsaufgaben aus dem Chat heraus."
|
||||
},
|
||||
"searxngLocalSearch": {
|
||||
"description": "Datenschutzfreundliche Web-Suche über die interne SearXNG-Instanz der Plattform. Durchsuchen Sie Web, Bilder und News ohne externe API-Aufrufe oder Tracker."
|
||||
},
|
||||
"gog": {
|
||||
"description": "Gebündelter Zugriff auf Gmail, Kalender, Drive, Docs, Sheets und Kontakte via Google OAuth. Setup erfordert ein Google-Cloud-Projekt — wenden Sie sich an den PieCed-Support für die Einrichtung.",
|
||||
"clientIdLabel": "Google OAuth Client-ID",
|
||||
"clientIdPlaceholder": "xxxxxxxxxxx.apps.googleusercontent.com",
|
||||
"clientSecretLabel": "Google OAuth Client-Secret",
|
||||
"clientSecretPlaceholder": "GOCSPX-…",
|
||||
"refreshTokenLabel": "Google OAuth Refresh-Token",
|
||||
"refreshTokenPlaceholder": "1//0g…",
|
||||
"instructions": "Die Google-Workspace-Integration verwendet OAuth und erfordert derzeit manuelles Onboarding. Bitte eröffnen Sie ein Support-Ticket, um den Setup-Prozess zu starten — wir tauschen die Client-Zugangsdaten und ein Refresh-Token offline aus und aktivieren dann dieses Paket für Ihren Mandanten.",
|
||||
"disclaimer": "Mit der Aktivierung der Google-Workspace-Integration autorisieren Sie PieCed, in Ihrem Namen auf Gmail, Kalender, Drive, Docs, Sheets und Kontakte zuzugreifen. Daten fliessen über die Google-APIs, vorbehaltlich der Google-Bedingungen."
|
||||
},
|
||||
"mail": {
|
||||
"description": "E-Mails über IMAP lesen, suchen und verwalten; senden über SMTP. Funktioniert mit Gmail (mit App-Passwort), Outlook, Fastmail und jedem standardkonformen IMAP/SMTP-Host.",
|
||||
"imapHostLabel": "IMAP-Host",
|
||||
"imapHostPlaceholder": "imap.example.com",
|
||||
"imapUserLabel": "IMAP-Benutzername",
|
||||
"imapUserPlaceholder": "benutzer@example.com",
|
||||
"imapPassLabel": "IMAP-Passwort",
|
||||
"imapPassPlaceholder": "••••••••",
|
||||
"smtpHostLabel": "SMTP-Host",
|
||||
"smtpHostPlaceholder": "smtp.example.com",
|
||||
"smtpUserLabel": "SMTP-Benutzername",
|
||||
"smtpUserPlaceholder": "benutzer@example.com",
|
||||
"smtpPassLabel": "SMTP-Passwort",
|
||||
"smtpPassPlaceholder": "••••••••",
|
||||
"instructions": "1. Für Gmail: Aktivieren Sie die 2-Faktor-Authentifizierung, erstellen Sie dann unter https://myaccount.google.com/apppasswords ein App-Passwort und verwenden Sie es als IMAP- und SMTP-Passwort.\n2. Für Outlook / Microsoft 365 mit MFA: Generieren Sie ein App-Passwort in den Sicherheitseinstellungen Ihres Kontos.\n3. Für andere Anbieter: Konsultieren Sie deren IMAP/SMTP-Dokumentation für Hostnamen und Ports.\n4. Typische IMAP-Hosts: imap.gmail.com, outlook.office365.com.\n5. Typische SMTP-Hosts: smtp.gmail.com, smtp.office365.com.",
|
||||
"disclaimer": "Der Assistent erhält Lese- und Schreibzugriff auf das von Ihnen konfigurierte Postfach. Verwenden Sie eine dedizierte Adresse anstelle eines persönlichen Postfachs, wenn Sie den Umfang einschränken möchten."
|
||||
},
|
||||
"threema": {
|
||||
"description": "Senden und empfangen Sie Nachrichten über Threema. Jede eingehende und ausgehende Nachricht läuft über den gemeinsamen PieCed-Messaging-Dienst und verursacht eine Gebühr pro Nachricht bei Threema — eine Drittanbieter-Kostenposition, unabhängig von Ihrem PieCed-Abonnement.",
|
||||
"instructions": "1. Aktivieren Sie dieses Paket.\n2. Öffnen Sie Threema auf Ihrem Telefon, scannen Sie den QR-Code unter Autorisierte Benutzer → threema und akzeptieren Sie den Kontakt.\n3. Tragen Sie Ihre eigene Threema-ID unter Autorisierte Benutzer → threema ein, damit der Assistent Ihre Nachrichten erkennt.\n4. Schreiben Sie eine Nachricht aus Threema, um das Gespräch zu beginnen.",
|
||||
"disclaimer": "Nachrichten zwischen Threema und PieCed werden Ende-zu-Ende verschlüsselt bis zum PieCed-Messaging-Dienst, wo sie entschlüsselt und an Ihren Assistenten weitergeleitet werden. Jede gesendete oder empfangene Nachricht wird gemäss Threema-Tarif pro Nachricht abgerechnet — die aktuellen Preise finden Sie in Ihrem Plan."
|
||||
}
|
||||
},
|
||||
"admin": {
|
||||
"title": "Plattform-Admin",
|
||||
@@ -334,7 +383,8 @@
|
||||
"statusDown": "Ausgefallen",
|
||||
"spendChf": "Kosten (CHF)",
|
||||
"resumeRequestBadge": "Wieder",
|
||||
"resumeRequestTooltip": "Reaktivierungsanfrage für einen bestehenden Tenant. Bei Genehmigung wird der Tenant wieder aktiviert; keine Provisionierung läuft."
|
||||
"resumeRequestTooltip": "Reaktivierungsanfrage für einen bestehenden Tenant. Bei Genehmigung wird der Tenant wieder aktiviert; keine Provisionierung läuft.",
|
||||
"openclawTool": "OpenClaw-Versionen"
|
||||
},
|
||||
"channelUsers": {
|
||||
"title": "Autorisierte Benutzer",
|
||||
@@ -346,7 +396,15 @@
|
||||
"alreadyAdded": "Diese Benutzer-ID ist bereits autorisiert.",
|
||||
"telegramIdHelp": "So finden Sie Ihre Telegram-Benutzer-ID:\n1. Öffnen Sie Telegram und schreiben Sie @userinfobot\n2. Der Bot antwortet sofort mit Ihrer numerischen ID\n3. Geben Sie diese Nummer hier ein",
|
||||
"discordIdHelp": "So finden Sie Ihre Discord-Benutzer-ID:\n1. Aktivieren Sie den Entwicklermodus in den Discord-Einstellungen (Erweitert)\n2. Rechtsklick auf Ihren Namen → Benutzer-ID kopieren\n3. Geben Sie diese Nummer hier ein",
|
||||
"emailIdHelp": "Geben Sie die E-Mail-Adresse ein, die zur Interaktion mit dem Assistenten autorisiert werden soll."
|
||||
"threemaIdHelp": "Geben Sie Ihre eigene Threema-ID ein — die 8 Zeichen, die in Ihrer Threema-App unter Einstellungen → Meine Threema-ID angezeigt werden. Anschliessend können Sie direkt aus Threema mit dem Assistenten chatten.",
|
||||
"threemaSetup": {
|
||||
"title": "Assistenten zu Threema hinzufügen",
|
||||
"step1": "Öffnen Sie Threema auf Ihrem Telefon.",
|
||||
"step2": "Tippen Sie auf das Scan-Symbol und scannen Sie diesen QR-Code, um den Assistenten als Kontakt hinzuzufügen.",
|
||||
"step3": "Fügen Sie anschliessend unten Ihre eigene Threema-ID hinzu.",
|
||||
"qrAlt": "QR-Code, um {gateway} als Threema-Kontakt hinzuzufügen",
|
||||
"showQr": "QR anzeigen"
|
||||
}
|
||||
},
|
||||
"team": {
|
||||
"title": "Team",
|
||||
@@ -474,5 +532,24 @@
|
||||
"resolvedBanner": "Dieses Ticket ist erledigt. Antworten Sie unten, falls Sie nachfragen möchten — das öffnet es erneut.",
|
||||
"adminControlsTitle": "Admin-Steuerung",
|
||||
"updateFailed": "Änderungen konnten nicht gespeichert werden. Bitte erneut versuchen."
|
||||
},
|
||||
"openclawAdmin": {
|
||||
"title": "OpenClaw-Versionen",
|
||||
"subtitle": "Plattform-Standard-Tag und Tenant-spezifische Overrides für das Testen neuer Releases konfigurieren.",
|
||||
"defaultSection": "Plattform-Standard",
|
||||
"defaultDescription": "Wird von jedem Tenant ohne eigenen Override verwendet.",
|
||||
"fieldTag": "Tag",
|
||||
"emptyHint": "Leer lassen, um den eingebauten Operator-Standard zu verwenden.",
|
||||
"saveDefault": "Standard speichern",
|
||||
"defaultSaved": "Standard gespeichert. Tenants ohne Override übernehmen den Wert beim nächsten Reconcile.",
|
||||
"saveFailed": "Speichern fehlgeschlagen. Bitte erneut versuchen.",
|
||||
"overridesSection": "Tenant-Overrides",
|
||||
"noTenants": "Keine Tenants im Cluster.",
|
||||
"statusOverridden": "Override",
|
||||
"statusFollowsDefault": "Folgt Standard",
|
||||
"builtinFallback": "(eingebauter Fallback)",
|
||||
"defaultPrefix": "Standard:",
|
||||
"saveOverride": "Override speichern",
|
||||
"clearOverride": "Override entfernen"
|
||||
}
|
||||
}
|
||||
|
||||
@@ -192,8 +192,7 @@
|
||||
"requests": "requests",
|
||||
"budgetEdit": "Edit",
|
||||
"budgetEditTitle": "Set spending budget",
|
||||
"budgetEditDescription": "Cap how much your assistants can spend before requests start being declined.",
|
||||
"budgetOrgScopeWarning": "This budget applies to all tenants in your organization, not just this one. If you have multiple tenants, they share the same cap.",
|
||||
"budgetEditDescription": "Cap how much this tenant's assistants can spend before requests start being declined.",
|
||||
"budgetModeUnlimited": "No limit",
|
||||
"budgetModeUnlimitedDescription": "Spend as much as needed; no cap.",
|
||||
"budgetModeCapped": "Set a cap",
|
||||
@@ -215,7 +214,8 @@
|
||||
"packages": {
|
||||
"categories": {
|
||||
"channels": "Channels",
|
||||
"skills": "Skills"
|
||||
"skills": "Skills",
|
||||
"core": "Core"
|
||||
},
|
||||
"enable": "Enable",
|
||||
"disable": "Disable",
|
||||
@@ -240,29 +240,78 @@
|
||||
"botTokenLabel": "Discord Bot Token",
|
||||
"botTokenPlaceholder": "MTAxNjQ0OTk2NjAz...",
|
||||
"instructions": "1. Go to discord.com/developers/applications\n2. Create a new application and add a bot\n3. Copy the bot token",
|
||||
"disclaimer": "I confirm I own this Discord bot and authorize PieCed IT to connect it to my AI assistant."
|
||||
"disclaimer": "I confirm I own this Discord bot and authorize PieCed IT to connect it to my AI assistant.",
|
||||
"appIdLabel": "Discord Application ID",
|
||||
"appIdPlaceholder": "18-19 digit numeric ID from Developer Portal"
|
||||
},
|
||||
"email": {
|
||||
"description": "Enable your AI assistant to send and receive email.",
|
||||
"statusEnabled": "enabled",
|
||||
"statusDisabled": "disabled",
|
||||
"coreHeartbeat": {
|
||||
"description": "Periodic agent run every 30 minutes that lets your assistant check inbox, calendar, and other configured sources and message you proactively when something needs attention. Without this, the assistant only responds when you message it first."
|
||||
},
|
||||
"coreCron": {
|
||||
"description": "Allow the assistant to run scheduled tasks (daily briefings, recurring reminders, periodic reports). Off by default. When off, the agent's cron tool stays available but no scheduled job ever fires."
|
||||
},
|
||||
"coreActiveMemory": {
|
||||
"description": "Lets the assistant recall stable preferences, recurring habits, and long-term context from past conversations during a chat. Uses an extra sub-agent turn per inbound message to query the memory store. Direct-message sessions only. Adds a small token cost in exchange for continuity and personalisation."
|
||||
},
|
||||
"coreVoice": {
|
||||
"description": "Speech-to-text on incoming voice notes and text-to-speech on replies, routed through the PieCed LiteLLM gateway so audio cost is tracked per tenant alongside chat. Runtime wiring lands in the next platform release; toggling now stores the preference for that rollout."
|
||||
},
|
||||
"gitCli": {
|
||||
"description": "Standalone git command-line operations (clone, commit, branch, diff, log, status). For private repositories, configure credentials in your workspace."
|
||||
},
|
||||
"github": {
|
||||
"description": "Interact with GitHub repositories via the gh CLI — issues, pull requests, CI runs, releases, gists. Requires a personal access token.",
|
||||
"tokenLabel": "GitHub Personal Access Token",
|
||||
"tokenPlaceholder": "ghp_… or github_pat_…",
|
||||
"instructions": "1. Open https://github.com/settings/tokens\n2. Generate a fine-grained personal access token with the repo scopes you want the assistant to use\n3. Copy the token (it's shown only once)"
|
||||
},
|
||||
"gitea": {
|
||||
"description": "Interact with a Gitea instance — repositories, issues, pull requests, releases. Defaults to the PieCed-platform Gitea at git.c5ai.ch.",
|
||||
"tokenLabel": "Gitea Access Token",
|
||||
"tokenPlaceholder": "Generated under Settings → Applications",
|
||||
"instructions": "1. Log in to your Gitea instance (default https://git.c5ai.ch)\n2. Go to Settings → Applications → Generate New Token\n3. Grant the scopes you want the assistant to use (repo, issue, user)\n4. Copy the token"
|
||||
},
|
||||
"whisperSelfHosted": {
|
||||
"description": "Transcribe audio files via the platform's self-hosted Whisper instance. Useful for ad-hoc transcription tasks initiated from chat."
|
||||
},
|
||||
"searxngLocalSearch": {
|
||||
"description": "Privacy-respecting web search via the platform's internal SearXNG instance. Search the web, images, and news without external API calls or trackers."
|
||||
},
|
||||
"gog": {
|
||||
"description": "Bundled access to Gmail, Calendar, Drive, Docs, Sheets, and Contacts via Google OAuth. Setup requires a Google Cloud project — contact PieCed support to onboard.",
|
||||
"clientIdLabel": "Google OAuth Client ID",
|
||||
"clientIdPlaceholder": "xxxxxxxxxxx.apps.googleusercontent.com",
|
||||
"clientSecretLabel": "Google OAuth Client Secret",
|
||||
"clientSecretPlaceholder": "GOCSPX-…",
|
||||
"refreshTokenLabel": "Google OAuth Refresh Token",
|
||||
"refreshTokenPlaceholder": "1//0g…",
|
||||
"instructions": "Google Workspace integration uses OAuth and requires manual onboarding for now. Please open a support ticket to start the setup — we'll exchange the client credentials and a refresh token offline, then enable this package on your tenant.",
|
||||
"disclaimer": "By enabling Google Workspace integration you authorize PieCed to access Gmail, Calendar, Drive, Docs, Sheets, and Contacts on your behalf. Data flows through Google's APIs subject to Google's terms."
|
||||
},
|
||||
"mail": {
|
||||
"description": "Read, search, and manage email via IMAP; send via SMTP. Works with Gmail (with an app password), Outlook, Fastmail, and any standard IMAP/SMTP host.",
|
||||
"imapHostLabel": "IMAP Host",
|
||||
"imapHostPlaceholder": "imap.example.com",
|
||||
"imapUserLabel": "IMAP Username",
|
||||
"imapUserPlaceholder": "user@example.com",
|
||||
"imapPassLabel": "IMAP Password",
|
||||
"imapPassPlaceholder": "••••••••",
|
||||
"smtpHostLabel": "SMTP Host",
|
||||
"smtpHostPlaceholder": "smtp.example.com",
|
||||
"smtpUserLabel": "SMTP Username",
|
||||
"smtpUserPlaceholder": "user@example.com",
|
||||
"smtpPasswordLabel": "SMTP Password",
|
||||
"smtpPasswordPlaceholder": "••••••••",
|
||||
"imapHostLabel": "IMAP Host",
|
||||
"imapHostPlaceholder": "imap.example.com",
|
||||
"instructions": "Provide SMTP and IMAP credentials. The assistant uses these to send and monitor messages.",
|
||||
"disclaimer": "I confirm I am authorized to use these email credentials and that PieCed IT may access this mailbox."
|
||||
"smtpPassLabel": "SMTP Password",
|
||||
"smtpPassPlaceholder": "••••••••",
|
||||
"instructions": "1. For Gmail: enable 2-Step Verification, then create an App Password at https://myaccount.google.com/apppasswords and use it as both IMAP and SMTP password.\n2. For Outlook / Microsoft 365 with MFA: generate an app password in your account's security settings.\n3. For other providers: refer to their IMAP/SMTP documentation for host names and ports.\n4. Typical IMAP hosts: imap.gmail.com, outlook.office365.com.\n5. Typical SMTP hosts: smtp.gmail.com, smtp.office365.com.",
|
||||
"disclaimer": "The assistant gains read/write access to the mailbox you configure. Consider using a dedicated address rather than a personal inbox if you want to limit scope."
|
||||
},
|
||||
"webSearch": {
|
||||
"description": "Give your AI assistant the ability to search the web."
|
||||
},
|
||||
"documentProcessing": {
|
||||
"description": "Enable document parsing, summarization, and extraction."
|
||||
},
|
||||
"statusEnabled": "enabled",
|
||||
"statusDisabled": "disabled"
|
||||
"threema": {
|
||||
"description": "Send and receive messages through Threema. Each inbound and outbound message uses the shared PieCed messaging service and incurs a per-message charge from Threema — a third-party cost, separate from your PieCed subscription.",
|
||||
"instructions": "1. Enable this package.\n2. Open Threema on your phone, scan the QR code shown under Authorized Users → threema, and accept the contact.\n3. Add your own Threema ID under Authorized Users → threema so the assistant recognises your messages.\n4. Send a message from Threema to start chatting with the assistant.",
|
||||
"disclaimer": "Messages between Threema and PieCed are end-to-end encrypted up to PieCed's messaging service, where they are decrypted to be routed to your assistant. Each message sent or received is counted toward Threema's per-message billing — see your plan for current rates."
|
||||
}
|
||||
},
|
||||
"admin": {
|
||||
"title": "Platform Admin",
|
||||
@@ -334,7 +383,8 @@
|
||||
"statusDown": "Down",
|
||||
"spendChf": "Spend (CHF)",
|
||||
"resumeRequestBadge": "Resume",
|
||||
"resumeRequestTooltip": "Reactivation request for an existing tenant. Approving will un-suspend the tenant; no provisioning runs."
|
||||
"resumeRequestTooltip": "Reactivation request for an existing tenant. Approving will un-suspend the tenant; no provisioning runs.",
|
||||
"openclawTool": "OpenClaw versions"
|
||||
},
|
||||
"channelUsers": {
|
||||
"title": "Authorized Users",
|
||||
@@ -346,7 +396,15 @@
|
||||
"alreadyAdded": "This user ID is already authorized.",
|
||||
"telegramIdHelp": "To find your Telegram user ID:\n1. Open Telegram and message @userinfobot\n2. It instantly replies with your numeric ID\n3. Enter that number here",
|
||||
"discordIdHelp": "To find your Discord user ID:\n1. Enable Developer Mode in Discord settings (Advanced)\n2. Right-click your name → Copy User ID\n3. Enter that number here",
|
||||
"emailIdHelp": "Enter the email address that should be authorized to interact with the assistant."
|
||||
"threemaIdHelp": "Enter your own Threema ID — the 8 characters shown in your Threema app under Settings → My Threema ID. Once added, you'll be able to chat with the assistant directly from Threema.",
|
||||
"threemaSetup": {
|
||||
"title": "Add the assistant to your Threema",
|
||||
"step1": "Open Threema on your phone.",
|
||||
"step2": "Tap the scan icon and scan this QR code to add the assistant as a contact.",
|
||||
"step3": "Then add your own Threema ID below.",
|
||||
"qrAlt": "QR code to add {gateway} as a Threema contact",
|
||||
"showQr": "Show QR"
|
||||
}
|
||||
},
|
||||
"team": {
|
||||
"title": "Team",
|
||||
@@ -474,5 +532,24 @@
|
||||
"resolvedBanner": "This ticket is resolved. Reply below if you need to follow up — that will reopen it.",
|
||||
"adminControlsTitle": "Admin controls",
|
||||
"updateFailed": "Could not save changes. Please try again."
|
||||
},
|
||||
"openclawAdmin": {
|
||||
"title": "OpenClaw versions",
|
||||
"subtitle": "Configure the platform-default OpenClaw image tag and per-tenant overrides for testing new releases.",
|
||||
"defaultSection": "Platform default",
|
||||
"defaultDescription": "Used by every tenant that doesn't have its own override.",
|
||||
"fieldTag": "Tag",
|
||||
"emptyHint": "Leave empty to fall back to the operator's built-in default.",
|
||||
"saveDefault": "Save default",
|
||||
"defaultSaved": "Default saved. Tenants without overrides will pick this up on the next reconcile.",
|
||||
"saveFailed": "Could not save. Please try again.",
|
||||
"overridesSection": "Tenant overrides",
|
||||
"noTenants": "No tenants in the cluster.",
|
||||
"statusOverridden": "Override",
|
||||
"statusFollowsDefault": "Follows default",
|
||||
"builtinFallback": "(operator built-in fallback)",
|
||||
"defaultPrefix": "Default:",
|
||||
"saveOverride": "Save override",
|
||||
"clearOverride": "Clear override"
|
||||
}
|
||||
}
|
||||
|
||||
@@ -192,8 +192,7 @@
|
||||
"requests": "requêtes",
|
||||
"budgetEdit": "Modifier",
|
||||
"budgetEditTitle": "Définir un budget",
|
||||
"budgetEditDescription": "Limitez la dépense de vos assistants avant que les requêtes ne soient refusées.",
|
||||
"budgetOrgScopeWarning": "Ce budget s'applique à tous les locataires de votre organisation, pas seulement à celui-ci. Si vous avez plusieurs locataires, ils partagent le même plafond.",
|
||||
"budgetEditDescription": "Limitez la dépense des assistants de ce locataire avant que les requêtes ne soient refusées.",
|
||||
"budgetModeUnlimited": "Aucune limite",
|
||||
"budgetModeUnlimitedDescription": "Dépense libre, sans plafond.",
|
||||
"budgetModeCapped": "Définir un plafond",
|
||||
@@ -215,7 +214,8 @@
|
||||
"packages": {
|
||||
"categories": {
|
||||
"channels": "Canaux",
|
||||
"skills": "Compétences"
|
||||
"skills": "Compétences",
|
||||
"core": "Cœur"
|
||||
},
|
||||
"enable": "Activer",
|
||||
"disable": "Désactiver",
|
||||
@@ -240,29 +240,78 @@
|
||||
"botTokenLabel": "Token du bot Discord",
|
||||
"botTokenPlaceholder": "MTAxNjQ0OTk2NjAz...",
|
||||
"instructions": "1. Allez sur discord.com/developers/applications\n2. Créez une nouvelle application et ajoutez un bot\n3. Copiez le token du bot",
|
||||
"disclaimer": "Je confirme que je possède ce bot Discord et autorise PieCed IT à le connecter à mon assistant IA."
|
||||
"disclaimer": "Je confirme que je possède ce bot Discord et autorise PieCed IT à le connecter à mon assistant IA.",
|
||||
"appIdLabel": "ID d'application Discord",
|
||||
"appIdPlaceholder": "ID numérique de 18–19 chiffres depuis le Developer Portal"
|
||||
},
|
||||
"email": {
|
||||
"description": "Permettez à votre assistant IA d'envoyer et de recevoir des e-mails.",
|
||||
"statusEnabled": "activé",
|
||||
"statusDisabled": "désactivé",
|
||||
"coreHeartbeat": {
|
||||
"description": "Exécution périodique de l'agent toutes les 30 minutes pour vérifier votre boîte mail, votre agenda et d'autres sources configurées, et vous notifier de manière proactive lorsqu'une attention est requise. Sans cette option, l'assistant ne répond que lorsque vous lui écrivez."
|
||||
},
|
||||
"coreCron": {
|
||||
"description": "Permet à l'assistant d'exécuter des tâches programmées (briefings quotidiens, rappels récurrents, rapports périodiques). Désactivé par défaut. Lorsqu'il est désactivé, l'outil cron reste disponible mais aucune tâche planifiée ne s'exécute."
|
||||
},
|
||||
"coreActiveMemory": {
|
||||
"description": "Permet à l'assistant de se rappeler des préférences stables, des habitudes récurrentes et du contexte à long terme issu de conversations passées. Utilise un tour de sous-agent supplémentaire par message entrant pour interroger la mémoire. Uniquement en messages directs. Légère consommation de tokens supplémentaire en échange de continuité et de personnalisation."
|
||||
},
|
||||
"coreVoice": {
|
||||
"description": "Reconnaissance vocale sur les notes vocales entrantes et synthèse vocale sur les réponses, via la passerelle PieCed LiteLLM pour un suivi du coût audio par tenant. L'intégration runtime arrive dans la prochaine version de la plateforme ; basculer le commutateur enregistre dès maintenant la préférence."
|
||||
},
|
||||
"gitCli": {
|
||||
"description": "Opérations git en ligne de commande autonomes (clone, commit, branch, diff, log, status). Pour les dépôts privés, configurez les identifiants dans votre espace de travail."
|
||||
},
|
||||
"github": {
|
||||
"description": "Interagissez avec les dépôts GitHub via la CLI gh — issues, pull requests, exécutions CI, releases, gists. Nécessite un jeton d'accès personnel.",
|
||||
"tokenLabel": "Jeton d'accès personnel GitHub",
|
||||
"tokenPlaceholder": "ghp_… ou github_pat_…",
|
||||
"instructions": "1. Ouvrez https://github.com/settings/tokens\n2. Générez un jeton d'accès personnel fin avec les portées de dépôt souhaitées\n3. Copiez le jeton (il n'est affiché qu'une fois)"
|
||||
},
|
||||
"gitea": {
|
||||
"description": "Interagissez avec une instance Gitea — dépôts, issues, pull requests, releases. Par défaut, l'instance Gitea PieCed à git.c5ai.ch.",
|
||||
"tokenLabel": "Jeton d'accès Gitea",
|
||||
"tokenPlaceholder": "Généré sous Paramètres → Applications",
|
||||
"instructions": "1. Connectez-vous à votre instance Gitea (par défaut https://git.c5ai.ch)\n2. Allez dans Paramètres → Applications → Générer un nouveau jeton\n3. Accordez les portées souhaitées (repo, issue, user)\n4. Copiez le jeton"
|
||||
},
|
||||
"whisperSelfHosted": {
|
||||
"description": "Transcrivez des fichiers audio via l'instance Whisper auto-hébergée de la plateforme. Utile pour les transcriptions ad hoc initiées depuis le chat."
|
||||
},
|
||||
"searxngLocalSearch": {
|
||||
"description": "Recherche web respectueuse de la vie privée via l'instance SearXNG interne de la plateforme. Recherchez le web, les images et les actualités sans appels d'API externes ni traqueurs."
|
||||
},
|
||||
"gog": {
|
||||
"description": "Accès groupé à Gmail, Agenda, Drive, Docs, Sheets et Contacts via Google OAuth. La configuration nécessite un projet Google Cloud — contactez le support PieCed pour l'intégration.",
|
||||
"clientIdLabel": "ID client Google OAuth",
|
||||
"clientIdPlaceholder": "xxxxxxxxxxx.apps.googleusercontent.com",
|
||||
"clientSecretLabel": "Secret client Google OAuth",
|
||||
"clientSecretPlaceholder": "GOCSPX-…",
|
||||
"refreshTokenLabel": "Jeton de rafraîchissement Google OAuth",
|
||||
"refreshTokenPlaceholder": "1//0g…",
|
||||
"instructions": "L'intégration de Google Workspace utilise OAuth et nécessite actuellement une intégration manuelle. Veuillez ouvrir un ticket de support pour démarrer la configuration — nous échangerons hors ligne les identifiants client et un jeton de rafraîchissement, puis activerons ce package sur votre tenant.",
|
||||
"disclaimer": "En activant l'intégration de Google Workspace, vous autorisez PieCed à accéder à Gmail, Agenda, Drive, Docs, Sheets et Contacts en votre nom. Les données transitent par les API de Google, soumises aux conditions de Google."
|
||||
},
|
||||
"mail": {
|
||||
"description": "Lisez, recherchez et gérez vos e-mails via IMAP ; envoyez via SMTP. Compatible avec Gmail (avec un mot de passe d'application), Outlook, Fastmail et tout hôte IMAP/SMTP standard.",
|
||||
"imapHostLabel": "Hôte IMAP",
|
||||
"imapHostPlaceholder": "imap.example.com",
|
||||
"imapUserLabel": "Nom d'utilisateur IMAP",
|
||||
"imapUserPlaceholder": "utilisateur@example.com",
|
||||
"imapPassLabel": "Mot de passe IMAP",
|
||||
"imapPassPlaceholder": "••••••••",
|
||||
"smtpHostLabel": "Hôte SMTP",
|
||||
"smtpHostPlaceholder": "smtp.example.com",
|
||||
"smtpUserLabel": "Nom d'utilisateur SMTP",
|
||||
"smtpUserPlaceholder": "user@example.com",
|
||||
"smtpPasswordLabel": "Mot de passe SMTP",
|
||||
"smtpPasswordPlaceholder": "••••••••",
|
||||
"imapHostLabel": "Hôte IMAP",
|
||||
"imapHostPlaceholder": "imap.example.com",
|
||||
"instructions": "Fournissez les identifiants SMTP et IMAP. L'assistant les utilise pour envoyer et surveiller les messages.",
|
||||
"disclaimer": "Je confirme que je suis autorisé à utiliser ces identifiants e-mail et que PieCed IT peut accéder à cette boîte mail."
|
||||
"smtpUserPlaceholder": "utilisateur@example.com",
|
||||
"smtpPassLabel": "Mot de passe SMTP",
|
||||
"smtpPassPlaceholder": "••••••••",
|
||||
"instructions": "1. Pour Gmail : activez la validation en deux étapes, puis créez un mot de passe d'application sur https://myaccount.google.com/apppasswords et utilisez-le comme mot de passe IMAP et SMTP.\n2. Pour Outlook / Microsoft 365 avec MFA : générez un mot de passe d'application dans les paramètres de sécurité de votre compte.\n3. Pour les autres fournisseurs : consultez leur documentation IMAP/SMTP pour les noms d'hôte et les ports.\n4. Hôtes IMAP typiques : imap.gmail.com, outlook.office365.com.\n5. Hôtes SMTP typiques : smtp.gmail.com, smtp.office365.com.",
|
||||
"disclaimer": "L'assistant obtient un accès en lecture/écriture à la boîte aux lettres que vous configurez. Envisagez d'utiliser une adresse dédiée plutôt qu'une boîte personnelle si vous souhaitez limiter la portée."
|
||||
},
|
||||
"webSearch": {
|
||||
"description": "Donnez à votre assistant IA la capacité de rechercher sur le web."
|
||||
},
|
||||
"documentProcessing": {
|
||||
"description": "Activez l'analyse, le résumé et l'extraction de documents."
|
||||
},
|
||||
"statusEnabled": "activé",
|
||||
"statusDisabled": "désactivé"
|
||||
"threema": {
|
||||
"description": "Envoyez et recevez des messages via Threema. Chaque message entrant ou sortant transite par le service de messagerie PieCed partagé et entraîne des frais par message facturés par Threema — un coût tiers, distinct de votre abonnement PieCed.",
|
||||
"instructions": "1. Activez ce package.\n2. Ouvrez Threema sur votre téléphone, scannez le QR code affiché dans Utilisateurs autorisés → threema, puis acceptez le contact.\n3. Ajoutez votre propre identifiant Threema sous Utilisateurs autorisés → threema afin que l'assistant reconnaisse vos messages.\n4. Envoyez un message depuis Threema pour commencer la conversation.",
|
||||
"disclaimer": "Les messages entre Threema et PieCed sont chiffrés de bout en bout jusqu'au service de messagerie PieCed, où ils sont déchiffrés pour être acheminés vers votre assistant. Chaque message envoyé ou reçu est facturé par Threema selon son tarif par message — consultez votre plan pour les tarifs en vigueur."
|
||||
}
|
||||
},
|
||||
"admin": {
|
||||
"title": "Admin plateforme",
|
||||
@@ -334,7 +383,8 @@
|
||||
"statusDown": "Hors service",
|
||||
"spendChf": "Coûts (CHF)",
|
||||
"resumeRequestBadge": "Reprise",
|
||||
"resumeRequestTooltip": "Demande de réactivation d'un locataire existant. L'approbation le réactivera ; aucun provisionnement ne s'exécute."
|
||||
"resumeRequestTooltip": "Demande de réactivation d'un locataire existant. L'approbation le réactivera ; aucun provisionnement ne s'exécute.",
|
||||
"openclawTool": "Versions OpenClaw"
|
||||
},
|
||||
"channelUsers": {
|
||||
"title": "Utilisateurs autorisés",
|
||||
@@ -346,7 +396,15 @@
|
||||
"alreadyAdded": "Cet identifiant est déjà autorisé.",
|
||||
"telegramIdHelp": "Pour trouver votre identifiant Telegram :\n1. Ouvrez Telegram et envoyez un message à @userinfobot\n2. Il répond instantanément avec votre identifiant numérique\n3. Entrez ce numéro ici",
|
||||
"discordIdHelp": "Pour trouver votre identifiant Discord :\n1. Activez le mode développeur dans les paramètres Discord (Avancé)\n2. Clic droit sur votre nom → Copier l'identifiant\n3. Entrez ce numéro ici",
|
||||
"emailIdHelp": "Entrez l'adresse e-mail qui doit être autorisée à interagir avec l'assistant."
|
||||
"threemaIdHelp": "Saisissez votre propre identifiant Threema — les 8 caractères affichés dans votre application Threema sous Réglages → Mon identifiant Threema. Une fois ajouté, vous pourrez discuter directement avec l'assistant depuis Threema.",
|
||||
"threemaSetup": {
|
||||
"title": "Ajouter l'assistant à Threema",
|
||||
"step1": "Ouvrez Threema sur votre téléphone.",
|
||||
"step2": "Appuyez sur l'icône de scan et scannez ce QR code pour ajouter l'assistant comme contact.",
|
||||
"step3": "Puis ajoutez votre propre identifiant Threema ci-dessous.",
|
||||
"qrAlt": "QR code pour ajouter {gateway} comme contact Threema",
|
||||
"showQr": "Afficher le QR"
|
||||
}
|
||||
},
|
||||
"team": {
|
||||
"title": "Équipe",
|
||||
@@ -474,5 +532,24 @@
|
||||
"resolvedBanner": "Ce ticket est résolu. Répondez ci-dessous si vous avez besoin d'un suivi — cela le rouvrira.",
|
||||
"adminControlsTitle": "Contrôles admin",
|
||||
"updateFailed": "Impossible d'enregistrer les modifications. Veuillez réessayer."
|
||||
},
|
||||
"openclawAdmin": {
|
||||
"title": "Versions OpenClaw",
|
||||
"subtitle": "Configurer le tag par défaut de la plateforme et les surcharges par locataire pour tester les nouvelles versions.",
|
||||
"defaultSection": "Défaut de la plateforme",
|
||||
"defaultDescription": "Utilisé par tous les locataires sans surcharge propre.",
|
||||
"fieldTag": "Tag",
|
||||
"emptyHint": "Laisser vide pour utiliser le défaut intégré de l'opérateur.",
|
||||
"saveDefault": "Enregistrer le défaut",
|
||||
"defaultSaved": "Défaut enregistré. Les locataires sans surcharge l'appliqueront au prochain réconcile.",
|
||||
"saveFailed": "Échec de l'enregistrement. Veuillez réessayer.",
|
||||
"overridesSection": "Surcharges par locataire",
|
||||
"noTenants": "Aucun locataire dans le cluster.",
|
||||
"statusOverridden": "Surcharge",
|
||||
"statusFollowsDefault": "Suit le défaut",
|
||||
"builtinFallback": "(repli intégré)",
|
||||
"defaultPrefix": "Défaut :",
|
||||
"saveOverride": "Enregistrer la surcharge",
|
||||
"clearOverride": "Supprimer la surcharge"
|
||||
}
|
||||
}
|
||||
|
||||
@@ -192,8 +192,7 @@
|
||||
"requests": "richieste",
|
||||
"budgetEdit": "Modifica",
|
||||
"budgetEditTitle": "Imposta budget",
|
||||
"budgetEditDescription": "Limita quanto i tuoi assistenti possono spendere prima che le richieste vengano rifiutate.",
|
||||
"budgetOrgScopeWarning": "Questo budget si applica a tutti i tenant della tua organizzazione, non solo a questo. Se hai più tenant, condividono lo stesso limite.",
|
||||
"budgetEditDescription": "Limita quanto gli assistenti di questo tenant possono spendere prima che le richieste vengano rifiutate.",
|
||||
"budgetModeUnlimited": "Nessun limite",
|
||||
"budgetModeUnlimitedDescription": "Spesa libera, nessun tetto.",
|
||||
"budgetModeCapped": "Imposta un tetto",
|
||||
@@ -215,7 +214,8 @@
|
||||
"packages": {
|
||||
"categories": {
|
||||
"channels": "Canali",
|
||||
"skills": "Capacità"
|
||||
"skills": "Capacità",
|
||||
"core": "Core"
|
||||
},
|
||||
"enable": "Attiva",
|
||||
"disable": "Disattiva",
|
||||
@@ -240,29 +240,78 @@
|
||||
"botTokenLabel": "Token bot Discord",
|
||||
"botTokenPlaceholder": "MTAxNjQ0OTk2NjAz...",
|
||||
"instructions": "1. Vai su discord.com/developers/applications\n2. Crea una nuova applicazione e aggiungi un bot\n3. Copia il token del bot",
|
||||
"disclaimer": "Confermo di possedere questo bot Discord e autorizzo PieCed IT a collegarlo al mio assistente IA."
|
||||
},
|
||||
"email": {
|
||||
"description": "Permetti al tuo assistente IA di inviare e ricevere e-mail.",
|
||||
"smtpHostLabel": "Host SMTP",
|
||||
"smtpHostPlaceholder": "smtp.example.com",
|
||||
"smtpUserLabel": "Nome utente SMTP",
|
||||
"smtpUserPlaceholder": "user@example.com",
|
||||
"smtpPasswordLabel": "Password SMTP",
|
||||
"smtpPasswordPlaceholder": "••••••••",
|
||||
"imapHostLabel": "Host IMAP",
|
||||
"imapHostPlaceholder": "imap.example.com",
|
||||
"instructions": "Fornisci le credenziali SMTP e IMAP. L'assistente le usa per inviare e monitorare i messaggi.",
|
||||
"disclaimer": "Confermo di essere autorizzato a utilizzare queste credenziali e-mail e che PieCed IT può accedere a questa casella di posta."
|
||||
},
|
||||
"webSearch": {
|
||||
"description": "Dai al tuo assistente IA la capacità di cercare nel web."
|
||||
},
|
||||
"documentProcessing": {
|
||||
"description": "Attiva l'analisi, il riassunto e l'estrazione di documenti."
|
||||
"disclaimer": "Confermo di possedere questo bot Discord e autorizzo PieCed IT a collegarlo al mio assistente IA.",
|
||||
"appIdLabel": "ID applicazione Discord",
|
||||
"appIdPlaceholder": "ID numerico di 18–19 cifre dal Developer Portal"
|
||||
},
|
||||
"statusEnabled": "abilitato",
|
||||
"statusDisabled": "disabilitato"
|
||||
"statusDisabled": "disabilitato",
|
||||
"coreHeartbeat": {
|
||||
"description": "Esecuzione periodica dell'agente ogni 30 minuti che consente all'assistente di controllare posta, calendario e altre fonti configurate e di avvisarti proattivamente quando serve attenzione. Senza questa opzione, l'assistente risponde solo quando lo contatti."
|
||||
},
|
||||
"coreCron": {
|
||||
"description": "Consente all'assistente di eseguire attività pianificate (briefing giornalieri, promemoria ricorrenti, report periodici). Disattivato per impostazione predefinita. Quando è disattivato, lo strumento cron resta disponibile ma nessuna attività pianificata viene eseguita."
|
||||
},
|
||||
"coreActiveMemory": {
|
||||
"description": "Consente all'assistente di richiamare preferenze stabili, abitudini ricorrenti e contesto a lungo termine dalle conversazioni precedenti. Utilizza un turno extra di sub-agente per ogni messaggio in entrata per interrogare lo store di memoria. Solo messaggi diretti. Aggiunge un piccolo costo in token in cambio di continuità e personalizzazione."
|
||||
},
|
||||
"coreVoice": {
|
||||
"description": "Riconoscimento vocale sui messaggi audio in entrata e sintesi vocale sulle risposte, instradati attraverso il gateway PieCed LiteLLM per tracciare il costo audio per tenant. L'integrazione runtime arriverà nel prossimo rilascio della piattaforma; attivare ora salva la preferenza per quel rilascio."
|
||||
},
|
||||
"gitCli": {
|
||||
"description": "Operazioni git da riga di comando autonome (clone, commit, branch, diff, log, status). Per i repository privati, configura le credenziali nel tuo workspace."
|
||||
},
|
||||
"github": {
|
||||
"description": "Interagisci con repository GitHub tramite la CLI gh — issue, pull request, esecuzioni CI, release, gist. Richiede un token di accesso personale.",
|
||||
"tokenLabel": "Token di accesso personale GitHub",
|
||||
"tokenPlaceholder": "ghp_… o github_pat_…",
|
||||
"instructions": "1. Apri https://github.com/settings/tokens\n2. Genera un token di accesso personale fine con gli ambiti repo desiderati\n3. Copia il token (viene mostrato una sola volta)"
|
||||
},
|
||||
"gitea": {
|
||||
"description": "Interagisci con un'istanza Gitea — repository, issue, pull request, release. Per impostazione predefinita, l'istanza Gitea PieCed su git.c5ai.ch.",
|
||||
"tokenLabel": "Token di accesso Gitea",
|
||||
"tokenPlaceholder": "Generato in Impostazioni → Applicazioni",
|
||||
"instructions": "1. Accedi alla tua istanza Gitea (predefinito https://git.c5ai.ch)\n2. Vai a Impostazioni → Applicazioni → Genera nuovo token\n3. Concedi gli ambiti desiderati (repo, issue, user)\n4. Copia il token"
|
||||
},
|
||||
"whisperSelfHosted": {
|
||||
"description": "Trascrivi file audio tramite l'istanza Whisper auto-ospitata della piattaforma. Utile per attività di trascrizione ad hoc avviate dalla chat."
|
||||
},
|
||||
"searxngLocalSearch": {
|
||||
"description": "Ricerca web rispettosa della privacy tramite l'istanza SearXNG interna della piattaforma. Cerca sul web, nelle immagini e nelle notizie senza chiamate ad API esterne né tracker."
|
||||
},
|
||||
"gog": {
|
||||
"description": "Accesso integrato a Gmail, Calendar, Drive, Docs, Sheets e Contatti tramite Google OAuth. La configurazione richiede un progetto Google Cloud — contatta il supporto PieCed per l'onboarding.",
|
||||
"clientIdLabel": "ID client Google OAuth",
|
||||
"clientIdPlaceholder": "xxxxxxxxxxx.apps.googleusercontent.com",
|
||||
"clientSecretLabel": "Client secret Google OAuth",
|
||||
"clientSecretPlaceholder": "GOCSPX-…",
|
||||
"refreshTokenLabel": "Token di refresh Google OAuth",
|
||||
"refreshTokenPlaceholder": "1//0g…",
|
||||
"instructions": "L'integrazione con Google Workspace utilizza OAuth e richiede attualmente un onboarding manuale. Apri un ticket di supporto per avviare la configurazione — scambieremo le credenziali del client e un token di refresh offline, quindi abiliteremo questo pacchetto sul tuo tenant.",
|
||||
"disclaimer": "Abilitando l'integrazione con Google Workspace autorizzi PieCed ad accedere per tuo conto a Gmail, Calendar, Drive, Docs, Sheets e Contatti. I dati transitano attraverso le API di Google, soggetti ai termini di Google."
|
||||
},
|
||||
"mail": {
|
||||
"description": "Leggi, cerca e gestisci le e-mail via IMAP; invia tramite SMTP. Funziona con Gmail (con una password per app), Outlook, Fastmail e qualsiasi host IMAP/SMTP standard.",
|
||||
"imapHostLabel": "Host IMAP",
|
||||
"imapHostPlaceholder": "imap.example.com",
|
||||
"imapUserLabel": "Username IMAP",
|
||||
"imapUserPlaceholder": "utente@example.com",
|
||||
"imapPassLabel": "Password IMAP",
|
||||
"imapPassPlaceholder": "••••••••",
|
||||
"smtpHostLabel": "Host SMTP",
|
||||
"smtpHostPlaceholder": "smtp.example.com",
|
||||
"smtpUserLabel": "Username SMTP",
|
||||
"smtpUserPlaceholder": "utente@example.com",
|
||||
"smtpPassLabel": "Password SMTP",
|
||||
"smtpPassPlaceholder": "••••••••",
|
||||
"instructions": "1. Per Gmail: abilita la verifica in due passaggi, quindi crea una password per app su https://myaccount.google.com/apppasswords e usala come password IMAP e SMTP.\n2. Per Outlook / Microsoft 365 con MFA: genera una password per app nelle impostazioni di sicurezza del tuo account.\n3. Per altri provider: consulta la loro documentazione IMAP/SMTP per nomi host e porte.\n4. Host IMAP tipici: imap.gmail.com, outlook.office365.com.\n5. Host SMTP tipici: smtp.gmail.com, smtp.office365.com.",
|
||||
"disclaimer": "L'assistente ottiene accesso in lettura/scrittura alla casella di posta che configuri. Valuta l'uso di un indirizzo dedicato anziché di una casella personale se vuoi limitare la portata."
|
||||
},
|
||||
"threema": {
|
||||
"description": "Invia e ricevi messaggi tramite Threema. Ogni messaggio in entrata e in uscita passa attraverso il servizio di messaggistica condiviso di PieCed e comporta un addebito per messaggio da parte di Threema — un costo di terzi, separato dall'abbonamento PieCed.",
|
||||
"instructions": "1. Attiva questo pacchetto.\n2. Apri Threema sul tuo telefono, scansiona il QR code mostrato in Utenti autorizzati → threema e accetta il contatto.\n3. Aggiungi il tuo ID Threema sotto Utenti autorizzati → threema affinché l'assistente riconosca i tuoi messaggi.\n4. Invia un messaggio da Threema per iniziare la conversazione.",
|
||||
"disclaimer": "I messaggi tra Threema e PieCed sono cifrati end-to-end fino al servizio di messaggistica PieCed, dove vengono decifrati per essere inoltrati al tuo assistente. Ogni messaggio inviato o ricevuto viene addebitato da Threema secondo la sua tariffa per messaggio — consulta il tuo piano per i prezzi attuali."
|
||||
}
|
||||
},
|
||||
"admin": {
|
||||
"title": "Admin piattaforma",
|
||||
@@ -334,7 +383,8 @@
|
||||
"statusDown": "Non disponibile",
|
||||
"spendChf": "Costi (CHF)",
|
||||
"resumeRequestBadge": "Ripresa",
|
||||
"resumeRequestTooltip": "Richiesta di riattivazione di un tenant esistente. L'approvazione lo riattiverà; non viene eseguito alcun provisioning."
|
||||
"resumeRequestTooltip": "Richiesta di riattivazione di un tenant esistente. L'approvazione lo riattiverà; non viene eseguito alcun provisioning.",
|
||||
"openclawTool": "Versioni OpenClaw"
|
||||
},
|
||||
"channelUsers": {
|
||||
"title": "Utenti autorizzati",
|
||||
@@ -346,7 +396,15 @@
|
||||
"alreadyAdded": "Questo ID utente è già autorizzato.",
|
||||
"telegramIdHelp": "Per trovare il tuo ID Telegram:\n1. Apri Telegram e invia un messaggio a @userinfobot\n2. Risponde istantaneamente con il tuo ID numerico\n3. Inserisci quel numero qui",
|
||||
"discordIdHelp": "Per trovare il tuo ID Discord:\n1. Attiva la Modalità sviluppatore nelle impostazioni Discord (Avanzate)\n2. Clic destro sul tuo nome → Copia ID utente\n3. Inserisci quel numero qui",
|
||||
"emailIdHelp": "Inserisci l'indirizzo e-mail che deve essere autorizzato a interagire con l'assistente."
|
||||
"threemaIdHelp": "Inserisci il tuo ID Threema — gli 8 caratteri mostrati nella tua app Threema sotto Impostazioni → Il mio ID Threema. Una volta aggiunto, potrai conversare con l'assistente direttamente da Threema.",
|
||||
"threemaSetup": {
|
||||
"title": "Aggiungi l'assistente a Threema",
|
||||
"step1": "Apri Threema sul tuo telefono.",
|
||||
"step2": "Tocca l'icona di scansione e scansiona questo QR code per aggiungere l'assistente ai contatti.",
|
||||
"step3": "Quindi aggiungi il tuo ID Threema qui sotto.",
|
||||
"qrAlt": "QR code per aggiungere {gateway} come contatto Threema",
|
||||
"showQr": "Mostra QR"
|
||||
}
|
||||
},
|
||||
"team": {
|
||||
"title": "Team",
|
||||
@@ -474,5 +532,24 @@
|
||||
"resolvedBanner": "Questo ticket è risolto. Rispondi qui sotto se hai bisogno di un seguito — questo lo riaprirà.",
|
||||
"adminControlsTitle": "Controlli admin",
|
||||
"updateFailed": "Impossibile salvare le modifiche. Riprova."
|
||||
},
|
||||
"openclawAdmin": {
|
||||
"title": "Versioni OpenClaw",
|
||||
"subtitle": "Configura il tag predefinito della piattaforma e gli override per tenant per testare nuove release.",
|
||||
"defaultSection": "Predefinito piattaforma",
|
||||
"defaultDescription": "Usato da ogni tenant senza override proprio.",
|
||||
"fieldTag": "Tag",
|
||||
"emptyHint": "Lascia vuoto per usare il predefinito integrato dell'operatore.",
|
||||
"saveDefault": "Salva predefinito",
|
||||
"defaultSaved": "Predefinito salvato. I tenant senza override lo applicheranno al prossimo reconcile.",
|
||||
"saveFailed": "Salvataggio fallito. Riprova.",
|
||||
"overridesSection": "Override per tenant",
|
||||
"noTenants": "Nessun tenant nel cluster.",
|
||||
"statusOverridden": "Override",
|
||||
"statusFollowsDefault": "Segue predefinito",
|
||||
"builtinFallback": "(fallback integrato)",
|
||||
"defaultPrefix": "Predefinito:",
|
||||
"saveOverride": "Salva override",
|
||||
"clearOverride": "Rimuovi override"
|
||||
}
|
||||
}
|
||||
|
||||
@@ -75,6 +75,18 @@ export interface PiecedTenantSpec {
|
||||
workspaceFiles?: Record<string, string>;
|
||||
channelUsers?: Record<string, string[]>;
|
||||
suspend?: boolean;
|
||||
/**
|
||||
* Per-tenant OpenClaw image override (tag). Set only by platform
|
||||
* admins via the portal admin UI. Customers never see this field.
|
||||
* When unset or with empty Tag, the operator uses the platform
|
||||
* default from the pieced-openclaw-config ConfigMap.
|
||||
*
|
||||
* Tag-only by design — see operator notes for rationale (single
|
||||
* image-selector field avoids SSA field-ownership ambiguity).
|
||||
*/
|
||||
openClawImage?: {
|
||||
tag?: string;
|
||||
};
|
||||
}
|
||||
|
||||
export interface PiecedTenantStatus {
|
||||
|
||||
Reference in New Issue
Block a user