mirror of
https://github.com/openclaw/openclaw.git
synced 2026-06-03 16:04:07 +00:00
* refactor: share talk event metric extraction * refactor: reuse shared coercion helpers * refactor: reuse shared primitive guards * refactor: reuse shared record guard * refactor: reuse shared primitive helpers * refactor: reuse shared string guards * refactor: reuse shared non-empty string guard * refactor: share plugin primitive coercion helpers * refactor: reuse plugin coercion helpers * refactor: reuse plugin coercion helpers in more plugins * refactor: reuse channel coercion helpers * refactor: reuse monitor coercion helpers * refactor: reuse provider coercion helpers * refactor: reuse core coercion helpers * refactor: reuse runtime coercion helpers * refactor: reuse helper coercion in codex paths * refactor: reuse helper coercion in runtime paths * refactor: reuse codex app-server coercion helpers * refactor: reuse codex record helpers * refactor: reuse migration and qa record helpers * refactor: reuse feishu and core helper guards * refactor: reuse browser and policy coercion helpers * refactor: reuse memory wiki record helper * refactor: share boolean coercion helpers * refactor: reuse finite number coercion * refactor: reuse trimmed string list helpers * refactor: reuse string list normalization * refactor: reuse remaining string list helpers * refactor: reuse string entry normalizer * refactor: share sorted string helpers * refactor: share string list normalization * test: preserve command registry browser imports * refactor: reuse trimmed list helpers * refactor: reuse string dedupe helpers * refactor: reuse local dedupe helpers * refactor: reuse more string dedupe helpers * refactor: reuse command string dedupe helpers * refactor: dedupe memory path lists with helper * refactor: expose string dedupe helpers to plugins * refactor: reuse core string dedupe helpers * refactor: reuse shared unique value helpers * refactor: reuse unique helpers in agent utilities * refactor: reuse unique helpers in config plumbing * refactor: reuse unique helpers in extensions * refactor: reuse unique helpers in core utilities * refactor: reuse unique helpers in qa plugins * refactor: reuse unique helpers in memory plugins * refactor: reuse unique helpers in channel plugins * refactor: reuse unique helpers in core tails * refactor: reuse unique helper in comfy workflow * refactor: reuse unique helpers in test utilities * refactor: expose unique value helper to plugins * refactor: reuse unique helpers for numeric lists * refactor: replace index dedupe filters * refactor: reuse string entry normalization * refactor: reuse string normalization in plugin helpers * refactor: reuse string normalization in extension helpers * refactor: reuse string normalization in channel parsers * refactor: reuse string normalization in memory search * refactor: reuse string normalization in provider parsers * refactor: reuse string normalization in qa helpers * refactor: reuse string normalization in infra parsers * refactor: reuse string normalization in messaging parsers * refactor: reuse string normalization in core parsers * refactor: reuse string normalization in extension parsers * refactor: reuse string normalization in remaining parsers * refactor: reuse string normalization in final parser spots * refactor: reuse string normalization in qa media helpers * refactor: reuse normalization in provider and media lists * refactor: reuse normalization for remaining set filters * refactor: reuse normalization in policy allowlists * refactor: reuse normalization in session and owner lists * refactor: centralize primitive string lists * refactor: reuse lowercase entry helpers * refactor: reuse sorted string helpers * refactor: reuse unique trimmed helpers * refactor: reuse string normalization helpers * refactor: reuse catalog string helpers * refactor: reuse remaining string helpers * refactor: simplify remaining list normalization * refactor: reuse codex auth order normalization * chore: refresh plugin sdk api baseline * fix: make shared string sorting deterministic * chore: refresh plugin sdk api baseline * fix: align host env security ordering
164 lines
4.8 KiB
TypeScript
164 lines
4.8 KiB
TypeScript
import {
|
|
createProviderHttpError,
|
|
resolveProviderRequestHeaders,
|
|
} from "openclaw/plugin-sdk/provider-http";
|
|
import { captureWsEvent } from "openclaw/plugin-sdk/proxy-capture";
|
|
import { fetchWithSsrFGuard } from "openclaw/plugin-sdk/ssrf-runtime";
|
|
import {
|
|
asFiniteNumber,
|
|
asOptionalRecord as asObjectRecord,
|
|
normalizeOptionalString,
|
|
} from "openclaw/plugin-sdk/string-coerce-runtime";
|
|
|
|
export const trimToUndefined = normalizeOptionalString;
|
|
export { asFiniteNumber, asObjectRecord };
|
|
|
|
export function readRealtimeErrorDetail(error: unknown): string {
|
|
if (typeof error === "string" && error) {
|
|
return error;
|
|
}
|
|
const message = asObjectRecord(error)?.message;
|
|
if (typeof message === "string" && message) {
|
|
return message;
|
|
}
|
|
return "Unknown error";
|
|
}
|
|
|
|
export function resolveOpenAIProviderConfigRecord(
|
|
config: Record<string, unknown>,
|
|
): Record<string, unknown> | undefined {
|
|
const providers = asObjectRecord(config.providers);
|
|
return (
|
|
asObjectRecord(providers?.openai) ?? asObjectRecord(config.openai) ?? asObjectRecord(config)
|
|
);
|
|
}
|
|
|
|
export function captureOpenAIRealtimeWsClose(params: {
|
|
url: string;
|
|
flowId: string;
|
|
capability: "realtime-transcription" | "realtime-voice";
|
|
code: unknown;
|
|
reasonBuffer: unknown;
|
|
}): void {
|
|
captureWsEvent({
|
|
url: params.url,
|
|
direction: "local",
|
|
kind: "ws-close",
|
|
flowId: params.flowId,
|
|
closeCode: typeof params.code === "number" ? params.code : undefined,
|
|
meta: {
|
|
provider: "openai",
|
|
capability: params.capability,
|
|
reason:
|
|
Buffer.isBuffer(params.reasonBuffer) && params.reasonBuffer.length > 0
|
|
? params.reasonBuffer.toString("utf8")
|
|
: undefined,
|
|
},
|
|
});
|
|
}
|
|
|
|
export type OpenAIRealtimeClientSecretResult = {
|
|
value: string;
|
|
expiresAt?: number;
|
|
};
|
|
|
|
type OpenAIRealtimeSecretRequest = {
|
|
authToken: string;
|
|
auditContext: string;
|
|
url: string;
|
|
body: unknown;
|
|
errorMessage: string;
|
|
missingValueMessage: string;
|
|
};
|
|
|
|
function readStringField(value: unknown, key: string): string | undefined {
|
|
if (!value || typeof value !== "object") {
|
|
return undefined;
|
|
}
|
|
const raw = (value as Record<string, unknown>)[key];
|
|
return typeof raw === "string" && raw.trim() ? raw.trim() : undefined;
|
|
}
|
|
|
|
async function createOpenAIRealtimeSecret(
|
|
params: OpenAIRealtimeSecretRequest,
|
|
): Promise<OpenAIRealtimeClientSecretResult> {
|
|
const { response, release } = await fetchWithSsrFGuard({
|
|
url: params.url,
|
|
init: {
|
|
method: "POST",
|
|
headers: resolveProviderRequestHeaders({
|
|
provider: "openai",
|
|
baseUrl: params.url,
|
|
capability: "audio",
|
|
transport: "http",
|
|
defaultHeaders: {
|
|
Authorization: `Bearer ${params.authToken}`,
|
|
"Content-Type": "application/json",
|
|
},
|
|
}) ?? {
|
|
Authorization: `Bearer ${params.authToken}`,
|
|
"Content-Type": "application/json",
|
|
},
|
|
body: JSON.stringify(params.body),
|
|
},
|
|
auditContext: params.auditContext,
|
|
});
|
|
const payload = await (async () => {
|
|
try {
|
|
if (!response.ok) {
|
|
throw await createProviderHttpError(response, params.errorMessage);
|
|
}
|
|
return (await response.json()) as unknown;
|
|
} finally {
|
|
await release();
|
|
}
|
|
})();
|
|
const nestedSecret =
|
|
payload && typeof payload === "object"
|
|
? (payload as Record<string, unknown>).client_secret
|
|
: undefined;
|
|
const clientSecret = readStringField(payload, "value") ?? readStringField(nestedSecret, "value");
|
|
if (!clientSecret) {
|
|
throw new Error(params.missingValueMessage);
|
|
}
|
|
const expiresAt =
|
|
payload && typeof payload === "object"
|
|
? (payload as Record<string, unknown>).expires_at
|
|
: undefined;
|
|
return {
|
|
value: clientSecret,
|
|
...(typeof expiresAt === "number" ? { expiresAt } : {}),
|
|
};
|
|
}
|
|
|
|
export async function createOpenAIRealtimeClientSecret(params: {
|
|
authToken: string;
|
|
auditContext: string;
|
|
session: Record<string, unknown>;
|
|
}): Promise<OpenAIRealtimeClientSecretResult> {
|
|
const url = "https://api.openai.com/v1/realtime/client_secrets";
|
|
return createOpenAIRealtimeSecret({
|
|
...params,
|
|
url,
|
|
body: { session: params.session },
|
|
errorMessage: "OpenAI Realtime client secret failed",
|
|
missingValueMessage: "OpenAI Realtime client secret response did not include a value",
|
|
});
|
|
}
|
|
|
|
export async function createOpenAIRealtimeTranscriptionClientSecret(params: {
|
|
authToken: string;
|
|
auditContext: string;
|
|
session: Record<string, unknown>;
|
|
}): Promise<OpenAIRealtimeClientSecretResult> {
|
|
const url = "https://api.openai.com/v1/realtime/transcription_sessions";
|
|
return createOpenAIRealtimeSecret({
|
|
...params,
|
|
url,
|
|
body: params.session,
|
|
errorMessage: "OpenAI Realtime transcription client secret failed",
|
|
missingValueMessage:
|
|
"OpenAI Realtime transcription client secret response did not include a value",
|
|
});
|
|
}
|