Full set of admin surfaces on top of /platform/* and /admin/* endpoints,
plus a migration of /assistant onto @crema/llm-providers-ui.
Buckets (/buckets):
S3-level CRUD over /platform/buckets — list, create, delete (with the
6-digit confirmation flow the backend enforces), per-bucket configure
for versioning / CORS rules / policy JSON, plus an object browser
with FileGrid/FileList from @crema/file-ui and presigned-URL reveal.
Storage-config picker scopes the view to one credential at a time.
Monitoring (/monitoring):
Live dashboard. Service health board derived from indirect signals
(status-ui OverallStatus + ComponentRow). KPI tiles for sessions,
jobs, audit. Tabs: background jobs (Donut + BarChart + retry recent),
sessions (Sparkline of last 24h sign-ins), audit activity (BarChart
of severity / top resource types), infrastructure (DO summary +
WorldMapSvg coloured by droplet region + droplet list + Spaces),
rate limits. 30s auto-refresh.
Memberships (/memberships):
M:N glue between users and tenants over /admin/memberships. Add /
edit / suspend / activate / remove with role multi-select.
Networking (/networking):
Tabs over /platform/{firewalls,vpcs,domains,floating_ips}.
Read/delete on firewalls, read on VPCs, full DNS-record CRUD, and
inline assign/unassign for floating IPs.
SSO (/sso):
/sso/identity-providers CRUD with PEM cert as write-only field, plus
/sso/sessions list with destroy.
Announcements (/announcements):
/admin/announcements CRUD. Platform-wide vs per-tenant audience,
schedule windows, dismissible + active toggles.
Status page (/status-page):
/admin/status-page/{components,incidents,subscribers}. Components
CRUD, incidents with timeline + post-update + resolve flow,
subscriber list. Public preview at the top using StatusBoard +
IncidentTimeline from @crema/status-ui.
Assistant migration:
/assistant now uses @crema/llm-providers-ui (provider catalog +
vault key resolution) instead of ~/lib/llm-settings. Same async
buildAdapter() flow used by /ai. The legacy lib file is now
unreferenced and can be removed when ready.
New sibling libs wired (cloned from CremaUIStudio):
lib-file-ui, lib-card-ui, lib-dashboard-ui, lib-chart-ui,
lib-map-ui, lib-status-ui.
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
80 lines
2.0 KiB
TypeScript
80 lines
2.0 KiB
TypeScript
// Platform announcements helpers.
|
|
// Backend: /api/v1/admin/announcements (admin CRUD).
|
|
|
|
import type { ArcadiaClient } from "@crema/arcadia-client"
|
|
|
|
export type AnnouncementType =
|
|
| "info"
|
|
| "warning"
|
|
| "maintenance"
|
|
| "incident"
|
|
| "feature"
|
|
| string
|
|
|
|
export type AnnouncementAudience = "all" | "tenant" | "platform" | string
|
|
|
|
export interface Announcement {
|
|
id: string
|
|
tenant_id: string | null
|
|
announcement_type: AnnouncementType
|
|
title: string
|
|
body: string | null
|
|
action_label: string | null
|
|
action_url: string | null
|
|
starts_at: string | null
|
|
ends_at: string | null
|
|
audience: AnnouncementAudience
|
|
dismissible: boolean
|
|
active: boolean
|
|
created_by_id: string | null
|
|
inserted_at: string
|
|
updated_at: string
|
|
}
|
|
|
|
export interface AnnouncementInput {
|
|
title: string
|
|
body?: string
|
|
announcement_type?: AnnouncementType
|
|
audience?: AnnouncementAudience
|
|
action_label?: string | null
|
|
action_url?: string | null
|
|
starts_at?: string | null
|
|
ends_at?: string | null
|
|
dismissible?: boolean
|
|
active?: boolean
|
|
/** Platform-wide if null, otherwise scoped. */
|
|
tenant_id?: string | null
|
|
}
|
|
|
|
const BASE = "/api/v1/admin/announcements"
|
|
|
|
export async function listAnnouncements(arcadia: ArcadiaClient): Promise<Announcement[]> {
|
|
const res = await arcadia.GET<{ data: Announcement[] }>(BASE)
|
|
return res.data
|
|
}
|
|
|
|
export async function createAnnouncement(
|
|
arcadia: ArcadiaClient,
|
|
input: AnnouncementInput,
|
|
): Promise<Announcement> {
|
|
const res = await arcadia.POST<{ data: Announcement }>(BASE, {
|
|
body: { announcement: input },
|
|
})
|
|
return res.data
|
|
}
|
|
|
|
export async function updateAnnouncement(
|
|
arcadia: ArcadiaClient,
|
|
id: string,
|
|
input: Partial<AnnouncementInput>,
|
|
): Promise<Announcement> {
|
|
const res = await arcadia.PUT<{ data: Announcement }>(`${BASE}/${id}`, {
|
|
body: { announcement: input },
|
|
})
|
|
return res.data
|
|
}
|
|
|
|
export async function deleteAnnouncement(arcadia: ArcadiaClient, id: string): Promise<void> {
|
|
await arcadia.DELETE(`${BASE}/${id}`)
|
|
}
|