@camstack/addon-pipeline 0.1.20 → 0.2.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (101) hide show
  1. package/dist/audio-analyzer/index.js +736 -719
  2. package/dist/audio-analyzer/index.mjs +726 -679
  3. package/dist/audio-codec-nodeav/index.js +304 -461
  4. package/dist/audio-codec-nodeav/index.mjs +300 -462
  5. package/dist/chunk-BdkLduGY.mjs +5 -0
  6. package/dist/chunk-D6vf50IK.js +28 -0
  7. package/dist/codec-runtime-BOk-13PN.js +202 -0
  8. package/dist/codec-runtime-BsqlEjPi.mjs +197 -0
  9. package/dist/constants-B_b0a-6h.mjs +3119 -0
  10. package/dist/{index-CMcx_k6Y.js → constants-D65v6yp6.js} +3107 -2935
  11. package/dist/decoder-nodeav/index.js +1374 -1444
  12. package/dist/decoder-nodeav/index.mjs +1369 -1425
  13. package/dist/detection-pipeline/index.js +6462 -5613
  14. package/dist/detection-pipeline/index.mjs +6451 -5574
  15. package/dist/dist-7ewQjTle.js +22454 -0
  16. package/dist/dist-C5jnNl0n.mjs +22089 -0
  17. package/dist/motion-wasm/index.js +469 -467
  18. package/dist/motion-wasm/index.mjs +464 -446
  19. package/dist/pipeline-runner/index.js +2029 -1827
  20. package/dist/pipeline-runner/index.mjs +2025 -1811
  21. package/dist/recorder/index.js +2045 -2157
  22. package/dist/recorder/index.mjs +2042 -2156
  23. package/dist/stream-broker/_stub.js +1806 -1352
  24. package/dist/stream-broker/_virtual_mf-localSharedImportMap___mfe_internal__addon_stream_broker_widgets-D4-DHanK.mjs +156 -0
  25. package/dist/stream-broker/_virtual_mf___mfe_internal__addon_stream_broker_widgets__loadShare___mf_0_camstack_mf_1_types__loadShare__.js-Tf-HACFd.mjs +26 -0
  26. package/dist/stream-broker/_virtual_mf___mfe_internal__addon_stream_broker_widgets__loadShare___mf_0_camstack_mf_1_ui_mf_2_library__loadShare__.js-C9WX5HNw.mjs +26 -0
  27. package/dist/stream-broker/_virtual_mf___mfe_internal__addon_stream_broker_widgets__loadShare___mf_0_tanstack_mf_1_react_mf_2_query__loadShare__.js-BO7TIbJV.mjs +26 -0
  28. package/dist/stream-broker/_virtual_mf___mfe_internal__addon_stream_broker_widgets__loadShare__react__loadShare__.js-C9j-2lBe.mjs +26 -0
  29. package/dist/stream-broker/_virtual_mf___mfe_internal__addon_stream_broker_widgets__loadShare__react_mf_1_jsx_mf_2_runtime__loadShare__.js-XO0-Pyu6.mjs +26 -0
  30. package/dist/stream-broker/dist-CYZr2fwk.mjs +2726 -0
  31. package/dist/stream-broker/hostInit-Di6vceAU.mjs +129 -0
  32. package/dist/stream-broker/index.js +17778 -15470
  33. package/dist/stream-broker/index.mjs +17769 -15465
  34. package/dist/stream-broker/remoteEntry.js +134 -2973
  35. package/dist/stream-broker/remoteEntry.ssr.js +33 -0
  36. package/dist/stream-broker/virtualExposes-dYNvIwoR.mjs +27 -0
  37. package/dist/stream-broker/virtual_mf-exposes-ssr___mfe_internal__addon_stream_broker_widgets__remoteEntry_js-Cmqfp4i_.mjs +10 -0
  38. package/embed-dist/assets/index-B8VlSD0-.js +150 -0
  39. package/embed-dist/assets/index-ZhDdp1Nd.css +2 -0
  40. package/embed-dist/index.html +13 -0
  41. package/package.json +25 -7
  42. package/wasm/assembly/index.ts +41 -16
  43. package/dist/audio-analyzer/index.js.map +0 -1
  44. package/dist/audio-analyzer/index.mjs.map +0 -1
  45. package/dist/audio-codec-nodeav/index.js.map +0 -1
  46. package/dist/audio-codec-nodeav/index.mjs.map +0 -1
  47. package/dist/decoder-nodeav/index.js.map +0 -1
  48. package/dist/decoder-nodeav/index.mjs.map +0 -1
  49. package/dist/detection-pipeline/index.js.map +0 -1
  50. package/dist/detection-pipeline/index.mjs.map +0 -1
  51. package/dist/index-5aYef068.mjs +0 -17514
  52. package/dist/index-5aYef068.mjs.map +0 -1
  53. package/dist/index-B36NMAdu.js +0 -17513
  54. package/dist/index-B36NMAdu.js.map +0 -1
  55. package/dist/index-CMcx_k6Y.js.map +0 -1
  56. package/dist/index-CYb7cFrv.mjs +0 -5790
  57. package/dist/index-CYb7cFrv.mjs.map +0 -1
  58. package/dist/motion-wasm/index.js.map +0 -1
  59. package/dist/motion-wasm/index.mjs.map +0 -1
  60. package/dist/pipeline-runner/index.js.map +0 -1
  61. package/dist/pipeline-runner/index.mjs.map +0 -1
  62. package/dist/recorder/index.js.map +0 -1
  63. package/dist/recorder/index.mjs.map +0 -1
  64. package/dist/stream-broker/@mf-types/compiled-types/stream-broker/widgets/FfmpegParamsField.d.ts +0 -41
  65. package/dist/stream-broker/@mf-types/compiled-types/stream-broker/widgets/GeometryBuilder.d.ts +0 -54
  66. package/dist/stream-broker/@mf-types/compiled-types/stream-broker/widgets/StreamBrokerPanel.d.ts +0 -21
  67. package/dist/stream-broker/@mf-types/compiled-types/stream-broker/widgets/format-ua.d.ts +0 -13
  68. package/dist/stream-broker/@mf-types/compiled-types/stream-broker/widgets/index.d.ts +0 -15
  69. package/dist/stream-broker/@mf-types/widgets.d.ts +0 -2
  70. package/dist/stream-broker/@mf-types.d.ts +0 -3
  71. package/dist/stream-broker/@mf-types.zip +0 -0
  72. package/dist/stream-broker/__mfe_internal__addon_stream_broker_widgets__loadShare___mf_0_camstack_mf_1_sdk__loadShare__.mjs-lantnv8e.mjs +0 -12
  73. package/dist/stream-broker/__mfe_internal__addon_stream_broker_widgets__loadShare___mf_0_camstack_mf_1_types__loadShare__.mjs-DJ3UNg7O.mjs +0 -30
  74. package/dist/stream-broker/__mfe_internal__addon_stream_broker_widgets__loadShare___mf_0_camstack_mf_1_ui_mf_2_library__loadShare__.mjs-CYXy_bhS.mjs +0 -21
  75. package/dist/stream-broker/__mfe_internal__addon_stream_broker_widgets__loadShare___mf_0_tanstack_mf_1_react_mf_2_query__loadShare__.mjs-U1EUeEPs.mjs +0 -104
  76. package/dist/stream-broker/__mfe_internal__addon_stream_broker_widgets__loadShare___mf_0_trpc_mf_1_client__loadShare__.mjs-DeouEaSs.mjs +0 -85
  77. package/dist/stream-broker/__mfe_internal__addon_stream_broker_widgets__loadShare___mf_0_trpc_mf_1_react_mf_2_query__loadShare__.mjs-DHUwjbb9.mjs +0 -62
  78. package/dist/stream-broker/__mfe_internal__addon_stream_broker_widgets__loadShare__react__loadShare__.mjs-CaDEYBIU.mjs +0 -89
  79. package/dist/stream-broker/__mfe_internal__addon_stream_broker_widgets__loadShare__react__loadShare__.mjs_commonjs-proxy-D6EROtlA.mjs +0 -29
  80. package/dist/stream-broker/__mfe_internal__addon_stream_broker_widgets__loadShare__react_mf_1_jsx_mf_2_runtime__loadShare__.mjs-x6pP3Ghk.mjs +0 -36
  81. package/dist/stream-broker/__mfe_internal__addon_stream_broker_widgets__loadShare__react_mf_2_dom__loadShare__.mjs-DYEKzzY-.mjs +0 -45
  82. package/dist/stream-broker/__mfe_internal__addon_stream_broker_widgets__loadShare__react_mf_2_dom__loadShare__.mjs_commonjs-proxy-CcnN6sbA.mjs +0 -6
  83. package/dist/stream-broker/__mfe_internal__addon_stream_broker_widgets__loadShare__react_mf_2_dom_mf_1_client__loadShare__.mjs-DICOtMTl.mjs +0 -34
  84. package/dist/stream-broker/_virtual_mf-localSharedImportMap___mfe_internal__addon_stream_broker_widgets-CL9DR49k.mjs +0 -156
  85. package/dist/stream-broker/client-BvTmMOQu.mjs +0 -9836
  86. package/dist/stream-broker/getErrorShape-BPSzUA7W-TlK8ipWe.mjs +0 -211
  87. package/dist/stream-broker/hostInit-ChmiMPS0.mjs +0 -168
  88. package/dist/stream-broker/index-BxsFuFmE.mjs +0 -2603
  89. package/dist/stream-broker/index-C-248uOU.mjs +0 -725
  90. package/dist/stream-broker/index-C05B6jqp.mjs +0 -185
  91. package/dist/stream-broker/index-CWkKuNLr.mjs +0 -232
  92. package/dist/stream-broker/index-DOJoSShD.mjs +0 -67784
  93. package/dist/stream-broker/index-DtOI1aTU.mjs +0 -18504
  94. package/dist/stream-broker/index-oMq6ilgR.mjs +0 -1641
  95. package/dist/stream-broker/index-vIWZQBIL.mjs +0 -435
  96. package/dist/stream-broker/index-xncRG7-x.mjs +0 -2713
  97. package/dist/stream-broker/index.js.map +0 -1
  98. package/dist/stream-broker/index.mjs.map +0 -1
  99. package/dist/stream-broker/jsx-runtime-BRT_HL0A.mjs +0 -55
  100. package/dist/stream-broker/schemas-B7L0qZtq.mjs +0 -3599
  101. package/dist/stream-broker/virtualExposes-pCd777Rp.mjs +0 -42
