mirror of
https://github.com/kennethnym/aris.git
synced 2026-03-25 03:11:17 +00:00
* feat: add admin dashboard app - React + Vite + TanStack Router + TanStack Query - Auth with better-auth (login, session, admin guard) - Source config management (WeatherKit credentials, user config) - Feed query panel - Location push card - General settings with health check - CORS middleware for cross-origin auth - Disable CSRF check in dev mode - Sonner toasts for mutation feedback Co-authored-by: Ona <no-reply@ona.com> * fix: use useQuery instead of getQueryData Co-authored-by: Ona <no-reply@ona.com> * refactor: remove backend changes from dashboard PR Backend CORS/CSRF changes moved to #92. Source registry removed (sources hardcoded in frontend). Co-authored-by: Ona <no-reply@ona.com> --------- Co-authored-by: Ona <no-reply@ona.com>
165 lines
4.9 KiB
TypeScript
165 lines
4.9 KiB
TypeScript
import { getServerUrl } from "./server-url"
|
|
|
|
function apiBase() {
|
|
return `${getServerUrl()}/api/admin`
|
|
}
|
|
|
|
function serverBase() {
|
|
return `${getServerUrl()}/api`
|
|
}
|
|
|
|
export interface ConfigFieldDef {
|
|
type: "string" | "number" | "select"
|
|
label: string
|
|
required?: boolean
|
|
description?: string
|
|
secret?: boolean
|
|
defaultValue?: string | number
|
|
options?: { label: string; value: string }[]
|
|
}
|
|
|
|
export interface SourceDefinition {
|
|
id: string
|
|
name: string
|
|
description: string
|
|
alwaysEnabled?: boolean
|
|
fields: Record<string, ConfigFieldDef>
|
|
}
|
|
|
|
export interface SourceConfig {
|
|
sourceId: string
|
|
enabled: boolean
|
|
config: Record<string, unknown>
|
|
}
|
|
|
|
const sourceDefinitions: SourceDefinition[] = [
|
|
{
|
|
id: "aelis.location",
|
|
name: "Location",
|
|
description: "Device location provider. Always enabled as a dependency for other sources.",
|
|
alwaysEnabled: true,
|
|
fields: {},
|
|
},
|
|
{
|
|
id: "aelis.weather",
|
|
name: "WeatherKit",
|
|
description: "Apple WeatherKit weather data. Requires Apple Developer credentials.",
|
|
fields: {
|
|
privateKey: { type: "string", label: "Private Key", required: true, secret: true, description: "Apple WeatherKit private key (PEM format)" },
|
|
keyId: { type: "string", label: "Key ID", required: true, secret: true },
|
|
teamId: { type: "string", label: "Team ID", required: true, secret: true },
|
|
serviceId: { type: "string", label: "Service ID", required: true, secret: true },
|
|
units: { type: "select", label: "Units", options: [{ label: "Metric", value: "metric" }, { label: "Imperial", value: "imperial" }], defaultValue: "metric" },
|
|
hourlyLimit: { type: "number", label: "Hourly Forecast Limit", defaultValue: 12, description: "Number of hourly forecasts to include" },
|
|
dailyLimit: { type: "number", label: "Daily Forecast Limit", defaultValue: 7, description: "Number of daily forecasts to include" },
|
|
},
|
|
},
|
|
]
|
|
|
|
export function fetchSources(): Promise<SourceDefinition[]> {
|
|
return Promise.resolve(sourceDefinitions)
|
|
}
|
|
|
|
export async function fetchSourceConfig(
|
|
sourceId: string,
|
|
): Promise<SourceConfig | null> {
|
|
const res = await fetch(`${serverBase()}/sources/${sourceId}`, {
|
|
credentials: "include",
|
|
})
|
|
if (res.status === 404) return null
|
|
if (!res.ok) throw new Error(`Failed to fetch source config: ${res.status}`)
|
|
const data = (await res.json()) as { enabled: boolean; config: Record<string, unknown> }
|
|
return { sourceId, enabled: data.enabled, config: data.config }
|
|
}
|
|
|
|
export async function fetchConfigs(): Promise<SourceConfig[]> {
|
|
const results = await Promise.all(
|
|
sourceDefinitions.map((s) => fetchSourceConfig(s.id)),
|
|
)
|
|
return results.filter((c): c is SourceConfig => c !== null)
|
|
}
|
|
|
|
export async function replaceSource(
|
|
sourceId: string,
|
|
body: { enabled: boolean; config: unknown },
|
|
): Promise<void> {
|
|
const res = await fetch(`${serverBase()}/sources/${sourceId}`, {
|
|
method: "PUT",
|
|
headers: { "Content-Type": "application/json" },
|
|
credentials: "include",
|
|
body: JSON.stringify(body),
|
|
})
|
|
if (!res.ok) {
|
|
const data = (await res.json()) as { error?: string }
|
|
throw new Error(data.error ?? `Failed to replace source config: ${res.status}`)
|
|
}
|
|
}
|
|
|
|
export async function updateProviderConfig(
|
|
sourceId: string,
|
|
body: Record<string, unknown>,
|
|
): Promise<void> {
|
|
const res = await fetch(`${apiBase()}/${sourceId}/config`, {
|
|
method: "PUT",
|
|
headers: { "Content-Type": "application/json" },
|
|
credentials: "include",
|
|
body: JSON.stringify(body),
|
|
})
|
|
if (!res.ok) {
|
|
const data = (await res.json()) as { error?: string }
|
|
throw new Error(data.error ?? `Failed to update provider config: ${res.status}`)
|
|
}
|
|
}
|
|
|
|
export interface LocationInput {
|
|
lat: number
|
|
lng: number
|
|
accuracy: number
|
|
}
|
|
|
|
export async function pushLocation(location: LocationInput): Promise<void> {
|
|
const res = await fetch(`${serverBase()}/location`, {
|
|
method: "POST",
|
|
headers: { "Content-Type": "application/json" },
|
|
credentials: "include",
|
|
body: JSON.stringify({
|
|
...location,
|
|
timestamp: new Date().toISOString(),
|
|
}),
|
|
})
|
|
if (!res.ok) {
|
|
const data = (await res.json()) as { error?: string }
|
|
throw new Error(data.error ?? `Failed to push location: ${res.status}`)
|
|
}
|
|
}
|
|
|
|
export interface FeedItemSlot {
|
|
description: string
|
|
content: string | null
|
|
}
|
|
|
|
export interface FeedItem {
|
|
id: string
|
|
sourceId: string
|
|
type: string
|
|
timestamp: string
|
|
data: Record<string, unknown>
|
|
signals?: {
|
|
urgency?: number
|
|
timeRelevance?: string
|
|
}
|
|
slots?: Record<string, FeedItemSlot>
|
|
ui?: unknown
|
|
}
|
|
|
|
export interface FeedResponse {
|
|
items: FeedItem[]
|
|
errors: { sourceId: string; error: string }[]
|
|
}
|
|
|
|
export async function fetchFeed(): Promise<FeedResponse> {
|
|
const res = await fetch(`${serverBase()}/feed`, { credentials: "include" })
|
|
if (!res.ok) throw new Error(`Failed to fetch feed: ${res.status}`)
|
|
return res.json() as Promise<FeedResponse>
|
|
}
|