@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.
- package/.env.example +22 -0
- package/.github/workflows/publish.yml +20 -0
- package/LICENSE.txt +201 -0
- package/README.md +621 -0
- package/babel.config.js +29 -0
- package/dist/adapters/__tests__/expo-router-adapter.test.d.ts +1 -0
- package/dist/adapters/expo-router-adapter.d.ts +16 -0
- package/dist/adapters/expo-router-adapter.js +521 -0
- package/dist/adapters/navigation-adapter.d.ts +20 -0
- package/dist/adapters/navigation-adapter.js +137 -0
- package/dist/audio-visualizer.d.ts +14 -0
- package/dist/audio-visualizer.js +123 -0
- package/dist/current-selection.d.ts +27 -0
- package/dist/current-selection.js +94 -0
- package/dist/errors.d.ts +19 -0
- package/dist/errors.js +37 -0
- package/dist/genie/DateTime.d.ts +66 -0
- package/dist/genie/DateTime.js +399 -0
- package/dist/genie/TimeDelta.d.ts +35 -0
- package/dist/genie/TimeDelta.js +169 -0
- package/dist/genie-view-wrapper.d.ts +1 -0
- package/dist/genie-view-wrapper.js +377 -0
- package/dist/hooks/__tests__/useSpeechRecognition.test.d.ts +1 -0
- package/dist/hooks/useSpeechRecognition.d.ts +28 -0
- package/dist/hooks/useSpeechRecognition.js +118 -0
- package/dist/index.d.ts +30 -0
- package/dist/index.js +469 -0
- package/dist/logger.d.ts +23 -0
- package/dist/logger.js +597 -0
- package/dist/logger.remote.test.d.ts +0 -0
- package/dist/modality-provider-v2.d.ts +28 -0
- package/dist/modality-provider-v2.js +1321 -0
- package/dist/modality-provider.d.ts +22 -0
- package/dist/modality-provider.js +373 -0
- package/dist/native-visibility.d.ts +28 -0
- package/dist/native-visibility.js +50 -0
- package/dist/platform/VoiceRecognitionBar.d.ts +17 -0
- package/dist/platform/VoiceRecognitionBar.js +332 -0
- package/dist/platform/components.d.ts +32 -0
- package/dist/platform/components.js +351 -0
- package/dist/platform/events.d.ts +31 -0
- package/dist/platform/events.js +274 -0
- package/dist/platform/index.d.ts +3 -0
- package/dist/platform/index.js +39 -0
- package/dist/platform/types.d.ts +79 -0
- package/dist/platform/types.js +97 -0
- package/dist/react-decorators.d.ts +87 -0
- package/dist/react-decorators.js +368 -0
- package/dist/shared-store.d.ts +74 -0
- package/dist/shared-store.js +589 -0
- package/dist/speech-recognition/__tests__/speech-recognition-groq-transport.test.d.ts +1 -0
- package/dist/speech-recognition/__tests__/speech-recognition-native.test.d.ts +1 -0
- package/dist/speech-recognition/__tests__/speech-recognition-openai-native.test.d.ts +1 -0
- package/dist/speech-recognition/__tests__/speech-recognition-openai.test.d.ts +1 -0
- package/dist/speech-recognition/__tests__/speech-recognition-unified-import.test.d.ts +0 -0
- package/dist/speech-recognition/__tests__/speech-recognition-unified.test.d.ts +1 -0
- package/dist/speech-recognition/speech-recognition-groq.d.ts +21 -0
- package/dist/speech-recognition/speech-recognition-groq.js +409 -0
- package/dist/speech-recognition/speech-recognition-mlx.d.ts +15 -0
- package/dist/speech-recognition/speech-recognition-mlx.js +393 -0
- package/dist/speech-recognition/speech-recognition-native.d.ts +24 -0
- package/dist/speech-recognition/speech-recognition-native.js +632 -0
- package/dist/speech-recognition/speech-recognition-openai-native.d.ts +40 -0
- package/dist/speech-recognition/speech-recognition-openai-native.js +653 -0
- package/dist/speech-recognition/speech-recognition-openai.d.ts +39 -0
- package/dist/speech-recognition/speech-recognition-openai.js +718 -0
- package/dist/speech-recognition/speech-recognition-unified.d.ts +93 -0
- package/dist/speech-recognition/speech-recognition-unified.js +589 -0
- package/dist/speech-recognition/utils/groq-transcription.d.ts +41 -0
- package/dist/speech-recognition/utils/groq-transcription.js +382 -0
- package/dist/speech-recognition.d.ts +7 -0
- package/dist/speech-recognition.js +61 -0
- package/dist/voice-pipeline-telemetry.d.ts +26 -0
- package/dist/voice-pipeline-telemetry.js +15 -0
- package/garrix82-reactgenie-lib-1.3.0.tgz +0 -0
- package/metro/index.js +3 -0
- package/metro/with-genie-registry.js +47 -0
- package/package.json +111 -0
- package/scripts/dry-run.js +23 -0
- package/scripts/generate-genie-registry.js +278 -0
- package/scripts/log-file-test.js +51 -0
- package/scripts/parse.js +26 -0
- package/scripts/prompt.js +19 -0
- package/scripts/set-script.js +200 -0
- package/tsconfig.json +36 -0
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
File without changes
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -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;
|