@@ -1,467 +1,310 @@
1
- "use strict";
2
- Object.defineProperties(exports, { __esModule: { value: true }, [Symbol.toStringTag]: { value: "Module" } });
3
- const crypto = require("node:crypto");
4
- const index = require("../index-B36NMAdu.js");
5
- const nodeAv = require("node-av");
6
- const CODEC_ID_BY_NAME = {
7
- aac: nodeAv.AV_CODEC_ID_AAC,
8
- aac_latm: nodeAv.AV_CODEC_ID_AAC_LATM,
9
- opus: nodeAv.AV_CODEC_ID_OPUS,
10
- pcm_alaw: nodeAv.AV_CODEC_ID_PCM_ALAW,
11
- pcm_mulaw: nodeAv.AV_CODEC_ID_PCM_MULAW
12
- };
13
- function resolveCodecId(codec) {
14
- return CODEC_ID_BY_NAME[codec.toLowerCase()] ?? null;
15
- }
16
- function buildChannelLayout(channels) {
17
- if (channels === 1) {
18
- return { nbChannels: 1, order: nodeAv.AV_CHANNEL_ORDER_NATIVE, mask: nodeAv.AV_CH_LAYOUT_MONO };
19
- }
20
- if (channels === 2) {
21
- return { nbChannels: 2, order: nodeAv.AV_CHANNEL_ORDER_NATIVE, mask: nodeAv.AV_CH_LAYOUT_STEREO };
22
- }
23
- return { nbChannels: channels, order: nodeAv.AV_CHANNEL_ORDER_NATIVE, mask: 0n };
24
- }
25
- function pcmFormatToAv(format) {
26
- switch (format) {
27
- case "f32le":
28
- return nodeAv.AV_SAMPLE_FMT_FLT;
29
- case "s16le":
30
- return nodeAv.AV_SAMPLE_FMT_S16;
31
- }
32
- }
33
- function bytesPerSample(format) {
34
- switch (format) {
35
- case "f32le":
36
- return 4;
37
- case "s16le":
38
- return 2;
39
- }
40
- }
41
- class DecodeRuntime {
42
- ctx;
43
- swr;
44
- packet;
45
- inFrame;
46
- outFrame;
47
- cfg;
48
- inLayout;
49
- outLayout;
50
- inSampleFmt;
51
- outSampleFmt;
52
- outBytesPerSample;
53
- nextPts = 0;
54
- closed = false;
55
- static create(cfg) {
56
- const codecId = resolveCodecId(cfg.codec);
57
- if (codecId === null) {
58
- throw new Error(`audio-codec: unknown codec '${cfg.codec}' for decode runtime`);
59
- }
60
- const codec = nodeAv.Codec.findDecoder(codecId);
61
- if (!codec) {
62
- throw new Error(`audio-codec: decoder not registered for codec '${cfg.codec}'`);
63
- }
64
- const ctx = new nodeAv.CodecContext();
65
- ctx.allocContext3(codec);
66
- ctx.sampleRate = cfg.sourceSampleRate;
67
- ctx.channels = cfg.sourceChannels;
68
- const inLayout = buildChannelLayout(cfg.sourceChannels);
69
- ctx.channelLayout = inLayout;
70
- if (cfg.extraData && cfg.extraData.byteLength > 0) {
71
- ctx.extraData = Buffer.from(cfg.extraData);
72
- }
73
- const ret = ctx.open2Sync(codec, null);
74
- if (ret < 0) {
75
- ctx.freeContext();
76
- throw new Error(`audio-codec: open2 failed for '${cfg.codec}' (ret=${ret})`);
77
- }
78
- const inSampleFmt = ctx.sampleFormat ?? nodeAv.AV_SAMPLE_FMT_FLT;
79
- const outSampleFmt = pcmFormatToAv(cfg.targetFormat);
80
- const outLayout = buildChannelLayout(cfg.targetChannels);
81
- const swr = new nodeAv.SoftwareResampleContext();
82
- const allocRet = swr.allocSetOpts2(
83
- outLayout,
84
- outSampleFmt,
85
- cfg.targetSampleRate,
86
- inLayout,
87
- inSampleFmt,
88
- cfg.sourceSampleRate
89
- );
90
- if (allocRet < 0) {
91
- ctx.freeContext();
92
- throw new Error(`audio-codec: swr allocSetOpts2 failed (ret=${allocRet})`);
93
- }
94
- const initRet = swr.init();
95
- if (initRet < 0) {
96
- ctx.freeContext();
97
- throw new Error(`audio-codec: swr init failed (ret=${initRet})`);
98
- }
99
- const packet = new nodeAv.Packet();
100
- packet.alloc();
101
- const inFrame = new nodeAv.Frame();
102
- inFrame.alloc();
103
- const outFrame = new nodeAv.Frame();
104
- outFrame.alloc();
105
- return new DecodeRuntime(ctx, swr, packet, inFrame, outFrame, cfg, inLayout, outLayout, inSampleFmt, outSampleFmt);
106
- }
107
- constructor(ctx, swr, packet, inFrame, outFrame, cfg, inLayout, outLayout, inSampleFmt, outSampleFmt) {
108
- this.ctx = ctx;
109
- this.swr = swr;
110
- this.packet = packet;
111
- this.inFrame = inFrame;
112
- this.outFrame = outFrame;
113
- this.cfg = cfg;
114
- this.inLayout = inLayout;
115
- this.outLayout = outLayout;
116
- this.inSampleFmt = inSampleFmt;
117
- this.outSampleFmt = outSampleFmt;
118
- this.outBytesPerSample = bytesPerSample(cfg.targetFormat);
119
- }
120
- decode(buf, pts) {
121
- if (this.closed) return [];
122
- this.packet.data = Buffer.from(buf);
123
- if (pts !== void 0) {
124
- this.packet.pts = BigInt(Math.round(pts));
125
- this.packet.dts = BigInt(Math.round(pts));
126
- }
127
- const sendRet = this.ctx.sendPacketSync(this.packet);
128
- if (sendRet < 0 && sendRet !== nodeAv.AVERROR_EAGAIN) {
129
- throw new Error(`audio-codec: sendPacket failed (ret=${sendRet})`);
130
- }
131
- const out = [];
132
- while (true) {
133
- const recvRet = this.ctx.receiveFrameSync(this.inFrame);
134
- if (recvRet === nodeAv.AVERROR_EAGAIN || recvRet === nodeAv.AVERROR_EOF) break;
135
- if (recvRet < 0) {
136
- throw new Error(`audio-codec: receiveFrame failed (ret=${recvRet})`);
137
- }
138
- const pcm = this.resampleFrame(this.inFrame);
139
- if (pcm) out.push(pcm);
140
- this.inFrame.unref();
141
- }
142
- return out;
143
- }
144
- resampleFrame(inFrame) {
145
- const inSamples = inFrame.nbSamples;
146
- const outSamples = Math.ceil(inSamples * this.cfg.targetSampleRate / this.cfg.sourceSampleRate) + 32;
147
- this.outFrame.unref();
148
- this.outFrame.format = this.outSampleFmt;
149
- this.outFrame.sampleRate = this.cfg.targetSampleRate;
150
- this.outFrame.channelLayout = this.outLayout;
151
- this.outFrame.nbSamples = outSamples;
152
- const allocRet = this.outFrame.getBuffer(0);
153
- if (allocRet < 0) {
154
- throw new Error(`audio-codec: outFrame.getBuffer failed (ret=${allocRet})`);
155
- }
156
- const convRet = this.swr.convertFrame(this.outFrame, inFrame);
157
- if (convRet < 0) {
158
- throw new Error(`audio-codec: swr.convertFrame failed (ret=${convRet})`);
159
- }
160
- const producedSamples = this.outFrame.nbSamples;
161
- if (producedSamples <= 0) return null;
162
- const planes = this.outFrame.extendedData;
163
- if (!planes || planes.length === 0 || !planes[0]) return null;
164
- const bytes = producedSamples * this.cfg.targetChannels * this.outBytesPerSample;
165
- const src = planes[0];
166
- const out = new Uint8Array(new ArrayBuffer(bytes));
167
- out.set(src.subarray(0, bytes));
168
- const ptsMs = this.nextPts;
169
- this.nextPts = ptsMs + Math.round(producedSamples * 1e3 / this.cfg.targetSampleRate);
170
- return {
171
- data: out,
172
- sampleRate: this.cfg.targetSampleRate,
173
- channels: this.cfg.targetChannels,
174
- format: this.cfg.targetFormat,
175
- pts: ptsMs
176
- };
177
- }
178
- close() {
179
- if (this.closed) return;
180
- this.closed = true;
181
- try {
182
- this.outFrame.free();
183
- } catch {
184
- }
185
- try {
186
- this.inFrame.free();
187
- } catch {
188
- }
189
- try {
190
- this.packet.free();
191
- } catch {
192
- }
193
- try {
194
- this.swr.free();
195
- } catch {
196
- }
197
- try {
198
- this.ctx.freeContext();
199
- } catch {
200
- }
201
- }
202
- /** Surfaced for assertion-style tests to inspect computed layouts. */
203
- describe() {
204
- return {
205
- inLayoutMask: this.inLayout.mask,
206
- outLayoutMask: this.outLayout.mask,
207
- inSampleFormat: this.inSampleFmt,
208
- outSampleFormat: this.outSampleFmt
209
- };
210
- }
211
- }
212
- const CODEC_CATALOG = [
213
- { codec: "pcm_mulaw", canDecode: true, canEncode: true, label: "PCM µ-law (G.711)" },
214
- { codec: "pcm_alaw", canDecode: true, canEncode: true, label: "PCM A-law (G.711)" },
215
- { codec: "aac", canDecode: true, canEncode: true, label: "AAC" },
216
- { codec: "aac_latm", canDecode: true, canEncode: false, label: "AAC LATM" },
217
- { codec: "mpeg4-generic", canDecode: true, canEncode: false, label: "AAC (MPEG4-GENERIC)" },
218
- { codec: "opus", canDecode: true, canEncode: true, label: "Opus" }
1
+ Object.defineProperties(exports, {
2
+ __esModule: { value: true },
3
+ [Symbol.toStringTag]: { value: "Module" }
4
+ });
5
+ const require_dist = require("../dist-7ewQjTle.js");
6
+ const require_codec_runtime = require("../codec-runtime-BOk-13PN.js");
7
+ let node_crypto = require("node:crypto");
8
+ //#region src/audio-codec-nodeav/addon/index.ts
9
+ var CODEC_CATALOG = [
10
+ {
11
+ codec: "pcm_mulaw",
12
+ canDecode: true,
13
+ canEncode: true,
14
+ label: "PCM µ-law (G.711)"
15
+ },
16
+ {
17
+ codec: "pcm_alaw",
18
+ canDecode: true,
19
+ canEncode: true,
20
+ label: "PCM A-law (G.711)"
21
+ },
22
+ {
23
+ codec: "aac",
24
+ canDecode: true,
25
+ canEncode: true,
26
+ label: "AAC"
27
+ },
28
+ {
29
+ codec: "aac_latm",
30
+ canDecode: true,
31
+ canEncode: false,
32
+ label: "AAC LATM"
33
+ },
34
+ {
35
+ codec: "mpeg4-generic",
36
+ canDecode: true,
37
+ canEncode: false,
38
+ label: "AAC (MPEG4-GENERIC)"
39
+ },
40
+ {
41
+ codec: "opus",
42
+ canDecode: true,
43
+ canEncode: true,
44
+ label: "Opus"
45
+ }
219
46
  ];
220
47
  function resolveCodecAlias(codec) {
221
- const c = codec.toLowerCase();
222
- if (c === "mpeg4-generic") return "aac";
223
- if (c === "l16") return "pcm_s16be";
224
- return c;
48
+ const c = codec.toLowerCase();
49
+ if (c === "mpeg4-generic") return "aac";
50
+ if (c === "l16") return "pcm_s16be";
51
+ return c;
225
52
  }
226
- const DEFAULT_IDLE_MS = 3e4;
227
- const REAPER_INTERVAL_MS = 5e3;
228
- const DEFAULT_GLOBAL_CONFIG = {
229
- defaultIdleMs: DEFAULT_IDLE_MS
53
+ var DEFAULT_IDLE_MS = 3e4;
54
+ var REAPER_INTERVAL_MS = 5e3;
55
+ var DEFAULT_GLOBAL_CONFIG = { defaultIdleMs: DEFAULT_IDLE_MS };
56
+ /**
57
+ * Audio codec I/O box backed by node-av (libavcodec + libswresample).
58
+ *
59
+ * Each `createDecodeSession` / `createEncodeSession` allocates an
60
+ * independent libav codec context + resampler so consumers don't share
61
+ * resamplers — a 16kHz mono ASA subscriber and a 48kHz stereo WebRTC
62
+ * subscriber on the same source codec each get their own session.
63
+ *
64
+ * Phase 1 scaffolds the cap surface + session bookkeeping. The actual
65
+ * libav decode/encode wiring is filled in by follow-up commits — the
66
+ * stub keeps the cap registered so wiring on the broker side can land
67
+ * before the audio backend matures.
68
+ */
69
+ var AudioCodecNodeAvAddon = class extends require_dist.BaseAddon {
70
+ sessions = /* @__PURE__ */ new Map();
71
+ reaperTimer = null;
72
+ logger = null;
73
+ constructor() {
74
+ super(DEFAULT_GLOBAL_CONFIG);
75
+ }
76
+ async onInitialize() {
77
+ this.logger = this.ctx.logger;
78
+ this.reaperTimer = setInterval(() => this.reapIdleSessions(), REAPER_INTERVAL_MS);
79
+ if (typeof this.reaperTimer.unref === "function") this.reaperTimer.unref();
80
+ return [{
81
+ capability: require_dist.audioCodecCapability,
82
+ provider: this
83
+ }];
84
+ }
85
+ async onShutdown() {
86
+ if (this.reaperTimer) {
87
+ clearInterval(this.reaperTimer);
88
+ this.reaperTimer = null;
89
+ }
90
+ for (const s of [...this.sessions.values()]) this.disposeSession(s);
91
+ this.sessions.clear();
92
+ }
93
+ async listSupportedCodecs() {
94
+ return CODEC_CATALOG.map((e) => ({
95
+ codec: e.codec,
96
+ canDecode: e.canDecode,
97
+ canEncode: e.canEncode,
98
+ ...e.label ? { label: e.label } : {}
99
+ }));
100
+ }
101
+ async canHandle(input) {
102
+ const resolved = resolveCodecAlias(input.codec);
103
+ const entry = CODEC_CATALOG.find((e) => e.codec === resolved);
104
+ if (!entry) return false;
105
+ return input.kind === "decode" ? entry.canDecode : entry.canEncode;
106
+ }
107
+ async createDecodeSession(input) {
108
+ const codec = resolveCodecAlias(input.codec);
109
+ const entry = CODEC_CATALOG.find((e) => e.codec === codec);
110
+ if (!entry || !entry.canDecode) throw new Error(`audio-codec: decode unsupported for codec '${input.codec}'`);
111
+ const sessionId = `dec-${(0, node_crypto.randomUUID)()}`;
112
+ const state = {
113
+ sessionId,
114
+ kind: "decode",
115
+ config: {
116
+ ...input,
117
+ codec
118
+ },
119
+ ...input.tag ? { tag: input.tag } : {},
120
+ createdAtMs: Date.now(),
121
+ lastActivityMs: Date.now(),
122
+ framesIn: 0,
123
+ framesOut: 0,
124
+ pcmQueue: [],
125
+ runtime: null
126
+ };
127
+ this.sessions.set(sessionId, state);
128
+ this.logger?.info("audio-codec: decode session created", {
129
+ tags: { sessionId },
130
+ meta: {
131
+ codec,
132
+ target: `${input.targetSampleRate}Hz×${input.targetChannels}`
133
+ }
134
+ });
135
+ return {
136
+ sessionId,
137
+ nodeId: this.ctx.kernel.localNodeId ?? "local"
138
+ };
139
+ }
140
+ async createEncodeSession(input) {
141
+ const codec = resolveCodecAlias(input.codec);
142
+ const entry = CODEC_CATALOG.find((e) => e.codec === codec);
143
+ if (!entry || !entry.canEncode) throw new Error(`audio-codec: encode unsupported for codec '${input.codec}'`);
144
+ const sessionId = `enc-${(0, node_crypto.randomUUID)()}`;
145
+ const state = {
146
+ sessionId,
147
+ kind: "encode",
148
+ config: {
149
+ ...input,
150
+ codec
151
+ },
152
+ ...input.tag ? { tag: input.tag } : {},
153
+ createdAtMs: Date.now(),
154
+ lastActivityMs: Date.now(),
155
+ framesIn: 0,
156
+ framesOut: 0,
157
+ encodedQueue: []
158
+ };
159
+ this.sessions.set(sessionId, state);
160
+ this.logger?.info("audio-codec: encode session created", {
161
+ tags: { sessionId },
162
+ meta: {
163
+ codec,
164
+ target: `${input.targetSampleRate}Hz×${input.targetChannels}`
165
+ }
166
+ });
167
+ return {
168
+ sessionId,
169
+ nodeId: this.ctx.kernel.localNodeId ?? "local"
170
+ };
171
+ }
172
+ async closeSession(input) {
173
+ const s = this.sessions.get(input.sessionId);
174
+ if (!s) return;
175
+ this.disposeSession(s);
176
+ this.sessions.delete(input.sessionId);
177
+ }
178
+ async pushEncodedFrame(input) {
179
+ const s = this.sessions.get(input.sessionId);
180
+ if (!s || s.kind !== "decode") throw new Error(`audio-codec: decode session '${input.sessionId}' not found`);
181
+ s.lastActivityMs = Date.now();
182
+ s.framesIn++;
183
+ if (!s.runtime) try {
184
+ s.runtime = require_codec_runtime.DecodeRuntime.create({
185
+ codec: s.config.codec,
186
+ sourceSampleRate: s.config.sourceSampleRate,
187
+ sourceChannels: s.config.sourceChannels,
188
+ ...s.config.extraData ? { extraData: s.config.extraData } : {},
189
+ targetSampleRate: s.config.targetSampleRate,
190
+ targetChannels: s.config.targetChannels,
191
+ targetFormat: s.config.targetFormat ?? "f32le"
192
+ });
193
+ } catch (err) {
194
+ this.logger?.error("audio-codec: decode runtime init failed", {
195
+ tags: { sessionId: input.sessionId },
196
+ meta: {
197
+ codec: s.config.codec,
198
+ error: require_dist.errMsg(err)
199
+ }
200
+ });
201
+ return;
202
+ }
203
+ try {
204
+ const pcms = s.runtime.decode(input.data, input.pts);
205
+ for (const pcm of pcms) s.pcmQueue.push(pcm);
206
+ } catch (err) {
207
+ this.logger?.warn("audio-codec: decode frame failed", {
208
+ tags: { sessionId: input.sessionId },
209
+ meta: {
210
+ codec: s.config.codec,
211
+ error: require_dist.errMsg(err)
212
+ }
213
+ });
214
+ }
215
+ }
216
+ async pullPcm(input) {
217
+ const s = this.sessions.get(input.sessionId);
218
+ if (!s || s.kind !== "decode") throw new Error(`audio-codec: decode session '${input.sessionId}' not found`);
219
+ s.lastActivityMs = Date.now();
220
+ const out = s.pcmQueue.splice(0, input.maxCount);
221
+ s.framesOut += out.length;
222
+ return out;
223
+ }
224
+ async pushPcm(input) {
225
+ const s = this.sessions.get(input.sessionId);
226
+ if (!s || s.kind !== "encode") throw new Error(`audio-codec: encode session '${input.sessionId}' not found`);
227
+ s.lastActivityMs = Date.now();
228
+ s.framesIn++;
229
+ input.data;
230
+ input.pts;
231
+ }
232
+ async pullEncoded(input) {
233
+ const s = this.sessions.get(input.sessionId);
234
+ if (!s || s.kind !== "encode") throw new Error(`audio-codec: encode session '${input.sessionId}' not found`);
235
+ s.lastActivityMs = Date.now();
236
+ const out = s.encodedQueue.splice(0, input.maxCount);
237
+ s.framesOut += out.length;
238
+ return out;
239
+ }
240
+ async flushEncode(input) {
241
+ const s = this.sessions.get(input.sessionId);
242
+ if (!s || s.kind !== "encode") throw new Error(`audio-codec: encode session '${input.sessionId}' not found`);
243
+ s.lastActivityMs = Date.now();
244
+ const out = s.encodedQueue.splice(0);
245
+ s.framesOut += out.length;
246
+ return out;
247
+ }
248
+ async listActiveSessions() {
249
+ return [...this.sessions.values()].map((s) => ({
250
+ sessionId: s.sessionId,
251
+ kind: s.kind,
252
+ codec: s.config.codec,
253
+ sourceSampleRate: s.config.sourceSampleRate,
254
+ sourceChannels: s.config.sourceChannels,
255
+ targetSampleRate: s.config.targetSampleRate,
256
+ targetChannels: s.config.targetChannels,
257
+ format: this.resolveFormat(s),
258
+ ...s.tag ? { tag: s.tag } : {},
259
+ createdAtMs: s.createdAtMs,
260
+ lastActivityMs: s.lastActivityMs,
261
+ framesIn: s.framesIn,
262
+ framesOut: s.framesOut
263
+ }));
264
+ }
265
+ resolveFormat(s) {
266
+ if (s.kind === "decode") return s.config.targetFormat ?? "f32le";
267
+ return s.config.sourceFormat ?? "f32le";
268
+ }
269
+ reapIdleSessions() {
270
+ const now = Date.now();
271
+ for (const [id, s] of this.sessions) {
272
+ const limit = s.config.idleMs ?? this.config.defaultIdleMs ?? DEFAULT_IDLE_MS;
273
+ if (now - s.lastActivityMs > limit) {
274
+ this.logger?.info("audio-codec: reaping idle session", {
275
+ tags: { sessionId: id },
276
+ meta: {
277
+ kind: s.kind,
278
+ idleMs: now - s.lastActivityMs,
279
+ limit
280
+ }
281
+ });
282
+ try {
283
+ this.disposeSession(s);
284
+ } catch (err) {
285
+ this.logger?.warn("audio-codec: dispose failed during reap", {
286
+ tags: { sessionId: id },
287
+ meta: { error: require_dist.errMsg(err) }
288
+ });
289
+ }
290
+ this.sessions.delete(id);
291
+ }
292
+ }
293
+ }
294
+ disposeSession(s) {
295
+ if (s.kind === "decode" && s.runtime) {
296
+ try {
297
+ s.runtime.close();
298
+ } catch (err) {
299
+ this.logger?.warn("audio-codec: decode runtime close failed", {
300
+ tags: { sessionId: s.sessionId },
301
+ meta: { error: require_dist.errMsg(err) }
302
+ });
303
+ }
304
+ s.runtime = null;
305
+ }
306
+ }
230
307
  };
231
- class AudioCodecNodeAvAddon extends index.BaseAddon {
232
- sessions = /* @__PURE__ */ new Map();
233
- reaperTimer = null;
234
- logger = null;
235
- constructor() {
236
- super(DEFAULT_GLOBAL_CONFIG);
237
- }
238
- async onInitialize() {
239
- this.logger = this.ctx.logger;
240
- this.reaperTimer = setInterval(() => this.reapIdleSessions(), REAPER_INTERVAL_MS);
241
- if (typeof this.reaperTimer.unref === "function") this.reaperTimer.unref();
242
- return [{ capability: index.audioCodecCapability, provider: this }];
243
- }
244
- async onShutdown() {
245
- if (this.reaperTimer) {
246
- clearInterval(this.reaperTimer);
247
- this.reaperTimer = null;
248
- }
249
- for (const s of [...this.sessions.values()]) {
250
- this.disposeSession(s);
251
- }
252
- this.sessions.clear();
253
- }
254
- // ── Discovery ─────────────────────────────────────────────────────────────
255
- async listSupportedCodecs() {
256
- return CODEC_CATALOG.map((e) => ({
257
- codec: e.codec,
258
- canDecode: e.canDecode,
259
- canEncode: e.canEncode,
260
- ...e.label ? { label: e.label } : {}
261
- }));
262
- }
263
- async canHandle(input) {
264
- const resolved = resolveCodecAlias(input.codec);
265
- const entry = CODEC_CATALOG.find((e) => e.codec === resolved);
266
- if (!entry) return false;
267
- return input.kind === "decode" ? entry.canDecode : entry.canEncode;
268
- }
269
- // ── Session lifecycle ─────────────────────────────────────────────────────
270
- async createDecodeSession(input) {
271
- const codec = resolveCodecAlias(input.codec);
272
- const entry = CODEC_CATALOG.find((e) => e.codec === codec);
273
- if (!entry || !entry.canDecode) {
274
- throw new Error(`audio-codec: decode unsupported for codec '${input.codec}'`);
275
- }
276
- const sessionId = `dec-${crypto.randomUUID()}`;
277
- const state = {
278
- sessionId,
279
- kind: "decode",
280
- config: { ...input, codec },
281
- ...input.tag ? { tag: input.tag } : {},
282
- createdAtMs: Date.now(),
283
- lastActivityMs: Date.now(),
284
- framesIn: 0,
285
- framesOut: 0,
286
- pcmQueue: [],
287
- runtime: null
288
- };
289
- this.sessions.set(sessionId, state);
290
- this.logger?.info("audio-codec: decode session created", {
291
- tags: { sessionId },
292
- meta: { codec, target: `${input.targetSampleRate}Hz×${input.targetChannels}` }
293
- });
294
- return { sessionId, nodeId: this.ctx.kernel.localNodeId ?? "local" };
295
- }
296
- async createEncodeSession(input) {
297
- const codec = resolveCodecAlias(input.codec);
298
- const entry = CODEC_CATALOG.find((e) => e.codec === codec);
299
- if (!entry || !entry.canEncode) {
300
- throw new Error(`audio-codec: encode unsupported for codec '${input.codec}'`);
301
- }
302
- const sessionId = `enc-${crypto.randomUUID()}`;
303
- const state = {
304
- sessionId,
305
- kind: "encode",
306
- config: { ...input, codec },
307
- ...input.tag ? { tag: input.tag } : {},
308
- createdAtMs: Date.now(),
309
- lastActivityMs: Date.now(),
310
- framesIn: 0,
311
- framesOut: 0,
312
- encodedQueue: []
313
- };
314
- this.sessions.set(sessionId, state);
315
- this.logger?.info("audio-codec: encode session created", {
316
- tags: { sessionId },
317
- meta: { codec, target: `${input.targetSampleRate}Hz×${input.targetChannels}` }
318
- });
319
- return { sessionId, nodeId: this.ctx.kernel.localNodeId ?? "local" };
320
- }
321
- async closeSession(input) {
322
- const s = this.sessions.get(input.sessionId);
323
- if (!s) return;
324
- this.disposeSession(s);
325
- this.sessions.delete(input.sessionId);
326
- }
327
- // ── Decode data plane ─────────────────────────────────────────────────────
328
- async pushEncodedFrame(input) {
329
- const s = this.sessions.get(input.sessionId);
330
- if (!s || s.kind !== "decode") {
331
- throw new Error(`audio-codec: decode session '${input.sessionId}' not found`);
332
- }
333
- s.lastActivityMs = Date.now();
334
- s.framesIn++;
335
- if (!s.runtime) {
336
- try {
337
- s.runtime = DecodeRuntime.create({
338
- codec: s.config.codec,
339
- sourceSampleRate: s.config.sourceSampleRate,
340
- sourceChannels: s.config.sourceChannels,
341
- ...s.config.extraData ? { extraData: s.config.extraData } : {},
342
- targetSampleRate: s.config.targetSampleRate,
343
- targetChannels: s.config.targetChannels,
344
- targetFormat: s.config.targetFormat ?? "f32le"
345
- });
346
- } catch (err) {
347
- this.logger?.error("audio-codec: decode runtime init failed", {
348
- tags: { sessionId: input.sessionId },
349
- meta: { codec: s.config.codec, error: index.errMsg(err) }
350
- });
351
- return;
352
- }
353
- }
354
- try {
355
- const pcms = s.runtime.decode(input.data, input.pts);
356
- for (const pcm of pcms) {
357
- s.pcmQueue.push(pcm);
358
- }
359
- } catch (err) {
360
- this.logger?.warn("audio-codec: decode frame failed", {
361
- tags: { sessionId: input.sessionId },
362
- meta: { codec: s.config.codec, error: index.errMsg(err) }
363
- });
364
- }
365
- }
366
- async pullPcm(input) {
367
- const s = this.sessions.get(input.sessionId);
368
- if (!s || s.kind !== "decode") {
369
- throw new Error(`audio-codec: decode session '${input.sessionId}' not found`);
370
- }
371
- s.lastActivityMs = Date.now();
372
- const out = s.pcmQueue.splice(0, input.maxCount);
373
- s.framesOut += out.length;
374
- return out;
375
- }
376
- // ── Encode data plane ─────────────────────────────────────────────────────
377
- async pushPcm(input) {
378
- const s = this.sessions.get(input.sessionId);
379
- if (!s || s.kind !== "encode") {
380
- throw new Error(`audio-codec: encode session '${input.sessionId}' not found`);
381
- }
382
- s.lastActivityMs = Date.now();
383
- s.framesIn++;
384
- void input.data;
385
- void input.pts;
386
- }
387
- async pullEncoded(input) {
388
- const s = this.sessions.get(input.sessionId);
389
- if (!s || s.kind !== "encode") {
390
- throw new Error(`audio-codec: encode session '${input.sessionId}' not found`);
391
- }
392
- s.lastActivityMs = Date.now();
393
- const out = s.encodedQueue.splice(0, input.maxCount);
394
- s.framesOut += out.length;
395
- return out;
396
- }
397
- async flushEncode(input) {
398
- const s = this.sessions.get(input.sessionId);
399
- if (!s || s.kind !== "encode") {
400
- throw new Error(`audio-codec: encode session '${input.sessionId}' not found`);
401
- }
402
- s.lastActivityMs = Date.now();
403
- const out = s.encodedQueue.splice(0);
404
- s.framesOut += out.length;
405
- return out;
406
- }
407
- // ── Inventory ─────────────────────────────────────────────────────────────
408
- async listActiveSessions() {
409
- return [...this.sessions.values()].map((s) => ({
410
- sessionId: s.sessionId,
411
- kind: s.kind,
412
- codec: s.config.codec,
413
- sourceSampleRate: s.config.sourceSampleRate,
414
- sourceChannels: s.config.sourceChannels,
415
- targetSampleRate: s.config.targetSampleRate,
416
- targetChannels: s.config.targetChannels,
417
- format: this.resolveFormat(s),
418
- ...s.tag ? { tag: s.tag } : {},
419
- createdAtMs: s.createdAtMs,
420
- lastActivityMs: s.lastActivityMs,
421
- framesIn: s.framesIn,
422
- framesOut: s.framesOut
423
- }));
424
- }
425
- // ── Internals ─────────────────────────────────────────────────────────────
426
- resolveFormat(s) {
427
- if (s.kind === "decode") return s.config.targetFormat ?? "f32le";
428
- return s.config.sourceFormat ?? "f32le";
429
- }
430
- reapIdleSessions() {
431
- const now = Date.now();
432
- for (const [id, s] of this.sessions) {
433
- const limit = s.config.idleMs ?? this.config.defaultIdleMs ?? DEFAULT_IDLE_MS;
434
- if (now - s.lastActivityMs > limit) {
435
- this.logger?.info("audio-codec: reaping idle session", {
436
- tags: { sessionId: id },
437
- meta: { kind: s.kind, idleMs: now - s.lastActivityMs, limit }
438
- });
439
- try {
440
- this.disposeSession(s);
441
- } catch (err) {
442
- this.logger?.warn("audio-codec: dispose failed during reap", {
443
- tags: { sessionId: id },
444
- meta: { error: index.errMsg(err) }
445
- });
446
- }
447
- this.sessions.delete(id);
448
- }
449
- }
450
- }
451
- disposeSession(s) {
452
- if (s.kind === "decode" && s.runtime) {
453
- try {
454
- s.runtime.close();
455
- } catch (err) {
456
- this.logger?.warn("audio-codec: decode runtime close failed", {
457
- tags: { sessionId: s.sessionId },
458
- meta: { error: index.errMsg(err) }
459
- });
460
- }
461
- s.runtime = null;
462
- }
463
- }
464
- }
308
+ //#endregion
465
309
  exports.AudioCodecNodeAvAddon = AudioCodecNodeAvAddon;
466
310
  exports.default = AudioCodecNodeAvAddon;
467
- //# sourceMappingURL=index.js.map