Compare commits
3 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 395d2f43cc | |||
| 6f42b56ad5 | |||
| 85c4302f7a |
@@ -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__
|
||||
|
||||
176
README.md
176
README.md
@@ -1,100 +1,116 @@
|
||||
# PieCed Portal
|
||||
# Threema UX rework + QR code
|
||||
|
||||
Customer self-service portal for the PieCed IT multi-tenant OpenClaw platform.
|
||||
## Files
|
||||
|
||||
## Stack
|
||||
```
|
||||
src/lib/threema-gateway-config.ts # NEW — single source of truth for gateway ID + QR path
|
||||
src/components/channel-users/threema-setup.tsx # NEW — QR + 3-step instructions component
|
||||
src/components/channel-users/channel-users.tsx # MODIFIED — renders <ThreemaSetup /> for the threema channel
|
||||
deploy/patch-i18n-threema.mjs # REPLACES earlier version — customer-friendly texts in 4 langs
|
||||
public/threema/qr_code_AIAGENT.png # NEW — the QR you uploaded
|
||||
```
|
||||
|
||||
| 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` |
|
||||
## What changed (UX, customer-facing)
|
||||
|
||||
## Setup
|
||||
1. **Package description / instructions / disclaimer.** No mention of
|
||||
"Gateway account", "Gateway credentials", or anything about asterisks.
|
||||
The disclaimer now explicitly says messages are end-to-end encrypted
|
||||
only up to PieCed's messaging service (where they get decrypted to
|
||||
route to the assistant) — accurate, not overclaiming. Crucially it
|
||||
also says **Threema charges per message** so customers know that
|
||||
sending/receiving on this channel has a cost separate from their
|
||||
PieCed subscription.
|
||||
|
||||
### 1. ZITADEL Application
|
||||
2. **Threema ID help text.** Now tells the customer to add **their
|
||||
own** ID, with explicit instructions for finding it in the Threema
|
||||
app (Settings → My Threema ID). Drops the asterisk / Gateway-prefix
|
||||
explanation entirely.
|
||||
|
||||
In ZITADEL console (`auth.pieced.ch`), project "OpenClaw Platform":
|
||||
3. **QR code component (NEW).** Shown above the help text in the
|
||||
channel-users panel when the threema channel is enabled. Three-step
|
||||
flow: open Threema, scan, add your own ID below.
|
||||
|
||||
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
|
||||
4. **All four languages** (en/de/fr/it) updated consistently.
|
||||
|
||||
### 2. OpenBao Secrets
|
||||
## What changed (technical)
|
||||
|
||||
- Gateway constants centralised in `src/lib/threema-gateway-config.ts`.
|
||||
Today hardcoded to `*AIAGENT` + `/threema/qr_code_AIAGENT.png`. When
|
||||
you need multiple gateway accounts, edit that one file (and the
|
||||
inline TODO block tells the next person how).
|
||||
- The component uses Next.js `<Image>` — automatic optimisation,
|
||||
lazy-loaded by default (no `priority`).
|
||||
- The PNG goes in `public/threema/`, served as a static asset under
|
||||
`/threema/qr_code_AIAGENT.png` — no API route, no auth wrapper, the
|
||||
QR encodes a Threema invitation anyone can scan anyway.
|
||||
|
||||
## 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
|
||||
|
||||
# Drop in new files (overwrites prior channel-users.tsx and i18n script)
|
||||
cp -r <unzipped>/* .
|
||||
|
||||
# Quick TS check (the new component uses next/image — should be fine)
|
||||
npx tsc --noEmit
|
||||
|
||||
# Patch the message files
|
||||
node deploy/patch-i18n-threema.mjs
|
||||
# Should print 4 lines, one per language
|
||||
|
||||
# Commit + push
|
||||
git add -A
|
||||
git status # eyeball the changes
|
||||
git commit -m "Threema: customer-friendly texts + QR setup component"
|
||||
git push
|
||||
```
|
||||
|
||||
### 3. Build & Push
|
||||
## Verify after redeploy
|
||||
|
||||
1. Open the portal as a customer admin, pick the test tenant.
|
||||
2. Disable + re-enable Threema in the package list (or just look at
|
||||
the channel-users panel — the QR shows there regardless).
|
||||
3. Authorized Users → threema section should show:
|
||||
- QR code on the left
|
||||
- "AIAGENT" label under the QR (no asterisk)
|
||||
- 3-step instruction list
|
||||
- Help text below: "Enter your own Threema ID..."
|
||||
- Existing user pills + Add input
|
||||
|
||||
Scan the QR with your phone — it should prompt to add `*AIAGENT` as a
|
||||
contact in Threema with trust level "verified by the operator" (the
|
||||
QR encodes the contact's public key).
|
||||
|
||||
## Billing log query (re-run after sending a few messages)
|
||||
|
||||
```bash
|
||||
docker build -t registry.c5ai.ch/pieced/pieced-portal:0.1.0 .
|
||||
docker push registry.c5ai.ch/pieced/pieced-portal:0.1.0
|
||||
POD=$(kubectl -n threema-gateway get pods -l 'cnpg.io/cluster=pieced-threema-gateway-db,role=primary' -o jsonpath='{.items[0].metadata.name}')
|
||||
DBPASS=$(kubectl -n threema-gateway get secret pieced-threema-gateway-db-app -o jsonpath='{.data.password}' | base64 -d)
|
||||
|
||||
# Per-tenant in/out counts
|
||||
kubectl -n threema-gateway exec $POD -- env PGPASSWORD="$DBPASS" \
|
||||
psql -U relay -d relay -c \
|
||||
"SELECT tenant_name, direction, count(*), max(created_at) FROM messages GROUP BY tenant_name, direction ORDER BY tenant_name, direction;"
|
||||
```
|
||||
|
||||
Update image tag in `pieced-gitops/apps/portal/deployment.yaml`, push, ArgoCD syncs.
|
||||
That's your billing source — count(*) × Threema's per-message rate ×
|
||||
direction-specific multiplier = customer charge.
|
||||
|
||||
### 4. DNS
|
||||
## Future: dynamic gateway accounts
|
||||
|
||||
Ensure `app.pieced.ch` A record → MetalLB ingress IP (or ExternalDNS handles it).
|
||||
Today's hardcoded `*AIAGENT` works for one shared gateway. When you
|
||||
move to per-tenant gateway accounts (Threema's bigger plans, or to
|
||||
isolate billing per tenant):
|
||||
|
||||
## Local Development
|
||||
1. Update `src/lib/threema-gateway-config.ts`:
|
||||
- Make the constants a lookup keyed by tenant
|
||||
- Or fetch from `/api/tenants/<name>/threema` (new admin route)
|
||||
2. Update `threema-setup.tsx` to accept gateway info as props
|
||||
3. Update the parent `channel-users.tsx` to pass tenant-specific values
|
||||
4. Replace the static PNG with a server route that generates per-tenant
|
||||
QR codes from each gateway's `id` + `publicKey`
|
||||
|
||||
```bash
|
||||
cp .env.example .env.local
|
||||
# Fill in values — K8s client uses ~/.kube/config locally
|
||||
npm install
|
||||
npm run dev
|
||||
```
|
||||
|
||||
## Project Structure
|
||||
|
||||
```
|
||||
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
|
||||
```
|
||||
|
||||
## 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)
|
||||
The single config file means this refactor is contained — every
|
||||
consumer reads from one place, so once that one place returns the
|
||||
right values, everything else falls in line.
|
||||
|
||||
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.
|
||||
118
deploy/patch-i18n-threema.mjs
Normal file
118
deploy/patch-i18n-threema.mjs
Normal file
@@ -0,0 +1,118 @@
|
||||
#!/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",
|
||||
},
|
||||
},
|
||||
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",
|
||||
},
|
||||
},
|
||||
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",
|
||||
},
|
||||
},
|
||||
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",
|
||||
},
|
||||
},
|
||||
};
|
||||
|
||||
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 |
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 });
|
||||
}
|
||||
@@ -8,12 +8,26 @@ import { useRouter } from "next/navigation";
|
||||
const CHANNEL_ID_HELP: Record<string, string> = {
|
||||
telegram: "telegramIdHelp",
|
||||
discord: "discordIdHelp",
|
||||
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"]) */
|
||||
@@ -63,6 +77,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();
|
||||
@@ -74,18 +152,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,
|
||||
@@ -93,7 +180,7 @@ export function ChannelUsers({
|
||||
};
|
||||
updateChannelUsers(updated);
|
||||
},
|
||||
[channelUsers, updateChannelUsers]
|
||||
[channelUsers, updateChannelUsers, removeFromRelayChannel]
|
||||
);
|
||||
|
||||
if (enabledChannels.length === 0) return null;
|
||||
|
||||
@@ -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
|
||||
|
||||
@@ -19,6 +19,17 @@
|
||||
* 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 {
|
||||
@@ -38,6 +49,14 @@ export interface PackageDef {
|
||||
instructionsKey?: string;
|
||||
disclaimerKey?: string;
|
||||
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[] = [
|
||||
@@ -121,6 +140,21 @@ export const PACKAGE_CATALOG: PackageDef[] = [
|
||||
disclaimerKey: "packages.discord.disclaimer",
|
||||
category: "channel",
|
||||
},
|
||||
{
|
||||
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
|
||||
|
||||
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;
|
||||
}
|
||||
@@ -306,6 +306,11 @@
|
||||
"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": {
|
||||
@@ -390,7 +395,15 @@
|
||||
"remove": "Entfernen",
|
||||
"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"
|
||||
"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",
|
||||
"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"
|
||||
}
|
||||
},
|
||||
"team": {
|
||||
"title": "Team",
|
||||
|
||||
@@ -306,6 +306,11 @@
|
||||
"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."
|
||||
},
|
||||
"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": {
|
||||
@@ -390,7 +395,15 @@
|
||||
"remove": "Remove",
|
||||
"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"
|
||||
"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",
|
||||
"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"
|
||||
}
|
||||
},
|
||||
"team": {
|
||||
"title": "Team",
|
||||
|
||||
@@ -306,6 +306,11 @@
|
||||
"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."
|
||||
},
|
||||
"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": {
|
||||
@@ -390,7 +395,15 @@
|
||||
"remove": "Supprimer",
|
||||
"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"
|
||||
"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",
|
||||
"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"
|
||||
}
|
||||
},
|
||||
"team": {
|
||||
"title": "Équipe",
|
||||
|
||||
@@ -306,6 +306,11 @@
|
||||
"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": {
|
||||
@@ -390,7 +395,15 @@
|
||||
"remove": "Rimuovi",
|
||||
"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"
|
||||
"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",
|
||||
"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"
|
||||
}
|
||||
},
|
||||
"team": {
|
||||
"title": "Team",
|
||||
|
||||
Reference in New Issue
Block a user