@tekyzinc/stt-component 0.1.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/README.md +279 -0
- package/dist/index.cjs +482 -0
- package/dist/index.cjs.map +1 -0
- package/dist/index.d.cts +223 -0
- package/dist/index.d.ts +223 -0
- package/dist/index.js +445 -0
- package/dist/index.js.map +1 -0
- package/package.json +56 -0
package/dist/index.cjs
ADDED
|
@@ -0,0 +1,482 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __defProp = Object.defineProperty;
|
|
3
|
+
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
|
|
4
|
+
var __getOwnPropNames = Object.getOwnPropertyNames;
|
|
5
|
+
var __hasOwnProp = Object.prototype.hasOwnProperty;
|
|
6
|
+
var __export = (target, all) => {
|
|
7
|
+
for (var name in all)
|
|
8
|
+
__defProp(target, name, { get: all[name], enumerable: true });
|
|
9
|
+
};
|
|
10
|
+
var __copyProps = (to, from, except, desc) => {
|
|
11
|
+
if (from && typeof from === "object" || typeof from === "function") {
|
|
12
|
+
for (let key of __getOwnPropNames(from))
|
|
13
|
+
if (!__hasOwnProp.call(to, key) && key !== except)
|
|
14
|
+
__defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
|
|
15
|
+
}
|
|
16
|
+
return to;
|
|
17
|
+
};
|
|
18
|
+
var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
|
|
19
|
+
|
|
20
|
+
// src/index.ts
|
|
21
|
+
var index_exports = {};
|
|
22
|
+
__export(index_exports, {
|
|
23
|
+
CorrectionOrchestrator: () => CorrectionOrchestrator,
|
|
24
|
+
DEFAULT_STT_CONFIG: () => DEFAULT_STT_CONFIG,
|
|
25
|
+
STTEngine: () => STTEngine,
|
|
26
|
+
TypedEventEmitter: () => TypedEventEmitter,
|
|
27
|
+
WorkerManager: () => WorkerManager,
|
|
28
|
+
resampleAudio: () => resampleAudio,
|
|
29
|
+
resolveConfig: () => resolveConfig,
|
|
30
|
+
snapshotAudio: () => snapshotAudio,
|
|
31
|
+
startCapture: () => startCapture,
|
|
32
|
+
stopCapture: () => stopCapture
|
|
33
|
+
});
|
|
34
|
+
module.exports = __toCommonJS(index_exports);
|
|
35
|
+
|
|
36
|
+
// src/types.ts
|
|
37
|
+
var DEFAULT_STT_CONFIG = {
|
|
38
|
+
model: "tiny",
|
|
39
|
+
backend: "auto",
|
|
40
|
+
language: "en",
|
|
41
|
+
dtype: "q4",
|
|
42
|
+
correction: {
|
|
43
|
+
enabled: true,
|
|
44
|
+
pauseThreshold: 3e3,
|
|
45
|
+
forcedInterval: 5e3
|
|
46
|
+
},
|
|
47
|
+
chunking: {
|
|
48
|
+
chunkLengthS: 30,
|
|
49
|
+
strideLengthS: 5
|
|
50
|
+
}
|
|
51
|
+
};
|
|
52
|
+
function resolveConfig(config) {
|
|
53
|
+
return {
|
|
54
|
+
model: config?.model ?? DEFAULT_STT_CONFIG.model,
|
|
55
|
+
backend: config?.backend ?? DEFAULT_STT_CONFIG.backend,
|
|
56
|
+
language: config?.language ?? DEFAULT_STT_CONFIG.language,
|
|
57
|
+
dtype: config?.dtype ?? DEFAULT_STT_CONFIG.dtype,
|
|
58
|
+
correction: {
|
|
59
|
+
enabled: config?.correction?.enabled ?? DEFAULT_STT_CONFIG.correction.enabled,
|
|
60
|
+
pauseThreshold: config?.correction?.pauseThreshold ?? DEFAULT_STT_CONFIG.correction.pauseThreshold,
|
|
61
|
+
forcedInterval: config?.correction?.forcedInterval ?? DEFAULT_STT_CONFIG.correction.forcedInterval
|
|
62
|
+
},
|
|
63
|
+
chunking: {
|
|
64
|
+
chunkLengthS: config?.chunking?.chunkLengthS ?? DEFAULT_STT_CONFIG.chunking.chunkLengthS,
|
|
65
|
+
strideLengthS: config?.chunking?.strideLengthS ?? DEFAULT_STT_CONFIG.chunking.strideLengthS
|
|
66
|
+
}
|
|
67
|
+
};
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
// src/event-emitter.ts
|
|
71
|
+
var TypedEventEmitter = class {
|
|
72
|
+
listeners = /* @__PURE__ */ new Map();
|
|
73
|
+
/** Subscribe to an event. */
|
|
74
|
+
on(event, listener) {
|
|
75
|
+
let set = this.listeners.get(event);
|
|
76
|
+
if (!set) {
|
|
77
|
+
set = /* @__PURE__ */ new Set();
|
|
78
|
+
this.listeners.set(event, set);
|
|
79
|
+
}
|
|
80
|
+
set.add(listener);
|
|
81
|
+
}
|
|
82
|
+
/** Unsubscribe a specific listener. No-op if not registered. */
|
|
83
|
+
off(event, listener) {
|
|
84
|
+
this.listeners.get(event)?.delete(listener);
|
|
85
|
+
}
|
|
86
|
+
/** Emit an event, calling all registered listeners in insertion order. */
|
|
87
|
+
emit(event, ...args) {
|
|
88
|
+
const set = this.listeners.get(event);
|
|
89
|
+
if (!set) return;
|
|
90
|
+
for (const listener of set) {
|
|
91
|
+
listener(...args);
|
|
92
|
+
}
|
|
93
|
+
}
|
|
94
|
+
/** Remove all listeners, optionally for a single event. */
|
|
95
|
+
removeAllListeners(event) {
|
|
96
|
+
if (event !== void 0) {
|
|
97
|
+
this.listeners.delete(event);
|
|
98
|
+
} else {
|
|
99
|
+
this.listeners.clear();
|
|
100
|
+
}
|
|
101
|
+
}
|
|
102
|
+
};
|
|
103
|
+
|
|
104
|
+
// src/audio-capture.ts
|
|
105
|
+
var TARGET_SAMPLE_RATE = 16e3;
|
|
106
|
+
async function startCapture() {
|
|
107
|
+
const stream = await navigator.mediaDevices.getUserMedia({
|
|
108
|
+
audio: { channelCount: 1 }
|
|
109
|
+
});
|
|
110
|
+
const audioCtx = new AudioContext();
|
|
111
|
+
if (audioCtx.state === "suspended") {
|
|
112
|
+
await audioCtx.resume();
|
|
113
|
+
}
|
|
114
|
+
const source = audioCtx.createMediaStreamSource(stream);
|
|
115
|
+
const samples = [];
|
|
116
|
+
const processor = audioCtx.createScriptProcessor(4096, 1, 1);
|
|
117
|
+
processor.onaudioprocess = (e) => {
|
|
118
|
+
samples.push(new Float32Array(e.inputBuffer.getChannelData(0)));
|
|
119
|
+
};
|
|
120
|
+
const silencer = audioCtx.createGain();
|
|
121
|
+
silencer.gain.value = 0;
|
|
122
|
+
source.connect(processor);
|
|
123
|
+
processor.connect(silencer);
|
|
124
|
+
silencer.connect(audioCtx.destination);
|
|
125
|
+
return { audioCtx, stream, samples, _processor: processor };
|
|
126
|
+
}
|
|
127
|
+
function snapshotAudio(capture) {
|
|
128
|
+
return [...capture.samples];
|
|
129
|
+
}
|
|
130
|
+
async function resampleAudio(samples, nativeSr) {
|
|
131
|
+
const totalLength = samples.reduce((sum, s) => sum + s.length, 0);
|
|
132
|
+
if (totalLength === 0) return new Float32Array(0);
|
|
133
|
+
const fullAudio = new Float32Array(totalLength);
|
|
134
|
+
let offset = 0;
|
|
135
|
+
for (const s of samples) {
|
|
136
|
+
fullAudio.set(s, offset);
|
|
137
|
+
offset += s.length;
|
|
138
|
+
}
|
|
139
|
+
if (nativeSr === TARGET_SAMPLE_RATE) return fullAudio;
|
|
140
|
+
const duration = fullAudio.length / nativeSr;
|
|
141
|
+
const outLength = Math.round(duration * TARGET_SAMPLE_RATE);
|
|
142
|
+
const offline = new OfflineAudioContext(1, outLength, TARGET_SAMPLE_RATE);
|
|
143
|
+
const buffer = offline.createBuffer(1, fullAudio.length, nativeSr);
|
|
144
|
+
buffer.getChannelData(0).set(fullAudio);
|
|
145
|
+
const src = offline.createBufferSource();
|
|
146
|
+
src.buffer = buffer;
|
|
147
|
+
src.connect(offline.destination);
|
|
148
|
+
src.start(0);
|
|
149
|
+
const resampled = await offline.startRendering();
|
|
150
|
+
return resampled.getChannelData(0);
|
|
151
|
+
}
|
|
152
|
+
async function stopCapture(capture) {
|
|
153
|
+
const { audioCtx, stream, samples, _processor } = capture;
|
|
154
|
+
try {
|
|
155
|
+
_processor.disconnect();
|
|
156
|
+
} catch {
|
|
157
|
+
}
|
|
158
|
+
for (const track of stream.getTracks()) {
|
|
159
|
+
track.stop();
|
|
160
|
+
}
|
|
161
|
+
const nativeSr = audioCtx.sampleRate;
|
|
162
|
+
await audioCtx.close();
|
|
163
|
+
return resampleAudio(samples, nativeSr);
|
|
164
|
+
}
|
|
165
|
+
|
|
166
|
+
// src/worker-manager.ts
|
|
167
|
+
var import_meta = {};
|
|
168
|
+
var WorkerManager = class extends TypedEventEmitter {
|
|
169
|
+
worker = null;
|
|
170
|
+
transcribeResolve = null;
|
|
171
|
+
modelReadyResolve = null;
|
|
172
|
+
modelReadyReject = null;
|
|
173
|
+
/** Spawn the Web Worker. Must be called before loadModel/transcribe. */
|
|
174
|
+
spawn(workerUrl) {
|
|
175
|
+
if (this.worker) return;
|
|
176
|
+
const url = workerUrl ?? new URL("./whisper-worker.js", import_meta.url);
|
|
177
|
+
this.worker = new Worker(url, { type: "module" });
|
|
178
|
+
this.worker.onmessage = (e) => {
|
|
179
|
+
this.handleMessage(e.data);
|
|
180
|
+
};
|
|
181
|
+
this.worker.onerror = (e) => {
|
|
182
|
+
this.emit("error", e.message ?? "Worker error");
|
|
183
|
+
};
|
|
184
|
+
}
|
|
185
|
+
/** Load the Whisper model in the worker. Resolves when ready. */
|
|
186
|
+
async loadModel(config) {
|
|
187
|
+
if (!this.worker) throw new Error("Worker not spawned");
|
|
188
|
+
return new Promise((resolve, reject) => {
|
|
189
|
+
this.modelReadyResolve = resolve;
|
|
190
|
+
this.modelReadyReject = reject;
|
|
191
|
+
this.worker.postMessage({
|
|
192
|
+
type: "load",
|
|
193
|
+
config: {
|
|
194
|
+
model: config.model,
|
|
195
|
+
backend: config.backend,
|
|
196
|
+
language: config.language,
|
|
197
|
+
dtype: config.dtype,
|
|
198
|
+
chunkLengthS: config.chunking.chunkLengthS,
|
|
199
|
+
strideLengthS: config.chunking.strideLengthS
|
|
200
|
+
}
|
|
201
|
+
});
|
|
202
|
+
});
|
|
203
|
+
}
|
|
204
|
+
/** Send audio to the worker for transcription. Resolves with text. */
|
|
205
|
+
async transcribe(audio) {
|
|
206
|
+
if (!this.worker) throw new Error("Worker not spawned");
|
|
207
|
+
if (audio.length === 0) return "";
|
|
208
|
+
return new Promise((resolve) => {
|
|
209
|
+
this.transcribeResolve = resolve;
|
|
210
|
+
this.worker.postMessage({ type: "transcribe", audio }, [audio.buffer]);
|
|
211
|
+
});
|
|
212
|
+
}
|
|
213
|
+
/** Cancel any in-flight transcription. */
|
|
214
|
+
cancel() {
|
|
215
|
+
this.worker?.postMessage({ type: "cancel" });
|
|
216
|
+
if (this.transcribeResolve) {
|
|
217
|
+
this.transcribeResolve("");
|
|
218
|
+
this.transcribeResolve = null;
|
|
219
|
+
}
|
|
220
|
+
}
|
|
221
|
+
/** Terminate the worker and release resources. */
|
|
222
|
+
destroy() {
|
|
223
|
+
this.cancel();
|
|
224
|
+
this.worker?.terminate();
|
|
225
|
+
this.worker = null;
|
|
226
|
+
this.removeAllListeners();
|
|
227
|
+
}
|
|
228
|
+
handleMessage(msg) {
|
|
229
|
+
switch (msg.type) {
|
|
230
|
+
case "progress":
|
|
231
|
+
this.emit("progress", msg.data);
|
|
232
|
+
break;
|
|
233
|
+
case "ready":
|
|
234
|
+
this.emit("ready");
|
|
235
|
+
this.modelReadyResolve?.();
|
|
236
|
+
this.modelReadyResolve = null;
|
|
237
|
+
this.modelReadyReject = null;
|
|
238
|
+
break;
|
|
239
|
+
case "result":
|
|
240
|
+
this.emit("result", msg.data);
|
|
241
|
+
this.transcribeResolve?.(msg.data);
|
|
242
|
+
this.transcribeResolve = null;
|
|
243
|
+
break;
|
|
244
|
+
case "error": {
|
|
245
|
+
const errMsg = msg.data;
|
|
246
|
+
this.emit("error", errMsg);
|
|
247
|
+
if (this.modelReadyReject) {
|
|
248
|
+
this.modelReadyReject(new Error(errMsg));
|
|
249
|
+
this.modelReadyResolve = null;
|
|
250
|
+
this.modelReadyReject = null;
|
|
251
|
+
}
|
|
252
|
+
if (this.transcribeResolve) {
|
|
253
|
+
this.transcribeResolve("");
|
|
254
|
+
this.transcribeResolve = null;
|
|
255
|
+
}
|
|
256
|
+
break;
|
|
257
|
+
}
|
|
258
|
+
}
|
|
259
|
+
}
|
|
260
|
+
};
|
|
261
|
+
|
|
262
|
+
// src/correction-orchestrator.ts
|
|
263
|
+
var CorrectionOrchestrator = class {
|
|
264
|
+
forcedTimer = null;
|
|
265
|
+
lastCorrectionTime = 0;
|
|
266
|
+
correctionFn = null;
|
|
267
|
+
config;
|
|
268
|
+
/** Create a new correction orchestrator with the given timing config. */
|
|
269
|
+
constructor(config) {
|
|
270
|
+
this.config = config;
|
|
271
|
+
}
|
|
272
|
+
/** Set the function to call when a correction is triggered. */
|
|
273
|
+
setCorrectionFn(fn) {
|
|
274
|
+
this.correctionFn = fn;
|
|
275
|
+
}
|
|
276
|
+
/** Start the correction orchestrator (begin forced interval timer). */
|
|
277
|
+
start() {
|
|
278
|
+
if (!this.config.enabled) return;
|
|
279
|
+
this.lastCorrectionTime = Date.now();
|
|
280
|
+
this.startForcedTimer();
|
|
281
|
+
}
|
|
282
|
+
/** Stop the orchestrator (clear all timers). */
|
|
283
|
+
stop() {
|
|
284
|
+
this.stopForcedTimer();
|
|
285
|
+
}
|
|
286
|
+
/** Called when a speech pause is detected. Triggers correction if cooldown elapsed. */
|
|
287
|
+
onPauseDetected() {
|
|
288
|
+
if (!this.config.enabled) return;
|
|
289
|
+
const now = Date.now();
|
|
290
|
+
if (now - this.lastCorrectionTime < this.config.pauseThreshold) return;
|
|
291
|
+
this.triggerCorrection();
|
|
292
|
+
}
|
|
293
|
+
/** Force a correction now (resets timer). */
|
|
294
|
+
forceCorrection() {
|
|
295
|
+
this.triggerCorrection();
|
|
296
|
+
}
|
|
297
|
+
triggerCorrection() {
|
|
298
|
+
this.lastCorrectionTime = Date.now();
|
|
299
|
+
this.correctionFn?.();
|
|
300
|
+
this.restartForcedTimer();
|
|
301
|
+
}
|
|
302
|
+
startForcedTimer() {
|
|
303
|
+
this.stopForcedTimer();
|
|
304
|
+
this.forcedTimer = setInterval(() => {
|
|
305
|
+
this.triggerCorrection();
|
|
306
|
+
}, this.config.forcedInterval);
|
|
307
|
+
}
|
|
308
|
+
stopForcedTimer() {
|
|
309
|
+
if (this.forcedTimer) {
|
|
310
|
+
clearInterval(this.forcedTimer);
|
|
311
|
+
this.forcedTimer = null;
|
|
312
|
+
}
|
|
313
|
+
}
|
|
314
|
+
restartForcedTimer() {
|
|
315
|
+
if (this.forcedTimer) {
|
|
316
|
+
this.startForcedTimer();
|
|
317
|
+
}
|
|
318
|
+
}
|
|
319
|
+
};
|
|
320
|
+
|
|
321
|
+
// src/stt-engine.ts
|
|
322
|
+
var STTEngine = class extends TypedEventEmitter {
|
|
323
|
+
config;
|
|
324
|
+
workerManager;
|
|
325
|
+
correctionOrchestrator;
|
|
326
|
+
capture = null;
|
|
327
|
+
state;
|
|
328
|
+
workerUrl;
|
|
329
|
+
/**
|
|
330
|
+
* Create a new STT engine instance.
|
|
331
|
+
* @param config - Optional configuration overrides (model, backend, language, etc.).
|
|
332
|
+
* @param workerUrl - Optional custom URL for the Whisper Web Worker script.
|
|
333
|
+
*/
|
|
334
|
+
constructor(config, workerUrl) {
|
|
335
|
+
super();
|
|
336
|
+
this.config = resolveConfig(config);
|
|
337
|
+
this.workerManager = new WorkerManager();
|
|
338
|
+
this.correctionOrchestrator = new CorrectionOrchestrator(this.config.correction);
|
|
339
|
+
this.workerUrl = workerUrl;
|
|
340
|
+
this.state = {
|
|
341
|
+
status: "idle",
|
|
342
|
+
isModelLoaded: false,
|
|
343
|
+
loadProgress: 0,
|
|
344
|
+
backend: null,
|
|
345
|
+
error: null
|
|
346
|
+
};
|
|
347
|
+
this.correctionOrchestrator.setCorrectionFn(() => {
|
|
348
|
+
this.performCorrection();
|
|
349
|
+
});
|
|
350
|
+
this.setupWorkerListeners();
|
|
351
|
+
}
|
|
352
|
+
/** Initialize the engine: spawn worker and load model. */
|
|
353
|
+
async init() {
|
|
354
|
+
this.updateStatus("loading");
|
|
355
|
+
this.workerManager.spawn(this.workerUrl);
|
|
356
|
+
try {
|
|
357
|
+
await this.workerManager.loadModel(this.config);
|
|
358
|
+
this.state.isModelLoaded = true;
|
|
359
|
+
this.updateStatus("ready");
|
|
360
|
+
} catch (err) {
|
|
361
|
+
this.emitError("MODEL_LOAD_FAILED", err instanceof Error ? err.message : String(err));
|
|
362
|
+
this.updateStatus("idle");
|
|
363
|
+
throw err;
|
|
364
|
+
}
|
|
365
|
+
}
|
|
366
|
+
/** Start recording audio and enable correction cycles. */
|
|
367
|
+
async start() {
|
|
368
|
+
if (this.state.status !== "ready") {
|
|
369
|
+
throw new Error(`Cannot start: engine is "${this.state.status}", expected "ready"`);
|
|
370
|
+
}
|
|
371
|
+
try {
|
|
372
|
+
this.capture = await startCapture();
|
|
373
|
+
this.updateStatus("recording");
|
|
374
|
+
this.correctionOrchestrator.start();
|
|
375
|
+
} catch (err) {
|
|
376
|
+
this.emitError(
|
|
377
|
+
"MIC_DENIED",
|
|
378
|
+
err instanceof Error ? err.message : "Microphone access denied. Check browser permissions."
|
|
379
|
+
);
|
|
380
|
+
}
|
|
381
|
+
}
|
|
382
|
+
/** Stop recording, run final transcription, return text. */
|
|
383
|
+
async stop() {
|
|
384
|
+
if (!this.capture) return "";
|
|
385
|
+
this.correctionOrchestrator.stop();
|
|
386
|
+
this.workerManager.cancel();
|
|
387
|
+
this.updateStatus("processing");
|
|
388
|
+
try {
|
|
389
|
+
const audio = await stopCapture(this.capture);
|
|
390
|
+
this.capture = null;
|
|
391
|
+
if (audio.length === 0) {
|
|
392
|
+
this.updateStatus("ready");
|
|
393
|
+
return "";
|
|
394
|
+
}
|
|
395
|
+
const text = await this.workerManager.transcribe(audio);
|
|
396
|
+
this.emit("correction", text);
|
|
397
|
+
this.updateStatus("ready");
|
|
398
|
+
return text;
|
|
399
|
+
} catch (err) {
|
|
400
|
+
this.emitError(
|
|
401
|
+
"TRANSCRIPTION_FAILED",
|
|
402
|
+
err instanceof Error ? err.message : "Final transcription failed."
|
|
403
|
+
);
|
|
404
|
+
this.updateStatus("ready");
|
|
405
|
+
return "";
|
|
406
|
+
}
|
|
407
|
+
}
|
|
408
|
+
/** Destroy the engine: terminate worker, release all resources. */
|
|
409
|
+
destroy() {
|
|
410
|
+
this.correctionOrchestrator.stop();
|
|
411
|
+
if (this.capture) {
|
|
412
|
+
for (const track of this.capture.stream.getTracks()) {
|
|
413
|
+
track.stop();
|
|
414
|
+
}
|
|
415
|
+
this.capture.audioCtx.close().catch(() => {
|
|
416
|
+
});
|
|
417
|
+
this.capture = null;
|
|
418
|
+
}
|
|
419
|
+
this.workerManager.destroy();
|
|
420
|
+
this.updateStatus("idle");
|
|
421
|
+
this.removeAllListeners();
|
|
422
|
+
}
|
|
423
|
+
/** Get current engine state. */
|
|
424
|
+
getState() {
|
|
425
|
+
return { ...this.state };
|
|
426
|
+
}
|
|
427
|
+
/** Notify the correction orchestrator of a speech pause. */
|
|
428
|
+
notifyPause() {
|
|
429
|
+
this.correctionOrchestrator.onPauseDetected();
|
|
430
|
+
}
|
|
431
|
+
async performCorrection() {
|
|
432
|
+
if (!this.capture || !this.state.isModelLoaded) return;
|
|
433
|
+
this.workerManager.cancel();
|
|
434
|
+
try {
|
|
435
|
+
const samples = snapshotAudio(this.capture);
|
|
436
|
+
const nativeSr = this.capture.audioCtx.sampleRate;
|
|
437
|
+
const audio = await resampleAudio(samples, nativeSr);
|
|
438
|
+
if (audio.length === 0) return;
|
|
439
|
+
const text = await this.workerManager.transcribe(audio);
|
|
440
|
+
if (text.trim() && this.capture) {
|
|
441
|
+
this.emit("correction", text);
|
|
442
|
+
}
|
|
443
|
+
} catch (err) {
|
|
444
|
+
this.emitError(
|
|
445
|
+
"TRANSCRIPTION_FAILED",
|
|
446
|
+
err instanceof Error ? err.message : "Correction transcription failed."
|
|
447
|
+
);
|
|
448
|
+
}
|
|
449
|
+
}
|
|
450
|
+
setupWorkerListeners() {
|
|
451
|
+
this.workerManager.on("progress", (percent) => {
|
|
452
|
+
this.state.loadProgress = percent;
|
|
453
|
+
this.emit("status", { ...this.state });
|
|
454
|
+
});
|
|
455
|
+
this.workerManager.on("error", (message) => {
|
|
456
|
+
this.emitError("WORKER_ERROR", message);
|
|
457
|
+
});
|
|
458
|
+
}
|
|
459
|
+
updateStatus(status) {
|
|
460
|
+
this.state.status = status;
|
|
461
|
+
this.state.error = null;
|
|
462
|
+
this.emit("status", { ...this.state });
|
|
463
|
+
}
|
|
464
|
+
emitError(code, message) {
|
|
465
|
+
this.state.error = message;
|
|
466
|
+
this.emit("error", { code, message });
|
|
467
|
+
}
|
|
468
|
+
};
|
|
469
|
+
// Annotate the CommonJS export names for ESM import in node:
|
|
470
|
+
0 && (module.exports = {
|
|
471
|
+
CorrectionOrchestrator,
|
|
472
|
+
DEFAULT_STT_CONFIG,
|
|
473
|
+
STTEngine,
|
|
474
|
+
TypedEventEmitter,
|
|
475
|
+
WorkerManager,
|
|
476
|
+
resampleAudio,
|
|
477
|
+
resolveConfig,
|
|
478
|
+
snapshotAudio,
|
|
479
|
+
startCapture,
|
|
480
|
+
stopCapture
|
|
481
|
+
});
|
|
482
|
+
//# sourceMappingURL=index.cjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/index.ts","../src/types.ts","../src/event-emitter.ts","../src/audio-capture.ts","../src/worker-manager.ts","../src/correction-orchestrator.ts","../src/stt-engine.ts"],"sourcesContent":["// Public types\nexport type {\n STTModelSize,\n STTBackend,\n STTStatus,\n STTCorrectionConfig,\n STTChunkingConfig,\n STTConfig,\n ResolvedSTTConfig,\n STTState,\n STTError,\n STTEvents,\n AudioCaptureHandle,\n} from './types.js';\n\n// Public values\nexport { DEFAULT_STT_CONFIG, resolveConfig } from './types.js';\n\n// Event emitter\nexport { TypedEventEmitter } from './event-emitter.js';\n\n// Audio capture\nexport { startCapture, snapshotAudio, resampleAudio, stopCapture } from './audio-capture.js';\n\n// Worker manager\nexport { WorkerManager } from './worker-manager.js';\nexport type { WorkerManagerEvents } from './worker-manager.js';\n\n// Correction orchestrator\nexport { CorrectionOrchestrator } from './correction-orchestrator.js';\n\n// STT Engine (main public API)\nexport { STTEngine } from './stt-engine.js';\n","/** Supported Whisper model sizes. */\nexport type STTModelSize = 'tiny' | 'base' | 'small' | 'medium';\n\n/** Supported compute backends. */\nexport type STTBackend = 'webgpu' | 'wasm' | 'auto';\n\n/** Engine lifecycle states. */\nexport type STTStatus = 'idle' | 'loading' | 'ready' | 'recording' | 'processing';\n\n/** Correction timing configuration. */\nexport interface STTCorrectionConfig {\n /** Enable mid-recording Whisper correction. Default: true */\n enabled?: boolean;\n /** Silence duration (ms) before triggering correction. Default: 3000 */\n pauseThreshold?: number;\n /** Maximum interval (ms) between forced corrections. Default: 5000 */\n forcedInterval?: number;\n}\n\n/** Audio chunking configuration for long-form audio. */\nexport interface STTChunkingConfig {\n /** Chunk length in seconds for Whisper processing. Default: 30 */\n chunkLengthS?: number;\n /** Stride length in seconds for overlapping chunks. Default: 5 */\n strideLengthS?: number;\n}\n\n/** Full engine configuration. All fields optional — sensible defaults applied. */\nexport interface STTConfig {\n /** Whisper model size. Default: 'tiny' */\n model?: STTModelSize;\n /** Compute backend preference. Default: 'auto' (WebGPU with WASM fallback) */\n backend?: STTBackend;\n /** Transcription language. Default: 'en' */\n language?: string;\n /** Model quantization dtype. Default: 'q4' */\n dtype?: string;\n /** Mid-recording correction settings. */\n correction?: STTCorrectionConfig;\n /** Audio chunking settings for long-form audio. */\n chunking?: STTChunkingConfig;\n}\n\n/** Resolved configuration with all defaults applied. */\nexport interface ResolvedSTTConfig {\n model: STTModelSize;\n backend: STTBackend;\n language: string;\n dtype: string;\n correction: Required<STTCorrectionConfig>;\n chunking: Required<STTChunkingConfig>;\n}\n\n/** Engine state exposed to consumers via status events. */\nexport interface STTState {\n status: STTStatus;\n isModelLoaded: boolean;\n /** Model download progress (0–100). */\n loadProgress: number;\n /** Active compute backend, or null if not yet determined. */\n backend: 'webgpu' | 'wasm' | null;\n error: string | null;\n}\n\n/** Structured error emitted via the 'error' event. */\nexport interface STTError {\n code: string;\n message: string;\n}\n\n/** Event map for the typed event emitter. */\nexport type STTEvents = {\n /** Streaming interim text during recording. */\n transcript: (text: string) => void;\n /** Whisper-corrected text replacing interim text. */\n correction: (text: string) => void;\n /** Actionable error (mic denied, model fail, transcription fail). */\n error: (error: STTError) => void;\n /** Engine state change. */\n status: (state: STTState) => void;\n};\n\n/** Handle returned by audio capture — used internally. */\nexport interface AudioCaptureHandle {\n audioCtx: AudioContext;\n stream: MediaStream;\n samples: Float32Array[];\n /** Retain reference to prevent GC from stopping audio processing. */\n _processor: ScriptProcessorNode;\n}\n\n/** Message sent from main thread to Whisper worker. */\nexport interface WorkerMessage {\n type: 'load' | 'transcribe' | 'cancel';\n audio?: Float32Array;\n config?: {\n model: string;\n backend: STTBackend;\n language: string;\n dtype: string;\n chunkLengthS: number;\n strideLengthS: number;\n };\n}\n\n/** Response sent from Whisper worker to main thread. */\nexport interface WorkerResponse {\n type: 'progress' | 'ready' | 'result' | 'error';\n data?: unknown;\n}\n\n/** Default configuration values. */\nexport const DEFAULT_STT_CONFIG: ResolvedSTTConfig = {\n model: 'tiny',\n backend: 'auto',\n language: 'en',\n dtype: 'q4',\n correction: {\n enabled: true,\n pauseThreshold: 3_000,\n forcedInterval: 5_000,\n },\n chunking: {\n chunkLengthS: 30,\n strideLengthS: 5,\n },\n};\n\n/** Merge user config with defaults to produce resolved config. */\nexport function resolveConfig(config?: STTConfig): ResolvedSTTConfig {\n return {\n model: config?.model ?? DEFAULT_STT_CONFIG.model,\n backend: config?.backend ?? DEFAULT_STT_CONFIG.backend,\n language: config?.language ?? DEFAULT_STT_CONFIG.language,\n dtype: config?.dtype ?? DEFAULT_STT_CONFIG.dtype,\n correction: {\n enabled: config?.correction?.enabled ?? DEFAULT_STT_CONFIG.correction.enabled,\n pauseThreshold:\n config?.correction?.pauseThreshold ?? DEFAULT_STT_CONFIG.correction.pauseThreshold,\n forcedInterval:\n config?.correction?.forcedInterval ?? DEFAULT_STT_CONFIG.correction.forcedInterval,\n },\n chunking: {\n chunkLengthS: config?.chunking?.chunkLengthS ?? DEFAULT_STT_CONFIG.chunking.chunkLengthS,\n strideLengthS: config?.chunking?.strideLengthS ?? DEFAULT_STT_CONFIG.chunking.strideLengthS,\n },\n };\n}\n","/**\n * A generic, typed event emitter.\n *\n * Type parameter `T` is a map of event names to listener signatures,\n * giving consumers compile-time safety on event names and callback args.\n */\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nexport class TypedEventEmitter<T extends Record<string, (...args: any[]) => void>> {\n private listeners = new Map<keyof T, Set<T[keyof T]>>();\n\n /** Subscribe to an event. */\n on<K extends keyof T>(event: K, listener: T[K]): void {\n let set = this.listeners.get(event);\n if (!set) {\n set = new Set();\n this.listeners.set(event, set);\n }\n set.add(listener as T[keyof T]);\n }\n\n /** Unsubscribe a specific listener. No-op if not registered. */\n off<K extends keyof T>(event: K, listener: T[K]): void {\n this.listeners.get(event)?.delete(listener as T[keyof T]);\n }\n\n /** Emit an event, calling all registered listeners in insertion order. */\n emit<K extends keyof T>(event: K, ...args: Parameters<T[K]>): void {\n const set = this.listeners.get(event);\n if (!set) return;\n for (const listener of set) {\n (listener as (...a: Parameters<T[K]>) => void)(...args);\n }\n }\n\n /** Remove all listeners, optionally for a single event. */\n removeAllListeners(event?: keyof T): void {\n if (event !== undefined) {\n this.listeners.delete(event);\n } else {\n this.listeners.clear();\n }\n }\n}\n","import type { AudioCaptureHandle } from './types.js';\n\nconst TARGET_SAMPLE_RATE = 16_000;\n\n/**\n * Start capturing raw PCM audio from the microphone.\n * Uses ScriptProcessorNode to collect Float32Array samples directly.\n */\nexport async function startCapture(): Promise<AudioCaptureHandle> {\n const stream = await navigator.mediaDevices.getUserMedia({\n audio: { channelCount: 1 },\n });\n const audioCtx = new AudioContext();\n\n // Chrome may suspend AudioContext — must resume within user gesture\n if (audioCtx.state === 'suspended') {\n await audioCtx.resume();\n }\n\n const source = audioCtx.createMediaStreamSource(stream);\n const samples: Float32Array[] = [];\n\n const processor = audioCtx.createScriptProcessor(4096, 1, 1);\n processor.onaudioprocess = (e: AudioProcessingEvent) => {\n samples.push(new Float32Array(e.inputBuffer.getChannelData(0)));\n };\n\n // Connect through a silent gain node so mic audio doesn't play back\n const silencer = audioCtx.createGain();\n silencer.gain.value = 0;\n source.connect(processor);\n processor.connect(silencer);\n silencer.connect(audioCtx.destination);\n\n return { audioCtx, stream, samples, _processor: processor };\n}\n\n/**\n * Copy current audio buffer without stopping capture.\n * Returns a shallow copy of the samples array (each chunk is shared, not cloned).\n */\nexport function snapshotAudio(capture: AudioCaptureHandle): Float32Array[] {\n return [...capture.samples];\n}\n\n/**\n * Concatenate sample chunks and resample to 16kHz for Whisper.\n */\nexport async function resampleAudio(\n samples: Float32Array[],\n nativeSr: number,\n): Promise<Float32Array> {\n const totalLength = samples.reduce((sum, s) => sum + s.length, 0);\n if (totalLength === 0) return new Float32Array(0);\n\n const fullAudio = new Float32Array(totalLength);\n let offset = 0;\n for (const s of samples) {\n fullAudio.set(s, offset);\n offset += s.length;\n }\n\n if (nativeSr === TARGET_SAMPLE_RATE) return fullAudio;\n\n const duration = fullAudio.length / nativeSr;\n const outLength = Math.round(duration * TARGET_SAMPLE_RATE);\n const offline = new OfflineAudioContext(1, outLength, TARGET_SAMPLE_RATE);\n const buffer = offline.createBuffer(1, fullAudio.length, nativeSr);\n buffer.getChannelData(0).set(fullAudio);\n const src = offline.createBufferSource();\n src.buffer = buffer;\n src.connect(offline.destination);\n src.start(0);\n const resampled = await offline.startRendering();\n return resampled.getChannelData(0);\n}\n\n/**\n * Stop capturing and return resampled audio at 16kHz.\n */\nexport async function stopCapture(capture: AudioCaptureHandle): Promise<Float32Array> {\n const { audioCtx, stream, samples, _processor } = capture;\n\n // Disconnect processor to stop capturing\n try {\n _processor.disconnect();\n } catch {\n /* already disconnected */\n }\n\n // Stop microphone tracks\n for (const track of stream.getTracks()) {\n track.stop();\n }\n\n const nativeSr = audioCtx.sampleRate;\n await audioCtx.close();\n\n return resampleAudio(samples, nativeSr);\n}\n","import type { ResolvedSTTConfig, WorkerResponse } from './types.js';\nimport { TypedEventEmitter } from './event-emitter.js';\n\n/** Events emitted by the WorkerManager. */\nexport type WorkerManagerEvents = {\n progress: (percent: number) => void;\n ready: () => void;\n result: (text: string) => void;\n error: (message: string) => void;\n};\n\n/**\n * Manages the Whisper Web Worker lifecycle.\n * Provides typed message passing and a promise-based transcription API.\n */\nexport class WorkerManager extends TypedEventEmitter<WorkerManagerEvents> {\n private worker: Worker | null = null;\n private transcribeResolve: ((text: string) => void) | null = null;\n private modelReadyResolve: (() => void) | null = null;\n private modelReadyReject: ((err: Error) => void) | null = null;\n\n /** Spawn the Web Worker. Must be called before loadModel/transcribe. */\n spawn(workerUrl?: URL): void {\n if (this.worker) return;\n\n const url = workerUrl ?? new URL('./whisper-worker.js', import.meta.url);\n\n this.worker = new Worker(url, { type: 'module' });\n this.worker.onmessage = (e: MessageEvent<WorkerResponse>) => {\n this.handleMessage(e.data);\n };\n this.worker.onerror = (e: ErrorEvent) => {\n this.emit('error', e.message ?? 'Worker error');\n };\n }\n\n /** Load the Whisper model in the worker. Resolves when ready. */\n async loadModel(config: ResolvedSTTConfig): Promise<void> {\n if (!this.worker) throw new Error('Worker not spawned');\n\n return new Promise<void>((resolve, reject) => {\n this.modelReadyResolve = resolve;\n this.modelReadyReject = reject;\n this.worker!.postMessage({\n type: 'load',\n config: {\n model: config.model,\n backend: config.backend,\n language: config.language,\n dtype: config.dtype,\n chunkLengthS: config.chunking.chunkLengthS,\n strideLengthS: config.chunking.strideLengthS,\n },\n });\n });\n }\n\n /** Send audio to the worker for transcription. Resolves with text. */\n async transcribe(audio: Float32Array): Promise<string> {\n if (!this.worker) throw new Error('Worker not spawned');\n if (audio.length === 0) return '';\n\n return new Promise<string>((resolve) => {\n this.transcribeResolve = resolve;\n this.worker!.postMessage({ type: 'transcribe', audio }, [audio.buffer]);\n });\n }\n\n /** Cancel any in-flight transcription. */\n cancel(): void {\n this.worker?.postMessage({ type: 'cancel' });\n if (this.transcribeResolve) {\n this.transcribeResolve('');\n this.transcribeResolve = null;\n }\n }\n\n /** Terminate the worker and release resources. */\n destroy(): void {\n this.cancel();\n this.worker?.terminate();\n this.worker = null;\n this.removeAllListeners();\n }\n\n private handleMessage(msg: WorkerResponse): void {\n switch (msg.type) {\n case 'progress':\n this.emit('progress', msg.data as number);\n break;\n case 'ready':\n this.emit('ready');\n this.modelReadyResolve?.();\n this.modelReadyResolve = null;\n this.modelReadyReject = null;\n break;\n case 'result':\n this.emit('result', msg.data as string);\n this.transcribeResolve?.(msg.data as string);\n this.transcribeResolve = null;\n break;\n case 'error': {\n const errMsg = msg.data as string;\n this.emit('error', errMsg);\n // Reject model load if still pending\n if (this.modelReadyReject) {\n this.modelReadyReject(new Error(errMsg));\n this.modelReadyResolve = null;\n this.modelReadyReject = null;\n }\n // Resolve transcribe with empty string on error\n if (this.transcribeResolve) {\n this.transcribeResolve('');\n this.transcribeResolve = null;\n }\n break;\n }\n }\n }\n}\n","import type { ResolvedSTTConfig } from './types.js';\n\n/**\n * Manages mid-recording correction timing.\n * Two triggers: pause detection and forced interval.\n */\nexport class CorrectionOrchestrator {\n private forcedTimer: ReturnType<typeof setInterval> | null = null;\n private lastCorrectionTime = 0;\n private correctionFn: (() => void) | null = null;\n private config: ResolvedSTTConfig['correction'];\n\n /** Create a new correction orchestrator with the given timing config. */\n constructor(config: ResolvedSTTConfig['correction']) {\n this.config = config;\n }\n\n /** Set the function to call when a correction is triggered. */\n setCorrectionFn(fn: () => void): void {\n this.correctionFn = fn;\n }\n\n /** Start the correction orchestrator (begin forced interval timer). */\n start(): void {\n if (!this.config.enabled) return;\n\n this.lastCorrectionTime = Date.now();\n this.startForcedTimer();\n }\n\n /** Stop the orchestrator (clear all timers). */\n stop(): void {\n this.stopForcedTimer();\n }\n\n /** Called when a speech pause is detected. Triggers correction if cooldown elapsed. */\n onPauseDetected(): void {\n if (!this.config.enabled) return;\n\n const now = Date.now();\n if (now - this.lastCorrectionTime < this.config.pauseThreshold) return;\n\n this.triggerCorrection();\n }\n\n /** Force a correction now (resets timer). */\n forceCorrection(): void {\n this.triggerCorrection();\n }\n\n private triggerCorrection(): void {\n this.lastCorrectionTime = Date.now();\n this.correctionFn?.();\n // Reset forced timer after any correction\n this.restartForcedTimer();\n }\n\n private startForcedTimer(): void {\n this.stopForcedTimer();\n this.forcedTimer = setInterval(() => {\n this.triggerCorrection();\n }, this.config.forcedInterval);\n }\n\n private stopForcedTimer(): void {\n if (this.forcedTimer) {\n clearInterval(this.forcedTimer);\n this.forcedTimer = null;\n }\n }\n\n private restartForcedTimer(): void {\n if (this.forcedTimer) {\n this.startForcedTimer();\n }\n }\n}\n","import type {\n STTConfig,\n STTState,\n STTEvents,\n STTStatus,\n ResolvedSTTConfig,\n AudioCaptureHandle,\n} from './types.js';\nimport { resolveConfig } from './types.js';\nimport { TypedEventEmitter } from './event-emitter.js';\nimport { startCapture, snapshotAudio, resampleAudio, stopCapture } from './audio-capture.js';\nimport { WorkerManager } from './worker-manager.js';\nimport { CorrectionOrchestrator } from './correction-orchestrator.js';\n\n/**\n * Main STT engine — the public API for speech-to-text with Whisper correction.\n *\n * Usage:\n * ```typescript\n * const engine = new STTEngine({ model: 'tiny' });\n * engine.on('transcript', (text) => console.log(text));\n * engine.on('correction', (text) => console.log('corrected:', text));\n * await engine.init();\n * await engine.start();\n * const finalText = await engine.stop();\n * ```\n */\nexport class STTEngine extends TypedEventEmitter<STTEvents> {\n private config: ResolvedSTTConfig;\n private workerManager: WorkerManager;\n private correctionOrchestrator: CorrectionOrchestrator;\n private capture: AudioCaptureHandle | null = null;\n private state: STTState;\n private workerUrl?: URL;\n\n /**\n * Create a new STT engine instance.\n * @param config - Optional configuration overrides (model, backend, language, etc.).\n * @param workerUrl - Optional custom URL for the Whisper Web Worker script.\n */\n constructor(config?: STTConfig, workerUrl?: URL) {\n super();\n this.config = resolveConfig(config);\n this.workerManager = new WorkerManager();\n this.correctionOrchestrator = new CorrectionOrchestrator(this.config.correction);\n this.workerUrl = workerUrl;\n\n this.state = {\n status: 'idle',\n isModelLoaded: false,\n loadProgress: 0,\n backend: null,\n error: null,\n };\n\n this.correctionOrchestrator.setCorrectionFn(() => {\n this.performCorrection();\n });\n\n this.setupWorkerListeners();\n }\n\n /** Initialize the engine: spawn worker and load model. */\n async init(): Promise<void> {\n this.updateStatus('loading');\n this.workerManager.spawn(this.workerUrl);\n\n try {\n await this.workerManager.loadModel(this.config);\n this.state.isModelLoaded = true;\n this.updateStatus('ready');\n } catch (err) {\n this.emitError('MODEL_LOAD_FAILED', err instanceof Error ? err.message : String(err));\n this.updateStatus('idle');\n throw err;\n }\n }\n\n /** Start recording audio and enable correction cycles. */\n async start(): Promise<void> {\n if (this.state.status !== 'ready') {\n throw new Error(`Cannot start: engine is \"${this.state.status}\", expected \"ready\"`);\n }\n\n try {\n this.capture = await startCapture();\n this.updateStatus('recording');\n this.correctionOrchestrator.start();\n } catch (err) {\n this.emitError(\n 'MIC_DENIED',\n err instanceof Error ? err.message : 'Microphone access denied. Check browser permissions.',\n );\n }\n }\n\n /** Stop recording, run final transcription, return text. */\n async stop(): Promise<string> {\n if (!this.capture) return '';\n\n this.correctionOrchestrator.stop();\n this.workerManager.cancel();\n\n this.updateStatus('processing');\n\n try {\n const audio = await stopCapture(this.capture);\n this.capture = null;\n\n if (audio.length === 0) {\n this.updateStatus('ready');\n return '';\n }\n\n const text = await this.workerManager.transcribe(audio);\n this.emit('correction', text);\n this.updateStatus('ready');\n return text;\n } catch (err) {\n this.emitError(\n 'TRANSCRIPTION_FAILED',\n err instanceof Error ? err.message : 'Final transcription failed.',\n );\n this.updateStatus('ready');\n return '';\n }\n }\n\n /** Destroy the engine: terminate worker, release all resources. */\n destroy(): void {\n this.correctionOrchestrator.stop();\n\n if (this.capture) {\n for (const track of this.capture.stream.getTracks()) {\n track.stop();\n }\n this.capture.audioCtx.close().catch(() => {});\n this.capture = null;\n }\n\n this.workerManager.destroy();\n this.updateStatus('idle');\n this.removeAllListeners();\n }\n\n /** Get current engine state. */\n getState(): Readonly<STTState> {\n return { ...this.state };\n }\n\n /** Notify the correction orchestrator of a speech pause. */\n notifyPause(): void {\n this.correctionOrchestrator.onPauseDetected();\n }\n\n private async performCorrection(): Promise<void> {\n if (!this.capture || !this.state.isModelLoaded) return;\n\n this.workerManager.cancel();\n\n try {\n const samples = snapshotAudio(this.capture);\n const nativeSr = this.capture.audioCtx.sampleRate;\n const audio = await resampleAudio(samples, nativeSr);\n\n if (audio.length === 0) return;\n\n const text = await this.workerManager.transcribe(audio);\n if (text.trim() && this.capture) {\n this.emit('correction', text);\n }\n } catch (err) {\n this.emitError(\n 'TRANSCRIPTION_FAILED',\n err instanceof Error ? err.message : 'Correction transcription failed.',\n );\n // Recording continues — error is non-fatal\n }\n }\n\n private setupWorkerListeners(): void {\n this.workerManager.on('progress', (percent) => {\n this.state.loadProgress = percent;\n this.emit('status', { ...this.state });\n });\n\n this.workerManager.on('error', (message) => {\n this.emitError('WORKER_ERROR', message);\n });\n }\n\n private updateStatus(status: STTStatus): void {\n this.state.status = status;\n this.state.error = null;\n this.emit('status', { ...this.state });\n }\n\n private emitError(code: string, message: string): void {\n this.state.error = message;\n this.emit('error', { code, message });\n }\n}\n"],"mappings":"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","names":[]}
|