mirror of
https://github.com/openclaw/openclaw.git
synced 2026-05-30 06:53:39 +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
214 lines
6.0 KiB
TypeScript
214 lines
6.0 KiB
TypeScript
import { normalizeOptionalString } from "../shared/string-coerce.js";
|
|
import {
|
|
createTalkEventSequencer,
|
|
type TalkBrain,
|
|
type TalkEvent,
|
|
type TalkEventContext,
|
|
type TalkEventInput,
|
|
type TalkEventSequencer,
|
|
type TalkMode,
|
|
type TalkTransport,
|
|
} from "./talk-events.js";
|
|
|
|
export type TalkTurnFailureReason = "no_active_turn" | "stale_turn";
|
|
|
|
export type TalkTurnSuccess = {
|
|
event: TalkEvent;
|
|
ok: true;
|
|
turnId: string;
|
|
};
|
|
|
|
export type TalkTurnFailure = {
|
|
ok: false;
|
|
reason: TalkTurnFailureReason;
|
|
};
|
|
|
|
export type TalkTurnResult = TalkTurnSuccess | TalkTurnFailure;
|
|
|
|
export type TalkEnsureTurnResult = {
|
|
event?: TalkEvent;
|
|
turnId: string;
|
|
};
|
|
|
|
export type TalkSessionController = {
|
|
readonly activeTurnId: string | undefined;
|
|
readonly context: TalkEventContext;
|
|
readonly outputAudioActive: boolean;
|
|
readonly recentEvents: readonly TalkEvent[];
|
|
clearActiveTurn(): void;
|
|
emit<TPayload>(input: TalkEventInput<TPayload>): TalkEvent<TPayload>;
|
|
ensureTurn(params?: { payload?: unknown; turnId?: string }): TalkEnsureTurnResult;
|
|
startTurn(params?: { payload?: unknown; turnId?: string }): TalkEnsureTurnResult;
|
|
endTurn(params?: { payload?: unknown; turnId?: string }): TalkTurnResult;
|
|
cancelTurn(params?: { payload?: unknown; turnId?: string }): TalkTurnResult;
|
|
finishOutputAudio(params?: { payload?: unknown; turnId?: string }): TalkEvent | undefined;
|
|
startOutputAudio(params?: { payload?: unknown; turnId?: string }): TalkEnsureTurnResult;
|
|
};
|
|
|
|
export type TalkSessionControllerParams = TalkEventContext & {
|
|
maxRecentEvents?: number;
|
|
turnIdPrefix?: string;
|
|
};
|
|
|
|
export type TalkSessionControllerOptions = {
|
|
now?: () => Date | string;
|
|
onEvent?: (event: TalkEvent) => void;
|
|
sequencer?: TalkEventSequencer;
|
|
};
|
|
|
|
export function createTalkSessionController(
|
|
params: TalkSessionControllerParams,
|
|
options: TalkSessionControllerOptions = {},
|
|
): TalkSessionController {
|
|
const { maxRecentEvents = 20, turnIdPrefix = "turn", ...context } = params;
|
|
const sequencer = options.sequencer ?? createTalkEventSequencer(context, { now: options.now });
|
|
const recentEvents: TalkEvent[] = [];
|
|
let activeTurnId: string | undefined;
|
|
let outputAudioActive = false;
|
|
let turnSeq = 0;
|
|
|
|
const remember = <TPayload>(event: TalkEvent<TPayload>): TalkEvent<TPayload> => {
|
|
recentEvents.push(event as TalkEvent);
|
|
if (recentEvents.length > maxRecentEvents) {
|
|
recentEvents.splice(0, recentEvents.length - maxRecentEvents);
|
|
}
|
|
try {
|
|
options.onEvent?.(event as TalkEvent);
|
|
} catch {
|
|
// Diagnostics hooks must not break Talk delivery.
|
|
}
|
|
return event;
|
|
};
|
|
|
|
const emit = <TPayload>(input: TalkEventInput<TPayload>): TalkEvent<TPayload> => {
|
|
return remember(sequencer.next(input));
|
|
};
|
|
|
|
const resolveActiveTurn = (requestedTurnId: string | undefined): string | TalkTurnFailure => {
|
|
if (!activeTurnId) {
|
|
return { ok: false, reason: "no_active_turn" };
|
|
}
|
|
const normalizedRequested = normalizeOptionalString(requestedTurnId);
|
|
if (normalizedRequested && normalizedRequested !== activeTurnId) {
|
|
return { ok: false, reason: "stale_turn" };
|
|
}
|
|
return activeTurnId;
|
|
};
|
|
|
|
const ensureTurn = (ensureParams: { payload?: unknown; turnId?: string } = {}) => {
|
|
if (activeTurnId) {
|
|
return { turnId: activeTurnId };
|
|
}
|
|
return startTurn(ensureParams);
|
|
};
|
|
|
|
const startTurn = (startParams: { payload?: unknown; turnId?: string } = {}) => {
|
|
const turnId = normalizeOptionalString(startParams.turnId) ?? `${turnIdPrefix}-${++turnSeq}`;
|
|
outputAudioActive = false;
|
|
activeTurnId = turnId;
|
|
return {
|
|
turnId,
|
|
event: emit({
|
|
type: "turn.started",
|
|
turnId,
|
|
payload: startParams.payload ?? {},
|
|
}),
|
|
};
|
|
};
|
|
|
|
const finishTurn = (
|
|
type: "turn.ended" | "turn.cancelled",
|
|
paramsForTurn: { payload?: unknown; turnId?: string } = {},
|
|
): TalkTurnResult => {
|
|
const turnId = resolveActiveTurn(paramsForTurn.turnId);
|
|
if (typeof turnId !== "string") {
|
|
return turnId;
|
|
}
|
|
outputAudioActive = false;
|
|
activeTurnId = undefined;
|
|
return {
|
|
ok: true,
|
|
turnId,
|
|
event: emit({
|
|
type,
|
|
turnId,
|
|
payload: paramsForTurn.payload ?? {},
|
|
final: true,
|
|
}),
|
|
};
|
|
};
|
|
|
|
return {
|
|
get activeTurnId() {
|
|
return activeTurnId;
|
|
},
|
|
context,
|
|
get outputAudioActive() {
|
|
return outputAudioActive;
|
|
},
|
|
get recentEvents() {
|
|
return recentEvents;
|
|
},
|
|
clearActiveTurn() {
|
|
activeTurnId = undefined;
|
|
outputAudioActive = false;
|
|
},
|
|
emit,
|
|
ensureTurn,
|
|
startTurn,
|
|
endTurn(paramsForTurn) {
|
|
return finishTurn("turn.ended", paramsForTurn);
|
|
},
|
|
cancelTurn(paramsForTurn) {
|
|
return finishTurn("turn.cancelled", paramsForTurn);
|
|
},
|
|
finishOutputAudio(paramsForOutput = {}) {
|
|
if (!outputAudioActive) {
|
|
return undefined;
|
|
}
|
|
const turnId = resolveActiveTurn(paramsForOutput.turnId);
|
|
if (typeof turnId !== "string") {
|
|
return undefined;
|
|
}
|
|
outputAudioActive = false;
|
|
return emit({
|
|
type: "output.audio.done",
|
|
turnId,
|
|
payload: paramsForOutput.payload ?? {},
|
|
final: true,
|
|
});
|
|
},
|
|
startOutputAudio(paramsForOutput = {}) {
|
|
const turn = ensureTurn({ turnId: paramsForOutput.turnId, payload: {} });
|
|
if (outputAudioActive) {
|
|
return { turnId: turn.turnId };
|
|
}
|
|
outputAudioActive = true;
|
|
return {
|
|
turnId: turn.turnId,
|
|
event: emit({
|
|
type: "output.audio.started",
|
|
turnId: turn.turnId,
|
|
payload: paramsForOutput.payload ?? {},
|
|
}),
|
|
};
|
|
},
|
|
};
|
|
}
|
|
|
|
export function normalizeTalkTransport(value: string | undefined): string | undefined {
|
|
const normalized = normalizeOptionalString(value);
|
|
if (!normalized) {
|
|
return undefined;
|
|
}
|
|
if (normalized === "webrtc-sdp") {
|
|
return "webrtc";
|
|
}
|
|
if (normalized === "json-pcm-websocket") {
|
|
return "provider-websocket";
|
|
}
|
|
return normalized;
|
|
}
|
|
|
|
export type { TalkBrain, TalkEvent, TalkEventContext, TalkEventInput, TalkMode, TalkTransport };
|