@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,21 @@
1
+ import React from 'react';
2
+ export interface GroqSpeechRecognizerProps {
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 GroqSpeechRecognizer: React.FC<GroqSpeechRecognizerProps>;
16
+ export declare function isGroqSpeechAvailable(): boolean;
17
+ export declare function getGroqSpeechCapabilities(): {
18
+ available: boolean;
19
+ supportedMimeTypes: string[];
20
+ platform: "ios" | "android" | "windows" | "macos" | "web";
21
+ };
@@ -0,0 +1,409 @@
1
+ "use strict";
2
+
3
+ Object.defineProperty(exports, "__esModule", {
4
+ value: true
5
+ });
6
+ exports.GroqSpeechRecognizer = void 0;
7
+ exports.getGroqSpeechCapabilities = getGroqSpeechCapabilities;
8
+ exports.isGroqSpeechAvailable = isGroqSpeechAvailable;
9
+ var _react = _interopRequireWildcard(require("react"));
10
+ var _reactNative = require("react-native");
11
+ var _logger = require("../logger");
12
+ var _groqTranscription = require("./utils/groq-transcription");
13
+ 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); }
14
+ /**
15
+ * Groq-based Speech Recognition using Whisper API
16
+ * Provides real-time streaming transcription with interim results
17
+ * Streams audio chunks to Groq as they accumulate, displaying interim transcripts
18
+ */
19
+
20
+ /**
21
+ * Groq-based speech recognizer using Whisper-large-v3 model
22
+ * Supports real-time streaming and batch transcription
23
+ */
24
+ const GroqSpeechRecognizer = ({
25
+ shouldListen,
26
+ speechStatusCallback,
27
+ speechResultCallback,
28
+ speechTranslationCallback,
29
+ clientSecret,
30
+ groqApiBaseUrl,
31
+ groqTranscriptionEndpoint,
32
+ stream = true,
33
+ language = 'en',
34
+ onError,
35
+ minAudioSize = 1000
36
+ }) => {
37
+ const [isRecording, setIsRecording] = (0, _react.useState)(false);
38
+ const mediaRecorderRef = (0, _react.useRef)(null);
39
+ const audioChunksRef = (0, _react.useRef)([]);
40
+ const streamRef = (0, _react.useRef)(null);
41
+ const startTimeRef = (0, _react.useRef)(0);
42
+ const lastMimeTypeRef = (0, _react.useRef)('audio/webm');
43
+
44
+ // Real-time streaming state
45
+ const streamAbortRef = (0, _react.useRef)(null);
46
+ const transcriptionStateRef = (0, _react.useRef)({
47
+ isStreaming: false,
48
+ lastSendTime: 0,
49
+ lastInterim: ''
50
+ });
51
+ const finalTranscriptRef = (0, _react.useRef)('');
52
+ const MIN_CHUNK_SIZE = 500; // bytes (before first send)
53
+ const SEND_INTERVAL = 1000; // ms (debounce interval)
54
+
55
+ /**
56
+ * Start audio recording (matching MLX approach)
57
+ */
58
+ const startRecording = (0, _react.useCallback)(async () => {
59
+ try {
60
+ _logger.logger.debug('Starting Groq recording...');
61
+
62
+ // Browser-specific audio constraints
63
+ const isSafari = /^((?!chrome|android).)*safari/i.test(navigator.userAgent);
64
+ _logger.logger.debug(`Browser detection: Safari = ${isSafari}`);
65
+ const stream = await navigator.mediaDevices.getUserMedia({
66
+ audio: isSafari ? {
67
+ // Safari prefers simpler constraints
68
+ echoCancellation: false,
69
+ noiseSuppression: false,
70
+ autoGainControl: false
71
+ } : {
72
+ echoCancellation: true,
73
+ noiseSuppression: true,
74
+ autoGainControl: true,
75
+ sampleRate: 16000
76
+ }
77
+ });
78
+ _logger.logger.debug(`Audio stream tracks: ${stream.getAudioTracks().length}`);
79
+ stream.getAudioTracks().forEach((track, index) => {
80
+ _logger.logger.debug(`Track ${index}: ${track.label}, enabled: ${track.enabled}, readyState: ${track.readyState}`);
81
+ });
82
+ streamRef.current = stream;
83
+ audioChunksRef.current = [];
84
+ finalTranscriptRef.current = '';
85
+
86
+ // Create MediaRecorder with supported format
87
+ const mimeType = getSupportedMimeType();
88
+ _logger.logger.debug(`Using MIME type: ${mimeType}`);
89
+ lastMimeTypeRef.current = mimeType;
90
+
91
+ // Try creating MediaRecorder with minimal options first
92
+ let mediaRecorder;
93
+ try {
94
+ mediaRecorder = new MediaRecorder(stream, {
95
+ mimeType
96
+ });
97
+ _logger.logger.debug('MediaRecorder created with minimal options');
98
+ } catch (error) {
99
+ _logger.logger.warn('MediaRecorder constructor failed with minimal options, trying fallback:', error);
100
+ // Fallback without mimeType
101
+ mediaRecorder = new MediaRecorder(stream);
102
+ _logger.logger.debug('MediaRecorder created with fallback options');
103
+ }
104
+ mediaRecorderRef.current = mediaRecorder;
105
+
106
+ // Handle audio data - trigger streaming checks
107
+ mediaRecorder.ondataavailable = event => {
108
+ if (event.data.size > 0) {
109
+ audioChunksRef.current.push(event.data);
110
+ _logger.logger.debug(`Audio chunk: ${event.data.size} bytes (total: ${audioChunksRef.current.length} chunks)`);
111
+ // Check if we should stream
112
+ checkAndStreamAudio();
113
+ }
114
+ };
115
+
116
+ // Handle start
117
+ mediaRecorder.onstart = () => {
118
+ _logger.logger.debug('MediaRecorder started, state:', mediaRecorder.state);
119
+ };
120
+
121
+ // Handle recording stop
122
+ mediaRecorder.onstop = async () => {
123
+ _logger.logger.debug('Recording stopped, cleaning up stream...');
124
+ setIsRecording(false);
125
+
126
+ // Cancel active stream
127
+ if (streamAbortRef.current) {
128
+ streamAbortRef.current.abort();
129
+ streamAbortRef.current = null;
130
+ }
131
+
132
+ // Clean up stream (close mic immediately)
133
+ if (streamRef.current) {
134
+ streamRef.current.getTracks().forEach(track => track.stop());
135
+ streamRef.current = null;
136
+ }
137
+ _logger.logger.info('🛑 Groq recording stopped');
138
+ const finalTranscript = (finalTranscriptRef.current || transcriptionStateRef.current.lastInterim || '').trim();
139
+ let parseTranscript = finalTranscript;
140
+ try {
141
+ if (audioChunksRef.current.length > 0 && language && !language.startsWith('en')) {
142
+ const mimeType = lastMimeTypeRef.current || getSupportedMimeType();
143
+ const audioBlob = new Blob(audioChunksRef.current, {
144
+ type: mimeType
145
+ });
146
+ if (audioBlob.size > 0) {
147
+ const translated = await (0, _groqTranscription.translateGroqAudio)({
148
+ apiKey: clientSecret,
149
+ audioBlob,
150
+ mimeType,
151
+ language,
152
+ endpoint: resolveGroqTranslationEndpoint(groqApiBaseUrl, groqTranscriptionEndpoint)
153
+ });
154
+ if (translated) {
155
+ if (speechTranslationCallback) {
156
+ speechTranslationCallback(translated);
157
+ }
158
+ parseTranscript = translated;
159
+ }
160
+ }
161
+ }
162
+ } catch (error) {
163
+ _logger.logger.warn('Groq translation failed:', error);
164
+ } finally {
165
+ audioChunksRef.current = [];
166
+ }
167
+
168
+ // Stream will naturally end, just signal end of recording
169
+ speechStatusCallback(false, '');
170
+ if (parseTranscript) {
171
+ speechResultCallback(parseTranscript);
172
+ }
173
+ };
174
+
175
+ // Handle errors
176
+ mediaRecorder.onerror = event => {
177
+ _logger.logger.error('MediaRecorder error:', event);
178
+ setIsRecording(false);
179
+ const error = new Error('Recording failed');
180
+ if (onError) onError(error);
181
+ speechStatusCallback(false, '');
182
+ };
183
+
184
+ // Start recording with 100ms timeslice (matching MLX)
185
+ mediaRecorder.start(100);
186
+ _logger.logger.debug('MediaRecorder.start(100) called');
187
+ startTimeRef.current = Date.now();
188
+ setIsRecording(true);
189
+ // Don't send empty interim on start - wait for actual transcripts
190
+ // speechStatusCallback(true, '');
191
+ _logger.logger.info('🎤 Groq recording started');
192
+ } catch (error) {
193
+ _logger.logger.error('Failed to start recording:', error);
194
+ const err = error instanceof Error ? error : new Error(String(error));
195
+ if (onError) onError(err);
196
+ speechStatusCallback(false, '');
197
+ }
198
+ }, [speechStatusCallback, speechResultCallback, speechTranslationCallback, clientSecret, groqApiBaseUrl, groqTranscriptionEndpoint, language, onError]);
199
+
200
+ /**
201
+ * Stop audio recording
202
+ */
203
+ const stopRecording = (0, _react.useCallback)(() => {
204
+ _logger.logger.debug('Stop recording called');
205
+
206
+ // Stop all audio tracks immediately
207
+ if (streamRef.current) {
208
+ streamRef.current.getTracks().forEach(track => {
209
+ track.stop();
210
+ });
211
+ streamRef.current = null;
212
+ }
213
+ if (mediaRecorderRef.current && mediaRecorderRef.current.state === 'recording') {
214
+ mediaRecorderRef.current.stop();
215
+ }
216
+ }, []);
217
+
218
+ /**
219
+ * Stream audio to Groq and handle SSE responses
220
+ * Sends accumulated chunks and streams interim transcripts
221
+ */
222
+ const streamAudio = (0, _react.useCallback)(async audioBlob => {
223
+ if (transcriptionStateRef.current.isStreaming) {
224
+ _logger.logger.warn('Already streaming, skipping');
225
+ return;
226
+ }
227
+ transcriptionStateRef.current.isStreaming = true;
228
+ transcriptionStateRef.current.lastSendTime = Date.now();
229
+ try {
230
+ const mimeType = getSupportedMimeType();
231
+ _logger.logger.info(`📤 Streaming ${audioBlob.size} bytes to Groq (interim mode)...`);
232
+ streamAbortRef.current = new AbortController();
233
+ const state = transcriptionStateRef.current;
234
+ await (0, _groqTranscription.streamGroqTranscription)({
235
+ apiKey: clientSecret,
236
+ audioBlob,
237
+ mimeType,
238
+ language,
239
+ responseFormat: 'verbose_json',
240
+ stream,
241
+ endpoint: resolveGroqTranscriptionEndpoint(groqApiBaseUrl, groqTranscriptionEndpoint),
242
+ signal: streamAbortRef.current.signal,
243
+ onInterim: interimText => {
244
+ if (!interimText || interimText === state.lastInterim) return;
245
+ state.lastInterim = interimText;
246
+ finalTranscriptRef.current = interimText;
247
+ _logger.logger.info(`📝 Interim: "${interimText}"`);
248
+ speechStatusCallback(true, interimText);
249
+ }
250
+ });
251
+ } catch (error) {
252
+ if (error instanceof Error && error.name !== 'AbortError') {
253
+ _logger.logger.error('❌ Stream error:', error);
254
+ if (onError) onError(error);
255
+ }
256
+ } finally {
257
+ transcriptionStateRef.current.isStreaming = false;
258
+ }
259
+ }, [clientSecret, groqApiBaseUrl, groqTranscriptionEndpoint, language, onError, stream]);
260
+
261
+ /**
262
+ * Check if audio threshold met and stream to Groq
263
+ * Implements debounced streaming with minimum chunk size
264
+ */
265
+ const checkAndStreamAudio = (0, _react.useCallback)(() => {
266
+ const mimeType = getSupportedMimeType();
267
+ const audioBlob = new Blob(audioChunksRef.current, {
268
+ type: mimeType
269
+ });
270
+ const now = Date.now();
271
+ const state = transcriptionStateRef.current;
272
+
273
+ // Check: minimum chunk size reached?
274
+ if (audioBlob.size < MIN_CHUNK_SIZE) {
275
+ return;
276
+ }
277
+
278
+ // Check: already streaming?
279
+ if (state.isStreaming) {
280
+ return;
281
+ }
282
+
283
+ // Check: debounce interval respected?
284
+ if (now - state.lastSendTime < SEND_INTERVAL) {
285
+ return;
286
+ }
287
+
288
+ // All checks passed - start streaming
289
+ streamAudio(audioBlob);
290
+ }, [streamAudio]);
291
+
292
+ // SSE handling moved to shared helper (groq-transcription.ts)
293
+
294
+ /**
295
+ * Process collected audio chunks and send to Groq for transcription (matching MLX)
296
+ * CRITICAL: Guard with processingRef to prevent duplicate uploads
297
+ */
298
+ const processAudioChunks = (0, _react.useCallback)(async () => {
299
+ if (audioChunksRef.current.length === 0) {
300
+ _logger.logger.warn('No audio chunks to process');
301
+ return;
302
+ }
303
+
304
+ // In streaming mode, audio already processed via stream
305
+ _logger.logger.info('✅ Streaming completed');
306
+ audioChunksRef.current = [];
307
+ }, []);
308
+
309
+ /**
310
+ * Handle shouldListen prop changes
311
+ */
312
+ (0, _react.useEffect)(() => {
313
+ _logger.logger.debug(`shouldListen changed: ${shouldListen}, isRecording: ${isRecording}`);
314
+ if (shouldListen && !isRecording) {
315
+ _logger.logger.debug('✅ Starting recording');
316
+ startRecording();
317
+ } else if (!shouldListen && isRecording) {
318
+ _logger.logger.debug('❌ Stopping recording');
319
+ stopRecording();
320
+ }
321
+ }, [shouldListen, isRecording, startRecording, stopRecording]);
322
+
323
+ /**
324
+ * Cleanup on unmount
325
+ */
326
+ (0, _react.useEffect)(() => {
327
+ return () => {
328
+ if (streamAbortRef.current) {
329
+ streamAbortRef.current.abort();
330
+ }
331
+ if (mediaRecorderRef.current && mediaRecorderRef.current.state === 'recording') {
332
+ mediaRecorderRef.current.stop();
333
+ }
334
+ if (streamRef.current) {
335
+ streamRef.current.getTracks().forEach(track => track.stop());
336
+ }
337
+ };
338
+ }, []);
339
+ return /*#__PURE__*/_react.default.createElement(_reactNative.View, null);
340
+ };
341
+
342
+ /**
343
+ * Get supported MIME type for MediaRecorder
344
+ */
345
+ exports.GroqSpeechRecognizer = GroqSpeechRecognizer;
346
+ function getSupportedMimeType() {
347
+ const types = ['audio/webm;codecs=opus', 'audio/webm', 'audio/ogg;codecs=opus', 'audio/mp4', 'audio/wav'];
348
+ for (const type of types) {
349
+ if (MediaRecorder.isTypeSupported(type)) {
350
+ return type;
351
+ }
352
+ }
353
+
354
+ // Fallback
355
+ return 'audio/webm';
356
+ }
357
+ function resolveGroqTranscriptionEndpoint(baseUrl, overrideEndpoint) {
358
+ if (overrideEndpoint) return overrideEndpoint;
359
+ if (!baseUrl) return undefined;
360
+ const trimmed = baseUrl.replace(/\/+$/, '');
361
+ if (trimmed.endsWith('/openai/v1')) {
362
+ return `${trimmed}/audio/transcriptions`;
363
+ }
364
+ return `${trimmed}/openai/v1/audio/transcriptions`;
365
+ }
366
+ function resolveGroqTranslationEndpoint(baseUrl, overrideEndpoint) {
367
+ if (overrideEndpoint) {
368
+ return overrideEndpoint.includes('/audio/transcriptions') ? overrideEndpoint.replace('/audio/transcriptions', '/audio/translations') : overrideEndpoint;
369
+ }
370
+ if (!baseUrl) return undefined;
371
+ const trimmed = baseUrl.replace(/\/+$/, '');
372
+ if (trimmed.endsWith('/openai/v1')) {
373
+ return `${trimmed}/audio/translations`;
374
+ }
375
+ return `${trimmed}/openai/v1/audio/translations`;
376
+ }
377
+
378
+ /**
379
+ * Get file extension from MIME type
380
+ */
381
+ /**
382
+ * Check if Groq speech recognition is available
383
+ */
384
+ function isGroqSpeechAvailable() {
385
+ // Check if running in browser
386
+ if (typeof window === 'undefined') {
387
+ return false;
388
+ }
389
+
390
+ // Check for required APIs
391
+ const hasMediaDevices = navigator?.mediaDevices?.getUserMedia !== undefined;
392
+ const hasMediaRecorder = typeof MediaRecorder !== 'undefined';
393
+ return hasMediaDevices && hasMediaRecorder;
394
+ }
395
+
396
+ /**
397
+ * Get browser capabilities for Groq speech
398
+ */
399
+ function getGroqSpeechCapabilities() {
400
+ return {
401
+ available: isGroqSpeechAvailable(),
402
+ supportedMimeTypes: ['audio/webm;codecs=opus', 'audio/webm', 'audio/ogg;codecs=opus', 'audio/mp4', 'audio/wav'].filter(type => {
403
+ if (typeof MediaRecorder === 'undefined') return false;
404
+ return MediaRecorder.isTypeSupported(type);
405
+ }),
406
+ platform: _reactNative.Platform.OS
407
+ };
408
+ }
409
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_react","_interopRequireWildcard","require","_reactNative","_logger","_groqTranscription","e","t","WeakMap","r","n","__esModule","o","i","f","__proto__","default","has","get","set","hasOwnProperty","call","Object","defineProperty","getOwnPropertyDescriptor","GroqSpeechRecognizer","shouldListen","speechStatusCallback","speechResultCallback","speechTranslationCallback","clientSecret","groqApiBaseUrl","groqTranscriptionEndpoint","stream","language","onError","minAudioSize","isRecording","setIsRecording","useState","mediaRecorderRef","useRef","audioChunksRef","streamRef","startTimeRef","lastMimeTypeRef","streamAbortRef","transcriptionStateRef","isStreaming","lastSendTime","lastInterim","finalTranscriptRef","MIN_CHUNK_SIZE","SEND_INTERVAL","startRecording","useCallback","logger","debug","isSafari","test","navigator","userAgent","mediaDevices","getUserMedia","audio","echoCancellation","noiseSuppression","autoGainControl","sampleRate","getAudioTracks","length","forEach","track","index","label","enabled","readyState","current","mimeType","getSupportedMimeType","mediaRecorder","MediaRecorder","error","warn","ondataavailable","event","data","size","push","checkAndStreamAudio","onstart","state","onstop","abort","getTracks","stop","info","finalTranscript","trim","parseTranscript","startsWith","audioBlob","Blob","type","translated","translateGroqAudio","apiKey","endpoint","resolveGroqTranslationEndpoint","onerror","Error","start","Date","now","err","String","stopRecording","streamAudio","AbortController","streamGroqTranscription","responseFormat","resolveGroqTranscriptionEndpoint","signal","onInterim","interimText","name","processAudioChunks","useEffect","createElement","View","exports","types","isTypeSupported","baseUrl","overrideEndpoint","undefined","trimmed","replace","endsWith","includes","isGroqSpeechAvailable","window","hasMediaDevices","hasMediaRecorder","getGroqSpeechCapabilities","available","supportedMimeTypes","filter","platform","Platform","OS"],"sources":["../../src/speech-recognition/speech-recognition-groq.tsx"],"sourcesContent":["/**\n * Groq-based Speech Recognition using Whisper API\n * Provides real-time streaming transcription with interim results\n * Streams audio chunks to Groq as they accumulate, displaying interim transcripts\n */\n\nimport React, { useEffect, useState, useRef, useCallback } from 'react';\nimport { View, Platform } from 'react-native';\nimport { logger } from '../logger';\nimport { streamGroqTranscription, translateGroqAudio } from './utils/groq-transcription';\n\nexport interface GroqSpeechRecognizerProps {\n  shouldListen: boolean;\n  speechStatusCallback: (status: boolean, transcript: string) => void;\n  speechResultCallback: (result: string) => void;\n  speechTranslationCallback?: (translation: string) => void;\n  clientSecret: string;\n  groqApiBaseUrl?: string;\n  groqTranscriptionEndpoint?: string;\n  stream?: boolean;\n  language?: string;\n  onError?: (error: Error) => void;\n  minAudioSize?: number; // Minimum audio size in bytes (default 1000)\n}\n\n/**\n * Groq-based speech recognizer using Whisper-large-v3 model\n * Supports real-time streaming and batch transcription\n */\nexport const GroqSpeechRecognizer: React.FC<GroqSpeechRecognizerProps> = ({\n  shouldListen,\n  speechStatusCallback,\n  speechResultCallback,\n  speechTranslationCallback,\n  clientSecret,\n  groqApiBaseUrl,\n  groqTranscriptionEndpoint,\n  stream = true,\n  language = 'en',\n  onError,\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 startTimeRef = useRef<number>(0);\n  const lastMimeTypeRef = useRef<string>('audio/webm');\n  \n  // Real-time streaming state\n  const streamAbortRef = useRef<AbortController | null>(null);\n  const transcriptionStateRef = useRef({\n    isStreaming: false,\n    lastSendTime: 0,\n    lastInterim: '',\n  });\n  const finalTranscriptRef = useRef<string>('');\n  const MIN_CHUNK_SIZE = 500; // bytes (before first send)\n  const SEND_INTERVAL = 1000; // ms (debounce interval)\n\n  /**\n   * Start audio recording (matching MLX approach)\n   */\n  const startRecording = useCallback(async () => {\n    try {\n      logger.debug('Starting Groq recording...');\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      finalTranscriptRef.current = '';\n\n      // Create MediaRecorder with supported format\n      const mimeType = getSupportedMimeType();\n      logger.debug(`Using MIME type: ${mimeType}`);\n      lastMimeTypeRef.current = 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 - trigger streaming checks\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 (total: ${audioChunksRef.current.length} chunks)`);\n          // Check if we should stream\n          checkAndStreamAudio();\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, cleaning up stream...');\n        setIsRecording(false);\n        \n        // Cancel active stream\n        if (streamAbortRef.current) {\n          streamAbortRef.current.abort();\n          streamAbortRef.current = null;\n        }\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('🛑 Groq recording stopped');\n\n        const finalTranscript = (\n          finalTranscriptRef.current ||\n          transcriptionStateRef.current.lastInterim ||\n          ''\n        ).trim();\n        let parseTranscript = finalTranscript;\n\n        try {\n          if (\n            audioChunksRef.current.length > 0 &&\n            language &&\n            !language.startsWith('en')\n          ) {\n            const mimeType = lastMimeTypeRef.current || getSupportedMimeType();\n            const audioBlob = new Blob(audioChunksRef.current, { type: mimeType });\n            if (audioBlob.size > 0) {\n              const translated = await translateGroqAudio({\n                apiKey: clientSecret,\n                audioBlob,\n                mimeType,\n                language,\n                endpoint: resolveGroqTranslationEndpoint(\n                  groqApiBaseUrl,\n                  groqTranscriptionEndpoint,\n                ),\n              });\n              if (translated) {\n                if (speechTranslationCallback) {\n                  speechTranslationCallback(translated);\n                }\n                parseTranscript = translated;\n              }\n            }\n          }\n        } catch (error) {\n          logger.warn('Groq translation failed:', error);\n        } finally {\n          audioChunksRef.current = [];\n        }\n\n        // Stream will naturally end, just signal end of recording\n        speechStatusCallback(false, '');\n        if (parseTranscript) {\n          speechResultCallback(parseTranscript);\n        }\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 100ms timeslice (matching MLX)\n      mediaRecorder.start(100);\n      logger.debug('MediaRecorder.start(100) called');\n\n      startTimeRef.current = Date.now();\n      setIsRecording(true);\n      // Don't send empty interim on start - wait for actual transcripts\n      // speechStatusCallback(true, '');\n      logger.info('🎤 Groq recording started');\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  }, [\n    speechStatusCallback,\n    speechResultCallback,\n    speechTranslationCallback,\n    clientSecret,\n    groqApiBaseUrl,\n    groqTranscriptionEndpoint,\n    language,\n    onError,\n  ]);\n\n  /**\n   * Stop audio recording\n   */\n  const stopRecording = useCallback(() => {\n    logger.debug('Stop recording called');\n    \n    // Stop all audio tracks immediately\n    if (streamRef.current) {\n      streamRef.current.getTracks().forEach((track) => {\n        track.stop();\n      });\n      streamRef.current = null;\n    }\n    \n    if (mediaRecorderRef.current && mediaRecorderRef.current.state === 'recording') {\n      mediaRecorderRef.current.stop();\n    }\n  }, []);\n\n  /**\n   * Stream audio to Groq and handle SSE responses\n   * Sends accumulated chunks and streams interim transcripts\n   */\n  const streamAudio = useCallback(async (audioBlob: Blob) => {\n    if (transcriptionStateRef.current.isStreaming) {\n      logger.warn('Already streaming, skipping');\n      return;\n    }\n\n    transcriptionStateRef.current.isStreaming = true;\n    transcriptionStateRef.current.lastSendTime = Date.now();\n\n    try {\n      const mimeType = getSupportedMimeType();\n      logger.info(`📤 Streaming ${audioBlob.size} bytes to Groq (interim mode)...`);\n\n      streamAbortRef.current = new AbortController();\n      const state = transcriptionStateRef.current;\n\n      await streamGroqTranscription({\n        apiKey: clientSecret,\n        audioBlob,\n        mimeType,\n        language,\n        responseFormat: 'verbose_json',\n        stream,\n        endpoint: resolveGroqTranscriptionEndpoint(\n          groqApiBaseUrl,\n          groqTranscriptionEndpoint,\n        ),\n        signal: streamAbortRef.current.signal,\n        onInterim: (interimText) => {\n          if (!interimText || interimText === state.lastInterim) return;\n          state.lastInterim = interimText;\n          finalTranscriptRef.current = interimText;\n          logger.info(`📝 Interim: \"${interimText}\"`);\n          speechStatusCallback(true, interimText);\n        },\n      });\n    } catch (error) {\n      if (error instanceof Error && error.name !== 'AbortError') {\n        logger.error('❌ Stream error:', error);\n        if (onError) onError(error);\n      }\n    } finally {\n      transcriptionStateRef.current.isStreaming = false;\n    }\n  }, [clientSecret, groqApiBaseUrl, groqTranscriptionEndpoint, language, onError, stream]);\n\n  /**\n   * Check if audio threshold met and stream to Groq\n   * Implements debounced streaming with minimum chunk size\n   */\n  const checkAndStreamAudio = useCallback(() => {\n    const mimeType = getSupportedMimeType();\n    const audioBlob = new Blob(audioChunksRef.current, { type: mimeType });\n    const now = Date.now();\n\n    const state = transcriptionStateRef.current;\n\n    // Check: minimum chunk size reached?\n    if (audioBlob.size < MIN_CHUNK_SIZE) {\n      return;\n    }\n\n    // Check: already streaming?\n    if (state.isStreaming) {\n      return;\n    }\n\n    // Check: debounce interval respected?\n    if (now - state.lastSendTime < SEND_INTERVAL) {\n      return;\n    }\n\n    // All checks passed - start streaming\n    streamAudio(audioBlob);\n  }, [streamAudio]);\n\n  // SSE handling moved to shared helper (groq-transcription.ts)\n\n  /**\n   * Process collected audio chunks and send to Groq for transcription (matching MLX)\n   * CRITICAL: Guard with processingRef to prevent duplicate uploads\n   */\n  const processAudioChunks = useCallback(async () => {\n    if (audioChunksRef.current.length === 0) {\n      logger.warn('No audio chunks to process');\n      return;\n    }\n\n    // In streaming mode, audio already processed via stream\n    logger.info('✅ Streaming completed');\n    audioChunksRef.current = [];\n  }, []);\n\n  /**\n   * Handle shouldListen prop changes\n   */\n  useEffect(() => {\n    logger.debug(`shouldListen changed: ${shouldListen}, isRecording: ${isRecording}`);\n\n    if (shouldListen && !isRecording) {\n      logger.debug('✅ Starting recording');\n      startRecording();\n    } else if (!shouldListen && isRecording) {\n      logger.debug('❌ Stopping recording');\n      stopRecording();\n    }\n  }, [shouldListen, isRecording, startRecording, stopRecording]);\n\n  /**\n   * Cleanup on unmount\n   */\n  useEffect(() => {\n    return () => {\n      if (streamAbortRef.current) {\n        streamAbortRef.current.abort();\n      }\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    };\n  }, []);\n\n  return <View />;\n};\n\n/**\n * Get supported MIME type for MediaRecorder\n */\nfunction getSupportedMimeType(): string {\n  const types = [\n    'audio/webm;codecs=opus',\n    'audio/webm',\n    'audio/ogg;codecs=opus',\n    'audio/mp4',\n    'audio/wav',\n  ];\n\n  for (const type of types) {\n    if (MediaRecorder.isTypeSupported(type)) {\n      return type;\n    }\n  }\n\n  // Fallback\n  return 'audio/webm';\n}\n\nfunction resolveGroqTranscriptionEndpoint(\n  baseUrl?: string,\n  overrideEndpoint?: string,\n): string | undefined {\n  if (overrideEndpoint) return overrideEndpoint;\n  if (!baseUrl) return undefined;\n  const trimmed = baseUrl.replace(/\\/+$/, '');\n  if (trimmed.endsWith('/openai/v1')) {\n    return `${trimmed}/audio/transcriptions`;\n  }\n  return `${trimmed}/openai/v1/audio/transcriptions`;\n}\n\nfunction resolveGroqTranslationEndpoint(\n  baseUrl?: string,\n  overrideEndpoint?: string,\n): string | undefined {\n  if (overrideEndpoint) {\n    return overrideEndpoint.includes('/audio/transcriptions')\n      ? overrideEndpoint.replace('/audio/transcriptions', '/audio/translations')\n      : overrideEndpoint;\n  }\n  if (!baseUrl) return undefined;\n  const trimmed = baseUrl.replace(/\\/+$/, '');\n  if (trimmed.endsWith('/openai/v1')) {\n    return `${trimmed}/audio/translations`;\n  }\n  return `${trimmed}/openai/v1/audio/translations`;\n}\n\n/**\n * Get file extension from MIME type\n */\n/**\n * Check if Groq speech recognition is available\n */\nexport function isGroqSpeechAvailable(): boolean {\n  // Check if running in browser\n  if (typeof window === 'undefined') {\n    return false;\n  }\n\n  // Check for required APIs\n  const hasMediaDevices = \n    navigator?.mediaDevices?.getUserMedia !== undefined;\n  const hasMediaRecorder = typeof MediaRecorder !== 'undefined';\n\n  return hasMediaDevices && hasMediaRecorder;\n}\n\n/**\n * Get browser capabilities for Groq speech\n */\nexport function getGroqSpeechCapabilities() {\n  return {\n    available: isGroqSpeechAvailable(),\n    supportedMimeTypes: [\n      'audio/webm;codecs=opus',\n      'audio/webm',\n      'audio/ogg;codecs=opus',\n      'audio/mp4',\n      'audio/wav',\n    ].filter((type) => {\n      if (typeof MediaRecorder === 'undefined') return false;\n      return MediaRecorder.isTypeSupported(type);\n    }),\n    platform: Platform.OS,\n  };\n}\n"],"mappings":"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","ignoreList":[]}
@@ -0,0 +1,15 @@
1
+ import React from 'react';
2
+ export type LocalWhisperMode = 'oneshot' | 'sse' | 'realtime';
3
+ export interface MLXSpeechRecognizerProps {
4
+ shouldListen: boolean;
5
+ speechStatusCallback: (status: boolean, transcript: string) => void;
6
+ speechResultCallback: (result: string) => void;
7
+ language?: string;
8
+ onError?: (error: Error) => void;
9
+ baseUrl?: string;
10
+ mode?: LocalWhisperMode;
11
+ modelName?: string;
12
+ minAudioSize?: number;
13
+ }
14
+ export declare const MLXSpeechRecognizer: React.FC<MLXSpeechRecognizerProps>;
15
+ export default MLXSpeechRecognizer;