@garrix82/reactgenie-lib 1.3.0

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.
Files changed (85) hide show
  1. package/.env.example +22 -0
  2. package/.github/workflows/publish.yml +20 -0
  3. package/LICENSE.txt +201 -0
  4. package/README.md +621 -0
  5. package/babel.config.js +29 -0
  6. package/dist/adapters/__tests__/expo-router-adapter.test.d.ts +1 -0
  7. package/dist/adapters/expo-router-adapter.d.ts +16 -0
  8. package/dist/adapters/expo-router-adapter.js +521 -0
  9. package/dist/adapters/navigation-adapter.d.ts +20 -0
  10. package/dist/adapters/navigation-adapter.js +137 -0
  11. package/dist/audio-visualizer.d.ts +14 -0
  12. package/dist/audio-visualizer.js +123 -0
  13. package/dist/current-selection.d.ts +27 -0
  14. package/dist/current-selection.js +94 -0
  15. package/dist/errors.d.ts +19 -0
  16. package/dist/errors.js +37 -0
  17. package/dist/genie/DateTime.d.ts +66 -0
  18. package/dist/genie/DateTime.js +399 -0
  19. package/dist/genie/TimeDelta.d.ts +35 -0
  20. package/dist/genie/TimeDelta.js +169 -0
  21. package/dist/genie-view-wrapper.d.ts +1 -0
  22. package/dist/genie-view-wrapper.js +377 -0
  23. package/dist/hooks/__tests__/useSpeechRecognition.test.d.ts +1 -0
  24. package/dist/hooks/useSpeechRecognition.d.ts +28 -0
  25. package/dist/hooks/useSpeechRecognition.js +118 -0
  26. package/dist/index.d.ts +30 -0
  27. package/dist/index.js +469 -0
  28. package/dist/logger.d.ts +23 -0
  29. package/dist/logger.js +597 -0
  30. package/dist/logger.remote.test.d.ts +0 -0
  31. package/dist/modality-provider-v2.d.ts +28 -0
  32. package/dist/modality-provider-v2.js +1321 -0
  33. package/dist/modality-provider.d.ts +22 -0
  34. package/dist/modality-provider.js +373 -0
  35. package/dist/native-visibility.d.ts +28 -0
  36. package/dist/native-visibility.js +50 -0
  37. package/dist/platform/VoiceRecognitionBar.d.ts +17 -0
  38. package/dist/platform/VoiceRecognitionBar.js +332 -0
  39. package/dist/platform/components.d.ts +32 -0
  40. package/dist/platform/components.js +351 -0
  41. package/dist/platform/events.d.ts +31 -0
  42. package/dist/platform/events.js +274 -0
  43. package/dist/platform/index.d.ts +3 -0
  44. package/dist/platform/index.js +39 -0
  45. package/dist/platform/types.d.ts +79 -0
  46. package/dist/platform/types.js +97 -0
  47. package/dist/react-decorators.d.ts +87 -0
  48. package/dist/react-decorators.js +368 -0
  49. package/dist/shared-store.d.ts +74 -0
  50. package/dist/shared-store.js +589 -0
  51. package/dist/speech-recognition/__tests__/speech-recognition-groq-transport.test.d.ts +1 -0
  52. package/dist/speech-recognition/__tests__/speech-recognition-native.test.d.ts +1 -0
  53. package/dist/speech-recognition/__tests__/speech-recognition-openai-native.test.d.ts +1 -0
  54. package/dist/speech-recognition/__tests__/speech-recognition-openai.test.d.ts +1 -0
  55. package/dist/speech-recognition/__tests__/speech-recognition-unified-import.test.d.ts +0 -0
  56. package/dist/speech-recognition/__tests__/speech-recognition-unified.test.d.ts +1 -0
  57. package/dist/speech-recognition/speech-recognition-groq.d.ts +21 -0
  58. package/dist/speech-recognition/speech-recognition-groq.js +409 -0
  59. package/dist/speech-recognition/speech-recognition-mlx.d.ts +15 -0
  60. package/dist/speech-recognition/speech-recognition-mlx.js +393 -0
  61. package/dist/speech-recognition/speech-recognition-native.d.ts +24 -0
  62. package/dist/speech-recognition/speech-recognition-native.js +632 -0
  63. package/dist/speech-recognition/speech-recognition-openai-native.d.ts +40 -0
  64. package/dist/speech-recognition/speech-recognition-openai-native.js +653 -0
  65. package/dist/speech-recognition/speech-recognition-openai.d.ts +39 -0
  66. package/dist/speech-recognition/speech-recognition-openai.js +718 -0
  67. package/dist/speech-recognition/speech-recognition-unified.d.ts +93 -0
  68. package/dist/speech-recognition/speech-recognition-unified.js +589 -0
  69. package/dist/speech-recognition/utils/groq-transcription.d.ts +41 -0
  70. package/dist/speech-recognition/utils/groq-transcription.js +382 -0
  71. package/dist/speech-recognition.d.ts +7 -0
  72. package/dist/speech-recognition.js +61 -0
  73. package/dist/voice-pipeline-telemetry.d.ts +26 -0
  74. package/dist/voice-pipeline-telemetry.js +15 -0
  75. package/garrix82-reactgenie-lib-1.3.0.tgz +0 -0
  76. package/metro/index.js +3 -0
  77. package/metro/with-genie-registry.js +47 -0
  78. package/package.json +111 -0
  79. package/scripts/dry-run.js +23 -0
  80. package/scripts/generate-genie-registry.js +278 -0
  81. package/scripts/log-file-test.js +51 -0
  82. package/scripts/parse.js +26 -0
  83. package/scripts/prompt.js +19 -0
  84. package/scripts/set-script.js +200 -0
  85. package/tsconfig.json +36 -0
