|
|
|
|
Quelle config.ts
Sprache: JAVA
|
|
Spracherkennung für: .ts vermutete Sprache: Unknown {[0] [0] [0]} [Methode: Schwerpunktbildung, einfache Gewichte, sechs Dimensionen]
import type { GatewayBrowserClient } from "../gateway.ts";
import type { ConfigSchemaResponse, ConfigSnapshot, ConfigUiHints } from "../types.ts";
import type { JsonSchema } from "../views/config-form.shared.ts";
import { coerceFormValues } from "./config/form-coerce.ts";
import {
cloneConfigObject,
removePathValue,
serializeConfigForm,
setPathValue,
} from "./config/form-utils.ts";
export type ConfigState = {
client: GatewayBrowserClient | null;
connected: boolean;
applySessionKey: string;
configLoading: boolean;
configRaw: string;
configRawOriginal: string;
configValid: boolean | null;
configIssues: unknown[];
configSaving: boolean;
configApplying: boolean;
updateRunning: boolean;
configSnapshot: ConfigSnapshot | null;
configSchema: unknown;
configSchemaVersion: string | null;
configSchemaLoading: boolean;
configUiHints: ConfigUiHints;
configForm: Record<string, unknown> | null;
configFormOriginal: Record<string, unknown> | null;
configFormDirty: boolean;
configFormMode: "form" | "raw";
configSearchQuery: string;
configActiveSection: string | null;
configActiveSubsection: string | null;
lastError: string | null;
};
export async function loadConfig(state: ConfigState) {
if (!state.client || !state.connected) {
return;
}
state.configLoading = true;
state.lastError = null;
try {
const res = await state.client.request<ConfigSnapshot>("config.get", {});
applyConfigSnapshot(state, res);
} catch (err) {
state.lastError = String(err);
} finally {
state.configLoading = false;
}
}
export async function loadConfigSchema(state: ConfigState) {
if (!state.client || !state.connected) {
return;
}
if (state.configSchemaLoading) {
return;
}
state.configSchemaLoading = true;
try {
const res = await state.client.request<ConfigSchemaResponse>("config.schema", {});
applyConfigSchema(state, res);
} catch (err) {
state.lastError = String(err);
} finally {
state.configSchemaLoading = false;
}
}
export function applyConfigSchema(state: ConfigState, res: ConfigSchemaResponse) {
state.configSchema = res.schema ?? null;
state.configUiHints = res.uiHints ?? {};
state.configSchemaVersion = res.version ?? null;
}
export function applyConfigSnapshot(state: ConfigState, snapshot: ConfigSnapshot) {
state.configSnapshot = snapshot;
const rawAvailable = typeof snapshot.raw === "string";
if (!rawAvailable && state.configFormMode === "raw") {
state.configFormMode = "form";
}
const rawFromSnapshot: string =
typeof snapshot.raw === "string"
? snapshot.raw
: snapshot.config && typeof snapshot.config === "object"
? serializeConfigForm(snapshot.config)
: state.configRaw;
if (!state.configFormDirty || state.configFormMode === "raw") {
state.configRaw = rawFromSnapshot;
} else if (state.configForm) {
state.configRaw = serializeConfigForm(state.configForm);
} else {
state.configRaw = rawFromSnapshot;
}
state.configValid = typeof snapshot.valid === "boolean" ? snapshot.valid : null;
state.configIssues = Array.isArray(snapshot.issues) ? snapshot.issues : [];
if (!state.configFormDirty) {
state.configForm = cloneConfigObject(snapshot.config ?? {});
state.configFormOriginal = cloneConfigObject(snapshot.config ?? {});
state.configRawOriginal = rawFromSnapshot;
}
}
function asJsonSchema(value: unknown): JsonSchema | null {
if (!value || typeof value !== "object" || Array.isArray(value)) {
return null;
}
return value as JsonSchema;
}
/**
* Serialize the form state for submission to `config.set` / `config.apply`.
*
* HTML `<input>` elements produce string `.value` properties, so numeric and
* boolean config fields can leak into `configForm` as strings. We coerce
* them back to their schema-defined types before JSON serialization so the
* gateway's Zod validation always sees correctly typed values.
*/
function serializeFormForSubmit(state: ConfigState): string {
if (state.configFormMode === "raw" && typeof state.configSnapshot?.raw !== "string") {
throw new Error("Raw config editing is unavailable for this snapshot. Switch to Form mode.");
}
if (state.configFormMode !== "form" || !state.configForm) {
return state.configRaw;
}
const schema = asJsonSchema(state.configSchema);
const form = schema
? (coerceFormValues(state.configForm, schema) as Record<string, unknown>)
: state.configForm;
return serializeConfigForm(form);
}
type ConfigSubmitMethod = "config.set" | "config.apply";
type ConfigSubmitBusyKey = "configSaving" | "configApplying";
async function submitConfigChange(
state: ConfigState,
method: ConfigSubmitMethod,
busyKey: ConfigSubmitBusyKey,
extraParams: Record<string, unknown> = {},
) {
if (!state.client || !state.connected) {
return;
}
state[busyKey] = true;
state.lastError = null;
try {
const raw = serializeFormForSubmit(state);
const baseHash = state.configSnapshot?.hash;
if (!baseHash) {
state.lastError = "Config hash missing; reload and retry.";
return;
}
await state.client.request(method, { raw, baseHash, ...extraParams });
state.configFormDirty = false;
await loadConfig(state);
} catch (err) {
state.lastError = String(err);
} finally {
state[busyKey] = false;
}
}
export async function saveConfig(state: ConfigState) {
await submitConfigChange(state, "config.set", "configSaving");
}
export async function applyConfig(state: ConfigState) {
await submitConfigChange(state, "config.apply", "configApplying", {
sessionKey: state.applySessionKey,
});
}
export async function runUpdate(state: ConfigState) {
if (!state.client || !state.connected) {
return;
}
state.updateRunning = true;
state.lastError = null;
try {
const res = await state.client.request<{
ok?: boolean;
result?: { status?: string; reason?: string };
}>("update.run", {
sessionKey: state.applySessionKey,
});
if (res && res.ok === false) {
const status = res.result?.status ?? "error";
const reason = res.result?.reason ?? "Update failed.";
state.lastError = `Update ${status}: ${reason}`;
}
} catch (err) {
state.lastError = String(err);
} finally {
state.updateRunning = false;
}
}
function mutateConfigForm(state: ConfigState, mutate: (draft: Record<string, unknown>) => void) {
const base = cloneConfigObject(state.configForm ?? state.configSnapshot?.config ?? {});
mutate(base);
state.configForm = base;
state.configFormDirty = true;
if (state.configFormMode === "form") {
state.configRaw = serializeConfigForm(base);
}
}
export function updateConfigFormValue(
state: ConfigState,
path: Array<string | number>,
value: unknown,
) {
mutateConfigForm(state, (draft) => setPathValue(draft, path, value));
}
export function resetConfigPendingChanges(state: ConfigState) {
state.configForm = cloneConfigObject(
state.configFormOriginal ?? state.configSnapshot?.config ?? {},
);
state.configRaw =
state.configRawOriginal ??
serializeConfigForm(state.configFormOriginal ?? state.configSnapshot?.config ?? {});
state.configFormDirty = false;
}
export function removeConfigFormValue(state: ConfigState, path: Array<string | number>) {
mutateConfigForm(state, (draft) => removePathValue(draft, path));
}
export function findAgentConfigEntryIndex(
config: Record<string, unknown> | null,
agentId: string,
): number {
const normalizedAgentId = agentId.trim();
if (!normalizedAgentId) {
return -1;
}
const list = (config as { agents?: { list?: unknown[] } } | null)?.agents?.list;
if (!Array.isArray(list)) {
return -1;
}
return list.findIndex(
(entry) =>
entry &&
typeof entry === "object" &&
"id" in entry &&
(entry as { id?: string }).id === normalizedAgentId,
);
}
export function ensureAgentConfigEntry(state: ConfigState, agentId: string): number {
const normalizedAgentId = agentId.trim();
if (!normalizedAgentId) {
return -1;
}
const source =
state.configForm ?? (state.configSnapshot?.config as Record<string, unknown> | null);
const existingIndex = findAgentConfigEntryIndex(source, normalizedAgentId);
if (existingIndex >= 0) {
return existingIndex;
}
const list = (source as { agents?: { list?: unknown[] } } | null)?.agents?.list;
const nextIndex = Array.isArray(list) ? list.length : 0;
updateConfigFormValue(state, ["agents", "list", nextIndex, "id"], normalizedAgentId);
return nextIndex;
}
export async function openConfigFile(state: ConfigState): Promise<void> {
if (!state.client || !state.connected) {
return;
}
try {
await state.client.request("config.openFile", {});
} catch {
const path = state.configSnapshot?.path;
if (path) {
try {
await navigator.clipboard.writeText(path);
} catch {
// ignore
}
}
}
}
¤ Dauer der Verarbeitung: 0.26 Sekunden
(vorverarbeitet am 2026-04-27)
¤
*© Formatika GbR, Deutschland
|
|
2026-05-26
|
|
|
|
|