@meframe/core 0.0.24 → 0.0.25
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/dist/worker/WorkerPool.d.ts +1 -0
- package/dist/worker/WorkerPool.d.ts.map +1 -1
- package/dist/worker/WorkerPool.js +19 -4
- package/dist/worker/WorkerPool.js.map +1 -1
- package/dist/worker/worker-manifest.d.ts +27 -0
- package/dist/worker/worker-manifest.d.ts.map +1 -0
- package/dist/worker/worker-manifest.js +37 -0
- package/dist/worker/worker-manifest.js.map +1 -0
- package/dist/workers/{BaseDecoder.js → BaseDecoder.Bk26nCBk.js} +1 -1
- package/dist/workers/BaseDecoder.Bk26nCBk.js.map +1 -0
- package/dist/workers/{MP4Demuxer.js → MP4Demuxer.CFHDkPYc.js} +1 -1
- package/dist/workers/MP4Demuxer.CFHDkPYc.js.map +1 -0
- package/dist/workers/{WorkerChannel.js → WorkerChannel.CE5euh3R.js} +1 -1
- package/dist/workers/WorkerChannel.CE5euh3R.js.map +1 -0
- package/dist/workers/stages/compose/{audio-compose.worker.js → audio-compose.worker.rW63uN6z.js} +2 -2
- package/dist/workers/stages/compose/audio-compose.worker.rW63uN6z.js.map +1 -0
- package/dist/workers/stages/compose/{video-compose.worker.js → video-compose.worker.B7xEVmN3.js} +2 -2
- package/dist/workers/stages/compose/video-compose.worker.B7xEVmN3.js.map +1 -0
- package/dist/workers/stages/decode/{audio-decode.worker.js → audio-decode.worker.Czb1P66a.js} +3 -3
- package/dist/workers/stages/decode/audio-decode.worker.Czb1P66a.js.map +1 -0
- package/dist/workers/stages/decode/{video-decode.worker.js → video-decode.worker.DLX8FRVc.js} +3 -3
- package/dist/workers/stages/decode/video-decode.worker.DLX8FRVc.js.map +1 -0
- package/dist/workers/stages/demux/{audio-demux.worker.js → audio-demux.worker.BTFPcY7P.js} +3 -3
- package/dist/workers/stages/demux/audio-demux.worker.BTFPcY7P.js.map +1 -0
- package/dist/workers/stages/demux/{video-demux.worker.js → video-demux.worker.D_WeHPkt.js} +3 -3
- package/dist/workers/stages/demux/video-demux.worker.D_WeHPkt.js.map +1 -0
- package/dist/workers/stages/encode/{video-encode.worker.js → video-encode.worker.u2o7iXCT.js} +2 -2
- package/dist/workers/stages/encode/video-encode.worker.u2o7iXCT.js.map +1 -0
- package/dist/workers/worker-manifest.json +9 -0
- package/package.json +2 -2
- package/dist/workers/BaseDecoder.js.map +0 -1
- package/dist/workers/MP4Demuxer.js.map +0 -1
- package/dist/workers/WorkerChannel.js.map +0 -1
- package/dist/workers/stages/compose/audio-compose.worker.js.map +0 -1
- package/dist/workers/stages/compose/video-compose.worker.js.map +0 -1
- package/dist/workers/stages/decode/audio-decode.worker.js.map +0 -1
- package/dist/workers/stages/decode/video-decode.worker.js.map +0 -1
- package/dist/workers/stages/demux/audio-demux.worker.js.map +0 -1
- package/dist/workers/stages/demux/video-demux.worker.js.map +0 -1
- package/dist/workers/stages/encode/video-encode.worker.js.map +0 -1
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"WorkerPool.d.ts","sourceRoot":"","sources":["../../src/worker/WorkerPool.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAEH,OAAO,EAAE,UAAU,EAAE,MAAM,cAAc,CAAC;AAC1C,OAAO,KAAK,EAAE,UAAU,EAAE,YAAY,EAA0B,MAAM,SAAS,CAAC;AAEhF,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,mBAAmB,CAAC;AAClD,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,iBAAiB,CAAC;
|
|
1
|
+
{"version":3,"file":"WorkerPool.d.ts","sourceRoot":"","sources":["../../src/worker/WorkerPool.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAEH,OAAO,EAAE,UAAU,EAAE,MAAM,cAAc,CAAC;AAC1C,OAAO,KAAK,EAAE,UAAU,EAAE,YAAY,EAA0B,MAAM,SAAS,CAAC;AAEhF,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,mBAAmB,CAAC;AAClD,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,iBAAiB,CAAC;AAGvD,MAAM,WAAW,gBAAgB;IAC/B,QAAQ,EAAE,QAAQ,CAAC,eAAe,CAAC,CAAC;IACpC,aAAa,CAAC,EAAE,MAAM,CAAC,UAAU,EAAE,GAAG,CAAC,CAAC;IACxC,2DAA2D;IAC3D,UAAU,CAAC,EAAE,MAAM,CAAC;IACpB,eAAe,CAAC,EAAE,MAAM,CAAC;CAC1B;AAeD,qBAAa,UAAU;IACrB,OAAO,CAAC,IAAI,CAAiC;IAC7C,OAAO,CAAC,QAAQ,CAA4B;IAC5C,OAAO,CAAC,aAAa,CAAsB;IAC3C,OAAO,CAAC,UAAU,CAAS;IAC3B,OAAO,CAAC,eAAe,CAAS;IAChC,OAAO,CAAC,eAAe,CAA+C;gBAE1D,MAAM,EAAE,gBAAgB;IAOpC;;OAEG;YACW,YAAY;IA4C1B,GAAG,CAAC,IAAI,EAAE,UAAU,EAAE,EAAE,CAAC,EAAE,MAAM,GAAG,UAAU,GAAG,SAAS;IAK1D;;;;;;OAMG;IACG,WAAW,CACf,IAAI,EAAE,UAAU,EAChB,EAAE,CAAC,EAAE,MAAM,EACX,OAAO,CAAC,EAAE;QAAE,IAAI,CAAC,EAAE,OAAO,CAAA;KAAE,GAC3B,OAAO,CAAC,UAAU,CAAC;IAsChB,SAAS,CAAC,IAAI,EAAE,UAAU,EAAE,MAAM,EAAE,GAAG,GAAG,OAAO,CAAC,IAAI,CAAC;IAgB7D;;OAEG;IACH,SAAS,CAAC,IAAI,EAAE,UAAU,EAAE,EAAE,CAAC,EAAE,MAAM,GAAG,IAAI;IAU9C;;OAEG;IACH,YAAY,IAAI,IAAI;IAOpB;;OAEG;IACH,IAAI,MAAM,IAAI,MAAM,CAAC,MAAM,EAAE,YAAY,CAAC,UAAU,CAAC,CAAC,CAQrD;IAED;;OAEG;IACH,IAAI,aAAa,IAAI,MAAM,EAAE,CAE5B;IAED;;OAEG;IACH,GAAG,CAAC,IAAI,EAAE,UAAU,EAAE,EAAE,CAAC,EAAE,MAAM,GAAG,OAAO;CAI5C"}
|
|
@@ -1,5 +1,6 @@
|
|
|
1
1
|
import { BaseWorker } from "./BaseWorker.js";
|
|
2
2
|
import { WorkerMessageType } from "./types.js";
|
|
3
|
+
import { loadWorkerManifest } from "./worker-manifest.js";
|
|
3
4
|
const WORKER_FILE_NAMES = {
|
|
4
5
|
videoDemux: "video-demux",
|
|
5
6
|
audioDemux: "audio-demux",
|
|
@@ -15,6 +16,7 @@ class WorkerPool {
|
|
|
15
16
|
workerConfigs;
|
|
16
17
|
workerPath;
|
|
17
18
|
workerExtension;
|
|
19
|
+
manifestPromise = null;
|
|
18
20
|
constructor(config) {
|
|
19
21
|
this.eventBus = config.eventBus;
|
|
20
22
|
this.workerConfigs = config.workerConfigs || {};
|
|
@@ -24,7 +26,7 @@ class WorkerPool {
|
|
|
24
26
|
/**
|
|
25
27
|
* Get worker URL for a specific worker type
|
|
26
28
|
*/
|
|
27
|
-
getWorkerUrl(type) {
|
|
29
|
+
async getWorkerUrl(type) {
|
|
28
30
|
const fileName = WORKER_FILE_NAMES[type];
|
|
29
31
|
const stageMap = {
|
|
30
32
|
"video-demux": "demux",
|
|
@@ -38,8 +40,21 @@ class WorkerPool {
|
|
|
38
40
|
encode: "encode"
|
|
39
41
|
};
|
|
40
42
|
const stage = stageMap[fileName];
|
|
41
|
-
|
|
42
|
-
|
|
43
|
+
if (this.workerExtension === ".ts") {
|
|
44
|
+
return `${this.workerPath}/stages/${stage}/${fileName}.worker.ts`;
|
|
45
|
+
}
|
|
46
|
+
if (!this.manifestPromise) {
|
|
47
|
+
this.manifestPromise = loadWorkerManifest(this.workerPath);
|
|
48
|
+
}
|
|
49
|
+
const manifest = await this.manifestPromise;
|
|
50
|
+
if (manifest) {
|
|
51
|
+
const manifestKey = `${fileName}.worker`;
|
|
52
|
+
const hashedPath = manifest[manifestKey];
|
|
53
|
+
if (hashedPath) {
|
|
54
|
+
return `${this.workerPath}/${hashedPath}`;
|
|
55
|
+
}
|
|
56
|
+
}
|
|
57
|
+
return `${this.workerPath}/stages/${stage}/${fileName}.worker.js`;
|
|
43
58
|
}
|
|
44
59
|
get(type, id) {
|
|
45
60
|
const key = id ? `${type}#${id}` : type;
|
|
@@ -56,7 +71,7 @@ class WorkerPool {
|
|
|
56
71
|
const key = id ? `${type}#${id}` : type;
|
|
57
72
|
const existing = this.pool.get(key);
|
|
58
73
|
if (!existing) {
|
|
59
|
-
const url = this.getWorkerUrl(type);
|
|
74
|
+
const url = await this.getWorkerUrl(type);
|
|
60
75
|
const worker = new BaseWorker({
|
|
61
76
|
type,
|
|
62
77
|
url,
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"WorkerPool.js","sources":["../../src/worker/WorkerPool.ts"],"sourcesContent":["/**\n * WorkerPool: Manages worker instances with key-based access\n * Provides lazy creation and unified management of workers\n */\n\nimport { BaseWorker } from './BaseWorker';\nimport type { WorkerType, WorkerStatus, WorkerConfigurePayload } from './types';\nimport { WorkerMessageType } from './types';\nimport type { EventBus } from '../event/EventBus';\nimport type { EventPayloadMap } from '../event/events';\n\nexport interface WorkerPoolConfig {\n eventBus: EventBus<EventPayloadMap>;\n workerConfigs?: Record<WorkerType, any>;\n /** Worker files base path (default: '/meframe-workers') */\n workerPath?: string;\n workerExtension?: string;\n}\n\n/**\n * Worker name mapping (worker type -> file name)\n */\nconst WORKER_FILE_NAMES: Record<WorkerType, string> = {\n videoDemux: 'video-demux',\n audioDemux: 'audio-demux',\n videoDecode: 'video-decode',\n audioDecode: 'audio-decode',\n videoCompose: 'video-compose',\n audioCompose: 'audio-compose',\n videoEncode: 'video-encode',\n};\n\nexport class WorkerPool {\n private pool = new Map<string, BaseWorker>();\n private eventBus: EventBus<EventPayloadMap>;\n private workerConfigs: Record<string, any>;\n private workerPath: string;\n private workerExtension: string;\n\n constructor(config: WorkerPoolConfig) {\n this.eventBus = config.eventBus;\n this.workerConfigs = config.workerConfigs || {};\n this.workerPath = config.workerPath || '/meframe-workers';\n this.workerExtension = config.workerExtension || '.js';\n }\n\n /**\n * Get worker URL for a specific worker type\n */\n private getWorkerUrl(type: WorkerType): string {\n const fileName = WORKER_FILE_NAMES[type];\n\n // Map worker type to its stage directory\n const stageMap: Record<string, string> = {\n 'video-demux': 'demux',\n 'audio-demux': 'demux',\n 'video-decode': 'decode',\n 'audio-decode': 'decode',\n decode: 'decode',\n 'video-compose': 'compose',\n 'audio-compose': 'compose',\n 'video-encode': 'encode',\n encode: 'encode',\n };\n const stage = stageMap[fileName];\n\n //
|
|
1
|
+
{"version":3,"file":"WorkerPool.js","sources":["../../src/worker/WorkerPool.ts"],"sourcesContent":["/**\n * WorkerPool: Manages worker instances with key-based access\n * Provides lazy creation and unified management of workers\n */\n\nimport { BaseWorker } from './BaseWorker';\nimport type { WorkerType, WorkerStatus, WorkerConfigurePayload } from './types';\nimport { WorkerMessageType } from './types';\nimport type { EventBus } from '../event/EventBus';\nimport type { EventPayloadMap } from '../event/events';\nimport { loadWorkerManifest, type WorkerManifest } from './worker-manifest';\n\nexport interface WorkerPoolConfig {\n eventBus: EventBus<EventPayloadMap>;\n workerConfigs?: Record<WorkerType, any>;\n /** Worker files base path (default: '/meframe-workers') */\n workerPath?: string;\n workerExtension?: string;\n}\n\n/**\n * Worker name mapping (worker type -> file name)\n */\nconst WORKER_FILE_NAMES: Record<WorkerType, string> = {\n videoDemux: 'video-demux',\n audioDemux: 'audio-demux',\n videoDecode: 'video-decode',\n audioDecode: 'audio-decode',\n videoCompose: 'video-compose',\n audioCompose: 'audio-compose',\n videoEncode: 'video-encode',\n};\n\nexport class WorkerPool {\n private pool = new Map<string, BaseWorker>();\n private eventBus: EventBus<EventPayloadMap>;\n private workerConfigs: Record<string, any>;\n private workerPath: string;\n private workerExtension: string;\n private manifestPromise: Promise<WorkerManifest | null> | null = null;\n\n constructor(config: WorkerPoolConfig) {\n this.eventBus = config.eventBus;\n this.workerConfigs = config.workerConfigs || {};\n this.workerPath = config.workerPath || '/meframe-workers';\n this.workerExtension = config.workerExtension || '.js';\n }\n\n /**\n * Get worker URL for a specific worker type\n */\n private async getWorkerUrl(type: WorkerType): Promise<string> {\n const fileName = WORKER_FILE_NAMES[type];\n\n // Map worker type to its stage directory\n const stageMap: Record<string, string> = {\n 'video-demux': 'demux',\n 'audio-demux': 'demux',\n 'video-decode': 'decode',\n 'audio-decode': 'decode',\n decode: 'decode',\n 'video-compose': 'compose',\n 'audio-compose': 'compose',\n 'video-encode': 'encode',\n encode: 'encode',\n };\n const stage = stageMap[fileName];\n\n // Dev mode: use .ts source files directly (no manifest needed)\n if (this.workerExtension === '.ts') {\n return `${this.workerPath}/stages/${stage}/${fileName}.worker.ts`;\n }\n\n // Production mode: try to load manifest for hashed filenames\n if (!this.manifestPromise) {\n this.manifestPromise = loadWorkerManifest(this.workerPath);\n }\n\n const manifest = await this.manifestPromise;\n\n if (manifest) {\n // Use hashed filename from manifest\n const manifestKey = `${fileName}.worker`;\n const hashedPath = manifest[manifestKey];\n\n if (hashedPath) {\n // Manifest contains full relative path from workers directory\n return `${this.workerPath}/${hashedPath}`;\n }\n }\n\n // Fallback: use non-hashed filename (backward compatibility)\n return `${this.workerPath}/stages/${stage}/${fileName}.worker.js`;\n }\n\n get(type: WorkerType, id?: string): BaseWorker | undefined {\n const key = id ? `${type}#${id}` : type;\n return this.pool.get(key) ?? undefined;\n }\n\n /**\n * Get or create a worker instance\n * @param type - Worker type\n * @param id - Optional ID for per-resource or per-clip workers\n * @param options - Optional configuration\n * - lazy: If true, skip initial configure (default: false)\n */\n async getOrCreate(\n type: WorkerType,\n id?: string,\n options?: { lazy?: boolean }\n ): Promise<BaseWorker> {\n const key = id ? `${type}#${id}` : type;\n\n const existing = this.pool.get(key);\n if (!existing) {\n // Generate worker URL based on worker path and type\n const url = await this.getWorkerUrl(type);\n\n const worker = new BaseWorker({\n type,\n url,\n eventBus: this.eventBus,\n clipId: id,\n });\n\n // Only initialize if not in lazy mode\n if (!options?.lazy) {\n const config = this.workerConfigs[type];\n await worker.initialize(config);\n }\n\n this.pool.set(key, worker);\n return worker;\n }\n\n if (!options?.lazy) {\n const config = this.workerConfigs[type];\n if (config) {\n await existing.send(WorkerMessageType.Configure, {\n config,\n initial: false,\n } as WorkerConfigurePayload);\n }\n }\n\n return existing;\n }\n\n async setConfig(type: WorkerType, config: any): Promise<void> {\n const existing = this.workerConfigs[type] || {};\n const mergedConfig = { ...existing, ...config };\n\n this.workerConfigs[type] = mergedConfig;\n\n for (const [key, worker] of this.pool.entries()) {\n if (key === type || key.startsWith(`${type}#`)) {\n await worker.send(WorkerMessageType.Configure, {\n config: mergedConfig,\n initial: false,\n } as WorkerConfigurePayload);\n }\n }\n }\n\n /**\n * Terminate a specific worker\n */\n terminate(type: WorkerType, id?: string): void {\n const key = id ? `${type}#${id}` : type;\n const worker = this.pool.get(key);\n\n if (worker) {\n worker.terminate();\n this.pool.delete(key);\n }\n }\n\n /**\n * Terminate all workers\n */\n terminateAll(): void {\n for (const worker of this.pool.values()) {\n worker.terminate();\n }\n this.pool.clear();\n }\n\n /**\n * Get status of all workers\n */\n get status(): Record<string, WorkerStatus[WorkerType]> {\n const result: Record<string, WorkerStatus[WorkerType]> = {};\n\n for (const [key, worker] of this.pool.entries()) {\n result[key] = worker.status;\n }\n\n return result;\n }\n\n /**\n * Get list of active worker keys\n */\n get activeWorkers(): string[] {\n return Array.from(this.pool.keys());\n }\n\n /**\n * Check if a worker exists\n */\n has(type: WorkerType, id?: string): boolean {\n const key = id ? `${type}#${id}` : type;\n return this.pool.has(key);\n }\n}\n"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Worker manifest for content-hashed worker filenames
|
|
3
|
+
*
|
|
4
|
+
* The manifest maps original worker names to hashed filenames,
|
|
5
|
+
* enabling cache busting for worker files.
|
|
6
|
+
*/
|
|
7
|
+
export interface WorkerManifest {
|
|
8
|
+
[workerName: string]: string;
|
|
9
|
+
}
|
|
10
|
+
/**
|
|
11
|
+
* Load worker manifest from the given worker path
|
|
12
|
+
*
|
|
13
|
+
* @param workerPath - Base path for worker files (e.g., '/meframe-workers')
|
|
14
|
+
* @returns Promise<WorkerManifest | null> - Manifest object or null if not found
|
|
15
|
+
*
|
|
16
|
+
* The manifest maps worker names to their hashed filenames:
|
|
17
|
+
* {
|
|
18
|
+
* "video-decode.worker": "stages/decode/video-decode.worker.a1b2c3d4.js",
|
|
19
|
+
* "video-demux.worker": "stages/demux/video-demux.worker.e5f6g7h8.js"
|
|
20
|
+
* }
|
|
21
|
+
*/
|
|
22
|
+
export declare function loadWorkerManifest(workerPath: string): Promise<WorkerManifest | null>;
|
|
23
|
+
/**
|
|
24
|
+
* Reset the cached manifest (useful for testing)
|
|
25
|
+
*/
|
|
26
|
+
export declare function resetManifestCache(): void;
|
|
27
|
+
//# sourceMappingURL=worker-manifest.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"worker-manifest.d.ts","sourceRoot":"","sources":["../../src/worker/worker-manifest.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AAEH,MAAM,WAAW,cAAc;IAC7B,CAAC,UAAU,EAAE,MAAM,GAAG,MAAM,CAAC;CAC9B;AAOD;;;;;;;;;;;GAWG;AACH,wBAAsB,kBAAkB,CAAC,UAAU,EAAE,MAAM,GAAG,OAAO,CAAC,cAAc,GAAG,IAAI,CAAC,CAyC3F;AAED;;GAEG;AACH,wBAAgB,kBAAkB,IAAI,IAAI,CAEzC"}
|
|
@@ -0,0 +1,37 @@
|
|
|
1
|
+
let cachedManifest = void 0;
|
|
2
|
+
async function loadWorkerManifest(workerPath) {
|
|
3
|
+
if (cachedManifest !== void 0) {
|
|
4
|
+
return cachedManifest;
|
|
5
|
+
}
|
|
6
|
+
try {
|
|
7
|
+
const manifestUrl = `${workerPath}/worker-manifest.json`;
|
|
8
|
+
const response = await fetch(manifestUrl);
|
|
9
|
+
if (!response.ok) {
|
|
10
|
+
console.warn(
|
|
11
|
+
`[WorkerManifest] Manifest not found at ${manifestUrl}, falling back to non-hashed filenames`
|
|
12
|
+
);
|
|
13
|
+
cachedManifest = null;
|
|
14
|
+
return null;
|
|
15
|
+
}
|
|
16
|
+
const manifest = await response.json();
|
|
17
|
+
if (typeof manifest !== "object" || manifest === null) {
|
|
18
|
+
console.error("[WorkerManifest] Invalid manifest format");
|
|
19
|
+
cachedManifest = null;
|
|
20
|
+
return null;
|
|
21
|
+
}
|
|
22
|
+
cachedManifest = manifest;
|
|
23
|
+
console.log(`[WorkerManifest] Loaded manifest with ${Object.keys(manifest).length} worker(s)`);
|
|
24
|
+
return manifest;
|
|
25
|
+
} catch (error) {
|
|
26
|
+
console.warn(
|
|
27
|
+
"[WorkerManifest] Failed to load manifest, using fallback:",
|
|
28
|
+
error instanceof Error ? error.message : String(error)
|
|
29
|
+
);
|
|
30
|
+
cachedManifest = null;
|
|
31
|
+
return null;
|
|
32
|
+
}
|
|
33
|
+
}
|
|
34
|
+
export {
|
|
35
|
+
loadWorkerManifest
|
|
36
|
+
};
|
|
37
|
+
//# sourceMappingURL=worker-manifest.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"worker-manifest.js","sources":["../../src/worker/worker-manifest.ts"],"sourcesContent":["/**\n * Worker manifest for content-hashed worker filenames\n *\n * The manifest maps original worker names to hashed filenames,\n * enabling cache busting for worker files.\n */\n\nexport interface WorkerManifest {\n [workerName: string]: string;\n}\n\n/**\n * Cached manifest to avoid repeated fetches\n */\nlet cachedManifest: WorkerManifest | null | undefined = undefined;\n\n/**\n * Load worker manifest from the given worker path\n *\n * @param workerPath - Base path for worker files (e.g., '/meframe-workers')\n * @returns Promise<WorkerManifest | null> - Manifest object or null if not found\n *\n * The manifest maps worker names to their hashed filenames:\n * {\n * \"video-decode.worker\": \"stages/decode/video-decode.worker.a1b2c3d4.js\",\n * \"video-demux.worker\": \"stages/demux/video-demux.worker.e5f6g7h8.js\"\n * }\n */\nexport async function loadWorkerManifest(workerPath: string): Promise<WorkerManifest | null> {\n // Return cached result if available\n if (cachedManifest !== undefined) {\n return cachedManifest;\n }\n\n try {\n const manifestUrl = `${workerPath}/worker-manifest.json`;\n\n const response = await fetch(manifestUrl);\n\n if (!response.ok) {\n console.warn(\n `[WorkerManifest] Manifest not found at ${manifestUrl}, falling back to non-hashed filenames`\n );\n cachedManifest = null;\n return null;\n }\n\n const manifest: WorkerManifest = await response.json();\n\n // Validate manifest structure\n if (typeof manifest !== 'object' || manifest === null) {\n console.error('[WorkerManifest] Invalid manifest format');\n cachedManifest = null;\n return null;\n }\n\n cachedManifest = manifest;\n console.log(`[WorkerManifest] Loaded manifest with ${Object.keys(manifest).length} worker(s)`);\n\n return manifest;\n } catch (error) {\n // Network error or invalid JSON - fall back to non-hashed filenames\n console.warn(\n '[WorkerManifest] Failed to load manifest, using fallback:',\n error instanceof Error ? error.message : String(error)\n );\n cachedManifest = null;\n return null;\n }\n}\n\n/**\n * Reset the cached manifest (useful for testing)\n */\nexport function resetManifestCache(): void {\n cachedManifest = undefined;\n}\n"],"names":[],"mappings":"AAcA,IAAI,iBAAoD;AAcxD,eAAsB,mBAAmB,YAAoD;AAE3F,MAAI,mBAAmB,QAAW;AAChC,WAAO;AAAA,EACT;AAEA,MAAI;AACF,UAAM,cAAc,GAAG,UAAU;AAEjC,UAAM,WAAW,MAAM,MAAM,WAAW;AAExC,QAAI,CAAC,SAAS,IAAI;AAChB,cAAQ;AAAA,QACN,0CAA0C,WAAW;AAAA,MAAA;AAEvD,uBAAiB;AACjB,aAAO;AAAA,IACT;AAEA,UAAM,WAA2B,MAAM,SAAS,KAAA;AAGhD,QAAI,OAAO,aAAa,YAAY,aAAa,MAAM;AACrD,cAAQ,MAAM,0CAA0C;AACxD,uBAAiB;AACjB,aAAO;AAAA,IACT;AAEA,qBAAiB;AACjB,YAAQ,IAAI,yCAAyC,OAAO,KAAK,QAAQ,EAAE,MAAM,YAAY;AAE7F,WAAO;AAAA,EACT,SAAS,OAAO;AAEd,YAAQ;AAAA,MACN;AAAA,MACA,iBAAiB,QAAQ,MAAM,UAAU,OAAO,KAAK;AAAA,IAAA;AAEvD,qBAAiB;AACjB,WAAO;AAAA,EACT;AACF;"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"BaseDecoder.Bk26nCBk.js","sources":["../../src/stages/decode/BaseDecoder.ts"],"sourcesContent":["/**\n * Base decoder class abstracting common WebCodecs decoder workflow.\n * Mirrors BaseEncoder but for decoding path.\n */\nimport { AudioDecoderConfig, DecodedVideoFrame, VideoDecoderConfig } from './types';\n\nexport abstract class BaseDecoder<\n TDecoder extends VideoDecoder | AudioDecoder,\n TConfig extends VideoDecoderConfig | AudioDecoderConfig,\n TInput extends EncodedVideoChunk | EncodedAudioChunk,\n TOutput extends VideoFrame | AudioData | DecodedVideoFrame,\n> {\n protected decoder?: TDecoder;\n protected config: TConfig;\n protected controller: TransformStreamDefaultController<TOutput> | null = null;\n\n constructor(config: TConfig) {\n this.config = config;\n }\n\n async initialize(): Promise<void> {\n if (this.decoder?.state === 'configured') {\n return;\n }\n const isSupported = await this.isConfigSupported(this.config);\n if (!isSupported.supported) {\n throw new Error(\n `Codec not supported: ${(this.config as any).codecString || (this.config as any).codec}`\n );\n }\n\n this.decoder = this.createDecoder({\n output: this.handleOutput.bind(this),\n error: this.handleError.bind(this),\n });\n\n await this.configureDecoder(this.config);\n }\n\n async reconfigure(config: Partial<TConfig>): Promise<void> {\n this.config = { ...this.config, ...config } as TConfig;\n\n if (!this.decoder) {\n await this.initialize();\n return;\n }\n\n // Flush before reconfiguring\n if (this.decoder.state === 'configured') {\n await this.decoder.flush();\n }\n\n const isSupported = await this.isConfigSupported(this.config);\n if (!isSupported.supported) {\n throw new Error(\n `New configuration not supported: ${(this.config as any).codecString || (this.config as any).codec}`\n );\n }\n\n await this.configureDecoder(this.config);\n }\n\n async flush(): Promise<void> {\n if (!this.decoder) return;\n await this.decoder.flush();\n }\n\n async reset(): Promise<void> {\n if (!this.decoder) return;\n this.decoder.reset();\n this.onReset();\n }\n\n async close(): Promise<void> {\n if (!this.decoder) return;\n\n if (this.decoder.state === 'configured') {\n await this.decoder.flush();\n }\n\n this.decoder.close();\n this.decoder = undefined;\n }\n\n get isReady(): boolean {\n return this.decoder?.state === 'configured';\n }\n\n get queueSize(): number {\n return (this.decoder as any)?.decodeQueueSize ?? 0;\n }\n\n protected handleOutput(data: TOutput): void {\n if (!this.controller) {\n this.closeIfPossible(data);\n return;\n }\n\n try {\n this.controller.enqueue(data);\n } catch (error) {\n // Stream closed or errored - silently drop frame\n if (error instanceof TypeError) {\n this.closeIfPossible(data);\n return;\n }\n\n throw error;\n }\n }\n\n protected handleError(error: DOMException): void {\n console.error(`${this.getDecoderType()} decoder error:`, error);\n this.controller?.error(error);\n }\n\n private closeIfPossible(data: TOutput): void {\n (data as any)?.close?.();\n (data as DecodedVideoFrame)?.frame?.close?.();\n }\n\n // Abstracts\n protected abstract isConfigSupported(config: TConfig): Promise<{ supported: boolean }>;\n protected abstract createDecoder(init: DecoderInit): TDecoder;\n protected abstract configureDecoder(config: TConfig): Promise<void>;\n protected abstract getDecoderType(): string;\n protected abstract decode(input: TInput): void;\n protected onReset(): void {}\n\n // Backpressure\n protected abstract readonly highWaterMark: number;\n protected abstract readonly decodeQueueThreshold: number;\n\n createStream(): TransformStream<TInput, TOutput> {\n return new TransformStream<TInput, TOutput>(\n {\n start: async (controller) => {\n this.controller = controller;\n if (!this.isReady) {\n await this.initialize();\n }\n },\n transform: async (input) => {\n if (!this.decoder || this.decoder.state !== 'configured') {\n throw new Error('Decoder not configured');\n }\n // backpressure\n if ((this.decoder as any).decodeQueueSize >= this.decodeQueueThreshold) {\n await new Promise<void>((resolve) => {\n const check = () => {\n if (\n !this.decoder ||\n (this.decoder as any).decodeQueueSize < this.decodeQueueThreshold - 1\n ) {\n resolve();\n } else {\n setTimeout(check, 10);\n }\n };\n check();\n });\n }\n this.decode(input);\n },\n flush: async () => {\n await this.flush();\n },\n },\n {\n highWaterMark: this.highWaterMark,\n size: () => 1,\n }\n );\n }\n}\n\ninterface DecoderInit {\n output: (data: any) => void;\n error: (error: DOMException) => void;\n}\n"],"names":[],"mappings":"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"}
|