@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,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
|
+
};
|