pup-recorder 0.0.8 → 0.0.10

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/.gitattributes ADDED
@@ -0,0 +1,2 @@
1
+ rust/** filter=lfs diff=lfs merge=lfs -text
2
+ x265/** filter=lfs diff=lfs merge=lfs -text
package/dist/cjs/app.cjs CHANGED
@@ -341,6 +341,18 @@ function stopSync(cdp) {
341
341
  });
342
342
  }
343
343
 
344
+ // src/base/html_check.ts
345
+ var SUPPORTED_PROTOCOLS = ["file:", "http:", "https:", "data:"];
346
+ var SOURCE_PATTERN = /^(file:|https?:|data:)/;
347
+ function checkHTML(source) {
348
+ if (SOURCE_PATTERN.test(source)) {
349
+ return;
350
+ }
351
+ const protocol = source.split(":")[0] + ":";
352
+ const message = SUPPORTED_PROTOCOLS.includes(protocol) ? `unsupported protocol: ${protocol}, expected ${SUPPORTED_PROTOCOLS.join(", ")}` : `invalid source format, expected ${SUPPORTED_PROTOCOLS.join(", ")}`;
353
+ throw new Error(message);
354
+ }
355
+
344
356
  // src/base/image.ts
345
357
  function isEmpty(image) {
346
358
  const size = image.getSize();
@@ -417,9 +429,7 @@ function useRetry({
417
429
  // src/base/record.ts
418
430
  var TAG2 = "[Record]";
419
431
  async function loadWindow(source, options) {
420
- if (!source.startsWith("file://") && !source.match(/^https?:\/\//)) {
421
- throw new Error("invalid source");
422
- }
432
+ checkHTML(source);
423
433
  const { width, height, useInnerProxy } = options;
424
434
  import_electron3.session.defaultSession.webRequest.onHeadersReceived((details, callback) => {
425
435
  const responseHeaders = { ...details.responseHeaders };
@@ -596,7 +606,7 @@ var DEFAULT_FPS = 30;
596
606
  var DEFAULT_DURATION = 5;
597
607
  var DEFAULT_OUT_DIR = "out";
598
608
  function makeCLI(name, callback) {
599
- import_commander.program.name(name).argument("<source>", "URL \u6216 HTML data").option("-w, --width <number>", "\u89C6\u9891\u5BBD\u5EA6", `${DEFAULT_WIDTH}`).option("-h, --height <number>", "\u89C6\u9891\u9AD8\u5EA6", `${DEFAULT_HEIGHT}`).option("-f, --fps <number>", "\u5E27\u7387", `${DEFAULT_FPS}`).option("-t, --duration <number>", "\u5F55\u5236\u65F6\u957F\uFF08\u79D2\uFF09", `${DEFAULT_DURATION}`).option("-o, --out-dir <path>", "\u8F93\u51FA\u76EE\u5F55", `${DEFAULT_OUT_DIR}`).option("-a, --with-alpha-channel", "\u8F93\u51FA\u5305\u542B alpha \u901A\u9053\u7684\u89C6\u9891", false).option(
609
+ import_commander.program.name(name).argument("<source>", "file://, http(s)://, \u6216 data: URI").option("-w, --width <number>", "\u89C6\u9891\u5BBD\u5EA6", `${DEFAULT_WIDTH}`).option("-h, --height <number>", "\u89C6\u9891\u9AD8\u5EA6", `${DEFAULT_HEIGHT}`).option("-f, --fps <number>", "\u5E27\u7387", `${DEFAULT_FPS}`).option("-t, --duration <number>", "\u5F55\u5236\u65F6\u957F\uFF08\u79D2\uFF09", `${DEFAULT_DURATION}`).option("-o, --out-dir <path>", "\u8F93\u51FA\u76EE\u5F55", `${DEFAULT_OUT_DIR}`).option("-a, --with-alpha-channel", "\u8F93\u51FA\u5305\u542B alpha \u901A\u9053\u7684\u89C6\u9891", false).option(
600
610
  "--use-inner-proxy",
601
611
  "\u4F7F\u7528 B \u7AD9\u5185\u7F51\u4EE3\u7406\u52A0\u901F\u8D44\u6E90\u8BBF\u95EE",
602
612
  pupUseInnerProxy
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
- "sources": ["../../src/app.ts", "../../src/base/electron.ts", "../../src/base/constants.ts", "../../src/base/env.ts", "../../src/base/parser.ts", "../../src/base/logging.ts", "../../src/base/process.ts", "../../src/base/record.ts", "../../src/rust/lib.ts", "../../src/base/frame_sync.ts", "../../src/base/image.ts", "../../src/base/proxy.ts", "../../src/base/retry.ts", "../../src/base/timing.ts", "../../src/common.ts", "../../src/base/noerr.ts"],
4
- "sourcesContent": ["// Created by Autokaka (qq1909698494@gmail.com) on 2026/01/30.\n\nimport { app } from \"electron\";\nimport { ELECTRON_OPTS } from \"./base/electron\";\nimport { record, type RecordOptions } from \"./base/record\";\nimport { makeCLI } from \"./common\";\n\nprocess.once(\"exit\", () => app.quit());\n\nmakeCLI(\"app\", async (source: string, options: RecordOptions) => {\n try {\n ELECTRON_OPTS.forEach((o) => app.commandLine.appendSwitch(o));\n app.dock?.hide();\n await app.whenReady();\n await record(source, options);\n } finally {\n app.quit();\n }\n});\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/25.\n\nimport electron, { type Size } from \"electron\";\nimport { exec } from \"./process\";\n\nexport const ELECTRON_OPTS = [\n \"no-sandbox\",\n \"disable-setuid-sandbox\",\n \"disable-gpu\",\n \"disable-dev-shm-usage\",\n \"disable-software-rasterizer\",\n \"disable-web-security\",\n \"disable-site-isolation-trials\",\n \"disable-features=IsolateOrigins,site-per-process\",\n \"allow-insecure-localhost\",\n \"ignore-certificate-errors\",\n \"disable-blink-features=AutomationControlled\",\n \"mute-audio\",\n \"disable-extensions\",\n \"disable-background-networking\",\n \"address-family=ipv4\",\n \"disable-async-dns\",\n \"force-device-scale-factor=1\",\n \"trace-warnings\",\n \"force-color-profile=srgb\",\n \"disable-color-correct-rendering\",\n \"log-level=3\",\n];\n\nexport function runElectronApp(size: Size, app: unknown, args: unknown[]) {\n const electronArgs = ELECTRON_OPTS.map((a) => `--${a}`);\n const cmdParts = [];\n if (process.platform === \"linux\") {\n cmdParts.push(\n \"xvfb-run\",\n \"--auto-servernum\",\n `--server-args='-screen 0 ${size.width}x${size.height}x24'`,\n );\n }\n cmdParts.push(electron, ...electronArgs, app);\n return exec(cmdParts.join(\" \"), {\n stdio: [\"ignore\", \"pipe\", \"pipe\"],\n shell: true,\n env: {\n ...process.env,\n ELECTRON_DISABLE_DBUS: \"1\",\n RUST_BACKTRACE: \"full\",\n __PUP_ARGS__: JSON.stringify(args),\n },\n });\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nimport { existsSync } from \"fs\";\nimport { resolve } from \"path\";\nimport { penv } from \"./env\";\nimport { parseNumber } from \"./parser\";\n\nconst pupAppSearchPaths = [\n resolve(__dirname, \"cjs/app.cjs\"), // process from dist\n resolve(__dirname, \"app.cjs\"), // process from dist/cjs\n resolve(__dirname, \"../../cjs/app.cjs\"), // process from src\n];\nexport const pupAppPath = pupAppSearchPaths.find(existsSync)!;\n\nconst env = process.env;\nexport const pupLogLevel = penv(\"PUP_LOG_LEVEL\", parseNumber, 2);\nexport const pupUseInnerProxy = env[\"PUP_USE_INNER_PROXY\"] === \"1\";\nexport const pupFFmpegPath = env[\"FFMPEG_BIN\"] ?? `ffmpeg`;\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/25.\n\nexport type EnvParser<T> = (value: unknown) => T;\n\nexport function penv<T>(name: string, parser: EnvParser<T>, defaultValue: T): T;\nexport function penv<T>(\n name: string,\n parser: EnvParser<T>,\n defaultValue?: T,\n): T | undefined;\nexport function penv<T>(\n name: string,\n parser: EnvParser<T>,\n defaultValue?: T,\n): T | undefined {\n try {\n return parser(process.env[name]);\n } catch {\n return defaultValue;\n }\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/01/30.\n\nexport function parseNumber(value: unknown): number {\n if (typeof value === \"number\") {\n return value;\n }\n const num = Number(value);\n if (Number.isNaN(num)) {\n throw new Error(`Value ${value} is not a valid number`);\n }\n return num;\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/06.\n\nimport { ChildProcess, type Serializable } from \"child_process\";\nimport { pupLogLevel } from \"./constants\";\n\nexport interface LoggerLike {\n debug?(this: void, ...messages: unknown[]): void;\n\n info?(this: void, ...messages: unknown[]): void;\n\n warn?(this: void, ...messages: unknown[]): void;\n\n error?(this: void, ...messages: unknown[]): void;\n}\n\nconst DEBUG = \"<pup@debug>\";\nconst INFO = \"<pup@info>\";\nconst WARN = \"<pup@warn>\";\nconst ERROR = \"<pup@error>\";\nconst FATAL = \"<pup@fatal>\";\n\nclass Logger implements LoggerLike {\n private _impl?: LoggerLike;\n\n get impl(): LoggerLike | undefined {\n return this._impl;\n }\n\n set impl(value: LoggerLike) {\n const debug = value.debug ?? console.debug;\n const info = value.info ?? console.info;\n const warn = value.warn ?? console.warn;\n const error = value.error ?? console.error;\n this._impl = {\n debug: pupLogLevel >= 3 ? debug : undefined,\n info: pupLogLevel >= 2 ? info : undefined,\n warn: pupLogLevel >= 1 ? warn : undefined,\n error: pupLogLevel >= 0 ? error : undefined,\n };\n }\n\n constructor() {\n this.impl = console;\n }\n\n debug(...messages: unknown[]): void {\n this.impl?.debug?.(DEBUG, ...messages);\n }\n\n info(...messages: unknown[]): void {\n this.impl?.info?.(INFO, ...messages);\n }\n\n warn(...messages: unknown[]): void {\n this.impl?.warn?.(WARN, ...messages);\n }\n\n error(...messages: unknown[]): void {\n this.impl?.error?.(ERROR, ...messages);\n }\n\n fatal(...messages: unknown[]): never {\n this.impl?.error?.(FATAL, ...messages);\n process.exit(1);\n }\n\n private dispatch(message: string) {\n if (message.startsWith(DEBUG)) {\n this.debug(message.slice(DEBUG.length + 1));\n } else if (message.startsWith(INFO)) {\n this.info(message.slice(INFO.length + 1));\n } else if (message.startsWith(WARN)) {\n this.warn(message.slice(WARN.length + 1));\n } else if (message.startsWith(ERROR)) {\n this.error(message.slice(ERROR.length + 1));\n } else {\n this.info(message);\n }\n }\n\n attach(proc: ChildProcess, name: string) {\n return new Promise<void>((resolve, reject) => {\n this.debug(`${name}.attach`);\n let fatal: string = \"\";\n const dispatch = (data: Buffer | Serializable) => {\n const message = data.toString();\n if (message.startsWith(FATAL)) {\n fatal += message.slice(FATAL.length + 1);\n } else {\n this.dispatch(message);\n }\n };\n proc.stderr?.on(\"data\", dispatch);\n proc.stdout?.on(\"data\", dispatch);\n proc\n .on(\"message\", dispatch)\n .on(\"error\", (err) => {\n fatal += err.message;\n proc.kill();\n })\n .once(\"close\", (code, signal) => {\n if (code || signal || fatal) {\n fatal ||= `command failed: ${proc.spawnargs.join(\" \")}`;\n this.error(`${name}.close`, { code, signal, fatal });\n reject(new Error(fatal));\n } else {\n this.debug(`${name}.close`);\n resolve();\n }\n })\n .on(\"unhandledRejection\", (reason) => {\n this.error(`${name}.unhandled`, reason);\n })\n .on(\"uncaughtExceptionMonitor\", (err) => {\n this.error(`${name}.unhandled`, err);\n });\n });\n }\n}\n\nconst logger = new Logger();\n\nexport { logger };\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/01/30.\n\nimport { spawn, type ChildProcess, type SpawnOptions } from \"child_process\";\nimport { logger } from \"./logging\";\n\nexport const PUP_ARGS_ENV_KEY = \"__PUP_ARGS__\";\n\nexport function pargs() {\n const pupArgs = process.env[PUP_ARGS_ENV_KEY];\n if (pupArgs) {\n const args = [\"exec\", ...process.argv.slice(-1)];\n args.push(...JSON.parse(pupArgs));\n logger.debug(\"pupargs\", args);\n return args;\n }\n\n logger.debug(\"procargv\", process.argv);\n return process.argv;\n}\n\nexport interface ProcessHandle {\n process: ChildProcess;\n wait: Promise<void>;\n}\n\nexport function exec(cmd: string, options?: SpawnOptions): ProcessHandle {\n const parts = cmd.split(\" \").filter((s) => s.length);\n const [command, ...args] = parts;\n if (!command) throw new Error(\"empty command\");\n const proc = spawn(command, args, {\n stdio: \"inherit\",\n ...options,\n });\n return { process: proc, wait: logger.attach(proc, command) };\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nimport { BrowserWindow, session, type NativeImage } from \"electron\";\nimport { mkdir, writeFile } from \"fs/promises\";\nimport { join } from \"path\";\nimport { FixedBufferWriter } from \"../rust/lib\";\nimport {\n buildWrapperHTML,\n decodeTimestamp,\n startSync,\n stopSync,\n} from \"./frame_sync\";\nimport { isEmpty } from \"./image\";\nimport { logger } from \"./logging\";\nimport { enableProxy, proxiedUrl } from \"./proxy\";\nimport { useRetry } from \"./retry\";\n\nconst TAG = \"[Record]\";\n\nexport interface RecordOptions {\n outDir: string;\n duration: number;\n fps: number;\n width: number;\n height: number;\n withAlphaChannel: boolean;\n useInnerProxy: boolean;\n}\n\nexport interface RecordResult {\n options: RecordOptions;\n written: number;\n bgraPath: string;\n}\n\nasync function loadWindow(source: string, options: RecordOptions) {\n if (!source.startsWith(\"file://\") && !source.match(/^https?:\\/\\//)) {\n throw new Error(\"invalid source\");\n }\n\n const { width, height, useInnerProxy } = options;\n\n // Remove X-Frame-Options and CSP headers to allow iframe loading\n // Note: Electron's onHeadersReceived uses REPLACE behavior (not append).\n // Only the last attached listener is used, so no cleanup needed.\n // Ref: https://www.electronjs.org/docs/latest/api/web-request\n session.defaultSession.webRequest.onHeadersReceived((details, callback) => {\n const responseHeaders = { ...details.responseHeaders };\n delete responseHeaders[\"x-frame-options\"];\n delete responseHeaders[\"X-Frame-Options\"];\n delete responseHeaders[\"content-security-policy\"];\n delete responseHeaders[\"Content-Security-Policy\"];\n callback({ cancel: false, responseHeaders });\n });\n\n let src = source;\n if (useInnerProxy) {\n src = proxiedUrl(source);\n enableProxy();\n }\n\n const win = new BrowserWindow({\n width: width,\n height: height + 1,\n show: false,\n transparent: true,\n backgroundColor: undefined,\n webPreferences: {\n offscreen: true,\n backgroundThrottling: false,\n nodeIntegration: true,\n contextIsolation: false,\n webSecurity: false,\n allowRunningInsecureContent: true,\n experimentalFeatures: true,\n },\n });\n\n win.webContents.on(\"console-message\", (event) => {\n if (event.level === \"error\") {\n logger.error(TAG, \"console:\", event.message);\n }\n });\n\n const wrapperHTML = buildWrapperHTML(src, { width, height });\n const dataURL = `data:text/html;charset=utf-8,${encodeURIComponent(wrapperHTML)}`;\n let token: NodeJS.Timeout | undefined;\n\n await new Promise<void>((resolve, reject) => {\n token = setTimeout(() => {\n reject(new Error(\"load window timeout\"));\n }, 20 * 1000);\n\n win.webContents.once(\"did-finish-load\", resolve);\n\n win.webContents.once(\"did-fail-load\", (_event, code, desc, url) => {\n reject(new Error(`failed to load ${url}: [${code}] ${desc}`));\n });\n\n win.webContents.once(\"render-process-gone\", (_event, details) => {\n const { exitCode, reason } = details;\n reject(new Error(`renderer crashed: ${exitCode}, ${reason}`));\n });\n\n win.loadURL(dataURL);\n });\n clearTimeout(token);\n return win;\n}\n\nexport async function record(\n source: string,\n options: RecordOptions,\n): Promise<void> {\n logger.info(TAG, `progress: 0%`);\n const { outDir, fps, width, height, duration } = options;\n\n const win = await useRetry({ fn: loadWindow, maxAttempts: 2 })(\n source,\n options,\n );\n\n await mkdir(outDir, { recursive: true });\n\n const cdp = win.webContents.debugger;\n cdp.attach(\"1.3\");\n\n win.webContents.setFrameRate(fps);\n if (!win.webContents.isPainting()) {\n win.webContents.startPainting();\n }\n\n const bgraPath = join(outDir, \"output.bgra\");\n const total = Math.ceil(fps * duration);\n const frameInterval = 1000 / fps;\n const bufferSize = width * height * 4;\n\n const writer = new FixedBufferWriter(bgraPath, bufferSize, fps);\n\n let written = 0;\n let lastWrittenTime: number | undefined;\n let progress = 0;\n let frameError: Error | undefined;\n let resolver: (() => void) | undefined;\n let rejecter: ((reason?: unknown) => void) | undefined;\n\n const scheduleWrite = (buffer: Buffer) => {\n written++;\n try {\n writer.write(buffer);\n } catch (error) {\n frameError ??= error as Error;\n }\n };\n\n const paint = (_e: unknown, _r: unknown, image: NativeImage) => {\n if (frameError) {\n rejecter?.(frameError);\n return;\n }\n\n if (written >= total) {\n resolver?.();\n return;\n }\n\n if (isEmpty(image)) return;\n\n const bitmap = image.toBitmap();\n const currentTime = decodeTimestamp(bitmap, image.getSize());\n if (currentTime === undefined) {\n frameError ??= new Error(`no timestamp @ ${written}`);\n return;\n }\n\n const bytesPerRow = width * 4;\n const cropped = bitmap.subarray(0, height * bytesPerRow);\n\n if (lastWrittenTime === undefined) {\n scheduleWrite(cropped);\n lastWrittenTime = currentTime;\n return;\n }\n\n const timeSinceLastFrame = currentTime - lastWrittenTime;\n if (timeSinceLastFrame < frameInterval * 0.8) {\n return;\n }\n\n if (timeSinceLastFrame <= frameInterval * 1.2) {\n scheduleWrite(cropped);\n } else {\n const framesToInsert = Math.round(timeSinceLastFrame / frameInterval);\n for (let i = 0; i < framesToInsert && written < total; i++) {\n scheduleWrite(cropped);\n }\n }\n lastWrittenTime = currentTime;\n\n const newProgress = Math.floor((written / total) * 100);\n if (Math.abs(newProgress - progress) > 10) {\n progress = newProgress;\n logger.info(TAG, `progress: ${Math.round(progress)}%`);\n }\n };\n\n win.webContents.on(\"paint\", paint);\n await startSync(cdp);\n try {\n await new Promise<void>((r, j) => ([resolver, rejecter] = [r, j]));\n } finally {\n await stopSync(cdp);\n win.webContents.off(\"paint\", paint);\n await writer.close();\n }\n\n if (frameError || written === 0) {\n throw frameError ?? new Error(\"no frames captured\");\n }\n\n try {\n const result: RecordResult = { options, written, bgraPath };\n await writeFile(join(outDir, \"record.json\"), JSON.stringify(result));\n logger.info(TAG, `progress: 100%, ${written} frames written`);\n } finally {\n win.close();\n }\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/10.\n\nimport { existsSync } from \"fs\";\nimport { join } from \"path\";\n\nconst { platform, arch } = process;\n\nconst rustPath = `rust/${platform}-${arch}.node`;\n\nconst nativeSearchPaths = [\n join(__dirname, `../../${rustPath}`), // process start from src\n join(__dirname, `../${rustPath}`), // process start from dist\n];\nconst mod = require(nativeSearchPaths.find(existsSync)!);\n\nexport interface FixedBufferWriter {\n new (\n path: string,\n bufferSize: number,\n queueDepth?: number,\n ): FixedBufferWriter;\n\n write(buffer: Buffer): void;\n\n close(): Promise<void>;\n}\n\nexport const FixedBufferWriter = mod.FixedBufferWriter as FixedBufferWriter;\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nimport type { Debugger, Size } from \"electron\";\n\nexport const FRAME_SYNC_MARKER_WIDTH = 32;\nexport const FRAME_SYNC_MARKER_HEIGHT = 1;\n\nexport function buildWrapperHTML(targetURL: string, size: Size): string {\n const { width, height } = size;\n return `<!DOCTYPE html>\n<html>\n<head>\n <meta charset=\"UTF-8\">\n <style>\n * { margin: 0; padding: 0; box-sizing: border-box; }\n html, body { width: ${width}px; height: ${height + 1}px; overflow: hidden; }\n #target { \n position: absolute; \n top: 0; \n left: 0; \n width: ${width}px; \n height: ${height}px; \n border: none; \n display: block;\n }\n #stego { \n position: absolute; \n top: ${height}px; \n left: 0; \n width: ${width}px; \n height: 1px; \n display: block;\n image-rendering: pixelated;\n }\n </style>\n</head>\n<body>\n <iframe id=\"target\" src=\"${targetURL}\"></iframe>\n <canvas id=\"stego\" width=\"${width}\" height=\"1\"></canvas>\n <script>\n (function() {\n const WIDTH = ${width};\n const MARKER_WIDTH = ${FRAME_SYNC_MARKER_WIDTH};\n const canvas = document.getElementById('stego');\n const ctx = canvas.getContext('2d', { willReadFrequently: true });\n let startTime = null;\n let rafId = null;\n\n function encodeTimestamp(timestampMs) {\n const imageData = ctx.createImageData(WIDTH, 1);\n const data = imageData.data;\n \n const timestampInt = Math.floor(timestampMs) >>> 0;\n \n for (let i = 0; i < MARKER_WIDTH; i++) {\n const bit = (timestampInt >>> (MARKER_WIDTH - 1 - i)) & 1;\n const value = bit ? 255 : 0;\n const idx = i * 4;\n data[idx] = value;\n data[idx + 1] = value;\n data[idx + 2] = value;\n data[idx + 3] = 255;\n }\n \n for (let i = MARKER_WIDTH; i < WIDTH; i++) {\n const idx = i * 4;\n data[idx] = 0;\n data[idx + 1] = 0;\n data[idx + 2] = 0;\n data[idx + 3] = 255;\n }\n \n ctx.putImageData(imageData, 0, 0);\n }\n\n function updateLoop() {\n if (startTime === null) return;\n const elapsed = performance.now() - startTime;\n encodeTimestamp(elapsed);\n rafId = requestAnimationFrame(updateLoop);\n }\n\n window.__pup_start_recording__ = () => {\n startTime = performance.now();\n encodeTimestamp(0);\n requestAnimationFrame(updateLoop);\n };\n\n window.__pup_stop_recording__ = () => {\n if (rafId !== null) {\n cancelAnimationFrame(rafId);\n rafId = null;\n }\n };\n })();\n </script>\n</body>\n</html>`;\n}\n\nexport function decodeTimestamp(\n bitmap: Buffer,\n size: Size,\n): number | undefined {\n const { width, height } = size;\n if (width < FRAME_SYNC_MARKER_WIDTH || height < 2) {\n return undefined;\n }\n\n const markerRow = height - 1;\n\n let timestamp = 0;\n for (let i = 0; i < FRAME_SYNC_MARKER_WIDTH; i++) {\n const pixelIdx = (markerRow * width + i) * 4;\n const r = bitmap[pixelIdx] ?? 0;\n const bit = r > 127 ? 1 : 0;\n timestamp = (timestamp << 1) | bit;\n }\n\n timestamp = timestamp >>> 0;\n\n if (!Number.isFinite(timestamp) || timestamp < 0 || timestamp > 1e7) {\n return undefined;\n }\n\n return timestamp;\n}\n\nexport function startSync(cdp: Debugger) {\n return cdp.sendCommand(\"Runtime.evaluate\", {\n expression: `window.__pup_start_recording__()`,\n });\n}\n\nexport function stopSync(cdp: Debugger) {\n return cdp.sendCommand(\"Runtime.evaluate\", {\n expression: `window.__pup_stop_recording__()`,\n });\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/06.\n\nimport type { NativeImage } from \"electron\";\n\nexport function isEmpty(image: NativeImage) {\n const size = image.getSize();\n if (size.width === 0 || size.height === 0) return true;\n return image.isEmpty();\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nimport { session } from \"electron\";\nimport { logger } from \"./logging\";\n\nconst TAG = \"[Proxy]\";\n\nexport function proxiedUrl(url: string) {\n if (!url.startsWith(\"http\")) {\n return url;\n }\n // Redirect boss.hdslb.com to boss.bilibili.co\n const match = url.match(/^https:\\/\\/([^-]+)-boss\\.hdslb\\.com(.*)$/);\n if (match) {\n const [, prefix, path] = match;\n return `http://${prefix}-boss.bilibili.co${path}`;\n }\n return url;\n}\n\nexport function enableProxy() {\n // Redirect boss.hdslb.com to boss.bilibili.co\n session.defaultSession.webRequest.onBeforeRequest((details, callback) => {\n const url = details.url;\n const proxied = proxiedUrl(url);\n if (proxied === url) {\n return callback({ cancel: false });\n } else {\n logger.debug(TAG, `${url} -> ${proxied}`);\n callback({ cancel: false, redirectURL: proxied });\n }\n });\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/05.\n\nimport { setTimeout } from \"timers/promises\";\nimport { sleep } from \"./timing\";\n\nexport interface RetryOptions<Args extends any[], Ret> {\n fn: (...args: Args) => Promise<Ret>;\n maxAttempts?: number;\n timeout?: number;\n}\n\nexport function useRetry<Args extends any[], Ret>({\n fn,\n maxAttempts = 3,\n timeout,\n}: RetryOptions<Args, Ret>) {\n const timeoutError = new Error(`timeout over ${timeout}ms`);\n return async function (...args: Args) {\n let attempt = 0;\n while (true) {\n try {\n const promises = [fn(...args)];\n if (timeout) {\n promises.push(\n setTimeout(timeout).then(() => {\n throw timeoutError;\n }),\n );\n }\n return await Promise.race(promises);\n } catch (e) {\n attempt++;\n if (attempt >= maxAttempts) {\n throw e;\n }\n await sleep(Math.pow(2, attempt) * 100 + Math.random() * 100);\n }\n }\n };\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nexport function sleep(ms: number) {\n return new Promise<void>((resolve) => setTimeout(resolve, ms));\n}\n\nexport function periodical(\n callback: (count: number) => Promise<void> | void,\n ms: number,\n) {\n let token: NodeJS.Timeout;\n let closed = false;\n async function tick(count: number) {\n await callback(count);\n if (closed) return;\n token = setTimeout(() => tick(count + 1), ms);\n }\n token = setTimeout(() => tick(0), ms);\n return () => {\n closed = true;\n clearTimeout(token);\n };\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nimport { program } from \"commander\";\nimport { pupUseInnerProxy } from \"./base/constants\";\nimport { logger } from \"./base/logging\";\nimport { noerr } from \"./base/noerr\";\nimport { parseNumber } from \"./base/parser\";\nimport { pargs } from \"./base/process\";\nimport type { RecordOptions } from \"./base/record\";\n\nexport const DEFAULT_WIDTH = 1920;\nexport const DEFAULT_HEIGHT = 1080;\nexport const DEFAULT_FPS = 30;\nexport const DEFAULT_DURATION = 5;\nexport const DEFAULT_OUT_DIR = \"out\";\n\nexport type CLICallback = (\n source: string,\n options: RecordOptions & Record<string, unknown>,\n) => Promise<unknown>;\n\nexport function makeCLI(name: string, callback: CLICallback) {\n program\n .name(name)\n .argument(\"<source>\", \"URL \u6216 HTML data\")\n .option(\"-w, --width <number>\", \"\u89C6\u9891\u5BBD\u5EA6\", `${DEFAULT_WIDTH}`)\n .option(\"-h, --height <number>\", \"\u89C6\u9891\u9AD8\u5EA6\", `${DEFAULT_HEIGHT}`)\n .option(\"-f, --fps <number>\", \"\u5E27\u7387\", `${DEFAULT_FPS}`)\n .option(\"-t, --duration <number>\", \"\u5F55\u5236\u65F6\u957F\uFF08\u79D2\uFF09\", `${DEFAULT_DURATION}`)\n .option(\"-o, --out-dir <path>\", \"\u8F93\u51FA\u76EE\u5F55\", `${DEFAULT_OUT_DIR}`)\n .option(\"-a, --with-alpha-channel\", \"\u8F93\u51FA\u5305\u542B alpha \u901A\u9053\u7684\u89C6\u9891\", false)\n .option(\n \"--use-inner-proxy\",\n \"\u4F7F\u7528 B \u7AD9\u5185\u7F51\u4EE3\u7406\u52A0\u901F\u8D44\u6E90\u8BBF\u95EE\",\n pupUseInnerProxy,\n )\n .action(async (source: string, opts) => {\n try {\n await callback(source, {\n width: noerr(parseNumber, DEFAULT_WIDTH)(opts.width),\n height: noerr(parseNumber, DEFAULT_HEIGHT)(opts.height),\n fps: noerr(parseNumber, DEFAULT_FPS)(opts.fps),\n duration: noerr(parseNumber, DEFAULT_DURATION)(opts.duration),\n outDir: opts.outDir ?? DEFAULT_OUT_DIR,\n withAlphaChannel: opts.withAlphaChannel ?? false,\n useInnerProxy: opts.useInnerProxy ?? pupUseInnerProxy,\n });\n } catch (e) {\n logger.fatal(e);\n }\n });\n program.parse(pargs());\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/24.\n\nexport function noerr<Fn extends (...args: any[]) => any, D>(\n fn: Fn,\n defaultValue: D,\n): (...args: Parameters<Fn>) => ReturnType<Fn> | D {\n return (...args) => {\n try {\n const ret = fn(...args);\n if (ret instanceof Promise) {\n return ret.catch(() => defaultValue);\n }\n return ret;\n } catch {\n return defaultValue;\n }\n };\n}\n"],
5
- "mappings": "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",
3
+ "sources": ["../../src/app.ts", "../../src/base/electron.ts", "../../src/base/constants.ts", "../../src/base/env.ts", "../../src/base/parser.ts", "../../src/base/logging.ts", "../../src/base/process.ts", "../../src/base/record.ts", "../../src/rust/lib.ts", "../../src/base/frame_sync.ts", "../../src/base/html_check.ts", "../../src/base/image.ts", "../../src/base/proxy.ts", "../../src/base/retry.ts", "../../src/base/timing.ts", "../../src/common.ts", "../../src/base/noerr.ts"],
4
+ "sourcesContent": ["// Created by Autokaka (qq1909698494@gmail.com) on 2026/01/30.\n\nimport { app } from \"electron\";\nimport { ELECTRON_OPTS } from \"./base/electron\";\nimport { record, type RecordOptions } from \"./base/record\";\nimport { makeCLI } from \"./common\";\n\nprocess.once(\"exit\", () => app.quit());\n\nmakeCLI(\"app\", async (source: string, options: RecordOptions) => {\n try {\n ELECTRON_OPTS.forEach((o) => app.commandLine.appendSwitch(o));\n app.dock?.hide();\n await app.whenReady();\n await record(source, options);\n } finally {\n app.quit();\n }\n});\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/25.\n\nimport electron, { type Size } from \"electron\";\nimport { exec } from \"./process\";\n\nexport const ELECTRON_OPTS = [\n \"no-sandbox\",\n \"disable-setuid-sandbox\",\n \"disable-gpu\",\n \"disable-dev-shm-usage\",\n \"disable-software-rasterizer\",\n \"disable-web-security\",\n \"disable-site-isolation-trials\",\n \"disable-features=IsolateOrigins,site-per-process\",\n \"allow-insecure-localhost\",\n \"ignore-certificate-errors\",\n \"disable-blink-features=AutomationControlled\",\n \"mute-audio\",\n \"disable-extensions\",\n \"disable-background-networking\",\n \"address-family=ipv4\",\n \"disable-async-dns\",\n \"force-device-scale-factor=1\",\n \"trace-warnings\",\n \"force-color-profile=srgb\",\n \"disable-color-correct-rendering\",\n \"log-level=3\",\n];\n\nexport function runElectronApp(size: Size, app: unknown, args: unknown[]) {\n const electronArgs = ELECTRON_OPTS.map((a) => `--${a}`);\n const cmdParts = [];\n if (process.platform === \"linux\") {\n cmdParts.push(\n \"xvfb-run\",\n \"--auto-servernum\",\n `--server-args='-screen 0 ${size.width}x${size.height}x24'`,\n );\n }\n cmdParts.push(electron, ...electronArgs, app);\n return exec(cmdParts.join(\" \"), {\n stdio: [\"ignore\", \"pipe\", \"pipe\"],\n shell: true,\n env: {\n ...process.env,\n ELECTRON_DISABLE_DBUS: \"1\",\n RUST_BACKTRACE: \"full\",\n __PUP_ARGS__: JSON.stringify(args),\n },\n });\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nimport { existsSync } from \"fs\";\nimport { resolve } from \"path\";\nimport { penv } from \"./env\";\nimport { parseNumber } from \"./parser\";\n\nconst pupAppSearchPaths = [\n resolve(__dirname, \"cjs/app.cjs\"), // process from dist\n resolve(__dirname, \"app.cjs\"), // process from dist/cjs\n resolve(__dirname, \"../../cjs/app.cjs\"), // process from src\n];\nexport const pupAppPath = pupAppSearchPaths.find(existsSync)!;\n\nconst env = process.env;\nexport const pupLogLevel = penv(\"PUP_LOG_LEVEL\", parseNumber, 2);\nexport const pupUseInnerProxy = env[\"PUP_USE_INNER_PROXY\"] === \"1\";\nexport const pupFFmpegPath = env[\"FFMPEG_BIN\"] ?? `ffmpeg`;\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/25.\n\nexport type EnvParser<T> = (value: unknown) => T;\n\nexport function penv<T>(name: string, parser: EnvParser<T>, defaultValue: T): T;\nexport function penv<T>(\n name: string,\n parser: EnvParser<T>,\n defaultValue?: T,\n): T | undefined;\nexport function penv<T>(\n name: string,\n parser: EnvParser<T>,\n defaultValue?: T,\n): T | undefined {\n try {\n return parser(process.env[name]);\n } catch {\n return defaultValue;\n }\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/01/30.\n\nexport function parseNumber(value: unknown): number {\n if (typeof value === \"number\") {\n return value;\n }\n const num = Number(value);\n if (Number.isNaN(num)) {\n throw new Error(`Value ${value} is not a valid number`);\n }\n return num;\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/06.\n\nimport { ChildProcess, type Serializable } from \"child_process\";\nimport { pupLogLevel } from \"./constants\";\n\nexport interface LoggerLike {\n debug?(this: void, ...messages: unknown[]): void;\n\n info?(this: void, ...messages: unknown[]): void;\n\n warn?(this: void, ...messages: unknown[]): void;\n\n error?(this: void, ...messages: unknown[]): void;\n}\n\nconst DEBUG = \"<pup@debug>\";\nconst INFO = \"<pup@info>\";\nconst WARN = \"<pup@warn>\";\nconst ERROR = \"<pup@error>\";\nconst FATAL = \"<pup@fatal>\";\n\nclass Logger implements LoggerLike {\n private _impl?: LoggerLike;\n\n get impl(): LoggerLike | undefined {\n return this._impl;\n }\n\n set impl(value: LoggerLike) {\n const debug = value.debug ?? console.debug;\n const info = value.info ?? console.info;\n const warn = value.warn ?? console.warn;\n const error = value.error ?? console.error;\n this._impl = {\n debug: pupLogLevel >= 3 ? debug : undefined,\n info: pupLogLevel >= 2 ? info : undefined,\n warn: pupLogLevel >= 1 ? warn : undefined,\n error: pupLogLevel >= 0 ? error : undefined,\n };\n }\n\n constructor() {\n this.impl = console;\n }\n\n debug(...messages: unknown[]): void {\n this.impl?.debug?.(DEBUG, ...messages);\n }\n\n info(...messages: unknown[]): void {\n this.impl?.info?.(INFO, ...messages);\n }\n\n warn(...messages: unknown[]): void {\n this.impl?.warn?.(WARN, ...messages);\n }\n\n error(...messages: unknown[]): void {\n this.impl?.error?.(ERROR, ...messages);\n }\n\n fatal(...messages: unknown[]): never {\n this.impl?.error?.(FATAL, ...messages);\n process.exit(1);\n }\n\n private dispatch(message: string) {\n if (message.startsWith(DEBUG)) {\n this.debug(message.slice(DEBUG.length + 1));\n } else if (message.startsWith(INFO)) {\n this.info(message.slice(INFO.length + 1));\n } else if (message.startsWith(WARN)) {\n this.warn(message.slice(WARN.length + 1));\n } else if (message.startsWith(ERROR)) {\n this.error(message.slice(ERROR.length + 1));\n } else {\n this.info(message);\n }\n }\n\n attach(proc: ChildProcess, name: string) {\n return new Promise<void>((resolve, reject) => {\n this.debug(`${name}.attach`);\n let fatal: string = \"\";\n const dispatch = (data: Buffer | Serializable) => {\n const message = data.toString();\n if (message.startsWith(FATAL)) {\n fatal += message.slice(FATAL.length + 1);\n } else {\n this.dispatch(message);\n }\n };\n proc.stderr?.on(\"data\", dispatch);\n proc.stdout?.on(\"data\", dispatch);\n proc\n .on(\"message\", dispatch)\n .on(\"error\", (err) => {\n fatal += err.message;\n proc.kill();\n })\n .once(\"close\", (code, signal) => {\n if (code || signal || fatal) {\n fatal ||= `command failed: ${proc.spawnargs.join(\" \")}`;\n this.error(`${name}.close`, { code, signal, fatal });\n reject(new Error(fatal));\n } else {\n this.debug(`${name}.close`);\n resolve();\n }\n })\n .on(\"unhandledRejection\", (reason) => {\n this.error(`${name}.unhandled`, reason);\n })\n .on(\"uncaughtExceptionMonitor\", (err) => {\n this.error(`${name}.unhandled`, err);\n });\n });\n }\n}\n\nconst logger = new Logger();\n\nexport { logger };\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/01/30.\n\nimport { spawn, type ChildProcess, type SpawnOptions } from \"child_process\";\nimport { logger } from \"./logging\";\n\nexport const PUP_ARGS_ENV_KEY = \"__PUP_ARGS__\";\n\nexport function pargs() {\n const pupArgs = process.env[PUP_ARGS_ENV_KEY];\n if (pupArgs) {\n const args = [\"exec\", ...process.argv.slice(-1)];\n args.push(...JSON.parse(pupArgs));\n logger.debug(\"pupargs\", args);\n return args;\n }\n\n logger.debug(\"procargv\", process.argv);\n return process.argv;\n}\n\nexport interface ProcessHandle {\n process: ChildProcess;\n wait: Promise<void>;\n}\n\nexport function exec(cmd: string, options?: SpawnOptions): ProcessHandle {\n const parts = cmd.split(\" \").filter((s) => s.length);\n const [command, ...args] = parts;\n if (!command) throw new Error(\"empty command\");\n const proc = spawn(command, args, {\n stdio: \"inherit\",\n ...options,\n });\n return { process: proc, wait: logger.attach(proc, command) };\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nimport { BrowserWindow, session, type NativeImage } from \"electron\";\nimport { mkdir, writeFile } from \"fs/promises\";\nimport { join } from \"path\";\nimport { FixedBufferWriter } from \"../rust/lib\";\nimport {\n buildWrapperHTML,\n decodeTimestamp,\n startSync,\n stopSync,\n} from \"./frame_sync\";\nimport { checkHTML } from \"./html_check\";\nimport { isEmpty } from \"./image\";\nimport { logger } from \"./logging\";\nimport { enableProxy, proxiedUrl } from \"./proxy\";\nimport { useRetry } from \"./retry\";\n\nconst TAG = \"[Record]\";\n\nexport interface RecordOptions {\n outDir: string;\n duration: number;\n fps: number;\n width: number;\n height: number;\n withAlphaChannel: boolean;\n useInnerProxy: boolean;\n}\n\nexport interface RecordResult {\n options: RecordOptions;\n written: number;\n bgraPath: string;\n}\n\nasync function loadWindow(source: string, options: RecordOptions) {\n checkHTML(source);\n\n const { width, height, useInnerProxy } = options;\n\n // Remove X-Frame-Options and CSP headers to allow iframe loading\n // Note: Electron's onHeadersReceived uses REPLACE behavior (not append).\n // Only the last attached listener is used, so no cleanup needed.\n // Ref: https://www.electronjs.org/docs/latest/api/web-request\n session.defaultSession.webRequest.onHeadersReceived((details, callback) => {\n const responseHeaders = { ...details.responseHeaders };\n delete responseHeaders[\"x-frame-options\"];\n delete responseHeaders[\"X-Frame-Options\"];\n delete responseHeaders[\"content-security-policy\"];\n delete responseHeaders[\"Content-Security-Policy\"];\n callback({ cancel: false, responseHeaders });\n });\n\n let src = source;\n if (useInnerProxy) {\n src = proxiedUrl(source);\n enableProxy();\n }\n\n const win = new BrowserWindow({\n width: width,\n height: height + 1,\n show: false,\n transparent: true,\n backgroundColor: undefined,\n webPreferences: {\n offscreen: true,\n backgroundThrottling: false,\n nodeIntegration: true,\n contextIsolation: false,\n webSecurity: false,\n allowRunningInsecureContent: true,\n experimentalFeatures: true,\n },\n });\n\n win.webContents.on(\"console-message\", (event) => {\n if (event.level === \"error\") {\n logger.error(TAG, \"console:\", event.message);\n }\n });\n\n const wrapperHTML = buildWrapperHTML(src, { width, height });\n const dataURL = `data:text/html;charset=utf-8,${encodeURIComponent(wrapperHTML)}`;\n let token: NodeJS.Timeout | undefined;\n\n await new Promise<void>((resolve, reject) => {\n token = setTimeout(() => {\n reject(new Error(\"load window timeout\"));\n }, 20 * 1000);\n\n win.webContents.once(\"did-finish-load\", resolve);\n\n win.webContents.once(\"did-fail-load\", (_event, code, desc, url) => {\n reject(new Error(`failed to load ${url}: [${code}] ${desc}`));\n });\n\n win.webContents.once(\"render-process-gone\", (_event, details) => {\n const { exitCode, reason } = details;\n reject(new Error(`renderer crashed: ${exitCode}, ${reason}`));\n });\n\n win.loadURL(dataURL);\n });\n clearTimeout(token);\n return win;\n}\n\nexport async function record(\n source: string,\n options: RecordOptions,\n): Promise<void> {\n logger.info(TAG, `progress: 0%`);\n const { outDir, fps, width, height, duration } = options;\n\n const win = await useRetry({ fn: loadWindow, maxAttempts: 2 })(\n source,\n options,\n );\n\n await mkdir(outDir, { recursive: true });\n\n const cdp = win.webContents.debugger;\n cdp.attach(\"1.3\");\n\n win.webContents.setFrameRate(fps);\n if (!win.webContents.isPainting()) {\n win.webContents.startPainting();\n }\n\n const bgraPath = join(outDir, \"output.bgra\");\n const total = Math.ceil(fps * duration);\n const frameInterval = 1000 / fps;\n const bufferSize = width * height * 4;\n\n const writer = new FixedBufferWriter(bgraPath, bufferSize, fps);\n\n let written = 0;\n let lastWrittenTime: number | undefined;\n let progress = 0;\n let frameError: Error | undefined;\n let resolver: (() => void) | undefined;\n let rejecter: ((reason?: unknown) => void) | undefined;\n\n const scheduleWrite = (buffer: Buffer) => {\n written++;\n try {\n writer.write(buffer);\n } catch (error) {\n frameError ??= error as Error;\n }\n };\n\n const paint = (_e: unknown, _r: unknown, image: NativeImage) => {\n if (frameError) {\n rejecter?.(frameError);\n return;\n }\n\n if (written >= total) {\n resolver?.();\n return;\n }\n\n if (isEmpty(image)) return;\n\n const bitmap = image.toBitmap();\n const currentTime = decodeTimestamp(bitmap, image.getSize());\n if (currentTime === undefined) {\n frameError ??= new Error(`no timestamp @ ${written}`);\n return;\n }\n\n const bytesPerRow = width * 4;\n const cropped = bitmap.subarray(0, height * bytesPerRow);\n\n if (lastWrittenTime === undefined) {\n scheduleWrite(cropped);\n lastWrittenTime = currentTime;\n return;\n }\n\n const timeSinceLastFrame = currentTime - lastWrittenTime;\n if (timeSinceLastFrame < frameInterval * 0.8) {\n return;\n }\n\n if (timeSinceLastFrame <= frameInterval * 1.2) {\n scheduleWrite(cropped);\n } else {\n const framesToInsert = Math.round(timeSinceLastFrame / frameInterval);\n for (let i = 0; i < framesToInsert && written < total; i++) {\n scheduleWrite(cropped);\n }\n }\n lastWrittenTime = currentTime;\n\n const newProgress = Math.floor((written / total) * 100);\n if (Math.abs(newProgress - progress) > 10) {\n progress = newProgress;\n logger.info(TAG, `progress: ${Math.round(progress)}%`);\n }\n };\n\n win.webContents.on(\"paint\", paint);\n await startSync(cdp);\n try {\n await new Promise<void>((r, j) => ([resolver, rejecter] = [r, j]));\n } finally {\n await stopSync(cdp);\n win.webContents.off(\"paint\", paint);\n await writer.close();\n }\n\n if (frameError || written === 0) {\n throw frameError ?? new Error(\"no frames captured\");\n }\n\n try {\n const result: RecordResult = { options, written, bgraPath };\n await writeFile(join(outDir, \"record.json\"), JSON.stringify(result));\n logger.info(TAG, `progress: 100%, ${written} frames written`);\n } finally {\n win.close();\n }\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/10.\n\nimport { existsSync } from \"fs\";\nimport { join } from \"path\";\n\nconst { platform, arch } = process;\n\nconst rustPath = `rust/${platform}-${arch}.node`;\n\nconst nativeSearchPaths = [\n join(__dirname, `../../${rustPath}`), // process start from src\n join(__dirname, `../${rustPath}`), // process start from dist\n];\nconst mod = require(nativeSearchPaths.find(existsSync)!);\n\nexport interface FixedBufferWriter {\n new (\n path: string,\n bufferSize: number,\n queueDepth?: number,\n ): FixedBufferWriter;\n\n write(buffer: Buffer): void;\n\n close(): Promise<void>;\n}\n\nexport const FixedBufferWriter = mod.FixedBufferWriter as FixedBufferWriter;\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nimport type { Debugger, Size } from \"electron\";\n\nexport const FRAME_SYNC_MARKER_WIDTH = 32;\nexport const FRAME_SYNC_MARKER_HEIGHT = 1;\n\nexport function buildWrapperHTML(targetURL: string, size: Size): string {\n const { width, height } = size;\n return `<!DOCTYPE html>\n<html>\n<head>\n <meta charset=\"UTF-8\">\n <style>\n * { margin: 0; padding: 0; box-sizing: border-box; }\n html, body { width: ${width}px; height: ${height + 1}px; overflow: hidden; }\n #target { \n position: absolute; \n top: 0; \n left: 0; \n width: ${width}px; \n height: ${height}px; \n border: none; \n display: block;\n }\n #stego { \n position: absolute; \n top: ${height}px; \n left: 0; \n width: ${width}px; \n height: 1px; \n display: block;\n image-rendering: pixelated;\n }\n </style>\n</head>\n<body>\n <iframe id=\"target\" src=\"${targetURL}\"></iframe>\n <canvas id=\"stego\" width=\"${width}\" height=\"1\"></canvas>\n <script>\n (function() {\n const WIDTH = ${width};\n const MARKER_WIDTH = ${FRAME_SYNC_MARKER_WIDTH};\n const canvas = document.getElementById('stego');\n const ctx = canvas.getContext('2d', { willReadFrequently: true });\n let startTime = null;\n let rafId = null;\n\n function encodeTimestamp(timestampMs) {\n const imageData = ctx.createImageData(WIDTH, 1);\n const data = imageData.data;\n \n const timestampInt = Math.floor(timestampMs) >>> 0;\n \n for (let i = 0; i < MARKER_WIDTH; i++) {\n const bit = (timestampInt >>> (MARKER_WIDTH - 1 - i)) & 1;\n const value = bit ? 255 : 0;\n const idx = i * 4;\n data[idx] = value;\n data[idx + 1] = value;\n data[idx + 2] = value;\n data[idx + 3] = 255;\n }\n \n for (let i = MARKER_WIDTH; i < WIDTH; i++) {\n const idx = i * 4;\n data[idx] = 0;\n data[idx + 1] = 0;\n data[idx + 2] = 0;\n data[idx + 3] = 255;\n }\n \n ctx.putImageData(imageData, 0, 0);\n }\n\n function updateLoop() {\n if (startTime === null) return;\n const elapsed = performance.now() - startTime;\n encodeTimestamp(elapsed);\n rafId = requestAnimationFrame(updateLoop);\n }\n\n window.__pup_start_recording__ = () => {\n startTime = performance.now();\n encodeTimestamp(0);\n requestAnimationFrame(updateLoop);\n };\n\n window.__pup_stop_recording__ = () => {\n if (rafId !== null) {\n cancelAnimationFrame(rafId);\n rafId = null;\n }\n };\n })();\n </script>\n</body>\n</html>`;\n}\n\nexport function decodeTimestamp(\n bitmap: Buffer,\n size: Size,\n): number | undefined {\n const { width, height } = size;\n if (width < FRAME_SYNC_MARKER_WIDTH || height < 2) {\n return undefined;\n }\n\n const markerRow = height - 1;\n\n let timestamp = 0;\n for (let i = 0; i < FRAME_SYNC_MARKER_WIDTH; i++) {\n const pixelIdx = (markerRow * width + i) * 4;\n const r = bitmap[pixelIdx] ?? 0;\n const bit = r > 127 ? 1 : 0;\n timestamp = (timestamp << 1) | bit;\n }\n\n timestamp = timestamp >>> 0;\n\n if (!Number.isFinite(timestamp) || timestamp < 0 || timestamp > 1e7) {\n return undefined;\n }\n\n return timestamp;\n}\n\nexport function startSync(cdp: Debugger) {\n return cdp.sendCommand(\"Runtime.evaluate\", {\n expression: `window.__pup_start_recording__()`,\n });\n}\n\nexport function stopSync(cdp: Debugger) {\n return cdp.sendCommand(\"Runtime.evaluate\", {\n expression: `window.__pup_stop_recording__()`,\n });\n}\n", "/*\n * Created by Lu Ao (luao@bilibili.com) on 2026/02/25.\n * Copyright \u00A9 2026 bilibili. All rights reserved.\n */\n\nconst SUPPORTED_PROTOCOLS = [\"file:\", \"http:\", \"https:\", \"data:\"];\nconst SOURCE_PATTERN = /^(file:|https?:|data:)/;\n\nexport function checkHTML(source: string): void {\n if (SOURCE_PATTERN.test(source)) {\n return;\n }\n\n const protocol = source.split(\":\")[0] + \":\";\n const message = SUPPORTED_PROTOCOLS.includes(protocol)\n ? `unsupported protocol: ${protocol}, expected ${SUPPORTED_PROTOCOLS.join(\", \")}`\n : `invalid source format, expected ${SUPPORTED_PROTOCOLS.join(\", \")}`;\n\n throw new Error(message);\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/06.\n\nimport type { NativeImage } from \"electron\";\n\nexport function isEmpty(image: NativeImage) {\n const size = image.getSize();\n if (size.width === 0 || size.height === 0) return true;\n return image.isEmpty();\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nimport { session } from \"electron\";\nimport { logger } from \"./logging\";\n\nconst TAG = \"[Proxy]\";\n\nexport function proxiedUrl(url: string) {\n if (!url.startsWith(\"http\")) {\n return url;\n }\n // Redirect boss.hdslb.com to boss.bilibili.co\n const match = url.match(/^https:\\/\\/([^-]+)-boss\\.hdslb\\.com(.*)$/);\n if (match) {\n const [, prefix, path] = match;\n return `http://${prefix}-boss.bilibili.co${path}`;\n }\n return url;\n}\n\nexport function enableProxy() {\n // Redirect boss.hdslb.com to boss.bilibili.co\n session.defaultSession.webRequest.onBeforeRequest((details, callback) => {\n const url = details.url;\n const proxied = proxiedUrl(url);\n if (proxied === url) {\n return callback({ cancel: false });\n } else {\n logger.debug(TAG, `${url} -> ${proxied}`);\n callback({ cancel: false, redirectURL: proxied });\n }\n });\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/05.\n\nimport { setTimeout } from \"timers/promises\";\nimport { sleep } from \"./timing\";\n\nexport interface RetryOptions<Args extends any[], Ret> {\n fn: (...args: Args) => Promise<Ret>;\n maxAttempts?: number;\n timeout?: number;\n}\n\nexport function useRetry<Args extends any[], Ret>({\n fn,\n maxAttempts = 3,\n timeout,\n}: RetryOptions<Args, Ret>) {\n const timeoutError = new Error(`timeout over ${timeout}ms`);\n return async function (...args: Args) {\n let attempt = 0;\n while (true) {\n try {\n const promises = [fn(...args)];\n if (timeout) {\n promises.push(\n setTimeout(timeout).then(() => {\n throw timeoutError;\n }),\n );\n }\n return await Promise.race(promises);\n } catch (e) {\n attempt++;\n if (attempt >= maxAttempts) {\n throw e;\n }\n await sleep(Math.pow(2, attempt) * 100 + Math.random() * 100);\n }\n }\n };\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nexport function sleep(ms: number) {\n return new Promise<void>((resolve) => setTimeout(resolve, ms));\n}\n\nexport function periodical(\n callback: (count: number) => Promise<void> | void,\n ms: number,\n) {\n let token: NodeJS.Timeout;\n let closed = false;\n async function tick(count: number) {\n await callback(count);\n if (closed) return;\n token = setTimeout(() => tick(count + 1), ms);\n }\n token = setTimeout(() => tick(0), ms);\n return () => {\n closed = true;\n clearTimeout(token);\n };\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nimport { program } from \"commander\";\nimport { pupUseInnerProxy } from \"./base/constants\";\nimport { logger } from \"./base/logging\";\nimport { noerr } from \"./base/noerr\";\nimport { parseNumber } from \"./base/parser\";\nimport { pargs } from \"./base/process\";\nimport type { RecordOptions } from \"./base/record\";\n\nexport const DEFAULT_WIDTH = 1920;\nexport const DEFAULT_HEIGHT = 1080;\nexport const DEFAULT_FPS = 30;\nexport const DEFAULT_DURATION = 5;\nexport const DEFAULT_OUT_DIR = \"out\";\n\nexport type CLICallback = (\n source: string,\n options: RecordOptions & Record<string, unknown>,\n) => Promise<unknown>;\n\nexport function makeCLI(name: string, callback: CLICallback) {\n program\n .name(name)\n .argument(\"<source>\", \"file://, http(s)://, \u6216 data: URI\")\n .option(\"-w, --width <number>\", \"\u89C6\u9891\u5BBD\u5EA6\", `${DEFAULT_WIDTH}`)\n .option(\"-h, --height <number>\", \"\u89C6\u9891\u9AD8\u5EA6\", `${DEFAULT_HEIGHT}`)\n .option(\"-f, --fps <number>\", \"\u5E27\u7387\", `${DEFAULT_FPS}`)\n .option(\"-t, --duration <number>\", \"\u5F55\u5236\u65F6\u957F\uFF08\u79D2\uFF09\", `${DEFAULT_DURATION}`)\n .option(\"-o, --out-dir <path>\", \"\u8F93\u51FA\u76EE\u5F55\", `${DEFAULT_OUT_DIR}`)\n .option(\"-a, --with-alpha-channel\", \"\u8F93\u51FA\u5305\u542B alpha \u901A\u9053\u7684\u89C6\u9891\", false)\n .option(\n \"--use-inner-proxy\",\n \"\u4F7F\u7528 B \u7AD9\u5185\u7F51\u4EE3\u7406\u52A0\u901F\u8D44\u6E90\u8BBF\u95EE\",\n pupUseInnerProxy,\n )\n .action(async (source: string, opts) => {\n try {\n await callback(source, {\n width: noerr(parseNumber, DEFAULT_WIDTH)(opts.width),\n height: noerr(parseNumber, DEFAULT_HEIGHT)(opts.height),\n fps: noerr(parseNumber, DEFAULT_FPS)(opts.fps),\n duration: noerr(parseNumber, DEFAULT_DURATION)(opts.duration),\n outDir: opts.outDir ?? DEFAULT_OUT_DIR,\n withAlphaChannel: opts.withAlphaChannel ?? false,\n useInnerProxy: opts.useInnerProxy ?? pupUseInnerProxy,\n });\n } catch (e) {\n logger.fatal(e);\n }\n });\n program.parse(pargs());\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/24.\n\nexport function noerr<Fn extends (...args: any[]) => any, D>(\n fn: Fn,\n defaultValue: D,\n): (...args: Parameters<Fn>) => ReturnType<Fn> | D {\n return (...args) => {\n try {\n const ret = fn(...args);\n if (ret instanceof Promise) {\n return ret.catch(() => defaultValue);\n }\n return ret;\n } catch {\n return defaultValue;\n }\n };\n}\n"],
5
+ "mappings": ";;;;;;;;;;;;;;;;;;;;;;;;;AAEA,IAAAA,mBAAoB;;;ACApB,sBAAoC;;;ACApC,gBAA2B;AAC3B,kBAAwB;;;ACOjB,SAAS,KACd,MACA,QACA,cACe;AACf,MAAI;AACF,WAAO,OAAO,QAAQ,IAAI,IAAI,CAAC;AAAA,EACjC,QAAQ;AACN,WAAO;AAAA,EACT;AACF;;;AClBO,SAAS,YAAY,OAAwB;AAClD,MAAI,OAAO,UAAU,UAAU;AAC7B,WAAO;AAAA,EACT;AACA,QAAM,MAAM,OAAO,KAAK;AACxB,MAAI,OAAO,MAAM,GAAG,GAAG;AACrB,UAAM,IAAI,MAAM,SAAS,KAAK,wBAAwB;AAAA,EACxD;AACA,SAAO;AACT;;;AFJA,IAAM,oBAAoB;AAAA,MACxB,qBAAQ,WAAW,aAAa;AAAA;AAAA,MAChC,qBAAQ,WAAW,SAAS;AAAA;AAAA,MAC5B,qBAAQ,WAAW,mBAAmB;AAAA;AACxC;AACO,IAAM,aAAa,kBAAkB,KAAK,oBAAU;AAE3D,IAAM,MAAM,QAAQ;AACb,IAAM,cAAc,KAAK,iBAAiB,aAAa,CAAC;AACxD,IAAM,mBAAmB,IAAI,qBAAqB,MAAM;AACxD,IAAM,gBAAgB,IAAI,YAAY,KAAK;;;AGFlD,IAAM,QAAQ;AACd,IAAM,OAAO;AACb,IAAM,OAAO;AACb,IAAM,QAAQ;AACd,IAAM,QAAQ;AAEd,IAAM,SAAN,MAAmC;AAAA,EACzB;AAAA,EAER,IAAI,OAA+B;AACjC,WAAO,KAAK;AAAA,EACd;AAAA,EAEA,IAAI,KAAK,OAAmB;AAC1B,UAAM,QAAQ,MAAM,SAAS,QAAQ;AACrC,UAAM,OAAO,MAAM,QAAQ,QAAQ;AACnC,UAAM,OAAO,MAAM,QAAQ,QAAQ;AACnC,UAAM,QAAQ,MAAM,SAAS,QAAQ;AACrC,SAAK,QAAQ;AAAA,MACX,OAAO,eAAe,IAAI,QAAQ;AAAA,MAClC,MAAM,eAAe,IAAI,OAAO;AAAA,MAChC,MAAM,eAAe,IAAI,OAAO;AAAA,MAChC,OAAO,eAAe,IAAI,QAAQ;AAAA,IACpC;AAAA,EACF;AAAA,EAEA,cAAc;AACZ,SAAK,OAAO;AAAA,EACd;AAAA,EAEA,SAAS,UAA2B;AAClC,SAAK,MAAM,QAAQ,OAAO,GAAG,QAAQ;AAAA,EACvC;AAAA,EAEA,QAAQ,UAA2B;AACjC,SAAK,MAAM,OAAO,MAAM,GAAG,QAAQ;AAAA,EACrC;AAAA,EAEA,QAAQ,UAA2B;AACjC,SAAK,MAAM,OAAO,MAAM,GAAG,QAAQ;AAAA,EACrC;AAAA,EAEA,SAAS,UAA2B;AAClC,SAAK,MAAM,QAAQ,OAAO,GAAG,QAAQ;AAAA,EACvC;AAAA,EAEA,SAAS,UAA4B;AACnC,SAAK,MAAM,QAAQ,OAAO,GAAG,QAAQ;AACrC,YAAQ,KAAK,CAAC;AAAA,EAChB;AAAA,EAEQ,SAAS,SAAiB;AAChC,QAAI,QAAQ,WAAW,KAAK,GAAG;AAC7B,WAAK,MAAM,QAAQ,MAAM,MAAM,SAAS,CAAC,CAAC;AAAA,IAC5C,WAAW,QAAQ,WAAW,IAAI,GAAG;AACnC,WAAK,KAAK,QAAQ,MAAM,KAAK,SAAS,CAAC,CAAC;AAAA,IAC1C,WAAW,QAAQ,WAAW,IAAI,GAAG;AACnC,WAAK,KAAK,QAAQ,MAAM,KAAK,SAAS,CAAC,CAAC;AAAA,IAC1C,WAAW,QAAQ,WAAW,KAAK,GAAG;AACpC,WAAK,MAAM,QAAQ,MAAM,MAAM,SAAS,CAAC,CAAC;AAAA,IAC5C,OAAO;AACL,WAAK,KAAK,OAAO;AAAA,IACnB;AAAA,EACF;AAAA,EAEA,OAAO,MAAoB,MAAc;AACvC,WAAO,IAAI,QAAc,CAACC,UAAS,WAAW;AAC5C,WAAK,MAAM,GAAG,IAAI,SAAS;AAC3B,UAAI,QAAgB;AACpB,YAAM,WAAW,CAAC,SAAgC;AAChD,cAAM,UAAU,KAAK,SAAS;AAC9B,YAAI,QAAQ,WAAW,KAAK,GAAG;AAC7B,mBAAS,QAAQ,MAAM,MAAM,SAAS,CAAC;AAAA,QACzC,OAAO;AACL,eAAK,SAAS,OAAO;AAAA,QACvB;AAAA,MACF;AACA,WAAK,QAAQ,GAAG,QAAQ,QAAQ;AAChC,WAAK,QAAQ,GAAG,QAAQ,QAAQ;AAChC,WACG,GAAG,WAAW,QAAQ,EACtB,GAAG,SAAS,CAAC,QAAQ;AACpB,iBAAS,IAAI;AACb,aAAK,KAAK;AAAA,MACZ,CAAC,EACA,KAAK,SAAS,CAAC,MAAM,WAAW;AAC/B,YAAI,QAAQ,UAAU,OAAO;AAC3B,oBAAU,mBAAmB,KAAK,UAAU,KAAK,GAAG,CAAC;AACrD,eAAK,MAAM,GAAG,IAAI,UAAU,EAAE,MAAM,QAAQ,MAAM,CAAC;AACnD,iBAAO,IAAI,MAAM,KAAK,CAAC;AAAA,QACzB,OAAO;AACL,eAAK,MAAM,GAAG,IAAI,QAAQ;AAC1B,UAAAA,SAAQ;AAAA,QACV;AAAA,MACF,CAAC,EACA,GAAG,sBAAsB,CAAC,WAAW;AACpC,aAAK,MAAM,GAAG,IAAI,cAAc,MAAM;AAAA,MACxC,CAAC,EACA,GAAG,4BAA4B,CAAC,QAAQ;AACvC,aAAK,MAAM,GAAG,IAAI,cAAc,GAAG;AAAA,MACrC,CAAC;AAAA,IACL,CAAC;AAAA,EACH;AACF;AAEA,IAAM,SAAS,IAAI,OAAO;;;ACnHnB,IAAM,mBAAmB;AAEzB,SAAS,QAAQ;AACtB,QAAM,UAAU,QAAQ,IAAI,gBAAgB;AAC5C,MAAI,SAAS;AACX,UAAM,OAAO,CAAC,QAAQ,GAAG,QAAQ,KAAK,MAAM,EAAE,CAAC;AAC/C,SAAK,KAAK,GAAG,KAAK,MAAM,OAAO,CAAC;AAChC,WAAO,MAAM,WAAW,IAAI;AAC5B,WAAO;AAAA,EACT;AAEA,SAAO,MAAM,YAAY,QAAQ,IAAI;AACrC,SAAO,QAAQ;AACjB;;;ALbO,IAAM,gBAAgB;AAAA,EAC3B;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AACF;;;AMzBA,IAAAC,mBAAyD;AACzD,IAAAC,mBAAiC;AACjC,IAAAC,eAAqB;;;ACFrB,IAAAC,aAA2B;AAC3B,IAAAC,eAAqB;AAErB,IAAM,EAAE,UAAU,KAAK,IAAI;AAE3B,IAAM,WAAW,QAAQ,QAAQ,IAAI,IAAI;AAEzC,IAAM,oBAAoB;AAAA,MACxB,mBAAK,WAAW,SAAS,QAAQ,EAAE;AAAA;AAAA,MACnC,mBAAK,WAAW,MAAM,QAAQ,EAAE;AAAA;AAClC;AACA,IAAM,MAAM,QAAQ,kBAAkB,KAAK,qBAAU,CAAE;AAchD,IAAM,oBAAoB,IAAI;;;ACvB9B,IAAM,0BAA0B;AAGhC,SAAS,iBAAiB,WAAmB,MAAoB;AACtE,QAAM,EAAE,OAAO,OAAO,IAAI;AAC1B,SAAO;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,0BAMiB,KAAK,eAAe,SAAS,CAAC;AAAA;AAAA;AAAA;AAAA;AAAA,eAKzC,KAAK;AAAA,gBACJ,MAAM;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,aAMT,MAAM;AAAA;AAAA,eAEJ,KAAK;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,6BAQS,SAAS;AAAA,8BACR,KAAK;AAAA;AAAA;AAAA,sBAGb,KAAK;AAAA,6BACE,uBAAuB;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAwDpD;AAEO,SAAS,gBACd,QACA,MACoB;AACpB,QAAM,EAAE,OAAO,OAAO,IAAI;AAC1B,MAAI,QAAQ,2BAA2B,SAAS,GAAG;AACjD,WAAO;AAAA,EACT;AAEA,QAAM,YAAY,SAAS;AAE3B,MAAI,YAAY;AAChB,WAAS,IAAI,GAAG,IAAI,yBAAyB,KAAK;AAChD,UAAM,YAAY,YAAY,QAAQ,KAAK;AAC3C,UAAM,IAAI,OAAO,QAAQ,KAAK;AAC9B,UAAM,MAAM,IAAI,MAAM,IAAI;AAC1B,gBAAa,aAAa,IAAK;AAAA,EACjC;AAEA,cAAY,cAAc;AAE1B,MAAI,CAAC,OAAO,SAAS,SAAS,KAAK,YAAY,KAAK,YAAY,KAAK;AACnE,WAAO;AAAA,EACT;AAEA,SAAO;AACT;AAEO,SAAS,UAAU,KAAe;AACvC,SAAO,IAAI,YAAY,oBAAoB;AAAA,IACzC,YAAY;AAAA,EACd,CAAC;AACH;AAEO,SAAS,SAAS,KAAe;AACtC,SAAO,IAAI,YAAY,oBAAoB;AAAA,IACzC,YAAY;AAAA,EACd,CAAC;AACH;;;ACrIA,IAAM,sBAAsB,CAAC,SAAS,SAAS,UAAU,OAAO;AAChE,IAAM,iBAAiB;AAEhB,SAAS,UAAU,QAAsB;AAC9C,MAAI,eAAe,KAAK,MAAM,GAAG;AAC/B;AAAA,EACF;AAEA,QAAM,WAAW,OAAO,MAAM,GAAG,EAAE,CAAC,IAAI;AACxC,QAAM,UAAU,oBAAoB,SAAS,QAAQ,IACjD,yBAAyB,QAAQ,cAAc,oBAAoB,KAAK,IAAI,CAAC,KAC7E,mCAAmC,oBAAoB,KAAK,IAAI,CAAC;AAErE,QAAM,IAAI,MAAM,OAAO;AACzB;;;ACfO,SAAS,QAAQ,OAAoB;AAC1C,QAAM,OAAO,MAAM,QAAQ;AAC3B,MAAI,KAAK,UAAU,KAAK,KAAK,WAAW,EAAG,QAAO;AAClD,SAAO,MAAM,QAAQ;AACvB;;;ACNA,IAAAC,mBAAwB;AAGxB,IAAM,MAAM;AAEL,SAAS,WAAW,KAAa;AACtC,MAAI,CAAC,IAAI,WAAW,MAAM,GAAG;AAC3B,WAAO;AAAA,EACT;AAEA,QAAM,QAAQ,IAAI,MAAM,0CAA0C;AAClE,MAAI,OAAO;AACT,UAAM,CAAC,EAAE,QAAQ,IAAI,IAAI;AACzB,WAAO,UAAU,MAAM,oBAAoB,IAAI;AAAA,EACjD;AACA,SAAO;AACT;AAEO,SAAS,cAAc;AAE5B,2BAAQ,eAAe,WAAW,gBAAgB,CAAC,SAAS,aAAa;AACvE,UAAM,MAAM,QAAQ;AACpB,UAAM,UAAU,WAAW,GAAG;AAC9B,QAAI,YAAY,KAAK;AACnB,aAAO,SAAS,EAAE,QAAQ,MAAM,CAAC;AAAA,IACnC,OAAO;AACL,aAAO,MAAM,KAAK,GAAG,GAAG,OAAO,OAAO,EAAE;AACxC,eAAS,EAAE,QAAQ,OAAO,aAAa,QAAQ,CAAC;AAAA,IAClD;AAAA,EACF,CAAC;AACH;;;AC9BA,sBAA2B;;;ACApB,SAAS,MAAM,IAAY;AAChC,SAAO,IAAI,QAAc,CAACC,aAAY,WAAWA,UAAS,EAAE,CAAC;AAC/D;;;ADOO,SAAS,SAAkC;AAAA,EAChD;AAAA,EACA,cAAc;AAAA,EACd;AACF,GAA4B;AAC1B,QAAM,eAAe,IAAI,MAAM,gBAAgB,OAAO,IAAI;AAC1D,SAAO,kBAAmB,MAAY;AACpC,QAAI,UAAU;AACd,WAAO,MAAM;AACX,UAAI;AACF,cAAM,WAAW,CAAC,GAAG,GAAG,IAAI,CAAC;AAC7B,YAAI,SAAS;AACX,mBAAS;AAAA,gBACP,4BAAW,OAAO,EAAE,KAAK,MAAM;AAC7B,oBAAM;AAAA,YACR,CAAC;AAAA,UACH;AAAA,QACF;AACA,eAAO,MAAM,QAAQ,KAAK,QAAQ;AAAA,MACpC,SAAS,GAAG;AACV;AACA,YAAI,WAAW,aAAa;AAC1B,gBAAM;AAAA,QACR;AACA,cAAM,MAAM,KAAK,IAAI,GAAG,OAAO,IAAI,MAAM,KAAK,OAAO,IAAI,GAAG;AAAA,MAC9D;AAAA,IACF;AAAA,EACF;AACF;;;ANrBA,IAAMC,OAAM;AAkBZ,eAAe,WAAW,QAAgB,SAAwB;AAChE,YAAU,MAAM;AAEhB,QAAM,EAAE,OAAO,QAAQ,cAAc,IAAI;AAMzC,2BAAQ,eAAe,WAAW,kBAAkB,CAAC,SAAS,aAAa;AACzE,UAAM,kBAAkB,EAAE,GAAG,QAAQ,gBAAgB;AACrD,WAAO,gBAAgB,iBAAiB;AACxC,WAAO,gBAAgB,iBAAiB;AACxC,WAAO,gBAAgB,yBAAyB;AAChD,WAAO,gBAAgB,yBAAyB;AAChD,aAAS,EAAE,QAAQ,OAAO,gBAAgB,CAAC;AAAA,EAC7C,CAAC;AAED,MAAI,MAAM;AACV,MAAI,eAAe;AACjB,UAAM,WAAW,MAAM;AACvB,gBAAY;AAAA,EACd;AAEA,QAAM,MAAM,IAAI,+BAAc;AAAA,IAC5B;AAAA,IACA,QAAQ,SAAS;AAAA,IACjB,MAAM;AAAA,IACN,aAAa;AAAA,IACb,iBAAiB;AAAA,IACjB,gBAAgB;AAAA,MACd,WAAW;AAAA,MACX,sBAAsB;AAAA,MACtB,iBAAiB;AAAA,MACjB,kBAAkB;AAAA,MAClB,aAAa;AAAA,MACb,6BAA6B;AAAA,MAC7B,sBAAsB;AAAA,IACxB;AAAA,EACF,CAAC;AAED,MAAI,YAAY,GAAG,mBAAmB,CAAC,UAAU;AAC/C,QAAI,MAAM,UAAU,SAAS;AAC3B,aAAO,MAAMA,MAAK,YAAY,MAAM,OAAO;AAAA,IAC7C;AAAA,EACF,CAAC;AAED,QAAM,cAAc,iBAAiB,KAAK,EAAE,OAAO,OAAO,CAAC;AAC3D,QAAM,UAAU,gCAAgC,mBAAmB,WAAW,CAAC;AAC/E,MAAI;AAEJ,QAAM,IAAI,QAAc,CAACC,UAAS,WAAW;AAC3C,YAAQ,WAAW,MAAM;AACvB,aAAO,IAAI,MAAM,qBAAqB,CAAC;AAAA,IACzC,GAAG,KAAK,GAAI;AAEZ,QAAI,YAAY,KAAK,mBAAmBA,QAAO;AAE/C,QAAI,YAAY,KAAK,iBAAiB,CAAC,QAAQ,MAAM,MAAM,QAAQ;AACjE,aAAO,IAAI,MAAM,kBAAkB,GAAG,MAAM,IAAI,KAAK,IAAI,EAAE,CAAC;AAAA,IAC9D,CAAC;AAED,QAAI,YAAY,KAAK,uBAAuB,CAAC,QAAQ,YAAY;AAC/D,YAAM,EAAE,UAAU,OAAO,IAAI;AAC7B,aAAO,IAAI,MAAM,qBAAqB,QAAQ,KAAK,MAAM,EAAE,CAAC;AAAA,IAC9D,CAAC;AAED,QAAI,QAAQ,OAAO;AAAA,EACrB,CAAC;AACD,eAAa,KAAK;AAClB,SAAO;AACT;AAEA,eAAsB,OACpB,QACA,SACe;AACf,SAAO,KAAKD,MAAK,cAAc;AAC/B,QAAM,EAAE,QAAQ,KAAK,OAAO,QAAQ,SAAS,IAAI;AAEjD,QAAM,MAAM,MAAM,SAAS,EAAE,IAAI,YAAY,aAAa,EAAE,CAAC;AAAA,IAC3D;AAAA,IACA;AAAA,EACF;AAEA,YAAM,wBAAM,QAAQ,EAAE,WAAW,KAAK,CAAC;AAEvC,QAAM,MAAM,IAAI,YAAY;AAC5B,MAAI,OAAO,KAAK;AAEhB,MAAI,YAAY,aAAa,GAAG;AAChC,MAAI,CAAC,IAAI,YAAY,WAAW,GAAG;AACjC,QAAI,YAAY,cAAc;AAAA,EAChC;AAEA,QAAM,eAAW,mBAAK,QAAQ,aAAa;AAC3C,QAAM,QAAQ,KAAK,KAAK,MAAM,QAAQ;AACtC,QAAM,gBAAgB,MAAO;AAC7B,QAAM,aAAa,QAAQ,SAAS;AAEpC,QAAM,SAAS,IAAI,kBAAkB,UAAU,YAAY,GAAG;AAE9D,MAAI,UAAU;AACd,MAAI;AACJ,MAAI,WAAW;AACf,MAAI;AACJ,MAAI;AACJ,MAAI;AAEJ,QAAM,gBAAgB,CAAC,WAAmB;AACxC;AACA,QAAI;AACF,aAAO,MAAM,MAAM;AAAA,IACrB,SAAS,OAAO;AACd,qBAAe;AAAA,IACjB;AAAA,EACF;AAEA,QAAM,QAAQ,CAAC,IAAa,IAAa,UAAuB;AAC9D,QAAI,YAAY;AACd,iBAAW,UAAU;AACrB;AAAA,IACF;AAEA,QAAI,WAAW,OAAO;AACpB,iBAAW;AACX;AAAA,IACF;AAEA,QAAI,QAAQ,KAAK,EAAG;AAEpB,UAAM,SAAS,MAAM,SAAS;AAC9B,UAAM,cAAc,gBAAgB,QAAQ,MAAM,QAAQ,CAAC;AAC3D,QAAI,gBAAgB,QAAW;AAC7B,qBAAe,IAAI,MAAM,kBAAkB,OAAO,EAAE;AACpD;AAAA,IACF;AAEA,UAAM,cAAc,QAAQ;AAC5B,UAAM,UAAU,OAAO,SAAS,GAAG,SAAS,WAAW;AAEvD,QAAI,oBAAoB,QAAW;AACjC,oBAAc,OAAO;AACrB,wBAAkB;AAClB;AAAA,IACF;AAEA,UAAM,qBAAqB,cAAc;AACzC,QAAI,qBAAqB,gBAAgB,KAAK;AAC5C;AAAA,IACF;AAEA,QAAI,sBAAsB,gBAAgB,KAAK;AAC7C,oBAAc,OAAO;AAAA,IACvB,OAAO;AACL,YAAM,iBAAiB,KAAK,MAAM,qBAAqB,aAAa;AACpE,eAAS,IAAI,GAAG,IAAI,kBAAkB,UAAU,OAAO,KAAK;AAC1D,sBAAc,OAAO;AAAA,MACvB;AAAA,IACF;AACA,sBAAkB;AAElB,UAAM,cAAc,KAAK,MAAO,UAAU,QAAS,GAAG;AACtD,QAAI,KAAK,IAAI,cAAc,QAAQ,IAAI,IAAI;AACzC,iBAAW;AACX,aAAO,KAAKA,MAAK,aAAa,KAAK,MAAM,QAAQ,CAAC,GAAG;AAAA,IACvD;AAAA,EACF;AAEA,MAAI,YAAY,GAAG,SAAS,KAAK;AACjC,QAAM,UAAU,GAAG;AACnB,MAAI;AACF,UAAM,IAAI,QAAc,CAAC,GAAG,MAAO,CAAC,UAAU,QAAQ,IAAI,CAAC,GAAG,CAAC,CAAE;AAAA,EACnE,UAAE;AACA,UAAM,SAAS,GAAG;AAClB,QAAI,YAAY,IAAI,SAAS,KAAK;AAClC,UAAM,OAAO,MAAM;AAAA,EACrB;AAEA,MAAI,cAAc,YAAY,GAAG;AAC/B,UAAM,cAAc,IAAI,MAAM,oBAAoB;AAAA,EACpD;AAEA,MAAI;AACF,UAAM,SAAuB,EAAE,SAAS,SAAS,SAAS;AAC1D,cAAM,gCAAU,mBAAK,QAAQ,aAAa,GAAG,KAAK,UAAU,MAAM,CAAC;AACnE,WAAO,KAAKA,MAAK,mBAAmB,OAAO,iBAAiB;AAAA,EAC9D,UAAE;AACA,QAAI,MAAM;AAAA,EACZ;AACF;;;AQhOA,uBAAwB;;;ACAjB,SAAS,MACd,IACA,cACiD;AACjD,SAAO,IAAI,SAAS;AAClB,QAAI;AACF,YAAM,MAAM,GAAG,GAAG,IAAI;AACtB,UAAI,eAAe,SAAS;AAC1B,eAAO,IAAI,MAAM,MAAM,YAAY;AAAA,MACrC;AACA,aAAO;AAAA,IACT,QAAQ;AACN,aAAO;AAAA,IACT;AAAA,EACF;AACF;;;ADPO,IAAM,gBAAgB;AACtB,IAAM,iBAAiB;AACvB,IAAM,cAAc;AACpB,IAAM,mBAAmB;AACzB,IAAM,kBAAkB;AAOxB,SAAS,QAAQ,MAAc,UAAuB;AAC3D,2BACG,KAAK,IAAI,EACT,SAAS,YAAY,uCAAkC,EACvD,OAAO,wBAAwB,4BAAQ,GAAG,aAAa,EAAE,EACzD,OAAO,yBAAyB,4BAAQ,GAAG,cAAc,EAAE,EAC3D,OAAO,sBAAsB,gBAAM,GAAG,WAAW,EAAE,EACnD,OAAO,2BAA2B,8CAAW,GAAG,gBAAgB,EAAE,EAClE,OAAO,wBAAwB,4BAAQ,GAAG,eAAe,EAAE,EAC3D,OAAO,4BAA4B,iEAAoB,KAAK,EAC5D;AAAA,IACC;AAAA,IACA;AAAA,IACA;AAAA,EACF,EACC,OAAO,OAAO,QAAgB,SAAS;AACtC,QAAI;AACF,YAAM,SAAS,QAAQ;AAAA,QACrB,OAAO,MAAM,aAAa,aAAa,EAAE,KAAK,KAAK;AAAA,QACnD,QAAQ,MAAM,aAAa,cAAc,EAAE,KAAK,MAAM;AAAA,QACtD,KAAK,MAAM,aAAa,WAAW,EAAE,KAAK,GAAG;AAAA,QAC7C,UAAU,MAAM,aAAa,gBAAgB,EAAE,KAAK,QAAQ;AAAA,QAC5D,QAAQ,KAAK,UAAU;AAAA,QACvB,kBAAkB,KAAK,oBAAoB;AAAA,QAC3C,eAAe,KAAK,iBAAiB;AAAA,MACvC,CAAC;AAAA,IACH,SAAS,GAAG;AACV,aAAO,MAAM,CAAC;AAAA,IAChB;AAAA,EACF,CAAC;AACH,2BAAQ,MAAM,MAAM,CAAC;AACvB;;;Af7CA,QAAQ,KAAK,QAAQ,MAAM,qBAAI,KAAK,CAAC;AAErC,QAAQ,OAAO,OAAO,QAAgB,YAA2B;AAC/D,MAAI;AACF,kBAAc,QAAQ,CAAC,MAAM,qBAAI,YAAY,aAAa,CAAC,CAAC;AAC5D,yBAAI,MAAM,KAAK;AACf,UAAM,qBAAI,UAAU;AACpB,UAAM,OAAO,QAAQ,OAAO;AAAA,EAC9B,UAAE;AACA,yBAAI,KAAK;AAAA,EACX;AACF,CAAC;",
6
6
  "names": ["import_electron", "resolve", "import_electron", "import_promises", "import_path", "import_fs", "import_path", "import_electron", "resolve", "TAG", "resolve"]
7
7
  }
package/dist/cjs/cli.cjs CHANGED
@@ -203,7 +203,7 @@ var DEFAULT_FPS = 30;
203
203
  var DEFAULT_DURATION = 5;
204
204
  var DEFAULT_OUT_DIR = "out";
205
205
  function makeCLI(name, callback) {
206
- import_commander.program.name(name).argument("<source>", "URL \u6216 HTML data").option("-w, --width <number>", "\u89C6\u9891\u5BBD\u5EA6", `${DEFAULT_WIDTH}`).option("-h, --height <number>", "\u89C6\u9891\u9AD8\u5EA6", `${DEFAULT_HEIGHT}`).option("-f, --fps <number>", "\u5E27\u7387", `${DEFAULT_FPS}`).option("-t, --duration <number>", "\u5F55\u5236\u65F6\u957F\uFF08\u79D2\uFF09", `${DEFAULT_DURATION}`).option("-o, --out-dir <path>", "\u8F93\u51FA\u76EE\u5F55", `${DEFAULT_OUT_DIR}`).option("-a, --with-alpha-channel", "\u8F93\u51FA\u5305\u542B alpha \u901A\u9053\u7684\u89C6\u9891", false).option(
206
+ import_commander.program.name(name).argument("<source>", "file://, http(s)://, \u6216 data: URI").option("-w, --width <number>", "\u89C6\u9891\u5BBD\u5EA6", `${DEFAULT_WIDTH}`).option("-h, --height <number>", "\u89C6\u9891\u9AD8\u5EA6", `${DEFAULT_HEIGHT}`).option("-f, --fps <number>", "\u5E27\u7387", `${DEFAULT_FPS}`).option("-t, --duration <number>", "\u5F55\u5236\u65F6\u957F\uFF08\u79D2\uFF09", `${DEFAULT_DURATION}`).option("-o, --out-dir <path>", "\u8F93\u51FA\u76EE\u5F55", `${DEFAULT_OUT_DIR}`).option("-a, --with-alpha-channel", "\u8F93\u51FA\u5305\u542B alpha \u901A\u9053\u7684\u89C6\u9891", false).option(
207
207
  "--use-inner-proxy",
208
208
  "\u4F7F\u7528 B \u7AD9\u5185\u7F51\u4EE3\u7406\u52A0\u901F\u8D44\u6E90\u8BBF\u95EE",
209
209
  pupUseInnerProxy
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../src/common.ts", "../../src/base/constants.ts", "../../src/base/env.ts", "../../src/base/parser.ts", "../../src/base/logging.ts", "../../src/base/noerr.ts", "../../src/base/process.ts", "../../src/pup.ts", "../../src/base/abort.ts", "../../src/base/electron.ts", "../../src/base/ffmpeg.ts", "../../src/base/encoder.ts", "../../src/base/limiter.ts", "../../src/base/stream.ts", "../../src/cli.ts"],
4
- "sourcesContent": ["// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nimport { program } from \"commander\";\nimport { pupUseInnerProxy } from \"./base/constants\";\nimport { logger } from \"./base/logging\";\nimport { noerr } from \"./base/noerr\";\nimport { parseNumber } from \"./base/parser\";\nimport { pargs } from \"./base/process\";\nimport type { RecordOptions } from \"./base/record\";\n\nexport const DEFAULT_WIDTH = 1920;\nexport const DEFAULT_HEIGHT = 1080;\nexport const DEFAULT_FPS = 30;\nexport const DEFAULT_DURATION = 5;\nexport const DEFAULT_OUT_DIR = \"out\";\n\nexport type CLICallback = (\n source: string,\n options: RecordOptions & Record<string, unknown>,\n) => Promise<unknown>;\n\nexport function makeCLI(name: string, callback: CLICallback) {\n program\n .name(name)\n .argument(\"<source>\", \"URL \u6216 HTML data\")\n .option(\"-w, --width <number>\", \"\u89C6\u9891\u5BBD\u5EA6\", `${DEFAULT_WIDTH}`)\n .option(\"-h, --height <number>\", \"\u89C6\u9891\u9AD8\u5EA6\", `${DEFAULT_HEIGHT}`)\n .option(\"-f, --fps <number>\", \"\u5E27\u7387\", `${DEFAULT_FPS}`)\n .option(\"-t, --duration <number>\", \"\u5F55\u5236\u65F6\u957F\uFF08\u79D2\uFF09\", `${DEFAULT_DURATION}`)\n .option(\"-o, --out-dir <path>\", \"\u8F93\u51FA\u76EE\u5F55\", `${DEFAULT_OUT_DIR}`)\n .option(\"-a, --with-alpha-channel\", \"\u8F93\u51FA\u5305\u542B alpha \u901A\u9053\u7684\u89C6\u9891\", false)\n .option(\n \"--use-inner-proxy\",\n \"\u4F7F\u7528 B \u7AD9\u5185\u7F51\u4EE3\u7406\u52A0\u901F\u8D44\u6E90\u8BBF\u95EE\",\n pupUseInnerProxy,\n )\n .action(async (source: string, opts) => {\n try {\n await callback(source, {\n width: noerr(parseNumber, DEFAULT_WIDTH)(opts.width),\n height: noerr(parseNumber, DEFAULT_HEIGHT)(opts.height),\n fps: noerr(parseNumber, DEFAULT_FPS)(opts.fps),\n duration: noerr(parseNumber, DEFAULT_DURATION)(opts.duration),\n outDir: opts.outDir ?? DEFAULT_OUT_DIR,\n withAlphaChannel: opts.withAlphaChannel ?? false,\n useInnerProxy: opts.useInnerProxy ?? pupUseInnerProxy,\n });\n } catch (e) {\n logger.fatal(e);\n }\n });\n program.parse(pargs());\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nimport { existsSync } from \"fs\";\nimport { resolve } from \"path\";\nimport { penv } from \"./env\";\nimport { parseNumber } from \"./parser\";\n\nconst pupAppSearchPaths = [\n resolve(__dirname, \"cjs/app.cjs\"), // process from dist\n resolve(__dirname, \"app.cjs\"), // process from dist/cjs\n resolve(__dirname, \"../../cjs/app.cjs\"), // process from src\n];\nexport const pupAppPath = pupAppSearchPaths.find(existsSync)!;\n\nconst env = process.env;\nexport const pupLogLevel = penv(\"PUP_LOG_LEVEL\", parseNumber, 2);\nexport const pupUseInnerProxy = env[\"PUP_USE_INNER_PROXY\"] === \"1\";\nexport const pupFFmpegPath = env[\"FFMPEG_BIN\"] ?? `ffmpeg`;\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/25.\n\nexport type EnvParser<T> = (value: unknown) => T;\n\nexport function penv<T>(name: string, parser: EnvParser<T>, defaultValue: T): T;\nexport function penv<T>(\n name: string,\n parser: EnvParser<T>,\n defaultValue?: T,\n): T | undefined;\nexport function penv<T>(\n name: string,\n parser: EnvParser<T>,\n defaultValue?: T,\n): T | undefined {\n try {\n return parser(process.env[name]);\n } catch {\n return defaultValue;\n }\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/01/30.\n\nexport function parseNumber(value: unknown): number {\n if (typeof value === \"number\") {\n return value;\n }\n const num = Number(value);\n if (Number.isNaN(num)) {\n throw new Error(`Value ${value} is not a valid number`);\n }\n return num;\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/06.\n\nimport { ChildProcess, type Serializable } from \"child_process\";\nimport { pupLogLevel } from \"./constants\";\n\nexport interface LoggerLike {\n debug?(this: void, ...messages: unknown[]): void;\n\n info?(this: void, ...messages: unknown[]): void;\n\n warn?(this: void, ...messages: unknown[]): void;\n\n error?(this: void, ...messages: unknown[]): void;\n}\n\nconst DEBUG = \"<pup@debug>\";\nconst INFO = \"<pup@info>\";\nconst WARN = \"<pup@warn>\";\nconst ERROR = \"<pup@error>\";\nconst FATAL = \"<pup@fatal>\";\n\nclass Logger implements LoggerLike {\n private _impl?: LoggerLike;\n\n get impl(): LoggerLike | undefined {\n return this._impl;\n }\n\n set impl(value: LoggerLike) {\n const debug = value.debug ?? console.debug;\n const info = value.info ?? console.info;\n const warn = value.warn ?? console.warn;\n const error = value.error ?? console.error;\n this._impl = {\n debug: pupLogLevel >= 3 ? debug : undefined,\n info: pupLogLevel >= 2 ? info : undefined,\n warn: pupLogLevel >= 1 ? warn : undefined,\n error: pupLogLevel >= 0 ? error : undefined,\n };\n }\n\n constructor() {\n this.impl = console;\n }\n\n debug(...messages: unknown[]): void {\n this.impl?.debug?.(DEBUG, ...messages);\n }\n\n info(...messages: unknown[]): void {\n this.impl?.info?.(INFO, ...messages);\n }\n\n warn(...messages: unknown[]): void {\n this.impl?.warn?.(WARN, ...messages);\n }\n\n error(...messages: unknown[]): void {\n this.impl?.error?.(ERROR, ...messages);\n }\n\n fatal(...messages: unknown[]): never {\n this.impl?.error?.(FATAL, ...messages);\n process.exit(1);\n }\n\n private dispatch(message: string) {\n if (message.startsWith(DEBUG)) {\n this.debug(message.slice(DEBUG.length + 1));\n } else if (message.startsWith(INFO)) {\n this.info(message.slice(INFO.length + 1));\n } else if (message.startsWith(WARN)) {\n this.warn(message.slice(WARN.length + 1));\n } else if (message.startsWith(ERROR)) {\n this.error(message.slice(ERROR.length + 1));\n } else {\n this.info(message);\n }\n }\n\n attach(proc: ChildProcess, name: string) {\n return new Promise<void>((resolve, reject) => {\n this.debug(`${name}.attach`);\n let fatal: string = \"\";\n const dispatch = (data: Buffer | Serializable) => {\n const message = data.toString();\n if (message.startsWith(FATAL)) {\n fatal += message.slice(FATAL.length + 1);\n } else {\n this.dispatch(message);\n }\n };\n proc.stderr?.on(\"data\", dispatch);\n proc.stdout?.on(\"data\", dispatch);\n proc\n .on(\"message\", dispatch)\n .on(\"error\", (err) => {\n fatal += err.message;\n proc.kill();\n })\n .once(\"close\", (code, signal) => {\n if (code || signal || fatal) {\n fatal ||= `command failed: ${proc.spawnargs.join(\" \")}`;\n this.error(`${name}.close`, { code, signal, fatal });\n reject(new Error(fatal));\n } else {\n this.debug(`${name}.close`);\n resolve();\n }\n })\n .on(\"unhandledRejection\", (reason) => {\n this.error(`${name}.unhandled`, reason);\n })\n .on(\"uncaughtExceptionMonitor\", (err) => {\n this.error(`${name}.unhandled`, err);\n });\n });\n }\n}\n\nconst logger = new Logger();\n\nexport { logger };\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/24.\n\nexport function noerr<Fn extends (...args: any[]) => any, D>(\n fn: Fn,\n defaultValue: D,\n): (...args: Parameters<Fn>) => ReturnType<Fn> | D {\n return (...args) => {\n try {\n const ret = fn(...args);\n if (ret instanceof Promise) {\n return ret.catch(() => defaultValue);\n }\n return ret;\n } catch {\n return defaultValue;\n }\n };\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/01/30.\n\nimport { spawn, type ChildProcess, type SpawnOptions } from \"child_process\";\nimport { logger } from \"./logging\";\n\nexport const PUP_ARGS_ENV_KEY = \"__PUP_ARGS__\";\n\nexport function pargs() {\n const pupArgs = process.env[PUP_ARGS_ENV_KEY];\n if (pupArgs) {\n const args = [\"exec\", ...process.argv.slice(-1)];\n args.push(...JSON.parse(pupArgs));\n logger.debug(\"pupargs\", args);\n return args;\n }\n\n logger.debug(\"procargv\", process.argv);\n return process.argv;\n}\n\nexport interface ProcessHandle {\n process: ChildProcess;\n wait: Promise<void>;\n}\n\nexport function exec(cmd: string, options?: SpawnOptions): ProcessHandle {\n const parts = cmd.split(\" \").filter((s) => s.length);\n const [command, ...args] = parts;\n if (!command) throw new Error(\"empty command\");\n const proc = spawn(command, args, {\n stdio: \"inherit\",\n ...options,\n });\n return { process: proc, wait: logger.attach(proc, command) };\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nimport { spawn } from \"child_process\";\nimport type { Size } from \"electron\";\nimport { readFile, rm } from \"fs/promises\";\nimport { join } from \"path\";\nimport { AbortLink, type AbortQuery } from \"./base/abort\";\nimport { pupAppPath } from \"./base/constants\";\nimport { runElectronApp } from \"./base/electron\";\nimport { encodeBgraFile, encodeBgraToMov } from \"./base/encoder\";\nimport { createCoverCommand } from \"./base/ffmpeg\";\nimport { ConcurrencyLimiter } from \"./base/limiter\";\nimport { logger } from \"./base/logging\";\nimport { parseNumber } from \"./base/parser\";\nimport { type ProcessHandle } from \"./base/process\";\nimport type { RecordResult } from \"./base/record\";\nimport { waitAll } from \"./base/stream\";\nimport type { VideoFilesWithCover, VideoSpec } from \"./base/types\";\nimport { DEFAULT_HEIGHT, DEFAULT_WIDTH } from \"./common\";\n\nconst TAG = \"[pup]\";\n\nexport type PupProgressCallback = (progress: number) => Promise<void> | void;\n\nexport interface PupOptions {\n withAlphaChannel?: boolean;\n width?: number;\n height?: number;\n fps?: number;\n duration?: number;\n outDir?: string;\n cancelQuery?: AbortQuery;\n onProgress?: PupProgressCallback;\n}\n\nconst PROGRESS_TAG = \" progress: \";\n\nfunction runPupApp(source: string, options: PupOptions) {\n logger.debug(TAG, `runPupApp`, source, options);\n\n const args: string[] = [source];\n if (options.width) args.push(\"--width\", `${options.width}`);\n if (options.height) args.push(\"--height\", `${options.height}`);\n if (options.fps) args.push(\"--fps\", `${options.fps}`);\n if (options.duration) args.push(\"--duration\", `${options.duration}`);\n if (options.outDir) args.push(\"--out-dir\", options.outDir);\n if (options.withAlphaChannel) args.push(\"--with-alpha-channel\");\n\n const w = options.width ?? DEFAULT_WIDTH;\n const h = options.height ?? DEFAULT_HEIGHT;\n const handle = runElectronApp({ width: w, height: h }, pupAppPath, args);\n const counter = new ConcurrencyLimiter(1);\n handle.process.stdout?.on(\"data\", (data: Buffer) => {\n let message = data.toString().trim();\n let start = message.indexOf(PROGRESS_TAG);\n if (start < 0) {\n return;\n }\n message = message.slice(start + PROGRESS_TAG.length);\n const end = message.indexOf(\"%\");\n if (end < 0) {\n return;\n }\n const progressStr = message.slice(0, end);\n const progress = parseNumber(progressStr);\n counter.schedule(async () => {\n await options.onProgress?.(progress);\n });\n });\n return { handle, counter };\n}\n\nexport async function pup(source: string, options: PupOptions) {\n logger.debug(TAG, `pup`, source, options);\n\n const link = AbortLink.start(options.cancelQuery);\n const outDir = options.outDir ?? \"out\";\n\n const t0 = performance.now();\n const { handle, counter } = runPupApp(source, { ...options, outDir });\n\n await link.wait(handle);\n await counter.end();\n logger.info(TAG, `capture cost ${Math.round(performance.now() - t0)}ms`);\n\n const metaPath = join(outDir, \"record.json\");\n const meta = JSON.parse(await readFile(metaPath, \"utf-8\")) as RecordResult;\n\n const { bgraPath, written, options: recordOptions } = meta;\n const { fps, width, height, withAlphaChannel } = recordOptions;\n const size: Size = { width, height };\n\n const outputs: VideoFilesWithCover = {\n mp4: withAlphaChannel ? undefined : join(outDir, \"output.mp4\"),\n webm: withAlphaChannel ? join(outDir, \"output.webm\") : undefined,\n mov: withAlphaChannel ? join(outDir, \"output.mov\") : undefined,\n cover: join(outDir, \"cover.png\"),\n };\n\n try {\n const t1 = performance.now();\n\n const spec: VideoSpec = { fps, frames: written, size };\n const handles: ProcessHandle[] = [];\n if (outputs.mp4) {\n handles.push(encodeBgraFile(bgraPath, outputs.mp4, spec, \"mp4\"));\n }\n if (outputs.webm) {\n handles.push(encodeBgraFile(bgraPath, outputs.webm, spec, \"webm\"));\n }\n if (outputs.mov) {\n handles.push(encodeBgraToMov(bgraPath, outputs.mov, spec));\n }\n await link.wait(...handles);\n\n const coverSrc = outputs.mov ?? outputs.webm ?? outputs.mp4;\n if (coverSrc) {\n const coverCmd = createCoverCommand(coverSrc, outputs.cover);\n await waitAll(\n spawn(coverCmd.command, coverCmd.args, { stdio: \"inherit\" }),\n );\n }\n\n link.stop();\n logger.info(TAG, `encoding cost ${Math.round(performance.now() - t1)}ms`);\n\n await Promise.all([\n rm(bgraPath, { force: true }),\n rm(metaPath, { force: true }),\n ]);\n return {\n ...outputs,\n width,\n height,\n fps,\n duration: Math.ceil(written / fps),\n };\n } catch (error) {\n await rm(outDir, { recursive: true, force: true });\n throw error;\n }\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/11.\n\nimport type { ProcessHandle } from \"./process\";\n\nexport type AsyncTask = () => Promise<void> | void;\nexport type AbortQuery = () => Promise<boolean> | boolean;\n\nexport class AbortLink {\n private _callback?: AsyncTask;\n private _aborted?: boolean;\n private _stopped = false;\n\n private constructor(\n readonly query?: AbortQuery,\n readonly interval: number = 1000,\n ) {\n if (query) {\n this.tick();\n }\n }\n\n static start(query?: AbortQuery, interval?: number) {\n return new AbortLink(query, interval);\n }\n\n get aborted() {\n return !this._stopped && this._aborted;\n }\n\n get stopped() {\n return this._stopped;\n }\n\n async onAbort(callback: AsyncTask) {\n if (this._aborted) {\n await callback();\n } else {\n this._callback = callback;\n }\n }\n\n wait(...handles: ProcessHandle[]) {\n const abort = new Promise((_, reject) => {\n this.onAbort(async () => {\n handles.forEach((h) => h.process.kill());\n reject(new Error(\"aborted\"));\n });\n });\n return Promise.race([\n abort,\n Promise.all(handles.map((h) => h.wait)), //\n ]);\n }\n\n stop() {\n this._stopped = true;\n }\n\n private tick() {\n setTimeout(async () => {\n if (this._stopped) {\n return;\n }\n this._aborted = await this.query?.();\n if (this._stopped) {\n return;\n }\n if (this._aborted) {\n await this._callback?.();\n } else {\n this.tick();\n }\n }, this.interval);\n }\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/25.\n\nimport electron, { type Size } from \"electron\";\nimport { exec } from \"./process\";\n\nexport const ELECTRON_OPTS = [\n \"no-sandbox\",\n \"disable-setuid-sandbox\",\n \"disable-gpu\",\n \"disable-dev-shm-usage\",\n \"disable-software-rasterizer\",\n \"disable-web-security\",\n \"disable-site-isolation-trials\",\n \"disable-features=IsolateOrigins,site-per-process\",\n \"allow-insecure-localhost\",\n \"ignore-certificate-errors\",\n \"disable-blink-features=AutomationControlled\",\n \"mute-audio\",\n \"disable-extensions\",\n \"disable-background-networking\",\n \"address-family=ipv4\",\n \"disable-async-dns\",\n \"force-device-scale-factor=1\",\n \"trace-warnings\",\n \"force-color-profile=srgb\",\n \"disable-color-correct-rendering\",\n \"log-level=3\",\n];\n\nexport function runElectronApp(size: Size, app: unknown, args: unknown[]) {\n const electronArgs = ELECTRON_OPTS.map((a) => `--${a}`);\n const cmdParts = [];\n if (process.platform === \"linux\") {\n cmdParts.push(\n \"xvfb-run\",\n \"--auto-servernum\",\n `--server-args='-screen 0 ${size.width}x${size.height}x24'`,\n );\n }\n cmdParts.push(electron, ...electronArgs, app);\n return exec(cmdParts.join(\" \"), {\n stdio: [\"ignore\", \"pipe\", \"pipe\"],\n shell: true,\n env: {\n ...process.env,\n ELECTRON_DISABLE_DBUS: \"1\",\n RUST_BACKTRACE: \"full\",\n __PUP_ARGS__: JSON.stringify(args),\n },\n });\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nimport { existsSync } from \"fs\";\nimport { resolve } from \"path\";\nimport { arch, platform } from \"process\";\nimport { pupFFmpegPath } from \"./constants\";\nimport type { VideoFiles, VideoSpec } from \"./types\";\n\ninterface Command {\n command: string;\n args: string[];\n}\n\nconst quiet = [\"-hide_banner\", \"-loglevel\", \"error\", \"-nostats\"];\n\nfunction resolveX265() {\n const path = `x265/${platform}-${arch}`;\n const dirs = [\n resolve(__dirname, `../../${path}`), // process from src\n resolve(__dirname, `../${path}`), // process from dist\n ];\n const found = dirs.find(existsSync);\n if (!found) {\n throw new Error(\"x265 not found\");\n }\n return found;\n}\n\nexport function createBgraFileCommand(\n bgraPath: string,\n spec: VideoSpec,\n files: VideoFiles,\n): Command {\n const { fps, frames } = spec;\n const args = [\n \"-y\",\n ...quiet,\n \"-f\",\n \"rawvideo\",\n \"-pix_fmt\",\n \"bgra\",\n \"-s\",\n `${spec.size.width}x${spec.size.height}`,\n \"-r\",\n `${fps}`,\n \"-i\",\n bgraPath,\n \"-frames:v\",\n `${Math.floor(frames)}`,\n ];\n if (files.mp4) {\n args.push(\n \"-colorspace\",\n \"bt709\",\n \"-color_primaries\",\n \"bt709\",\n \"-color_trc\",\n \"bt709\",\n \"-c:v\",\n \"libx264\",\n \"-preset\",\n \"fast\",\n \"-threads\",\n \"0\",\n \"-pix_fmt\",\n \"yuv420p\",\n \"-movflags\",\n \"+faststart\",\n files.mp4,\n );\n }\n if (files.webm) {\n args.push(\n \"-c:v\",\n \"libvpx-vp9\",\n \"-row-mt\",\n \"1\",\n \"-cpu-used\",\n \"1\",\n \"-threads\",\n \"0\",\n \"-pix_fmt\",\n \"yuva420p\",\n \"-auto-alt-ref\",\n \"0\",\n files.webm,\n );\n }\n return { command: pupFFmpegPath, args };\n}\n\ninterface X265Pipeline {\n raw: Command;\n x265: Command;\n mux: Command;\n}\n\nexport function createBgraToMovPipeline(\n bgraPath: string,\n spec: VideoSpec,\n mov: string,\n): X265Pipeline {\n const { fps, size } = spec;\n return {\n raw: {\n command: pupFFmpegPath,\n args: [\n \"-y\",\n ...quiet,\n \"-f\",\n \"rawvideo\",\n \"-pix_fmt\",\n \"bgra\",\n \"-s\",\n `${size.width}x${size.height}`,\n \"-r\",\n `${fps}`,\n \"-i\",\n bgraPath,\n \"-f\",\n \"rawvideo\",\n \"-pix_fmt\",\n \"yuva420p10le\",\n \"pipe:1\",\n ],\n },\n x265: {\n command: resolveX265(),\n args: [\n \"--no-progress\",\n \"--log-level\",\n \"error\",\n \"--input\",\n \"-\",\n \"--input-res\",\n `${size.width}x${size.height}`,\n \"--input-csp\",\n \"i420\",\n \"--input-depth\",\n \"10\",\n \"--output-depth\",\n \"10\",\n \"--fps\",\n `${fps}`,\n \"--alpha\",\n \"--bframes\",\n \"0\",\n \"--colorprim\",\n \"bt709\",\n \"--transfer\",\n \"bt709\",\n \"--colormatrix\",\n \"bt709\",\n \"--crf\",\n \"18\",\n \"--output\",\n \"-\",\n ],\n },\n mux: {\n command: pupFFmpegPath,\n args: [\n \"-y\",\n ...quiet,\n \"-f\",\n \"hevc\",\n \"-r\",\n `${fps}`,\n \"-i\",\n \"pipe:0\",\n \"-c:v\",\n \"copy\",\n \"-tag:v\",\n \"hvc1\",\n \"-movflags\",\n \"+faststart\",\n mov,\n ],\n },\n };\n}\n\nexport function createCoverCommand(src: string, dst: string): Command {\n return {\n command: pupFFmpegPath,\n args: [\"-y\", ...quiet, \"-i\", src, \"-frames:v\", \"1\", \"-q:v\", \"2\", dst],\n };\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nimport { createBgraFileCommand, createBgraToMovPipeline } from \"./ffmpeg\";\nimport { exec, type ProcessHandle } from \"./process\";\nimport type { VideoSpec } from \"./types\";\n\nexport function encodeBgraFile(\n bgraPath: string,\n outputPath: string,\n spec: VideoSpec,\n format: \"mp4\" | \"webm\",\n): ProcessHandle {\n const files = format === \"mp4\" ? { mp4: outputPath } : { webm: outputPath };\n const cmd = createBgraFileCommand(bgraPath, spec, files);\n return exec(`${cmd.command} ${cmd.args.join(\" \")}`, {\n stdio: [\"ignore\", \"inherit\", \"inherit\"],\n });\n}\n\nexport function encodeBgraToMov(\n bgraPath: string,\n movPath: string,\n spec: VideoSpec,\n): ProcessHandle {\n const x265 = createBgraToMovPipeline(bgraPath, spec, movPath);\n const shellCmd = [\n `${x265.raw.command} ${x265.raw.args.join(\" \")}`,\n `${x265.x265.command} ${x265.x265.args.join(\" \")}`,\n `${x265.mux.command} ${x265.mux.args.join(\" \")}`,\n ].join(\" | \");\n return exec(shellCmd, {\n stdio: [\"ignore\", \"inherit\", \"inherit\"],\n shell: true,\n });\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/01/30.\n\nexport class ConcurrencyLimiter {\n private _active = 0;\n private _queue: VoidFunction[] = [];\n private _pending = 0;\n private _ended = false;\n\n constructor(readonly maxConcurrency: number) {}\n\n get active(): number {\n return this._active;\n }\n\n get pending(): number {\n return this._pending;\n }\n\n async schedule<T>(fn: () => Promise<T>): Promise<T> {\n if (this._ended) {\n throw new Error(\"ended\");\n }\n return new Promise<T>((resolve, reject) => {\n const run = () => {\n this._active++;\n this._pending--;\n fn()\n .then(resolve)\n .catch(reject)\n .finally(() => {\n this._active--;\n this.next();\n });\n };\n this._pending++;\n if (this._active < this.maxConcurrency) {\n run();\n } else {\n this._queue.push(run);\n }\n });\n }\n\n async end() {\n if (!this._ended) {\n this._ended = true;\n while (this._active > 0 || this._pending > 0) {\n await new Promise((resolve) => setTimeout(resolve, 50));\n }\n }\n }\n\n private next() {\n if (this._active < this.maxConcurrency && this._queue.length > 0) {\n this._queue.shift()?.();\n }\n }\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nimport type { ChildProcess } from \"child_process\";\n\nexport interface StreamWriter {\n write(buffer: Buffer): Promise<void>;\n end(): void;\n}\n\nexport function createWriter(stdin: NodeJS.WritableStream): StreamWriter {\n const drainPromise = new Map<NodeJS.WritableStream, Promise<void>>();\n\n const waitDrain = (stream: NodeJS.WritableStream) => {\n const existing = drainPromise.get(stream);\n if (existing) return existing;\n const promise = new Promise<void>((resolve, reject) => {\n const cleanup = (fn: () => void) => {\n stream.off(\"drain\", onDrain);\n stream.off(\"error\", onError);\n stream.off(\"close\", onClose);\n drainPromise.delete(stream);\n fn();\n };\n const onDrain = () => cleanup(resolve);\n const onError = (err: Error) => cleanup(() => reject(err));\n const onClose = () => cleanup(() => reject(new Error(\"stream closed\")));\n stream.on(\"drain\", onDrain);\n stream.on(\"error\", onError);\n stream.on(\"close\", onClose);\n });\n drainPromise.set(stream, promise);\n return promise;\n };\n\n const destroyed = () => Reflect.get(stdin, \"destroyed\");\n return {\n async write(buffer: Buffer) {\n if (destroyed()) {\n throw new Error(\"stdin destroyed\");\n }\n if (!stdin.write(buffer)) {\n await waitDrain(stdin);\n }\n },\n end: () => {\n if (!destroyed()) stdin.end();\n },\n };\n}\n\nexport function pipeline(...procs: ChildProcess[]) {\n for (let i = 0; i < procs.length - 1; i++) {\n const src = procs[i];\n const dst = procs[i + 1];\n if (!src?.stdout || !dst?.stdin) throw new Error(\"pipeline broken\");\n src.stdout.pipe(dst.stdin);\n }\n}\n\nexport function waitAll(...procs: ChildProcess[]) {\n return Promise.all(\n procs.map(\n (proc) =>\n new Promise<void>((resolve, reject) => {\n proc.on(\"error\", reject);\n proc.on(\"close\", (code) =>\n code === 0\n ? resolve()\n : reject(new Error(`exit ${code ?? \"null\"}`)),\n );\n }),\n ),\n );\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/01/30.\n\nimport { makeCLI } from \"./common\";\nimport { pup } from \"./pup\";\n\nmakeCLI(\"pup\", pup);\n"],
5
- "mappings": "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",
4
+ "sourcesContent": ["// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nimport { program } from \"commander\";\nimport { pupUseInnerProxy } from \"./base/constants\";\nimport { logger } from \"./base/logging\";\nimport { noerr } from \"./base/noerr\";\nimport { parseNumber } from \"./base/parser\";\nimport { pargs } from \"./base/process\";\nimport type { RecordOptions } from \"./base/record\";\n\nexport const DEFAULT_WIDTH = 1920;\nexport const DEFAULT_HEIGHT = 1080;\nexport const DEFAULT_FPS = 30;\nexport const DEFAULT_DURATION = 5;\nexport const DEFAULT_OUT_DIR = \"out\";\n\nexport type CLICallback = (\n source: string,\n options: RecordOptions & Record<string, unknown>,\n) => Promise<unknown>;\n\nexport function makeCLI(name: string, callback: CLICallback) {\n program\n .name(name)\n .argument(\"<source>\", \"file://, http(s)://, \u6216 data: URI\")\n .option(\"-w, --width <number>\", \"\u89C6\u9891\u5BBD\u5EA6\", `${DEFAULT_WIDTH}`)\n .option(\"-h, --height <number>\", \"\u89C6\u9891\u9AD8\u5EA6\", `${DEFAULT_HEIGHT}`)\n .option(\"-f, --fps <number>\", \"\u5E27\u7387\", `${DEFAULT_FPS}`)\n .option(\"-t, --duration <number>\", \"\u5F55\u5236\u65F6\u957F\uFF08\u79D2\uFF09\", `${DEFAULT_DURATION}`)\n .option(\"-o, --out-dir <path>\", \"\u8F93\u51FA\u76EE\u5F55\", `${DEFAULT_OUT_DIR}`)\n .option(\"-a, --with-alpha-channel\", \"\u8F93\u51FA\u5305\u542B alpha \u901A\u9053\u7684\u89C6\u9891\", false)\n .option(\n \"--use-inner-proxy\",\n \"\u4F7F\u7528 B \u7AD9\u5185\u7F51\u4EE3\u7406\u52A0\u901F\u8D44\u6E90\u8BBF\u95EE\",\n pupUseInnerProxy,\n )\n .action(async (source: string, opts) => {\n try {\n await callback(source, {\n width: noerr(parseNumber, DEFAULT_WIDTH)(opts.width),\n height: noerr(parseNumber, DEFAULT_HEIGHT)(opts.height),\n fps: noerr(parseNumber, DEFAULT_FPS)(opts.fps),\n duration: noerr(parseNumber, DEFAULT_DURATION)(opts.duration),\n outDir: opts.outDir ?? DEFAULT_OUT_DIR,\n withAlphaChannel: opts.withAlphaChannel ?? false,\n useInnerProxy: opts.useInnerProxy ?? pupUseInnerProxy,\n });\n } catch (e) {\n logger.fatal(e);\n }\n });\n program.parse(pargs());\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nimport { existsSync } from \"fs\";\nimport { resolve } from \"path\";\nimport { penv } from \"./env\";\nimport { parseNumber } from \"./parser\";\n\nconst pupAppSearchPaths = [\n resolve(__dirname, \"cjs/app.cjs\"), // process from dist\n resolve(__dirname, \"app.cjs\"), // process from dist/cjs\n resolve(__dirname, \"../../cjs/app.cjs\"), // process from src\n];\nexport const pupAppPath = pupAppSearchPaths.find(existsSync)!;\n\nconst env = process.env;\nexport const pupLogLevel = penv(\"PUP_LOG_LEVEL\", parseNumber, 2);\nexport const pupUseInnerProxy = env[\"PUP_USE_INNER_PROXY\"] === \"1\";\nexport const pupFFmpegPath = env[\"FFMPEG_BIN\"] ?? `ffmpeg`;\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/25.\n\nexport type EnvParser<T> = (value: unknown) => T;\n\nexport function penv<T>(name: string, parser: EnvParser<T>, defaultValue: T): T;\nexport function penv<T>(\n name: string,\n parser: EnvParser<T>,\n defaultValue?: T,\n): T | undefined;\nexport function penv<T>(\n name: string,\n parser: EnvParser<T>,\n defaultValue?: T,\n): T | undefined {\n try {\n return parser(process.env[name]);\n } catch {\n return defaultValue;\n }\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/01/30.\n\nexport function parseNumber(value: unknown): number {\n if (typeof value === \"number\") {\n return value;\n }\n const num = Number(value);\n if (Number.isNaN(num)) {\n throw new Error(`Value ${value} is not a valid number`);\n }\n return num;\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/06.\n\nimport { ChildProcess, type Serializable } from \"child_process\";\nimport { pupLogLevel } from \"./constants\";\n\nexport interface LoggerLike {\n debug?(this: void, ...messages: unknown[]): void;\n\n info?(this: void, ...messages: unknown[]): void;\n\n warn?(this: void, ...messages: unknown[]): void;\n\n error?(this: void, ...messages: unknown[]): void;\n}\n\nconst DEBUG = \"<pup@debug>\";\nconst INFO = \"<pup@info>\";\nconst WARN = \"<pup@warn>\";\nconst ERROR = \"<pup@error>\";\nconst FATAL = \"<pup@fatal>\";\n\nclass Logger implements LoggerLike {\n private _impl?: LoggerLike;\n\n get impl(): LoggerLike | undefined {\n return this._impl;\n }\n\n set impl(value: LoggerLike) {\n const debug = value.debug ?? console.debug;\n const info = value.info ?? console.info;\n const warn = value.warn ?? console.warn;\n const error = value.error ?? console.error;\n this._impl = {\n debug: pupLogLevel >= 3 ? debug : undefined,\n info: pupLogLevel >= 2 ? info : undefined,\n warn: pupLogLevel >= 1 ? warn : undefined,\n error: pupLogLevel >= 0 ? error : undefined,\n };\n }\n\n constructor() {\n this.impl = console;\n }\n\n debug(...messages: unknown[]): void {\n this.impl?.debug?.(DEBUG, ...messages);\n }\n\n info(...messages: unknown[]): void {\n this.impl?.info?.(INFO, ...messages);\n }\n\n warn(...messages: unknown[]): void {\n this.impl?.warn?.(WARN, ...messages);\n }\n\n error(...messages: unknown[]): void {\n this.impl?.error?.(ERROR, ...messages);\n }\n\n fatal(...messages: unknown[]): never {\n this.impl?.error?.(FATAL, ...messages);\n process.exit(1);\n }\n\n private dispatch(message: string) {\n if (message.startsWith(DEBUG)) {\n this.debug(message.slice(DEBUG.length + 1));\n } else if (message.startsWith(INFO)) {\n this.info(message.slice(INFO.length + 1));\n } else if (message.startsWith(WARN)) {\n this.warn(message.slice(WARN.length + 1));\n } else if (message.startsWith(ERROR)) {\n this.error(message.slice(ERROR.length + 1));\n } else {\n this.info(message);\n }\n }\n\n attach(proc: ChildProcess, name: string) {\n return new Promise<void>((resolve, reject) => {\n this.debug(`${name}.attach`);\n let fatal: string = \"\";\n const dispatch = (data: Buffer | Serializable) => {\n const message = data.toString();\n if (message.startsWith(FATAL)) {\n fatal += message.slice(FATAL.length + 1);\n } else {\n this.dispatch(message);\n }\n };\n proc.stderr?.on(\"data\", dispatch);\n proc.stdout?.on(\"data\", dispatch);\n proc\n .on(\"message\", dispatch)\n .on(\"error\", (err) => {\n fatal += err.message;\n proc.kill();\n })\n .once(\"close\", (code, signal) => {\n if (code || signal || fatal) {\n fatal ||= `command failed: ${proc.spawnargs.join(\" \")}`;\n this.error(`${name}.close`, { code, signal, fatal });\n reject(new Error(fatal));\n } else {\n this.debug(`${name}.close`);\n resolve();\n }\n })\n .on(\"unhandledRejection\", (reason) => {\n this.error(`${name}.unhandled`, reason);\n })\n .on(\"uncaughtExceptionMonitor\", (err) => {\n this.error(`${name}.unhandled`, err);\n });\n });\n }\n}\n\nconst logger = new Logger();\n\nexport { logger };\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/24.\n\nexport function noerr<Fn extends (...args: any[]) => any, D>(\n fn: Fn,\n defaultValue: D,\n): (...args: Parameters<Fn>) => ReturnType<Fn> | D {\n return (...args) => {\n try {\n const ret = fn(...args);\n if (ret instanceof Promise) {\n return ret.catch(() => defaultValue);\n }\n return ret;\n } catch {\n return defaultValue;\n }\n };\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/01/30.\n\nimport { spawn, type ChildProcess, type SpawnOptions } from \"child_process\";\nimport { logger } from \"./logging\";\n\nexport const PUP_ARGS_ENV_KEY = \"__PUP_ARGS__\";\n\nexport function pargs() {\n const pupArgs = process.env[PUP_ARGS_ENV_KEY];\n if (pupArgs) {\n const args = [\"exec\", ...process.argv.slice(-1)];\n args.push(...JSON.parse(pupArgs));\n logger.debug(\"pupargs\", args);\n return args;\n }\n\n logger.debug(\"procargv\", process.argv);\n return process.argv;\n}\n\nexport interface ProcessHandle {\n process: ChildProcess;\n wait: Promise<void>;\n}\n\nexport function exec(cmd: string, options?: SpawnOptions): ProcessHandle {\n const parts = cmd.split(\" \").filter((s) => s.length);\n const [command, ...args] = parts;\n if (!command) throw new Error(\"empty command\");\n const proc = spawn(command, args, {\n stdio: \"inherit\",\n ...options,\n });\n return { process: proc, wait: logger.attach(proc, command) };\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nimport { spawn } from \"child_process\";\nimport type { Size } from \"electron\";\nimport { readFile, rm } from \"fs/promises\";\nimport { join } from \"path\";\nimport { AbortLink, type AbortQuery } from \"./base/abort\";\nimport { pupAppPath } from \"./base/constants\";\nimport { runElectronApp } from \"./base/electron\";\nimport { encodeBgraFile, encodeBgraToMov } from \"./base/encoder\";\nimport { createCoverCommand } from \"./base/ffmpeg\";\nimport { ConcurrencyLimiter } from \"./base/limiter\";\nimport { logger } from \"./base/logging\";\nimport { parseNumber } from \"./base/parser\";\nimport { type ProcessHandle } from \"./base/process\";\nimport type { RecordResult } from \"./base/record\";\nimport { waitAll } from \"./base/stream\";\nimport type { VideoFilesWithCover, VideoSpec } from \"./base/types\";\nimport { DEFAULT_HEIGHT, DEFAULT_WIDTH } from \"./common\";\n\nconst TAG = \"[pup]\";\n\nexport type PupProgressCallback = (progress: number) => Promise<void> | void;\n\nexport interface PupOptions {\n withAlphaChannel?: boolean;\n width?: number;\n height?: number;\n fps?: number;\n duration?: number;\n outDir?: string;\n cancelQuery?: AbortQuery;\n onProgress?: PupProgressCallback;\n}\n\nconst PROGRESS_TAG = \" progress: \";\n\nfunction runPupApp(source: string, options: PupOptions) {\n logger.debug(TAG, `runPupApp`, source, options);\n\n const args: string[] = [source];\n if (options.width) args.push(\"--width\", `${options.width}`);\n if (options.height) args.push(\"--height\", `${options.height}`);\n if (options.fps) args.push(\"--fps\", `${options.fps}`);\n if (options.duration) args.push(\"--duration\", `${options.duration}`);\n if (options.outDir) args.push(\"--out-dir\", options.outDir);\n if (options.withAlphaChannel) args.push(\"--with-alpha-channel\");\n\n const w = options.width ?? DEFAULT_WIDTH;\n const h = options.height ?? DEFAULT_HEIGHT;\n const handle = runElectronApp({ width: w, height: h }, pupAppPath, args);\n const counter = new ConcurrencyLimiter(1);\n handle.process.stdout?.on(\"data\", (data: Buffer) => {\n let message = data.toString().trim();\n let start = message.indexOf(PROGRESS_TAG);\n if (start < 0) {\n return;\n }\n message = message.slice(start + PROGRESS_TAG.length);\n const end = message.indexOf(\"%\");\n if (end < 0) {\n return;\n }\n const progressStr = message.slice(0, end);\n const progress = parseNumber(progressStr);\n counter.schedule(async () => {\n await options.onProgress?.(progress);\n });\n });\n return { handle, counter };\n}\n\nexport async function pup(source: string, options: PupOptions) {\n logger.debug(TAG, `pup`, source, options);\n\n const link = AbortLink.start(options.cancelQuery);\n const outDir = options.outDir ?? \"out\";\n\n const t0 = performance.now();\n const { handle, counter } = runPupApp(source, { ...options, outDir });\n\n await link.wait(handle);\n await counter.end();\n logger.info(TAG, `capture cost ${Math.round(performance.now() - t0)}ms`);\n\n const metaPath = join(outDir, \"record.json\");\n const meta = JSON.parse(await readFile(metaPath, \"utf-8\")) as RecordResult;\n\n const { bgraPath, written, options: recordOptions } = meta;\n const { fps, width, height, withAlphaChannel } = recordOptions;\n const size: Size = { width, height };\n\n const outputs: VideoFilesWithCover = {\n mp4: withAlphaChannel ? undefined : join(outDir, \"output.mp4\"),\n webm: withAlphaChannel ? join(outDir, \"output.webm\") : undefined,\n mov: withAlphaChannel ? join(outDir, \"output.mov\") : undefined,\n cover: join(outDir, \"cover.png\"),\n };\n\n try {\n const t1 = performance.now();\n\n const spec: VideoSpec = { fps, frames: written, size };\n const handles: ProcessHandle[] = [];\n if (outputs.mp4) {\n handles.push(encodeBgraFile(bgraPath, outputs.mp4, spec, \"mp4\"));\n }\n if (outputs.webm) {\n handles.push(encodeBgraFile(bgraPath, outputs.webm, spec, \"webm\"));\n }\n if (outputs.mov) {\n handles.push(encodeBgraToMov(bgraPath, outputs.mov, spec));\n }\n await link.wait(...handles);\n\n const coverSrc = outputs.mov ?? outputs.webm ?? outputs.mp4;\n if (coverSrc) {\n const coverCmd = createCoverCommand(coverSrc, outputs.cover);\n await waitAll(\n spawn(coverCmd.command, coverCmd.args, { stdio: \"inherit\" }),\n );\n }\n\n link.stop();\n logger.info(TAG, `encoding cost ${Math.round(performance.now() - t1)}ms`);\n\n await Promise.all([\n rm(bgraPath, { force: true }),\n rm(metaPath, { force: true }),\n ]);\n return {\n ...outputs,\n width,\n height,\n fps,\n duration: Math.ceil(written / fps),\n };\n } catch (error) {\n await rm(outDir, { recursive: true, force: true });\n throw error;\n }\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/11.\n\nimport type { ProcessHandle } from \"./process\";\n\nexport type AsyncTask = () => Promise<void> | void;\nexport type AbortQuery = () => Promise<boolean> | boolean;\n\nexport class AbortLink {\n private _callback?: AsyncTask;\n private _aborted?: boolean;\n private _stopped = false;\n\n private constructor(\n readonly query?: AbortQuery,\n readonly interval: number = 1000,\n ) {\n if (query) {\n this.tick();\n }\n }\n\n static start(query?: AbortQuery, interval?: number) {\n return new AbortLink(query, interval);\n }\n\n get aborted() {\n return !this._stopped && this._aborted;\n }\n\n get stopped() {\n return this._stopped;\n }\n\n async onAbort(callback: AsyncTask) {\n if (this._aborted) {\n await callback();\n } else {\n this._callback = callback;\n }\n }\n\n wait(...handles: ProcessHandle[]) {\n const abort = new Promise((_, reject) => {\n this.onAbort(async () => {\n handles.forEach((h) => h.process.kill());\n reject(new Error(\"aborted\"));\n });\n });\n return Promise.race([\n abort,\n Promise.all(handles.map((h) => h.wait)), //\n ]);\n }\n\n stop() {\n this._stopped = true;\n }\n\n private tick() {\n setTimeout(async () => {\n if (this._stopped) {\n return;\n }\n this._aborted = await this.query?.();\n if (this._stopped) {\n return;\n }\n if (this._aborted) {\n await this._callback?.();\n } else {\n this.tick();\n }\n }, this.interval);\n }\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/25.\n\nimport electron, { type Size } from \"electron\";\nimport { exec } from \"./process\";\n\nexport const ELECTRON_OPTS = [\n \"no-sandbox\",\n \"disable-setuid-sandbox\",\n \"disable-gpu\",\n \"disable-dev-shm-usage\",\n \"disable-software-rasterizer\",\n \"disable-web-security\",\n \"disable-site-isolation-trials\",\n \"disable-features=IsolateOrigins,site-per-process\",\n \"allow-insecure-localhost\",\n \"ignore-certificate-errors\",\n \"disable-blink-features=AutomationControlled\",\n \"mute-audio\",\n \"disable-extensions\",\n \"disable-background-networking\",\n \"address-family=ipv4\",\n \"disable-async-dns\",\n \"force-device-scale-factor=1\",\n \"trace-warnings\",\n \"force-color-profile=srgb\",\n \"disable-color-correct-rendering\",\n \"log-level=3\",\n];\n\nexport function runElectronApp(size: Size, app: unknown, args: unknown[]) {\n const electronArgs = ELECTRON_OPTS.map((a) => `--${a}`);\n const cmdParts = [];\n if (process.platform === \"linux\") {\n cmdParts.push(\n \"xvfb-run\",\n \"--auto-servernum\",\n `--server-args='-screen 0 ${size.width}x${size.height}x24'`,\n );\n }\n cmdParts.push(electron, ...electronArgs, app);\n return exec(cmdParts.join(\" \"), {\n stdio: [\"ignore\", \"pipe\", \"pipe\"],\n shell: true,\n env: {\n ...process.env,\n ELECTRON_DISABLE_DBUS: \"1\",\n RUST_BACKTRACE: \"full\",\n __PUP_ARGS__: JSON.stringify(args),\n },\n });\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nimport { existsSync } from \"fs\";\nimport { resolve } from \"path\";\nimport { arch, platform } from \"process\";\nimport { pupFFmpegPath } from \"./constants\";\nimport type { VideoFiles, VideoSpec } from \"./types\";\n\ninterface Command {\n command: string;\n args: string[];\n}\n\nconst quiet = [\"-hide_banner\", \"-loglevel\", \"error\", \"-nostats\"];\n\nfunction resolveX265() {\n const path = `x265/${platform}-${arch}`;\n const dirs = [\n resolve(__dirname, `../../${path}`), // process from src\n resolve(__dirname, `../${path}`), // process from dist\n ];\n const found = dirs.find(existsSync);\n if (!found) {\n throw new Error(\"x265 not found\");\n }\n return found;\n}\n\nexport function createBgraFileCommand(\n bgraPath: string,\n spec: VideoSpec,\n files: VideoFiles,\n): Command {\n const { fps, frames } = spec;\n const args = [\n \"-y\",\n ...quiet,\n \"-f\",\n \"rawvideo\",\n \"-pix_fmt\",\n \"bgra\",\n \"-s\",\n `${spec.size.width}x${spec.size.height}`,\n \"-r\",\n `${fps}`,\n \"-i\",\n bgraPath,\n \"-frames:v\",\n `${Math.floor(frames)}`,\n ];\n if (files.mp4) {\n args.push(\n \"-colorspace\",\n \"bt709\",\n \"-color_primaries\",\n \"bt709\",\n \"-color_trc\",\n \"bt709\",\n \"-c:v\",\n \"libx264\",\n \"-preset\",\n \"fast\",\n \"-threads\",\n \"0\",\n \"-pix_fmt\",\n \"yuv420p\",\n \"-movflags\",\n \"+faststart\",\n files.mp4,\n );\n }\n if (files.webm) {\n args.push(\n \"-c:v\",\n \"libvpx-vp9\",\n \"-row-mt\",\n \"1\",\n \"-cpu-used\",\n \"1\",\n \"-threads\",\n \"0\",\n \"-pix_fmt\",\n \"yuva420p\",\n \"-auto-alt-ref\",\n \"0\",\n files.webm,\n );\n }\n return { command: pupFFmpegPath, args };\n}\n\ninterface X265Pipeline {\n raw: Command;\n x265: Command;\n mux: Command;\n}\n\nexport function createBgraToMovPipeline(\n bgraPath: string,\n spec: VideoSpec,\n mov: string,\n): X265Pipeline {\n const { fps, size } = spec;\n return {\n raw: {\n command: pupFFmpegPath,\n args: [\n \"-y\",\n ...quiet,\n \"-f\",\n \"rawvideo\",\n \"-pix_fmt\",\n \"bgra\",\n \"-s\",\n `${size.width}x${size.height}`,\n \"-r\",\n `${fps}`,\n \"-i\",\n bgraPath,\n \"-f\",\n \"rawvideo\",\n \"-pix_fmt\",\n \"yuva420p10le\",\n \"pipe:1\",\n ],\n },\n x265: {\n command: resolveX265(),\n args: [\n \"--no-progress\",\n \"--log-level\",\n \"error\",\n \"--input\",\n \"-\",\n \"--input-res\",\n `${size.width}x${size.height}`,\n \"--input-csp\",\n \"i420\",\n \"--input-depth\",\n \"10\",\n \"--output-depth\",\n \"10\",\n \"--fps\",\n `${fps}`,\n \"--alpha\",\n \"--bframes\",\n \"0\",\n \"--colorprim\",\n \"bt709\",\n \"--transfer\",\n \"bt709\",\n \"--colormatrix\",\n \"bt709\",\n \"--crf\",\n \"18\",\n \"--output\",\n \"-\",\n ],\n },\n mux: {\n command: pupFFmpegPath,\n args: [\n \"-y\",\n ...quiet,\n \"-f\",\n \"hevc\",\n \"-r\",\n `${fps}`,\n \"-i\",\n \"pipe:0\",\n \"-c:v\",\n \"copy\",\n \"-tag:v\",\n \"hvc1\",\n \"-movflags\",\n \"+faststart\",\n mov,\n ],\n },\n };\n}\n\nexport function createCoverCommand(src: string, dst: string): Command {\n return {\n command: pupFFmpegPath,\n args: [\"-y\", ...quiet, \"-i\", src, \"-frames:v\", \"1\", \"-q:v\", \"2\", dst],\n };\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nimport { createBgraFileCommand, createBgraToMovPipeline } from \"./ffmpeg\";\nimport { exec, type ProcessHandle } from \"./process\";\nimport type { VideoSpec } from \"./types\";\n\nexport function encodeBgraFile(\n bgraPath: string,\n outputPath: string,\n spec: VideoSpec,\n format: \"mp4\" | \"webm\",\n): ProcessHandle {\n const files = format === \"mp4\" ? { mp4: outputPath } : { webm: outputPath };\n const cmd = createBgraFileCommand(bgraPath, spec, files);\n return exec(`${cmd.command} ${cmd.args.join(\" \")}`, {\n stdio: [\"ignore\", \"inherit\", \"inherit\"],\n });\n}\n\nexport function encodeBgraToMov(\n bgraPath: string,\n movPath: string,\n spec: VideoSpec,\n): ProcessHandle {\n const x265 = createBgraToMovPipeline(bgraPath, spec, movPath);\n const shellCmd = [\n `${x265.raw.command} ${x265.raw.args.join(\" \")}`,\n `${x265.x265.command} ${x265.x265.args.join(\" \")}`,\n `${x265.mux.command} ${x265.mux.args.join(\" \")}`,\n ].join(\" | \");\n return exec(shellCmd, {\n stdio: [\"ignore\", \"inherit\", \"inherit\"],\n shell: true,\n });\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/01/30.\n\nexport class ConcurrencyLimiter {\n private _active = 0;\n private _queue: VoidFunction[] = [];\n private _pending = 0;\n private _ended = false;\n\n constructor(readonly maxConcurrency: number) {}\n\n get active(): number {\n return this._active;\n }\n\n get pending(): number {\n return this._pending;\n }\n\n async schedule<T>(fn: () => Promise<T>): Promise<T> {\n if (this._ended) {\n throw new Error(\"ended\");\n }\n return new Promise<T>((resolve, reject) => {\n const run = () => {\n this._active++;\n this._pending--;\n fn()\n .then(resolve)\n .catch(reject)\n .finally(() => {\n this._active--;\n this.next();\n });\n };\n this._pending++;\n if (this._active < this.maxConcurrency) {\n run();\n } else {\n this._queue.push(run);\n }\n });\n }\n\n async end() {\n if (!this._ended) {\n this._ended = true;\n while (this._active > 0 || this._pending > 0) {\n await new Promise((resolve) => setTimeout(resolve, 50));\n }\n }\n }\n\n private next() {\n if (this._active < this.maxConcurrency && this._queue.length > 0) {\n this._queue.shift()?.();\n }\n }\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nimport type { ChildProcess } from \"child_process\";\n\nexport interface StreamWriter {\n write(buffer: Buffer): Promise<void>;\n end(): void;\n}\n\nexport function createWriter(stdin: NodeJS.WritableStream): StreamWriter {\n const drainPromise = new Map<NodeJS.WritableStream, Promise<void>>();\n\n const waitDrain = (stream: NodeJS.WritableStream) => {\n const existing = drainPromise.get(stream);\n if (existing) return existing;\n const promise = new Promise<void>((resolve, reject) => {\n const cleanup = (fn: () => void) => {\n stream.off(\"drain\", onDrain);\n stream.off(\"error\", onError);\n stream.off(\"close\", onClose);\n drainPromise.delete(stream);\n fn();\n };\n const onDrain = () => cleanup(resolve);\n const onError = (err: Error) => cleanup(() => reject(err));\n const onClose = () => cleanup(() => reject(new Error(\"stream closed\")));\n stream.on(\"drain\", onDrain);\n stream.on(\"error\", onError);\n stream.on(\"close\", onClose);\n });\n drainPromise.set(stream, promise);\n return promise;\n };\n\n const destroyed = () => Reflect.get(stdin, \"destroyed\");\n return {\n async write(buffer: Buffer) {\n if (destroyed()) {\n throw new Error(\"stdin destroyed\");\n }\n if (!stdin.write(buffer)) {\n await waitDrain(stdin);\n }\n },\n end: () => {\n if (!destroyed()) stdin.end();\n },\n };\n}\n\nexport function pipeline(...procs: ChildProcess[]) {\n for (let i = 0; i < procs.length - 1; i++) {\n const src = procs[i];\n const dst = procs[i + 1];\n if (!src?.stdout || !dst?.stdin) throw new Error(\"pipeline broken\");\n src.stdout.pipe(dst.stdin);\n }\n}\n\nexport function waitAll(...procs: ChildProcess[]) {\n return Promise.all(\n procs.map(\n (proc) =>\n new Promise<void>((resolve, reject) => {\n proc.on(\"error\", reject);\n proc.on(\"close\", (code) =>\n code === 0\n ? resolve()\n : reject(new Error(`exit ${code ?? \"null\"}`)),\n );\n }),\n ),\n );\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/01/30.\n\nimport { makeCLI } from \"./common\";\nimport { pup } from \"./pup\";\n\nmakeCLI(\"pup\", pup);\n"],
5
+ "mappings": "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",
6
6
  "names": ["resolve", "import_child_process", "import_path", "electron", "import_fs", "import_path", "import_process", "resolve", "resolve"]
7
7
  }
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../../src/index.ts", "../../src/base/constants.ts", "../../src/base/env.ts", "../../src/base/parser.ts", "../../src/base/lazy.ts", "../../src/base/limiter.ts", "../../src/base/logging.ts", "../../src/base/noerr.ts", "../../src/base/process.ts", "../../src/base/retry.ts", "../../src/base/timing.ts", "../../src/pup.ts", "../../src/base/abort.ts", "../../src/base/electron.ts", "../../src/base/ffmpeg.ts", "../../src/base/encoder.ts", "../../src/base/stream.ts", "../../src/common.ts"],
4
- "sourcesContent": ["// Created by Autokaka (qq1909698494@gmail.com) on 2026/01/30.\n\nexport * from \"./base/constants\";\nexport * from \"./base/env\";\nexport * from \"./base/lazy\";\nexport * from \"./base/limiter\";\nexport * from \"./base/logging\";\nexport * from \"./base/noerr\";\nexport * from \"./base/parser\";\nexport * from \"./base/process\";\nexport * from \"./base/retry\";\nexport * from \"./base/timing\";\n\nexport { pup, type PupOptions } from \"./pup\";\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nimport { existsSync } from \"fs\";\nimport { resolve } from \"path\";\nimport { penv } from \"./env\";\nimport { parseNumber } from \"./parser\";\n\nconst pupAppSearchPaths = [\n resolve(__dirname, \"cjs/app.cjs\"), // process from dist\n resolve(__dirname, \"app.cjs\"), // process from dist/cjs\n resolve(__dirname, \"../../cjs/app.cjs\"), // process from src\n];\nexport const pupAppPath = pupAppSearchPaths.find(existsSync)!;\n\nconst env = process.env;\nexport const pupLogLevel = penv(\"PUP_LOG_LEVEL\", parseNumber, 2);\nexport const pupUseInnerProxy = env[\"PUP_USE_INNER_PROXY\"] === \"1\";\nexport const pupFFmpegPath = env[\"FFMPEG_BIN\"] ?? `ffmpeg`;\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/25.\n\nexport type EnvParser<T> = (value: unknown) => T;\n\nexport function penv<T>(name: string, parser: EnvParser<T>, defaultValue: T): T;\nexport function penv<T>(\n name: string,\n parser: EnvParser<T>,\n defaultValue?: T,\n): T | undefined;\nexport function penv<T>(\n name: string,\n parser: EnvParser<T>,\n defaultValue?: T,\n): T | undefined {\n try {\n return parser(process.env[name]);\n } catch {\n return defaultValue;\n }\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/01/30.\n\nexport function parseNumber(value: unknown): number {\n if (typeof value === \"number\") {\n return value;\n }\n const num = Number(value);\n if (Number.isNaN(num)) {\n throw new Error(`Value ${value} is not a valid number`);\n }\n return num;\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/01/30.\n\nexport class Lazy<T> {\n constructor(readonly makeValue: () => T) {}\n\n get value(): T {\n if (!this._initialized) {\n this._value = this.makeValue();\n this._initialized = true;\n }\n return this._value!;\n }\n\n get initialized(): boolean {\n return this._initialized;\n }\n\n private _initialized = false;\n private _value: T | undefined;\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/01/30.\n\nexport class ConcurrencyLimiter {\n private _active = 0;\n private _queue: VoidFunction[] = [];\n private _pending = 0;\n private _ended = false;\n\n constructor(readonly maxConcurrency: number) {}\n\n get active(): number {\n return this._active;\n }\n\n get pending(): number {\n return this._pending;\n }\n\n async schedule<T>(fn: () => Promise<T>): Promise<T> {\n if (this._ended) {\n throw new Error(\"ended\");\n }\n return new Promise<T>((resolve, reject) => {\n const run = () => {\n this._active++;\n this._pending--;\n fn()\n .then(resolve)\n .catch(reject)\n .finally(() => {\n this._active--;\n this.next();\n });\n };\n this._pending++;\n if (this._active < this.maxConcurrency) {\n run();\n } else {\n this._queue.push(run);\n }\n });\n }\n\n async end() {\n if (!this._ended) {\n this._ended = true;\n while (this._active > 0 || this._pending > 0) {\n await new Promise((resolve) => setTimeout(resolve, 50));\n }\n }\n }\n\n private next() {\n if (this._active < this.maxConcurrency && this._queue.length > 0) {\n this._queue.shift()?.();\n }\n }\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/06.\n\nimport { ChildProcess, type Serializable } from \"child_process\";\nimport { pupLogLevel } from \"./constants\";\n\nexport interface LoggerLike {\n debug?(this: void, ...messages: unknown[]): void;\n\n info?(this: void, ...messages: unknown[]): void;\n\n warn?(this: void, ...messages: unknown[]): void;\n\n error?(this: void, ...messages: unknown[]): void;\n}\n\nconst DEBUG = \"<pup@debug>\";\nconst INFO = \"<pup@info>\";\nconst WARN = \"<pup@warn>\";\nconst ERROR = \"<pup@error>\";\nconst FATAL = \"<pup@fatal>\";\n\nclass Logger implements LoggerLike {\n private _impl?: LoggerLike;\n\n get impl(): LoggerLike | undefined {\n return this._impl;\n }\n\n set impl(value: LoggerLike) {\n const debug = value.debug ?? console.debug;\n const info = value.info ?? console.info;\n const warn = value.warn ?? console.warn;\n const error = value.error ?? console.error;\n this._impl = {\n debug: pupLogLevel >= 3 ? debug : undefined,\n info: pupLogLevel >= 2 ? info : undefined,\n warn: pupLogLevel >= 1 ? warn : undefined,\n error: pupLogLevel >= 0 ? error : undefined,\n };\n }\n\n constructor() {\n this.impl = console;\n }\n\n debug(...messages: unknown[]): void {\n this.impl?.debug?.(DEBUG, ...messages);\n }\n\n info(...messages: unknown[]): void {\n this.impl?.info?.(INFO, ...messages);\n }\n\n warn(...messages: unknown[]): void {\n this.impl?.warn?.(WARN, ...messages);\n }\n\n error(...messages: unknown[]): void {\n this.impl?.error?.(ERROR, ...messages);\n }\n\n fatal(...messages: unknown[]): never {\n this.impl?.error?.(FATAL, ...messages);\n process.exit(1);\n }\n\n private dispatch(message: string) {\n if (message.startsWith(DEBUG)) {\n this.debug(message.slice(DEBUG.length + 1));\n } else if (message.startsWith(INFO)) {\n this.info(message.slice(INFO.length + 1));\n } else if (message.startsWith(WARN)) {\n this.warn(message.slice(WARN.length + 1));\n } else if (message.startsWith(ERROR)) {\n this.error(message.slice(ERROR.length + 1));\n } else {\n this.info(message);\n }\n }\n\n attach(proc: ChildProcess, name: string) {\n return new Promise<void>((resolve, reject) => {\n this.debug(`${name}.attach`);\n let fatal: string = \"\";\n const dispatch = (data: Buffer | Serializable) => {\n const message = data.toString();\n if (message.startsWith(FATAL)) {\n fatal += message.slice(FATAL.length + 1);\n } else {\n this.dispatch(message);\n }\n };\n proc.stderr?.on(\"data\", dispatch);\n proc.stdout?.on(\"data\", dispatch);\n proc\n .on(\"message\", dispatch)\n .on(\"error\", (err) => {\n fatal += err.message;\n proc.kill();\n })\n .once(\"close\", (code, signal) => {\n if (code || signal || fatal) {\n fatal ||= `command failed: ${proc.spawnargs.join(\" \")}`;\n this.error(`${name}.close`, { code, signal, fatal });\n reject(new Error(fatal));\n } else {\n this.debug(`${name}.close`);\n resolve();\n }\n })\n .on(\"unhandledRejection\", (reason) => {\n this.error(`${name}.unhandled`, reason);\n })\n .on(\"uncaughtExceptionMonitor\", (err) => {\n this.error(`${name}.unhandled`, err);\n });\n });\n }\n}\n\nconst logger = new Logger();\n\nexport { logger };\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/24.\n\nexport function noerr<Fn extends (...args: any[]) => any, D>(\n fn: Fn,\n defaultValue: D,\n): (...args: Parameters<Fn>) => ReturnType<Fn> | D {\n return (...args) => {\n try {\n const ret = fn(...args);\n if (ret instanceof Promise) {\n return ret.catch(() => defaultValue);\n }\n return ret;\n } catch {\n return defaultValue;\n }\n };\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/01/30.\n\nimport { spawn, type ChildProcess, type SpawnOptions } from \"child_process\";\nimport { logger } from \"./logging\";\n\nexport const PUP_ARGS_ENV_KEY = \"__PUP_ARGS__\";\n\nexport function pargs() {\n const pupArgs = process.env[PUP_ARGS_ENV_KEY];\n if (pupArgs) {\n const args = [\"exec\", ...process.argv.slice(-1)];\n args.push(...JSON.parse(pupArgs));\n logger.debug(\"pupargs\", args);\n return args;\n }\n\n logger.debug(\"procargv\", process.argv);\n return process.argv;\n}\n\nexport interface ProcessHandle {\n process: ChildProcess;\n wait: Promise<void>;\n}\n\nexport function exec(cmd: string, options?: SpawnOptions): ProcessHandle {\n const parts = cmd.split(\" \").filter((s) => s.length);\n const [command, ...args] = parts;\n if (!command) throw new Error(\"empty command\");\n const proc = spawn(command, args, {\n stdio: \"inherit\",\n ...options,\n });\n return { process: proc, wait: logger.attach(proc, command) };\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/05.\n\nimport { setTimeout } from \"timers/promises\";\nimport { sleep } from \"./timing\";\n\nexport interface RetryOptions<Args extends any[], Ret> {\n fn: (...args: Args) => Promise<Ret>;\n maxAttempts?: number;\n timeout?: number;\n}\n\nexport function useRetry<Args extends any[], Ret>({\n fn,\n maxAttempts = 3,\n timeout,\n}: RetryOptions<Args, Ret>) {\n const timeoutError = new Error(`timeout over ${timeout}ms`);\n return async function (...args: Args) {\n let attempt = 0;\n while (true) {\n try {\n const promises = [fn(...args)];\n if (timeout) {\n promises.push(\n setTimeout(timeout).then(() => {\n throw timeoutError;\n }),\n );\n }\n return await Promise.race(promises);\n } catch (e) {\n attempt++;\n if (attempt >= maxAttempts) {\n throw e;\n }\n await sleep(Math.pow(2, attempt) * 100 + Math.random() * 100);\n }\n }\n };\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nexport function sleep(ms: number) {\n return new Promise<void>((resolve) => setTimeout(resolve, ms));\n}\n\nexport function periodical(\n callback: (count: number) => Promise<void> | void,\n ms: number,\n) {\n let token: NodeJS.Timeout;\n let closed = false;\n async function tick(count: number) {\n await callback(count);\n if (closed) return;\n token = setTimeout(() => tick(count + 1), ms);\n }\n token = setTimeout(() => tick(0), ms);\n return () => {\n closed = true;\n clearTimeout(token);\n };\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nimport { spawn } from \"child_process\";\nimport type { Size } from \"electron\";\nimport { readFile, rm } from \"fs/promises\";\nimport { join } from \"path\";\nimport { AbortLink, type AbortQuery } from \"./base/abort\";\nimport { pupAppPath } from \"./base/constants\";\nimport { runElectronApp } from \"./base/electron\";\nimport { encodeBgraFile, encodeBgraToMov } from \"./base/encoder\";\nimport { createCoverCommand } from \"./base/ffmpeg\";\nimport { ConcurrencyLimiter } from \"./base/limiter\";\nimport { logger } from \"./base/logging\";\nimport { parseNumber } from \"./base/parser\";\nimport { type ProcessHandle } from \"./base/process\";\nimport type { RecordResult } from \"./base/record\";\nimport { waitAll } from \"./base/stream\";\nimport type { VideoFilesWithCover, VideoSpec } from \"./base/types\";\nimport { DEFAULT_HEIGHT, DEFAULT_WIDTH } from \"./common\";\n\nconst TAG = \"[pup]\";\n\nexport type PupProgressCallback = (progress: number) => Promise<void> | void;\n\nexport interface PupOptions {\n withAlphaChannel?: boolean;\n width?: number;\n height?: number;\n fps?: number;\n duration?: number;\n outDir?: string;\n cancelQuery?: AbortQuery;\n onProgress?: PupProgressCallback;\n}\n\nconst PROGRESS_TAG = \" progress: \";\n\nfunction runPupApp(source: string, options: PupOptions) {\n logger.debug(TAG, `runPupApp`, source, options);\n\n const args: string[] = [source];\n if (options.width) args.push(\"--width\", `${options.width}`);\n if (options.height) args.push(\"--height\", `${options.height}`);\n if (options.fps) args.push(\"--fps\", `${options.fps}`);\n if (options.duration) args.push(\"--duration\", `${options.duration}`);\n if (options.outDir) args.push(\"--out-dir\", options.outDir);\n if (options.withAlphaChannel) args.push(\"--with-alpha-channel\");\n\n const w = options.width ?? DEFAULT_WIDTH;\n const h = options.height ?? DEFAULT_HEIGHT;\n const handle = runElectronApp({ width: w, height: h }, pupAppPath, args);\n const counter = new ConcurrencyLimiter(1);\n handle.process.stdout?.on(\"data\", (data: Buffer) => {\n let message = data.toString().trim();\n let start = message.indexOf(PROGRESS_TAG);\n if (start < 0) {\n return;\n }\n message = message.slice(start + PROGRESS_TAG.length);\n const end = message.indexOf(\"%\");\n if (end < 0) {\n return;\n }\n const progressStr = message.slice(0, end);\n const progress = parseNumber(progressStr);\n counter.schedule(async () => {\n await options.onProgress?.(progress);\n });\n });\n return { handle, counter };\n}\n\nexport async function pup(source: string, options: PupOptions) {\n logger.debug(TAG, `pup`, source, options);\n\n const link = AbortLink.start(options.cancelQuery);\n const outDir = options.outDir ?? \"out\";\n\n const t0 = performance.now();\n const { handle, counter } = runPupApp(source, { ...options, outDir });\n\n await link.wait(handle);\n await counter.end();\n logger.info(TAG, `capture cost ${Math.round(performance.now() - t0)}ms`);\n\n const metaPath = join(outDir, \"record.json\");\n const meta = JSON.parse(await readFile(metaPath, \"utf-8\")) as RecordResult;\n\n const { bgraPath, written, options: recordOptions } = meta;\n const { fps, width, height, withAlphaChannel } = recordOptions;\n const size: Size = { width, height };\n\n const outputs: VideoFilesWithCover = {\n mp4: withAlphaChannel ? undefined : join(outDir, \"output.mp4\"),\n webm: withAlphaChannel ? join(outDir, \"output.webm\") : undefined,\n mov: withAlphaChannel ? join(outDir, \"output.mov\") : undefined,\n cover: join(outDir, \"cover.png\"),\n };\n\n try {\n const t1 = performance.now();\n\n const spec: VideoSpec = { fps, frames: written, size };\n const handles: ProcessHandle[] = [];\n if (outputs.mp4) {\n handles.push(encodeBgraFile(bgraPath, outputs.mp4, spec, \"mp4\"));\n }\n if (outputs.webm) {\n handles.push(encodeBgraFile(bgraPath, outputs.webm, spec, \"webm\"));\n }\n if (outputs.mov) {\n handles.push(encodeBgraToMov(bgraPath, outputs.mov, spec));\n }\n await link.wait(...handles);\n\n const coverSrc = outputs.mov ?? outputs.webm ?? outputs.mp4;\n if (coverSrc) {\n const coverCmd = createCoverCommand(coverSrc, outputs.cover);\n await waitAll(\n spawn(coverCmd.command, coverCmd.args, { stdio: \"inherit\" }),\n );\n }\n\n link.stop();\n logger.info(TAG, `encoding cost ${Math.round(performance.now() - t1)}ms`);\n\n await Promise.all([\n rm(bgraPath, { force: true }),\n rm(metaPath, { force: true }),\n ]);\n return {\n ...outputs,\n width,\n height,\n fps,\n duration: Math.ceil(written / fps),\n };\n } catch (error) {\n await rm(outDir, { recursive: true, force: true });\n throw error;\n }\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/11.\n\nimport type { ProcessHandle } from \"./process\";\n\nexport type AsyncTask = () => Promise<void> | void;\nexport type AbortQuery = () => Promise<boolean> | boolean;\n\nexport class AbortLink {\n private _callback?: AsyncTask;\n private _aborted?: boolean;\n private _stopped = false;\n\n private constructor(\n readonly query?: AbortQuery,\n readonly interval: number = 1000,\n ) {\n if (query) {\n this.tick();\n }\n }\n\n static start(query?: AbortQuery, interval?: number) {\n return new AbortLink(query, interval);\n }\n\n get aborted() {\n return !this._stopped && this._aborted;\n }\n\n get stopped() {\n return this._stopped;\n }\n\n async onAbort(callback: AsyncTask) {\n if (this._aborted) {\n await callback();\n } else {\n this._callback = callback;\n }\n }\n\n wait(...handles: ProcessHandle[]) {\n const abort = new Promise((_, reject) => {\n this.onAbort(async () => {\n handles.forEach((h) => h.process.kill());\n reject(new Error(\"aborted\"));\n });\n });\n return Promise.race([\n abort,\n Promise.all(handles.map((h) => h.wait)), //\n ]);\n }\n\n stop() {\n this._stopped = true;\n }\n\n private tick() {\n setTimeout(async () => {\n if (this._stopped) {\n return;\n }\n this._aborted = await this.query?.();\n if (this._stopped) {\n return;\n }\n if (this._aborted) {\n await this._callback?.();\n } else {\n this.tick();\n }\n }, this.interval);\n }\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/25.\n\nimport electron, { type Size } from \"electron\";\nimport { exec } from \"./process\";\n\nexport const ELECTRON_OPTS = [\n \"no-sandbox\",\n \"disable-setuid-sandbox\",\n \"disable-gpu\",\n \"disable-dev-shm-usage\",\n \"disable-software-rasterizer\",\n \"disable-web-security\",\n \"disable-site-isolation-trials\",\n \"disable-features=IsolateOrigins,site-per-process\",\n \"allow-insecure-localhost\",\n \"ignore-certificate-errors\",\n \"disable-blink-features=AutomationControlled\",\n \"mute-audio\",\n \"disable-extensions\",\n \"disable-background-networking\",\n \"address-family=ipv4\",\n \"disable-async-dns\",\n \"force-device-scale-factor=1\",\n \"trace-warnings\",\n \"force-color-profile=srgb\",\n \"disable-color-correct-rendering\",\n \"log-level=3\",\n];\n\nexport function runElectronApp(size: Size, app: unknown, args: unknown[]) {\n const electronArgs = ELECTRON_OPTS.map((a) => `--${a}`);\n const cmdParts = [];\n if (process.platform === \"linux\") {\n cmdParts.push(\n \"xvfb-run\",\n \"--auto-servernum\",\n `--server-args='-screen 0 ${size.width}x${size.height}x24'`,\n );\n }\n cmdParts.push(electron, ...electronArgs, app);\n return exec(cmdParts.join(\" \"), {\n stdio: [\"ignore\", \"pipe\", \"pipe\"],\n shell: true,\n env: {\n ...process.env,\n ELECTRON_DISABLE_DBUS: \"1\",\n RUST_BACKTRACE: \"full\",\n __PUP_ARGS__: JSON.stringify(args),\n },\n });\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nimport { existsSync } from \"fs\";\nimport { resolve } from \"path\";\nimport { arch, platform } from \"process\";\nimport { pupFFmpegPath } from \"./constants\";\nimport type { VideoFiles, VideoSpec } from \"./types\";\n\ninterface Command {\n command: string;\n args: string[];\n}\n\nconst quiet = [\"-hide_banner\", \"-loglevel\", \"error\", \"-nostats\"];\n\nfunction resolveX265() {\n const path = `x265/${platform}-${arch}`;\n const dirs = [\n resolve(__dirname, `../../${path}`), // process from src\n resolve(__dirname, `../${path}`), // process from dist\n ];\n const found = dirs.find(existsSync);\n if (!found) {\n throw new Error(\"x265 not found\");\n }\n return found;\n}\n\nexport function createBgraFileCommand(\n bgraPath: string,\n spec: VideoSpec,\n files: VideoFiles,\n): Command {\n const { fps, frames } = spec;\n const args = [\n \"-y\",\n ...quiet,\n \"-f\",\n \"rawvideo\",\n \"-pix_fmt\",\n \"bgra\",\n \"-s\",\n `${spec.size.width}x${spec.size.height}`,\n \"-r\",\n `${fps}`,\n \"-i\",\n bgraPath,\n \"-frames:v\",\n `${Math.floor(frames)}`,\n ];\n if (files.mp4) {\n args.push(\n \"-colorspace\",\n \"bt709\",\n \"-color_primaries\",\n \"bt709\",\n \"-color_trc\",\n \"bt709\",\n \"-c:v\",\n \"libx264\",\n \"-preset\",\n \"fast\",\n \"-threads\",\n \"0\",\n \"-pix_fmt\",\n \"yuv420p\",\n \"-movflags\",\n \"+faststart\",\n files.mp4,\n );\n }\n if (files.webm) {\n args.push(\n \"-c:v\",\n \"libvpx-vp9\",\n \"-row-mt\",\n \"1\",\n \"-cpu-used\",\n \"1\",\n \"-threads\",\n \"0\",\n \"-pix_fmt\",\n \"yuva420p\",\n \"-auto-alt-ref\",\n \"0\",\n files.webm,\n );\n }\n return { command: pupFFmpegPath, args };\n}\n\ninterface X265Pipeline {\n raw: Command;\n x265: Command;\n mux: Command;\n}\n\nexport function createBgraToMovPipeline(\n bgraPath: string,\n spec: VideoSpec,\n mov: string,\n): X265Pipeline {\n const { fps, size } = spec;\n return {\n raw: {\n command: pupFFmpegPath,\n args: [\n \"-y\",\n ...quiet,\n \"-f\",\n \"rawvideo\",\n \"-pix_fmt\",\n \"bgra\",\n \"-s\",\n `${size.width}x${size.height}`,\n \"-r\",\n `${fps}`,\n \"-i\",\n bgraPath,\n \"-f\",\n \"rawvideo\",\n \"-pix_fmt\",\n \"yuva420p10le\",\n \"pipe:1\",\n ],\n },\n x265: {\n command: resolveX265(),\n args: [\n \"--no-progress\",\n \"--log-level\",\n \"error\",\n \"--input\",\n \"-\",\n \"--input-res\",\n `${size.width}x${size.height}`,\n \"--input-csp\",\n \"i420\",\n \"--input-depth\",\n \"10\",\n \"--output-depth\",\n \"10\",\n \"--fps\",\n `${fps}`,\n \"--alpha\",\n \"--bframes\",\n \"0\",\n \"--colorprim\",\n \"bt709\",\n \"--transfer\",\n \"bt709\",\n \"--colormatrix\",\n \"bt709\",\n \"--crf\",\n \"18\",\n \"--output\",\n \"-\",\n ],\n },\n mux: {\n command: pupFFmpegPath,\n args: [\n \"-y\",\n ...quiet,\n \"-f\",\n \"hevc\",\n \"-r\",\n `${fps}`,\n \"-i\",\n \"pipe:0\",\n \"-c:v\",\n \"copy\",\n \"-tag:v\",\n \"hvc1\",\n \"-movflags\",\n \"+faststart\",\n mov,\n ],\n },\n };\n}\n\nexport function createCoverCommand(src: string, dst: string): Command {\n return {\n command: pupFFmpegPath,\n args: [\"-y\", ...quiet, \"-i\", src, \"-frames:v\", \"1\", \"-q:v\", \"2\", dst],\n };\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nimport { createBgraFileCommand, createBgraToMovPipeline } from \"./ffmpeg\";\nimport { exec, type ProcessHandle } from \"./process\";\nimport type { VideoSpec } from \"./types\";\n\nexport function encodeBgraFile(\n bgraPath: string,\n outputPath: string,\n spec: VideoSpec,\n format: \"mp4\" | \"webm\",\n): ProcessHandle {\n const files = format === \"mp4\" ? { mp4: outputPath } : { webm: outputPath };\n const cmd = createBgraFileCommand(bgraPath, spec, files);\n return exec(`${cmd.command} ${cmd.args.join(\" \")}`, {\n stdio: [\"ignore\", \"inherit\", \"inherit\"],\n });\n}\n\nexport function encodeBgraToMov(\n bgraPath: string,\n movPath: string,\n spec: VideoSpec,\n): ProcessHandle {\n const x265 = createBgraToMovPipeline(bgraPath, spec, movPath);\n const shellCmd = [\n `${x265.raw.command} ${x265.raw.args.join(\" \")}`,\n `${x265.x265.command} ${x265.x265.args.join(\" \")}`,\n `${x265.mux.command} ${x265.mux.args.join(\" \")}`,\n ].join(\" | \");\n return exec(shellCmd, {\n stdio: [\"ignore\", \"inherit\", \"inherit\"],\n shell: true,\n });\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nimport type { ChildProcess } from \"child_process\";\n\nexport interface StreamWriter {\n write(buffer: Buffer): Promise<void>;\n end(): void;\n}\n\nexport function createWriter(stdin: NodeJS.WritableStream): StreamWriter {\n const drainPromise = new Map<NodeJS.WritableStream, Promise<void>>();\n\n const waitDrain = (stream: NodeJS.WritableStream) => {\n const existing = drainPromise.get(stream);\n if (existing) return existing;\n const promise = new Promise<void>((resolve, reject) => {\n const cleanup = (fn: () => void) => {\n stream.off(\"drain\", onDrain);\n stream.off(\"error\", onError);\n stream.off(\"close\", onClose);\n drainPromise.delete(stream);\n fn();\n };\n const onDrain = () => cleanup(resolve);\n const onError = (err: Error) => cleanup(() => reject(err));\n const onClose = () => cleanup(() => reject(new Error(\"stream closed\")));\n stream.on(\"drain\", onDrain);\n stream.on(\"error\", onError);\n stream.on(\"close\", onClose);\n });\n drainPromise.set(stream, promise);\n return promise;\n };\n\n const destroyed = () => Reflect.get(stdin, \"destroyed\");\n return {\n async write(buffer: Buffer) {\n if (destroyed()) {\n throw new Error(\"stdin destroyed\");\n }\n if (!stdin.write(buffer)) {\n await waitDrain(stdin);\n }\n },\n end: () => {\n if (!destroyed()) stdin.end();\n },\n };\n}\n\nexport function pipeline(...procs: ChildProcess[]) {\n for (let i = 0; i < procs.length - 1; i++) {\n const src = procs[i];\n const dst = procs[i + 1];\n if (!src?.stdout || !dst?.stdin) throw new Error(\"pipeline broken\");\n src.stdout.pipe(dst.stdin);\n }\n}\n\nexport function waitAll(...procs: ChildProcess[]) {\n return Promise.all(\n procs.map(\n (proc) =>\n new Promise<void>((resolve, reject) => {\n proc.on(\"error\", reject);\n proc.on(\"close\", (code) =>\n code === 0\n ? resolve()\n : reject(new Error(`exit ${code ?? \"null\"}`)),\n );\n }),\n ),\n );\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nimport { program } from \"commander\";\nimport { pupUseInnerProxy } from \"./base/constants\";\nimport { logger } from \"./base/logging\";\nimport { noerr } from \"./base/noerr\";\nimport { parseNumber } from \"./base/parser\";\nimport { pargs } from \"./base/process\";\nimport type { RecordOptions } from \"./base/record\";\n\nexport const DEFAULT_WIDTH = 1920;\nexport const DEFAULT_HEIGHT = 1080;\nexport const DEFAULT_FPS = 30;\nexport const DEFAULT_DURATION = 5;\nexport const DEFAULT_OUT_DIR = \"out\";\n\nexport type CLICallback = (\n source: string,\n options: RecordOptions & Record<string, unknown>,\n) => Promise<unknown>;\n\nexport function makeCLI(name: string, callback: CLICallback) {\n program\n .name(name)\n .argument(\"<source>\", \"URL \u6216 HTML data\")\n .option(\"-w, --width <number>\", \"\u89C6\u9891\u5BBD\u5EA6\", `${DEFAULT_WIDTH}`)\n .option(\"-h, --height <number>\", \"\u89C6\u9891\u9AD8\u5EA6\", `${DEFAULT_HEIGHT}`)\n .option(\"-f, --fps <number>\", \"\u5E27\u7387\", `${DEFAULT_FPS}`)\n .option(\"-t, --duration <number>\", \"\u5F55\u5236\u65F6\u957F\uFF08\u79D2\uFF09\", `${DEFAULT_DURATION}`)\n .option(\"-o, --out-dir <path>\", \"\u8F93\u51FA\u76EE\u5F55\", `${DEFAULT_OUT_DIR}`)\n .option(\"-a, --with-alpha-channel\", \"\u8F93\u51FA\u5305\u542B alpha \u901A\u9053\u7684\u89C6\u9891\", false)\n .option(\n \"--use-inner-proxy\",\n \"\u4F7F\u7528 B \u7AD9\u5185\u7F51\u4EE3\u7406\u52A0\u901F\u8D44\u6E90\u8BBF\u95EE\",\n pupUseInnerProxy,\n )\n .action(async (source: string, opts) => {\n try {\n await callback(source, {\n width: noerr(parseNumber, DEFAULT_WIDTH)(opts.width),\n height: noerr(parseNumber, DEFAULT_HEIGHT)(opts.height),\n fps: noerr(parseNumber, DEFAULT_FPS)(opts.fps),\n duration: noerr(parseNumber, DEFAULT_DURATION)(opts.duration),\n outDir: opts.outDir ?? DEFAULT_OUT_DIR,\n withAlphaChannel: opts.withAlphaChannel ?? false,\n useInnerProxy: opts.useInnerProxy ?? pupUseInnerProxy,\n });\n } catch (e) {\n logger.fatal(e);\n }\n });\n program.parse(pargs());\n}\n"],
4
+ "sourcesContent": ["// Created by Autokaka (qq1909698494@gmail.com) on 2026/01/30.\n\nexport * from \"./base/constants\";\nexport * from \"./base/env\";\nexport * from \"./base/lazy\";\nexport * from \"./base/limiter\";\nexport * from \"./base/logging\";\nexport * from \"./base/noerr\";\nexport * from \"./base/parser\";\nexport * from \"./base/process\";\nexport * from \"./base/retry\";\nexport * from \"./base/timing\";\n\nexport { pup, type PupOptions } from \"./pup\";\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nimport { existsSync } from \"fs\";\nimport { resolve } from \"path\";\nimport { penv } from \"./env\";\nimport { parseNumber } from \"./parser\";\n\nconst pupAppSearchPaths = [\n resolve(__dirname, \"cjs/app.cjs\"), // process from dist\n resolve(__dirname, \"app.cjs\"), // process from dist/cjs\n resolve(__dirname, \"../../cjs/app.cjs\"), // process from src\n];\nexport const pupAppPath = pupAppSearchPaths.find(existsSync)!;\n\nconst env = process.env;\nexport const pupLogLevel = penv(\"PUP_LOG_LEVEL\", parseNumber, 2);\nexport const pupUseInnerProxy = env[\"PUP_USE_INNER_PROXY\"] === \"1\";\nexport const pupFFmpegPath = env[\"FFMPEG_BIN\"] ?? `ffmpeg`;\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/25.\n\nexport type EnvParser<T> = (value: unknown) => T;\n\nexport function penv<T>(name: string, parser: EnvParser<T>, defaultValue: T): T;\nexport function penv<T>(\n name: string,\n parser: EnvParser<T>,\n defaultValue?: T,\n): T | undefined;\nexport function penv<T>(\n name: string,\n parser: EnvParser<T>,\n defaultValue?: T,\n): T | undefined {\n try {\n return parser(process.env[name]);\n } catch {\n return defaultValue;\n }\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/01/30.\n\nexport function parseNumber(value: unknown): number {\n if (typeof value === \"number\") {\n return value;\n }\n const num = Number(value);\n if (Number.isNaN(num)) {\n throw new Error(`Value ${value} is not a valid number`);\n }\n return num;\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/01/30.\n\nexport class Lazy<T> {\n constructor(readonly makeValue: () => T) {}\n\n get value(): T {\n if (!this._initialized) {\n this._value = this.makeValue();\n this._initialized = true;\n }\n return this._value!;\n }\n\n get initialized(): boolean {\n return this._initialized;\n }\n\n private _initialized = false;\n private _value: T | undefined;\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/01/30.\n\nexport class ConcurrencyLimiter {\n private _active = 0;\n private _queue: VoidFunction[] = [];\n private _pending = 0;\n private _ended = false;\n\n constructor(readonly maxConcurrency: number) {}\n\n get active(): number {\n return this._active;\n }\n\n get pending(): number {\n return this._pending;\n }\n\n async schedule<T>(fn: () => Promise<T>): Promise<T> {\n if (this._ended) {\n throw new Error(\"ended\");\n }\n return new Promise<T>((resolve, reject) => {\n const run = () => {\n this._active++;\n this._pending--;\n fn()\n .then(resolve)\n .catch(reject)\n .finally(() => {\n this._active--;\n this.next();\n });\n };\n this._pending++;\n if (this._active < this.maxConcurrency) {\n run();\n } else {\n this._queue.push(run);\n }\n });\n }\n\n async end() {\n if (!this._ended) {\n this._ended = true;\n while (this._active > 0 || this._pending > 0) {\n await new Promise((resolve) => setTimeout(resolve, 50));\n }\n }\n }\n\n private next() {\n if (this._active < this.maxConcurrency && this._queue.length > 0) {\n this._queue.shift()?.();\n }\n }\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/06.\n\nimport { ChildProcess, type Serializable } from \"child_process\";\nimport { pupLogLevel } from \"./constants\";\n\nexport interface LoggerLike {\n debug?(this: void, ...messages: unknown[]): void;\n\n info?(this: void, ...messages: unknown[]): void;\n\n warn?(this: void, ...messages: unknown[]): void;\n\n error?(this: void, ...messages: unknown[]): void;\n}\n\nconst DEBUG = \"<pup@debug>\";\nconst INFO = \"<pup@info>\";\nconst WARN = \"<pup@warn>\";\nconst ERROR = \"<pup@error>\";\nconst FATAL = \"<pup@fatal>\";\n\nclass Logger implements LoggerLike {\n private _impl?: LoggerLike;\n\n get impl(): LoggerLike | undefined {\n return this._impl;\n }\n\n set impl(value: LoggerLike) {\n const debug = value.debug ?? console.debug;\n const info = value.info ?? console.info;\n const warn = value.warn ?? console.warn;\n const error = value.error ?? console.error;\n this._impl = {\n debug: pupLogLevel >= 3 ? debug : undefined,\n info: pupLogLevel >= 2 ? info : undefined,\n warn: pupLogLevel >= 1 ? warn : undefined,\n error: pupLogLevel >= 0 ? error : undefined,\n };\n }\n\n constructor() {\n this.impl = console;\n }\n\n debug(...messages: unknown[]): void {\n this.impl?.debug?.(DEBUG, ...messages);\n }\n\n info(...messages: unknown[]): void {\n this.impl?.info?.(INFO, ...messages);\n }\n\n warn(...messages: unknown[]): void {\n this.impl?.warn?.(WARN, ...messages);\n }\n\n error(...messages: unknown[]): void {\n this.impl?.error?.(ERROR, ...messages);\n }\n\n fatal(...messages: unknown[]): never {\n this.impl?.error?.(FATAL, ...messages);\n process.exit(1);\n }\n\n private dispatch(message: string) {\n if (message.startsWith(DEBUG)) {\n this.debug(message.slice(DEBUG.length + 1));\n } else if (message.startsWith(INFO)) {\n this.info(message.slice(INFO.length + 1));\n } else if (message.startsWith(WARN)) {\n this.warn(message.slice(WARN.length + 1));\n } else if (message.startsWith(ERROR)) {\n this.error(message.slice(ERROR.length + 1));\n } else {\n this.info(message);\n }\n }\n\n attach(proc: ChildProcess, name: string) {\n return new Promise<void>((resolve, reject) => {\n this.debug(`${name}.attach`);\n let fatal: string = \"\";\n const dispatch = (data: Buffer | Serializable) => {\n const message = data.toString();\n if (message.startsWith(FATAL)) {\n fatal += message.slice(FATAL.length + 1);\n } else {\n this.dispatch(message);\n }\n };\n proc.stderr?.on(\"data\", dispatch);\n proc.stdout?.on(\"data\", dispatch);\n proc\n .on(\"message\", dispatch)\n .on(\"error\", (err) => {\n fatal += err.message;\n proc.kill();\n })\n .once(\"close\", (code, signal) => {\n if (code || signal || fatal) {\n fatal ||= `command failed: ${proc.spawnargs.join(\" \")}`;\n this.error(`${name}.close`, { code, signal, fatal });\n reject(new Error(fatal));\n } else {\n this.debug(`${name}.close`);\n resolve();\n }\n })\n .on(\"unhandledRejection\", (reason) => {\n this.error(`${name}.unhandled`, reason);\n })\n .on(\"uncaughtExceptionMonitor\", (err) => {\n this.error(`${name}.unhandled`, err);\n });\n });\n }\n}\n\nconst logger = new Logger();\n\nexport { logger };\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/24.\n\nexport function noerr<Fn extends (...args: any[]) => any, D>(\n fn: Fn,\n defaultValue: D,\n): (...args: Parameters<Fn>) => ReturnType<Fn> | D {\n return (...args) => {\n try {\n const ret = fn(...args);\n if (ret instanceof Promise) {\n return ret.catch(() => defaultValue);\n }\n return ret;\n } catch {\n return defaultValue;\n }\n };\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/01/30.\n\nimport { spawn, type ChildProcess, type SpawnOptions } from \"child_process\";\nimport { logger } from \"./logging\";\n\nexport const PUP_ARGS_ENV_KEY = \"__PUP_ARGS__\";\n\nexport function pargs() {\n const pupArgs = process.env[PUP_ARGS_ENV_KEY];\n if (pupArgs) {\n const args = [\"exec\", ...process.argv.slice(-1)];\n args.push(...JSON.parse(pupArgs));\n logger.debug(\"pupargs\", args);\n return args;\n }\n\n logger.debug(\"procargv\", process.argv);\n return process.argv;\n}\n\nexport interface ProcessHandle {\n process: ChildProcess;\n wait: Promise<void>;\n}\n\nexport function exec(cmd: string, options?: SpawnOptions): ProcessHandle {\n const parts = cmd.split(\" \").filter((s) => s.length);\n const [command, ...args] = parts;\n if (!command) throw new Error(\"empty command\");\n const proc = spawn(command, args, {\n stdio: \"inherit\",\n ...options,\n });\n return { process: proc, wait: logger.attach(proc, command) };\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/05.\n\nimport { setTimeout } from \"timers/promises\";\nimport { sleep } from \"./timing\";\n\nexport interface RetryOptions<Args extends any[], Ret> {\n fn: (...args: Args) => Promise<Ret>;\n maxAttempts?: number;\n timeout?: number;\n}\n\nexport function useRetry<Args extends any[], Ret>({\n fn,\n maxAttempts = 3,\n timeout,\n}: RetryOptions<Args, Ret>) {\n const timeoutError = new Error(`timeout over ${timeout}ms`);\n return async function (...args: Args) {\n let attempt = 0;\n while (true) {\n try {\n const promises = [fn(...args)];\n if (timeout) {\n promises.push(\n setTimeout(timeout).then(() => {\n throw timeoutError;\n }),\n );\n }\n return await Promise.race(promises);\n } catch (e) {\n attempt++;\n if (attempt >= maxAttempts) {\n throw e;\n }\n await sleep(Math.pow(2, attempt) * 100 + Math.random() * 100);\n }\n }\n };\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nexport function sleep(ms: number) {\n return new Promise<void>((resolve) => setTimeout(resolve, ms));\n}\n\nexport function periodical(\n callback: (count: number) => Promise<void> | void,\n ms: number,\n) {\n let token: NodeJS.Timeout;\n let closed = false;\n async function tick(count: number) {\n await callback(count);\n if (closed) return;\n token = setTimeout(() => tick(count + 1), ms);\n }\n token = setTimeout(() => tick(0), ms);\n return () => {\n closed = true;\n clearTimeout(token);\n };\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nimport { spawn } from \"child_process\";\nimport type { Size } from \"electron\";\nimport { readFile, rm } from \"fs/promises\";\nimport { join } from \"path\";\nimport { AbortLink, type AbortQuery } from \"./base/abort\";\nimport { pupAppPath } from \"./base/constants\";\nimport { runElectronApp } from \"./base/electron\";\nimport { encodeBgraFile, encodeBgraToMov } from \"./base/encoder\";\nimport { createCoverCommand } from \"./base/ffmpeg\";\nimport { ConcurrencyLimiter } from \"./base/limiter\";\nimport { logger } from \"./base/logging\";\nimport { parseNumber } from \"./base/parser\";\nimport { type ProcessHandle } from \"./base/process\";\nimport type { RecordResult } from \"./base/record\";\nimport { waitAll } from \"./base/stream\";\nimport type { VideoFilesWithCover, VideoSpec } from \"./base/types\";\nimport { DEFAULT_HEIGHT, DEFAULT_WIDTH } from \"./common\";\n\nconst TAG = \"[pup]\";\n\nexport type PupProgressCallback = (progress: number) => Promise<void> | void;\n\nexport interface PupOptions {\n withAlphaChannel?: boolean;\n width?: number;\n height?: number;\n fps?: number;\n duration?: number;\n outDir?: string;\n cancelQuery?: AbortQuery;\n onProgress?: PupProgressCallback;\n}\n\nconst PROGRESS_TAG = \" progress: \";\n\nfunction runPupApp(source: string, options: PupOptions) {\n logger.debug(TAG, `runPupApp`, source, options);\n\n const args: string[] = [source];\n if (options.width) args.push(\"--width\", `${options.width}`);\n if (options.height) args.push(\"--height\", `${options.height}`);\n if (options.fps) args.push(\"--fps\", `${options.fps}`);\n if (options.duration) args.push(\"--duration\", `${options.duration}`);\n if (options.outDir) args.push(\"--out-dir\", options.outDir);\n if (options.withAlphaChannel) args.push(\"--with-alpha-channel\");\n\n const w = options.width ?? DEFAULT_WIDTH;\n const h = options.height ?? DEFAULT_HEIGHT;\n const handle = runElectronApp({ width: w, height: h }, pupAppPath, args);\n const counter = new ConcurrencyLimiter(1);\n handle.process.stdout?.on(\"data\", (data: Buffer) => {\n let message = data.toString().trim();\n let start = message.indexOf(PROGRESS_TAG);\n if (start < 0) {\n return;\n }\n message = message.slice(start + PROGRESS_TAG.length);\n const end = message.indexOf(\"%\");\n if (end < 0) {\n return;\n }\n const progressStr = message.slice(0, end);\n const progress = parseNumber(progressStr);\n counter.schedule(async () => {\n await options.onProgress?.(progress);\n });\n });\n return { handle, counter };\n}\n\nexport async function pup(source: string, options: PupOptions) {\n logger.debug(TAG, `pup`, source, options);\n\n const link = AbortLink.start(options.cancelQuery);\n const outDir = options.outDir ?? \"out\";\n\n const t0 = performance.now();\n const { handle, counter } = runPupApp(source, { ...options, outDir });\n\n await link.wait(handle);\n await counter.end();\n logger.info(TAG, `capture cost ${Math.round(performance.now() - t0)}ms`);\n\n const metaPath = join(outDir, \"record.json\");\n const meta = JSON.parse(await readFile(metaPath, \"utf-8\")) as RecordResult;\n\n const { bgraPath, written, options: recordOptions } = meta;\n const { fps, width, height, withAlphaChannel } = recordOptions;\n const size: Size = { width, height };\n\n const outputs: VideoFilesWithCover = {\n mp4: withAlphaChannel ? undefined : join(outDir, \"output.mp4\"),\n webm: withAlphaChannel ? join(outDir, \"output.webm\") : undefined,\n mov: withAlphaChannel ? join(outDir, \"output.mov\") : undefined,\n cover: join(outDir, \"cover.png\"),\n };\n\n try {\n const t1 = performance.now();\n\n const spec: VideoSpec = { fps, frames: written, size };\n const handles: ProcessHandle[] = [];\n if (outputs.mp4) {\n handles.push(encodeBgraFile(bgraPath, outputs.mp4, spec, \"mp4\"));\n }\n if (outputs.webm) {\n handles.push(encodeBgraFile(bgraPath, outputs.webm, spec, \"webm\"));\n }\n if (outputs.mov) {\n handles.push(encodeBgraToMov(bgraPath, outputs.mov, spec));\n }\n await link.wait(...handles);\n\n const coverSrc = outputs.mov ?? outputs.webm ?? outputs.mp4;\n if (coverSrc) {\n const coverCmd = createCoverCommand(coverSrc, outputs.cover);\n await waitAll(\n spawn(coverCmd.command, coverCmd.args, { stdio: \"inherit\" }),\n );\n }\n\n link.stop();\n logger.info(TAG, `encoding cost ${Math.round(performance.now() - t1)}ms`);\n\n await Promise.all([\n rm(bgraPath, { force: true }),\n rm(metaPath, { force: true }),\n ]);\n return {\n ...outputs,\n width,\n height,\n fps,\n duration: Math.ceil(written / fps),\n };\n } catch (error) {\n await rm(outDir, { recursive: true, force: true });\n throw error;\n }\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/11.\n\nimport type { ProcessHandle } from \"./process\";\n\nexport type AsyncTask = () => Promise<void> | void;\nexport type AbortQuery = () => Promise<boolean> | boolean;\n\nexport class AbortLink {\n private _callback?: AsyncTask;\n private _aborted?: boolean;\n private _stopped = false;\n\n private constructor(\n readonly query?: AbortQuery,\n readonly interval: number = 1000,\n ) {\n if (query) {\n this.tick();\n }\n }\n\n static start(query?: AbortQuery, interval?: number) {\n return new AbortLink(query, interval);\n }\n\n get aborted() {\n return !this._stopped && this._aborted;\n }\n\n get stopped() {\n return this._stopped;\n }\n\n async onAbort(callback: AsyncTask) {\n if (this._aborted) {\n await callback();\n } else {\n this._callback = callback;\n }\n }\n\n wait(...handles: ProcessHandle[]) {\n const abort = new Promise((_, reject) => {\n this.onAbort(async () => {\n handles.forEach((h) => h.process.kill());\n reject(new Error(\"aborted\"));\n });\n });\n return Promise.race([\n abort,\n Promise.all(handles.map((h) => h.wait)), //\n ]);\n }\n\n stop() {\n this._stopped = true;\n }\n\n private tick() {\n setTimeout(async () => {\n if (this._stopped) {\n return;\n }\n this._aborted = await this.query?.();\n if (this._stopped) {\n return;\n }\n if (this._aborted) {\n await this._callback?.();\n } else {\n this.tick();\n }\n }, this.interval);\n }\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/25.\n\nimport electron, { type Size } from \"electron\";\nimport { exec } from \"./process\";\n\nexport const ELECTRON_OPTS = [\n \"no-sandbox\",\n \"disable-setuid-sandbox\",\n \"disable-gpu\",\n \"disable-dev-shm-usage\",\n \"disable-software-rasterizer\",\n \"disable-web-security\",\n \"disable-site-isolation-trials\",\n \"disable-features=IsolateOrigins,site-per-process\",\n \"allow-insecure-localhost\",\n \"ignore-certificate-errors\",\n \"disable-blink-features=AutomationControlled\",\n \"mute-audio\",\n \"disable-extensions\",\n \"disable-background-networking\",\n \"address-family=ipv4\",\n \"disable-async-dns\",\n \"force-device-scale-factor=1\",\n \"trace-warnings\",\n \"force-color-profile=srgb\",\n \"disable-color-correct-rendering\",\n \"log-level=3\",\n];\n\nexport function runElectronApp(size: Size, app: unknown, args: unknown[]) {\n const electronArgs = ELECTRON_OPTS.map((a) => `--${a}`);\n const cmdParts = [];\n if (process.platform === \"linux\") {\n cmdParts.push(\n \"xvfb-run\",\n \"--auto-servernum\",\n `--server-args='-screen 0 ${size.width}x${size.height}x24'`,\n );\n }\n cmdParts.push(electron, ...electronArgs, app);\n return exec(cmdParts.join(\" \"), {\n stdio: [\"ignore\", \"pipe\", \"pipe\"],\n shell: true,\n env: {\n ...process.env,\n ELECTRON_DISABLE_DBUS: \"1\",\n RUST_BACKTRACE: \"full\",\n __PUP_ARGS__: JSON.stringify(args),\n },\n });\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nimport { existsSync } from \"fs\";\nimport { resolve } from \"path\";\nimport { arch, platform } from \"process\";\nimport { pupFFmpegPath } from \"./constants\";\nimport type { VideoFiles, VideoSpec } from \"./types\";\n\ninterface Command {\n command: string;\n args: string[];\n}\n\nconst quiet = [\"-hide_banner\", \"-loglevel\", \"error\", \"-nostats\"];\n\nfunction resolveX265() {\n const path = `x265/${platform}-${arch}`;\n const dirs = [\n resolve(__dirname, `../../${path}`), // process from src\n resolve(__dirname, `../${path}`), // process from dist\n ];\n const found = dirs.find(existsSync);\n if (!found) {\n throw new Error(\"x265 not found\");\n }\n return found;\n}\n\nexport function createBgraFileCommand(\n bgraPath: string,\n spec: VideoSpec,\n files: VideoFiles,\n): Command {\n const { fps, frames } = spec;\n const args = [\n \"-y\",\n ...quiet,\n \"-f\",\n \"rawvideo\",\n \"-pix_fmt\",\n \"bgra\",\n \"-s\",\n `${spec.size.width}x${spec.size.height}`,\n \"-r\",\n `${fps}`,\n \"-i\",\n bgraPath,\n \"-frames:v\",\n `${Math.floor(frames)}`,\n ];\n if (files.mp4) {\n args.push(\n \"-colorspace\",\n \"bt709\",\n \"-color_primaries\",\n \"bt709\",\n \"-color_trc\",\n \"bt709\",\n \"-c:v\",\n \"libx264\",\n \"-preset\",\n \"fast\",\n \"-threads\",\n \"0\",\n \"-pix_fmt\",\n \"yuv420p\",\n \"-movflags\",\n \"+faststart\",\n files.mp4,\n );\n }\n if (files.webm) {\n args.push(\n \"-c:v\",\n \"libvpx-vp9\",\n \"-row-mt\",\n \"1\",\n \"-cpu-used\",\n \"1\",\n \"-threads\",\n \"0\",\n \"-pix_fmt\",\n \"yuva420p\",\n \"-auto-alt-ref\",\n \"0\",\n files.webm,\n );\n }\n return { command: pupFFmpegPath, args };\n}\n\ninterface X265Pipeline {\n raw: Command;\n x265: Command;\n mux: Command;\n}\n\nexport function createBgraToMovPipeline(\n bgraPath: string,\n spec: VideoSpec,\n mov: string,\n): X265Pipeline {\n const { fps, size } = spec;\n return {\n raw: {\n command: pupFFmpegPath,\n args: [\n \"-y\",\n ...quiet,\n \"-f\",\n \"rawvideo\",\n \"-pix_fmt\",\n \"bgra\",\n \"-s\",\n `${size.width}x${size.height}`,\n \"-r\",\n `${fps}`,\n \"-i\",\n bgraPath,\n \"-f\",\n \"rawvideo\",\n \"-pix_fmt\",\n \"yuva420p10le\",\n \"pipe:1\",\n ],\n },\n x265: {\n command: resolveX265(),\n args: [\n \"--no-progress\",\n \"--log-level\",\n \"error\",\n \"--input\",\n \"-\",\n \"--input-res\",\n `${size.width}x${size.height}`,\n \"--input-csp\",\n \"i420\",\n \"--input-depth\",\n \"10\",\n \"--output-depth\",\n \"10\",\n \"--fps\",\n `${fps}`,\n \"--alpha\",\n \"--bframes\",\n \"0\",\n \"--colorprim\",\n \"bt709\",\n \"--transfer\",\n \"bt709\",\n \"--colormatrix\",\n \"bt709\",\n \"--crf\",\n \"18\",\n \"--output\",\n \"-\",\n ],\n },\n mux: {\n command: pupFFmpegPath,\n args: [\n \"-y\",\n ...quiet,\n \"-f\",\n \"hevc\",\n \"-r\",\n `${fps}`,\n \"-i\",\n \"pipe:0\",\n \"-c:v\",\n \"copy\",\n \"-tag:v\",\n \"hvc1\",\n \"-movflags\",\n \"+faststart\",\n mov,\n ],\n },\n };\n}\n\nexport function createCoverCommand(src: string, dst: string): Command {\n return {\n command: pupFFmpegPath,\n args: [\"-y\", ...quiet, \"-i\", src, \"-frames:v\", \"1\", \"-q:v\", \"2\", dst],\n };\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nimport { createBgraFileCommand, createBgraToMovPipeline } from \"./ffmpeg\";\nimport { exec, type ProcessHandle } from \"./process\";\nimport type { VideoSpec } from \"./types\";\n\nexport function encodeBgraFile(\n bgraPath: string,\n outputPath: string,\n spec: VideoSpec,\n format: \"mp4\" | \"webm\",\n): ProcessHandle {\n const files = format === \"mp4\" ? { mp4: outputPath } : { webm: outputPath };\n const cmd = createBgraFileCommand(bgraPath, spec, files);\n return exec(`${cmd.command} ${cmd.args.join(\" \")}`, {\n stdio: [\"ignore\", \"inherit\", \"inherit\"],\n });\n}\n\nexport function encodeBgraToMov(\n bgraPath: string,\n movPath: string,\n spec: VideoSpec,\n): ProcessHandle {\n const x265 = createBgraToMovPipeline(bgraPath, spec, movPath);\n const shellCmd = [\n `${x265.raw.command} ${x265.raw.args.join(\" \")}`,\n `${x265.x265.command} ${x265.x265.args.join(\" \")}`,\n `${x265.mux.command} ${x265.mux.args.join(\" \")}`,\n ].join(\" | \");\n return exec(shellCmd, {\n stdio: [\"ignore\", \"inherit\", \"inherit\"],\n shell: true,\n });\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nimport type { ChildProcess } from \"child_process\";\n\nexport interface StreamWriter {\n write(buffer: Buffer): Promise<void>;\n end(): void;\n}\n\nexport function createWriter(stdin: NodeJS.WritableStream): StreamWriter {\n const drainPromise = new Map<NodeJS.WritableStream, Promise<void>>();\n\n const waitDrain = (stream: NodeJS.WritableStream) => {\n const existing = drainPromise.get(stream);\n if (existing) return existing;\n const promise = new Promise<void>((resolve, reject) => {\n const cleanup = (fn: () => void) => {\n stream.off(\"drain\", onDrain);\n stream.off(\"error\", onError);\n stream.off(\"close\", onClose);\n drainPromise.delete(stream);\n fn();\n };\n const onDrain = () => cleanup(resolve);\n const onError = (err: Error) => cleanup(() => reject(err));\n const onClose = () => cleanup(() => reject(new Error(\"stream closed\")));\n stream.on(\"drain\", onDrain);\n stream.on(\"error\", onError);\n stream.on(\"close\", onClose);\n });\n drainPromise.set(stream, promise);\n return promise;\n };\n\n const destroyed = () => Reflect.get(stdin, \"destroyed\");\n return {\n async write(buffer: Buffer) {\n if (destroyed()) {\n throw new Error(\"stdin destroyed\");\n }\n if (!stdin.write(buffer)) {\n await waitDrain(stdin);\n }\n },\n end: () => {\n if (!destroyed()) stdin.end();\n },\n };\n}\n\nexport function pipeline(...procs: ChildProcess[]) {\n for (let i = 0; i < procs.length - 1; i++) {\n const src = procs[i];\n const dst = procs[i + 1];\n if (!src?.stdout || !dst?.stdin) throw new Error(\"pipeline broken\");\n src.stdout.pipe(dst.stdin);\n }\n}\n\nexport function waitAll(...procs: ChildProcess[]) {\n return Promise.all(\n procs.map(\n (proc) =>\n new Promise<void>((resolve, reject) => {\n proc.on(\"error\", reject);\n proc.on(\"close\", (code) =>\n code === 0\n ? resolve()\n : reject(new Error(`exit ${code ?? \"null\"}`)),\n );\n }),\n ),\n );\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nimport { program } from \"commander\";\nimport { pupUseInnerProxy } from \"./base/constants\";\nimport { logger } from \"./base/logging\";\nimport { noerr } from \"./base/noerr\";\nimport { parseNumber } from \"./base/parser\";\nimport { pargs } from \"./base/process\";\nimport type { RecordOptions } from \"./base/record\";\n\nexport const DEFAULT_WIDTH = 1920;\nexport const DEFAULT_HEIGHT = 1080;\nexport const DEFAULT_FPS = 30;\nexport const DEFAULT_DURATION = 5;\nexport const DEFAULT_OUT_DIR = \"out\";\n\nexport type CLICallback = (\n source: string,\n options: RecordOptions & Record<string, unknown>,\n) => Promise<unknown>;\n\nexport function makeCLI(name: string, callback: CLICallback) {\n program\n .name(name)\n .argument(\"<source>\", \"file://, http(s)://, \u6216 data: URI\")\n .option(\"-w, --width <number>\", \"\u89C6\u9891\u5BBD\u5EA6\", `${DEFAULT_WIDTH}`)\n .option(\"-h, --height <number>\", \"\u89C6\u9891\u9AD8\u5EA6\", `${DEFAULT_HEIGHT}`)\n .option(\"-f, --fps <number>\", \"\u5E27\u7387\", `${DEFAULT_FPS}`)\n .option(\"-t, --duration <number>\", \"\u5F55\u5236\u65F6\u957F\uFF08\u79D2\uFF09\", `${DEFAULT_DURATION}`)\n .option(\"-o, --out-dir <path>\", \"\u8F93\u51FA\u76EE\u5F55\", `${DEFAULT_OUT_DIR}`)\n .option(\"-a, --with-alpha-channel\", \"\u8F93\u51FA\u5305\u542B alpha \u901A\u9053\u7684\u89C6\u9891\", false)\n .option(\n \"--use-inner-proxy\",\n \"\u4F7F\u7528 B \u7AD9\u5185\u7F51\u4EE3\u7406\u52A0\u901F\u8D44\u6E90\u8BBF\u95EE\",\n pupUseInnerProxy,\n )\n .action(async (source: string, opts) => {\n try {\n await callback(source, {\n width: noerr(parseNumber, DEFAULT_WIDTH)(opts.width),\n height: noerr(parseNumber, DEFAULT_HEIGHT)(opts.height),\n fps: noerr(parseNumber, DEFAULT_FPS)(opts.fps),\n duration: noerr(parseNumber, DEFAULT_DURATION)(opts.duration),\n outDir: opts.outDir ?? DEFAULT_OUT_DIR,\n withAlphaChannel: opts.withAlphaChannel ?? false,\n useInnerProxy: opts.useInnerProxy ?? pupUseInnerProxy,\n });\n } catch (e) {\n logger.fatal(e);\n }\n });\n program.parse(pargs());\n}\n"],
5
5
  "mappings": "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",
6
6
  "names": ["resolve", "resolve", "resolve", "import_child_process", "import_promises", "import_path", "electron", "import_fs", "import_path", "import_process", "resolve"]
7
7
  }
package/dist/cli.js CHANGED
@@ -179,7 +179,7 @@ var DEFAULT_FPS = 30;
179
179
  var DEFAULT_DURATION = 5;
180
180
  var DEFAULT_OUT_DIR = "out";
181
181
  function makeCLI(name, callback) {
182
- program.name(name).argument("<source>", "URL \u6216 HTML data").option("-w, --width <number>", "\u89C6\u9891\u5BBD\u5EA6", `${DEFAULT_WIDTH}`).option("-h, --height <number>", "\u89C6\u9891\u9AD8\u5EA6", `${DEFAULT_HEIGHT}`).option("-f, --fps <number>", "\u5E27\u7387", `${DEFAULT_FPS}`).option("-t, --duration <number>", "\u5F55\u5236\u65F6\u957F\uFF08\u79D2\uFF09", `${DEFAULT_DURATION}`).option("-o, --out-dir <path>", "\u8F93\u51FA\u76EE\u5F55", `${DEFAULT_OUT_DIR}`).option("-a, --with-alpha-channel", "\u8F93\u51FA\u5305\u542B alpha \u901A\u9053\u7684\u89C6\u9891", false).option(
182
+ program.name(name).argument("<source>", "file://, http(s)://, \u6216 data: URI").option("-w, --width <number>", "\u89C6\u9891\u5BBD\u5EA6", `${DEFAULT_WIDTH}`).option("-h, --height <number>", "\u89C6\u9891\u9AD8\u5EA6", `${DEFAULT_HEIGHT}`).option("-f, --fps <number>", "\u5E27\u7387", `${DEFAULT_FPS}`).option("-t, --duration <number>", "\u5F55\u5236\u65F6\u957F\uFF08\u79D2\uFF09", `${DEFAULT_DURATION}`).option("-o, --out-dir <path>", "\u8F93\u51FA\u76EE\u5F55", `${DEFAULT_OUT_DIR}`).option("-a, --with-alpha-channel", "\u8F93\u51FA\u5305\u542B alpha \u901A\u9053\u7684\u89C6\u9891", false).option(
183
183
  "--use-inner-proxy",
184
184
  "\u4F7F\u7528 B \u7AD9\u5185\u7F51\u4EE3\u7406\u52A0\u901F\u8D44\u6E90\u8BBF\u95EE",
185
185
  pupUseInnerProxy
package/dist/cli.js.map CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../src/common.ts", "../src/base/constants.ts", "../src/base/env.ts", "../src/base/parser.ts", "../src/base/logging.ts", "../src/base/noerr.ts", "../src/base/process.ts", "../src/pup.ts", "../src/base/abort.ts", "../src/base/electron.ts", "../src/base/ffmpeg.ts", "../src/base/encoder.ts", "../src/base/limiter.ts", "../src/base/stream.ts", "../src/cli.ts"],
4
- "sourcesContent": ["// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nimport { program } from \"commander\";\nimport { pupUseInnerProxy } from \"./base/constants\";\nimport { logger } from \"./base/logging\";\nimport { noerr } from \"./base/noerr\";\nimport { parseNumber } from \"./base/parser\";\nimport { pargs } from \"./base/process\";\nimport type { RecordOptions } from \"./base/record\";\n\nexport const DEFAULT_WIDTH = 1920;\nexport const DEFAULT_HEIGHT = 1080;\nexport const DEFAULT_FPS = 30;\nexport const DEFAULT_DURATION = 5;\nexport const DEFAULT_OUT_DIR = \"out\";\n\nexport type CLICallback = (\n source: string,\n options: RecordOptions & Record<string, unknown>,\n) => Promise<unknown>;\n\nexport function makeCLI(name: string, callback: CLICallback) {\n program\n .name(name)\n .argument(\"<source>\", \"URL \u6216 HTML data\")\n .option(\"-w, --width <number>\", \"\u89C6\u9891\u5BBD\u5EA6\", `${DEFAULT_WIDTH}`)\n .option(\"-h, --height <number>\", \"\u89C6\u9891\u9AD8\u5EA6\", `${DEFAULT_HEIGHT}`)\n .option(\"-f, --fps <number>\", \"\u5E27\u7387\", `${DEFAULT_FPS}`)\n .option(\"-t, --duration <number>\", \"\u5F55\u5236\u65F6\u957F\uFF08\u79D2\uFF09\", `${DEFAULT_DURATION}`)\n .option(\"-o, --out-dir <path>\", \"\u8F93\u51FA\u76EE\u5F55\", `${DEFAULT_OUT_DIR}`)\n .option(\"-a, --with-alpha-channel\", \"\u8F93\u51FA\u5305\u542B alpha \u901A\u9053\u7684\u89C6\u9891\", false)\n .option(\n \"--use-inner-proxy\",\n \"\u4F7F\u7528 B \u7AD9\u5185\u7F51\u4EE3\u7406\u52A0\u901F\u8D44\u6E90\u8BBF\u95EE\",\n pupUseInnerProxy,\n )\n .action(async (source: string, opts) => {\n try {\n await callback(source, {\n width: noerr(parseNumber, DEFAULT_WIDTH)(opts.width),\n height: noerr(parseNumber, DEFAULT_HEIGHT)(opts.height),\n fps: noerr(parseNumber, DEFAULT_FPS)(opts.fps),\n duration: noerr(parseNumber, DEFAULT_DURATION)(opts.duration),\n outDir: opts.outDir ?? DEFAULT_OUT_DIR,\n withAlphaChannel: opts.withAlphaChannel ?? false,\n useInnerProxy: opts.useInnerProxy ?? pupUseInnerProxy,\n });\n } catch (e) {\n logger.fatal(e);\n }\n });\n program.parse(pargs());\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nimport { existsSync } from \"fs\";\nimport { resolve } from \"path\";\nimport { penv } from \"./env\";\nimport { parseNumber } from \"./parser\";\n\nconst pupAppSearchPaths = [\n resolve(__dirname, \"cjs/app.cjs\"), // process from dist\n resolve(__dirname, \"app.cjs\"), // process from dist/cjs\n resolve(__dirname, \"../../cjs/app.cjs\"), // process from src\n];\nexport const pupAppPath = pupAppSearchPaths.find(existsSync)!;\n\nconst env = process.env;\nexport const pupLogLevel = penv(\"PUP_LOG_LEVEL\", parseNumber, 2);\nexport const pupUseInnerProxy = env[\"PUP_USE_INNER_PROXY\"] === \"1\";\nexport const pupFFmpegPath = env[\"FFMPEG_BIN\"] ?? `ffmpeg`;\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/25.\n\nexport type EnvParser<T> = (value: unknown) => T;\n\nexport function penv<T>(name: string, parser: EnvParser<T>, defaultValue: T): T;\nexport function penv<T>(\n name: string,\n parser: EnvParser<T>,\n defaultValue?: T,\n): T | undefined;\nexport function penv<T>(\n name: string,\n parser: EnvParser<T>,\n defaultValue?: T,\n): T | undefined {\n try {\n return parser(process.env[name]);\n } catch {\n return defaultValue;\n }\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/01/30.\n\nexport function parseNumber(value: unknown): number {\n if (typeof value === \"number\") {\n return value;\n }\n const num = Number(value);\n if (Number.isNaN(num)) {\n throw new Error(`Value ${value} is not a valid number`);\n }\n return num;\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/06.\n\nimport { ChildProcess, type Serializable } from \"child_process\";\nimport { pupLogLevel } from \"./constants\";\n\nexport interface LoggerLike {\n debug?(this: void, ...messages: unknown[]): void;\n\n info?(this: void, ...messages: unknown[]): void;\n\n warn?(this: void, ...messages: unknown[]): void;\n\n error?(this: void, ...messages: unknown[]): void;\n}\n\nconst DEBUG = \"<pup@debug>\";\nconst INFO = \"<pup@info>\";\nconst WARN = \"<pup@warn>\";\nconst ERROR = \"<pup@error>\";\nconst FATAL = \"<pup@fatal>\";\n\nclass Logger implements LoggerLike {\n private _impl?: LoggerLike;\n\n get impl(): LoggerLike | undefined {\n return this._impl;\n }\n\n set impl(value: LoggerLike) {\n const debug = value.debug ?? console.debug;\n const info = value.info ?? console.info;\n const warn = value.warn ?? console.warn;\n const error = value.error ?? console.error;\n this._impl = {\n debug: pupLogLevel >= 3 ? debug : undefined,\n info: pupLogLevel >= 2 ? info : undefined,\n warn: pupLogLevel >= 1 ? warn : undefined,\n error: pupLogLevel >= 0 ? error : undefined,\n };\n }\n\n constructor() {\n this.impl = console;\n }\n\n debug(...messages: unknown[]): void {\n this.impl?.debug?.(DEBUG, ...messages);\n }\n\n info(...messages: unknown[]): void {\n this.impl?.info?.(INFO, ...messages);\n }\n\n warn(...messages: unknown[]): void {\n this.impl?.warn?.(WARN, ...messages);\n }\n\n error(...messages: unknown[]): void {\n this.impl?.error?.(ERROR, ...messages);\n }\n\n fatal(...messages: unknown[]): never {\n this.impl?.error?.(FATAL, ...messages);\n process.exit(1);\n }\n\n private dispatch(message: string) {\n if (message.startsWith(DEBUG)) {\n this.debug(message.slice(DEBUG.length + 1));\n } else if (message.startsWith(INFO)) {\n this.info(message.slice(INFO.length + 1));\n } else if (message.startsWith(WARN)) {\n this.warn(message.slice(WARN.length + 1));\n } else if (message.startsWith(ERROR)) {\n this.error(message.slice(ERROR.length + 1));\n } else {\n this.info(message);\n }\n }\n\n attach(proc: ChildProcess, name: string) {\n return new Promise<void>((resolve, reject) => {\n this.debug(`${name}.attach`);\n let fatal: string = \"\";\n const dispatch = (data: Buffer | Serializable) => {\n const message = data.toString();\n if (message.startsWith(FATAL)) {\n fatal += message.slice(FATAL.length + 1);\n } else {\n this.dispatch(message);\n }\n };\n proc.stderr?.on(\"data\", dispatch);\n proc.stdout?.on(\"data\", dispatch);\n proc\n .on(\"message\", dispatch)\n .on(\"error\", (err) => {\n fatal += err.message;\n proc.kill();\n })\n .once(\"close\", (code, signal) => {\n if (code || signal || fatal) {\n fatal ||= `command failed: ${proc.spawnargs.join(\" \")}`;\n this.error(`${name}.close`, { code, signal, fatal });\n reject(new Error(fatal));\n } else {\n this.debug(`${name}.close`);\n resolve();\n }\n })\n .on(\"unhandledRejection\", (reason) => {\n this.error(`${name}.unhandled`, reason);\n })\n .on(\"uncaughtExceptionMonitor\", (err) => {\n this.error(`${name}.unhandled`, err);\n });\n });\n }\n}\n\nconst logger = new Logger();\n\nexport { logger };\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/24.\n\nexport function noerr<Fn extends (...args: any[]) => any, D>(\n fn: Fn,\n defaultValue: D,\n): (...args: Parameters<Fn>) => ReturnType<Fn> | D {\n return (...args) => {\n try {\n const ret = fn(...args);\n if (ret instanceof Promise) {\n return ret.catch(() => defaultValue);\n }\n return ret;\n } catch {\n return defaultValue;\n }\n };\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/01/30.\n\nimport { spawn, type ChildProcess, type SpawnOptions } from \"child_process\";\nimport { logger } from \"./logging\";\n\nexport const PUP_ARGS_ENV_KEY = \"__PUP_ARGS__\";\n\nexport function pargs() {\n const pupArgs = process.env[PUP_ARGS_ENV_KEY];\n if (pupArgs) {\n const args = [\"exec\", ...process.argv.slice(-1)];\n args.push(...JSON.parse(pupArgs));\n logger.debug(\"pupargs\", args);\n return args;\n }\n\n logger.debug(\"procargv\", process.argv);\n return process.argv;\n}\n\nexport interface ProcessHandle {\n process: ChildProcess;\n wait: Promise<void>;\n}\n\nexport function exec(cmd: string, options?: SpawnOptions): ProcessHandle {\n const parts = cmd.split(\" \").filter((s) => s.length);\n const [command, ...args] = parts;\n if (!command) throw new Error(\"empty command\");\n const proc = spawn(command, args, {\n stdio: \"inherit\",\n ...options,\n });\n return { process: proc, wait: logger.attach(proc, command) };\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nimport { spawn } from \"child_process\";\nimport type { Size } from \"electron\";\nimport { readFile, rm } from \"fs/promises\";\nimport { join } from \"path\";\nimport { AbortLink, type AbortQuery } from \"./base/abort\";\nimport { pupAppPath } from \"./base/constants\";\nimport { runElectronApp } from \"./base/electron\";\nimport { encodeBgraFile, encodeBgraToMov } from \"./base/encoder\";\nimport { createCoverCommand } from \"./base/ffmpeg\";\nimport { ConcurrencyLimiter } from \"./base/limiter\";\nimport { logger } from \"./base/logging\";\nimport { parseNumber } from \"./base/parser\";\nimport { type ProcessHandle } from \"./base/process\";\nimport type { RecordResult } from \"./base/record\";\nimport { waitAll } from \"./base/stream\";\nimport type { VideoFilesWithCover, VideoSpec } from \"./base/types\";\nimport { DEFAULT_HEIGHT, DEFAULT_WIDTH } from \"./common\";\n\nconst TAG = \"[pup]\";\n\nexport type PupProgressCallback = (progress: number) => Promise<void> | void;\n\nexport interface PupOptions {\n withAlphaChannel?: boolean;\n width?: number;\n height?: number;\n fps?: number;\n duration?: number;\n outDir?: string;\n cancelQuery?: AbortQuery;\n onProgress?: PupProgressCallback;\n}\n\nconst PROGRESS_TAG = \" progress: \";\n\nfunction runPupApp(source: string, options: PupOptions) {\n logger.debug(TAG, `runPupApp`, source, options);\n\n const args: string[] = [source];\n if (options.width) args.push(\"--width\", `${options.width}`);\n if (options.height) args.push(\"--height\", `${options.height}`);\n if (options.fps) args.push(\"--fps\", `${options.fps}`);\n if (options.duration) args.push(\"--duration\", `${options.duration}`);\n if (options.outDir) args.push(\"--out-dir\", options.outDir);\n if (options.withAlphaChannel) args.push(\"--with-alpha-channel\");\n\n const w = options.width ?? DEFAULT_WIDTH;\n const h = options.height ?? DEFAULT_HEIGHT;\n const handle = runElectronApp({ width: w, height: h }, pupAppPath, args);\n const counter = new ConcurrencyLimiter(1);\n handle.process.stdout?.on(\"data\", (data: Buffer) => {\n let message = data.toString().trim();\n let start = message.indexOf(PROGRESS_TAG);\n if (start < 0) {\n return;\n }\n message = message.slice(start + PROGRESS_TAG.length);\n const end = message.indexOf(\"%\");\n if (end < 0) {\n return;\n }\n const progressStr = message.slice(0, end);\n const progress = parseNumber(progressStr);\n counter.schedule(async () => {\n await options.onProgress?.(progress);\n });\n });\n return { handle, counter };\n}\n\nexport async function pup(source: string, options: PupOptions) {\n logger.debug(TAG, `pup`, source, options);\n\n const link = AbortLink.start(options.cancelQuery);\n const outDir = options.outDir ?? \"out\";\n\n const t0 = performance.now();\n const { handle, counter } = runPupApp(source, { ...options, outDir });\n\n await link.wait(handle);\n await counter.end();\n logger.info(TAG, `capture cost ${Math.round(performance.now() - t0)}ms`);\n\n const metaPath = join(outDir, \"record.json\");\n const meta = JSON.parse(await readFile(metaPath, \"utf-8\")) as RecordResult;\n\n const { bgraPath, written, options: recordOptions } = meta;\n const { fps, width, height, withAlphaChannel } = recordOptions;\n const size: Size = { width, height };\n\n const outputs: VideoFilesWithCover = {\n mp4: withAlphaChannel ? undefined : join(outDir, \"output.mp4\"),\n webm: withAlphaChannel ? join(outDir, \"output.webm\") : undefined,\n mov: withAlphaChannel ? join(outDir, \"output.mov\") : undefined,\n cover: join(outDir, \"cover.png\"),\n };\n\n try {\n const t1 = performance.now();\n\n const spec: VideoSpec = { fps, frames: written, size };\n const handles: ProcessHandle[] = [];\n if (outputs.mp4) {\n handles.push(encodeBgraFile(bgraPath, outputs.mp4, spec, \"mp4\"));\n }\n if (outputs.webm) {\n handles.push(encodeBgraFile(bgraPath, outputs.webm, spec, \"webm\"));\n }\n if (outputs.mov) {\n handles.push(encodeBgraToMov(bgraPath, outputs.mov, spec));\n }\n await link.wait(...handles);\n\n const coverSrc = outputs.mov ?? outputs.webm ?? outputs.mp4;\n if (coverSrc) {\n const coverCmd = createCoverCommand(coverSrc, outputs.cover);\n await waitAll(\n spawn(coverCmd.command, coverCmd.args, { stdio: \"inherit\" }),\n );\n }\n\n link.stop();\n logger.info(TAG, `encoding cost ${Math.round(performance.now() - t1)}ms`);\n\n await Promise.all([\n rm(bgraPath, { force: true }),\n rm(metaPath, { force: true }),\n ]);\n return {\n ...outputs,\n width,\n height,\n fps,\n duration: Math.ceil(written / fps),\n };\n } catch (error) {\n await rm(outDir, { recursive: true, force: true });\n throw error;\n }\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/11.\n\nimport type { ProcessHandle } from \"./process\";\n\nexport type AsyncTask = () => Promise<void> | void;\nexport type AbortQuery = () => Promise<boolean> | boolean;\n\nexport class AbortLink {\n private _callback?: AsyncTask;\n private _aborted?: boolean;\n private _stopped = false;\n\n private constructor(\n readonly query?: AbortQuery,\n readonly interval: number = 1000,\n ) {\n if (query) {\n this.tick();\n }\n }\n\n static start(query?: AbortQuery, interval?: number) {\n return new AbortLink(query, interval);\n }\n\n get aborted() {\n return !this._stopped && this._aborted;\n }\n\n get stopped() {\n return this._stopped;\n }\n\n async onAbort(callback: AsyncTask) {\n if (this._aborted) {\n await callback();\n } else {\n this._callback = callback;\n }\n }\n\n wait(...handles: ProcessHandle[]) {\n const abort = new Promise((_, reject) => {\n this.onAbort(async () => {\n handles.forEach((h) => h.process.kill());\n reject(new Error(\"aborted\"));\n });\n });\n return Promise.race([\n abort,\n Promise.all(handles.map((h) => h.wait)), //\n ]);\n }\n\n stop() {\n this._stopped = true;\n }\n\n private tick() {\n setTimeout(async () => {\n if (this._stopped) {\n return;\n }\n this._aborted = await this.query?.();\n if (this._stopped) {\n return;\n }\n if (this._aborted) {\n await this._callback?.();\n } else {\n this.tick();\n }\n }, this.interval);\n }\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/25.\n\nimport electron, { type Size } from \"electron\";\nimport { exec } from \"./process\";\n\nexport const ELECTRON_OPTS = [\n \"no-sandbox\",\n \"disable-setuid-sandbox\",\n \"disable-gpu\",\n \"disable-dev-shm-usage\",\n \"disable-software-rasterizer\",\n \"disable-web-security\",\n \"disable-site-isolation-trials\",\n \"disable-features=IsolateOrigins,site-per-process\",\n \"allow-insecure-localhost\",\n \"ignore-certificate-errors\",\n \"disable-blink-features=AutomationControlled\",\n \"mute-audio\",\n \"disable-extensions\",\n \"disable-background-networking\",\n \"address-family=ipv4\",\n \"disable-async-dns\",\n \"force-device-scale-factor=1\",\n \"trace-warnings\",\n \"force-color-profile=srgb\",\n \"disable-color-correct-rendering\",\n \"log-level=3\",\n];\n\nexport function runElectronApp(size: Size, app: unknown, args: unknown[]) {\n const electronArgs = ELECTRON_OPTS.map((a) => `--${a}`);\n const cmdParts = [];\n if (process.platform === \"linux\") {\n cmdParts.push(\n \"xvfb-run\",\n \"--auto-servernum\",\n `--server-args='-screen 0 ${size.width}x${size.height}x24'`,\n );\n }\n cmdParts.push(electron, ...electronArgs, app);\n return exec(cmdParts.join(\" \"), {\n stdio: [\"ignore\", \"pipe\", \"pipe\"],\n shell: true,\n env: {\n ...process.env,\n ELECTRON_DISABLE_DBUS: \"1\",\n RUST_BACKTRACE: \"full\",\n __PUP_ARGS__: JSON.stringify(args),\n },\n });\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nimport { existsSync } from \"fs\";\nimport { resolve } from \"path\";\nimport { arch, platform } from \"process\";\nimport { pupFFmpegPath } from \"./constants\";\nimport type { VideoFiles, VideoSpec } from \"./types\";\n\ninterface Command {\n command: string;\n args: string[];\n}\n\nconst quiet = [\"-hide_banner\", \"-loglevel\", \"error\", \"-nostats\"];\n\nfunction resolveX265() {\n const path = `x265/${platform}-${arch}`;\n const dirs = [\n resolve(__dirname, `../../${path}`), // process from src\n resolve(__dirname, `../${path}`), // process from dist\n ];\n const found = dirs.find(existsSync);\n if (!found) {\n throw new Error(\"x265 not found\");\n }\n return found;\n}\n\nexport function createBgraFileCommand(\n bgraPath: string,\n spec: VideoSpec,\n files: VideoFiles,\n): Command {\n const { fps, frames } = spec;\n const args = [\n \"-y\",\n ...quiet,\n \"-f\",\n \"rawvideo\",\n \"-pix_fmt\",\n \"bgra\",\n \"-s\",\n `${spec.size.width}x${spec.size.height}`,\n \"-r\",\n `${fps}`,\n \"-i\",\n bgraPath,\n \"-frames:v\",\n `${Math.floor(frames)}`,\n ];\n if (files.mp4) {\n args.push(\n \"-colorspace\",\n \"bt709\",\n \"-color_primaries\",\n \"bt709\",\n \"-color_trc\",\n \"bt709\",\n \"-c:v\",\n \"libx264\",\n \"-preset\",\n \"fast\",\n \"-threads\",\n \"0\",\n \"-pix_fmt\",\n \"yuv420p\",\n \"-movflags\",\n \"+faststart\",\n files.mp4,\n );\n }\n if (files.webm) {\n args.push(\n \"-c:v\",\n \"libvpx-vp9\",\n \"-row-mt\",\n \"1\",\n \"-cpu-used\",\n \"1\",\n \"-threads\",\n \"0\",\n \"-pix_fmt\",\n \"yuva420p\",\n \"-auto-alt-ref\",\n \"0\",\n files.webm,\n );\n }\n return { command: pupFFmpegPath, args };\n}\n\ninterface X265Pipeline {\n raw: Command;\n x265: Command;\n mux: Command;\n}\n\nexport function createBgraToMovPipeline(\n bgraPath: string,\n spec: VideoSpec,\n mov: string,\n): X265Pipeline {\n const { fps, size } = spec;\n return {\n raw: {\n command: pupFFmpegPath,\n args: [\n \"-y\",\n ...quiet,\n \"-f\",\n \"rawvideo\",\n \"-pix_fmt\",\n \"bgra\",\n \"-s\",\n `${size.width}x${size.height}`,\n \"-r\",\n `${fps}`,\n \"-i\",\n bgraPath,\n \"-f\",\n \"rawvideo\",\n \"-pix_fmt\",\n \"yuva420p10le\",\n \"pipe:1\",\n ],\n },\n x265: {\n command: resolveX265(),\n args: [\n \"--no-progress\",\n \"--log-level\",\n \"error\",\n \"--input\",\n \"-\",\n \"--input-res\",\n `${size.width}x${size.height}`,\n \"--input-csp\",\n \"i420\",\n \"--input-depth\",\n \"10\",\n \"--output-depth\",\n \"10\",\n \"--fps\",\n `${fps}`,\n \"--alpha\",\n \"--bframes\",\n \"0\",\n \"--colorprim\",\n \"bt709\",\n \"--transfer\",\n \"bt709\",\n \"--colormatrix\",\n \"bt709\",\n \"--crf\",\n \"18\",\n \"--output\",\n \"-\",\n ],\n },\n mux: {\n command: pupFFmpegPath,\n args: [\n \"-y\",\n ...quiet,\n \"-f\",\n \"hevc\",\n \"-r\",\n `${fps}`,\n \"-i\",\n \"pipe:0\",\n \"-c:v\",\n \"copy\",\n \"-tag:v\",\n \"hvc1\",\n \"-movflags\",\n \"+faststart\",\n mov,\n ],\n },\n };\n}\n\nexport function createCoverCommand(src: string, dst: string): Command {\n return {\n command: pupFFmpegPath,\n args: [\"-y\", ...quiet, \"-i\", src, \"-frames:v\", \"1\", \"-q:v\", \"2\", dst],\n };\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nimport { createBgraFileCommand, createBgraToMovPipeline } from \"./ffmpeg\";\nimport { exec, type ProcessHandle } from \"./process\";\nimport type { VideoSpec } from \"./types\";\n\nexport function encodeBgraFile(\n bgraPath: string,\n outputPath: string,\n spec: VideoSpec,\n format: \"mp4\" | \"webm\",\n): ProcessHandle {\n const files = format === \"mp4\" ? { mp4: outputPath } : { webm: outputPath };\n const cmd = createBgraFileCommand(bgraPath, spec, files);\n return exec(`${cmd.command} ${cmd.args.join(\" \")}`, {\n stdio: [\"ignore\", \"inherit\", \"inherit\"],\n });\n}\n\nexport function encodeBgraToMov(\n bgraPath: string,\n movPath: string,\n spec: VideoSpec,\n): ProcessHandle {\n const x265 = createBgraToMovPipeline(bgraPath, spec, movPath);\n const shellCmd = [\n `${x265.raw.command} ${x265.raw.args.join(\" \")}`,\n `${x265.x265.command} ${x265.x265.args.join(\" \")}`,\n `${x265.mux.command} ${x265.mux.args.join(\" \")}`,\n ].join(\" | \");\n return exec(shellCmd, {\n stdio: [\"ignore\", \"inherit\", \"inherit\"],\n shell: true,\n });\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/01/30.\n\nexport class ConcurrencyLimiter {\n private _active = 0;\n private _queue: VoidFunction[] = [];\n private _pending = 0;\n private _ended = false;\n\n constructor(readonly maxConcurrency: number) {}\n\n get active(): number {\n return this._active;\n }\n\n get pending(): number {\n return this._pending;\n }\n\n async schedule<T>(fn: () => Promise<T>): Promise<T> {\n if (this._ended) {\n throw new Error(\"ended\");\n }\n return new Promise<T>((resolve, reject) => {\n const run = () => {\n this._active++;\n this._pending--;\n fn()\n .then(resolve)\n .catch(reject)\n .finally(() => {\n this._active--;\n this.next();\n });\n };\n this._pending++;\n if (this._active < this.maxConcurrency) {\n run();\n } else {\n this._queue.push(run);\n }\n });\n }\n\n async end() {\n if (!this._ended) {\n this._ended = true;\n while (this._active > 0 || this._pending > 0) {\n await new Promise((resolve) => setTimeout(resolve, 50));\n }\n }\n }\n\n private next() {\n if (this._active < this.maxConcurrency && this._queue.length > 0) {\n this._queue.shift()?.();\n }\n }\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nimport type { ChildProcess } from \"child_process\";\n\nexport interface StreamWriter {\n write(buffer: Buffer): Promise<void>;\n end(): void;\n}\n\nexport function createWriter(stdin: NodeJS.WritableStream): StreamWriter {\n const drainPromise = new Map<NodeJS.WritableStream, Promise<void>>();\n\n const waitDrain = (stream: NodeJS.WritableStream) => {\n const existing = drainPromise.get(stream);\n if (existing) return existing;\n const promise = new Promise<void>((resolve, reject) => {\n const cleanup = (fn: () => void) => {\n stream.off(\"drain\", onDrain);\n stream.off(\"error\", onError);\n stream.off(\"close\", onClose);\n drainPromise.delete(stream);\n fn();\n };\n const onDrain = () => cleanup(resolve);\n const onError = (err: Error) => cleanup(() => reject(err));\n const onClose = () => cleanup(() => reject(new Error(\"stream closed\")));\n stream.on(\"drain\", onDrain);\n stream.on(\"error\", onError);\n stream.on(\"close\", onClose);\n });\n drainPromise.set(stream, promise);\n return promise;\n };\n\n const destroyed = () => Reflect.get(stdin, \"destroyed\");\n return {\n async write(buffer: Buffer) {\n if (destroyed()) {\n throw new Error(\"stdin destroyed\");\n }\n if (!stdin.write(buffer)) {\n await waitDrain(stdin);\n }\n },\n end: () => {\n if (!destroyed()) stdin.end();\n },\n };\n}\n\nexport function pipeline(...procs: ChildProcess[]) {\n for (let i = 0; i < procs.length - 1; i++) {\n const src = procs[i];\n const dst = procs[i + 1];\n if (!src?.stdout || !dst?.stdin) throw new Error(\"pipeline broken\");\n src.stdout.pipe(dst.stdin);\n }\n}\n\nexport function waitAll(...procs: ChildProcess[]) {\n return Promise.all(\n procs.map(\n (proc) =>\n new Promise<void>((resolve, reject) => {\n proc.on(\"error\", reject);\n proc.on(\"close\", (code) =>\n code === 0\n ? resolve()\n : reject(new Error(`exit ${code ?? \"null\"}`)),\n );\n }),\n ),\n );\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/01/30.\n\nimport { makeCLI } from \"./common\";\nimport { pup } from \"./pup\";\n\nmakeCLI(\"pup\", pup);\n"],
5
- "mappings": "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",
4
+ "sourcesContent": ["// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nimport { program } from \"commander\";\nimport { pupUseInnerProxy } from \"./base/constants\";\nimport { logger } from \"./base/logging\";\nimport { noerr } from \"./base/noerr\";\nimport { parseNumber } from \"./base/parser\";\nimport { pargs } from \"./base/process\";\nimport type { RecordOptions } from \"./base/record\";\n\nexport const DEFAULT_WIDTH = 1920;\nexport const DEFAULT_HEIGHT = 1080;\nexport const DEFAULT_FPS = 30;\nexport const DEFAULT_DURATION = 5;\nexport const DEFAULT_OUT_DIR = \"out\";\n\nexport type CLICallback = (\n source: string,\n options: RecordOptions & Record<string, unknown>,\n) => Promise<unknown>;\n\nexport function makeCLI(name: string, callback: CLICallback) {\n program\n .name(name)\n .argument(\"<source>\", \"file://, http(s)://, \u6216 data: URI\")\n .option(\"-w, --width <number>\", \"\u89C6\u9891\u5BBD\u5EA6\", `${DEFAULT_WIDTH}`)\n .option(\"-h, --height <number>\", \"\u89C6\u9891\u9AD8\u5EA6\", `${DEFAULT_HEIGHT}`)\n .option(\"-f, --fps <number>\", \"\u5E27\u7387\", `${DEFAULT_FPS}`)\n .option(\"-t, --duration <number>\", \"\u5F55\u5236\u65F6\u957F\uFF08\u79D2\uFF09\", `${DEFAULT_DURATION}`)\n .option(\"-o, --out-dir <path>\", \"\u8F93\u51FA\u76EE\u5F55\", `${DEFAULT_OUT_DIR}`)\n .option(\"-a, --with-alpha-channel\", \"\u8F93\u51FA\u5305\u542B alpha \u901A\u9053\u7684\u89C6\u9891\", false)\n .option(\n \"--use-inner-proxy\",\n \"\u4F7F\u7528 B \u7AD9\u5185\u7F51\u4EE3\u7406\u52A0\u901F\u8D44\u6E90\u8BBF\u95EE\",\n pupUseInnerProxy,\n )\n .action(async (source: string, opts) => {\n try {\n await callback(source, {\n width: noerr(parseNumber, DEFAULT_WIDTH)(opts.width),\n height: noerr(parseNumber, DEFAULT_HEIGHT)(opts.height),\n fps: noerr(parseNumber, DEFAULT_FPS)(opts.fps),\n duration: noerr(parseNumber, DEFAULT_DURATION)(opts.duration),\n outDir: opts.outDir ?? DEFAULT_OUT_DIR,\n withAlphaChannel: opts.withAlphaChannel ?? false,\n useInnerProxy: opts.useInnerProxy ?? pupUseInnerProxy,\n });\n } catch (e) {\n logger.fatal(e);\n }\n });\n program.parse(pargs());\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nimport { existsSync } from \"fs\";\nimport { resolve } from \"path\";\nimport { penv } from \"./env\";\nimport { parseNumber } from \"./parser\";\n\nconst pupAppSearchPaths = [\n resolve(__dirname, \"cjs/app.cjs\"), // process from dist\n resolve(__dirname, \"app.cjs\"), // process from dist/cjs\n resolve(__dirname, \"../../cjs/app.cjs\"), // process from src\n];\nexport const pupAppPath = pupAppSearchPaths.find(existsSync)!;\n\nconst env = process.env;\nexport const pupLogLevel = penv(\"PUP_LOG_LEVEL\", parseNumber, 2);\nexport const pupUseInnerProxy = env[\"PUP_USE_INNER_PROXY\"] === \"1\";\nexport const pupFFmpegPath = env[\"FFMPEG_BIN\"] ?? `ffmpeg`;\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/25.\n\nexport type EnvParser<T> = (value: unknown) => T;\n\nexport function penv<T>(name: string, parser: EnvParser<T>, defaultValue: T): T;\nexport function penv<T>(\n name: string,\n parser: EnvParser<T>,\n defaultValue?: T,\n): T | undefined;\nexport function penv<T>(\n name: string,\n parser: EnvParser<T>,\n defaultValue?: T,\n): T | undefined {\n try {\n return parser(process.env[name]);\n } catch {\n return defaultValue;\n }\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/01/30.\n\nexport function parseNumber(value: unknown): number {\n if (typeof value === \"number\") {\n return value;\n }\n const num = Number(value);\n if (Number.isNaN(num)) {\n throw new Error(`Value ${value} is not a valid number`);\n }\n return num;\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/06.\n\nimport { ChildProcess, type Serializable } from \"child_process\";\nimport { pupLogLevel } from \"./constants\";\n\nexport interface LoggerLike {\n debug?(this: void, ...messages: unknown[]): void;\n\n info?(this: void, ...messages: unknown[]): void;\n\n warn?(this: void, ...messages: unknown[]): void;\n\n error?(this: void, ...messages: unknown[]): void;\n}\n\nconst DEBUG = \"<pup@debug>\";\nconst INFO = \"<pup@info>\";\nconst WARN = \"<pup@warn>\";\nconst ERROR = \"<pup@error>\";\nconst FATAL = \"<pup@fatal>\";\n\nclass Logger implements LoggerLike {\n private _impl?: LoggerLike;\n\n get impl(): LoggerLike | undefined {\n return this._impl;\n }\n\n set impl(value: LoggerLike) {\n const debug = value.debug ?? console.debug;\n const info = value.info ?? console.info;\n const warn = value.warn ?? console.warn;\n const error = value.error ?? console.error;\n this._impl = {\n debug: pupLogLevel >= 3 ? debug : undefined,\n info: pupLogLevel >= 2 ? info : undefined,\n warn: pupLogLevel >= 1 ? warn : undefined,\n error: pupLogLevel >= 0 ? error : undefined,\n };\n }\n\n constructor() {\n this.impl = console;\n }\n\n debug(...messages: unknown[]): void {\n this.impl?.debug?.(DEBUG, ...messages);\n }\n\n info(...messages: unknown[]): void {\n this.impl?.info?.(INFO, ...messages);\n }\n\n warn(...messages: unknown[]): void {\n this.impl?.warn?.(WARN, ...messages);\n }\n\n error(...messages: unknown[]): void {\n this.impl?.error?.(ERROR, ...messages);\n }\n\n fatal(...messages: unknown[]): never {\n this.impl?.error?.(FATAL, ...messages);\n process.exit(1);\n }\n\n private dispatch(message: string) {\n if (message.startsWith(DEBUG)) {\n this.debug(message.slice(DEBUG.length + 1));\n } else if (message.startsWith(INFO)) {\n this.info(message.slice(INFO.length + 1));\n } else if (message.startsWith(WARN)) {\n this.warn(message.slice(WARN.length + 1));\n } else if (message.startsWith(ERROR)) {\n this.error(message.slice(ERROR.length + 1));\n } else {\n this.info(message);\n }\n }\n\n attach(proc: ChildProcess, name: string) {\n return new Promise<void>((resolve, reject) => {\n this.debug(`${name}.attach`);\n let fatal: string = \"\";\n const dispatch = (data: Buffer | Serializable) => {\n const message = data.toString();\n if (message.startsWith(FATAL)) {\n fatal += message.slice(FATAL.length + 1);\n } else {\n this.dispatch(message);\n }\n };\n proc.stderr?.on(\"data\", dispatch);\n proc.stdout?.on(\"data\", dispatch);\n proc\n .on(\"message\", dispatch)\n .on(\"error\", (err) => {\n fatal += err.message;\n proc.kill();\n })\n .once(\"close\", (code, signal) => {\n if (code || signal || fatal) {\n fatal ||= `command failed: ${proc.spawnargs.join(\" \")}`;\n this.error(`${name}.close`, { code, signal, fatal });\n reject(new Error(fatal));\n } else {\n this.debug(`${name}.close`);\n resolve();\n }\n })\n .on(\"unhandledRejection\", (reason) => {\n this.error(`${name}.unhandled`, reason);\n })\n .on(\"uncaughtExceptionMonitor\", (err) => {\n this.error(`${name}.unhandled`, err);\n });\n });\n }\n}\n\nconst logger = new Logger();\n\nexport { logger };\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/24.\n\nexport function noerr<Fn extends (...args: any[]) => any, D>(\n fn: Fn,\n defaultValue: D,\n): (...args: Parameters<Fn>) => ReturnType<Fn> | D {\n return (...args) => {\n try {\n const ret = fn(...args);\n if (ret instanceof Promise) {\n return ret.catch(() => defaultValue);\n }\n return ret;\n } catch {\n return defaultValue;\n }\n };\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/01/30.\n\nimport { spawn, type ChildProcess, type SpawnOptions } from \"child_process\";\nimport { logger } from \"./logging\";\n\nexport const PUP_ARGS_ENV_KEY = \"__PUP_ARGS__\";\n\nexport function pargs() {\n const pupArgs = process.env[PUP_ARGS_ENV_KEY];\n if (pupArgs) {\n const args = [\"exec\", ...process.argv.slice(-1)];\n args.push(...JSON.parse(pupArgs));\n logger.debug(\"pupargs\", args);\n return args;\n }\n\n logger.debug(\"procargv\", process.argv);\n return process.argv;\n}\n\nexport interface ProcessHandle {\n process: ChildProcess;\n wait: Promise<void>;\n}\n\nexport function exec(cmd: string, options?: SpawnOptions): ProcessHandle {\n const parts = cmd.split(\" \").filter((s) => s.length);\n const [command, ...args] = parts;\n if (!command) throw new Error(\"empty command\");\n const proc = spawn(command, args, {\n stdio: \"inherit\",\n ...options,\n });\n return { process: proc, wait: logger.attach(proc, command) };\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nimport { spawn } from \"child_process\";\nimport type { Size } from \"electron\";\nimport { readFile, rm } from \"fs/promises\";\nimport { join } from \"path\";\nimport { AbortLink, type AbortQuery } from \"./base/abort\";\nimport { pupAppPath } from \"./base/constants\";\nimport { runElectronApp } from \"./base/electron\";\nimport { encodeBgraFile, encodeBgraToMov } from \"./base/encoder\";\nimport { createCoverCommand } from \"./base/ffmpeg\";\nimport { ConcurrencyLimiter } from \"./base/limiter\";\nimport { logger } from \"./base/logging\";\nimport { parseNumber } from \"./base/parser\";\nimport { type ProcessHandle } from \"./base/process\";\nimport type { RecordResult } from \"./base/record\";\nimport { waitAll } from \"./base/stream\";\nimport type { VideoFilesWithCover, VideoSpec } from \"./base/types\";\nimport { DEFAULT_HEIGHT, DEFAULT_WIDTH } from \"./common\";\n\nconst TAG = \"[pup]\";\n\nexport type PupProgressCallback = (progress: number) => Promise<void> | void;\n\nexport interface PupOptions {\n withAlphaChannel?: boolean;\n width?: number;\n height?: number;\n fps?: number;\n duration?: number;\n outDir?: string;\n cancelQuery?: AbortQuery;\n onProgress?: PupProgressCallback;\n}\n\nconst PROGRESS_TAG = \" progress: \";\n\nfunction runPupApp(source: string, options: PupOptions) {\n logger.debug(TAG, `runPupApp`, source, options);\n\n const args: string[] = [source];\n if (options.width) args.push(\"--width\", `${options.width}`);\n if (options.height) args.push(\"--height\", `${options.height}`);\n if (options.fps) args.push(\"--fps\", `${options.fps}`);\n if (options.duration) args.push(\"--duration\", `${options.duration}`);\n if (options.outDir) args.push(\"--out-dir\", options.outDir);\n if (options.withAlphaChannel) args.push(\"--with-alpha-channel\");\n\n const w = options.width ?? DEFAULT_WIDTH;\n const h = options.height ?? DEFAULT_HEIGHT;\n const handle = runElectronApp({ width: w, height: h }, pupAppPath, args);\n const counter = new ConcurrencyLimiter(1);\n handle.process.stdout?.on(\"data\", (data: Buffer) => {\n let message = data.toString().trim();\n let start = message.indexOf(PROGRESS_TAG);\n if (start < 0) {\n return;\n }\n message = message.slice(start + PROGRESS_TAG.length);\n const end = message.indexOf(\"%\");\n if (end < 0) {\n return;\n }\n const progressStr = message.slice(0, end);\n const progress = parseNumber(progressStr);\n counter.schedule(async () => {\n await options.onProgress?.(progress);\n });\n });\n return { handle, counter };\n}\n\nexport async function pup(source: string, options: PupOptions) {\n logger.debug(TAG, `pup`, source, options);\n\n const link = AbortLink.start(options.cancelQuery);\n const outDir = options.outDir ?? \"out\";\n\n const t0 = performance.now();\n const { handle, counter } = runPupApp(source, { ...options, outDir });\n\n await link.wait(handle);\n await counter.end();\n logger.info(TAG, `capture cost ${Math.round(performance.now() - t0)}ms`);\n\n const metaPath = join(outDir, \"record.json\");\n const meta = JSON.parse(await readFile(metaPath, \"utf-8\")) as RecordResult;\n\n const { bgraPath, written, options: recordOptions } = meta;\n const { fps, width, height, withAlphaChannel } = recordOptions;\n const size: Size = { width, height };\n\n const outputs: VideoFilesWithCover = {\n mp4: withAlphaChannel ? undefined : join(outDir, \"output.mp4\"),\n webm: withAlphaChannel ? join(outDir, \"output.webm\") : undefined,\n mov: withAlphaChannel ? join(outDir, \"output.mov\") : undefined,\n cover: join(outDir, \"cover.png\"),\n };\n\n try {\n const t1 = performance.now();\n\n const spec: VideoSpec = { fps, frames: written, size };\n const handles: ProcessHandle[] = [];\n if (outputs.mp4) {\n handles.push(encodeBgraFile(bgraPath, outputs.mp4, spec, \"mp4\"));\n }\n if (outputs.webm) {\n handles.push(encodeBgraFile(bgraPath, outputs.webm, spec, \"webm\"));\n }\n if (outputs.mov) {\n handles.push(encodeBgraToMov(bgraPath, outputs.mov, spec));\n }\n await link.wait(...handles);\n\n const coverSrc = outputs.mov ?? outputs.webm ?? outputs.mp4;\n if (coverSrc) {\n const coverCmd = createCoverCommand(coverSrc, outputs.cover);\n await waitAll(\n spawn(coverCmd.command, coverCmd.args, { stdio: \"inherit\" }),\n );\n }\n\n link.stop();\n logger.info(TAG, `encoding cost ${Math.round(performance.now() - t1)}ms`);\n\n await Promise.all([\n rm(bgraPath, { force: true }),\n rm(metaPath, { force: true }),\n ]);\n return {\n ...outputs,\n width,\n height,\n fps,\n duration: Math.ceil(written / fps),\n };\n } catch (error) {\n await rm(outDir, { recursive: true, force: true });\n throw error;\n }\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/11.\n\nimport type { ProcessHandle } from \"./process\";\n\nexport type AsyncTask = () => Promise<void> | void;\nexport type AbortQuery = () => Promise<boolean> | boolean;\n\nexport class AbortLink {\n private _callback?: AsyncTask;\n private _aborted?: boolean;\n private _stopped = false;\n\n private constructor(\n readonly query?: AbortQuery,\n readonly interval: number = 1000,\n ) {\n if (query) {\n this.tick();\n }\n }\n\n static start(query?: AbortQuery, interval?: number) {\n return new AbortLink(query, interval);\n }\n\n get aborted() {\n return !this._stopped && this._aborted;\n }\n\n get stopped() {\n return this._stopped;\n }\n\n async onAbort(callback: AsyncTask) {\n if (this._aborted) {\n await callback();\n } else {\n this._callback = callback;\n }\n }\n\n wait(...handles: ProcessHandle[]) {\n const abort = new Promise((_, reject) => {\n this.onAbort(async () => {\n handles.forEach((h) => h.process.kill());\n reject(new Error(\"aborted\"));\n });\n });\n return Promise.race([\n abort,\n Promise.all(handles.map((h) => h.wait)), //\n ]);\n }\n\n stop() {\n this._stopped = true;\n }\n\n private tick() {\n setTimeout(async () => {\n if (this._stopped) {\n return;\n }\n this._aborted = await this.query?.();\n if (this._stopped) {\n return;\n }\n if (this._aborted) {\n await this._callback?.();\n } else {\n this.tick();\n }\n }, this.interval);\n }\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/25.\n\nimport electron, { type Size } from \"electron\";\nimport { exec } from \"./process\";\n\nexport const ELECTRON_OPTS = [\n \"no-sandbox\",\n \"disable-setuid-sandbox\",\n \"disable-gpu\",\n \"disable-dev-shm-usage\",\n \"disable-software-rasterizer\",\n \"disable-web-security\",\n \"disable-site-isolation-trials\",\n \"disable-features=IsolateOrigins,site-per-process\",\n \"allow-insecure-localhost\",\n \"ignore-certificate-errors\",\n \"disable-blink-features=AutomationControlled\",\n \"mute-audio\",\n \"disable-extensions\",\n \"disable-background-networking\",\n \"address-family=ipv4\",\n \"disable-async-dns\",\n \"force-device-scale-factor=1\",\n \"trace-warnings\",\n \"force-color-profile=srgb\",\n \"disable-color-correct-rendering\",\n \"log-level=3\",\n];\n\nexport function runElectronApp(size: Size, app: unknown, args: unknown[]) {\n const electronArgs = ELECTRON_OPTS.map((a) => `--${a}`);\n const cmdParts = [];\n if (process.platform === \"linux\") {\n cmdParts.push(\n \"xvfb-run\",\n \"--auto-servernum\",\n `--server-args='-screen 0 ${size.width}x${size.height}x24'`,\n );\n }\n cmdParts.push(electron, ...electronArgs, app);\n return exec(cmdParts.join(\" \"), {\n stdio: [\"ignore\", \"pipe\", \"pipe\"],\n shell: true,\n env: {\n ...process.env,\n ELECTRON_DISABLE_DBUS: \"1\",\n RUST_BACKTRACE: \"full\",\n __PUP_ARGS__: JSON.stringify(args),\n },\n });\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nimport { existsSync } from \"fs\";\nimport { resolve } from \"path\";\nimport { arch, platform } from \"process\";\nimport { pupFFmpegPath } from \"./constants\";\nimport type { VideoFiles, VideoSpec } from \"./types\";\n\ninterface Command {\n command: string;\n args: string[];\n}\n\nconst quiet = [\"-hide_banner\", \"-loglevel\", \"error\", \"-nostats\"];\n\nfunction resolveX265() {\n const path = `x265/${platform}-${arch}`;\n const dirs = [\n resolve(__dirname, `../../${path}`), // process from src\n resolve(__dirname, `../${path}`), // process from dist\n ];\n const found = dirs.find(existsSync);\n if (!found) {\n throw new Error(\"x265 not found\");\n }\n return found;\n}\n\nexport function createBgraFileCommand(\n bgraPath: string,\n spec: VideoSpec,\n files: VideoFiles,\n): Command {\n const { fps, frames } = spec;\n const args = [\n \"-y\",\n ...quiet,\n \"-f\",\n \"rawvideo\",\n \"-pix_fmt\",\n \"bgra\",\n \"-s\",\n `${spec.size.width}x${spec.size.height}`,\n \"-r\",\n `${fps}`,\n \"-i\",\n bgraPath,\n \"-frames:v\",\n `${Math.floor(frames)}`,\n ];\n if (files.mp4) {\n args.push(\n \"-colorspace\",\n \"bt709\",\n \"-color_primaries\",\n \"bt709\",\n \"-color_trc\",\n \"bt709\",\n \"-c:v\",\n \"libx264\",\n \"-preset\",\n \"fast\",\n \"-threads\",\n \"0\",\n \"-pix_fmt\",\n \"yuv420p\",\n \"-movflags\",\n \"+faststart\",\n files.mp4,\n );\n }\n if (files.webm) {\n args.push(\n \"-c:v\",\n \"libvpx-vp9\",\n \"-row-mt\",\n \"1\",\n \"-cpu-used\",\n \"1\",\n \"-threads\",\n \"0\",\n \"-pix_fmt\",\n \"yuva420p\",\n \"-auto-alt-ref\",\n \"0\",\n files.webm,\n );\n }\n return { command: pupFFmpegPath, args };\n}\n\ninterface X265Pipeline {\n raw: Command;\n x265: Command;\n mux: Command;\n}\n\nexport function createBgraToMovPipeline(\n bgraPath: string,\n spec: VideoSpec,\n mov: string,\n): X265Pipeline {\n const { fps, size } = spec;\n return {\n raw: {\n command: pupFFmpegPath,\n args: [\n \"-y\",\n ...quiet,\n \"-f\",\n \"rawvideo\",\n \"-pix_fmt\",\n \"bgra\",\n \"-s\",\n `${size.width}x${size.height}`,\n \"-r\",\n `${fps}`,\n \"-i\",\n bgraPath,\n \"-f\",\n \"rawvideo\",\n \"-pix_fmt\",\n \"yuva420p10le\",\n \"pipe:1\",\n ],\n },\n x265: {\n command: resolveX265(),\n args: [\n \"--no-progress\",\n \"--log-level\",\n \"error\",\n \"--input\",\n \"-\",\n \"--input-res\",\n `${size.width}x${size.height}`,\n \"--input-csp\",\n \"i420\",\n \"--input-depth\",\n \"10\",\n \"--output-depth\",\n \"10\",\n \"--fps\",\n `${fps}`,\n \"--alpha\",\n \"--bframes\",\n \"0\",\n \"--colorprim\",\n \"bt709\",\n \"--transfer\",\n \"bt709\",\n \"--colormatrix\",\n \"bt709\",\n \"--crf\",\n \"18\",\n \"--output\",\n \"-\",\n ],\n },\n mux: {\n command: pupFFmpegPath,\n args: [\n \"-y\",\n ...quiet,\n \"-f\",\n \"hevc\",\n \"-r\",\n `${fps}`,\n \"-i\",\n \"pipe:0\",\n \"-c:v\",\n \"copy\",\n \"-tag:v\",\n \"hvc1\",\n \"-movflags\",\n \"+faststart\",\n mov,\n ],\n },\n };\n}\n\nexport function createCoverCommand(src: string, dst: string): Command {\n return {\n command: pupFFmpegPath,\n args: [\"-y\", ...quiet, \"-i\", src, \"-frames:v\", \"1\", \"-q:v\", \"2\", dst],\n };\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nimport { createBgraFileCommand, createBgraToMovPipeline } from \"./ffmpeg\";\nimport { exec, type ProcessHandle } from \"./process\";\nimport type { VideoSpec } from \"./types\";\n\nexport function encodeBgraFile(\n bgraPath: string,\n outputPath: string,\n spec: VideoSpec,\n format: \"mp4\" | \"webm\",\n): ProcessHandle {\n const files = format === \"mp4\" ? { mp4: outputPath } : { webm: outputPath };\n const cmd = createBgraFileCommand(bgraPath, spec, files);\n return exec(`${cmd.command} ${cmd.args.join(\" \")}`, {\n stdio: [\"ignore\", \"inherit\", \"inherit\"],\n });\n}\n\nexport function encodeBgraToMov(\n bgraPath: string,\n movPath: string,\n spec: VideoSpec,\n): ProcessHandle {\n const x265 = createBgraToMovPipeline(bgraPath, spec, movPath);\n const shellCmd = [\n `${x265.raw.command} ${x265.raw.args.join(\" \")}`,\n `${x265.x265.command} ${x265.x265.args.join(\" \")}`,\n `${x265.mux.command} ${x265.mux.args.join(\" \")}`,\n ].join(\" | \");\n return exec(shellCmd, {\n stdio: [\"ignore\", \"inherit\", \"inherit\"],\n shell: true,\n });\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/01/30.\n\nexport class ConcurrencyLimiter {\n private _active = 0;\n private _queue: VoidFunction[] = [];\n private _pending = 0;\n private _ended = false;\n\n constructor(readonly maxConcurrency: number) {}\n\n get active(): number {\n return this._active;\n }\n\n get pending(): number {\n return this._pending;\n }\n\n async schedule<T>(fn: () => Promise<T>): Promise<T> {\n if (this._ended) {\n throw new Error(\"ended\");\n }\n return new Promise<T>((resolve, reject) => {\n const run = () => {\n this._active++;\n this._pending--;\n fn()\n .then(resolve)\n .catch(reject)\n .finally(() => {\n this._active--;\n this.next();\n });\n };\n this._pending++;\n if (this._active < this.maxConcurrency) {\n run();\n } else {\n this._queue.push(run);\n }\n });\n }\n\n async end() {\n if (!this._ended) {\n this._ended = true;\n while (this._active > 0 || this._pending > 0) {\n await new Promise((resolve) => setTimeout(resolve, 50));\n }\n }\n }\n\n private next() {\n if (this._active < this.maxConcurrency && this._queue.length > 0) {\n this._queue.shift()?.();\n }\n }\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nimport type { ChildProcess } from \"child_process\";\n\nexport interface StreamWriter {\n write(buffer: Buffer): Promise<void>;\n end(): void;\n}\n\nexport function createWriter(stdin: NodeJS.WritableStream): StreamWriter {\n const drainPromise = new Map<NodeJS.WritableStream, Promise<void>>();\n\n const waitDrain = (stream: NodeJS.WritableStream) => {\n const existing = drainPromise.get(stream);\n if (existing) return existing;\n const promise = new Promise<void>((resolve, reject) => {\n const cleanup = (fn: () => void) => {\n stream.off(\"drain\", onDrain);\n stream.off(\"error\", onError);\n stream.off(\"close\", onClose);\n drainPromise.delete(stream);\n fn();\n };\n const onDrain = () => cleanup(resolve);\n const onError = (err: Error) => cleanup(() => reject(err));\n const onClose = () => cleanup(() => reject(new Error(\"stream closed\")));\n stream.on(\"drain\", onDrain);\n stream.on(\"error\", onError);\n stream.on(\"close\", onClose);\n });\n drainPromise.set(stream, promise);\n return promise;\n };\n\n const destroyed = () => Reflect.get(stdin, \"destroyed\");\n return {\n async write(buffer: Buffer) {\n if (destroyed()) {\n throw new Error(\"stdin destroyed\");\n }\n if (!stdin.write(buffer)) {\n await waitDrain(stdin);\n }\n },\n end: () => {\n if (!destroyed()) stdin.end();\n },\n };\n}\n\nexport function pipeline(...procs: ChildProcess[]) {\n for (let i = 0; i < procs.length - 1; i++) {\n const src = procs[i];\n const dst = procs[i + 1];\n if (!src?.stdout || !dst?.stdin) throw new Error(\"pipeline broken\");\n src.stdout.pipe(dst.stdin);\n }\n}\n\nexport function waitAll(...procs: ChildProcess[]) {\n return Promise.all(\n procs.map(\n (proc) =>\n new Promise<void>((resolve, reject) => {\n proc.on(\"error\", reject);\n proc.on(\"close\", (code) =>\n code === 0\n ? resolve()\n : reject(new Error(`exit ${code ?? \"null\"}`)),\n );\n }),\n ),\n );\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/01/30.\n\nimport { makeCLI } from \"./common\";\nimport { pup } from \"./pup\";\n\nmakeCLI(\"pup\", pup);\n"],
5
+ "mappings": "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",
6
6
  "names": ["resolve", "spawn", "existsSync", "resolve", "resolve", "existsSync", "resolve", "resolve", "spawn"]
7
7
  }
package/dist/index.js.map CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "version": 3,
3
3
  "sources": ["../src/base/constants.ts", "../src/base/env.ts", "../src/base/parser.ts", "../src/base/lazy.ts", "../src/base/limiter.ts", "../src/base/logging.ts", "../src/base/noerr.ts", "../src/base/process.ts", "../src/base/retry.ts", "../src/base/timing.ts", "../src/pup.ts", "../src/base/abort.ts", "../src/base/electron.ts", "../src/base/ffmpeg.ts", "../src/base/encoder.ts", "../src/base/stream.ts", "../src/common.ts"],
4
- "sourcesContent": ["// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nimport { existsSync } from \"fs\";\nimport { resolve } from \"path\";\nimport { penv } from \"./env\";\nimport { parseNumber } from \"./parser\";\n\nconst pupAppSearchPaths = [\n resolve(__dirname, \"cjs/app.cjs\"), // process from dist\n resolve(__dirname, \"app.cjs\"), // process from dist/cjs\n resolve(__dirname, \"../../cjs/app.cjs\"), // process from src\n];\nexport const pupAppPath = pupAppSearchPaths.find(existsSync)!;\n\nconst env = process.env;\nexport const pupLogLevel = penv(\"PUP_LOG_LEVEL\", parseNumber, 2);\nexport const pupUseInnerProxy = env[\"PUP_USE_INNER_PROXY\"] === \"1\";\nexport const pupFFmpegPath = env[\"FFMPEG_BIN\"] ?? `ffmpeg`;\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/25.\n\nexport type EnvParser<T> = (value: unknown) => T;\n\nexport function penv<T>(name: string, parser: EnvParser<T>, defaultValue: T): T;\nexport function penv<T>(\n name: string,\n parser: EnvParser<T>,\n defaultValue?: T,\n): T | undefined;\nexport function penv<T>(\n name: string,\n parser: EnvParser<T>,\n defaultValue?: T,\n): T | undefined {\n try {\n return parser(process.env[name]);\n } catch {\n return defaultValue;\n }\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/01/30.\n\nexport function parseNumber(value: unknown): number {\n if (typeof value === \"number\") {\n return value;\n }\n const num = Number(value);\n if (Number.isNaN(num)) {\n throw new Error(`Value ${value} is not a valid number`);\n }\n return num;\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/01/30.\n\nexport class Lazy<T> {\n constructor(readonly makeValue: () => T) {}\n\n get value(): T {\n if (!this._initialized) {\n this._value = this.makeValue();\n this._initialized = true;\n }\n return this._value!;\n }\n\n get initialized(): boolean {\n return this._initialized;\n }\n\n private _initialized = false;\n private _value: T | undefined;\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/01/30.\n\nexport class ConcurrencyLimiter {\n private _active = 0;\n private _queue: VoidFunction[] = [];\n private _pending = 0;\n private _ended = false;\n\n constructor(readonly maxConcurrency: number) {}\n\n get active(): number {\n return this._active;\n }\n\n get pending(): number {\n return this._pending;\n }\n\n async schedule<T>(fn: () => Promise<T>): Promise<T> {\n if (this._ended) {\n throw new Error(\"ended\");\n }\n return new Promise<T>((resolve, reject) => {\n const run = () => {\n this._active++;\n this._pending--;\n fn()\n .then(resolve)\n .catch(reject)\n .finally(() => {\n this._active--;\n this.next();\n });\n };\n this._pending++;\n if (this._active < this.maxConcurrency) {\n run();\n } else {\n this._queue.push(run);\n }\n });\n }\n\n async end() {\n if (!this._ended) {\n this._ended = true;\n while (this._active > 0 || this._pending > 0) {\n await new Promise((resolve) => setTimeout(resolve, 50));\n }\n }\n }\n\n private next() {\n if (this._active < this.maxConcurrency && this._queue.length > 0) {\n this._queue.shift()?.();\n }\n }\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/06.\n\nimport { ChildProcess, type Serializable } from \"child_process\";\nimport { pupLogLevel } from \"./constants\";\n\nexport interface LoggerLike {\n debug?(this: void, ...messages: unknown[]): void;\n\n info?(this: void, ...messages: unknown[]): void;\n\n warn?(this: void, ...messages: unknown[]): void;\n\n error?(this: void, ...messages: unknown[]): void;\n}\n\nconst DEBUG = \"<pup@debug>\";\nconst INFO = \"<pup@info>\";\nconst WARN = \"<pup@warn>\";\nconst ERROR = \"<pup@error>\";\nconst FATAL = \"<pup@fatal>\";\n\nclass Logger implements LoggerLike {\n private _impl?: LoggerLike;\n\n get impl(): LoggerLike | undefined {\n return this._impl;\n }\n\n set impl(value: LoggerLike) {\n const debug = value.debug ?? console.debug;\n const info = value.info ?? console.info;\n const warn = value.warn ?? console.warn;\n const error = value.error ?? console.error;\n this._impl = {\n debug: pupLogLevel >= 3 ? debug : undefined,\n info: pupLogLevel >= 2 ? info : undefined,\n warn: pupLogLevel >= 1 ? warn : undefined,\n error: pupLogLevel >= 0 ? error : undefined,\n };\n }\n\n constructor() {\n this.impl = console;\n }\n\n debug(...messages: unknown[]): void {\n this.impl?.debug?.(DEBUG, ...messages);\n }\n\n info(...messages: unknown[]): void {\n this.impl?.info?.(INFO, ...messages);\n }\n\n warn(...messages: unknown[]): void {\n this.impl?.warn?.(WARN, ...messages);\n }\n\n error(...messages: unknown[]): void {\n this.impl?.error?.(ERROR, ...messages);\n }\n\n fatal(...messages: unknown[]): never {\n this.impl?.error?.(FATAL, ...messages);\n process.exit(1);\n }\n\n private dispatch(message: string) {\n if (message.startsWith(DEBUG)) {\n this.debug(message.slice(DEBUG.length + 1));\n } else if (message.startsWith(INFO)) {\n this.info(message.slice(INFO.length + 1));\n } else if (message.startsWith(WARN)) {\n this.warn(message.slice(WARN.length + 1));\n } else if (message.startsWith(ERROR)) {\n this.error(message.slice(ERROR.length + 1));\n } else {\n this.info(message);\n }\n }\n\n attach(proc: ChildProcess, name: string) {\n return new Promise<void>((resolve, reject) => {\n this.debug(`${name}.attach`);\n let fatal: string = \"\";\n const dispatch = (data: Buffer | Serializable) => {\n const message = data.toString();\n if (message.startsWith(FATAL)) {\n fatal += message.slice(FATAL.length + 1);\n } else {\n this.dispatch(message);\n }\n };\n proc.stderr?.on(\"data\", dispatch);\n proc.stdout?.on(\"data\", dispatch);\n proc\n .on(\"message\", dispatch)\n .on(\"error\", (err) => {\n fatal += err.message;\n proc.kill();\n })\n .once(\"close\", (code, signal) => {\n if (code || signal || fatal) {\n fatal ||= `command failed: ${proc.spawnargs.join(\" \")}`;\n this.error(`${name}.close`, { code, signal, fatal });\n reject(new Error(fatal));\n } else {\n this.debug(`${name}.close`);\n resolve();\n }\n })\n .on(\"unhandledRejection\", (reason) => {\n this.error(`${name}.unhandled`, reason);\n })\n .on(\"uncaughtExceptionMonitor\", (err) => {\n this.error(`${name}.unhandled`, err);\n });\n });\n }\n}\n\nconst logger = new Logger();\n\nexport { logger };\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/24.\n\nexport function noerr<Fn extends (...args: any[]) => any, D>(\n fn: Fn,\n defaultValue: D,\n): (...args: Parameters<Fn>) => ReturnType<Fn> | D {\n return (...args) => {\n try {\n const ret = fn(...args);\n if (ret instanceof Promise) {\n return ret.catch(() => defaultValue);\n }\n return ret;\n } catch {\n return defaultValue;\n }\n };\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/01/30.\n\nimport { spawn, type ChildProcess, type SpawnOptions } from \"child_process\";\nimport { logger } from \"./logging\";\n\nexport const PUP_ARGS_ENV_KEY = \"__PUP_ARGS__\";\n\nexport function pargs() {\n const pupArgs = process.env[PUP_ARGS_ENV_KEY];\n if (pupArgs) {\n const args = [\"exec\", ...process.argv.slice(-1)];\n args.push(...JSON.parse(pupArgs));\n logger.debug(\"pupargs\", args);\n return args;\n }\n\n logger.debug(\"procargv\", process.argv);\n return process.argv;\n}\n\nexport interface ProcessHandle {\n process: ChildProcess;\n wait: Promise<void>;\n}\n\nexport function exec(cmd: string, options?: SpawnOptions): ProcessHandle {\n const parts = cmd.split(\" \").filter((s) => s.length);\n const [command, ...args] = parts;\n if (!command) throw new Error(\"empty command\");\n const proc = spawn(command, args, {\n stdio: \"inherit\",\n ...options,\n });\n return { process: proc, wait: logger.attach(proc, command) };\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/05.\n\nimport { setTimeout } from \"timers/promises\";\nimport { sleep } from \"./timing\";\n\nexport interface RetryOptions<Args extends any[], Ret> {\n fn: (...args: Args) => Promise<Ret>;\n maxAttempts?: number;\n timeout?: number;\n}\n\nexport function useRetry<Args extends any[], Ret>({\n fn,\n maxAttempts = 3,\n timeout,\n}: RetryOptions<Args, Ret>) {\n const timeoutError = new Error(`timeout over ${timeout}ms`);\n return async function (...args: Args) {\n let attempt = 0;\n while (true) {\n try {\n const promises = [fn(...args)];\n if (timeout) {\n promises.push(\n setTimeout(timeout).then(() => {\n throw timeoutError;\n }),\n );\n }\n return await Promise.race(promises);\n } catch (e) {\n attempt++;\n if (attempt >= maxAttempts) {\n throw e;\n }\n await sleep(Math.pow(2, attempt) * 100 + Math.random() * 100);\n }\n }\n };\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nexport function sleep(ms: number) {\n return new Promise<void>((resolve) => setTimeout(resolve, ms));\n}\n\nexport function periodical(\n callback: (count: number) => Promise<void> | void,\n ms: number,\n) {\n let token: NodeJS.Timeout;\n let closed = false;\n async function tick(count: number) {\n await callback(count);\n if (closed) return;\n token = setTimeout(() => tick(count + 1), ms);\n }\n token = setTimeout(() => tick(0), ms);\n return () => {\n closed = true;\n clearTimeout(token);\n };\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nimport { spawn } from \"child_process\";\nimport type { Size } from \"electron\";\nimport { readFile, rm } from \"fs/promises\";\nimport { join } from \"path\";\nimport { AbortLink, type AbortQuery } from \"./base/abort\";\nimport { pupAppPath } from \"./base/constants\";\nimport { runElectronApp } from \"./base/electron\";\nimport { encodeBgraFile, encodeBgraToMov } from \"./base/encoder\";\nimport { createCoverCommand } from \"./base/ffmpeg\";\nimport { ConcurrencyLimiter } from \"./base/limiter\";\nimport { logger } from \"./base/logging\";\nimport { parseNumber } from \"./base/parser\";\nimport { type ProcessHandle } from \"./base/process\";\nimport type { RecordResult } from \"./base/record\";\nimport { waitAll } from \"./base/stream\";\nimport type { VideoFilesWithCover, VideoSpec } from \"./base/types\";\nimport { DEFAULT_HEIGHT, DEFAULT_WIDTH } from \"./common\";\n\nconst TAG = \"[pup]\";\n\nexport type PupProgressCallback = (progress: number) => Promise<void> | void;\n\nexport interface PupOptions {\n withAlphaChannel?: boolean;\n width?: number;\n height?: number;\n fps?: number;\n duration?: number;\n outDir?: string;\n cancelQuery?: AbortQuery;\n onProgress?: PupProgressCallback;\n}\n\nconst PROGRESS_TAG = \" progress: \";\n\nfunction runPupApp(source: string, options: PupOptions) {\n logger.debug(TAG, `runPupApp`, source, options);\n\n const args: string[] = [source];\n if (options.width) args.push(\"--width\", `${options.width}`);\n if (options.height) args.push(\"--height\", `${options.height}`);\n if (options.fps) args.push(\"--fps\", `${options.fps}`);\n if (options.duration) args.push(\"--duration\", `${options.duration}`);\n if (options.outDir) args.push(\"--out-dir\", options.outDir);\n if (options.withAlphaChannel) args.push(\"--with-alpha-channel\");\n\n const w = options.width ?? DEFAULT_WIDTH;\n const h = options.height ?? DEFAULT_HEIGHT;\n const handle = runElectronApp({ width: w, height: h }, pupAppPath, args);\n const counter = new ConcurrencyLimiter(1);\n handle.process.stdout?.on(\"data\", (data: Buffer) => {\n let message = data.toString().trim();\n let start = message.indexOf(PROGRESS_TAG);\n if (start < 0) {\n return;\n }\n message = message.slice(start + PROGRESS_TAG.length);\n const end = message.indexOf(\"%\");\n if (end < 0) {\n return;\n }\n const progressStr = message.slice(0, end);\n const progress = parseNumber(progressStr);\n counter.schedule(async () => {\n await options.onProgress?.(progress);\n });\n });\n return { handle, counter };\n}\n\nexport async function pup(source: string, options: PupOptions) {\n logger.debug(TAG, `pup`, source, options);\n\n const link = AbortLink.start(options.cancelQuery);\n const outDir = options.outDir ?? \"out\";\n\n const t0 = performance.now();\n const { handle, counter } = runPupApp(source, { ...options, outDir });\n\n await link.wait(handle);\n await counter.end();\n logger.info(TAG, `capture cost ${Math.round(performance.now() - t0)}ms`);\n\n const metaPath = join(outDir, \"record.json\");\n const meta = JSON.parse(await readFile(metaPath, \"utf-8\")) as RecordResult;\n\n const { bgraPath, written, options: recordOptions } = meta;\n const { fps, width, height, withAlphaChannel } = recordOptions;\n const size: Size = { width, height };\n\n const outputs: VideoFilesWithCover = {\n mp4: withAlphaChannel ? undefined : join(outDir, \"output.mp4\"),\n webm: withAlphaChannel ? join(outDir, \"output.webm\") : undefined,\n mov: withAlphaChannel ? join(outDir, \"output.mov\") : undefined,\n cover: join(outDir, \"cover.png\"),\n };\n\n try {\n const t1 = performance.now();\n\n const spec: VideoSpec = { fps, frames: written, size };\n const handles: ProcessHandle[] = [];\n if (outputs.mp4) {\n handles.push(encodeBgraFile(bgraPath, outputs.mp4, spec, \"mp4\"));\n }\n if (outputs.webm) {\n handles.push(encodeBgraFile(bgraPath, outputs.webm, spec, \"webm\"));\n }\n if (outputs.mov) {\n handles.push(encodeBgraToMov(bgraPath, outputs.mov, spec));\n }\n await link.wait(...handles);\n\n const coverSrc = outputs.mov ?? outputs.webm ?? outputs.mp4;\n if (coverSrc) {\n const coverCmd = createCoverCommand(coverSrc, outputs.cover);\n await waitAll(\n spawn(coverCmd.command, coverCmd.args, { stdio: \"inherit\" }),\n );\n }\n\n link.stop();\n logger.info(TAG, `encoding cost ${Math.round(performance.now() - t1)}ms`);\n\n await Promise.all([\n rm(bgraPath, { force: true }),\n rm(metaPath, { force: true }),\n ]);\n return {\n ...outputs,\n width,\n height,\n fps,\n duration: Math.ceil(written / fps),\n };\n } catch (error) {\n await rm(outDir, { recursive: true, force: true });\n throw error;\n }\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/11.\n\nimport type { ProcessHandle } from \"./process\";\n\nexport type AsyncTask = () => Promise<void> | void;\nexport type AbortQuery = () => Promise<boolean> | boolean;\n\nexport class AbortLink {\n private _callback?: AsyncTask;\n private _aborted?: boolean;\n private _stopped = false;\n\n private constructor(\n readonly query?: AbortQuery,\n readonly interval: number = 1000,\n ) {\n if (query) {\n this.tick();\n }\n }\n\n static start(query?: AbortQuery, interval?: number) {\n return new AbortLink(query, interval);\n }\n\n get aborted() {\n return !this._stopped && this._aborted;\n }\n\n get stopped() {\n return this._stopped;\n }\n\n async onAbort(callback: AsyncTask) {\n if (this._aborted) {\n await callback();\n } else {\n this._callback = callback;\n }\n }\n\n wait(...handles: ProcessHandle[]) {\n const abort = new Promise((_, reject) => {\n this.onAbort(async () => {\n handles.forEach((h) => h.process.kill());\n reject(new Error(\"aborted\"));\n });\n });\n return Promise.race([\n abort,\n Promise.all(handles.map((h) => h.wait)), //\n ]);\n }\n\n stop() {\n this._stopped = true;\n }\n\n private tick() {\n setTimeout(async () => {\n if (this._stopped) {\n return;\n }\n this._aborted = await this.query?.();\n if (this._stopped) {\n return;\n }\n if (this._aborted) {\n await this._callback?.();\n } else {\n this.tick();\n }\n }, this.interval);\n }\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/25.\n\nimport electron, { type Size } from \"electron\";\nimport { exec } from \"./process\";\n\nexport const ELECTRON_OPTS = [\n \"no-sandbox\",\n \"disable-setuid-sandbox\",\n \"disable-gpu\",\n \"disable-dev-shm-usage\",\n \"disable-software-rasterizer\",\n \"disable-web-security\",\n \"disable-site-isolation-trials\",\n \"disable-features=IsolateOrigins,site-per-process\",\n \"allow-insecure-localhost\",\n \"ignore-certificate-errors\",\n \"disable-blink-features=AutomationControlled\",\n \"mute-audio\",\n \"disable-extensions\",\n \"disable-background-networking\",\n \"address-family=ipv4\",\n \"disable-async-dns\",\n \"force-device-scale-factor=1\",\n \"trace-warnings\",\n \"force-color-profile=srgb\",\n \"disable-color-correct-rendering\",\n \"log-level=3\",\n];\n\nexport function runElectronApp(size: Size, app: unknown, args: unknown[]) {\n const electronArgs = ELECTRON_OPTS.map((a) => `--${a}`);\n const cmdParts = [];\n if (process.platform === \"linux\") {\n cmdParts.push(\n \"xvfb-run\",\n \"--auto-servernum\",\n `--server-args='-screen 0 ${size.width}x${size.height}x24'`,\n );\n }\n cmdParts.push(electron, ...electronArgs, app);\n return exec(cmdParts.join(\" \"), {\n stdio: [\"ignore\", \"pipe\", \"pipe\"],\n shell: true,\n env: {\n ...process.env,\n ELECTRON_DISABLE_DBUS: \"1\",\n RUST_BACKTRACE: \"full\",\n __PUP_ARGS__: JSON.stringify(args),\n },\n });\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nimport { existsSync } from \"fs\";\nimport { resolve } from \"path\";\nimport { arch, platform } from \"process\";\nimport { pupFFmpegPath } from \"./constants\";\nimport type { VideoFiles, VideoSpec } from \"./types\";\n\ninterface Command {\n command: string;\n args: string[];\n}\n\nconst quiet = [\"-hide_banner\", \"-loglevel\", \"error\", \"-nostats\"];\n\nfunction resolveX265() {\n const path = `x265/${platform}-${arch}`;\n const dirs = [\n resolve(__dirname, `../../${path}`), // process from src\n resolve(__dirname, `../${path}`), // process from dist\n ];\n const found = dirs.find(existsSync);\n if (!found) {\n throw new Error(\"x265 not found\");\n }\n return found;\n}\n\nexport function createBgraFileCommand(\n bgraPath: string,\n spec: VideoSpec,\n files: VideoFiles,\n): Command {\n const { fps, frames } = spec;\n const args = [\n \"-y\",\n ...quiet,\n \"-f\",\n \"rawvideo\",\n \"-pix_fmt\",\n \"bgra\",\n \"-s\",\n `${spec.size.width}x${spec.size.height}`,\n \"-r\",\n `${fps}`,\n \"-i\",\n bgraPath,\n \"-frames:v\",\n `${Math.floor(frames)}`,\n ];\n if (files.mp4) {\n args.push(\n \"-colorspace\",\n \"bt709\",\n \"-color_primaries\",\n \"bt709\",\n \"-color_trc\",\n \"bt709\",\n \"-c:v\",\n \"libx264\",\n \"-preset\",\n \"fast\",\n \"-threads\",\n \"0\",\n \"-pix_fmt\",\n \"yuv420p\",\n \"-movflags\",\n \"+faststart\",\n files.mp4,\n );\n }\n if (files.webm) {\n args.push(\n \"-c:v\",\n \"libvpx-vp9\",\n \"-row-mt\",\n \"1\",\n \"-cpu-used\",\n \"1\",\n \"-threads\",\n \"0\",\n \"-pix_fmt\",\n \"yuva420p\",\n \"-auto-alt-ref\",\n \"0\",\n files.webm,\n );\n }\n return { command: pupFFmpegPath, args };\n}\n\ninterface X265Pipeline {\n raw: Command;\n x265: Command;\n mux: Command;\n}\n\nexport function createBgraToMovPipeline(\n bgraPath: string,\n spec: VideoSpec,\n mov: string,\n): X265Pipeline {\n const { fps, size } = spec;\n return {\n raw: {\n command: pupFFmpegPath,\n args: [\n \"-y\",\n ...quiet,\n \"-f\",\n \"rawvideo\",\n \"-pix_fmt\",\n \"bgra\",\n \"-s\",\n `${size.width}x${size.height}`,\n \"-r\",\n `${fps}`,\n \"-i\",\n bgraPath,\n \"-f\",\n \"rawvideo\",\n \"-pix_fmt\",\n \"yuva420p10le\",\n \"pipe:1\",\n ],\n },\n x265: {\n command: resolveX265(),\n args: [\n \"--no-progress\",\n \"--log-level\",\n \"error\",\n \"--input\",\n \"-\",\n \"--input-res\",\n `${size.width}x${size.height}`,\n \"--input-csp\",\n \"i420\",\n \"--input-depth\",\n \"10\",\n \"--output-depth\",\n \"10\",\n \"--fps\",\n `${fps}`,\n \"--alpha\",\n \"--bframes\",\n \"0\",\n \"--colorprim\",\n \"bt709\",\n \"--transfer\",\n \"bt709\",\n \"--colormatrix\",\n \"bt709\",\n \"--crf\",\n \"18\",\n \"--output\",\n \"-\",\n ],\n },\n mux: {\n command: pupFFmpegPath,\n args: [\n \"-y\",\n ...quiet,\n \"-f\",\n \"hevc\",\n \"-r\",\n `${fps}`,\n \"-i\",\n \"pipe:0\",\n \"-c:v\",\n \"copy\",\n \"-tag:v\",\n \"hvc1\",\n \"-movflags\",\n \"+faststart\",\n mov,\n ],\n },\n };\n}\n\nexport function createCoverCommand(src: string, dst: string): Command {\n return {\n command: pupFFmpegPath,\n args: [\"-y\", ...quiet, \"-i\", src, \"-frames:v\", \"1\", \"-q:v\", \"2\", dst],\n };\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nimport { createBgraFileCommand, createBgraToMovPipeline } from \"./ffmpeg\";\nimport { exec, type ProcessHandle } from \"./process\";\nimport type { VideoSpec } from \"./types\";\n\nexport function encodeBgraFile(\n bgraPath: string,\n outputPath: string,\n spec: VideoSpec,\n format: \"mp4\" | \"webm\",\n): ProcessHandle {\n const files = format === \"mp4\" ? { mp4: outputPath } : { webm: outputPath };\n const cmd = createBgraFileCommand(bgraPath, spec, files);\n return exec(`${cmd.command} ${cmd.args.join(\" \")}`, {\n stdio: [\"ignore\", \"inherit\", \"inherit\"],\n });\n}\n\nexport function encodeBgraToMov(\n bgraPath: string,\n movPath: string,\n spec: VideoSpec,\n): ProcessHandle {\n const x265 = createBgraToMovPipeline(bgraPath, spec, movPath);\n const shellCmd = [\n `${x265.raw.command} ${x265.raw.args.join(\" \")}`,\n `${x265.x265.command} ${x265.x265.args.join(\" \")}`,\n `${x265.mux.command} ${x265.mux.args.join(\" \")}`,\n ].join(\" | \");\n return exec(shellCmd, {\n stdio: [\"ignore\", \"inherit\", \"inherit\"],\n shell: true,\n });\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nimport type { ChildProcess } from \"child_process\";\n\nexport interface StreamWriter {\n write(buffer: Buffer): Promise<void>;\n end(): void;\n}\n\nexport function createWriter(stdin: NodeJS.WritableStream): StreamWriter {\n const drainPromise = new Map<NodeJS.WritableStream, Promise<void>>();\n\n const waitDrain = (stream: NodeJS.WritableStream) => {\n const existing = drainPromise.get(stream);\n if (existing) return existing;\n const promise = new Promise<void>((resolve, reject) => {\n const cleanup = (fn: () => void) => {\n stream.off(\"drain\", onDrain);\n stream.off(\"error\", onError);\n stream.off(\"close\", onClose);\n drainPromise.delete(stream);\n fn();\n };\n const onDrain = () => cleanup(resolve);\n const onError = (err: Error) => cleanup(() => reject(err));\n const onClose = () => cleanup(() => reject(new Error(\"stream closed\")));\n stream.on(\"drain\", onDrain);\n stream.on(\"error\", onError);\n stream.on(\"close\", onClose);\n });\n drainPromise.set(stream, promise);\n return promise;\n };\n\n const destroyed = () => Reflect.get(stdin, \"destroyed\");\n return {\n async write(buffer: Buffer) {\n if (destroyed()) {\n throw new Error(\"stdin destroyed\");\n }\n if (!stdin.write(buffer)) {\n await waitDrain(stdin);\n }\n },\n end: () => {\n if (!destroyed()) stdin.end();\n },\n };\n}\n\nexport function pipeline(...procs: ChildProcess[]) {\n for (let i = 0; i < procs.length - 1; i++) {\n const src = procs[i];\n const dst = procs[i + 1];\n if (!src?.stdout || !dst?.stdin) throw new Error(\"pipeline broken\");\n src.stdout.pipe(dst.stdin);\n }\n}\n\nexport function waitAll(...procs: ChildProcess[]) {\n return Promise.all(\n procs.map(\n (proc) =>\n new Promise<void>((resolve, reject) => {\n proc.on(\"error\", reject);\n proc.on(\"close\", (code) =>\n code === 0\n ? resolve()\n : reject(new Error(`exit ${code ?? \"null\"}`)),\n );\n }),\n ),\n );\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nimport { program } from \"commander\";\nimport { pupUseInnerProxy } from \"./base/constants\";\nimport { logger } from \"./base/logging\";\nimport { noerr } from \"./base/noerr\";\nimport { parseNumber } from \"./base/parser\";\nimport { pargs } from \"./base/process\";\nimport type { RecordOptions } from \"./base/record\";\n\nexport const DEFAULT_WIDTH = 1920;\nexport const DEFAULT_HEIGHT = 1080;\nexport const DEFAULT_FPS = 30;\nexport const DEFAULT_DURATION = 5;\nexport const DEFAULT_OUT_DIR = \"out\";\n\nexport type CLICallback = (\n source: string,\n options: RecordOptions & Record<string, unknown>,\n) => Promise<unknown>;\n\nexport function makeCLI(name: string, callback: CLICallback) {\n program\n .name(name)\n .argument(\"<source>\", \"URL \u6216 HTML data\")\n .option(\"-w, --width <number>\", \"\u89C6\u9891\u5BBD\u5EA6\", `${DEFAULT_WIDTH}`)\n .option(\"-h, --height <number>\", \"\u89C6\u9891\u9AD8\u5EA6\", `${DEFAULT_HEIGHT}`)\n .option(\"-f, --fps <number>\", \"\u5E27\u7387\", `${DEFAULT_FPS}`)\n .option(\"-t, --duration <number>\", \"\u5F55\u5236\u65F6\u957F\uFF08\u79D2\uFF09\", `${DEFAULT_DURATION}`)\n .option(\"-o, --out-dir <path>\", \"\u8F93\u51FA\u76EE\u5F55\", `${DEFAULT_OUT_DIR}`)\n .option(\"-a, --with-alpha-channel\", \"\u8F93\u51FA\u5305\u542B alpha \u901A\u9053\u7684\u89C6\u9891\", false)\n .option(\n \"--use-inner-proxy\",\n \"\u4F7F\u7528 B \u7AD9\u5185\u7F51\u4EE3\u7406\u52A0\u901F\u8D44\u6E90\u8BBF\u95EE\",\n pupUseInnerProxy,\n )\n .action(async (source: string, opts) => {\n try {\n await callback(source, {\n width: noerr(parseNumber, DEFAULT_WIDTH)(opts.width),\n height: noerr(parseNumber, DEFAULT_HEIGHT)(opts.height),\n fps: noerr(parseNumber, DEFAULT_FPS)(opts.fps),\n duration: noerr(parseNumber, DEFAULT_DURATION)(opts.duration),\n outDir: opts.outDir ?? DEFAULT_OUT_DIR,\n withAlphaChannel: opts.withAlphaChannel ?? false,\n useInnerProxy: opts.useInnerProxy ?? pupUseInnerProxy,\n });\n } catch (e) {\n logger.fatal(e);\n }\n });\n program.parse(pargs());\n}\n"],
4
+ "sourcesContent": ["// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nimport { existsSync } from \"fs\";\nimport { resolve } from \"path\";\nimport { penv } from \"./env\";\nimport { parseNumber } from \"./parser\";\n\nconst pupAppSearchPaths = [\n resolve(__dirname, \"cjs/app.cjs\"), // process from dist\n resolve(__dirname, \"app.cjs\"), // process from dist/cjs\n resolve(__dirname, \"../../cjs/app.cjs\"), // process from src\n];\nexport const pupAppPath = pupAppSearchPaths.find(existsSync)!;\n\nconst env = process.env;\nexport const pupLogLevel = penv(\"PUP_LOG_LEVEL\", parseNumber, 2);\nexport const pupUseInnerProxy = env[\"PUP_USE_INNER_PROXY\"] === \"1\";\nexport const pupFFmpegPath = env[\"FFMPEG_BIN\"] ?? `ffmpeg`;\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/25.\n\nexport type EnvParser<T> = (value: unknown) => T;\n\nexport function penv<T>(name: string, parser: EnvParser<T>, defaultValue: T): T;\nexport function penv<T>(\n name: string,\n parser: EnvParser<T>,\n defaultValue?: T,\n): T | undefined;\nexport function penv<T>(\n name: string,\n parser: EnvParser<T>,\n defaultValue?: T,\n): T | undefined {\n try {\n return parser(process.env[name]);\n } catch {\n return defaultValue;\n }\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/01/30.\n\nexport function parseNumber(value: unknown): number {\n if (typeof value === \"number\") {\n return value;\n }\n const num = Number(value);\n if (Number.isNaN(num)) {\n throw new Error(`Value ${value} is not a valid number`);\n }\n return num;\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/01/30.\n\nexport class Lazy<T> {\n constructor(readonly makeValue: () => T) {}\n\n get value(): T {\n if (!this._initialized) {\n this._value = this.makeValue();\n this._initialized = true;\n }\n return this._value!;\n }\n\n get initialized(): boolean {\n return this._initialized;\n }\n\n private _initialized = false;\n private _value: T | undefined;\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/01/30.\n\nexport class ConcurrencyLimiter {\n private _active = 0;\n private _queue: VoidFunction[] = [];\n private _pending = 0;\n private _ended = false;\n\n constructor(readonly maxConcurrency: number) {}\n\n get active(): number {\n return this._active;\n }\n\n get pending(): number {\n return this._pending;\n }\n\n async schedule<T>(fn: () => Promise<T>): Promise<T> {\n if (this._ended) {\n throw new Error(\"ended\");\n }\n return new Promise<T>((resolve, reject) => {\n const run = () => {\n this._active++;\n this._pending--;\n fn()\n .then(resolve)\n .catch(reject)\n .finally(() => {\n this._active--;\n this.next();\n });\n };\n this._pending++;\n if (this._active < this.maxConcurrency) {\n run();\n } else {\n this._queue.push(run);\n }\n });\n }\n\n async end() {\n if (!this._ended) {\n this._ended = true;\n while (this._active > 0 || this._pending > 0) {\n await new Promise((resolve) => setTimeout(resolve, 50));\n }\n }\n }\n\n private next() {\n if (this._active < this.maxConcurrency && this._queue.length > 0) {\n this._queue.shift()?.();\n }\n }\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/06.\n\nimport { ChildProcess, type Serializable } from \"child_process\";\nimport { pupLogLevel } from \"./constants\";\n\nexport interface LoggerLike {\n debug?(this: void, ...messages: unknown[]): void;\n\n info?(this: void, ...messages: unknown[]): void;\n\n warn?(this: void, ...messages: unknown[]): void;\n\n error?(this: void, ...messages: unknown[]): void;\n}\n\nconst DEBUG = \"<pup@debug>\";\nconst INFO = \"<pup@info>\";\nconst WARN = \"<pup@warn>\";\nconst ERROR = \"<pup@error>\";\nconst FATAL = \"<pup@fatal>\";\n\nclass Logger implements LoggerLike {\n private _impl?: LoggerLike;\n\n get impl(): LoggerLike | undefined {\n return this._impl;\n }\n\n set impl(value: LoggerLike) {\n const debug = value.debug ?? console.debug;\n const info = value.info ?? console.info;\n const warn = value.warn ?? console.warn;\n const error = value.error ?? console.error;\n this._impl = {\n debug: pupLogLevel >= 3 ? debug : undefined,\n info: pupLogLevel >= 2 ? info : undefined,\n warn: pupLogLevel >= 1 ? warn : undefined,\n error: pupLogLevel >= 0 ? error : undefined,\n };\n }\n\n constructor() {\n this.impl = console;\n }\n\n debug(...messages: unknown[]): void {\n this.impl?.debug?.(DEBUG, ...messages);\n }\n\n info(...messages: unknown[]): void {\n this.impl?.info?.(INFO, ...messages);\n }\n\n warn(...messages: unknown[]): void {\n this.impl?.warn?.(WARN, ...messages);\n }\n\n error(...messages: unknown[]): void {\n this.impl?.error?.(ERROR, ...messages);\n }\n\n fatal(...messages: unknown[]): never {\n this.impl?.error?.(FATAL, ...messages);\n process.exit(1);\n }\n\n private dispatch(message: string) {\n if (message.startsWith(DEBUG)) {\n this.debug(message.slice(DEBUG.length + 1));\n } else if (message.startsWith(INFO)) {\n this.info(message.slice(INFO.length + 1));\n } else if (message.startsWith(WARN)) {\n this.warn(message.slice(WARN.length + 1));\n } else if (message.startsWith(ERROR)) {\n this.error(message.slice(ERROR.length + 1));\n } else {\n this.info(message);\n }\n }\n\n attach(proc: ChildProcess, name: string) {\n return new Promise<void>((resolve, reject) => {\n this.debug(`${name}.attach`);\n let fatal: string = \"\";\n const dispatch = (data: Buffer | Serializable) => {\n const message = data.toString();\n if (message.startsWith(FATAL)) {\n fatal += message.slice(FATAL.length + 1);\n } else {\n this.dispatch(message);\n }\n };\n proc.stderr?.on(\"data\", dispatch);\n proc.stdout?.on(\"data\", dispatch);\n proc\n .on(\"message\", dispatch)\n .on(\"error\", (err) => {\n fatal += err.message;\n proc.kill();\n })\n .once(\"close\", (code, signal) => {\n if (code || signal || fatal) {\n fatal ||= `command failed: ${proc.spawnargs.join(\" \")}`;\n this.error(`${name}.close`, { code, signal, fatal });\n reject(new Error(fatal));\n } else {\n this.debug(`${name}.close`);\n resolve();\n }\n })\n .on(\"unhandledRejection\", (reason) => {\n this.error(`${name}.unhandled`, reason);\n })\n .on(\"uncaughtExceptionMonitor\", (err) => {\n this.error(`${name}.unhandled`, err);\n });\n });\n }\n}\n\nconst logger = new Logger();\n\nexport { logger };\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/24.\n\nexport function noerr<Fn extends (...args: any[]) => any, D>(\n fn: Fn,\n defaultValue: D,\n): (...args: Parameters<Fn>) => ReturnType<Fn> | D {\n return (...args) => {\n try {\n const ret = fn(...args);\n if (ret instanceof Promise) {\n return ret.catch(() => defaultValue);\n }\n return ret;\n } catch {\n return defaultValue;\n }\n };\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/01/30.\n\nimport { spawn, type ChildProcess, type SpawnOptions } from \"child_process\";\nimport { logger } from \"./logging\";\n\nexport const PUP_ARGS_ENV_KEY = \"__PUP_ARGS__\";\n\nexport function pargs() {\n const pupArgs = process.env[PUP_ARGS_ENV_KEY];\n if (pupArgs) {\n const args = [\"exec\", ...process.argv.slice(-1)];\n args.push(...JSON.parse(pupArgs));\n logger.debug(\"pupargs\", args);\n return args;\n }\n\n logger.debug(\"procargv\", process.argv);\n return process.argv;\n}\n\nexport interface ProcessHandle {\n process: ChildProcess;\n wait: Promise<void>;\n}\n\nexport function exec(cmd: string, options?: SpawnOptions): ProcessHandle {\n const parts = cmd.split(\" \").filter((s) => s.length);\n const [command, ...args] = parts;\n if (!command) throw new Error(\"empty command\");\n const proc = spawn(command, args, {\n stdio: \"inherit\",\n ...options,\n });\n return { process: proc, wait: logger.attach(proc, command) };\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/05.\n\nimport { setTimeout } from \"timers/promises\";\nimport { sleep } from \"./timing\";\n\nexport interface RetryOptions<Args extends any[], Ret> {\n fn: (...args: Args) => Promise<Ret>;\n maxAttempts?: number;\n timeout?: number;\n}\n\nexport function useRetry<Args extends any[], Ret>({\n fn,\n maxAttempts = 3,\n timeout,\n}: RetryOptions<Args, Ret>) {\n const timeoutError = new Error(`timeout over ${timeout}ms`);\n return async function (...args: Args) {\n let attempt = 0;\n while (true) {\n try {\n const promises = [fn(...args)];\n if (timeout) {\n promises.push(\n setTimeout(timeout).then(() => {\n throw timeoutError;\n }),\n );\n }\n return await Promise.race(promises);\n } catch (e) {\n attempt++;\n if (attempt >= maxAttempts) {\n throw e;\n }\n await sleep(Math.pow(2, attempt) * 100 + Math.random() * 100);\n }\n }\n };\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nexport function sleep(ms: number) {\n return new Promise<void>((resolve) => setTimeout(resolve, ms));\n}\n\nexport function periodical(\n callback: (count: number) => Promise<void> | void,\n ms: number,\n) {\n let token: NodeJS.Timeout;\n let closed = false;\n async function tick(count: number) {\n await callback(count);\n if (closed) return;\n token = setTimeout(() => tick(count + 1), ms);\n }\n token = setTimeout(() => tick(0), ms);\n return () => {\n closed = true;\n clearTimeout(token);\n };\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nimport { spawn } from \"child_process\";\nimport type { Size } from \"electron\";\nimport { readFile, rm } from \"fs/promises\";\nimport { join } from \"path\";\nimport { AbortLink, type AbortQuery } from \"./base/abort\";\nimport { pupAppPath } from \"./base/constants\";\nimport { runElectronApp } from \"./base/electron\";\nimport { encodeBgraFile, encodeBgraToMov } from \"./base/encoder\";\nimport { createCoverCommand } from \"./base/ffmpeg\";\nimport { ConcurrencyLimiter } from \"./base/limiter\";\nimport { logger } from \"./base/logging\";\nimport { parseNumber } from \"./base/parser\";\nimport { type ProcessHandle } from \"./base/process\";\nimport type { RecordResult } from \"./base/record\";\nimport { waitAll } from \"./base/stream\";\nimport type { VideoFilesWithCover, VideoSpec } from \"./base/types\";\nimport { DEFAULT_HEIGHT, DEFAULT_WIDTH } from \"./common\";\n\nconst TAG = \"[pup]\";\n\nexport type PupProgressCallback = (progress: number) => Promise<void> | void;\n\nexport interface PupOptions {\n withAlphaChannel?: boolean;\n width?: number;\n height?: number;\n fps?: number;\n duration?: number;\n outDir?: string;\n cancelQuery?: AbortQuery;\n onProgress?: PupProgressCallback;\n}\n\nconst PROGRESS_TAG = \" progress: \";\n\nfunction runPupApp(source: string, options: PupOptions) {\n logger.debug(TAG, `runPupApp`, source, options);\n\n const args: string[] = [source];\n if (options.width) args.push(\"--width\", `${options.width}`);\n if (options.height) args.push(\"--height\", `${options.height}`);\n if (options.fps) args.push(\"--fps\", `${options.fps}`);\n if (options.duration) args.push(\"--duration\", `${options.duration}`);\n if (options.outDir) args.push(\"--out-dir\", options.outDir);\n if (options.withAlphaChannel) args.push(\"--with-alpha-channel\");\n\n const w = options.width ?? DEFAULT_WIDTH;\n const h = options.height ?? DEFAULT_HEIGHT;\n const handle = runElectronApp({ width: w, height: h }, pupAppPath, args);\n const counter = new ConcurrencyLimiter(1);\n handle.process.stdout?.on(\"data\", (data: Buffer) => {\n let message = data.toString().trim();\n let start = message.indexOf(PROGRESS_TAG);\n if (start < 0) {\n return;\n }\n message = message.slice(start + PROGRESS_TAG.length);\n const end = message.indexOf(\"%\");\n if (end < 0) {\n return;\n }\n const progressStr = message.slice(0, end);\n const progress = parseNumber(progressStr);\n counter.schedule(async () => {\n await options.onProgress?.(progress);\n });\n });\n return { handle, counter };\n}\n\nexport async function pup(source: string, options: PupOptions) {\n logger.debug(TAG, `pup`, source, options);\n\n const link = AbortLink.start(options.cancelQuery);\n const outDir = options.outDir ?? \"out\";\n\n const t0 = performance.now();\n const { handle, counter } = runPupApp(source, { ...options, outDir });\n\n await link.wait(handle);\n await counter.end();\n logger.info(TAG, `capture cost ${Math.round(performance.now() - t0)}ms`);\n\n const metaPath = join(outDir, \"record.json\");\n const meta = JSON.parse(await readFile(metaPath, \"utf-8\")) as RecordResult;\n\n const { bgraPath, written, options: recordOptions } = meta;\n const { fps, width, height, withAlphaChannel } = recordOptions;\n const size: Size = { width, height };\n\n const outputs: VideoFilesWithCover = {\n mp4: withAlphaChannel ? undefined : join(outDir, \"output.mp4\"),\n webm: withAlphaChannel ? join(outDir, \"output.webm\") : undefined,\n mov: withAlphaChannel ? join(outDir, \"output.mov\") : undefined,\n cover: join(outDir, \"cover.png\"),\n };\n\n try {\n const t1 = performance.now();\n\n const spec: VideoSpec = { fps, frames: written, size };\n const handles: ProcessHandle[] = [];\n if (outputs.mp4) {\n handles.push(encodeBgraFile(bgraPath, outputs.mp4, spec, \"mp4\"));\n }\n if (outputs.webm) {\n handles.push(encodeBgraFile(bgraPath, outputs.webm, spec, \"webm\"));\n }\n if (outputs.mov) {\n handles.push(encodeBgraToMov(bgraPath, outputs.mov, spec));\n }\n await link.wait(...handles);\n\n const coverSrc = outputs.mov ?? outputs.webm ?? outputs.mp4;\n if (coverSrc) {\n const coverCmd = createCoverCommand(coverSrc, outputs.cover);\n await waitAll(\n spawn(coverCmd.command, coverCmd.args, { stdio: \"inherit\" }),\n );\n }\n\n link.stop();\n logger.info(TAG, `encoding cost ${Math.round(performance.now() - t1)}ms`);\n\n await Promise.all([\n rm(bgraPath, { force: true }),\n rm(metaPath, { force: true }),\n ]);\n return {\n ...outputs,\n width,\n height,\n fps,\n duration: Math.ceil(written / fps),\n };\n } catch (error) {\n await rm(outDir, { recursive: true, force: true });\n throw error;\n }\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/11.\n\nimport type { ProcessHandle } from \"./process\";\n\nexport type AsyncTask = () => Promise<void> | void;\nexport type AbortQuery = () => Promise<boolean> | boolean;\n\nexport class AbortLink {\n private _callback?: AsyncTask;\n private _aborted?: boolean;\n private _stopped = false;\n\n private constructor(\n readonly query?: AbortQuery,\n readonly interval: number = 1000,\n ) {\n if (query) {\n this.tick();\n }\n }\n\n static start(query?: AbortQuery, interval?: number) {\n return new AbortLink(query, interval);\n }\n\n get aborted() {\n return !this._stopped && this._aborted;\n }\n\n get stopped() {\n return this._stopped;\n }\n\n async onAbort(callback: AsyncTask) {\n if (this._aborted) {\n await callback();\n } else {\n this._callback = callback;\n }\n }\n\n wait(...handles: ProcessHandle[]) {\n const abort = new Promise((_, reject) => {\n this.onAbort(async () => {\n handles.forEach((h) => h.process.kill());\n reject(new Error(\"aborted\"));\n });\n });\n return Promise.race([\n abort,\n Promise.all(handles.map((h) => h.wait)), //\n ]);\n }\n\n stop() {\n this._stopped = true;\n }\n\n private tick() {\n setTimeout(async () => {\n if (this._stopped) {\n return;\n }\n this._aborted = await this.query?.();\n if (this._stopped) {\n return;\n }\n if (this._aborted) {\n await this._callback?.();\n } else {\n this.tick();\n }\n }, this.interval);\n }\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/25.\n\nimport electron, { type Size } from \"electron\";\nimport { exec } from \"./process\";\n\nexport const ELECTRON_OPTS = [\n \"no-sandbox\",\n \"disable-setuid-sandbox\",\n \"disable-gpu\",\n \"disable-dev-shm-usage\",\n \"disable-software-rasterizer\",\n \"disable-web-security\",\n \"disable-site-isolation-trials\",\n \"disable-features=IsolateOrigins,site-per-process\",\n \"allow-insecure-localhost\",\n \"ignore-certificate-errors\",\n \"disable-blink-features=AutomationControlled\",\n \"mute-audio\",\n \"disable-extensions\",\n \"disable-background-networking\",\n \"address-family=ipv4\",\n \"disable-async-dns\",\n \"force-device-scale-factor=1\",\n \"trace-warnings\",\n \"force-color-profile=srgb\",\n \"disable-color-correct-rendering\",\n \"log-level=3\",\n];\n\nexport function runElectronApp(size: Size, app: unknown, args: unknown[]) {\n const electronArgs = ELECTRON_OPTS.map((a) => `--${a}`);\n const cmdParts = [];\n if (process.platform === \"linux\") {\n cmdParts.push(\n \"xvfb-run\",\n \"--auto-servernum\",\n `--server-args='-screen 0 ${size.width}x${size.height}x24'`,\n );\n }\n cmdParts.push(electron, ...electronArgs, app);\n return exec(cmdParts.join(\" \"), {\n stdio: [\"ignore\", \"pipe\", \"pipe\"],\n shell: true,\n env: {\n ...process.env,\n ELECTRON_DISABLE_DBUS: \"1\",\n RUST_BACKTRACE: \"full\",\n __PUP_ARGS__: JSON.stringify(args),\n },\n });\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nimport { existsSync } from \"fs\";\nimport { resolve } from \"path\";\nimport { arch, platform } from \"process\";\nimport { pupFFmpegPath } from \"./constants\";\nimport type { VideoFiles, VideoSpec } from \"./types\";\n\ninterface Command {\n command: string;\n args: string[];\n}\n\nconst quiet = [\"-hide_banner\", \"-loglevel\", \"error\", \"-nostats\"];\n\nfunction resolveX265() {\n const path = `x265/${platform}-${arch}`;\n const dirs = [\n resolve(__dirname, `../../${path}`), // process from src\n resolve(__dirname, `../${path}`), // process from dist\n ];\n const found = dirs.find(existsSync);\n if (!found) {\n throw new Error(\"x265 not found\");\n }\n return found;\n}\n\nexport function createBgraFileCommand(\n bgraPath: string,\n spec: VideoSpec,\n files: VideoFiles,\n): Command {\n const { fps, frames } = spec;\n const args = [\n \"-y\",\n ...quiet,\n \"-f\",\n \"rawvideo\",\n \"-pix_fmt\",\n \"bgra\",\n \"-s\",\n `${spec.size.width}x${spec.size.height}`,\n \"-r\",\n `${fps}`,\n \"-i\",\n bgraPath,\n \"-frames:v\",\n `${Math.floor(frames)}`,\n ];\n if (files.mp4) {\n args.push(\n \"-colorspace\",\n \"bt709\",\n \"-color_primaries\",\n \"bt709\",\n \"-color_trc\",\n \"bt709\",\n \"-c:v\",\n \"libx264\",\n \"-preset\",\n \"fast\",\n \"-threads\",\n \"0\",\n \"-pix_fmt\",\n \"yuv420p\",\n \"-movflags\",\n \"+faststart\",\n files.mp4,\n );\n }\n if (files.webm) {\n args.push(\n \"-c:v\",\n \"libvpx-vp9\",\n \"-row-mt\",\n \"1\",\n \"-cpu-used\",\n \"1\",\n \"-threads\",\n \"0\",\n \"-pix_fmt\",\n \"yuva420p\",\n \"-auto-alt-ref\",\n \"0\",\n files.webm,\n );\n }\n return { command: pupFFmpegPath, args };\n}\n\ninterface X265Pipeline {\n raw: Command;\n x265: Command;\n mux: Command;\n}\n\nexport function createBgraToMovPipeline(\n bgraPath: string,\n spec: VideoSpec,\n mov: string,\n): X265Pipeline {\n const { fps, size } = spec;\n return {\n raw: {\n command: pupFFmpegPath,\n args: [\n \"-y\",\n ...quiet,\n \"-f\",\n \"rawvideo\",\n \"-pix_fmt\",\n \"bgra\",\n \"-s\",\n `${size.width}x${size.height}`,\n \"-r\",\n `${fps}`,\n \"-i\",\n bgraPath,\n \"-f\",\n \"rawvideo\",\n \"-pix_fmt\",\n \"yuva420p10le\",\n \"pipe:1\",\n ],\n },\n x265: {\n command: resolveX265(),\n args: [\n \"--no-progress\",\n \"--log-level\",\n \"error\",\n \"--input\",\n \"-\",\n \"--input-res\",\n `${size.width}x${size.height}`,\n \"--input-csp\",\n \"i420\",\n \"--input-depth\",\n \"10\",\n \"--output-depth\",\n \"10\",\n \"--fps\",\n `${fps}`,\n \"--alpha\",\n \"--bframes\",\n \"0\",\n \"--colorprim\",\n \"bt709\",\n \"--transfer\",\n \"bt709\",\n \"--colormatrix\",\n \"bt709\",\n \"--crf\",\n \"18\",\n \"--output\",\n \"-\",\n ],\n },\n mux: {\n command: pupFFmpegPath,\n args: [\n \"-y\",\n ...quiet,\n \"-f\",\n \"hevc\",\n \"-r\",\n `${fps}`,\n \"-i\",\n \"pipe:0\",\n \"-c:v\",\n \"copy\",\n \"-tag:v\",\n \"hvc1\",\n \"-movflags\",\n \"+faststart\",\n mov,\n ],\n },\n };\n}\n\nexport function createCoverCommand(src: string, dst: string): Command {\n return {\n command: pupFFmpegPath,\n args: [\"-y\", ...quiet, \"-i\", src, \"-frames:v\", \"1\", \"-q:v\", \"2\", dst],\n };\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nimport { createBgraFileCommand, createBgraToMovPipeline } from \"./ffmpeg\";\nimport { exec, type ProcessHandle } from \"./process\";\nimport type { VideoSpec } from \"./types\";\n\nexport function encodeBgraFile(\n bgraPath: string,\n outputPath: string,\n spec: VideoSpec,\n format: \"mp4\" | \"webm\",\n): ProcessHandle {\n const files = format === \"mp4\" ? { mp4: outputPath } : { webm: outputPath };\n const cmd = createBgraFileCommand(bgraPath, spec, files);\n return exec(`${cmd.command} ${cmd.args.join(\" \")}`, {\n stdio: [\"ignore\", \"inherit\", \"inherit\"],\n });\n}\n\nexport function encodeBgraToMov(\n bgraPath: string,\n movPath: string,\n spec: VideoSpec,\n): ProcessHandle {\n const x265 = createBgraToMovPipeline(bgraPath, spec, movPath);\n const shellCmd = [\n `${x265.raw.command} ${x265.raw.args.join(\" \")}`,\n `${x265.x265.command} ${x265.x265.args.join(\" \")}`,\n `${x265.mux.command} ${x265.mux.args.join(\" \")}`,\n ].join(\" | \");\n return exec(shellCmd, {\n stdio: [\"ignore\", \"inherit\", \"inherit\"],\n shell: true,\n });\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nimport type { ChildProcess } from \"child_process\";\n\nexport interface StreamWriter {\n write(buffer: Buffer): Promise<void>;\n end(): void;\n}\n\nexport function createWriter(stdin: NodeJS.WritableStream): StreamWriter {\n const drainPromise = new Map<NodeJS.WritableStream, Promise<void>>();\n\n const waitDrain = (stream: NodeJS.WritableStream) => {\n const existing = drainPromise.get(stream);\n if (existing) return existing;\n const promise = new Promise<void>((resolve, reject) => {\n const cleanup = (fn: () => void) => {\n stream.off(\"drain\", onDrain);\n stream.off(\"error\", onError);\n stream.off(\"close\", onClose);\n drainPromise.delete(stream);\n fn();\n };\n const onDrain = () => cleanup(resolve);\n const onError = (err: Error) => cleanup(() => reject(err));\n const onClose = () => cleanup(() => reject(new Error(\"stream closed\")));\n stream.on(\"drain\", onDrain);\n stream.on(\"error\", onError);\n stream.on(\"close\", onClose);\n });\n drainPromise.set(stream, promise);\n return promise;\n };\n\n const destroyed = () => Reflect.get(stdin, \"destroyed\");\n return {\n async write(buffer: Buffer) {\n if (destroyed()) {\n throw new Error(\"stdin destroyed\");\n }\n if (!stdin.write(buffer)) {\n await waitDrain(stdin);\n }\n },\n end: () => {\n if (!destroyed()) stdin.end();\n },\n };\n}\n\nexport function pipeline(...procs: ChildProcess[]) {\n for (let i = 0; i < procs.length - 1; i++) {\n const src = procs[i];\n const dst = procs[i + 1];\n if (!src?.stdout || !dst?.stdin) throw new Error(\"pipeline broken\");\n src.stdout.pipe(dst.stdin);\n }\n}\n\nexport function waitAll(...procs: ChildProcess[]) {\n return Promise.all(\n procs.map(\n (proc) =>\n new Promise<void>((resolve, reject) => {\n proc.on(\"error\", reject);\n proc.on(\"close\", (code) =>\n code === 0\n ? resolve()\n : reject(new Error(`exit ${code ?? \"null\"}`)),\n );\n }),\n ),\n );\n}\n", "// Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.\n\nimport { program } from \"commander\";\nimport { pupUseInnerProxy } from \"./base/constants\";\nimport { logger } from \"./base/logging\";\nimport { noerr } from \"./base/noerr\";\nimport { parseNumber } from \"./base/parser\";\nimport { pargs } from \"./base/process\";\nimport type { RecordOptions } from \"./base/record\";\n\nexport const DEFAULT_WIDTH = 1920;\nexport const DEFAULT_HEIGHT = 1080;\nexport const DEFAULT_FPS = 30;\nexport const DEFAULT_DURATION = 5;\nexport const DEFAULT_OUT_DIR = \"out\";\n\nexport type CLICallback = (\n source: string,\n options: RecordOptions & Record<string, unknown>,\n) => Promise<unknown>;\n\nexport function makeCLI(name: string, callback: CLICallback) {\n program\n .name(name)\n .argument(\"<source>\", \"file://, http(s)://, \u6216 data: URI\")\n .option(\"-w, --width <number>\", \"\u89C6\u9891\u5BBD\u5EA6\", `${DEFAULT_WIDTH}`)\n .option(\"-h, --height <number>\", \"\u89C6\u9891\u9AD8\u5EA6\", `${DEFAULT_HEIGHT}`)\n .option(\"-f, --fps <number>\", \"\u5E27\u7387\", `${DEFAULT_FPS}`)\n .option(\"-t, --duration <number>\", \"\u5F55\u5236\u65F6\u957F\uFF08\u79D2\uFF09\", `${DEFAULT_DURATION}`)\n .option(\"-o, --out-dir <path>\", \"\u8F93\u51FA\u76EE\u5F55\", `${DEFAULT_OUT_DIR}`)\n .option(\"-a, --with-alpha-channel\", \"\u8F93\u51FA\u5305\u542B alpha \u901A\u9053\u7684\u89C6\u9891\", false)\n .option(\n \"--use-inner-proxy\",\n \"\u4F7F\u7528 B \u7AD9\u5185\u7F51\u4EE3\u7406\u52A0\u901F\u8D44\u6E90\u8BBF\u95EE\",\n pupUseInnerProxy,\n )\n .action(async (source: string, opts) => {\n try {\n await callback(source, {\n width: noerr(parseNumber, DEFAULT_WIDTH)(opts.width),\n height: noerr(parseNumber, DEFAULT_HEIGHT)(opts.height),\n fps: noerr(parseNumber, DEFAULT_FPS)(opts.fps),\n duration: noerr(parseNumber, DEFAULT_DURATION)(opts.duration),\n outDir: opts.outDir ?? DEFAULT_OUT_DIR,\n withAlphaChannel: opts.withAlphaChannel ?? false,\n useInnerProxy: opts.useInnerProxy ?? pupUseInnerProxy,\n });\n } catch (e) {\n logger.fatal(e);\n }\n });\n program.parse(pargs());\n}\n"],
5
5
  "mappings": "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",
6
6
  "names": ["resolve", "resolve", "setTimeout", "resolve", "setTimeout", "spawn", "existsSync", "resolve", "resolve", "existsSync", "resolve", "spawn"]
7
7
  }
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "pup-recorder",
3
- "version": "0.0.8",
3
+ "version": "0.0.10",
4
4
  "description": "High-performance webview recording tool.",
5
5
  "type": "module",
6
6
  "bin": {
Binary file
Binary file
@@ -0,0 +1,17 @@
1
+ // Created by Autokaka (qq1909698494@gmail.com) on 2026/02/09.
2
+
3
+ const SUPPORTED_PROTOCOLS = ["file:", "http:", "https:", "data:"];
4
+ const SOURCE_PATTERN = /^(file:|https?:|data:)/;
5
+
6
+ export function checkHTML(source: string): void {
7
+ if (SOURCE_PATTERN.test(source)) {
8
+ return;
9
+ }
10
+
11
+ const protocol = source.split(":")[0] + ":";
12
+ const message = SUPPORTED_PROTOCOLS.includes(protocol)
13
+ ? `unsupported protocol: ${protocol}, expected ${SUPPORTED_PROTOCOLS.join(", ")}`
14
+ : `invalid source format, expected ${SUPPORTED_PROTOCOLS.join(", ")}`;
15
+
16
+ throw new Error(message);
17
+ }
@@ -10,6 +10,7 @@ import {
10
10
  startSync,
11
11
  stopSync,
12
12
  } from "./frame_sync";
13
+ import { checkHTML } from "./html_check";
13
14
  import { isEmpty } from "./image";
14
15
  import { logger } from "./logging";
15
16
  import { enableProxy, proxiedUrl } from "./proxy";
@@ -34,9 +35,7 @@ export interface RecordResult {
34
35
  }
35
36
 
36
37
  async function loadWindow(source: string, options: RecordOptions) {
37
- if (!source.startsWith("file://") && !source.match(/^https?:\/\//)) {
38
- throw new Error("invalid source");
39
- }
38
+ checkHTML(source);
40
39
 
41
40
  const { width, height, useInnerProxy } = options;
42
41
 
package/src/common.ts CHANGED
@@ -22,7 +22,7 @@ export type CLICallback = (
22
22
  export function makeCLI(name: string, callback: CLICallback) {
23
23
  program
24
24
  .name(name)
25
- .argument("<source>", "URLHTML data")
25
+ .argument("<source>", "file://, http(s)://, data: URI")
26
26
  .option("-w, --width <number>", "视频宽度", `${DEFAULT_WIDTH}`)
27
27
  .option("-h, --height <number>", "视频高度", `${DEFAULT_HEIGHT}`)
28
28
  .option("-f, --fps <number>", "帧率", `${DEFAULT_FPS}`)