@@ -0,0 +1,393 @@
1
+ "use strict";
2
+
3
+ Object.defineProperty(exports, "__esModule", {
4
+ value: true
5
+ });
6
+ exports.default = exports.MLXSpeechRecognizer = void 0;
7
+ var _react = _interopRequireWildcard(require("react"));
8
+ var _reactNative = require("react-native");
9
+ var _logger = _interopRequireDefault(require("../logger"));
10
+ function _interopRequireDefault(e) { return e && e.__esModule ? e : { default: e }; }
11
+ function _interopRequireWildcard(e, t) { if ("function" == typeof WeakMap) var r = new WeakMap(), n = new WeakMap(); return (_interopRequireWildcard = function (e, t) { if (!t && e && e.__esModule) return e; var o, i, f = { __proto__: null, default: e }; if (null === e || "object" != typeof e && "function" != typeof e) return f; if (o = t ? n : r) { if (o.has(e)) return o.get(e); o.set(e, f); } for (const t in e) "default" !== t && {}.hasOwnProperty.call(e, t) && ((i = (o = Object.defineProperty) && Object.getOwnPropertyDescriptor(e, t)) && (i.get || i.set) ? o(f, t, i) : f[t] = e[t]); return f; })(e, t); }
12
+ /**
13
+ * MLX Whisper-based Speech Recognition using a local FastAPI server.
14
+ * Simplified version with manual start/stop control.
15
+ */
16
+
17
+ const MLXSpeechRecognizer = ({
18
+ shouldListen,
19
+ speechStatusCallback,
20
+ speechResultCallback,
21
+ language = 'en',
22
+ onError,
23
+ baseUrl = 'http://127.0.0.1:8000',
24
+ mode,
25
+ modelName = 'large-v3-turbo',
26
+ minAudioSize = 1000
27
+ }) => {
28
+ const [isRecording, setIsRecording] = (0, _react.useState)(false);
29
+ const mediaRecorderRef = (0, _react.useRef)(null);
30
+ const audioChunksRef = (0, _react.useRef)([]);
31
+ const streamRef = (0, _react.useRef)(null);
32
+ const abortControllerRef = (0, _react.useRef)(null);
33
+ const startTimeRef = (0, _react.useRef)(0);
34
+ // Prevent concurrent uploads / duplicate calls
35
+ const processingRef = (0, _react.useRef)(false);
36
+ // Track last sent transcript for finalization
37
+ const lastSentTranscriptRef = (0, _react.useRef)('');
38
+
39
+ /**
40
+ * Start recording audio from microphone.
41
+ */
42
+ const startRecording = (0, _react.useCallback)(async () => {
43
+ try {
44
+ _logger.default.debug('Starting MLX Whisper recording...');
45
+ _logger.default.debug(`shouldListen=${shouldListen}, isRecording=${isRecording}, processing=${processingRef.current}`);
46
+
47
+ // Browser-specific audio constraints
48
+ const isSafari = /^((?!chrome|android).)*safari/i.test(navigator.userAgent);
49
+ _logger.default.debug(`Browser detection: Safari = ${isSafari}`);
50
+ const stream = await navigator.mediaDevices.getUserMedia({
51
+ audio: isSafari ? {
52
+ // Safari prefers simpler constraints
53
+ echoCancellation: false,
54
+ noiseSuppression: false,
55
+ autoGainControl: false
56
+ } : {
57
+ echoCancellation: true,
58
+ noiseSuppression: true,
59
+ autoGainControl: true,
60
+ sampleRate: 16000
61
+ }
62
+ });
63
+ _logger.default.debug(`Audio stream tracks: ${stream.getAudioTracks().length}`);
64
+ stream.getAudioTracks().forEach((track, index) => {
65
+ _logger.default.debug(`Track ${index}: ${track.label}, enabled: ${track.enabled}, readyState: ${track.readyState}`);
66
+ });
67
+ streamRef.current = stream;
68
+ audioChunksRef.current = [];
69
+
70
+ // Create MediaRecorder with supported format
71
+ const mimeType = getSupportedMimeType();
72
+ _logger.default.debug(`Using MIME type: ${mimeType}`);
73
+
74
+ // Try creating MediaRecorder with minimal options first
75
+ let mediaRecorder;
76
+ try {
77
+ mediaRecorder = new MediaRecorder(stream, {
78
+ mimeType
79
+ });
80
+ _logger.default.debug('MediaRecorder created with minimal options');
81
+ } catch (error) {
82
+ _logger.default.warn('MediaRecorder constructor failed with minimal options, trying fallback:', error);
83
+ // Fallback without mimeType
84
+ mediaRecorder = new MediaRecorder(stream);
85
+ _logger.default.debug('MediaRecorder created with fallback options');
86
+ }
87
+ mediaRecorderRef.current = mediaRecorder;
88
+
89
+ // Handle audio data
90
+ mediaRecorder.ondataavailable = event => {
91
+ if (event.data.size > 0) {
92
+ audioChunksRef.current.push(event.data);
93
+ _logger.default.debug(`Audio chunk: ${event.data.size} bytes`);
94
+ } else {
95
+ _logger.default.debug('ondataavailable with empty chunk');
96
+ }
97
+ };
98
+
99
+ // Handle start
100
+ mediaRecorder.onstart = () => {
101
+ _logger.default.debug('MediaRecorder started, state:', mediaRecorder.state);
102
+ };
103
+
104
+ // Handle recording stop
105
+ mediaRecorder.onstop = async () => {
106
+ _logger.default.debug('Recording stopped, processing audio...');
107
+ setIsRecording(false);
108
+
109
+ // Clean up stream (close mic immediately)
110
+ if (streamRef.current) {
111
+ streamRef.current.getTracks().forEach(track => track.stop());
112
+ streamRef.current = null;
113
+ }
114
+ _logger.default.info('🛑 MLX Whisper recording stopped');
115
+ _logger.default.debug(`Collected ${audioChunksRef.current.length} audio chunks`);
116
+ await processAudioChunks();
117
+ };
118
+
119
+ // Handle errors
120
+ mediaRecorder.onerror = event => {
121
+ _logger.default.error('MediaRecorder error:', event);
122
+ setIsRecording(false);
123
+ const error = new Error('Recording failed');
124
+ if (onError) onError(error);
125
+ speechStatusCallback(false, '');
126
+ };
127
+
128
+ // Start recording with a small timeslice to ensure data collection
129
+ mediaRecorder.start(100); // Collect data every 100ms
130
+ _logger.default.debug('MediaRecorder.start() called');
131
+ startTimeRef.current = Date.now();
132
+ setIsRecording(true);
133
+ speechStatusCallback(true, '');
134
+ _logger.default.info('🎤 MLX Whisper recording started');
135
+ } catch (error) {
136
+ _logger.default.error('Failed to start recording:', error);
137
+ const err = error instanceof Error ? error : new Error(String(error));
138
+ if (onError) onError(err);
139
+ speechStatusCallback(false, '');
140
+ }
141
+ }, [speechStatusCallback, onError]);
142
+
143
+ /**
144
+ * Stop recording and process audio.
145
+ */
146
+ const stopRecording = (0, _react.useCallback)(() => {
147
+ const elapsed = Date.now() - startTimeRef.current;
148
+ const minDuration = 1000; // Minimum 1 second recording
149
+
150
+ if (elapsed < minDuration) {
151
+ // Schedule stop after minimum duration
152
+ const delay = minDuration - elapsed;
153
+ setTimeout(() => {
154
+ if (mediaRecorderRef.current && mediaRecorderRef.current.state === 'recording') {
155
+ try {
156
+ // Flush any pending data
157
+ mediaRecorderRef.current.requestData?.();
158
+ } catch (e) {}
159
+ mediaRecorderRef.current.stop();
160
+ }
161
+ }, delay);
162
+ _logger.default.debug(`Recording too short (${elapsed}ms), delaying stop by ${delay}ms`);
163
+ } else {
164
+ // Stop immediately if minimum duration met
165
+ if (mediaRecorderRef.current && mediaRecorderRef.current.state === 'recording') {
166
+ try {
167
+ mediaRecorderRef.current.requestData?.();
168
+ } catch (e) {}
169
+ mediaRecorderRef.current.stop();
170
+ }
171
+ }
172
+
173
+ // Note: setIsRecording(false) moved to onstop handler
174
+ }, []);
175
+
176
+ /**
177
+ * Process collected audio chunks and send to server.
178
+ */
179
+ const processAudioChunks = (0, _react.useCallback)(async () => {
180
+ if (audioChunksRef.current.length === 0) {
181
+ _logger.default.warn('No audio chunks to process');
182
+ speechStatusCallback(false, '');
183
+ return;
184
+ }
185
+
186
+ // Prevent duplicate uploads
187
+ if (processingRef.current) {
188
+ _logger.default.warn('Already processing an utterance, skipping duplicate upload');
189
+ return;
190
+ }
191
+ processingRef.current = true;
192
+ // Give feedback to UI that we are uploading/transcribing
193
+ try {
194
+ // indicate remote processing started (no fake interim text)
195
+ speechStatusCallback(true, '');
196
+ // Create audio blob
197
+ const mimeType = getSupportedMimeType();
198
+ const audioBlob = new Blob(audioChunksRef.current, {
199
+ type: mimeType
200
+ });
201
+ _logger.default.debug(`Audio blob size: ${audioBlob.size} bytes`);
202
+ if (audioBlob.size < minAudioSize) {
203
+ _logger.default.warn(`Audio too short (${audioBlob.size} < ${minAudioSize} bytes), skipping`);
204
+ speechStatusCallback(false, '');
205
+ speechResultCallback('');
206
+ return;
207
+ }
208
+
209
+ // Create file for upload
210
+ const extension = getFileExtension(mimeType);
211
+ const audioFile = new File([audioBlob], `audio.${extension}`, {
212
+ type: mimeType
213
+ });
214
+
215
+ // Use single REST endpoint for oneshot and SSE
216
+ const endpoint = `${baseUrl}/v1/audio/transcriptions`;
217
+
218
+ // Prepare request
219
+ abortControllerRef.current = new AbortController();
220
+ const formData = new FormData();
221
+ formData.append('file', audioFile);
222
+ formData.append('model', modelName);
223
+ formData.append('response_format', 'json');
224
+ if (language) formData.append('language', language);
225
+ if (mode === 'oneshot') {
226
+ formData.append('stream', 'false');
227
+ } else {
228
+ formData.append('stream', 'true');
229
+ }
230
+ _logger.default.info(`📤 Sending ${audioBlob.size} bytes to ${endpoint}`);
231
+ _logger.default.info(`🗣️ Mode: ${mode}, Model: ${modelName}, endpoint: ${endpoint}`);
232
+
233
+ // Send to server
234
+ const response = await fetch(endpoint, {
235
+ method: 'POST',
236
+ body: formData,
237
+ signal: abortControllerRef.current.signal
238
+ });
239
+ if (!response.ok) {
240
+ throw new Error(`Server error: ${response.status}`);
241
+ }
242
+
243
+ // Handle response
244
+ if (mode === 'oneshot') {
245
+ const result = await response.json();
246
+ const transcript = result?.text?.trim() || '';
247
+ _logger.default.debug('Received transcription response:', result);
248
+ _logger.default.debug(`Transcription text: "${transcript}"`);
249
+ // Deliver transcript first, then signal end so ModalityProvider processes it once
250
+ lastSentTranscriptRef.current = transcript;
251
+ speechResultCallback(transcript);
252
+ speechStatusCallback(false, '');
253
+ } else {
254
+ // Handle streaming response (SSE)
255
+ await handleStreamingResponse(response);
256
+ }
257
+ } catch (error) {
258
+ _logger.default.error('Transcription error:', error);
259
+ const err = error instanceof Error ? error : new Error(String(error));
260
+ if (onError) onError(err);
261
+ speechStatusCallback(false, '');
262
+ speechResultCallback('');
263
+ } finally {
264
+ audioChunksRef.current = [];
265
+ if (abortControllerRef.current) {
266
+ abortControllerRef.current = null;
267
+ }
268
+ processingRef.current = false;
269
+ }
270
+ }, [speechStatusCallback, speechResultCallback, baseUrl, mode, modelName, language, minAudioSize, onError]);
271
+
272
+ /**
273
+ * Handle streaming transcription response.
274
+ */
275
+ const handleStreamingResponse = (0, _react.useCallback)(async response => {
276
+ const reader = response.body?.getReader();
277
+ if (!reader) throw new Error('No response reader');
278
+ const decoder = new TextDecoder();
279
+ let buffer = '';
280
+ let lastTranscript = '';
281
+ let sawDone = false;
282
+ try {
283
+ let done = false;
284
+ while (!done) {
285
+ const result = await reader.read();
286
+ done = result.done;
287
+ if (done) break;
288
+ buffer += decoder.decode(result.value, {
289
+ stream: true
290
+ });
291
+ const lines = buffer.split('\n');
292
+ buffer = lines.pop() || '';
293
+ for (const line of lines) {
294
+ if (line.startsWith('data: ')) {
295
+ const data = line.slice(6).trim();
296
+ if (data === '[DONE]') {
297
+ sawDone = true;
298
+ break;
299
+ }
300
+ try {
301
+ const parsedResult = JSON.parse(data);
302
+ const transcript = parsedResult?.text?.trim();
303
+ if (transcript) {
304
+ _logger.default.info(`📝 Streaming: "${transcript}"`);
305
+ lastTranscript = transcript;
306
+ // deliver partial to UI immediately
307
+ speechResultCallback(transcript);
308
+ }
309
+ } catch (e) {
310
+ // Ignore parse errors
311
+ }
312
+ }
313
+ }
314
+ if (sawDone) break;
315
+ }
316
+ } finally {
317
+ // Small delay to ensure UI has applied last partial
318
+ try {
319
+ await new Promise(res => setTimeout(res, 60));
320
+ } catch (e) {}
321
+ if (lastTranscript) {
322
+ _logger.default.debug(`Final streaming transcript before end: "${lastTranscript}"`);
323
+ try {
324
+ speechResultCallback(lastTranscript);
325
+ } catch (e) {}
326
+ }
327
+
328
+ // Signal stream end so ModalityProvider triggers the Genie interpreter once
329
+ try {
330
+ speechStatusCallback(false, '');
331
+ } catch (e) {}
332
+ reader.releaseLock();
333
+ }
334
+ }, [speechStatusCallback, speechResultCallback]);
335
+
336
+ /**
337
+ * React to shouldListen changes.
338
+ */
339
+ (0, _react.useEffect)(() => {
340
+ _logger.default.debug(`shouldListen prop changed: ${shouldListen}`);
341
+ // Avoid starting a new recording while previous utterance is processing
342
+ if (shouldListen && !isRecording && !processingRef.current) {
343
+ startRecording();
344
+ } else if (!shouldListen && isRecording) {
345
+ stopRecording();
346
+ }
347
+ }, [shouldListen, isRecording, startRecording, stopRecording]);
348
+
349
+ /**
350
+ * Cleanup on unmount.
351
+ */
352
+ (0, _react.useEffect)(() => {
353
+ return () => {
354
+ if (mediaRecorderRef.current && mediaRecorderRef.current.state === 'recording') {
355
+ mediaRecorderRef.current.stop();
356
+ }
357
+ if (streamRef.current) {
358
+ streamRef.current.getTracks().forEach(track => track.stop());
359
+ }
360
+ if (abortControllerRef.current) {
361
+ abortControllerRef.current.abort();
362
+ }
363
+ };
364
+ }, []);
365
+ return /*#__PURE__*/_react.default.createElement(_reactNative.View, null);
366
+ };
367
+
368
+ /**
369
+ * Get supported MIME type for MediaRecorder.
370
+ */
371
+ exports.MLXSpeechRecognizer = MLXSpeechRecognizer;
372
+ function getSupportedMimeType() {
373
+ const types = ['audio/webm', 'audio/mp4', 'audio/wav', 'audio/ogg'];
374
+ for (const type of types) {
375
+ if (MediaRecorder.isTypeSupported(type)) {
376
+ return type;
377
+ }
378
+ }
379
+ return 'audio/webm';
380
+ }
381
+
382
+ /**
383
+ * Get file extension for MIME type.
384
+ */
385
+ function getFileExtension(mimeType) {
386
+ if (mimeType.includes('webm')) return 'webm';
387
+ if (mimeType.includes('mp4')) return 'mp4';
388
+ if (mimeType.includes('wav')) return 'wav';
389
+ if (mimeType.includes('ogg')) return 'ogg';
390
+ return 'webm';
391
+ }
392
+ var _default = exports.default = MLXSpeechRecognizer;
393
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_react","_interopRequireWildcard","require","_reactNative","_logger","_interopRequireDefault","e","__esModule","default","t","WeakMap","r","n","o","i","f","__proto__","has","get","set","hasOwnProperty","call","Object","defineProperty","getOwnPropertyDescriptor","MLXSpeechRecognizer","shouldListen","speechStatusCallback","speechResultCallback","language","onError","baseUrl","mode","modelName","minAudioSize","isRecording","setIsRecording","useState","mediaRecorderRef","useRef","audioChunksRef","streamRef","abortControllerRef","startTimeRef","processingRef","lastSentTranscriptRef","startRecording","useCallback","logger","debug","current","isSafari","test","navigator","userAgent","stream","mediaDevices","getUserMedia","audio","echoCancellation","noiseSuppression","autoGainControl","sampleRate","getAudioTracks","length","forEach","track","index","label","enabled","readyState","mimeType","getSupportedMimeType","mediaRecorder","MediaRecorder","error","warn","ondataavailable","event","data","size","push","onstart","state","onstop","getTracks","stop","info","processAudioChunks","onerror","Error","start","Date","now","err","String","stopRecording","elapsed","minDuration","delay","setTimeout","requestData","audioBlob","Blob","type","extension","getFileExtension","audioFile","File","endpoint","AbortController","formData","FormData","append","response","fetch","method","body","signal","ok","status","result","json","transcript","text","trim","handleStreamingResponse","reader","getReader","decoder","TextDecoder","buffer","lastTranscript","sawDone","done","read","decode","value","lines","split","pop","line","startsWith","slice","parsedResult","JSON","parse","Promise","res","releaseLock","useEffect","abort","createElement","View","exports","types","isTypeSupported","includes","_default"],"sources":["../../src/speech-recognition/speech-recognition-mlx.tsx"],"sourcesContent":["/**\n * MLX Whisper-based Speech Recognition using a local FastAPI server.\n * Simplified version with manual start/stop control.\n */\n\nimport React, { useState, useRef, useCallback, useEffect } from 'react';\nimport { View } from 'react-native';\nimport logger from '../logger';\n\nexport type LocalWhisperMode = 'oneshot' | 'sse' | 'realtime';\n\nexport interface MLXSpeechRecognizerProps {\n  shouldListen: boolean;\n  speechStatusCallback: (status: boolean, transcript: string) => void;\n  speechResultCallback: (result: string) => void;\n  language?: string;\n  onError?: (error: Error) => void;\n  baseUrl?: string;\n  mode?: LocalWhisperMode;\n  modelName?: string;\n  minAudioSize?: number;\n}\n\nexport const MLXSpeechRecognizer: React.FC<MLXSpeechRecognizerProps> = ({\n  shouldListen,\n  speechStatusCallback,\n  speechResultCallback,\n  language = 'en',\n  onError,\n  baseUrl = 'http://127.0.0.1:8000',\n  mode,\n  modelName = 'large-v3-turbo',\n  minAudioSize = 1000,\n}) => {\n  const [isRecording, setIsRecording] = useState(false);\n  const mediaRecorderRef = useRef<MediaRecorder | null>(null);\n  const audioChunksRef = useRef<Blob[]>([]);\n  const streamRef = useRef<MediaStream | null>(null);\n  const abortControllerRef = useRef<AbortController | null>(null);\n  const startTimeRef = useRef<number>(0);\n  // Prevent concurrent uploads / duplicate calls\n  const processingRef = useRef<boolean>(false);\n  // Track last sent transcript for finalization\n  const lastSentTranscriptRef = useRef<string>('');\n\n  /**\n   * Start recording audio from microphone.\n   */\n  const startRecording = useCallback(async () => {\n    try {\n      logger.debug('Starting MLX Whisper recording...');\n      logger.debug(`shouldListen=${shouldListen}, isRecording=${isRecording}, processing=${processingRef.current}`);\n\n      // Browser-specific audio constraints\n      const isSafari = /^((?!chrome|android).)*safari/i.test(navigator.userAgent);\n      logger.debug(`Browser detection: Safari = ${isSafari}`);\n\n      const stream = await navigator.mediaDevices.getUserMedia({\n        audio: isSafari ? {\n          // Safari prefers simpler constraints\n          echoCancellation: false,\n          noiseSuppression: false,\n          autoGainControl: false,\n        } : {\n          echoCancellation: true,\n          noiseSuppression: true,\n          autoGainControl: true,\n          sampleRate: 16000,\n        },\n      });\n\n      logger.debug(`Audio stream tracks: ${stream.getAudioTracks().length}`);\n      stream.getAudioTracks().forEach((track, index) => {\n        logger.debug(`Track ${index}: ${track.label}, enabled: ${track.enabled}, readyState: ${track.readyState}`);\n      });\n\n      streamRef.current = stream;\n      audioChunksRef.current = [];\n\n      // Create MediaRecorder with supported format\n      const mimeType = getSupportedMimeType();\n      logger.debug(`Using MIME type: ${mimeType}`);\n\n      // Try creating MediaRecorder with minimal options first\n      let mediaRecorder: MediaRecorder;\n      try {\n        mediaRecorder = new MediaRecorder(stream, { mimeType });\n        logger.debug('MediaRecorder created with minimal options');\n      } catch (error) {\n        logger.warn('MediaRecorder constructor failed with minimal options, trying fallback:', error);\n        // Fallback without mimeType\n        mediaRecorder = new MediaRecorder(stream);\n        logger.debug('MediaRecorder created with fallback options');\n      }\n\n      mediaRecorderRef.current = mediaRecorder;\n\n      // Handle audio data\n      mediaRecorder.ondataavailable = (event: BlobEvent) => {\n        if (event.data.size > 0) {\n          audioChunksRef.current.push(event.data);\n          logger.debug(`Audio chunk: ${event.data.size} bytes`);\n        } else {\n          logger.debug('ondataavailable with empty chunk');\n        }\n      };\n\n      // Handle start\n      mediaRecorder.onstart = () => {\n        logger.debug('MediaRecorder started, state:', mediaRecorder.state);\n      };\n\n      // Handle recording stop\n      mediaRecorder.onstop = async () => {\n        logger.debug('Recording stopped, processing audio...');\n        setIsRecording(false);\n\n        // Clean up stream (close mic immediately)\n        if (streamRef.current) {\n          streamRef.current.getTracks().forEach(track => track.stop());\n          streamRef.current = null;\n        }\n\n        logger.info('🛑 MLX Whisper recording stopped');\n        logger.debug(`Collected ${audioChunksRef.current.length} audio chunks`);\n        await processAudioChunks();\n      };\n\n      // Handle errors\n      mediaRecorder.onerror = (event: Event) => {\n        logger.error('MediaRecorder error:', event);\n        setIsRecording(false);\n        const error = new Error('Recording failed');\n        if (onError) onError(error);\n        speechStatusCallback(false, '');\n      };\n\n      // Start recording with a small timeslice to ensure data collection\n      mediaRecorder.start(100); // Collect data every 100ms\n      logger.debug('MediaRecorder.start() called');\n\n      startTimeRef.current = Date.now();\n      setIsRecording(true);\n      speechStatusCallback(true, '');\n      logger.info('🎤 MLX Whisper recording started');\n\n    } catch (error) {\n      logger.error('Failed to start recording:', error);\n      const err = error instanceof Error ? error : new Error(String(error));\n      if (onError) onError(err);\n      speechStatusCallback(false, '');\n    }\n  }, [speechStatusCallback, onError]);\n\n  /**\n   * Stop recording and process audio.\n   */\n  const stopRecording = useCallback(() => {\n    const elapsed = Date.now() - startTimeRef.current;\n    const minDuration = 1000; // Minimum 1 second recording\n\n    if (elapsed < minDuration) {\n      // Schedule stop after minimum duration\n      const delay = minDuration - elapsed;\n      setTimeout(() => {\n        if (mediaRecorderRef.current && mediaRecorderRef.current.state === 'recording') {\n          try {\n            // Flush any pending data\n            (mediaRecorderRef.current as any).requestData?.();\n          } catch (e) {}\n          mediaRecorderRef.current.stop();\n        }\n      }, delay);\n      logger.debug(`Recording too short (${elapsed}ms), delaying stop by ${delay}ms`);\n    } else {\n      // Stop immediately if minimum duration met\n      if (mediaRecorderRef.current && mediaRecorderRef.current.state === 'recording') {\n        try { (mediaRecorderRef.current as any).requestData?.(); } catch (e) {}\n        mediaRecorderRef.current.stop();\n      }\n    }\n\n    // Note: setIsRecording(false) moved to onstop handler\n  }, []);\n\n  /**\n   * Process collected audio chunks and send to server.\n   */\n  const processAudioChunks = useCallback(async () => {\n    if (audioChunksRef.current.length === 0) {\n      logger.warn('No audio chunks to process');\n      speechStatusCallback(false, '');\n      return;\n    }\n\n    // Prevent duplicate uploads\n    if (processingRef.current) {\n      logger.warn('Already processing an utterance, skipping duplicate upload');\n      return;\n    }\n\n    processingRef.current = true;\n    // Give feedback to UI that we are uploading/transcribing\n    try {\n      // indicate remote processing started (no fake interim text)\n      speechStatusCallback(true, '');\n      // Create audio blob\n      const mimeType = getSupportedMimeType();\n      const audioBlob = new Blob(audioChunksRef.current, { type: mimeType });\n\n      logger.debug(`Audio blob size: ${audioBlob.size} bytes`);\n\n      if (audioBlob.size < minAudioSize) {\n        logger.warn(`Audio too short (${audioBlob.size} < ${minAudioSize} bytes), skipping`);\n        speechStatusCallback(false, '');\n        speechResultCallback('');\n        return;\n      }\n\n      // Create file for upload\n      const extension = getFileExtension(mimeType);\n      const audioFile = new File([audioBlob], `audio.${extension}`, { type: mimeType });\n\n      // Use single REST endpoint for oneshot and SSE\n      const endpoint = `${baseUrl}/v1/audio/transcriptions`;\n\n      // Prepare request\n      abortControllerRef.current = new AbortController();\n      const formData = new FormData();\n      formData.append('file', audioFile);\n      formData.append('model', modelName);\n      formData.append('response_format', 'json');\n      if (language) formData.append('language', language);\n\n      if (mode === 'oneshot') {\n        formData.append('stream', 'false');\n      } else {\n        formData.append('stream', 'true');\n      }\n\n      logger.info(`📤 Sending ${audioBlob.size} bytes to ${endpoint}`);\n      logger.info(`🗣️ Mode: ${mode}, Model: ${modelName}, endpoint: ${endpoint}`);\n\n      // Send to server\n      const response = await fetch(endpoint, {\n        method: 'POST',\n        body: formData,\n        signal: abortControllerRef.current.signal,\n      });\n\n      if (!response.ok) {\n        throw new Error(`Server error: ${response.status}`);\n      }\n\n      // Handle response\n      if (mode === 'oneshot') {\n        const result = await response.json();\n        const transcript = result?.text?.trim() || '';\n        logger.debug('Received transcription response:', result);\n        logger.debug(`Transcription text: \"${transcript}\"`);\n        // Deliver transcript first, then signal end so ModalityProvider processes it once\n        lastSentTranscriptRef.current = transcript;\n        speechResultCallback(transcript);\n        speechStatusCallback(false, '');\n      } else {\n        // Handle streaming response (SSE)\n        await handleStreamingResponse(response);\n      }\n\n    } catch (error) {\n      logger.error('Transcription error:', error);\n      const err = error instanceof Error ? error : new Error(String(error));\n      if (onError) onError(err);\n      speechStatusCallback(false, '');\n      speechResultCallback('');\n    } finally {\n      audioChunksRef.current = [];\n      if (abortControllerRef.current) {\n        abortControllerRef.current = null;\n      }\n      processingRef.current = false;\n    }\n  }, [speechStatusCallback, speechResultCallback, baseUrl, mode, modelName, language, minAudioSize, onError]);\n\n  /**\n   * Handle streaming transcription response.\n   */\n  const handleStreamingResponse = useCallback(async (response: Response) => {\n    const reader = response.body?.getReader();\n    if (!reader) throw new Error('No response reader');\n\n    const decoder = new TextDecoder();\n    let buffer = '';\n    let lastTranscript = '';\n    let sawDone = false;\n\n    try {\n      let done = false;\n      while (!done) {\n        const result = await reader.read();\n        done = result.done;\n\n        if (done) break;\n\n        buffer += decoder.decode(result.value, { stream: true });\n        const lines = buffer.split('\\n');\n        buffer = lines.pop() || '';\n\n        for (const line of lines) {\n          if (line.startsWith('data: ')) {\n            const data = line.slice(6).trim();\n            if (data === '[DONE]') {\n              sawDone = true;\n              break;\n            }\n\n            try {\n              const parsedResult = JSON.parse(data);\n              const transcript = parsedResult?.text?.trim();\n              if (transcript) {\n                logger.info(`📝 Streaming: \"${transcript}\"`);\n                lastTranscript = transcript;\n                // deliver partial to UI immediately\n                speechResultCallback(transcript);\n              }\n            } catch (e) {\n              // Ignore parse errors\n            }\n          }\n        }\n\n        if (sawDone) break;\n      }\n    } finally {\n      // Small delay to ensure UI has applied last partial\n      try { await new Promise((res) => setTimeout(res, 60)); } catch (e) {}\n\n      if (lastTranscript) {\n        logger.debug(`Final streaming transcript before end: \"${lastTranscript}\"`);\n        try { speechResultCallback(lastTranscript); } catch (e) {}\n      }\n\n      // Signal stream end so ModalityProvider triggers the Genie interpreter once\n      try { speechStatusCallback(false, ''); } catch (e) {}\n\n      reader.releaseLock();\n    }\n  }, [speechStatusCallback, speechResultCallback]);\n\n  /**\n   * React to shouldListen changes.\n   */\n  useEffect(() => {\n    logger.debug(`shouldListen prop changed: ${shouldListen}`);\n    // Avoid starting a new recording while previous utterance is processing\n    if (shouldListen && !isRecording && !processingRef.current) {\n      startRecording();\n    } else if (!shouldListen && isRecording) {\n      stopRecording();\n    }\n  }, [shouldListen, isRecording, startRecording, stopRecording]);\n\n  /**\n   * Cleanup on unmount.\n   */\n  useEffect(() => {\n    return () => {\n      if (mediaRecorderRef.current && mediaRecorderRef.current.state === 'recording') {\n        mediaRecorderRef.current.stop();\n      }\n      if (streamRef.current) {\n        streamRef.current.getTracks().forEach(track => track.stop());\n      }\n      if (abortControllerRef.current) {\n        abortControllerRef.current.abort();\n      }\n    };\n  }, []);\n\n  return <View />;\n};\n\n/**\n * Get supported MIME type for MediaRecorder.\n */\nfunction getSupportedMimeType(): string {\n  const types = ['audio/webm', 'audio/mp4', 'audio/wav', 'audio/ogg'];\n\n  for (const type of types) {\n    if (MediaRecorder.isTypeSupported(type)) {\n      return type;\n    }\n  }\n\n  return 'audio/webm';\n}\n\n/**\n * Get file extension for MIME type.\n */\nfunction getFileExtension(mimeType: string): string {\n  if (mimeType.includes('webm')) return 'webm';\n  if (mimeType.includes('mp4')) return 'mp4';\n  if (mimeType.includes('wav')) return 'wav';\n  if (mimeType.includes('ogg')) return 'ogg';\n  return 'webm';\n}\n\nexport default MLXSpeechRecognizer;\n"],"mappings":"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","ignoreList":[]}
@@ -0,0 +1,24 @@
1
+ import React from 'react';
2
+ export interface NativeSpeechRecognizerProps {
3
+ shouldListen: boolean;
4
+ speechStatusCallback: (status: boolean, transcript: string) => void;
5
+ speechResultCallback: (result: string) => void;
6
+ speechTranslationCallback?: (translation: string) => void;
7
+ clientSecret: string;
8
+ groqApiBaseUrl?: string;
9
+ groqTranscriptionEndpoint?: string;
10
+ stream?: boolean;
11
+ language?: string;
12
+ onError?: (error: Error) => void;
13
+ minAudioSize?: number;
14
+ }
15
+ export declare const NativeSpeechRecognizer: React.FC<NativeSpeechRecognizerProps>;
16
+ export declare function isNativeSpeechAvailable(): boolean;
17
+ export declare function getNativeSpeechCapabilities(): {
18
+ available: boolean;
19
+ platform: "ios" | "android" | "windows" | "macos" | "web";
20
+ supportsVAD: boolean;
21
+ supportsAudioLevels: boolean;
22
+ requiresExpoAudio: boolean;
23
+ backend: string;
24
+ };