@estuary-ai/sdk 0.1.21 → 0.1.23
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +21 -21
- package/README.md +2 -2
- package/dist/{chunk-6M5LSBMK.mjs → chunk-W5QYPYX3.mjs} +2 -2
- package/dist/chunk-W5QYPYX3.mjs.map +1 -0
- package/dist/index.d.mts +105 -15
- package/dist/index.d.ts +105 -15
- package/dist/index.js +64 -13
- package/dist/index.js.map +1 -1
- package/dist/index.mjs +59 -15
- package/dist/index.mjs.map +1 -1
- package/dist/{livekit-voice-RWXL7IXC.mjs → livekit-voice-A52TC6XZ.mjs} +3 -3
- package/dist/livekit-voice-A52TC6XZ.mjs.map +1 -0
- package/dist/{websocket-voice-UDGQPVQP.mjs → websocket-voice-6DMYBGHP.mjs} +36 -34
- package/dist/websocket-voice-6DMYBGHP.mjs.map +1 -0
- package/package.json +71 -71
- package/dist/chunk-6M5LSBMK.mjs.map +0 -1
- package/dist/livekit-voice-RWXL7IXC.mjs.map +0 -1
- package/dist/websocket-voice-UDGQPVQP.mjs.map +0 -1
package/dist/index.mjs
CHANGED
|
@@ -1,5 +1,5 @@
|
|
|
1
|
-
import { EstuaryError } from './chunk-
|
|
2
|
-
export { ErrorCode, EstuaryError } from './chunk-
|
|
1
|
+
import { EstuaryError } from './chunk-W5QYPYX3.mjs';
|
|
2
|
+
export { ErrorCode, EstuaryError } from './chunk-W5QYPYX3.mjs';
|
|
3
3
|
import { io } from 'socket.io-client';
|
|
4
4
|
|
|
5
5
|
// src/utils/event-emitter.ts
|
|
@@ -361,36 +361,39 @@ async function isLiveKitAvailable() {
|
|
|
361
361
|
}
|
|
362
362
|
async function createVoiceManager(transport, socketManager, sampleRate, logger) {
|
|
363
363
|
if (transport === "websocket") {
|
|
364
|
-
const { WebSocketVoiceManager } = await import('./websocket-voice-
|
|
364
|
+
const { WebSocketVoiceManager } = await import('./websocket-voice-6DMYBGHP.mjs');
|
|
365
365
|
return new WebSocketVoiceManager(socketManager, sampleRate, logger);
|
|
366
366
|
}
|
|
367
367
|
if (transport === "livekit") {
|
|
368
368
|
if (await isLiveKitAvailable()) {
|
|
369
|
-
const { LiveKitVoiceManager } = await import('./livekit-voice-
|
|
369
|
+
const { LiveKitVoiceManager } = await import('./livekit-voice-A52TC6XZ.mjs');
|
|
370
370
|
return new LiveKitVoiceManager(socketManager, logger);
|
|
371
371
|
}
|
|
372
372
|
logger.warn("livekit-client not installed, falling back to WebSocket voice");
|
|
373
|
-
const { WebSocketVoiceManager } = await import('./websocket-voice-
|
|
373
|
+
const { WebSocketVoiceManager } = await import('./websocket-voice-6DMYBGHP.mjs');
|
|
374
374
|
return new WebSocketVoiceManager(socketManager, sampleRate, logger);
|
|
375
375
|
}
|
|
376
376
|
if (transport === "auto") {
|
|
377
377
|
if (await isLiveKitAvailable()) {
|
|
378
|
-
const { LiveKitVoiceManager } = await import('./livekit-voice-
|
|
378
|
+
const { LiveKitVoiceManager } = await import('./livekit-voice-A52TC6XZ.mjs');
|
|
379
379
|
return new LiveKitVoiceManager(socketManager, logger);
|
|
380
380
|
}
|
|
381
|
-
const { WebSocketVoiceManager } = await import('./websocket-voice-
|
|
381
|
+
const { WebSocketVoiceManager } = await import('./websocket-voice-6DMYBGHP.mjs');
|
|
382
382
|
return new WebSocketVoiceManager(socketManager, sampleRate, logger);
|
|
383
383
|
}
|
|
384
384
|
return null;
|
|
385
385
|
}
|
|
386
386
|
|
|
387
387
|
// src/rest/rest-client.ts
|
|
388
|
+
var DEFAULT_TIMEOUT_MS = 1e4;
|
|
388
389
|
var RestClient = class {
|
|
389
390
|
baseUrl;
|
|
390
391
|
apiKey;
|
|
391
|
-
|
|
392
|
+
timeoutMs;
|
|
393
|
+
constructor(baseUrl, apiKey, timeoutMs = DEFAULT_TIMEOUT_MS) {
|
|
392
394
|
this.baseUrl = baseUrl.replace(/\/+$/, "");
|
|
393
395
|
this.apiKey = apiKey;
|
|
396
|
+
this.timeoutMs = timeoutMs;
|
|
394
397
|
}
|
|
395
398
|
async get(path, params) {
|
|
396
399
|
const url = this.buildUrl(path, params);
|
|
@@ -425,7 +428,11 @@ var RestClient = class {
|
|
|
425
428
|
async request(url, init) {
|
|
426
429
|
const headers = new Headers(init.headers);
|
|
427
430
|
headers.set("X-API-Key", this.apiKey);
|
|
428
|
-
const response = await fetch(url, {
|
|
431
|
+
const response = await fetch(url, {
|
|
432
|
+
...init,
|
|
433
|
+
headers,
|
|
434
|
+
signal: AbortSignal.timeout(this.timeoutMs)
|
|
435
|
+
});
|
|
429
436
|
if (!response.ok) {
|
|
430
437
|
let detail;
|
|
431
438
|
try {
|
|
@@ -481,6 +488,30 @@ var MemoryClient = class {
|
|
|
481
488
|
}
|
|
482
489
|
};
|
|
483
490
|
|
|
491
|
+
// src/rest/character-client.ts
|
|
492
|
+
var CharacterClient = class {
|
|
493
|
+
rest;
|
|
494
|
+
constructor(rest) {
|
|
495
|
+
this.rest = rest;
|
|
496
|
+
}
|
|
497
|
+
/** Fetch character details including 3D model and avatar URLs. */
|
|
498
|
+
async getCharacter(characterId) {
|
|
499
|
+
const raw = await this.rest.get(`/api/agents/${characterId}`);
|
|
500
|
+
return {
|
|
501
|
+
id: raw.id,
|
|
502
|
+
name: raw.name,
|
|
503
|
+
tagline: raw.tagline ?? null,
|
|
504
|
+
avatar: raw.avatar ?? null,
|
|
505
|
+
modelUrl: raw.modelUrl ?? null,
|
|
506
|
+
modelPreviewUrl: raw.modelPreviewUrl ?? null,
|
|
507
|
+
modelStatus: raw.modelStatus ?? null,
|
|
508
|
+
sourceImageUrl: raw.sourceImageUrl ?? null
|
|
509
|
+
};
|
|
510
|
+
}
|
|
511
|
+
dispose() {
|
|
512
|
+
}
|
|
513
|
+
};
|
|
514
|
+
|
|
484
515
|
// src/audio/audio-player.ts
|
|
485
516
|
var AudioPlayer = class {
|
|
486
517
|
sampleRate;
|
|
@@ -728,6 +759,7 @@ var EstuaryClient = class extends TypedEventEmitter {
|
|
|
728
759
|
voiceManager = null;
|
|
729
760
|
audioPlayer = null;
|
|
730
761
|
_memory;
|
|
762
|
+
_character;
|
|
731
763
|
_sessionInfo = null;
|
|
732
764
|
actionParsers = /* @__PURE__ */ new Map();
|
|
733
765
|
_hasAutoInterrupted = false;
|
|
@@ -740,11 +772,16 @@ var EstuaryClient = class extends TypedEventEmitter {
|
|
|
740
772
|
this.forwardSocketEvents();
|
|
741
773
|
const restClient = new RestClient(config.serverUrl, config.apiKey);
|
|
742
774
|
this._memory = new MemoryClient(restClient, config.characterId, config.playerId);
|
|
775
|
+
this._character = new CharacterClient(restClient);
|
|
743
776
|
}
|
|
744
777
|
/** Memory API client for querying memories, graphs, and facts */
|
|
745
778
|
get memory() {
|
|
746
779
|
return this._memory;
|
|
747
780
|
}
|
|
781
|
+
/** Fetch character details including 3D model and avatar URLs. */
|
|
782
|
+
async getCharacter(characterId) {
|
|
783
|
+
return this._character.getCharacter(characterId ?? this.config.characterId);
|
|
784
|
+
}
|
|
748
785
|
/** Current session info (null if not connected) */
|
|
749
786
|
get session() {
|
|
750
787
|
return this._sessionInfo;
|
|
@@ -765,20 +802,20 @@ var EstuaryClient = class extends TypedEventEmitter {
|
|
|
765
802
|
return session;
|
|
766
803
|
}
|
|
767
804
|
/** Disconnect from the server */
|
|
768
|
-
disconnect() {
|
|
805
|
+
async disconnect() {
|
|
769
806
|
this.logger.info("Disconnecting...");
|
|
770
807
|
if (this._autoInterruptGraceTimer) {
|
|
771
808
|
clearTimeout(this._autoInterruptGraceTimer);
|
|
772
809
|
this._autoInterruptGraceTimer = null;
|
|
773
810
|
}
|
|
774
|
-
this.stopVoice();
|
|
811
|
+
await this.stopVoice();
|
|
775
812
|
this.audioPlayer?.dispose();
|
|
776
813
|
this.audioPlayer = null;
|
|
777
814
|
this.socketManager.disconnect();
|
|
778
815
|
this._sessionInfo = null;
|
|
779
816
|
}
|
|
780
|
-
/** Send a text message to the character */
|
|
781
|
-
sendText(text, textOnly =
|
|
817
|
+
/** Send a text message to the character. Defaults to textOnly=true (no TTS audio response). Pass textOnly=false to receive voice audio. */
|
|
818
|
+
sendText(text, textOnly = true) {
|
|
782
819
|
this.ensureConnected();
|
|
783
820
|
this.socketManager.emitEvent("text", { text, textOnly });
|
|
784
821
|
}
|
|
@@ -856,9 +893,9 @@ var EstuaryClient = class extends TypedEventEmitter {
|
|
|
856
893
|
this.emit("voiceStarted");
|
|
857
894
|
}
|
|
858
895
|
/** Stop voice input */
|
|
859
|
-
stopVoice() {
|
|
896
|
+
async stopVoice() {
|
|
860
897
|
if (this.voiceManager?.isActive) {
|
|
861
|
-
this.voiceManager.stop();
|
|
898
|
+
await this.voiceManager.stop();
|
|
862
899
|
this.voiceManager.dispose();
|
|
863
900
|
this.voiceManager = null;
|
|
864
901
|
this.emit("voiceStopped");
|
|
@@ -875,6 +912,13 @@ var EstuaryClient = class extends TypedEventEmitter {
|
|
|
875
912
|
get isMuted() {
|
|
876
913
|
return this.voiceManager?.isMuted ?? false;
|
|
877
914
|
}
|
|
915
|
+
/** Get/set suppressMicDuringPlayback at runtime (no reconnect needed) */
|
|
916
|
+
get suppressMicDuringPlayback() {
|
|
917
|
+
return this.config.suppressMicDuringPlayback ?? false;
|
|
918
|
+
}
|
|
919
|
+
set suppressMicDuringPlayback(enabled) {
|
|
920
|
+
this.config.suppressMicDuringPlayback = enabled;
|
|
921
|
+
}
|
|
878
922
|
/** Whether voice is currently active */
|
|
879
923
|
get isVoiceActive() {
|
|
880
924
|
return this.voiceManager?.isActive ?? false;
|
package/dist/index.mjs.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/utils/event-emitter.ts","../src/types.ts","../src/connection/socket-manager.ts","../src/voice/voice-manager.ts","../src/rest/rest-client.ts","../src/rest/memory-client.ts","../src/audio/audio-player.ts","../src/utils/logger.ts","../src/utils/action-parser.ts","../src/client.ts"],"names":["ConnectionState"],"mappings":"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","file":"index.mjs","sourcesContent":["export class TypedEventEmitter<T extends Record<string, unknown>> {\r\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\r\n private listeners = new Map<keyof T, Set<any>>();\r\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\r\n private onceListeners = new Map<keyof T, Set<any>>();\r\n\r\n on<K extends keyof T>(event: K, listener: T[K]): this {\r\n if (!this.listeners.has(event)) {\r\n this.listeners.set(event, new Set());\r\n }\r\n this.listeners.get(event)!.add(listener);\r\n return this;\r\n }\r\n\r\n off<K extends keyof T>(event: K, listener: T[K]): this {\r\n this.listeners.get(event)?.delete(listener);\r\n this.onceListeners.get(event)?.delete(listener);\r\n return this;\r\n }\r\n\r\n once<K extends keyof T>(event: K, listener: T[K]): this {\r\n if (!this.onceListeners.has(event)) {\r\n this.onceListeners.set(event, new Set());\r\n }\r\n this.onceListeners.get(event)!.add(listener);\r\n this.on(event, listener);\r\n return this;\r\n }\r\n\r\n protected emit<K extends keyof T>(\r\n event: K,\r\n ...args: T[K] extends (...a: infer A) => void ? A : never[]\r\n ): boolean {\r\n const listeners = this.listeners.get(event);\r\n if (!listeners || listeners.size === 0) return false;\r\n\r\n for (const listener of listeners) {\r\n try {\r\n (listener as Function)(...args);\r\n } catch {\r\n // Listener threw — swallow to not break other listeners\r\n }\r\n }\r\n\r\n // Remove once listeners after firing\r\n const onceSet = this.onceListeners.get(event);\r\n if (onceSet) {\r\n for (const listener of onceSet) {\r\n this.listeners.get(event)?.delete(listener);\r\n }\r\n onceSet.clear();\r\n }\r\n\r\n return true;\r\n }\r\n\r\n removeAllListeners<K extends keyof T>(event?: K): this {\r\n if (event) {\r\n this.listeners.delete(event);\r\n this.onceListeners.delete(event);\r\n } else {\r\n this.listeners.clear();\r\n this.onceListeners.clear();\r\n }\r\n return this;\r\n }\r\n\r\n listenerCount<K extends keyof T>(event: K): number {\r\n return this.listeners.get(event)?.size ?? 0;\r\n }\r\n}\r\n","// ─── Configuration ───────────────────────────────────────────────\r\n\r\nexport interface EstuaryConfig {\r\n /** Base URL of the Estuary server (e.g., \"https://api.estuary-ai.com\") */\r\n serverUrl: string;\r\n /** API key (starts with \"est_\") */\r\n apiKey: string;\r\n /** Character (agent) ID */\r\n characterId: string;\r\n /** Unique identifier for the end user */\r\n playerId: string;\r\n /** Audio sample rate in Hz (default: 16000) */\r\n audioSampleRate?: number;\r\n /** Auto-reconnect on disconnect (default: true) */\r\n autoReconnect?: boolean;\r\n /** Max reconnect attempts (default: 5) */\r\n maxReconnectAttempts?: number;\r\n /** Delay between reconnect attempts in ms (default: 2000) */\r\n reconnectDelayMs?: number;\r\n /** Enable debug logging (default: false) */\r\n debug?: boolean;\r\n /** Voice transport: 'websocket' | 'livekit' | 'auto' (default: 'auto') */\r\n voiceTransport?: VoiceTransport;\r\n /** Enable real-time memory extraction after each response (default: false) */\r\n realtimeMemory?: boolean;\r\n /** Suppress mic during TTS playback (software AEC fallback, disables barge-in). Default: false */\r\n suppressMicDuringPlayback?: boolean;\r\n /** Proactively interrupt bot audio when user starts speaking (default: true) */\r\n autoInterruptOnSpeech?: boolean;\r\n}\r\n\r\nexport type VoiceTransport = 'websocket' | 'livekit' | 'auto';\r\n\r\n// ─── Connection State ────────────────────────────────────────────\r\n\r\nexport enum ConnectionState {\r\n Disconnected = 'disconnected',\r\n Connecting = 'connecting',\r\n Connected = 'connected',\r\n Reconnecting = 'reconnecting',\r\n Error = 'error',\r\n}\r\n\r\n// ─── Wire Format Types (snake_case from server) ─────────────────\r\n\r\n/** @internal */\r\nexport interface WireSessionInfo {\r\n session_id: string;\r\n conversation_id: string;\r\n character_id: string;\r\n player_id: string;\r\n}\r\n\r\n/** @internal */\r\nexport interface WireBotResponse {\r\n text: string;\r\n is_final: boolean;\r\n partial: string;\r\n message_id: string;\r\n chunk_index: number;\r\n is_interjection: boolean;\r\n token_stream?: boolean;\r\n}\r\n\r\n/** @internal */\r\nexport interface WireBotVoice {\r\n audio: string;\r\n message_id: string;\r\n chunk_index: number;\r\n is_final: boolean;\r\n}\r\n\r\n/** @internal */\r\nexport interface WireSttResponse {\r\n text: string;\r\n is_final: boolean;\r\n}\r\n\r\n/** @internal */\r\nexport interface WireInterruptData {\r\n message_id?: string;\r\n reason?: string;\r\n interrupted_at?: string;\r\n}\r\n\r\n/** @internal */\r\nexport interface WireQuotaExceededData {\r\n message: string;\r\n current: number;\r\n limit: number;\r\n remaining: number;\r\n tier: string;\r\n}\r\n\r\n/** @internal */\r\nexport interface WireLiveKitTokenResponse {\r\n token: string;\r\n url: string;\r\n room: string;\r\n}\r\n\r\n/** @internal */\r\nexport interface WireCameraCaptureRequest {\r\n request_id: string;\r\n text?: string;\r\n}\r\n\r\n/** @internal — new_memories items are already camelCase from Memory.to_dict() */\r\nexport interface WireMemoryUpdated {\r\n agent_id: string;\r\n player_id: string;\r\n memories_extracted: number;\r\n facts_extracted: number;\r\n conversation_id: string;\r\n new_memories: MemoryData[];\r\n timestamp: string;\r\n}\r\n\r\n// ─── Public Types (camelCase) ────────────────────────────────────\r\n\r\nexport interface SessionInfo {\r\n sessionId: string;\r\n conversationId: string;\r\n characterId: string;\r\n playerId: string;\r\n}\r\n\r\nexport interface BotResponse {\r\n text: string;\r\n isFinal: boolean;\r\n partial: string;\r\n messageId: string;\r\n chunkIndex: number;\r\n isInterjection: boolean;\r\n tokenStream?: boolean;\r\n}\r\n\r\nexport interface BotVoice {\r\n audio: string;\r\n messageId: string;\r\n chunkIndex: number;\r\n isFinal: boolean;\r\n}\r\n\r\nexport interface SttResponse {\r\n text: string;\r\n isFinal: boolean;\r\n}\r\n\r\nexport interface InterruptData {\r\n messageId?: string;\r\n reason?: string;\r\n interruptedAt?: string;\r\n}\r\n\r\nexport interface QuotaExceededData {\r\n message: string;\r\n current: number;\r\n limit: number;\r\n remaining: number;\r\n tier: string;\r\n}\r\n\r\nexport interface LiveKitTokenResponse {\r\n token: string;\r\n url: string;\r\n room: string;\r\n}\r\n\r\nexport interface CameraCaptureRequest {\r\n requestId: string;\r\n text?: string;\r\n}\r\n\r\nexport interface MemoryData {\r\n id: string;\r\n content: string;\r\n memoryType: string;\r\n confidence: number;\r\n status: string;\r\n sourceConversationId: string;\r\n sourceQuote?: string;\r\n topic?: string;\r\n createdAt: string;\r\n}\r\n\r\nexport interface MemoryUpdatedEvent {\r\n agentId: string;\r\n playerId: string;\r\n memoriesExtracted: number;\r\n factsExtracted: number;\r\n conversationId: string;\r\n newMemories: MemoryData[];\r\n timestamp: string;\r\n}\r\n\r\n// ─── Wire → Public Converters ────────────────────────────────────\r\n\r\n/** @internal */\r\nexport function toSessionInfo(wire: WireSessionInfo): SessionInfo {\r\n return {\r\n sessionId: wire.session_id,\r\n conversationId: wire.conversation_id,\r\n characterId: wire.character_id,\r\n playerId: wire.player_id,\r\n };\r\n}\r\n\r\n/** @internal */\r\nexport function toBotResponse(wire: WireBotResponse): BotResponse {\r\n return {\r\n text: wire.text,\r\n isFinal: wire.is_final,\r\n partial: wire.partial,\r\n messageId: wire.message_id,\r\n chunkIndex: wire.chunk_index,\r\n isInterjection: wire.is_interjection,\r\n tokenStream: wire.token_stream,\r\n };\r\n}\r\n\r\n/** @internal */\r\nexport function toBotVoice(wire: WireBotVoice): BotVoice {\r\n return {\r\n audio: wire.audio,\r\n messageId: wire.message_id,\r\n chunkIndex: wire.chunk_index,\r\n isFinal: wire.is_final,\r\n };\r\n}\r\n\r\n/** @internal */\r\nexport function toSttResponse(wire: WireSttResponse): SttResponse {\r\n return {\r\n text: wire.text,\r\n isFinal: wire.is_final,\r\n };\r\n}\r\n\r\n/** @internal */\r\nexport function toInterruptData(wire: WireInterruptData): InterruptData {\r\n return {\r\n messageId: wire.message_id,\r\n reason: wire.reason,\r\n interruptedAt: wire.interrupted_at,\r\n };\r\n}\r\n\r\n/** @internal */\r\nexport function toQuotaExceededData(wire: WireQuotaExceededData): QuotaExceededData {\r\n return {\r\n message: wire.message,\r\n current: wire.current,\r\n limit: wire.limit,\r\n remaining: wire.remaining,\r\n tier: wire.tier,\r\n };\r\n}\r\n\r\n/** @internal */\r\nexport function toLiveKitTokenResponse(wire: WireLiveKitTokenResponse): LiveKitTokenResponse {\r\n return {\r\n token: wire.token,\r\n url: wire.url,\r\n room: wire.room,\r\n };\r\n}\r\n\r\n/** @internal */\r\nexport function toCameraCaptureRequest(wire: WireCameraCaptureRequest): CameraCaptureRequest {\r\n return {\r\n requestId: wire.request_id,\r\n text: wire.text,\r\n };\r\n}\r\n\r\n/** @internal */\r\nexport function toMemoryUpdatedEvent(wire: WireMemoryUpdated): MemoryUpdatedEvent {\r\n return {\r\n agentId: wire.agent_id,\r\n playerId: wire.player_id,\r\n memoriesExtracted: wire.memories_extracted,\r\n factsExtracted: wire.facts_extracted,\r\n conversationId: wire.conversation_id,\r\n newMemories: wire.new_memories ?? [],\r\n timestamp: wire.timestamp,\r\n };\r\n}\r\n\r\n// ─── Character Actions ───────────────────────────────────────────\r\n\r\nexport interface CharacterAction {\r\n /** Action name (e.g., \"follow_user\", \"sit\", \"look_at\") */\r\n name: string;\r\n /** Action parameters as key-value pairs */\r\n params: Record<string, string>;\r\n /** Message ID of the bot response that contained this action */\r\n messageId: string;\r\n}\r\n\r\n// ─── Event Map ───────────────────────────────────────────────────\r\n\r\nexport type EstuaryEventMap = {\r\n connected: (session: SessionInfo) => void;\r\n disconnected: (reason: string) => void;\r\n reconnecting: (attempt: number) => void;\r\n connectionStateChanged: (state: ConnectionState) => void;\r\n botResponse: (response: BotResponse) => void;\r\n botVoice: (voice: BotVoice) => void;\r\n sttResponse: (response: SttResponse) => void;\r\n interrupt: (data: InterruptData) => void;\r\n error: (error: Error) => void;\r\n authError: (error: string) => void;\r\n quotaExceeded: (data: QuotaExceededData) => void;\r\n cameraCaptureRequest: (request: CameraCaptureRequest) => void;\r\n characterAction: (action: CharacterAction) => void;\r\n voiceStarted: () => void;\r\n voiceStopped: () => void;\r\n livekitConnected: (room: string) => void;\r\n livekitDisconnected: () => void;\r\n audioPlaybackStarted: (messageId: string) => void;\r\n audioPlaybackComplete: (messageId: string) => void;\r\n memoryUpdated: (event: MemoryUpdatedEvent) => void;\r\n}\r\n\r\n// ─── Voice Manager Interface ─────────────────────────────────────\r\n\r\nexport interface VoiceManager {\r\n start(): Promise<void>;\r\n stop(): Promise<void>;\r\n toggleMute(): void;\r\n /** Suppress audio sending (software AEC). No-op if not supported. */\r\n setSuppressed?(suppressed: boolean): void;\r\n readonly isMuted: boolean;\r\n readonly isActive: boolean;\r\n dispose(): void;\r\n}\r\n\r\n// ─── Memory Types ────────────────────────────────────────────────\r\n\r\nexport interface MemoryListOptions {\r\n memoryType?: string;\r\n status?: string;\r\n limit?: number;\r\n offset?: number;\r\n sortBy?: 'created_at' | 'confidence' | 'last_accessed_at';\r\n sortOrder?: 'asc' | 'desc';\r\n}\r\n\r\nexport interface MemoryTimelineOptions {\r\n startDate?: string;\r\n endDate?: string;\r\n groupBy?: 'day' | 'week' | 'month';\r\n}\r\n\r\nexport interface MemoryGraphOptions {\r\n includeEntities?: boolean;\r\n includeCharacterMemories?: boolean;\r\n}\r\n\r\nexport interface MemorySearchOptions {\r\n query: string;\r\n limit?: number;\r\n}\r\n\r\nexport interface MemoryListResponse {\r\n memories: Record<string, unknown>[];\r\n total: number;\r\n limit: number;\r\n offset: number;\r\n}\r\n\r\nexport interface MemoryTimelineResponse {\r\n timeline: { date: string; memories: Record<string, unknown>[] }[];\r\n totalMemories: number;\r\n groupBy: string;\r\n}\r\n\r\nexport interface MemoryStatsResponse {\r\n [key: string]: unknown;\r\n}\r\n\r\nexport interface MemoryGraphResponse {\r\n nodes: Record<string, unknown>[];\r\n edges: Record<string, unknown>[];\r\n stats: Record<string, unknown>;\r\n}\r\n\r\nexport interface MemorySearchResponse {\r\n results: { memory: Record<string, unknown>; score: number; similarityScore: number }[];\r\n query: string;\r\n total: number;\r\n}\r\n\r\nexport interface CoreFactsResponse {\r\n coreFacts: Record<string, unknown>[];\r\n}\r\n","import { io, Socket } from 'socket.io-client';\r\nimport { TypedEventEmitter } from '../utils/event-emitter';\r\nimport { Logger } from '../utils/logger';\r\nimport { EstuaryError, ErrorCode } from '../errors';\r\nimport {\r\n EstuaryConfig,\r\n EstuaryEventMap,\r\n ConnectionState,\r\n WireSessionInfo,\r\n WireBotResponse,\r\n WireBotVoice,\r\n WireSttResponse,\r\n WireInterruptData,\r\n WireQuotaExceededData,\r\n WireCameraCaptureRequest,\r\n WireLiveKitTokenResponse,\r\n WireMemoryUpdated,\r\n SessionInfo,\r\n LiveKitTokenResponse,\r\n toSessionInfo,\r\n toBotResponse,\r\n toBotVoice,\r\n toSttResponse,\r\n toInterruptData,\r\n toQuotaExceededData,\r\n toCameraCaptureRequest,\r\n toLiveKitTokenResponse,\r\n toMemoryUpdatedEvent,\r\n} from '../types';\r\n\r\nexport class SocketManager extends TypedEventEmitter<EstuaryEventMap> {\r\n private socket: Socket | null = null;\r\n private config: EstuaryConfig;\r\n private logger: Logger;\r\n private connectionState: ConnectionState = ConnectionState.Disconnected;\r\n private reconnectAttempt = 0;\r\n private reconnectTimer: ReturnType<typeof setTimeout> | null = null;\r\n private sessionInfo: SessionInfo | null = null;\r\n\r\n constructor(config: EstuaryConfig, logger: Logger) {\r\n super();\r\n this.config = config;\r\n this.logger = logger;\r\n }\r\n\r\n get state(): ConnectionState {\r\n return this.connectionState;\r\n }\r\n\r\n get session(): SessionInfo | null {\r\n return this.sessionInfo;\r\n }\r\n\r\n get isConnected(): boolean {\r\n return this.connectionState === ConnectionState.Connected;\r\n }\r\n\r\n get rawSocket(): Socket | null {\r\n return this.socket;\r\n }\r\n\r\n connect(): Promise<SessionInfo> {\r\n return new Promise((resolve, reject) => {\r\n if (this.socket?.connected) {\r\n if (this.sessionInfo) {\r\n resolve(this.sessionInfo);\r\n return;\r\n }\r\n }\r\n\r\n this.setConnectionState(ConnectionState.Connecting);\r\n this.reconnectAttempt = 0;\r\n\r\n const url = `${this.config.serverUrl}/sdk`;\r\n this.logger.debug('Connecting to', url);\r\n\r\n this.socket = io(url, {\r\n transports: ['websocket'],\r\n timeout: 10000,\r\n reconnection: false, // We handle reconnection ourselves\r\n path: '/socket.io/',\r\n });\r\n\r\n let settled = false;\r\n\r\n const onConnect = () => {\r\n this.logger.debug('Socket connected, authenticating...');\r\n this.socket!.emit('authenticate', {\r\n api_key: this.config.apiKey,\r\n character_id: this.config.characterId,\r\n player_id: this.config.playerId,\r\n audio_sample_rate: this.config.audioSampleRate ?? 16000,\r\n realtime_memory: this.config.realtimeMemory ?? false,\r\n });\r\n };\r\n\r\n const onSessionInfo = (data: WireSessionInfo) => {\r\n this.sessionInfo = toSessionInfo(data);\r\n this.setConnectionState(ConnectionState.Connected);\r\n this.reconnectAttempt = 0;\r\n this.logger.info('Connected, session:', this.sessionInfo.sessionId);\r\n this.emit('connected', this.sessionInfo);\r\n if (!settled) {\r\n settled = true;\r\n resolve(this.sessionInfo);\r\n }\r\n };\r\n\r\n const onAuthError = (data: { error: string }) => {\r\n this.logger.error('Auth error:', data.error);\r\n this.setConnectionState(ConnectionState.Error);\r\n this.emit('authError', data.error);\r\n if (!settled) {\r\n settled = true;\r\n reject(new EstuaryError(ErrorCode.AUTH_FAILED, data.error));\r\n }\r\n };\r\n\r\n const onConnectError = (err: Error) => {\r\n this.logger.error('Connection error:', err.message);\r\n if (!settled) {\r\n settled = true;\r\n this.setConnectionState(ConnectionState.Error);\r\n reject(new EstuaryError(ErrorCode.CONNECTION_FAILED, err.message));\r\n } else {\r\n this.handleDisconnect('connect_error');\r\n }\r\n };\r\n\r\n const onDisconnect = (reason: string) => {\r\n this.logger.info('Disconnected:', reason);\r\n this.sessionInfo = null;\r\n if (!settled) {\r\n settled = true;\r\n this.setConnectionState(ConnectionState.Disconnected);\r\n reject(new EstuaryError(ErrorCode.CONNECTION_FAILED, `Disconnected: ${reason}`));\r\n } else {\r\n this.handleDisconnect(reason);\r\n }\r\n };\r\n\r\n this.socket.on('connect', onConnect);\r\n this.socket.on('session_info', onSessionInfo);\r\n this.socket.on('auth_error', onAuthError);\r\n this.socket.on('connect_error', onConnectError);\r\n this.socket.on('disconnect', onDisconnect);\r\n\r\n // Wire up all server → client events\r\n this.bindServerEvents();\r\n });\r\n }\r\n\r\n disconnect(): void {\r\n this.clearReconnectTimer();\r\n this.sessionInfo = null;\r\n if (this.socket) {\r\n this.socket.removeAllListeners();\r\n this.socket.disconnect();\r\n this.socket = null;\r\n }\r\n this.setConnectionState(ConnectionState.Disconnected);\r\n this.emit('disconnected', 'manual');\r\n }\r\n\r\n emitEvent(event: string, data?: unknown): void {\r\n if (!this.socket?.connected) {\r\n throw new EstuaryError(ErrorCode.NOT_CONNECTED, 'Not connected to server');\r\n }\r\n this.socket.emit(event, data);\r\n }\r\n\r\n private bindServerEvents(): void {\r\n if (!this.socket) return;\r\n\r\n this.socket.on('bot_response', (data: WireBotResponse) => {\r\n this.emit('botResponse', toBotResponse(data));\r\n });\r\n\r\n this.socket.on('bot_voice', (data: WireBotVoice) => {\r\n this.emit('botVoice', toBotVoice(data));\r\n });\r\n\r\n this.socket.on('stt_response', (data: WireSttResponse) => {\r\n this.emit('sttResponse', toSttResponse(data));\r\n });\r\n\r\n this.socket.on('interrupt', (data: WireInterruptData) => {\r\n this.emit('interrupt', toInterruptData(data));\r\n });\r\n\r\n this.socket.on('quota_exceeded', (data: WireQuotaExceededData) => {\r\n this.emit('quotaExceeded', toQuotaExceededData(data));\r\n });\r\n\r\n this.socket.on('camera_capture', (data: WireCameraCaptureRequest) => {\r\n this.emit('cameraCaptureRequest', toCameraCaptureRequest(data));\r\n });\r\n\r\n this.socket.on('error', (data: { message: string }) => {\r\n this.emit('error', new EstuaryError(ErrorCode.UNKNOWN, data.message));\r\n });\r\n\r\n this.socket.on('livekit_token', (data: WireLiveKitTokenResponse) => {\r\n // This is handled by LiveKitVoiceManager via a separate listener\r\n // We store it here so the voice manager can access it\r\n (this as unknown as { _livekitTokenCallback?: (d: LiveKitTokenResponse) => void })\r\n ._livekitTokenCallback?.(toLiveKitTokenResponse(data));\r\n });\r\n\r\n this.socket.on('livekit_ready', (data: { room: string }) => {\r\n this.emit('livekitConnected', data.room);\r\n });\r\n\r\n this.socket.on('memory_updated', (data: WireMemoryUpdated) => {\r\n this.emit('memoryUpdated', toMemoryUpdatedEvent(data));\r\n });\r\n }\r\n\r\n /** Register a callback for livekit_token events (used by LiveKitVoiceManager) */\r\n onLiveKitToken(callback: (data: LiveKitTokenResponse) => void): void {\r\n (this as unknown as { _livekitTokenCallback?: (d: LiveKitTokenResponse) => void })\r\n ._livekitTokenCallback = callback;\r\n }\r\n\r\n private handleDisconnect(reason: string): void {\r\n this.sessionInfo = null;\r\n this.setConnectionState(ConnectionState.Disconnected);\r\n this.emit('disconnected', reason);\r\n\r\n const autoReconnect = this.config.autoReconnect ?? true;\r\n const maxAttempts = this.config.maxReconnectAttempts ?? 5;\r\n\r\n if (autoReconnect && this.reconnectAttempt < maxAttempts) {\r\n this.attemptReconnect();\r\n }\r\n }\r\n\r\n private attemptReconnect(): void {\r\n const maxAttempts = this.config.maxReconnectAttempts ?? 5;\r\n const delay = this.config.reconnectDelayMs ?? 2000;\r\n\r\n if (this.reconnectAttempt >= maxAttempts) {\r\n this.logger.warn('Max reconnect attempts reached');\r\n this.setConnectionState(ConnectionState.Error);\r\n this.emit('error', new EstuaryError(\r\n ErrorCode.CONNECTION_FAILED,\r\n `Failed to reconnect after ${maxAttempts} attempts`,\r\n ));\r\n return;\r\n }\r\n\r\n this.reconnectAttempt++;\r\n this.setConnectionState(ConnectionState.Reconnecting);\r\n this.emit('reconnecting', this.reconnectAttempt);\r\n this.logger.info(`Reconnecting (attempt ${this.reconnectAttempt}/${maxAttempts})...`);\r\n\r\n this.clearReconnectTimer();\r\n this.reconnectTimer = setTimeout(() => {\r\n this.connect().catch((err) => {\r\n this.logger.error('Reconnect failed:', err.message);\r\n // Will trigger another attempt via handleDisconnect\r\n });\r\n }, delay * this.reconnectAttempt); // Exponential-ish backoff\r\n }\r\n\r\n private clearReconnectTimer(): void {\r\n if (this.reconnectTimer) {\r\n clearTimeout(this.reconnectTimer);\r\n this.reconnectTimer = null;\r\n }\r\n }\r\n\r\n private setConnectionState(state: ConnectionState): void {\r\n if (this.connectionState !== state) {\r\n this.connectionState = state;\r\n this.emit('connectionStateChanged', state);\r\n }\r\n }\r\n}\r\n","import type { VoiceManager, VoiceTransport } from '../types';\r\nimport type { SocketManager } from '../connection/socket-manager';\r\nimport type { Logger } from '../utils/logger';\r\n\r\nexport type { VoiceManager };\r\n\r\nasync function isLiveKitAvailable(): Promise<boolean> {\r\n try {\r\n await import('livekit-client');\r\n return true;\r\n } catch {\r\n return false;\r\n }\r\n}\r\n\r\nexport async function createVoiceManager(\r\n transport: VoiceTransport,\r\n socketManager: SocketManager,\r\n sampleRate: number,\r\n logger: Logger,\r\n): Promise<VoiceManager | null> {\r\n if (transport === 'websocket') {\r\n const { WebSocketVoiceManager } = await import('./websocket-voice');\r\n return new WebSocketVoiceManager(socketManager, sampleRate, logger);\r\n }\r\n\r\n if (transport === 'livekit') {\r\n if (await isLiveKitAvailable()) {\r\n const { LiveKitVoiceManager } = await import('./livekit-voice');\r\n return new LiveKitVoiceManager(socketManager, logger);\r\n }\r\n logger.warn('livekit-client not installed, falling back to WebSocket voice');\r\n const { WebSocketVoiceManager } = await import('./websocket-voice');\r\n return new WebSocketVoiceManager(socketManager, sampleRate, logger);\r\n }\r\n\r\n // auto: prefer LiveKit if available, else WebSocket\r\n if (transport === 'auto') {\r\n if (await isLiveKitAvailable()) {\r\n const { LiveKitVoiceManager } = await import('./livekit-voice');\r\n return new LiveKitVoiceManager(socketManager, logger);\r\n }\r\n const { WebSocketVoiceManager } = await import('./websocket-voice');\r\n return new WebSocketVoiceManager(socketManager, sampleRate, logger);\r\n }\r\n\r\n return null;\r\n}\r\n","import { EstuaryError, ErrorCode } from '../errors';\r\n\r\nexport class RestClient {\r\n private baseUrl: string;\r\n private apiKey: string;\r\n\r\n constructor(baseUrl: string, apiKey: string) {\r\n this.baseUrl = baseUrl.replace(/\\/+$/, '');\r\n this.apiKey = apiKey;\r\n }\r\n\r\n async get<T>(path: string, params?: Record<string, string | number | boolean | undefined>): Promise<T> {\r\n const url = this.buildUrl(path, params);\r\n return this.request<T>(url, { method: 'GET' });\r\n }\r\n\r\n async post<T>(path: string, body?: unknown): Promise<T> {\r\n const url = this.buildUrl(path);\r\n const init: RequestInit = { method: 'POST' };\r\n if (body !== undefined) {\r\n init.headers = { 'Content-Type': 'application/json' };\r\n init.body = JSON.stringify(body);\r\n }\r\n return this.request<T>(url, init);\r\n }\r\n\r\n async delete<T>(path: string, params?: Record<string, string | number | boolean | undefined>): Promise<T> {\r\n const url = this.buildUrl(path, params);\r\n return this.request<T>(url, { method: 'DELETE' });\r\n }\r\n\r\n dispose(): void {\r\n // No persistent resources to clean up\r\n }\r\n\r\n private buildUrl(path: string, params?: Record<string, string | number | boolean | undefined>): string {\r\n const url = new URL(path, this.baseUrl);\r\n if (params) {\r\n for (const [key, value] of Object.entries(params)) {\r\n if (value !== undefined) {\r\n url.searchParams.set(key, String(value));\r\n }\r\n }\r\n }\r\n return url.toString();\r\n }\r\n\r\n private async request<T>(url: string, init: RequestInit): Promise<T> {\r\n const headers = new Headers(init.headers);\r\n headers.set('X-API-Key', this.apiKey);\r\n\r\n const response = await fetch(url, { ...init, headers });\r\n\r\n if (!response.ok) {\r\n let detail: unknown;\r\n try {\r\n detail = await response.json();\r\n } catch {\r\n detail = await response.text().catch(() => null);\r\n }\r\n throw new EstuaryError(\r\n ErrorCode.REST_ERROR,\r\n `HTTP ${response.status}: ${response.statusText}`,\r\n detail,\r\n );\r\n }\r\n\r\n return response.json() as Promise<T>;\r\n }\r\n}\r\n","import type {\r\n MemoryListOptions,\r\n MemoryListResponse,\r\n MemoryTimelineOptions,\r\n MemoryTimelineResponse,\r\n MemoryStatsResponse,\r\n CoreFactsResponse,\r\n MemoryGraphOptions,\r\n MemoryGraphResponse,\r\n MemorySearchResponse,\r\n} from '../types';\r\nimport type { RestClient } from './rest-client';\r\n\r\nexport class MemoryClient {\r\n private rest: RestClient;\r\n private basePath: string;\r\n\r\n constructor(rest: RestClient, agentId: string, playerId: string) {\r\n this.rest = rest;\r\n this.basePath = `/api/agents/${agentId}/players/${playerId}/memories`;\r\n }\r\n\r\n async getMemories(options?: MemoryListOptions): Promise<MemoryListResponse> {\r\n return this.rest.get<MemoryListResponse>(this.basePath, options as Record<string, string | number | boolean | undefined>);\r\n }\r\n\r\n async getTimeline(options?: MemoryTimelineOptions): Promise<MemoryTimelineResponse> {\r\n return this.rest.get<MemoryTimelineResponse>(`${this.basePath}/timeline`, options as Record<string, string | number | boolean | undefined>);\r\n }\r\n\r\n async getStats(): Promise<MemoryStatsResponse> {\r\n return this.rest.get<MemoryStatsResponse>(`${this.basePath}/stats`);\r\n }\r\n\r\n async getCoreFacts(): Promise<CoreFactsResponse> {\r\n return this.rest.get<CoreFactsResponse>(`${this.basePath}/core-facts`);\r\n }\r\n\r\n async getGraph(options?: MemoryGraphOptions): Promise<MemoryGraphResponse> {\r\n const params: Record<string, string | number | boolean | undefined> = {};\r\n if (options) {\r\n if (options.includeEntities !== undefined) params.include_entities = options.includeEntities;\r\n if (options.includeCharacterMemories !== undefined) params.include_character_memories = options.includeCharacterMemories;\r\n }\r\n return this.rest.get<MemoryGraphResponse>(`${this.basePath}/graph`, params);\r\n }\r\n\r\n async search(query: string, limit?: number): Promise<MemorySearchResponse> {\r\n return this.rest.get<MemorySearchResponse>(`${this.basePath}/search`, { q: query, limit });\r\n }\r\n\r\n async deleteAll(confirm: boolean): Promise<{ message: string; deletedCount: number }> {\r\n return this.rest.delete<{ message: string; deletedCount: number }>(this.basePath, { confirm });\r\n }\r\n\r\n dispose(): void {\r\n // No persistent resources to clean up\r\n }\r\n}\r\n","import type { BotVoice } from '../types';\r\n\r\nexport type AudioPlaybackEvent =\r\n | { type: 'started'; messageId: string }\r\n | { type: 'complete'; messageId: string };\r\n\r\nexport class AudioPlayer {\r\n private sampleRate: number;\r\n private onEvent: (event: AudioPlaybackEvent) => void;\r\n private audioContext: AudioContext | null = null;\r\n private mediaStreamDest: MediaStreamAudioDestinationNode | null = null;\r\n private audioElement: HTMLAudioElement | null = null;\r\n private queue: { buffer: AudioBuffer; messageId: string }[] = [];\r\n private currentSource: AudioBufferSourceNode | null = null;\r\n private currentMessageId: string | null = null;\r\n private isPlaying = false;\r\n private _isCleared = false;\r\n private _interruptedMessageId: string | null = null;\r\n\r\n constructor(sampleRate: number, onEvent: (event: AudioPlaybackEvent) => void) {\r\n this.sampleRate = sampleRate;\r\n this.onEvent = onEvent;\r\n }\r\n\r\n /** Whether audio is currently playing */\r\n get playing(): boolean {\r\n return this.isPlaying;\r\n }\r\n\r\n /** The messageId of the currently playing audio, or null */\r\n get playingMessageId(): string | null {\r\n return this.currentMessageId;\r\n }\r\n\r\n /** Mark a messageId as interrupted so late-arriving chunks are dropped */\r\n setInterruptedMessageId(id: string | null): void {\r\n this._interruptedMessageId = id;\r\n }\r\n\r\n enqueue(voice: BotVoice): void {\r\n // Drop chunks belonging to the interrupted message\r\n if (voice.messageId === this._interruptedMessageId) return;\r\n // Clear the filter once a new message arrives\r\n if (this._interruptedMessageId && voice.messageId !== this._interruptedMessageId) {\r\n this._interruptedMessageId = null;\r\n }\r\n\r\n this._isCleared = false;\r\n\r\n const ctx = this.getAudioContext();\r\n if (!ctx) return;\r\n\r\n const pcm16 = base64ToInt16Array(voice.audio);\r\n const float32 = int16ToFloat32(pcm16);\r\n\r\n const buffer = ctx.createBuffer(1, float32.length, this.sampleRate);\r\n buffer.getChannelData(0).set(float32);\r\n\r\n this.queue.push({ buffer, messageId: voice.messageId });\r\n\r\n if (!this.isPlaying) {\r\n this.playNext();\r\n }\r\n }\r\n\r\n clear(): void {\r\n this._isCleared = true;\r\n this.queue.length = 0;\r\n if (this.currentSource) {\r\n try {\r\n this.currentSource.onended = null;\r\n this.currentSource.stop();\r\n this.currentSource.disconnect();\r\n } catch {\r\n // Already stopped\r\n }\r\n this.currentSource = null;\r\n }\r\n // Push silence through mediaStreamDest to flush the <audio> element's\r\n // internal pipeline buffer. Without this, mobile browsers replay the last\r\n // buffered frames. We can't touch the <audio> element directly (pause or\r\n // srcObject reassignment revokes the autoplay privilege on mobile).\r\n if (this.audioContext && this.mediaStreamDest) {\r\n const silence = this.audioContext.createBuffer(\r\n 1,\r\n this.sampleRate * 0.25,\r\n this.sampleRate,\r\n );\r\n const src = this.audioContext.createBufferSource();\r\n src.buffer = silence;\r\n src.connect(this.mediaStreamDest);\r\n src.start();\r\n }\r\n this.isPlaying = false;\r\n this.currentMessageId = null;\r\n }\r\n\r\n dispose(): void {\r\n this.clear();\r\n if (this.audioElement) {\r\n this.audioElement.pause();\r\n this.audioElement.srcObject = null;\r\n this.audioElement.remove();\r\n this.audioElement = null;\r\n }\r\n if (this.mediaStreamDest) {\r\n this.mediaStreamDest.disconnect();\r\n this.mediaStreamDest = null;\r\n }\r\n if (this.audioContext) {\r\n this.audioContext.close().catch(() => {});\r\n this.audioContext = null;\r\n }\r\n }\r\n\r\n private getAudioContext(): AudioContext | null {\r\n if (this.audioContext) return this.audioContext;\r\n\r\n if (typeof AudioContext === 'undefined' && typeof (globalThis as any).webkitAudioContext === 'undefined') {\r\n return null;\r\n }\r\n\r\n const AudioCtx = globalThis.AudioContext || (globalThis as any).webkitAudioContext;\r\n const ctx = new AudioCtx({ sampleRate: this.sampleRate });\r\n this.audioContext = ctx;\r\n\r\n // Route through a MediaStreamDestination → hidden <audio> element so the\r\n // browser's AEC pipeline can see our playback as the echo reference signal.\r\n if (typeof document !== 'undefined') {\r\n this.mediaStreamDest = ctx.createMediaStreamDestination();\r\n const el = document.createElement('audio');\r\n el.srcObject = this.mediaStreamDest.stream;\r\n el.autoplay = true;\r\n // Hidden but in the DOM so the browser treats it as a real media element\r\n el.style.display = 'none';\r\n document.body.appendChild(el);\r\n el.play().catch(() => {});\r\n this.audioElement = el;\r\n }\r\n\r\n return ctx;\r\n }\r\n\r\n private playNext(): void {\r\n if (this._isCleared) return;\r\n\r\n const ctx = this.getAudioContext();\r\n if (!ctx || this.queue.length === 0) {\r\n if (this.isPlaying && this.currentMessageId) {\r\n this.onEvent({ type: 'complete', messageId: this.currentMessageId });\r\n }\r\n this.isPlaying = false;\r\n this.currentMessageId = null;\r\n return;\r\n }\r\n\r\n const { buffer, messageId } = this.queue.shift()!;\r\n\r\n // Emit started if this is a new message\r\n if (messageId !== this.currentMessageId) {\r\n if (this.currentMessageId) {\r\n this.onEvent({ type: 'complete', messageId: this.currentMessageId });\r\n }\r\n this.currentMessageId = messageId;\r\n this.onEvent({ type: 'started', messageId });\r\n }\r\n\r\n this.isPlaying = true;\r\n const source = ctx.createBufferSource();\r\n source.buffer = buffer;\r\n source.connect(this.mediaStreamDest ?? ctx.destination);\r\n this.currentSource = source;\r\n\r\n source.onended = () => {\r\n if (this._isCleared) return;\r\n this.currentSource = null;\r\n this.playNext();\r\n };\r\n\r\n if (this.audioElement) {\r\n this.audioElement.play().catch(() => {});\r\n }\r\n ctx.resume().catch(() => {});\r\n source.start();\r\n }\r\n}\r\n\r\nfunction base64ToInt16Array(base64: string): Int16Array {\r\n let bytes: Uint8Array;\r\n if (typeof atob === 'function') {\r\n const binary = atob(base64);\r\n bytes = new Uint8Array(binary.length);\r\n for (let i = 0; i < binary.length; i++) {\r\n bytes[i] = binary.charCodeAt(i);\r\n }\r\n } else {\r\n // Node.js fallback\r\n bytes = new Uint8Array(Buffer.from(base64, 'base64'));\r\n }\r\n return new Int16Array(bytes.buffer, bytes.byteOffset, bytes.byteLength / 2);\r\n}\r\n\r\nfunction int16ToFloat32(int16: Int16Array): Float32Array {\r\n const float32 = new Float32Array(int16.length);\r\n for (let i = 0; i < int16.length; i++) {\r\n float32[i] = int16[i] / (int16[i] < 0 ? 0x8000 : 0x7fff);\r\n }\r\n return float32;\r\n}\r\n","export class Logger {\r\n private enabled: boolean;\r\n\r\n constructor(debug = false) {\r\n this.enabled = debug;\r\n }\r\n\r\n setDebug(enabled: boolean): void {\r\n this.enabled = enabled;\r\n }\r\n\r\n debug(...args: unknown[]): void {\r\n if (this.enabled) console.debug('[Estuary]', ...args);\r\n }\r\n\r\n info(...args: unknown[]): void {\r\n if (this.enabled) console.info('[Estuary]', ...args);\r\n }\r\n\r\n warn(...args: unknown[]): void {\r\n console.warn('[Estuary]', ...args);\r\n }\r\n\r\n error(...args: unknown[]): void {\r\n console.error('[Estuary]', ...args);\r\n }\r\n}\r\n","/**\r\n * Parses `<action name=\"...\" .../>` XML tags from bot response text.\r\n *\r\n * Designed for streaming: call `parse()` with the accumulated text on each\r\n * chunk and it returns only newly-discovered actions since the last call.\r\n */\r\n\r\nexport interface ParsedAction {\r\n name: string;\r\n params: Record<string, string>;\r\n}\r\n\r\n// Matches self-closing <action .../> tags (with optional whitespace before />)\r\nconst ACTION_TAG_RE = /<action\\s+([^>]*?)\\/>/gi;\r\n// Matches key=\"value\" or key='value' attribute pairs\r\nconst ATTR_RE = /(\\w+)\\s*=\\s*\"([^\"]*)\"|(\\w+)\\s*=\\s*'([^']*)'/g;\r\n\r\nfunction parseAttributes(attrString: string): Record<string, string> {\r\n const attrs: Record<string, string> = {};\r\n let match: RegExpExecArray | null;\r\n while ((match = ATTR_RE.exec(attrString)) !== null) {\r\n const key = match[1] ?? match[3];\r\n const value = match[2] ?? match[4];\r\n attrs[key] = value;\r\n }\r\n return attrs;\r\n}\r\n\r\nexport class StreamingActionParser {\r\n private emittedCount = 0;\r\n\r\n /**\r\n * Parse the accumulated response text and return any new actions found\r\n * since the last call. Also returns the text with all action tags stripped.\r\n */\r\n parse(accumulatedText: string): { actions: ParsedAction[]; cleanText: string } {\r\n const allActions: ParsedAction[] = [];\r\n let match: RegExpExecArray | null;\r\n\r\n ACTION_TAG_RE.lastIndex = 0;\r\n while ((match = ACTION_TAG_RE.exec(accumulatedText)) !== null) {\r\n const attrs = parseAttributes(match[1]);\r\n const name = attrs.name;\r\n if (name) {\r\n delete attrs.name;\r\n allActions.push({ name, params: attrs });\r\n }\r\n }\r\n\r\n const newActions = allActions.slice(this.emittedCount);\r\n this.emittedCount = allActions.length;\r\n\r\n const cleanText = accumulatedText.replace(ACTION_TAG_RE, '').replace(/\\s{2,}/g, ' ').trimEnd();\r\n\r\n return { actions: newActions, cleanText };\r\n }\r\n\r\n reset(): void {\r\n this.emittedCount = 0;\r\n }\r\n}\r\n\r\n/**\r\n * One-shot parse: extract all actions and return clean text.\r\n * Useful for non-streaming contexts.\r\n */\r\nexport function parseActions(text: string): { actions: ParsedAction[]; cleanText: string } {\r\n const parser = new StreamingActionParser();\r\n return parser.parse(text);\r\n}\r\n","import { SocketManager } from './connection/socket-manager';\r\nimport { createVoiceManager } from './voice/voice-manager';\r\nimport { RestClient } from './rest/rest-client';\r\nimport { MemoryClient } from './rest/memory-client';\r\nimport { AudioPlayer } from './audio/audio-player';\r\nimport { TypedEventEmitter } from './utils/event-emitter';\r\nimport { Logger } from './utils/logger';\r\nimport { EstuaryError, ErrorCode } from './errors';\r\nimport {\r\n EstuaryConfig,\r\n EstuaryEventMap,\r\n ConnectionState,\r\n SessionInfo,\r\n BotResponse,\r\n BotVoice,\r\n SttResponse,\r\n VoiceManager,\r\n} from './types';\r\nimport { StreamingActionParser } from './utils/action-parser';\r\n\r\nconst DEFAULT_SAMPLE_RATE = 16000;\r\n\r\nexport class EstuaryClient extends TypedEventEmitter<EstuaryEventMap> {\r\n private config: EstuaryConfig;\r\n private logger: Logger;\r\n private socketManager: SocketManager;\r\n private voiceManager: VoiceManager | null = null;\r\n private audioPlayer: AudioPlayer | null = null;\r\n private _memory: MemoryClient;\r\n private _sessionInfo: SessionInfo | null = null;\r\n private actionParsers = new Map<string, StreamingActionParser>();\r\n private _hasAutoInterrupted = false;\r\n private _autoInterruptGraceTimer: ReturnType<typeof setTimeout> | null = null;\r\n\r\n constructor(config: EstuaryConfig) {\r\n super();\r\n this.config = config;\r\n this.logger = new Logger(config.debug ?? false);\r\n this.socketManager = new SocketManager(config, this.logger);\r\n this.forwardSocketEvents();\r\n\r\n // Set up REST client for memory API\r\n const restClient = new RestClient(config.serverUrl, config.apiKey);\r\n this._memory = new MemoryClient(restClient, config.characterId, config.playerId);\r\n }\r\n\r\n /** Memory API client for querying memories, graphs, and facts */\r\n get memory(): MemoryClient {\r\n return this._memory;\r\n }\r\n\r\n /** Current session info (null if not connected) */\r\n get session(): SessionInfo | null {\r\n return this._sessionInfo;\r\n }\r\n\r\n /** Current connection state */\r\n get connectionState(): ConnectionState {\r\n return this.socketManager.state;\r\n }\r\n\r\n /** Whether the client is connected and authenticated */\r\n get isConnected(): boolean {\r\n return this.socketManager.isConnected;\r\n }\r\n\r\n /** Connect to the Estuary server and authenticate */\r\n async connect(): Promise<SessionInfo> {\r\n this.logger.info('Connecting...');\r\n const session = await this.socketManager.connect();\r\n this._sessionInfo = session;\r\n return session;\r\n }\r\n\r\n /** Disconnect from the server */\r\n disconnect(): void {\r\n this.logger.info('Disconnecting...');\r\n if (this._autoInterruptGraceTimer) {\r\n clearTimeout(this._autoInterruptGraceTimer);\r\n this._autoInterruptGraceTimer = null;\r\n }\r\n this.stopVoice();\r\n this.audioPlayer?.dispose();\r\n this.audioPlayer = null;\r\n this.socketManager.disconnect();\r\n this._sessionInfo = null;\r\n }\r\n\r\n /** Send a text message to the character */\r\n sendText(text: string, textOnly = false): void {\r\n this.ensureConnected();\r\n this.socketManager.emitEvent('text', { text, textOnly });\r\n }\r\n\r\n /** Interrupt the current bot response */\r\n interrupt(messageId?: string): void {\r\n this.ensureConnected();\r\n this.socketManager.emitEvent('client_interrupt', { message_id: messageId });\r\n this.audioPlayer?.setInterruptedMessageId(messageId ?? this.audioPlayer.playingMessageId);\r\n this.audioPlayer?.clear();\r\n this._hasAutoInterrupted = true;\r\n if (this._autoInterruptGraceTimer) {\r\n clearTimeout(this._autoInterruptGraceTimer);\r\n this._autoInterruptGraceTimer = null;\r\n }\r\n if (this.config.suppressMicDuringPlayback) {\r\n this.voiceManager?.setSuppressed?.(false);\r\n }\r\n }\r\n\r\n /** Send a camera image for vision processing */\r\n sendCameraImage(imageBase64: string, mimeType: string, requestId?: string, text?: string): void {\r\n this.ensureConnected();\r\n this.socketManager.emitEvent('camera_image', {\r\n image: imageBase64,\r\n mime_type: mimeType,\r\n request_id: requestId,\r\n text,\r\n });\r\n }\r\n\r\n /** Update session preferences */\r\n updatePreferences(preferences: { enableVisionAcknowledgment?: boolean }): void {\r\n this.ensureConnected();\r\n this.socketManager.emitEvent('update_preferences', preferences);\r\n }\r\n\r\n /** Notify server that audio playback completed for a message */\r\n notifyAudioPlaybackComplete(messageId?: string): void {\r\n this.ensureConnected();\r\n this.socketManager.emitEvent('audio_playback_complete', { message_id: messageId });\r\n }\r\n\r\n // ─── Voice ───────────────────────────────────────────────────\r\n\r\n /** Start voice input (requests microphone permission) */\r\n async startVoice(): Promise<void> {\r\n this.ensureConnected();\r\n\r\n if (this.voiceManager?.isActive) {\r\n throw new EstuaryError(ErrorCode.VOICE_ALREADY_ACTIVE, 'Voice is already active');\r\n }\r\n\r\n const transport = this.config.voiceTransport ?? 'auto';\r\n const sampleRate = this.config.audioSampleRate ?? DEFAULT_SAMPLE_RATE;\r\n\r\n this.voiceManager = await createVoiceManager(transport, this.socketManager, sampleRate, this.logger);\r\n if (!this.voiceManager) {\r\n throw new EstuaryError(ErrorCode.VOICE_NOT_SUPPORTED, 'No voice transport available');\r\n }\r\n\r\n // Set up audio player for bot voice responses (browser only)\r\n if (!this.audioPlayer && typeof AudioContext !== 'undefined') {\r\n this.audioPlayer = new AudioPlayer(sampleRate, (event) => {\r\n if (event.type === 'started') {\r\n // Suppress auto-interrupt during grace period so trailing STT partials\r\n // from the user's previous speech don't kill the new audio.\r\n this._hasAutoInterrupted = true;\r\n if (this._autoInterruptGraceTimer) clearTimeout(this._autoInterruptGraceTimer);\r\n this._autoInterruptGraceTimer = setTimeout(() => {\r\n this._hasAutoInterrupted = false;\r\n this._autoInterruptGraceTimer = null;\r\n }, 1500);\r\n this.emit('audioPlaybackStarted', event.messageId);\r\n if (this.config.suppressMicDuringPlayback) {\r\n this.voiceManager?.setSuppressed?.(true);\r\n }\r\n } else if (event.type === 'complete') {\r\n this.emit('audioPlaybackComplete', event.messageId);\r\n this.notifyAudioPlaybackComplete(event.messageId);\r\n if (this.config.suppressMicDuringPlayback) {\r\n this.voiceManager?.setSuppressed?.(false);\r\n }\r\n }\r\n });\r\n }\r\n\r\n await this.voiceManager.start();\r\n this.emit('voiceStarted');\r\n }\r\n\r\n /** Stop voice input */\r\n stopVoice(): void {\r\n if (this.voiceManager?.isActive) {\r\n this.voiceManager.stop();\r\n this.voiceManager.dispose();\r\n this.voiceManager = null;\r\n this.emit('voiceStopped');\r\n }\r\n }\r\n\r\n /** Toggle microphone mute */\r\n toggleMute(): void {\r\n if (!this.voiceManager?.isActive) {\r\n throw new EstuaryError(ErrorCode.VOICE_NOT_ACTIVE, 'Voice is not active');\r\n }\r\n this.voiceManager.toggleMute();\r\n }\r\n\r\n /** Whether the microphone is muted */\r\n get isMuted(): boolean {\r\n return this.voiceManager?.isMuted ?? false;\r\n }\r\n\r\n /** Whether voice is currently active */\r\n get isVoiceActive(): boolean {\r\n return this.voiceManager?.isActive ?? false;\r\n }\r\n\r\n // ─── Internal ────────────────────────────────────────────────\r\n\r\n private ensureConnected(): void {\r\n if (!this.socketManager.isConnected) {\r\n throw new EstuaryError(ErrorCode.NOT_CONNECTED, 'Not connected to server. Call connect() first.');\r\n }\r\n }\r\n\r\n private forwardSocketEvents(): void {\r\n // Forward all socket manager events to this client's event emitter\r\n this.socketManager.on('connected', (session) => {\r\n this._sessionInfo = session;\r\n this.emit('connected', session);\r\n });\r\n this.socketManager.on('disconnected', (reason) => {\r\n this._sessionInfo = null;\r\n this.actionParsers.clear();\r\n this.emit('disconnected', reason);\r\n });\r\n this.socketManager.on('reconnecting', (attempt) => this.emit('reconnecting', attempt));\r\n this.socketManager.on('connectionStateChanged', (state) => this.emit('connectionStateChanged', state));\r\n this.socketManager.on('botResponse', (response) => this.handleBotResponse(response));\r\n this.socketManager.on('botVoice', (voice) => this.handleBotVoice(voice));\r\n this.socketManager.on('sttResponse', (response) => {\r\n this.maybeAutoInterrupt(response);\r\n this.emit('sttResponse', response);\r\n });\r\n this.socketManager.on('interrupt', (data) => {\r\n this.audioPlayer?.setInterruptedMessageId(data.messageId ?? null);\r\n this.audioPlayer?.clear();\r\n this.actionParsers.clear();\r\n if (this.config.suppressMicDuringPlayback) {\r\n this.voiceManager?.setSuppressed?.(false);\r\n }\r\n this.emit('interrupt', data);\r\n });\r\n this.socketManager.on('error', (error) => this.emit('error', error));\r\n this.socketManager.on('authError', (error) => this.emit('authError', error));\r\n this.socketManager.on('quotaExceeded', (data) => this.emit('quotaExceeded', data));\r\n this.socketManager.on('cameraCaptureRequest', (request) => this.emit('cameraCaptureRequest', request));\r\n this.socketManager.on('livekitConnected', (room) => this.emit('livekitConnected', room));\r\n this.socketManager.on('livekitDisconnected', () => this.emit('livekitDisconnected'));\r\n this.socketManager.on('memoryUpdated', (event) => this.emit('memoryUpdated', event));\r\n }\r\n\r\n private handleBotResponse(response: BotResponse): void {\r\n const { messageId } = response;\r\n\r\n // Get or create a parser for this message stream\r\n if (!this.actionParsers.has(messageId)) {\r\n this.actionParsers.set(messageId, new StreamingActionParser());\r\n }\r\n const parser = this.actionParsers.get(messageId)!;\r\n\r\n // Parse actions from accumulated text\r\n const { actions, cleanText } = parser.parse(response.text);\r\n\r\n // Emit characterAction events for newly discovered actions\r\n for (const action of actions) {\r\n this.emit('characterAction', {\r\n name: action.name,\r\n params: action.params,\r\n messageId,\r\n });\r\n }\r\n\r\n // Forward botResponse with cleaned text\r\n this.emit('botResponse', {\r\n ...response,\r\n text: cleanText,\r\n });\r\n\r\n // Clean up parser when message is final\r\n if (response.isFinal) {\r\n this.actionParsers.delete(messageId);\r\n }\r\n }\r\n\r\n private handleBotVoice(voice: BotVoice): void {\r\n this.emit('botVoice', voice);\r\n // Enqueue audio for playback if we have an audio player\r\n this.audioPlayer?.enqueue(voice);\r\n }\r\n\r\n private maybeAutoInterrupt(stt: SttResponse): void {\r\n if ((this.config.autoInterruptOnSpeech ?? true) === false) return;\r\n if (this.config.suppressMicDuringPlayback) return;\r\n if (stt.isFinal) return;\r\n if (!this.audioPlayer?.playing) return;\r\n if (this._hasAutoInterrupted) return;\r\n\r\n this._hasAutoInterrupted = true;\r\n this.interrupt();\r\n }\r\n}\r\n"]}
|
|
1
|
+
{"version":3,"sources":["../src/utils/event-emitter.ts","../src/types.ts","../src/connection/socket-manager.ts","../src/voice/voice-manager.ts","../src/rest/rest-client.ts","../src/rest/memory-client.ts","../src/rest/character-client.ts","../src/audio/audio-player.ts","../src/utils/logger.ts","../src/utils/action-parser.ts","../src/client.ts"],"names":["ConnectionState"],"mappings":"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","file":"index.mjs","sourcesContent":["export class TypedEventEmitter<T extends Record<string, unknown>> {\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n private listeners = new Map<keyof T, Set<any>>();\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n private onceListeners = new Map<keyof T, Set<any>>();\n\n on<K extends keyof T>(event: K, listener: T[K]): this {\n if (!this.listeners.has(event)) {\n this.listeners.set(event, new Set());\n }\n this.listeners.get(event)!.add(listener);\n return this;\n }\n\n off<K extends keyof T>(event: K, listener: T[K]): this {\n this.listeners.get(event)?.delete(listener);\n this.onceListeners.get(event)?.delete(listener);\n return this;\n }\n\n once<K extends keyof T>(event: K, listener: T[K]): this {\n if (!this.onceListeners.has(event)) {\n this.onceListeners.set(event, new Set());\n }\n this.onceListeners.get(event)!.add(listener);\n this.on(event, listener);\n return this;\n }\n\n protected emit<K extends keyof T>(\n event: K,\n ...args: T[K] extends (...a: infer A) => void ? A : never[]\n ): boolean {\n const listeners = this.listeners.get(event);\n if (!listeners || listeners.size === 0) return false;\n\n for (const listener of listeners) {\n try {\n (listener as Function)(...args);\n } catch {\n // Listener threw — swallow to not break other listeners\n }\n }\n\n // Remove once listeners after firing\n const onceSet = this.onceListeners.get(event);\n if (onceSet) {\n for (const listener of onceSet) {\n this.listeners.get(event)?.delete(listener);\n }\n onceSet.clear();\n }\n\n return true;\n }\n\n removeAllListeners<K extends keyof T>(event?: K): this {\n if (event) {\n this.listeners.delete(event);\n this.onceListeners.delete(event);\n } else {\n this.listeners.clear();\n this.onceListeners.clear();\n }\n return this;\n }\n\n listenerCount<K extends keyof T>(event: K): number {\n return this.listeners.get(event)?.size ?? 0;\n }\n}\n","// ─── Configuration ───────────────────────────────────────────────\n\nexport interface EstuaryConfig {\n /** Base URL of the Estuary server (e.g., \"https://api.estuary-ai.com\") */\n serverUrl: string;\n /** API key (starts with \"est_\") */\n apiKey: string;\n /** Character (agent) ID */\n characterId: string;\n /** Unique identifier for the end user */\n playerId: string;\n /** Audio sample rate in Hz (default: 16000) */\n audioSampleRate?: number;\n /** Auto-reconnect on disconnect (default: true) */\n autoReconnect?: boolean;\n /** Max reconnect attempts (default: 5) */\n maxReconnectAttempts?: number;\n /** Base delay between reconnect attempts in ms (default: 2000). Actual delay is baseDelay × attemptNumber (linear backoff). */\n reconnectDelayMs?: number;\n /** Enable debug logging (default: false) */\n debug?: boolean;\n /** Voice transport: 'websocket' | 'livekit' | 'auto' (default: 'auto') */\n voiceTransport?: VoiceTransport;\n /** Enable real-time memory extraction after each response (default: false) */\n realtimeMemory?: boolean;\n /** Suppress mic during TTS playback (software AEC fallback, disables barge-in). Default: false */\n suppressMicDuringPlayback?: boolean;\n /** Proactively interrupt bot audio when user starts speaking (default: true) */\n autoInterruptOnSpeech?: boolean;\n}\n\nexport type VoiceTransport = 'websocket' | 'livekit' | 'auto';\n\n// ─── Connection State ────────────────────────────────────────────\n\nexport enum ConnectionState {\n Disconnected = 'disconnected',\n Connecting = 'connecting',\n Connected = 'connected',\n Reconnecting = 'reconnecting',\n Error = 'error',\n}\n\n// ─── Wire Format Types (snake_case from server) ─────────────────\n\n/** @internal */\nexport interface WireSessionInfo {\n session_id: string;\n conversation_id: string;\n character_id: string;\n player_id: string;\n}\n\n/** @internal */\nexport interface WireBotResponse {\n text: string;\n is_final: boolean;\n partial: string;\n message_id: string;\n chunk_index: number;\n is_interjection: boolean;\n token_stream?: boolean;\n}\n\n/** @internal */\nexport interface WireBotVoice {\n audio: string;\n message_id: string;\n chunk_index: number;\n is_final: boolean;\n}\n\n/** @internal */\nexport interface WireSttResponse {\n text: string;\n is_final: boolean;\n}\n\n/** @internal */\nexport interface WireInterruptData {\n message_id?: string;\n reason?: string;\n interrupted_at?: string;\n}\n\n/** @internal */\nexport interface WireQuotaExceededData {\n message: string;\n current: number;\n limit: number;\n remaining: number;\n tier: string;\n}\n\n/** @internal */\nexport interface WireLiveKitTokenResponse {\n token: string;\n url: string;\n room: string;\n}\n\n/** @internal */\nexport interface WireCameraCaptureRequest {\n request_id: string;\n text?: string;\n}\n\n/** @internal — new_memories items are already camelCase from Memory.to_dict() */\nexport interface WireMemoryUpdated {\n agent_id: string;\n player_id: string;\n memories_extracted: number;\n facts_extracted: number;\n conversation_id: string;\n new_memories: MemoryData[];\n timestamp: string;\n}\n\n// ─── Public Types (camelCase) ────────────────────────────────────\n\nexport interface SessionInfo {\n sessionId: string;\n conversationId: string;\n characterId: string;\n playerId: string;\n}\n\nexport interface BotResponse {\n text: string;\n isFinal: boolean;\n partial: string;\n messageId: string;\n chunkIndex: number;\n isInterjection: boolean;\n tokenStream?: boolean;\n}\n\nexport interface BotVoice {\n audio: string;\n messageId: string;\n chunkIndex: number;\n isFinal: boolean;\n}\n\nexport interface SttResponse {\n text: string;\n isFinal: boolean;\n}\n\nexport interface InterruptData {\n messageId?: string;\n reason?: string;\n interruptedAt?: string;\n}\n\nexport interface QuotaExceededData {\n message: string;\n current: number;\n limit: number;\n remaining: number;\n tier: string;\n}\n\nexport interface LiveKitTokenResponse {\n token: string;\n url: string;\n room: string;\n}\n\nexport interface CameraCaptureRequest {\n requestId: string;\n text?: string;\n}\n\nexport interface MemoryData {\n id: string;\n userId: string;\n agentId: string;\n playerId: string;\n content: string;\n memoryType: string;\n confidence: number;\n status: string;\n sourceConversationId: string;\n sourceQuote?: string;\n source?: string;\n topic?: string;\n secondaryTopics?: string[];\n lastAccessedAt?: string | null;\n accessCount?: number;\n extractedAt?: string | null;\n createdAt?: string | null;\n updatedAt?: string | null;\n}\n\nexport interface MemoryUpdatedEvent {\n agentId: string;\n playerId: string;\n memoriesExtracted: number;\n factsExtracted: number;\n conversationId: string;\n newMemories: MemoryData[];\n timestamp: string;\n}\n\n// ─── Wire → Public Converters ────────────────────────────────────\n\n/** @internal */\nexport function toSessionInfo(wire: WireSessionInfo): SessionInfo {\n return {\n sessionId: wire.session_id,\n conversationId: wire.conversation_id,\n characterId: wire.character_id,\n playerId: wire.player_id,\n };\n}\n\n/** @internal */\nexport function toBotResponse(wire: WireBotResponse): BotResponse {\n return {\n text: wire.text,\n isFinal: wire.is_final,\n partial: wire.partial,\n messageId: wire.message_id,\n chunkIndex: wire.chunk_index,\n isInterjection: wire.is_interjection,\n tokenStream: wire.token_stream,\n };\n}\n\n/** @internal */\nexport function toBotVoice(wire: WireBotVoice): BotVoice {\n return {\n audio: wire.audio,\n messageId: wire.message_id,\n chunkIndex: wire.chunk_index,\n isFinal: wire.is_final,\n };\n}\n\n/** @internal */\nexport function toSttResponse(wire: WireSttResponse): SttResponse {\n return {\n text: wire.text,\n isFinal: wire.is_final,\n };\n}\n\n/** @internal */\nexport function toInterruptData(wire: WireInterruptData): InterruptData {\n return {\n messageId: wire.message_id,\n reason: wire.reason,\n interruptedAt: wire.interrupted_at,\n };\n}\n\n/** @internal */\nexport function toQuotaExceededData(wire: WireQuotaExceededData): QuotaExceededData {\n return {\n message: wire.message,\n current: wire.current,\n limit: wire.limit,\n remaining: wire.remaining,\n tier: wire.tier,\n };\n}\n\n/** @internal */\nexport function toLiveKitTokenResponse(wire: WireLiveKitTokenResponse): LiveKitTokenResponse {\n return {\n token: wire.token,\n url: wire.url,\n room: wire.room,\n };\n}\n\n/** @internal */\nexport function toCameraCaptureRequest(wire: WireCameraCaptureRequest): CameraCaptureRequest {\n return {\n requestId: wire.request_id,\n text: wire.text,\n };\n}\n\n/** @internal */\nexport function toMemoryUpdatedEvent(wire: WireMemoryUpdated): MemoryUpdatedEvent {\n return {\n agentId: wire.agent_id,\n playerId: wire.player_id,\n memoriesExtracted: wire.memories_extracted,\n factsExtracted: wire.facts_extracted,\n conversationId: wire.conversation_id,\n newMemories: wire.new_memories ?? [],\n timestamp: wire.timestamp,\n };\n}\n\n// ─── Character Info ──────────────────────────────────────────────\n\nexport interface CharacterInfo {\n id: string;\n name: string;\n tagline: string | null;\n avatar: string | null;\n modelUrl: string | null;\n modelPreviewUrl: string | null;\n modelStatus: string | null;\n sourceImageUrl: string | null;\n}\n\n// ─── Character Actions ───────────────────────────────────────────\n\nexport interface CharacterAction {\n /** Action name (e.g., \"follow_user\", \"sit\", \"look_at\") */\n name: string;\n /** Action parameters as key-value pairs */\n params: Record<string, string>;\n /** Message ID of the bot response that contained this action */\n messageId: string;\n}\n\n// ─── Event Map ───────────────────────────────────────────────────\n\nexport type EstuaryEventMap = {\n connected: (session: SessionInfo) => void;\n disconnected: (reason: string) => void;\n reconnecting: (attempt: number) => void;\n connectionStateChanged: (state: ConnectionState) => void;\n botResponse: (response: BotResponse) => void;\n botVoice: (voice: BotVoice) => void;\n sttResponse: (response: SttResponse) => void;\n interrupt: (data: InterruptData) => void;\n error: (error: Error) => void;\n authError: (error: string) => void;\n quotaExceeded: (data: QuotaExceededData) => void;\n cameraCaptureRequest: (request: CameraCaptureRequest) => void;\n characterAction: (action: CharacterAction) => void;\n voiceStarted: () => void;\n voiceStopped: () => void;\n livekitConnected: (room: string) => void;\n livekitDisconnected: () => void;\n audioPlaybackStarted: (messageId: string) => void;\n audioPlaybackComplete: (messageId: string) => void;\n memoryUpdated: (event: MemoryUpdatedEvent) => void;\n}\n\n// ─── Voice Manager Interface ─────────────────────────────────────\n\nexport interface VoiceManager {\n start(): Promise<void>;\n stop(): Promise<void>;\n toggleMute(): void;\n /** Suppress audio sending (software AEC). No-op if not supported. */\n setSuppressed?(suppressed: boolean): void;\n readonly isMuted: boolean;\n readonly isActive: boolean;\n dispose(): void;\n}\n\n// ─── Memory Types ────────────────────────────────────────────────\n\nexport interface MemoryListOptions {\n memoryType?: string;\n status?: string;\n limit?: number;\n offset?: number;\n sortBy?: 'created_at' | 'confidence' | 'last_accessed_at';\n sortOrder?: 'asc' | 'desc';\n}\n\nexport interface MemoryTimelineOptions {\n startDate?: string;\n endDate?: string;\n groupBy?: 'day' | 'week' | 'month';\n}\n\nexport interface MemoryGraphOptions {\n includeEntities?: boolean;\n includeCharacterMemories?: boolean;\n}\n\nexport interface MemorySearchOptions {\n query: string;\n limit?: number;\n}\n\nexport interface MemoryListResponse {\n memories: MemoryData[];\n total: number;\n limit: number;\n offset: number;\n}\n\nexport interface MemoryTimelineResponse {\n timeline: { date: string; memories: MemoryData[] }[];\n totalMemories: number;\n groupBy: string;\n}\n\nexport interface MemoryStatsResponse {\n totalActive: number;\n totalSuperseded: number;\n totalDecayed: number;\n byType: Record<string, number>;\n coreFacts: number;\n}\n\nexport interface MemoryGraphNode {\n id: string;\n type: 'user' | 'cluster' | 'memory' | 'entity';\n label?: string;\n /** Cluster fields */\n level?: number;\n memoryCount?: number;\n typeDistribution?: Record<string, number>;\n expanded?: boolean;\n parentClusterId?: string | null;\n childClusterIds?: string[];\n labelPending?: boolean;\n /** Memory fields */\n memoryType?: string;\n content?: string;\n confidence?: number;\n clusterId?: string;\n sourceQuote?: string | null;\n sourceConversationId?: string | null;\n createdAt?: string | null;\n accessCount?: number;\n /** Entity fields */\n entityType?: string | null;\n name?: string;\n mentionCount?: number;\n extraData?: Record<string, string>;\n}\n\nexport interface MemoryGraphEdge {\n source: string;\n target: string;\n type: 'has_cluster' | 'contains' | 'mentions' | 'relationship';\n relationshipType?: string;\n label?: string | null;\n confidence?: number;\n}\n\nexport interface MemoryGraphResponse {\n nodes: MemoryGraphNode[];\n edges: MemoryGraphEdge[];\n stats: {\n totalMemories: number;\n totalEntities: number;\n clusterCount: number;\n clusters: Record<string, number>;\n };\n stale?: boolean;\n}\n\nexport interface MemorySearchResponse {\n results: { memory: MemoryData; score: number; similarityScore: number }[];\n query: string;\n total: number;\n}\n\nexport interface CoreFact {\n id: string;\n userId: string;\n agentId: string;\n playerId: string;\n factKey: string;\n factValue: string;\n sourceMemoryId?: string | null;\n createdAt?: string | null;\n updatedAt?: string | null;\n}\n\nexport interface CoreFactsResponse {\n coreFacts: CoreFact[];\n}\n","import { io, Socket } from 'socket.io-client';\nimport { TypedEventEmitter } from '../utils/event-emitter';\nimport { Logger } from '../utils/logger';\nimport { EstuaryError, ErrorCode } from '../errors';\nimport {\n EstuaryConfig,\n EstuaryEventMap,\n ConnectionState,\n WireSessionInfo,\n WireBotResponse,\n WireBotVoice,\n WireSttResponse,\n WireInterruptData,\n WireQuotaExceededData,\n WireCameraCaptureRequest,\n WireLiveKitTokenResponse,\n WireMemoryUpdated,\n SessionInfo,\n LiveKitTokenResponse,\n toSessionInfo,\n toBotResponse,\n toBotVoice,\n toSttResponse,\n toInterruptData,\n toQuotaExceededData,\n toCameraCaptureRequest,\n toLiveKitTokenResponse,\n toMemoryUpdatedEvent,\n} from '../types';\n\nexport class SocketManager extends TypedEventEmitter<EstuaryEventMap> {\n private socket: Socket | null = null;\n private config: EstuaryConfig;\n private logger: Logger;\n private connectionState: ConnectionState = ConnectionState.Disconnected;\n private reconnectAttempt = 0;\n private reconnectTimer: ReturnType<typeof setTimeout> | null = null;\n private sessionInfo: SessionInfo | null = null;\n\n constructor(config: EstuaryConfig, logger: Logger) {\n super();\n this.config = config;\n this.logger = logger;\n }\n\n get state(): ConnectionState {\n return this.connectionState;\n }\n\n get session(): SessionInfo | null {\n return this.sessionInfo;\n }\n\n get isConnected(): boolean {\n return this.connectionState === ConnectionState.Connected;\n }\n\n get rawSocket(): Socket | null {\n return this.socket;\n }\n\n connect(): Promise<SessionInfo> {\n return new Promise((resolve, reject) => {\n if (this.socket?.connected) {\n if (this.sessionInfo) {\n resolve(this.sessionInfo);\n return;\n }\n }\n\n this.setConnectionState(ConnectionState.Connecting);\n this.reconnectAttempt = 0;\n\n const url = `${this.config.serverUrl}/sdk`;\n this.logger.debug('Connecting to', url);\n\n this.socket = io(url, {\n transports: ['websocket'],\n timeout: 10000,\n reconnection: false, // We handle reconnection ourselves\n path: '/socket.io/',\n });\n\n let settled = false;\n\n const onConnect = () => {\n this.logger.debug('Socket connected, authenticating...');\n this.socket!.emit('authenticate', {\n api_key: this.config.apiKey,\n character_id: this.config.characterId,\n player_id: this.config.playerId,\n audio_sample_rate: this.config.audioSampleRate ?? 16000,\n realtime_memory: this.config.realtimeMemory ?? false,\n });\n };\n\n const onSessionInfo = (data: WireSessionInfo) => {\n this.sessionInfo = toSessionInfo(data);\n this.setConnectionState(ConnectionState.Connected);\n this.reconnectAttempt = 0;\n this.logger.info('Connected, session:', this.sessionInfo.sessionId);\n this.emit('connected', this.sessionInfo);\n if (!settled) {\n settled = true;\n resolve(this.sessionInfo);\n }\n };\n\n const onAuthError = (data: { error: string }) => {\n this.logger.error('Auth error:', data.error);\n this.setConnectionState(ConnectionState.Error);\n this.emit('authError', data.error);\n if (!settled) {\n settled = true;\n reject(new EstuaryError(ErrorCode.AUTH_FAILED, data.error));\n }\n };\n\n const onConnectError = (err: Error) => {\n this.logger.error('Connection error:', err.message);\n if (!settled) {\n settled = true;\n this.setConnectionState(ConnectionState.Error);\n reject(new EstuaryError(ErrorCode.CONNECTION_FAILED, err.message));\n } else {\n this.handleDisconnect('connect_error');\n }\n };\n\n const onDisconnect = (reason: string) => {\n this.logger.info('Disconnected:', reason);\n this.sessionInfo = null;\n if (!settled) {\n settled = true;\n this.setConnectionState(ConnectionState.Disconnected);\n reject(new EstuaryError(ErrorCode.CONNECTION_FAILED, `Disconnected: ${reason}`));\n } else {\n this.handleDisconnect(reason);\n }\n };\n\n this.socket.on('connect', onConnect);\n this.socket.on('session_info', onSessionInfo);\n this.socket.on('auth_error', onAuthError);\n this.socket.on('connect_error', onConnectError);\n this.socket.on('disconnect', onDisconnect);\n\n // Wire up all server → client events\n this.bindServerEvents();\n });\n }\n\n disconnect(): void {\n this.clearReconnectTimer();\n this.sessionInfo = null;\n if (this.socket) {\n this.socket.removeAllListeners();\n this.socket.disconnect();\n this.socket = null;\n }\n this.setConnectionState(ConnectionState.Disconnected);\n this.emit('disconnected', 'manual');\n }\n\n emitEvent(event: string, data?: unknown): void {\n if (!this.socket?.connected) {\n throw new EstuaryError(ErrorCode.NOT_CONNECTED, 'Not connected to server');\n }\n this.socket.emit(event, data);\n }\n\n private bindServerEvents(): void {\n if (!this.socket) return;\n\n this.socket.on('bot_response', (data: WireBotResponse) => {\n this.emit('botResponse', toBotResponse(data));\n });\n\n this.socket.on('bot_voice', (data: WireBotVoice) => {\n this.emit('botVoice', toBotVoice(data));\n });\n\n this.socket.on('stt_response', (data: WireSttResponse) => {\n this.emit('sttResponse', toSttResponse(data));\n });\n\n this.socket.on('interrupt', (data: WireInterruptData) => {\n this.emit('interrupt', toInterruptData(data));\n });\n\n this.socket.on('quota_exceeded', (data: WireQuotaExceededData) => {\n this.emit('quotaExceeded', toQuotaExceededData(data));\n });\n\n this.socket.on('camera_capture', (data: WireCameraCaptureRequest) => {\n this.emit('cameraCaptureRequest', toCameraCaptureRequest(data));\n });\n\n this.socket.on('error', (data: { message: string }) => {\n this.emit('error', new EstuaryError(ErrorCode.UNKNOWN, data.message));\n });\n\n this.socket.on('livekit_token', (data: WireLiveKitTokenResponse) => {\n // This is handled by LiveKitVoiceManager via a separate listener\n // We store it here so the voice manager can access it\n (this as unknown as { _livekitTokenCallback?: (d: LiveKitTokenResponse) => void })\n ._livekitTokenCallback?.(toLiveKitTokenResponse(data));\n });\n\n this.socket.on('livekit_ready', (data: { room: string }) => {\n this.emit('livekitConnected', data.room);\n });\n\n this.socket.on('memory_updated', (data: WireMemoryUpdated) => {\n this.emit('memoryUpdated', toMemoryUpdatedEvent(data));\n });\n }\n\n /** Register a callback for livekit_token events (used by LiveKitVoiceManager) */\n onLiveKitToken(callback: (data: LiveKitTokenResponse) => void): void {\n (this as unknown as { _livekitTokenCallback?: (d: LiveKitTokenResponse) => void })\n ._livekitTokenCallback = callback;\n }\n\n private handleDisconnect(reason: string): void {\n this.sessionInfo = null;\n this.setConnectionState(ConnectionState.Disconnected);\n this.emit('disconnected', reason);\n\n const autoReconnect = this.config.autoReconnect ?? true;\n const maxAttempts = this.config.maxReconnectAttempts ?? 5;\n\n if (autoReconnect && this.reconnectAttempt < maxAttempts) {\n this.attemptReconnect();\n }\n }\n\n private attemptReconnect(): void {\n const maxAttempts = this.config.maxReconnectAttempts ?? 5;\n const delay = this.config.reconnectDelayMs ?? 2000;\n\n if (this.reconnectAttempt >= maxAttempts) {\n this.logger.warn('Max reconnect attempts reached');\n this.setConnectionState(ConnectionState.Error);\n this.emit('error', new EstuaryError(\n ErrorCode.CONNECTION_FAILED,\n `Failed to reconnect after ${maxAttempts} attempts`,\n ));\n return;\n }\n\n this.reconnectAttempt++;\n this.setConnectionState(ConnectionState.Reconnecting);\n this.emit('reconnecting', this.reconnectAttempt);\n this.logger.info(`Reconnecting (attempt ${this.reconnectAttempt}/${maxAttempts})...`);\n\n this.clearReconnectTimer();\n this.reconnectTimer = setTimeout(() => {\n this.connect().catch((err) => {\n this.logger.error('Reconnect failed:', err.message);\n // Will trigger another attempt via handleDisconnect\n });\n }, delay * this.reconnectAttempt); // Linear backoff: delay * attempt\n }\n\n private clearReconnectTimer(): void {\n if (this.reconnectTimer) {\n clearTimeout(this.reconnectTimer);\n this.reconnectTimer = null;\n }\n }\n\n private setConnectionState(state: ConnectionState): void {\n if (this.connectionState !== state) {\n this.connectionState = state;\n this.emit('connectionStateChanged', state);\n }\n }\n}\n","import type { VoiceManager, VoiceTransport } from '../types';\nimport type { SocketManager } from '../connection/socket-manager';\nimport type { Logger } from '../utils/logger';\n\nexport type { VoiceManager };\n\nasync function isLiveKitAvailable(): Promise<boolean> {\n try {\n await import('livekit-client');\n return true;\n } catch {\n return false;\n }\n}\n\nexport async function createVoiceManager(\n transport: VoiceTransport,\n socketManager: SocketManager,\n sampleRate: number,\n logger: Logger,\n): Promise<VoiceManager | null> {\n if (transport === 'websocket') {\n const { WebSocketVoiceManager } = await import('./websocket-voice');\n return new WebSocketVoiceManager(socketManager, sampleRate, logger);\n }\n\n if (transport === 'livekit') {\n if (await isLiveKitAvailable()) {\n const { LiveKitVoiceManager } = await import('./livekit-voice');\n return new LiveKitVoiceManager(socketManager, logger);\n }\n logger.warn('livekit-client not installed, falling back to WebSocket voice');\n const { WebSocketVoiceManager } = await import('./websocket-voice');\n return new WebSocketVoiceManager(socketManager, sampleRate, logger);\n }\n\n // auto: prefer LiveKit if available, else WebSocket\n if (transport === 'auto') {\n if (await isLiveKitAvailable()) {\n const { LiveKitVoiceManager } = await import('./livekit-voice');\n return new LiveKitVoiceManager(socketManager, logger);\n }\n const { WebSocketVoiceManager } = await import('./websocket-voice');\n return new WebSocketVoiceManager(socketManager, sampleRate, logger);\n }\n\n return null;\n}\n","import { EstuaryError, ErrorCode } from '../errors';\n\nconst DEFAULT_TIMEOUT_MS = 10_000;\n\nexport class RestClient {\n private baseUrl: string;\n private apiKey: string;\n private timeoutMs: number;\n\n constructor(baseUrl: string, apiKey: string, timeoutMs: number = DEFAULT_TIMEOUT_MS) {\n this.baseUrl = baseUrl.replace(/\\/+$/, '');\n this.apiKey = apiKey;\n this.timeoutMs = timeoutMs;\n }\n\n async get<T>(path: string, params?: Record<string, string | number | boolean | undefined>): Promise<T> {\n const url = this.buildUrl(path, params);\n return this.request<T>(url, { method: 'GET' });\n }\n\n async post<T>(path: string, body?: unknown): Promise<T> {\n const url = this.buildUrl(path);\n const init: RequestInit = { method: 'POST' };\n if (body !== undefined) {\n init.headers = { 'Content-Type': 'application/json' };\n init.body = JSON.stringify(body);\n }\n return this.request<T>(url, init);\n }\n\n async delete<T>(path: string, params?: Record<string, string | number | boolean | undefined>): Promise<T> {\n const url = this.buildUrl(path, params);\n return this.request<T>(url, { method: 'DELETE' });\n }\n\n dispose(): void {\n // No persistent resources to clean up\n }\n\n private buildUrl(path: string, params?: Record<string, string | number | boolean | undefined>): string {\n const url = new URL(path, this.baseUrl);\n if (params) {\n for (const [key, value] of Object.entries(params)) {\n if (value !== undefined) {\n url.searchParams.set(key, String(value));\n }\n }\n }\n return url.toString();\n }\n\n private async request<T>(url: string, init: RequestInit): Promise<T> {\n const headers = new Headers(init.headers);\n headers.set('X-API-Key', this.apiKey);\n\n const response = await fetch(url, {\n ...init,\n headers,\n signal: AbortSignal.timeout(this.timeoutMs),\n });\n\n if (!response.ok) {\n let detail: unknown;\n try {\n detail = await response.json();\n } catch {\n detail = await response.text().catch(() => null);\n }\n throw new EstuaryError(\n ErrorCode.REST_ERROR,\n `HTTP ${response.status}: ${response.statusText}`,\n detail,\n );\n }\n\n return response.json() as Promise<T>;\n }\n}\n","import type {\n MemoryListOptions,\n MemoryListResponse,\n MemoryTimelineOptions,\n MemoryTimelineResponse,\n MemoryStatsResponse,\n CoreFactsResponse,\n MemoryGraphOptions,\n MemoryGraphResponse,\n MemorySearchResponse,\n} from '../types';\nimport type { RestClient } from './rest-client';\n\nexport class MemoryClient {\n private rest: RestClient;\n private basePath: string;\n\n constructor(rest: RestClient, agentId: string, playerId: string) {\n this.rest = rest;\n this.basePath = `/api/agents/${agentId}/players/${playerId}/memories`;\n }\n\n async getMemories(options?: MemoryListOptions): Promise<MemoryListResponse> {\n return this.rest.get<MemoryListResponse>(this.basePath, options as Record<string, string | number | boolean | undefined>);\n }\n\n async getTimeline(options?: MemoryTimelineOptions): Promise<MemoryTimelineResponse> {\n return this.rest.get<MemoryTimelineResponse>(`${this.basePath}/timeline`, options as Record<string, string | number | boolean | undefined>);\n }\n\n async getStats(): Promise<MemoryStatsResponse> {\n return this.rest.get<MemoryStatsResponse>(`${this.basePath}/stats`);\n }\n\n async getCoreFacts(): Promise<CoreFactsResponse> {\n return this.rest.get<CoreFactsResponse>(`${this.basePath}/core-facts`);\n }\n\n async getGraph(options?: MemoryGraphOptions): Promise<MemoryGraphResponse> {\n const params: Record<string, string | number | boolean | undefined> = {};\n if (options) {\n if (options.includeEntities !== undefined) params.include_entities = options.includeEntities;\n if (options.includeCharacterMemories !== undefined) params.include_character_memories = options.includeCharacterMemories;\n }\n return this.rest.get<MemoryGraphResponse>(`${this.basePath}/graph`, params);\n }\n\n async search(query: string, limit?: number): Promise<MemorySearchResponse> {\n return this.rest.get<MemorySearchResponse>(`${this.basePath}/search`, { q: query, limit });\n }\n\n async deleteAll(confirm: boolean): Promise<{ message: string; deletedCount: number }> {\n return this.rest.delete<{ message: string; deletedCount: number }>(this.basePath, { confirm });\n }\n\n dispose(): void {\n // No persistent resources to clean up\n }\n}\n","import type { CharacterInfo } from '../types';\nimport type { RestClient } from './rest-client';\n\nexport class CharacterClient {\n private rest: RestClient;\n\n constructor(rest: RestClient) {\n this.rest = rest;\n }\n\n /** Fetch character details including 3D model and avatar URLs. */\n async getCharacter(characterId: string): Promise<CharacterInfo> {\n const raw = await this.rest.get<Record<string, unknown>>(`/api/agents/${characterId}`);\n return {\n id: raw.id as string,\n name: raw.name as string,\n tagline: (raw.tagline as string) ?? null,\n avatar: (raw.avatar as string) ?? null,\n modelUrl: (raw.modelUrl as string) ?? null,\n modelPreviewUrl: (raw.modelPreviewUrl as string) ?? null,\n modelStatus: (raw.modelStatus as string) ?? null,\n sourceImageUrl: (raw.sourceImageUrl as string) ?? null,\n };\n }\n\n dispose(): void {\n // No persistent resources to clean up\n }\n}\n","import type { BotVoice } from '../types';\n\nexport type AudioPlaybackEvent =\n | { type: 'started'; messageId: string }\n | { type: 'complete'; messageId: string };\n\nexport class AudioPlayer {\n private sampleRate: number;\n private onEvent: (event: AudioPlaybackEvent) => void;\n private audioContext: AudioContext | null = null;\n private mediaStreamDest: MediaStreamAudioDestinationNode | null = null;\n private audioElement: HTMLAudioElement | null = null;\n private queue: { buffer: AudioBuffer; messageId: string }[] = [];\n private currentSource: AudioBufferSourceNode | null = null;\n private currentMessageId: string | null = null;\n private isPlaying = false;\n private _isCleared = false;\n private _interruptedMessageId: string | null = null;\n\n constructor(sampleRate: number, onEvent: (event: AudioPlaybackEvent) => void) {\n this.sampleRate = sampleRate;\n this.onEvent = onEvent;\n }\n\n /** Whether audio is currently playing */\n get playing(): boolean {\n return this.isPlaying;\n }\n\n /** The messageId of the currently playing audio, or null */\n get playingMessageId(): string | null {\n return this.currentMessageId;\n }\n\n /** Mark a messageId as interrupted so late-arriving chunks are dropped */\n setInterruptedMessageId(id: string | null): void {\n this._interruptedMessageId = id;\n }\n\n enqueue(voice: BotVoice): void {\n // Drop chunks belonging to the interrupted message\n if (voice.messageId === this._interruptedMessageId) return;\n // Clear the filter once a new message arrives\n if (this._interruptedMessageId && voice.messageId !== this._interruptedMessageId) {\n this._interruptedMessageId = null;\n }\n\n this._isCleared = false;\n\n const ctx = this.getAudioContext();\n if (!ctx) return;\n\n const pcm16 = base64ToInt16Array(voice.audio);\n const float32 = int16ToFloat32(pcm16);\n\n const buffer = ctx.createBuffer(1, float32.length, this.sampleRate);\n buffer.getChannelData(0).set(float32);\n\n this.queue.push({ buffer, messageId: voice.messageId });\n\n if (!this.isPlaying) {\n this.playNext();\n }\n }\n\n clear(): void {\n this._isCleared = true;\n this.queue.length = 0;\n if (this.currentSource) {\n try {\n this.currentSource.onended = null;\n this.currentSource.stop();\n this.currentSource.disconnect();\n } catch {\n // Already stopped\n }\n this.currentSource = null;\n }\n // Push silence through mediaStreamDest to flush the <audio> element's\n // internal pipeline buffer. Without this, mobile browsers replay the last\n // buffered frames. We can't touch the <audio> element directly (pause or\n // srcObject reassignment revokes the autoplay privilege on mobile).\n if (this.audioContext && this.mediaStreamDest) {\n const silence = this.audioContext.createBuffer(\n 1,\n this.sampleRate * 0.25,\n this.sampleRate,\n );\n const src = this.audioContext.createBufferSource();\n src.buffer = silence;\n src.connect(this.mediaStreamDest);\n src.start();\n }\n this.isPlaying = false;\n this.currentMessageId = null;\n }\n\n dispose(): void {\n this.clear();\n if (this.audioElement) {\n this.audioElement.pause();\n this.audioElement.srcObject = null;\n this.audioElement.remove();\n this.audioElement = null;\n }\n if (this.mediaStreamDest) {\n this.mediaStreamDest.disconnect();\n this.mediaStreamDest = null;\n }\n if (this.audioContext) {\n this.audioContext.close().catch(() => {});\n this.audioContext = null;\n }\n }\n\n private getAudioContext(): AudioContext | null {\n if (this.audioContext) return this.audioContext;\n\n if (typeof AudioContext === 'undefined' && typeof (globalThis as any).webkitAudioContext === 'undefined') {\n return null;\n }\n\n const AudioCtx = globalThis.AudioContext || (globalThis as any).webkitAudioContext;\n const ctx = new AudioCtx({ sampleRate: this.sampleRate });\n this.audioContext = ctx;\n\n // Route through a MediaStreamDestination → hidden <audio> element so the\n // browser's AEC pipeline can see our playback as the echo reference signal.\n if (typeof document !== 'undefined') {\n this.mediaStreamDest = ctx.createMediaStreamDestination();\n const el = document.createElement('audio');\n el.srcObject = this.mediaStreamDest.stream;\n el.autoplay = true;\n // Hidden but in the DOM so the browser treats it as a real media element\n el.style.display = 'none';\n document.body.appendChild(el);\n el.play().catch(() => {});\n this.audioElement = el;\n }\n\n return ctx;\n }\n\n private playNext(): void {\n if (this._isCleared) return;\n\n const ctx = this.getAudioContext();\n if (!ctx || this.queue.length === 0) {\n if (this.isPlaying && this.currentMessageId) {\n this.onEvent({ type: 'complete', messageId: this.currentMessageId });\n }\n this.isPlaying = false;\n this.currentMessageId = null;\n return;\n }\n\n const { buffer, messageId } = this.queue.shift()!;\n\n // Emit started if this is a new message\n if (messageId !== this.currentMessageId) {\n if (this.currentMessageId) {\n this.onEvent({ type: 'complete', messageId: this.currentMessageId });\n }\n this.currentMessageId = messageId;\n this.onEvent({ type: 'started', messageId });\n }\n\n this.isPlaying = true;\n const source = ctx.createBufferSource();\n source.buffer = buffer;\n source.connect(this.mediaStreamDest ?? ctx.destination);\n this.currentSource = source;\n\n source.onended = () => {\n if (this._isCleared) return;\n this.currentSource = null;\n this.playNext();\n };\n\n if (this.audioElement) {\n this.audioElement.play().catch(() => {});\n }\n ctx.resume().catch(() => {});\n source.start();\n }\n}\n\nfunction base64ToInt16Array(base64: string): Int16Array {\n let bytes: Uint8Array;\n if (typeof atob === 'function') {\n const binary = atob(base64);\n bytes = new Uint8Array(binary.length);\n for (let i = 0; i < binary.length; i++) {\n bytes[i] = binary.charCodeAt(i);\n }\n } else {\n // Node.js fallback\n bytes = new Uint8Array(Buffer.from(base64, 'base64'));\n }\n return new Int16Array(bytes.buffer, bytes.byteOffset, bytes.byteLength / 2);\n}\n\nfunction int16ToFloat32(int16: Int16Array): Float32Array {\n const float32 = new Float32Array(int16.length);\n for (let i = 0; i < int16.length; i++) {\n float32[i] = int16[i] / (int16[i] < 0 ? 0x8000 : 0x7fff);\n }\n return float32;\n}\n","export class Logger {\n private enabled: boolean;\n\n constructor(debug = false) {\n this.enabled = debug;\n }\n\n setDebug(enabled: boolean): void {\n this.enabled = enabled;\n }\n\n debug(...args: unknown[]): void {\n if (this.enabled) console.debug('[Estuary]', ...args);\n }\n\n info(...args: unknown[]): void {\n if (this.enabled) console.info('[Estuary]', ...args);\n }\n\n warn(...args: unknown[]): void {\n console.warn('[Estuary]', ...args);\n }\n\n error(...args: unknown[]): void {\n console.error('[Estuary]', ...args);\n }\n}\n","/**\n * Parses `<action name=\"...\" .../>` XML tags from bot response text.\n *\n * Designed for streaming: call `parse()` with the accumulated text on each\n * chunk and it returns only newly-discovered actions since the last call.\n */\n\nexport interface ParsedAction {\n name: string;\n params: Record<string, string>;\n}\n\n// Matches self-closing <action .../> tags (with optional whitespace before />)\nconst ACTION_TAG_RE = /<action\\s+([^>]*?)\\/>/gi;\n// Matches key=\"value\" or key='value' attribute pairs\nconst ATTR_RE = /(\\w+)\\s*=\\s*\"([^\"]*)\"|(\\w+)\\s*=\\s*'([^']*)'/g;\n\nfunction parseAttributes(attrString: string): Record<string, string> {\n const attrs: Record<string, string> = {};\n let match: RegExpExecArray | null;\n while ((match = ATTR_RE.exec(attrString)) !== null) {\n const key = match[1] ?? match[3];\n const value = match[2] ?? match[4];\n attrs[key] = value;\n }\n return attrs;\n}\n\nexport class StreamingActionParser {\n private emittedCount = 0;\n\n /**\n * Parse the accumulated response text and return any new actions found\n * since the last call. Also returns the text with all action tags stripped.\n */\n parse(accumulatedText: string): { actions: ParsedAction[]; cleanText: string } {\n const allActions: ParsedAction[] = [];\n let match: RegExpExecArray | null;\n\n ACTION_TAG_RE.lastIndex = 0;\n while ((match = ACTION_TAG_RE.exec(accumulatedText)) !== null) {\n const attrs = parseAttributes(match[1]);\n const name = attrs.name;\n if (name) {\n delete attrs.name;\n allActions.push({ name, params: attrs });\n }\n }\n\n const newActions = allActions.slice(this.emittedCount);\n this.emittedCount = allActions.length;\n\n const cleanText = accumulatedText.replace(ACTION_TAG_RE, '').replace(/\\s{2,}/g, ' ').trimEnd();\n\n return { actions: newActions, cleanText };\n }\n\n reset(): void {\n this.emittedCount = 0;\n }\n}\n\n/**\n * One-shot parse: extract all actions and return clean text.\n * Useful for non-streaming contexts.\n */\nexport function parseActions(text: string): { actions: ParsedAction[]; cleanText: string } {\n const parser = new StreamingActionParser();\n return parser.parse(text);\n}\n","import { SocketManager } from './connection/socket-manager';\nimport { createVoiceManager } from './voice/voice-manager';\nimport { RestClient } from './rest/rest-client';\nimport { MemoryClient } from './rest/memory-client';\nimport { CharacterClient } from './rest/character-client';\nimport { AudioPlayer } from './audio/audio-player';\nimport { TypedEventEmitter } from './utils/event-emitter';\nimport { Logger } from './utils/logger';\nimport { EstuaryError, ErrorCode } from './errors';\nimport {\n EstuaryConfig,\n EstuaryEventMap,\n ConnectionState,\n SessionInfo,\n CharacterInfo,\n BotResponse,\n BotVoice,\n SttResponse,\n VoiceManager,\n} from './types';\nimport { StreamingActionParser } from './utils/action-parser';\n\nconst DEFAULT_SAMPLE_RATE = 16000;\n\nexport class EstuaryClient extends TypedEventEmitter<EstuaryEventMap> {\n private config: EstuaryConfig;\n private logger: Logger;\n private socketManager: SocketManager;\n private voiceManager: VoiceManager | null = null;\n private audioPlayer: AudioPlayer | null = null;\n private _memory: MemoryClient;\n private _character: CharacterClient;\n private _sessionInfo: SessionInfo | null = null;\n private actionParsers = new Map<string, StreamingActionParser>();\n private _hasAutoInterrupted = false;\n private _autoInterruptGraceTimer: ReturnType<typeof setTimeout> | null = null;\n\n constructor(config: EstuaryConfig) {\n super();\n this.config = config;\n this.logger = new Logger(config.debug ?? false);\n this.socketManager = new SocketManager(config, this.logger);\n this.forwardSocketEvents();\n\n // Set up REST clients\n const restClient = new RestClient(config.serverUrl, config.apiKey);\n this._memory = new MemoryClient(restClient, config.characterId, config.playerId);\n this._character = new CharacterClient(restClient);\n }\n\n /** Memory API client for querying memories, graphs, and facts */\n get memory(): MemoryClient {\n return this._memory;\n }\n\n /** Fetch character details including 3D model and avatar URLs. */\n async getCharacter(characterId?: string): Promise<CharacterInfo> {\n return this._character.getCharacter(characterId ?? this.config.characterId);\n }\n\n /** Current session info (null if not connected) */\n get session(): SessionInfo | null {\n return this._sessionInfo;\n }\n\n /** Current connection state */\n get connectionState(): ConnectionState {\n return this.socketManager.state;\n }\n\n /** Whether the client is connected and authenticated */\n get isConnected(): boolean {\n return this.socketManager.isConnected;\n }\n\n /** Connect to the Estuary server and authenticate */\n async connect(): Promise<SessionInfo> {\n this.logger.info('Connecting...');\n const session = await this.socketManager.connect();\n this._sessionInfo = session;\n return session;\n }\n\n /** Disconnect from the server */\n async disconnect(): Promise<void> {\n this.logger.info('Disconnecting...');\n if (this._autoInterruptGraceTimer) {\n clearTimeout(this._autoInterruptGraceTimer);\n this._autoInterruptGraceTimer = null;\n }\n await this.stopVoice();\n this.audioPlayer?.dispose();\n this.audioPlayer = null;\n this.socketManager.disconnect();\n this._sessionInfo = null;\n }\n\n /** Send a text message to the character. Defaults to textOnly=true (no TTS audio response). Pass textOnly=false to receive voice audio. */\n sendText(text: string, textOnly = true): void {\n this.ensureConnected();\n this.socketManager.emitEvent('text', { text, textOnly });\n }\n\n /** Interrupt the current bot response */\n interrupt(messageId?: string): void {\n this.ensureConnected();\n this.socketManager.emitEvent('client_interrupt', { message_id: messageId });\n this.audioPlayer?.setInterruptedMessageId(messageId ?? this.audioPlayer.playingMessageId);\n this.audioPlayer?.clear();\n this._hasAutoInterrupted = true;\n if (this._autoInterruptGraceTimer) {\n clearTimeout(this._autoInterruptGraceTimer);\n this._autoInterruptGraceTimer = null;\n }\n if (this.config.suppressMicDuringPlayback) {\n this.voiceManager?.setSuppressed?.(false);\n }\n }\n\n /** Send a camera image for vision processing */\n sendCameraImage(imageBase64: string, mimeType: string, requestId?: string, text?: string): void {\n this.ensureConnected();\n this.socketManager.emitEvent('camera_image', {\n image: imageBase64,\n mime_type: mimeType,\n request_id: requestId,\n text,\n });\n }\n\n /** Update session preferences */\n updatePreferences(preferences: { enableVisionAcknowledgment?: boolean }): void {\n this.ensureConnected();\n this.socketManager.emitEvent('update_preferences', preferences);\n }\n\n /** Notify server that audio playback completed for a message */\n notifyAudioPlaybackComplete(messageId?: string): void {\n this.ensureConnected();\n this.socketManager.emitEvent('audio_playback_complete', { message_id: messageId });\n }\n\n // ─── Voice ───────────────────────────────────────────────────\n\n /** Start voice input (requests microphone permission) */\n async startVoice(): Promise<void> {\n this.ensureConnected();\n\n if (this.voiceManager?.isActive) {\n throw new EstuaryError(ErrorCode.VOICE_ALREADY_ACTIVE, 'Voice is already active');\n }\n\n const transport = this.config.voiceTransport ?? 'auto';\n const sampleRate = this.config.audioSampleRate ?? DEFAULT_SAMPLE_RATE;\n\n this.voiceManager = await createVoiceManager(transport, this.socketManager, sampleRate, this.logger);\n if (!this.voiceManager) {\n throw new EstuaryError(ErrorCode.VOICE_NOT_SUPPORTED, 'No voice transport available');\n }\n\n // Set up audio player for bot voice responses (browser only)\n if (!this.audioPlayer && typeof AudioContext !== 'undefined') {\n this.audioPlayer = new AudioPlayer(sampleRate, (event) => {\n if (event.type === 'started') {\n // Suppress auto-interrupt during grace period so trailing STT partials\n // from the user's previous speech don't kill the new audio.\n this._hasAutoInterrupted = true;\n if (this._autoInterruptGraceTimer) clearTimeout(this._autoInterruptGraceTimer);\n this._autoInterruptGraceTimer = setTimeout(() => {\n this._hasAutoInterrupted = false;\n this._autoInterruptGraceTimer = null;\n }, 1500);\n this.emit('audioPlaybackStarted', event.messageId);\n if (this.config.suppressMicDuringPlayback) {\n this.voiceManager?.setSuppressed?.(true);\n }\n } else if (event.type === 'complete') {\n this.emit('audioPlaybackComplete', event.messageId);\n this.notifyAudioPlaybackComplete(event.messageId);\n if (this.config.suppressMicDuringPlayback) {\n this.voiceManager?.setSuppressed?.(false);\n }\n }\n });\n }\n\n await this.voiceManager.start();\n this.emit('voiceStarted');\n }\n\n /** Stop voice input */\n async stopVoice(): Promise<void> {\n if (this.voiceManager?.isActive) {\n await this.voiceManager.stop();\n this.voiceManager.dispose();\n this.voiceManager = null;\n this.emit('voiceStopped');\n }\n }\n\n /** Toggle microphone mute */\n toggleMute(): void {\n if (!this.voiceManager?.isActive) {\n throw new EstuaryError(ErrorCode.VOICE_NOT_ACTIVE, 'Voice is not active');\n }\n this.voiceManager.toggleMute();\n }\n\n /** Whether the microphone is muted */\n get isMuted(): boolean {\n return this.voiceManager?.isMuted ?? false;\n }\n\n /** Get/set suppressMicDuringPlayback at runtime (no reconnect needed) */\n get suppressMicDuringPlayback(): boolean {\n return this.config.suppressMicDuringPlayback ?? false;\n }\n\n set suppressMicDuringPlayback(enabled: boolean) {\n this.config.suppressMicDuringPlayback = enabled;\n }\n\n /** Whether voice is currently active */\n get isVoiceActive(): boolean {\n return this.voiceManager?.isActive ?? false;\n }\n\n // ─── Internal ────────────────────────────────────────────────\n\n private ensureConnected(): void {\n if (!this.socketManager.isConnected) {\n throw new EstuaryError(ErrorCode.NOT_CONNECTED, 'Not connected to server. Call connect() first.');\n }\n }\n\n private forwardSocketEvents(): void {\n // Forward all socket manager events to this client's event emitter\n this.socketManager.on('connected', (session) => {\n this._sessionInfo = session;\n this.emit('connected', session);\n });\n this.socketManager.on('disconnected', (reason) => {\n this._sessionInfo = null;\n this.actionParsers.clear();\n this.emit('disconnected', reason);\n });\n this.socketManager.on('reconnecting', (attempt) => this.emit('reconnecting', attempt));\n this.socketManager.on('connectionStateChanged', (state) => this.emit('connectionStateChanged', state));\n this.socketManager.on('botResponse', (response) => this.handleBotResponse(response));\n this.socketManager.on('botVoice', (voice) => this.handleBotVoice(voice));\n this.socketManager.on('sttResponse', (response) => {\n this.maybeAutoInterrupt(response);\n this.emit('sttResponse', response);\n });\n this.socketManager.on('interrupt', (data) => {\n this.audioPlayer?.setInterruptedMessageId(data.messageId ?? null);\n this.audioPlayer?.clear();\n this.actionParsers.clear();\n if (this.config.suppressMicDuringPlayback) {\n this.voiceManager?.setSuppressed?.(false);\n }\n this.emit('interrupt', data);\n });\n this.socketManager.on('error', (error) => this.emit('error', error));\n this.socketManager.on('authError', (error) => this.emit('authError', error));\n this.socketManager.on('quotaExceeded', (data) => this.emit('quotaExceeded', data));\n this.socketManager.on('cameraCaptureRequest', (request) => this.emit('cameraCaptureRequest', request));\n this.socketManager.on('livekitConnected', (room) => this.emit('livekitConnected', room));\n this.socketManager.on('livekitDisconnected', () => this.emit('livekitDisconnected'));\n this.socketManager.on('memoryUpdated', (event) => this.emit('memoryUpdated', event));\n }\n\n private handleBotResponse(response: BotResponse): void {\n const { messageId } = response;\n\n // Get or create a parser for this message stream\n if (!this.actionParsers.has(messageId)) {\n this.actionParsers.set(messageId, new StreamingActionParser());\n }\n const parser = this.actionParsers.get(messageId)!;\n\n // Parse actions from accumulated text\n const { actions, cleanText } = parser.parse(response.text);\n\n // Emit characterAction events for newly discovered actions\n for (const action of actions) {\n this.emit('characterAction', {\n name: action.name,\n params: action.params,\n messageId,\n });\n }\n\n // Forward botResponse with cleaned text\n this.emit('botResponse', {\n ...response,\n text: cleanText,\n });\n\n // Clean up parser when message is final\n if (response.isFinal) {\n this.actionParsers.delete(messageId);\n }\n }\n\n private handleBotVoice(voice: BotVoice): void {\n this.emit('botVoice', voice);\n // Enqueue audio for playback if we have an audio player\n this.audioPlayer?.enqueue(voice);\n }\n\n private maybeAutoInterrupt(stt: SttResponse): void {\n if ((this.config.autoInterruptOnSpeech ?? true) === false) return;\n if (this.config.suppressMicDuringPlayback) return;\n if (stt.isFinal) return;\n if (!this.audioPlayer?.playing) return;\n if (this._hasAutoInterrupted) return;\n\n this._hasAutoInterrupted = true;\n this.interrupt();\n }\n}\n"]}
|
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import { EstuaryError } from './chunk-
|
|
1
|
+
import { EstuaryError } from './chunk-W5QYPYX3.mjs';
|
|
2
2
|
|
|
3
3
|
// src/voice/livekit-voice.ts
|
|
4
4
|
var LiveKitVoiceManager = class {
|
|
@@ -150,5 +150,5 @@ var LiveKitVoiceManager = class {
|
|
|
150
150
|
};
|
|
151
151
|
|
|
152
152
|
export { LiveKitVoiceManager };
|
|
153
|
-
//# sourceMappingURL=livekit-voice-
|
|
154
|
-
//# sourceMappingURL=livekit-voice-
|
|
153
|
+
//# sourceMappingURL=livekit-voice-A52TC6XZ.mjs.map
|
|
154
|
+
//# sourceMappingURL=livekit-voice-A52TC6XZ.mjs.map
|