zeldhash-miner 0.2.3 → 0.2.5

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/index.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"file":"index.js","sources":["../src/types.ts","../src/errors.ts","../src/coordinator.ts","../src/wasm.ts","../src/nonce.ts","../src/builder.ts","../src/index.ts"],"sourcesContent":["export enum ZeldMinerErrorCode {\n INVALID_ADDRESS = \"INVALID_ADDRESS\",\n UNSUPPORTED_ADDRESS_TYPE = \"UNSUPPORTED_ADDRESS_TYPE\",\n INSUFFICIENT_FUNDS = \"INSUFFICIENT_FUNDS\",\n NO_CHANGE_OUTPUT = \"NO_CHANGE_OUTPUT\",\n MULTIPLE_CHANGE_OUTPUTS = \"MULTIPLE_CHANGE_OUTPUTS\",\n INVALID_INPUT = \"INVALID_INPUT\",\n WEBGPU_NOT_AVAILABLE = \"WEBGPU_NOT_AVAILABLE\",\n WORKER_ERROR = \"WORKER_ERROR\",\n MINING_ABORTED = \"MINING_ABORTED\",\n DUST_OUTPUT = \"DUST_OUTPUT\",\n}\n\nexport type ZeldMinerErrorDetails = Record<string, unknown>;\n\nexport type Network = \"mainnet\" | \"testnet\" | \"signet\" | \"regtest\";\n\nexport interface ZeldMinerOptions {\n network: Network;\n batchSize: number;\n useWebGPU: boolean;\n workerThreads: number;\n satsPerVbyte: number;\n}\n\nexport interface MiningCoordinatorOptions {\n mode: WorkerMode;\n batchSize: number;\n workerThreads: number;\n}\n\nexport interface TxInput {\n txid: string;\n vout: number;\n scriptPubKey: string;\n amount: number;\n}\n\nexport interface TxOutput {\n address: string;\n amount?: number;\n change: boolean;\n}\n\nexport interface MineResult {\n psbt: string;\n txid: string;\n nonce: bigint;\n attempts: bigint;\n duration: number;\n hashRate: number;\n}\n\nexport interface ProgressEvent {\n hashesProcessed: bigint;\n hashRate: number;\n elapsedMs?: number;\n lastNonce?: bigint;\n workerId?: string;\n}\n\nexport type WorkerMode = \"cpu\" | \"gpu\";\n\nexport type WorkerMessage =\n | { type: \"init\"; mode: WorkerMode }\n | {\n type: \"mine\";\n inputs: TxInput[];\n outputs: TxOutput[];\n network: Network;\n satsPerVbyte: number;\n template: WorkerTemplate;\n startNonce: bigint;\n batchSize: number;\n targetZeros: number;\n nonceStep?: bigint;\n distribution?: bigint[];\n }\n | { type: \"stop\" };\n\nexport type WorkerResponse =\n | { type: \"ready\"; workerId?: string }\n | {\n type: \"progress\";\n hashesProcessed: bigint;\n hashRate: number;\n lastNonce?: bigint;\n workerId?: string;\n }\n | {\n type: \"found\";\n result: MineResult;\n hashesProcessed?: bigint;\n hashRate?: number;\n lastNonce?: bigint;\n workerId?: string;\n }\n | { type: \"batch_complete\"; lastNonce: bigint; workerId?: string }\n | {\n type: \"error\";\n message: string;\n code?: ZeldMinerErrorCode;\n details?: ZeldMinerErrorDetails;\n workerId?: string;\n };\n\nexport interface MiningState {\n mode: WorkerMode;\n running: boolean;\n startNonce: bigint;\n nextNonce: bigint;\n batchSize: number;\n targetZeros: number;\n hashesProcessed: bigint;\n lastHashRate?: number;\n}\n\nexport interface MiningTemplate {\n prefix: Uint8Array;\n suffix: Uint8Array;\n useCborNonce?: boolean;\n}\n\nexport interface WorkerTemplate extends MiningTemplate {\n nonceLength: number;\n useCborNonce?: boolean;\n}\n\nexport interface ValidationResult {\n ok: boolean;\n error?: string;\n addressType?: \"p2wpkh\" | \"p2tr\";\n network?: Network;\n}\n\nexport interface WasmExports {\n init_panic_hook: () => void;\n mine_batch_wasm: (\n txPrefix: Uint8Array,\n txSuffix: Uint8Array,\n startNonce: bigint,\n batchSize: number,\n targetZeros: number,\n useCborNonce?: boolean\n ) => ValidationResult | MineResult | null;\n mine_batch_gpu?: (\n txPrefix: Uint8Array,\n txSuffix: Uint8Array,\n startNonce: bigint,\n batchSize: number,\n targetZeros: number,\n useCborNonce?: boolean\n ) => Promise<ValidationResult | MineResult | null>;\n mine_range_wasm: (\n inputs: TxInput[],\n outputs: TxOutput[],\n network: Network,\n satsPerVbyte: bigint,\n startNonce: bigint,\n batchSize: number,\n targetZeros: number,\n distribution?: bigint[] | null\n ) => ValidationResult | MineResult | null;\n mine_range_gpu?: (\n inputs: TxInput[],\n outputs: TxOutput[],\n network: Network,\n satsPerVbyte: bigint,\n startNonce: bigint,\n batchSize: number,\n targetZeros: number,\n distribution?: bigint[] | null\n ) => Promise<ValidationResult | MineResult | null>;\n validate_address: (addr: string, network: Network) => ValidationResult;\n build_psbt: (\n inputs: TxInput[],\n outputs: TxOutput[],\n network: Network,\n satsPerVbyte: bigint,\n nonce: bigint,\n distribution?: bigint[] | null\n ) => string;\n build_mining_template: (\n inputs: TxInput[],\n outputs: TxOutput[],\n network: Network,\n satsPerVbyte: bigint,\n startNonce: bigint,\n batchSize: number,\n distribution?: bigint[] | null\n ) => MiningTemplate;\n compute_txid: (txBytes: Uint8Array) => string;\n init_gpu?: () => Promise<unknown>;\n calibrate_batch_size?: () => Promise<number>;\n}\n\nexport type ProgressStats = ProgressEvent;\n\nexport interface MineParams {\n inputs: TxInput[];\n outputs: TxOutput[];\n targetZeros: number;\n startNonce?: bigint;\n batchSize?: number;\n signal?: AbortSignal;\n distribution?: bigint[];\n}\n\n","import { ZeldMinerErrorCode, type ZeldMinerErrorDetails } from \"./types\";\n\nconst formatUnknownError = (err: unknown): string =>\n err instanceof Error ? err.message : String(err);\n\nexport class ZeldMinerError extends Error {\n readonly code: ZeldMinerErrorCode;\n readonly details?: ZeldMinerErrorDetails;\n\n constructor(\n code: ZeldMinerErrorCode,\n message: string,\n details?: ZeldMinerErrorDetails\n ) {\n super(message);\n this.name = \"ZeldMinerError\";\n this.code = code;\n this.details = details;\n }\n}\n\nexport const toZeldMinerError = (\n err: unknown,\n fallbackCode: ZeldMinerErrorCode = ZeldMinerErrorCode.WORKER_ERROR,\n details?: ZeldMinerErrorDetails\n): ZeldMinerError => {\n if (err instanceof ZeldMinerError) {\n return err;\n }\n\n const message = formatUnknownError(err);\n return new ZeldMinerError(fallbackCode, message, details);\n};\n\nexport const createMinerError = (\n code: ZeldMinerErrorCode,\n message: string,\n details?: ZeldMinerErrorDetails\n): ZeldMinerError => new ZeldMinerError(code, message, details);\n\n","import type {\n MineResult,\n MiningCoordinatorOptions,\n ProgressEvent as MiningProgressEvent,\n Network,\n TxInput,\n TxOutput,\n WorkerMessage,\n WorkerMode,\n WorkerResponse,\n WorkerTemplate,\n} from \"./types\";\nimport { ZeldMinerErrorCode } from \"./types\";\nimport { ZeldMinerError } from \"./errors\";\n\ntype CoordinatorEvent = \"ready\" | \"progress\" | \"found\" | \"error\" | \"stopped\";\n\ntype CoordinatorEventMap = {\n ready: void;\n progress: MiningProgressEvent;\n found: MineResult;\n error: ZeldMinerError;\n stopped: void;\n};\n\ntype CoordinatorListener<K extends CoordinatorEvent> = (\n payload: CoordinatorEventMap[K]\n) => void;\n\ninterface WorkerState {\n worker: Worker;\n hashesProcessed: bigint;\n hashRate: number;\n lastNonce?: bigint;\n nextNonce: bigint;\n processedBase: bigint;\n terminated?: boolean;\n}\n\ninterface StartParams {\n inputs: TxInput[];\n outputs: TxOutput[];\n network: Network;\n satsPerVbyte: number;\n template: WorkerTemplate;\n targetZeros: number;\n startNonce?: bigint;\n signal?: AbortSignal;\n distribution?: bigint[];\n}\n\nconst safeBigIntToNumber = (value: bigint): number => {\n const max = BigInt(Number.MAX_SAFE_INTEGER);\n if (value > max) return Number.MAX_SAFE_INTEGER;\n if (value < -max) return -Number.MAX_SAFE_INTEGER;\n return Number(value);\n};\n\nexport class MiningCoordinator {\n private readonly mode: WorkerMode;\n private readonly batchSize: number;\n private readonly workerCount: number;\n private readonly listeners: {\n [K in CoordinatorEvent]: Set<CoordinatorListener<K>>;\n };\n private readonly workers: WorkerState[] = [];\n private readonly readyPromise: Promise<void>;\n private readonly abortHandler: () => void;\n\n private stride: bigint;\n private cleanupExternalAbort = (): void => {\n if (this.externalAbort) {\n this.externalAbort.removeEventListener(\"abort\", this.abortHandler);\n this.externalAbort = undefined;\n }\n };\n private running = false;\n private paused = false;\n private startedAt: number | null = null;\n private txInputs?: TxInput[];\n private txOutputs?: TxOutput[];\n private txNetwork?: Network;\n private satsPerVbyte?: number;\n private template?: WorkerTemplate;\n private targetZeros?: number;\n private startNonce: bigint = 0n;\n private txDistribution?: bigint[];\n private externalAbort?: AbortSignal;\n private terminated = false;\n\n constructor(options: MiningCoordinatorOptions) {\n this.mode = options.mode;\n this.batchSize = options.batchSize;\n this.workerCount =\n this.mode === \"gpu\" ? 1 : Math.max(1, options.workerThreads);\n this.stride =\n this.mode === \"gpu\"\n ? BigInt(this.batchSize)\n : BigInt(this.batchSize) * BigInt(this.workerCount);\n\n this.listeners = {\n ready: new Set(),\n progress: new Set(),\n found: new Set(),\n error: new Set(),\n stopped: new Set(),\n };\n\n this.abortHandler = () => this.stop();\n this.readyPromise = this.spawnWorkers();\n }\n\n on<K extends CoordinatorEvent>(\n event: K,\n handler: CoordinatorListener<K>\n ): void {\n this.listeners[event].add(handler);\n }\n\n off<K extends CoordinatorEvent>(\n event: K,\n handler: CoordinatorListener<K>\n ): void {\n this.listeners[event].delete(handler);\n }\n\n private emit<K extends CoordinatorEvent>(\n event: K,\n payload: CoordinatorEventMap[K]\n ): void {\n this.listeners[event].forEach((handler) => handler(payload));\n }\n\n private resolveWorkerUrl(): string {\n const origin =\n typeof location !== \"undefined\" && location.origin\n ? location.origin\n : typeof self !== \"undefined\" &&\n (self as { location?: { origin?: string } }).location?.origin\n ? (self as { location?: { origin?: string } }).location!.origin\n : undefined;\n if (origin) return `${origin}/worker.js`;\n return new URL(\"./worker.ts\", import.meta.url).href;\n }\n\n private async spawnWorkers(): Promise<void> {\n const readySignals: Promise<void>[] = [];\n const workerUrl = this.resolveWorkerUrl();\n\n for (let i = 0; i < this.workerCount; i += 1) {\n const worker = new Worker(workerUrl, {\n type: \"module\",\n /* @vite-ignore */ name: `zeldminer-worker-${i}`,\n });\n\n const state: WorkerState = {\n worker,\n hashesProcessed: 0n,\n hashRate: 0,\n nextNonce: 0n,\n processedBase: 0n,\n };\n\n this.workers.push(state);\n\n readySignals.push(\n new Promise<void>((resolve) => {\n const handleReady = (event: MessageEvent<WorkerResponse>): void => {\n if (event.data.type === \"ready\") {\n worker.removeEventListener(\"message\", handleReady);\n resolve();\n }\n };\n\n worker.addEventListener(\"message\", handleReady);\n })\n );\n\n worker.addEventListener(\"message\", (event: MessageEvent<WorkerResponse>) =>\n this.handleWorkerMessage(state, event.data)\n );\n\n const initMessage: WorkerMessage = { type: \"init\", mode: this.mode };\n worker.postMessage(initMessage);\n }\n\n await Promise.all(readySignals);\n this.emit(\"ready\", undefined as unknown as void);\n }\n\n private handleWorkerMessage(\n state: WorkerState,\n message: WorkerResponse\n ): void {\n switch (message.type) {\n case \"ready\":\n break;\n case \"progress\":\n state.hashesProcessed = state.processedBase + message.hashesProcessed;\n state.hashRate = message.hashRate;\n state.lastNonce = message.lastNonce ?? state.lastNonce;\n this.emitProgress();\n break;\n case \"batch_complete\":\n state.lastNonce = message.lastNonce;\n state.nextNonce = this.computeNextNonce(message.lastNonce);\n break;\n case \"found\":\n state.hashesProcessed =\n state.processedBase + (message.hashesProcessed ?? state.hashesProcessed);\n state.hashRate = message.hashRate ?? state.hashRate;\n state.lastNonce = message.lastNonce ?? state.lastNonce;\n\n const attempts = this.workers.reduce(\n (total, worker) => total + worker.hashesProcessed,\n 0n\n );\n const duration = this.startedAt ? performance.now() - this.startedAt : 0;\n const hashRate =\n duration > 0\n ? safeBigIntToNumber(attempts) / (duration / 1000)\n : 0;\n\n const adjustedResult: MineResult = {\n ...message.result,\n attempts,\n duration,\n hashRate,\n };\n\n this.running = false;\n this.paused = false;\n this.terminateWorkers();\n this.cleanupExternalAbort();\n this.emit(\"found\", adjustedResult);\n break;\n case \"error\":\n this.running = false;\n this.paused = false;\n this.terminateWorkers();\n this.cleanupExternalAbort();\n this.emit(\n \"error\",\n new ZeldMinerError(\n message.code ?? ZeldMinerErrorCode.WORKER_ERROR,\n message.message,\n {\n workerId: message.workerId,\n ...(message.details ?? {}),\n }\n )\n );\n break;\n }\n }\n\n private emitProgress(): void {\n if (!this.running) return;\n\n const hashesProcessed = this.workers.reduce(\n (total, worker) => total + worker.hashesProcessed,\n 0n\n );\n const elapsedMs = this.startedAt ? performance.now() - this.startedAt : 0;\n // Derive rate from total attempts over wall-clock time to avoid overstating\n // throughput on multi-worker CPU runs.\n const hashRate =\n elapsedMs > 0\n ? safeBigIntToNumber(hashesProcessed) / (elapsedMs / 1000)\n : 0;\n\n const progress: MiningProgressEvent = {\n hashesProcessed,\n hashRate,\n elapsedMs,\n };\n\n this.emit(\"progress\", progress);\n }\n\n private computeNextNonce(lastNonce: bigint): bigint {\n const batchAdvance = this.stride - BigInt(this.batchSize);\n return lastNonce + 1n + batchAdvance;\n }\n\n private stopWorkers(): void {\n if (this.terminated) return;\n this.workers.forEach((state) =>\n state.worker.postMessage({ type: \"stop\" } satisfies WorkerMessage)\n );\n }\n\n private terminateWorkers(): void {\n if (this.terminated) return;\n this.stopWorkers();\n this.workers.forEach((state) => {\n try {\n state.worker.terminate();\n state.terminated = true;\n } catch {\n /* ignore termination errors */\n }\n });\n this.terminated = true;\n }\n\n async start(params: StartParams): Promise<void> {\n await this.readyPromise;\n this.stopWorkers();\n\n this.running = true;\n this.paused = false;\n this.startedAt = performance.now();\n this.txInputs = params.inputs;\n this.txOutputs = params.outputs;\n this.txNetwork = params.network === \"signet\" ? \"testnet\" : params.network;\n this.satsPerVbyte = params.satsPerVbyte;\n this.template = params.template;\n this.targetZeros = params.targetZeros;\n this.startNonce = params.startNonce ?? 0n;\n this.txDistribution = params.distribution;\n\n if (this.externalAbort && this.externalAbort !== params.signal) {\n this.cleanupExternalAbort();\n }\n\n if (params.signal) {\n this.externalAbort = params.signal;\n params.signal.addEventListener(\"abort\", this.abortHandler, { once: true });\n }\n\n if (\n !this.template ||\n !this.txInputs ||\n !this.txOutputs ||\n this.targetZeros === undefined ||\n this.satsPerVbyte === undefined ||\n !this.txNetwork\n ) {\n throw new Error(\"Mining parameters are missing\");\n }\n\n const inputs = this.txInputs;\n const outputs = this.txOutputs;\n const network = this.txNetwork;\n const satsPerVbyte = this.satsPerVbyte;\n const template = this.template;\n const targetZeros = this.targetZeros;\n const distribution = this.txDistribution;\n const stride = this.stride;\n\n this.workers.forEach((state, idx) => {\n state.processedBase = 0n;\n state.hashesProcessed = 0n;\n state.hashRate = 0;\n state.lastNonce = undefined;\n const workerStart =\n this.startNonce + BigInt(idx) * BigInt(this.batchSize);\n state.nextNonce = workerStart;\n\n const message: WorkerMessage = {\n type: \"mine\",\n inputs,\n outputs,\n network,\n satsPerVbyte,\n template,\n startNonce: workerStart,\n batchSize: this.batchSize,\n targetZeros,\n nonceStep: stride,\n distribution,\n };\n\n state.worker.postMessage(message);\n });\n }\n\n pause(): void {\n if (!this.running) return;\n this.running = false;\n this.paused = true;\n this.stopWorkers();\n }\n\n async resume(): Promise<void> {\n if (\n !this.paused ||\n !this.template ||\n !this.txInputs ||\n !this.txOutputs ||\n this.targetZeros === undefined ||\n this.satsPerVbyte === undefined ||\n !this.txNetwork\n )\n return;\n await this.readyPromise;\n\n this.running = true;\n this.paused = false;\n if (!this.startedAt) {\n this.startedAt = performance.now();\n }\n\n const inputs = this.txInputs;\n const outputs = this.txOutputs;\n const network = this.txNetwork;\n const satsPerVbyte = this.satsPerVbyte;\n const template = this.template;\n const targetZeros = this.targetZeros as number;\n const distribution = this.txDistribution;\n const stride = this.stride;\n\n this.workers.forEach((state, idx) => {\n state.processedBase = state.hashesProcessed;\n state.hashRate = 0;\n const workerStart =\n state.nextNonce ||\n this.startNonce + BigInt(idx) * BigInt(this.batchSize);\n\n const message: WorkerMessage = {\n type: \"mine\",\n inputs,\n outputs,\n network,\n satsPerVbyte,\n template,\n startNonce: workerStart,\n batchSize: this.batchSize,\n targetZeros,\n nonceStep: stride,\n distribution,\n };\n\n state.worker.postMessage(message);\n });\n }\n\n stop(): void {\n if (!this.running && !this.paused) return;\n this.running = false;\n this.paused = false;\n this.terminateWorkers();\n this.cleanupExternalAbort();\n this.emit(\"stopped\", undefined as unknown as void);\n }\n}\n\n","import type { WasmExports } from \"./types\";\n\n// Bootstrap: define a sensible default for WASM base path so consumers don't\n// need to patch the library. In Vite/dev the import.meta.url points inside\n// node_modules which is not served, so we derive a public path from the window\n// origin when available.\nif (typeof (globalThis as { __ZELDMINER_WASM_BASE__?: unknown }).__ZELDMINER_WASM_BASE__ === \"undefined\") {\n try {\n const origin =\n typeof window !== \"undefined\" && window.location?.origin\n ? window.location.origin\n : typeof self !== \"undefined\" && (self as { location?: { origin?: string } }).location?.origin\n ? (self as { location?: { origin?: string } }).location!.origin\n : \"http://localhost\";\n (globalThis as { __ZELDMINER_WASM_BASE__?: string }).__ZELDMINER_WASM_BASE__ = new URL(\"/wasm/\", origin!).href;\n } catch {\n (globalThis as { __ZELDMINER_WASM_BASE__?: string }).__ZELDMINER_WASM_BASE__ = \"/wasm/\";\n }\n}\n\nlet wasmModule: WasmExports | null = null;\nlet wasmInitPromise: Promise<WasmExports> | null = null;\n\n// Chrome/Dawn can reject requestDevice when optional limits (e.g.\n// maxInterStageShaderComponents) are present but unsupported. Strip any\n// limit keys the adapter doesn't expose to keep WebGPU initialization\n// portable across browser versions.\nlet webGpuLimitShimInstalled = false;\nconst installWebGpuLimitShim = (): void => {\n if (webGpuLimitShimInstalled) return;\n webGpuLimitShimInstalled = true;\n\n const adapterProto = (globalThis as typeof globalThis & { GPUAdapter?: { prototype?: unknown } })\n .GPUAdapter?.prototype as GPUAdapter | undefined;\n const requestDevice = adapterProto?.requestDevice;\n if (!adapterProto || typeof requestDevice !== \"function\") return;\n\n adapterProto.requestDevice = function patchedRequestDevice(\n this: GPUAdapter,\n descriptor?: GPUDeviceDescriptor\n ): Promise<GPUDevice> {\n if (descriptor?.requiredLimits && typeof this.limits === \"object\") {\n const limits = descriptor.requiredLimits as Record<string, unknown>;\n const supported = this.limits as unknown as Record<string, unknown>;\n for (const key of Object.keys(limits)) {\n if (!(key in supported) || supported[key] === undefined) {\n delete limits[key];\n }\n }\n }\n return requestDevice.call(this, descriptor);\n };\n};\n\nconst ensureTrailingSlash = (value: string): string =>\n value.endsWith(\"/\") ? value : `${value}/`;\n\nconst toAbsoluteBase = (base: string): string => {\n const trimmed = base.trim();\n if (!trimmed) return trimmed;\n // Use window origin when available so Vite treats it as an external URL and does not try to transform public assets.\n if (typeof window !== \"undefined\" && typeof window.location?.origin === \"string\") {\n return ensureTrailingSlash(new URL(trimmed, window.location.origin).href);\n }\n return ensureTrailingSlash(new URL(trimmed, import.meta.url).href);\n};\n\nconst resolveWasmBase = (): string => {\n const globalBase = (globalThis as { __ZELDMINER_WASM_BASE__?: unknown })\n .__ZELDMINER_WASM_BASE__;\n if (typeof globalBase === \"string\" && globalBase.trim()) {\n return toAbsoluteBase(globalBase);\n }\n\n const envBase = (import.meta as ImportMeta & { env?: Record<string, unknown> })\n .env?.VITE_ZELDMINER_WASM_BASE;\n if (typeof envBase === \"string\" && envBase.trim()) {\n return toAbsoluteBase(envBase);\n }\n\n const viteBase = (import.meta as ImportMeta & { env?: Record<string, unknown> }).env?.BASE_URL;\n if (typeof viteBase === \"string\" && viteBase.trim()) {\n return toAbsoluteBase(`${ensureTrailingSlash(viteBase.trim())}wasm/`);\n }\n\n return toAbsoluteBase(\"/wasm/\");\n};\n\nconst WASM_BASE_URL = resolveWasmBase();\nconst WASM_JS_PATH = `${WASM_BASE_URL}zeldhash_miner_wasm.js`;\nconst WASM_BINARY_PATH = `${WASM_BASE_URL}zeldhash_miner_wasm_bg.wasm`;\n\nconst dynamicImport = async <T>(specifier: string): Promise<T> => {\n // Use eval to avoid bundlers rewriting the import path when we intentionally\n // want to fetch from the public /wasm/ directory.\n const importer = (0, eval)(\"s => import(s)\") as (s: string) => Promise<T>;\n return importer(specifier);\n};\n\nconst formatError = (err: unknown): string =>\n err instanceof Error ? err.message : String(err);\n\nconst loadModule = async (): Promise<WasmExports> => {\n installWebGpuLimitShim();\n\n let bindings: unknown;\n try {\n bindings = await dynamicImport<WasmExports>(/* @vite-ignore */ WASM_JS_PATH);\n } catch (err) {\n throw new Error(\n `Failed to import WASM bundle (${WASM_JS_PATH}). ` +\n `Did you run ./scripts/build-wasm.sh? (${formatError(err)})`\n );\n }\n\n const init = (bindings as { default?: unknown }).default;\n if (typeof init !== \"function\") {\n throw new Error(\"WASM init function is missing from the bundle.\");\n }\n\n try {\n const wasmUrl = new URL(WASM_BINARY_PATH, import.meta.url);\n await init({ module_or_path: wasmUrl });\n } catch (err) {\n throw new Error(\n `Failed to initialize WASM bundle: ${formatError(err)}`\n );\n }\n\n const typedBindings = bindings as WasmExports;\n try {\n typedBindings.init_panic_hook?.();\n } catch {\n /* ignore optional panic hook failures */\n }\n\n return typedBindings;\n};\n\nexport const loadWasm = async (): Promise<WasmExports> => {\n if (wasmModule) return wasmModule;\n if (!wasmInitPromise) {\n wasmInitPromise = loadModule()\n .then((mod) => {\n wasmModule = mod;\n return mod;\n })\n .catch((err) => {\n wasmInitPromise = null;\n throw err;\n });\n }\n return wasmInitPromise;\n};\n\nexport const getWasm = (): WasmExports | null => wasmModule;\n\nexport const resetWasm = (): void => {\n wasmModule = null;\n wasmInitPromise = null;\n};\n\n","const MAX_U64 = (1n << 64n) - 1n;\n\nexport interface NonceSegment {\n start: bigint;\n size: number;\n nonceLength: number;\n}\n\nexport const nonceLength = (nonce: bigint): number => {\n if (nonce < 0n) {\n throw new Error(\"nonce must be non-negative\");\n }\n\n if (nonce === 0n) return 1;\n\n let len = 0;\n let value = nonce;\n while (value > 0n) {\n len += 1;\n value >>= 8n;\n }\n return len;\n};\n\nexport const cborNonceLength = (nonce: bigint): number => {\n if (nonce < 0n) {\n throw new Error(\"nonce must be non-negative\");\n }\n\n if (nonce <= 23n) return 1;\n if (nonce <= 0xffn) return 2;\n if (nonce <= 0xffffn) return 3;\n if (nonce <= 0xffff_ffffn) return 5;\n if (nonce <= MAX_U64) return 9;\n\n throw new Error(\"nonce range exceeds u64\");\n};\n\nconst maxValueForLength = (len: number): bigint => {\n if (!Number.isInteger(len) || len <= 0 || len > 8) {\n throw new Error(\"nonceLength must be between 1 and 8\");\n }\n return (1n << BigInt(len * 8)) - 1n;\n};\n\nconst maxValueForCborLength = (len: number): bigint => {\n switch (len) {\n case 1:\n return 23n;\n case 2:\n return 0xffn;\n case 3:\n return 0xffffn;\n case 5:\n return 0xffff_ffffn;\n case 9:\n return MAX_U64;\n default:\n throw new Error(\"cbor nonceLength must be one of 1, 2, 3, 5, 9\");\n }\n};\n\nexport const splitNonceSegments = (startNonce: bigint, span: number): NonceSegment[] => {\n if (startNonce < 0n) {\n throw new Error(\"startNonce must be non-negative\");\n }\n if (!Number.isInteger(span) || span <= 0) {\n throw new Error(\"batchSize must be a positive integer\");\n }\n\n const end = startNonce + BigInt(span - 1);\n if (end > MAX_U64) {\n throw new Error(\"nonce range exceeds u64\");\n }\n\n const segments: NonceSegment[] = [];\n let current = startNonce;\n\n while (current <= end) {\n const len = nonceLength(current);\n const maxForLen = maxValueForLength(len);\n const segmentEnd = end < maxForLen ? end : maxForLen;\n const segmentSize = segmentEnd - current + 1n;\n\n if (segmentSize > BigInt(Number.MAX_SAFE_INTEGER)) {\n throw new Error(\"segment size exceeds safe integer range\");\n }\n\n segments.push({\n start: current,\n size: Number(segmentSize),\n nonceLength: len,\n });\n\n if (segmentEnd === end) {\n break;\n }\n\n current = segmentEnd + 1n;\n }\n\n return segments;\n};\n\nexport const maxNonceForLength = (len: number): bigint => maxValueForLength(len);\n\nexport const splitNonceSegmentsCbor = (startNonce: bigint, span: number): NonceSegment[] => {\n if (startNonce < 0n) {\n throw new Error(\"startNonce must be non-negative\");\n }\n if (!Number.isInteger(span) || span <= 0) {\n throw new Error(\"batchSize must be a positive integer\");\n }\n\n const end = startNonce + BigInt(span - 1);\n if (end > MAX_U64) {\n throw new Error(\"nonce range exceeds u64\");\n }\n\n const segments: NonceSegment[] = [];\n let current = startNonce;\n\n while (current <= end) {\n const len = cborNonceLength(current);\n const maxForLen = maxValueForCborLength(len);\n const segmentEnd = end < maxForLen ? end : maxForLen;\n const segmentSize = segmentEnd - current + 1n;\n\n if (segmentSize > BigInt(Number.MAX_SAFE_INTEGER)) {\n throw new Error(\"segment size exceeds safe integer range\");\n }\n\n segments.push({\n start: current,\n size: Number(segmentSize),\n nonceLength: len,\n });\n\n if (segmentEnd === end) {\n break;\n }\n\n current = segmentEnd + 1n;\n }\n\n return segments;\n};\n\n\n","import type {\n MiningTemplate,\n Network,\n TxInput,\n TxOutput,\n ValidationResult,\n WasmExports,\n} from \"./types\";\nimport { ZeldMinerErrorCode } from \"./types\";\nimport { createMinerError, toZeldMinerError } from \"./errors\";\nimport { loadWasm } from \"./wasm\";\nimport { cborNonceLength, nonceLength } from \"./nonce\";\n\nconst TXID_REGEX = /^[0-9a-fA-F]{64}$/;\nconst HEX_REGEX = /^[0-9a-fA-F]+$/;\nconst MAX_U64 = (1n << 64n) - 1n;\nconst MAX_U32 = 0xffff_ffff;\n\nconst dustLimitForAddress = (addressType?: \"p2wpkh\" | \"p2tr\"): number => {\n if (addressType === \"p2tr\") return 330;\n if (addressType === \"p2wpkh\") return 310;\n return 546; // conservative fallback for unexpected types\n};\n\nconst formatError = (err: unknown): string =>\n err instanceof Error ? err.message : String(err);\n\nconst includesAny = (haystack: string, needles: string[]): boolean =>\n needles.some((needle) => haystack.includes(needle));\n\nconst mapWasmError = (\n err: unknown,\n context: string,\n details?: Record<string, unknown>\n): never => {\n const message = formatError(err);\n const normalized = message.toLowerCase();\n\n if (includesAny(normalized, [\"insufficient funds\", \"insufficient_funds\"])) {\n throw createMinerError(\n ZeldMinerErrorCode.INSUFFICIENT_FUNDS,\n \"Insufficient funds for outputs and fee\",\n { ...details, cause: message }\n );\n }\n\n if (includesAny(normalized, [\"change would be dust\", \"output amount below dust limit\", \"dust\"])) {\n throw createMinerError(\n ZeldMinerErrorCode.DUST_OUTPUT,\n \"Change would be dust\",\n { ...details, cause: message }\n );\n }\n\n throw toZeldMinerError(err, ZeldMinerErrorCode.WORKER_ERROR, {\n ...details,\n context,\n });\n};\n\nconst normalizeNetwork = (network: Network): Network =>\n network === \"signet\" ? \"testnet\" : network;\n\nexport class TransactionBuilder {\n private readonly network: Network;\n private readonly satsPerVbyte: number;\n\n constructor(network: Network, satsPerVbyte: number) {\n if (!Number.isFinite(satsPerVbyte) || satsPerVbyte <= 0) {\n throw createMinerError(\n ZeldMinerErrorCode.INVALID_INPUT,\n \"satsPerVbyte must be a positive number\",\n { field: \"satsPerVbyte\" }\n );\n }\n this.network = network;\n this.satsPerVbyte = satsPerVbyte;\n }\n\n private async getWasm(): Promise<WasmExports> {\n return loadWasm();\n }\n\n private assertNonceRange(startNonce: bigint, batchSize: number, useCborNonce: boolean): number {\n if (startNonce < 0n || startNonce > MAX_U64) {\n throw createMinerError(\n ZeldMinerErrorCode.INVALID_INPUT,\n \"startNonce must be between 0 and 2^64 - 1\",\n { startNonce }\n );\n }\n if (!Number.isInteger(batchSize) || batchSize <= 0 || batchSize > MAX_U32) {\n throw createMinerError(\n ZeldMinerErrorCode.INVALID_INPUT,\n \"batchSize must be a positive 32-bit integer\",\n { batchSize }\n );\n }\n\n const lastNonce = startNonce + BigInt(batchSize - 1);\n if (lastNonce > MAX_U64) {\n throw createMinerError(\n ZeldMinerErrorCode.INVALID_INPUT,\n \"nonce range exceeds u64\",\n { startNonce, batchSize }\n );\n }\n\n const lengthFn = useCborNonce ? cborNonceLength : nonceLength;\n const startLen = lengthFn(startNonce);\n const lastLen = lengthFn(lastNonce);\n if (startLen !== lastLen) {\n const boundaryLabel = useCborNonce ? \"CBOR length\" : \"byte-length\";\n throw createMinerError(\n ZeldMinerErrorCode.INVALID_INPUT,\n `nonce range crosses ${boundaryLabel} boundary; reduce batch size`,\n { startNonce, batchSize }\n );\n }\n\n return startLen;\n }\n\n private cloneInputs(inputs: TxInput[]): TxInput[] {\n return inputs.map((input) => ({ ...input }));\n }\n\n private cloneOutputs(outputs: TxOutput[]): TxOutput[] {\n return outputs.map((output) => ({ ...output }));\n }\n\n private validateInputs(inputs: TxInput[]): void {\n if (!Array.isArray(inputs) || inputs.length === 0) {\n throw createMinerError(\n ZeldMinerErrorCode.INVALID_INPUT,\n \"At least one input is required\"\n );\n }\n\n inputs.forEach((input, idx) => {\n if (!TXID_REGEX.test(input.txid)) {\n throw createMinerError(\n ZeldMinerErrorCode.INVALID_INPUT,\n `inputs[${idx}].txid must be a 64-character hex`,\n { index: idx }\n );\n }\n if (!Number.isInteger(input.vout) || input.vout < 0) {\n throw createMinerError(\n ZeldMinerErrorCode.INVALID_INPUT,\n `inputs[${idx}].vout must be a non-negative integer`,\n { index: idx }\n );\n }\n if (!Number.isInteger(input.amount) || input.amount <= 0) {\n throw createMinerError(\n ZeldMinerErrorCode.INVALID_INPUT,\n `inputs[${idx}].amount must be a positive integer`,\n { index: idx }\n );\n }\n if (\n typeof input.scriptPubKey !== \"string\" ||\n input.scriptPubKey.length === 0 ||\n input.scriptPubKey.length % 2 !== 0 ||\n !HEX_REGEX.test(input.scriptPubKey)\n ) {\n throw createMinerError(\n ZeldMinerErrorCode.INVALID_INPUT,\n `inputs[${idx}].scriptPubKey must be valid hex`,\n { index: idx }\n );\n }\n });\n }\n\n private validateAddressResult(\n validation: ValidationResult,\n idx: number,\n network: Network\n ): void {\n if (!validation.ok) {\n const reason = validation.error ?? \"invalid address\";\n const normalized = reason.toLowerCase();\n const code = normalized.includes(\"unsupported\")\n ? ZeldMinerErrorCode.UNSUPPORTED_ADDRESS_TYPE\n : ZeldMinerErrorCode.INVALID_ADDRESS;\n\n throw createMinerError(\n code,\n `outputs[${idx}].address is invalid (${reason})`,\n { index: idx }\n );\n }\n\n if (\n validation.network &&\n validation.network !== network &&\n !(validation.network === \"testnet\" && network === \"signet\")\n ) {\n throw createMinerError(\n ZeldMinerErrorCode.INVALID_ADDRESS,\n `outputs[${idx}].address network mismatch`,\n { index: idx }\n );\n }\n\n if (\n validation.addressType &&\n validation.addressType !== \"p2tr\" &&\n validation.addressType !== \"p2wpkh\"\n ) {\n throw createMinerError(\n ZeldMinerErrorCode.UNSUPPORTED_ADDRESS_TYPE,\n `outputs[${idx}].address uses an unsupported type`,\n { index: idx, addressType: validation.addressType }\n );\n }\n }\n\n private async validateOutputs(outputs: TxOutput[]): Promise<void> {\n if (!Array.isArray(outputs) || outputs.length === 0) {\n throw createMinerError(\n ZeldMinerErrorCode.INVALID_INPUT,\n \"At least one output is required\"\n );\n }\n\n const changeCount = outputs.filter((o) => o.change).length;\n if (changeCount !== 1) {\n const code =\n changeCount === 0\n ? ZeldMinerErrorCode.NO_CHANGE_OUTPUT\n : ZeldMinerErrorCode.MULTIPLE_CHANGE_OUTPUTS;\n throw createMinerError(\n code,\n \"Exactly one change output is required\",\n { changeCount }\n );\n }\n\n const wasm = await this.getWasm();\n const net = normalizeNetwork(this.network);\n\n outputs.forEach((output, idx) => {\n const validation = wasm.validate_address(output.address, net);\n this.validateAddressResult(validation, idx, net);\n const dustLimit = dustLimitForAddress(validation.addressType);\n\n if (output.change) {\n if (\n output.amount !== undefined &&\n (!Number.isInteger(output.amount) || output.amount < 0)\n ) {\n throw createMinerError(\n ZeldMinerErrorCode.INVALID_INPUT,\n `outputs[${idx}].amount must be a non-negative integer when provided`,\n { index: idx }\n );\n }\n } else {\n if (\n !Number.isInteger(output.amount) ||\n (output.amount as number) < dustLimit\n ) {\n throw createMinerError(\n ZeldMinerErrorCode.DUST_OUTPUT,\n `outputs[${idx}].amount must be at least ${dustLimit} sats`,\n { index: idx, addressType: validation.addressType }\n );\n }\n }\n });\n }\n\n private validateDistribution(\n outputs: TxOutput[],\n distribution?: bigint[]\n ): bigint[] | undefined {\n if (distribution === undefined) return undefined;\n if (!Array.isArray(distribution)) {\n throw createMinerError(\n ZeldMinerErrorCode.INVALID_INPUT,\n \"distribution must be an array of bigint values\"\n );\n }\n\n if (distribution.length !== outputs.length) {\n throw createMinerError(\n ZeldMinerErrorCode.INVALID_INPUT,\n \"distribution length must match number of outputs\",\n { expected: outputs.length, actual: distribution.length }\n );\n }\n\n return distribution.map((value, idx) => {\n if (typeof value !== \"bigint\") {\n throw createMinerError(\n ZeldMinerErrorCode.INVALID_INPUT,\n \"distribution values must be bigint\",\n { index: idx }\n );\n }\n if (value < 0n) {\n throw createMinerError(\n ZeldMinerErrorCode.INVALID_INPUT,\n \"distribution values must be non-negative\",\n { index: idx }\n );\n }\n return value;\n });\n }\n\n async buildMiningTemplate(params: {\n inputs: TxInput[];\n outputs: TxOutput[];\n startNonce: bigint;\n batchSize: number;\n distribution?: bigint[];\n }): Promise<MiningTemplate & { nonceLength: number }> {\n const { inputs, outputs, startNonce, batchSize, distribution } = params;\n this.validateInputs(inputs);\n await this.validateOutputs(outputs);\n const validatedDistribution = this.validateDistribution(outputs, distribution);\n const useCborNonce = Array.isArray(validatedDistribution);\n const nonceLength = this.assertNonceRange(startNonce, batchSize, useCborNonce);\n\n const wasm = await this.getWasm();\n try {\n const template = wasm.build_mining_template(\n this.cloneInputs(inputs),\n this.cloneOutputs(outputs),\n normalizeNetwork(this.network),\n BigInt(this.satsPerVbyte),\n startNonce,\n batchSize,\n validatedDistribution ?? null\n );\n\n if (\n !template ||\n !(template.prefix instanceof Uint8Array) ||\n !(template.suffix instanceof Uint8Array)\n ) {\n throw createMinerError(\n ZeldMinerErrorCode.WORKER_ERROR,\n \"WASM returned an invalid mining template\"\n );\n }\n\n const templateUsesCbor = template.useCborNonce ?? useCborNonce;\n return { ...template, nonceLength, useCborNonce: templateUsesCbor };\n } catch (err) {\n throw mapWasmError(err, \"build_mining_template\", {\n startNonce,\n batchSize,\n distribution: validatedDistribution,\n });\n }\n }\n\n async buildPsbt(params: {\n inputs: TxInput[];\n outputs: TxOutput[];\n nonce: bigint;\n distribution?: bigint[];\n }): Promise<string> {\n const { inputs, outputs, nonce, distribution } = params;\n this.validateInputs(inputs);\n await this.validateOutputs(outputs);\n const validatedDistribution = this.validateDistribution(outputs, distribution);\n\n if (nonce < 0n || nonce > MAX_U64) {\n throw createMinerError(\n ZeldMinerErrorCode.INVALID_INPUT,\n \"nonce must be between 0 and 2^64 - 1\",\n { nonce }\n );\n }\n\n const wasm = await this.getWasm();\n try {\n return wasm.build_psbt(\n this.cloneInputs(inputs),\n this.cloneOutputs(outputs),\n normalizeNetwork(this.network),\n BigInt(this.satsPerVbyte),\n nonce,\n validatedDistribution ?? null\n );\n } catch (err) {\n throw mapWasmError(err, \"build_psbt\", { nonce, distribution: validatedDistribution });\n }\n }\n}\n\n","import { MiningCoordinator } from \"./coordinator\";\nimport { TransactionBuilder } from \"./builder\";\nimport { splitNonceSegments, splitNonceSegmentsCbor } from \"./nonce\";\nimport type {\n MineParams,\n MineResult,\n ProgressStats,\n WorkerMode,\n ZeldMinerOptions,\n} from \"./types\";\nimport { ZeldMinerErrorCode } from \"./types\";\nimport { createMinerError, toZeldMinerError, ZeldMinerError } from \"./errors\";\n\ntype ZeldMinerEvent = \"progress\" | \"found\" | \"error\" | \"stopped\";\n\ntype ZeldMinerEventMap = {\n progress: ProgressStats;\n found: MineResult;\n error: ZeldMinerError;\n stopped: void;\n};\n\nconst MIN_TARGET_ZEROS = 1;\nconst MAX_TARGET_ZEROS = 32;\n\nexport class ZeldMiner {\n private readonly options: ZeldMinerOptions;\n private readonly builder: TransactionBuilder;\n private coordinator: MiningCoordinator | null = null;\n private readonly listeners: {\n [K in ZeldMinerEvent]: Set<(payload: ZeldMinerEventMap[K]) => void>;\n };\n private state: \"idle\" | \"running\" | \"paused\" = \"idle\";\n private stopRequested = false;\n\n constructor(options: ZeldMinerOptions) {\n this.options = this.validateOptions(options);\n this.builder = new TransactionBuilder(options.network, options.satsPerVbyte);\n this.listeners = {\n progress: new Set(),\n found: new Set(),\n error: new Set(),\n stopped: new Set(),\n };\n }\n\n on<K extends ZeldMinerEvent>(\n event: K,\n handler: (payload: ZeldMinerEventMap[K]) => void\n ): void {\n this.listeners[event].add(handler);\n }\n\n off<K extends ZeldMinerEvent>(\n event: K,\n handler: (payload: ZeldMinerEventMap[K]) => void\n ): void {\n this.listeners[event].delete(handler);\n }\n\n private emit<K extends ZeldMinerEvent>(\n event: K,\n payload: ZeldMinerEventMap[K]\n ): void {\n this.listeners[event].forEach((handler) => handler(payload));\n }\n\n private validateOptions(options: ZeldMinerOptions): ZeldMinerOptions {\n if (!Number.isInteger(options.batchSize) || options.batchSize <= 0) {\n throw createMinerError(\n ZeldMinerErrorCode.INVALID_INPUT,\n \"batchSize must be a positive integer\",\n { field: \"batchSize\" }\n );\n }\n if (!Number.isInteger(options.workerThreads) || options.workerThreads <= 0) {\n throw createMinerError(\n ZeldMinerErrorCode.INVALID_INPUT,\n \"workerThreads must be a positive integer\",\n { field: \"workerThreads\" }\n );\n }\n if (!Number.isFinite(options.satsPerVbyte) || options.satsPerVbyte <= 0) {\n throw createMinerError(\n ZeldMinerErrorCode.INVALID_INPUT,\n \"satsPerVbyte must be a positive number\",\n { field: \"satsPerVbyte\" }\n );\n }\n return options;\n }\n\n private async selectBackend(): Promise<WorkerMode> {\n if (!this.options.useWebGPU) return \"cpu\";\n if (typeof navigator === \"undefined\") return \"cpu\";\n\n const gpu = (navigator as Navigator & { gpu?: { requestAdapter?: () => Promise<unknown> } }).gpu;\n const adapter =\n typeof gpu?.requestAdapter === \"function\" ? await gpu.requestAdapter() : null;\n return adapter ? \"gpu\" : \"cpu\";\n }\n\n private clearCoordinatorHandlers(\n coordinator: MiningCoordinator,\n handlers: Partial<Record<ZeldMinerEvent, (...args: any[]) => void>>\n ): void {\n if (handlers.progress) {\n coordinator.off(\"progress\", handlers.progress as (payload: ProgressStats) => void);\n }\n if (handlers.found) {\n coordinator.off(\"found\", handlers.found as (payload: MineResult) => void);\n }\n if (handlers.error) {\n coordinator.off(\"error\", handlers.error as (payload: Error) => void);\n }\n if (handlers.stopped) {\n coordinator.off(\"stopped\", handlers.stopped as (payload: void) => void);\n }\n }\n\n async mineTransaction(params: MineParams): Promise<MineResult> {\n if (this.state === \"running\") {\n throw createMinerError(\n ZeldMinerErrorCode.INVALID_INPUT,\n \"Mining is already in progress\"\n );\n }\n\n if (\n !Number.isInteger(params.targetZeros) ||\n params.targetZeros < MIN_TARGET_ZEROS ||\n params.targetZeros > MAX_TARGET_ZEROS\n ) {\n throw createMinerError(\n ZeldMinerErrorCode.INVALID_INPUT,\n `targetZeros must be an integer between ${MIN_TARGET_ZEROS} and ${MAX_TARGET_ZEROS}`,\n { targetZeros: params.targetZeros }\n );\n }\n\n const startNonce = params.startNonce ?? 0n;\n const batchSize = params.batchSize ?? this.options.batchSize;\n\n if (!Number.isInteger(batchSize) || batchSize <= 0) {\n throw createMinerError(\n ZeldMinerErrorCode.INVALID_INPUT,\n \"batchSize must be a positive integer\",\n { batchSize }\n );\n }\n\n if (params.signal?.aborted) {\n throw createMinerError(\n ZeldMinerErrorCode.MINING_ABORTED,\n \"Abort signal already triggered\"\n );\n }\n\n const distribution = params.distribution;\n const useCborNonce = Boolean(distribution && distribution.length > 0);\n let firstSegmentSize = batchSize;\n try {\n const [firstSegment] = useCborNonce\n ? splitNonceSegmentsCbor(startNonce, batchSize)\n : splitNonceSegments(startNonce, batchSize);\n if (!firstSegment) {\n throw createMinerError(\n ZeldMinerErrorCode.INVALID_INPUT,\n \"Failed to compute nonce segments\",\n { startNonce, batchSize }\n );\n }\n firstSegmentSize = firstSegment.size;\n } catch (err) {\n throw createMinerError(\n ZeldMinerErrorCode.INVALID_INPUT,\n err instanceof Error ? err.message : String(err),\n { startNonce, batchSize }\n );\n }\n\n const mode = await this.selectBackend();\n const template = await this.builder.buildMiningTemplate({\n inputs: params.inputs,\n outputs: params.outputs,\n startNonce,\n batchSize: firstSegmentSize,\n distribution,\n });\n\n const network =\n this.options.network === \"signet\" ? \"testnet\" : this.options.network;\n const coordinator = new MiningCoordinator({\n mode,\n batchSize,\n workerThreads: this.options.workerThreads,\n });\n\n this.coordinator = coordinator;\n this.state = \"running\";\n this.stopRequested = false;\n\n return new Promise<MineResult>((resolve, reject) => {\n let settled = false;\n\n const cleanup = (): void => {\n this.clearCoordinatorHandlers(coordinator, handlers);\n if (this.coordinator === coordinator) {\n this.coordinator = null;\n }\n this.state = \"idle\";\n this.stopRequested = false;\n };\n\n const resolveOnce = (value: MineResult): void => {\n if (settled) return;\n settled = true;\n cleanup();\n resolve(value);\n };\n\n const rejectOnce = (err: unknown): void => {\n if (settled) return;\n settled = true;\n const error = toZeldMinerError(err);\n cleanup();\n this.emit(\"error\", error);\n reject(error);\n };\n\n const handlers: Partial<Record<ZeldMinerEvent, (...args: any[]) => void>> = {};\n\n handlers.progress = (stats: ProgressStats): void => {\n this.emit(\"progress\", stats);\n };\n\n handlers.found = async (result: MineResult): Promise<void> => {\n try {\n const psbt = await this.builder.buildPsbt({\n inputs: params.inputs,\n outputs: params.outputs,\n nonce: result.nonce,\n distribution,\n });\n\n const finalResult: MineResult = { ...result, psbt };\n this.emit(\"found\", finalResult);\n resolveOnce(finalResult);\n } catch (err) {\n rejectOnce(err);\n }\n };\n\n handlers.error = (err: Error): void => {\n rejectOnce(err);\n };\n\n handlers.stopped = (): void => {\n if (settled) return;\n const reason = this.stopRequested\n ? createMinerError(\n ZeldMinerErrorCode.MINING_ABORTED,\n \"Mining stopped by caller\"\n )\n : createMinerError(\n ZeldMinerErrorCode.MINING_ABORTED,\n \"Mining halted\"\n );\n this.emit(\"stopped\", undefined as unknown as void);\n rejectOnce(reason);\n };\n\n coordinator.on(\"progress\", handlers.progress);\n coordinator.on(\"found\", handlers.found);\n coordinator.on(\"error\", handlers.error);\n coordinator.on(\"stopped\", handlers.stopped);\n\n coordinator\n .start({\n inputs: params.inputs,\n outputs: params.outputs,\n network,\n satsPerVbyte: this.options.satsPerVbyte,\n template,\n targetZeros: params.targetZeros,\n startNonce,\n signal: params.signal,\n distribution,\n })\n .catch(rejectOnce);\n });\n }\n\n stop(): void {\n if (!this.coordinator) return;\n this.stopRequested = true;\n this.coordinator.stop();\n }\n\n pause(): void {\n if (this.state !== \"running\" || !this.coordinator) return;\n this.state = \"paused\";\n this.coordinator.pause();\n }\n\n async resume(): Promise<void> {\n if (this.state !== \"paused\" || !this.coordinator) return;\n this.state = \"running\";\n await this.coordinator.resume();\n }\n}\n\nexport type {\n ZeldMinerOptions,\n MineParams,\n MineResult,\n ProgressStats,\n Network,\n TxInput,\n TxOutput,\n} from \"./types\";\nexport { ZeldMinerErrorCode } from \"./types\";\nexport { TransactionBuilder } from \"./builder\";\nexport { MiningCoordinator } from \"./coordinator\";\nexport {\n ZeldMinerError,\n createMinerError,\n toZeldMinerError,\n} from \"./errors\";\n\n"],"names":["ZeldMinerErrorCode","formatUnknownError","err","ZeldMinerError","code","message","details","toZeldMinerError","fallbackCode","createMinerError","safeBigIntToNumber","value","max","MiningCoordinator","options","event","handler","payload","origin","readySignals","workerUrl","i","worker","state","resolve","handleReady","initMessage","attempts","total","duration","hashRate","adjustedResult","hashesProcessed","elapsedMs","progress","lastNonce","batchAdvance","params","inputs","outputs","network","satsPerVbyte","template","targetZeros","distribution","stride","idx","workerStart","wasmModule","wasmInitPromise","webGpuLimitShimInstalled","installWebGpuLimitShim","adapterProto","requestDevice","descriptor","limits","supported","key","ensureTrailingSlash","toAbsoluteBase","base","trimmed","resolveWasmBase","globalBase","envBase","__vite_import_meta_env__","viteBase","WASM_BASE_URL","WASM_JS_PATH","WASM_BINARY_PATH","dynamicImport","specifier","formatError","loadModule","bindings","init","wasmUrl","typedBindings","loadWasm","mod","MAX_U64","nonceLength","nonce","len","cborNonceLength","maxValueForLength","maxValueForCborLength","splitNonceSegments","startNonce","span","end","segments","current","maxForLen","segmentEnd","segmentSize","splitNonceSegmentsCbor","TXID_REGEX","HEX_REGEX","MAX_U32","dustLimitForAddress","addressType","includesAny","haystack","needles","needle","mapWasmError","context","normalized","normalizeNetwork","TransactionBuilder","batchSize","useCborNonce","lengthFn","startLen","lastLen","boundaryLabel","input","output","validation","reason","changeCount","o","wasm","net","dustLimit","validatedDistribution","templateUsesCbor","MIN_TARGET_ZEROS","MAX_TARGET_ZEROS","ZeldMiner","gpu","coordinator","handlers","firstSegmentSize","firstSegment","mode","reject","settled","cleanup","resolveOnce","rejectOnce","error","stats","result","psbt","finalResult"],"mappings":"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"}
1
+ {"version":3,"file":"index.js","sources":["../src/errors.ts","../src/coordinator.ts","../src/builder.ts","../src/index.ts"],"sourcesContent":["import { ZeldMinerErrorCode, type ZeldMinerErrorDetails } from \"./types\";\n\nconst formatUnknownError = (err: unknown): string =>\n err instanceof Error ? err.message : String(err);\n\nexport class ZeldMinerError extends Error {\n readonly code: ZeldMinerErrorCode;\n readonly details?: ZeldMinerErrorDetails;\n\n constructor(\n code: ZeldMinerErrorCode,\n message: string,\n details?: ZeldMinerErrorDetails\n ) {\n super(message);\n this.name = \"ZeldMinerError\";\n this.code = code;\n this.details = details;\n }\n}\n\nexport const toZeldMinerError = (\n err: unknown,\n fallbackCode: ZeldMinerErrorCode = ZeldMinerErrorCode.WORKER_ERROR,\n details?: ZeldMinerErrorDetails\n): ZeldMinerError => {\n if (err instanceof ZeldMinerError) {\n return err;\n }\n\n const message = formatUnknownError(err);\n return new ZeldMinerError(fallbackCode, message, details);\n};\n\nexport const createMinerError = (\n code: ZeldMinerErrorCode,\n message: string,\n details?: ZeldMinerErrorDetails\n): ZeldMinerError => new ZeldMinerError(code, message, details);\n\n","import type {\n MineResult,\n MiningCoordinatorOptions,\n ProgressEvent as MiningProgressEvent,\n Network,\n TxInput,\n TxOutput,\n WorkerMessage,\n WorkerMode,\n WorkerResponse,\n WorkerTemplate,\n} from \"./types\";\nimport { ZeldMinerErrorCode } from \"./types\";\nimport { ZeldMinerError } from \"./errors\";\n\ntype CoordinatorEvent = \"ready\" | \"progress\" | \"found\" | \"error\" | \"stopped\";\n\ntype CoordinatorEventMap = {\n ready: void;\n progress: MiningProgressEvent;\n found: MineResult;\n error: ZeldMinerError;\n stopped: void;\n};\n\ntype CoordinatorListener<K extends CoordinatorEvent> = (\n payload: CoordinatorEventMap[K]\n) => void;\n\ninterface WorkerState {\n worker: Worker;\n hashesProcessed: bigint;\n hashRate: number;\n lastNonce?: bigint;\n nextNonce: bigint;\n processedBase: bigint;\n terminated?: boolean;\n}\n\ninterface StartParams {\n inputs: TxInput[];\n outputs: TxOutput[];\n network: Network;\n satsPerVbyte: number;\n template: WorkerTemplate;\n targetZeros: number;\n startNonce?: bigint;\n signal?: AbortSignal;\n distribution?: bigint[];\n}\n\nconst safeBigIntToNumber = (value: bigint): number => {\n const max = BigInt(Number.MAX_SAFE_INTEGER);\n if (value > max) return Number.MAX_SAFE_INTEGER;\n if (value < -max) return -Number.MAX_SAFE_INTEGER;\n return Number(value);\n};\n\nexport class MiningCoordinator {\n private readonly mode: WorkerMode;\n private readonly batchSize: number;\n private readonly workerCount: number;\n private readonly listeners: {\n [K in CoordinatorEvent]: Set<CoordinatorListener<K>>;\n };\n private readonly workers: WorkerState[] = [];\n private readonly readyPromise: Promise<void>;\n private readonly abortHandler: () => void;\n\n private stride: bigint;\n private cleanupExternalAbort = (): void => {\n if (this.externalAbort) {\n this.externalAbort.removeEventListener(\"abort\", this.abortHandler);\n this.externalAbort = undefined;\n }\n };\n private running = false;\n private paused = false;\n private startedAt: number | null = null;\n private txInputs?: TxInput[];\n private txOutputs?: TxOutput[];\n private txNetwork?: Network;\n private satsPerVbyte?: number;\n private template?: WorkerTemplate;\n private targetZeros?: number;\n private startNonce: bigint = 0n;\n private txDistribution?: bigint[];\n private externalAbort?: AbortSignal;\n private terminated = false;\n\n constructor(options: MiningCoordinatorOptions) {\n this.mode = options.mode;\n this.batchSize = options.batchSize;\n this.workerCount =\n this.mode === \"gpu\" ? 1 : Math.max(1, options.workerThreads);\n this.stride =\n this.mode === \"gpu\"\n ? BigInt(this.batchSize)\n : BigInt(this.batchSize) * BigInt(this.workerCount);\n\n this.listeners = {\n ready: new Set(),\n progress: new Set(),\n found: new Set(),\n error: new Set(),\n stopped: new Set(),\n };\n\n this.abortHandler = () => this.stop();\n this.readyPromise = this.spawnWorkers();\n }\n\n on<K extends CoordinatorEvent>(\n event: K,\n handler: CoordinatorListener<K>\n ): void {\n this.listeners[event].add(handler);\n }\n\n off<K extends CoordinatorEvent>(\n event: K,\n handler: CoordinatorListener<K>\n ): void {\n this.listeners[event].delete(handler);\n }\n\n private emit<K extends CoordinatorEvent>(\n event: K,\n payload: CoordinatorEventMap[K]\n ): void {\n this.listeners[event].forEach((handler) => handler(payload));\n }\n\n private resolveWorkerUrl(): string {\n const origin =\n typeof location !== \"undefined\" && location.origin\n ? location.origin\n : typeof self !== \"undefined\" &&\n (self as { location?: { origin?: string } }).location?.origin\n ? (self as { location?: { origin?: string } }).location!.origin\n : undefined;\n if (origin) return `${origin}/worker.js`;\n return new URL(\"./worker.ts\", import.meta.url).href;\n }\n\n private async spawnWorkers(): Promise<void> {\n const readySignals: Promise<void>[] = [];\n const workerUrl = this.resolveWorkerUrl();\n\n for (let i = 0; i < this.workerCount; i += 1) {\n const worker = new Worker(workerUrl, {\n type: \"module\",\n /* @vite-ignore */ name: `zeldminer-worker-${i}`,\n });\n\n const state: WorkerState = {\n worker,\n hashesProcessed: 0n,\n hashRate: 0,\n nextNonce: 0n,\n processedBase: 0n,\n };\n\n this.workers.push(state);\n\n readySignals.push(\n new Promise<void>((resolve) => {\n const handleReady = (event: MessageEvent<WorkerResponse>): void => {\n if (event.data.type === \"ready\") {\n worker.removeEventListener(\"message\", handleReady);\n resolve();\n }\n };\n\n worker.addEventListener(\"message\", handleReady);\n })\n );\n\n worker.addEventListener(\"message\", (event: MessageEvent<WorkerResponse>) =>\n this.handleWorkerMessage(state, event.data)\n );\n\n const initMessage: WorkerMessage = { type: \"init\", mode: this.mode };\n worker.postMessage(initMessage);\n }\n\n await Promise.all(readySignals);\n this.emit(\"ready\", undefined as unknown as void);\n }\n\n private handleWorkerMessage(\n state: WorkerState,\n message: WorkerResponse\n ): void {\n switch (message.type) {\n case \"ready\":\n break;\n case \"progress\":\n state.hashesProcessed = state.processedBase + message.hashesProcessed;\n state.hashRate = message.hashRate;\n state.lastNonce = message.lastNonce ?? state.lastNonce;\n this.emitProgress();\n break;\n case \"batch_complete\":\n state.lastNonce = message.lastNonce;\n state.nextNonce = this.computeNextNonce(message.lastNonce);\n break;\n case \"found\":\n state.hashesProcessed =\n state.processedBase + (message.hashesProcessed ?? state.hashesProcessed);\n state.hashRate = message.hashRate ?? state.hashRate;\n state.lastNonce = message.lastNonce ?? state.lastNonce;\n\n const attempts = this.workers.reduce(\n (total, worker) => total + worker.hashesProcessed,\n 0n\n );\n const duration = this.startedAt ? performance.now() - this.startedAt : 0;\n const hashRate =\n duration > 0\n ? safeBigIntToNumber(attempts) / (duration / 1000)\n : 0;\n\n const adjustedResult: MineResult = {\n ...message.result,\n attempts,\n duration,\n hashRate,\n };\n\n this.running = false;\n this.paused = false;\n this.terminateWorkers();\n this.cleanupExternalAbort();\n this.emit(\"found\", adjustedResult);\n break;\n case \"error\":\n this.running = false;\n this.paused = false;\n this.terminateWorkers();\n this.cleanupExternalAbort();\n this.emit(\n \"error\",\n new ZeldMinerError(\n message.code ?? ZeldMinerErrorCode.WORKER_ERROR,\n message.message,\n {\n workerId: message.workerId,\n ...(message.details ?? {}),\n }\n )\n );\n break;\n }\n }\n\n private emitProgress(): void {\n if (!this.running) return;\n\n const hashesProcessed = this.workers.reduce(\n (total, worker) => total + worker.hashesProcessed,\n 0n\n );\n const elapsedMs = this.startedAt ? performance.now() - this.startedAt : 0;\n // Derive rate from total attempts over wall-clock time to avoid overstating\n // throughput on multi-worker CPU runs.\n const hashRate =\n elapsedMs > 0\n ? safeBigIntToNumber(hashesProcessed) / (elapsedMs / 1000)\n : 0;\n\n const progress: MiningProgressEvent = {\n hashesProcessed,\n hashRate,\n elapsedMs,\n };\n\n this.emit(\"progress\", progress);\n }\n\n private computeNextNonce(lastNonce: bigint): bigint {\n const batchAdvance = this.stride - BigInt(this.batchSize);\n return lastNonce + 1n + batchAdvance;\n }\n\n private stopWorkers(): void {\n if (this.terminated) return;\n this.workers.forEach((state) =>\n state.worker.postMessage({ type: \"stop\" } satisfies WorkerMessage)\n );\n }\n\n private terminateWorkers(): void {\n if (this.terminated) return;\n this.stopWorkers();\n this.workers.forEach((state) => {\n try {\n state.worker.terminate();\n state.terminated = true;\n } catch {\n /* ignore termination errors */\n }\n });\n this.terminated = true;\n }\n\n async start(params: StartParams): Promise<void> {\n await this.readyPromise;\n this.stopWorkers();\n\n this.running = true;\n this.paused = false;\n this.startedAt = performance.now();\n this.txInputs = params.inputs;\n this.txOutputs = params.outputs;\n this.txNetwork = params.network === \"signet\" ? \"testnet\" : params.network;\n this.satsPerVbyte = params.satsPerVbyte;\n this.template = params.template;\n this.targetZeros = params.targetZeros;\n this.startNonce = params.startNonce ?? 0n;\n this.txDistribution = params.distribution;\n\n if (this.externalAbort && this.externalAbort !== params.signal) {\n this.cleanupExternalAbort();\n }\n\n if (params.signal) {\n this.externalAbort = params.signal;\n params.signal.addEventListener(\"abort\", this.abortHandler, { once: true });\n }\n\n if (\n !this.template ||\n !this.txInputs ||\n !this.txOutputs ||\n this.targetZeros === undefined ||\n this.satsPerVbyte === undefined ||\n !this.txNetwork\n ) {\n throw new Error(\"Mining parameters are missing\");\n }\n\n const inputs = this.txInputs;\n const outputs = this.txOutputs;\n const network = this.txNetwork;\n const satsPerVbyte = this.satsPerVbyte;\n const template = this.template;\n const targetZeros = this.targetZeros;\n const distribution = this.txDistribution;\n const stride = this.stride;\n\n this.workers.forEach((state, idx) => {\n state.processedBase = 0n;\n state.hashesProcessed = 0n;\n state.hashRate = 0;\n state.lastNonce = undefined;\n const workerStart =\n this.startNonce + BigInt(idx) * BigInt(this.batchSize);\n state.nextNonce = workerStart;\n\n const message: WorkerMessage = {\n type: \"mine\",\n inputs,\n outputs,\n network,\n satsPerVbyte,\n template,\n startNonce: workerStart,\n batchSize: this.batchSize,\n targetZeros,\n nonceStep: stride,\n distribution,\n };\n\n state.worker.postMessage(message);\n });\n }\n\n pause(): void {\n if (!this.running) return;\n this.running = false;\n this.paused = true;\n this.stopWorkers();\n }\n\n async resume(): Promise<void> {\n if (\n !this.paused ||\n !this.template ||\n !this.txInputs ||\n !this.txOutputs ||\n this.targetZeros === undefined ||\n this.satsPerVbyte === undefined ||\n !this.txNetwork\n )\n return;\n await this.readyPromise;\n\n this.running = true;\n this.paused = false;\n if (!this.startedAt) {\n this.startedAt = performance.now();\n }\n\n const inputs = this.txInputs;\n const outputs = this.txOutputs;\n const network = this.txNetwork;\n const satsPerVbyte = this.satsPerVbyte;\n const template = this.template;\n const targetZeros = this.targetZeros as number;\n const distribution = this.txDistribution;\n const stride = this.stride;\n\n this.workers.forEach((state, idx) => {\n state.processedBase = state.hashesProcessed;\n state.hashRate = 0;\n const workerStart =\n state.nextNonce ||\n this.startNonce + BigInt(idx) * BigInt(this.batchSize);\n\n const message: WorkerMessage = {\n type: \"mine\",\n inputs,\n outputs,\n network,\n satsPerVbyte,\n template,\n startNonce: workerStart,\n batchSize: this.batchSize,\n targetZeros,\n nonceStep: stride,\n distribution,\n };\n\n state.worker.postMessage(message);\n });\n }\n\n stop(): void {\n if (!this.running && !this.paused) return;\n this.running = false;\n this.paused = false;\n this.terminateWorkers();\n this.cleanupExternalAbort();\n this.emit(\"stopped\", undefined as unknown as void);\n }\n}\n\n","import type {\n MiningTemplate,\n Network,\n TxInput,\n TxOutput,\n ValidationResult,\n WasmExports,\n} from \"./types\";\nimport { ZeldMinerErrorCode } from \"./types\";\nimport { createMinerError, toZeldMinerError } from \"./errors\";\nimport { loadWasm } from \"./wasm\";\nimport { cborNonceLength, nonceLength } from \"./nonce\";\n\nconst TXID_REGEX = /^[0-9a-fA-F]{64}$/;\nconst HEX_REGEX = /^[0-9a-fA-F]+$/;\nconst MAX_U64 = (1n << 64n) - 1n;\nconst MAX_U32 = 0xffff_ffff;\n\nconst dustLimitForAddress = (addressType?: \"p2wpkh\" | \"p2tr\"): number => {\n if (addressType === \"p2tr\") return 330;\n if (addressType === \"p2wpkh\") return 310;\n return 546; // conservative fallback for unexpected types\n};\n\nconst formatError = (err: unknown): string =>\n err instanceof Error ? err.message : String(err);\n\nconst includesAny = (haystack: string, needles: string[]): boolean =>\n needles.some((needle) => haystack.includes(needle));\n\nconst mapWasmError = (\n err: unknown,\n context: string,\n details?: Record<string, unknown>\n): never => {\n const message = formatError(err);\n const normalized = message.toLowerCase();\n\n if (includesAny(normalized, [\"insufficient funds\", \"insufficient_funds\"])) {\n throw createMinerError(\n ZeldMinerErrorCode.INSUFFICIENT_FUNDS,\n \"Insufficient funds for outputs and fee\",\n { ...details, cause: message }\n );\n }\n\n if (includesAny(normalized, [\"change would be dust\", \"output amount below dust limit\", \"dust\"])) {\n throw createMinerError(\n ZeldMinerErrorCode.DUST_OUTPUT,\n \"Change would be dust\",\n { ...details, cause: message }\n );\n }\n\n throw toZeldMinerError(err, ZeldMinerErrorCode.WORKER_ERROR, {\n ...details,\n context,\n });\n};\n\nconst normalizeNetwork = (network: Network): Network =>\n network === \"signet\" ? \"testnet\" : network;\n\nexport class TransactionBuilder {\n private readonly network: Network;\n private readonly satsPerVbyte: number;\n\n constructor(network: Network, satsPerVbyte: number) {\n if (!Number.isFinite(satsPerVbyte) || satsPerVbyte <= 0) {\n throw createMinerError(\n ZeldMinerErrorCode.INVALID_INPUT,\n \"satsPerVbyte must be a positive number\",\n { field: \"satsPerVbyte\" }\n );\n }\n this.network = network;\n this.satsPerVbyte = satsPerVbyte;\n }\n\n private async getWasm(): Promise<WasmExports> {\n return loadWasm();\n }\n\n private assertNonceRange(startNonce: bigint, batchSize: number, useCborNonce: boolean): number {\n if (startNonce < 0n || startNonce > MAX_U64) {\n throw createMinerError(\n ZeldMinerErrorCode.INVALID_INPUT,\n \"startNonce must be between 0 and 2^64 - 1\",\n { startNonce }\n );\n }\n if (!Number.isInteger(batchSize) || batchSize <= 0 || batchSize > MAX_U32) {\n throw createMinerError(\n ZeldMinerErrorCode.INVALID_INPUT,\n \"batchSize must be a positive 32-bit integer\",\n { batchSize }\n );\n }\n\n const lastNonce = startNonce + BigInt(batchSize - 1);\n if (lastNonce > MAX_U64) {\n throw createMinerError(\n ZeldMinerErrorCode.INVALID_INPUT,\n \"nonce range exceeds u64\",\n { startNonce, batchSize }\n );\n }\n\n const lengthFn = useCborNonce ? cborNonceLength : nonceLength;\n const startLen = lengthFn(startNonce);\n const lastLen = lengthFn(lastNonce);\n if (startLen !== lastLen) {\n const boundaryLabel = useCborNonce ? \"CBOR length\" : \"byte-length\";\n throw createMinerError(\n ZeldMinerErrorCode.INVALID_INPUT,\n `nonce range crosses ${boundaryLabel} boundary; reduce batch size`,\n { startNonce, batchSize }\n );\n }\n\n return startLen;\n }\n\n private cloneInputs(inputs: TxInput[]): TxInput[] {\n return inputs.map((input) => ({ ...input }));\n }\n\n private cloneOutputs(outputs: TxOutput[]): TxOutput[] {\n return outputs.map((output) => ({ ...output }));\n }\n\n private validateInputs(inputs: TxInput[]): void {\n if (!Array.isArray(inputs) || inputs.length === 0) {\n throw createMinerError(\n ZeldMinerErrorCode.INVALID_INPUT,\n \"At least one input is required\"\n );\n }\n\n inputs.forEach((input, idx) => {\n if (!TXID_REGEX.test(input.txid)) {\n throw createMinerError(\n ZeldMinerErrorCode.INVALID_INPUT,\n `inputs[${idx}].txid must be a 64-character hex`,\n { index: idx }\n );\n }\n if (!Number.isInteger(input.vout) || input.vout < 0) {\n throw createMinerError(\n ZeldMinerErrorCode.INVALID_INPUT,\n `inputs[${idx}].vout must be a non-negative integer`,\n { index: idx }\n );\n }\n if (!Number.isInteger(input.amount) || input.amount <= 0) {\n throw createMinerError(\n ZeldMinerErrorCode.INVALID_INPUT,\n `inputs[${idx}].amount must be a positive integer`,\n { index: idx }\n );\n }\n if (\n typeof input.scriptPubKey !== \"string\" ||\n input.scriptPubKey.length === 0 ||\n input.scriptPubKey.length % 2 !== 0 ||\n !HEX_REGEX.test(input.scriptPubKey)\n ) {\n throw createMinerError(\n ZeldMinerErrorCode.INVALID_INPUT,\n `inputs[${idx}].scriptPubKey must be valid hex`,\n { index: idx }\n );\n }\n });\n }\n\n private validateAddressResult(\n validation: ValidationResult,\n idx: number,\n network: Network\n ): void {\n if (!validation.ok) {\n const reason = validation.error ?? \"invalid address\";\n const normalized = reason.toLowerCase();\n const code = normalized.includes(\"unsupported\")\n ? ZeldMinerErrorCode.UNSUPPORTED_ADDRESS_TYPE\n : ZeldMinerErrorCode.INVALID_ADDRESS;\n\n throw createMinerError(\n code,\n `outputs[${idx}].address is invalid (${reason})`,\n { index: idx }\n );\n }\n\n if (\n validation.network &&\n validation.network !== network &&\n !(validation.network === \"testnet\" && network === \"signet\")\n ) {\n throw createMinerError(\n ZeldMinerErrorCode.INVALID_ADDRESS,\n `outputs[${idx}].address network mismatch`,\n { index: idx }\n );\n }\n\n if (\n validation.addressType &&\n validation.addressType !== \"p2tr\" &&\n validation.addressType !== \"p2wpkh\"\n ) {\n throw createMinerError(\n ZeldMinerErrorCode.UNSUPPORTED_ADDRESS_TYPE,\n `outputs[${idx}].address uses an unsupported type`,\n { index: idx, addressType: validation.addressType }\n );\n }\n }\n\n private async validateOutputs(outputs: TxOutput[]): Promise<void> {\n if (!Array.isArray(outputs) || outputs.length === 0) {\n throw createMinerError(\n ZeldMinerErrorCode.INVALID_INPUT,\n \"At least one output is required\"\n );\n }\n\n const changeCount = outputs.filter((o) => o.change).length;\n if (changeCount !== 1) {\n const code =\n changeCount === 0\n ? ZeldMinerErrorCode.NO_CHANGE_OUTPUT\n : ZeldMinerErrorCode.MULTIPLE_CHANGE_OUTPUTS;\n throw createMinerError(\n code,\n \"Exactly one change output is required\",\n { changeCount }\n );\n }\n\n const wasm = await this.getWasm();\n const net = normalizeNetwork(this.network);\n\n outputs.forEach((output, idx) => {\n const validation = wasm.validate_address(output.address, net);\n this.validateAddressResult(validation, idx, net);\n const dustLimit = dustLimitForAddress(validation.addressType);\n\n if (output.change) {\n if (\n output.amount !== undefined &&\n (!Number.isInteger(output.amount) || output.amount < 0)\n ) {\n throw createMinerError(\n ZeldMinerErrorCode.INVALID_INPUT,\n `outputs[${idx}].amount must be a non-negative integer when provided`,\n { index: idx }\n );\n }\n } else {\n if (\n !Number.isInteger(output.amount) ||\n (output.amount as number) < dustLimit\n ) {\n throw createMinerError(\n ZeldMinerErrorCode.DUST_OUTPUT,\n `outputs[${idx}].amount must be at least ${dustLimit} sats`,\n { index: idx, addressType: validation.addressType }\n );\n }\n }\n });\n }\n\n private validateDistribution(\n outputs: TxOutput[],\n distribution?: bigint[]\n ): bigint[] | undefined {\n if (distribution === undefined) return undefined;\n if (!Array.isArray(distribution)) {\n throw createMinerError(\n ZeldMinerErrorCode.INVALID_INPUT,\n \"distribution must be an array of bigint values\"\n );\n }\n\n if (distribution.length !== outputs.length) {\n throw createMinerError(\n ZeldMinerErrorCode.INVALID_INPUT,\n \"distribution length must match number of outputs\",\n { expected: outputs.length, actual: distribution.length }\n );\n }\n\n return distribution.map((value, idx) => {\n if (typeof value !== \"bigint\") {\n throw createMinerError(\n ZeldMinerErrorCode.INVALID_INPUT,\n \"distribution values must be bigint\",\n { index: idx }\n );\n }\n if (value < 0n) {\n throw createMinerError(\n ZeldMinerErrorCode.INVALID_INPUT,\n \"distribution values must be non-negative\",\n { index: idx }\n );\n }\n return value;\n });\n }\n\n async buildMiningTemplate(params: {\n inputs: TxInput[];\n outputs: TxOutput[];\n startNonce: bigint;\n batchSize: number;\n distribution?: bigint[];\n }): Promise<MiningTemplate & { nonceLength: number }> {\n const { inputs, outputs, startNonce, batchSize, distribution } = params;\n this.validateInputs(inputs);\n await this.validateOutputs(outputs);\n const validatedDistribution = this.validateDistribution(outputs, distribution);\n const useCborNonce = Array.isArray(validatedDistribution);\n const nonceLength = this.assertNonceRange(startNonce, batchSize, useCborNonce);\n\n const wasm = await this.getWasm();\n try {\n const template = wasm.build_mining_template(\n this.cloneInputs(inputs),\n this.cloneOutputs(outputs),\n normalizeNetwork(this.network),\n BigInt(this.satsPerVbyte),\n startNonce,\n batchSize,\n validatedDistribution ?? null\n );\n\n if (\n !template ||\n !(template.prefix instanceof Uint8Array) ||\n !(template.suffix instanceof Uint8Array)\n ) {\n throw createMinerError(\n ZeldMinerErrorCode.WORKER_ERROR,\n \"WASM returned an invalid mining template\"\n );\n }\n\n const templateUsesCbor = template.useCborNonce ?? useCborNonce;\n return { ...template, nonceLength, useCborNonce: templateUsesCbor };\n } catch (err) {\n throw mapWasmError(err, \"build_mining_template\", {\n startNonce,\n batchSize,\n distribution: validatedDistribution,\n });\n }\n }\n\n async buildPsbt(params: {\n inputs: TxInput[];\n outputs: TxOutput[];\n nonce: bigint;\n distribution?: bigint[];\n }): Promise<string> {\n const { inputs, outputs, nonce, distribution } = params;\n this.validateInputs(inputs);\n await this.validateOutputs(outputs);\n const validatedDistribution = this.validateDistribution(outputs, distribution);\n\n if (nonce < 0n || nonce > MAX_U64) {\n throw createMinerError(\n ZeldMinerErrorCode.INVALID_INPUT,\n \"nonce must be between 0 and 2^64 - 1\",\n { nonce }\n );\n }\n\n const wasm = await this.getWasm();\n try {\n return wasm.build_psbt(\n this.cloneInputs(inputs),\n this.cloneOutputs(outputs),\n normalizeNetwork(this.network),\n BigInt(this.satsPerVbyte),\n nonce,\n validatedDistribution ?? null\n );\n } catch (err) {\n throw mapWasmError(err, \"build_psbt\", { nonce, distribution: validatedDistribution });\n }\n }\n}\n\n","import { MiningCoordinator } from \"./coordinator\";\nimport { TransactionBuilder } from \"./builder\";\nimport { splitNonceSegments, splitNonceSegmentsCbor } from \"./nonce\";\nimport type {\n MineParams,\n MineResult,\n ProgressStats,\n WorkerMode,\n ZeldMinerOptions,\n} from \"./types\";\nimport { ZeldMinerErrorCode } from \"./types\";\nimport { createMinerError, toZeldMinerError, ZeldMinerError } from \"./errors\";\n\ntype ZeldMinerEvent = \"progress\" | \"found\" | \"error\" | \"stopped\";\n\ntype ZeldMinerEventMap = {\n progress: ProgressStats;\n found: MineResult;\n error: ZeldMinerError;\n stopped: void;\n};\n\nconst MIN_TARGET_ZEROS = 1;\nconst MAX_TARGET_ZEROS = 32;\n\nexport class ZeldMiner {\n private readonly options: ZeldMinerOptions;\n private readonly builder: TransactionBuilder;\n private coordinator: MiningCoordinator | null = null;\n private readonly listeners: {\n [K in ZeldMinerEvent]: Set<(payload: ZeldMinerEventMap[K]) => void>;\n };\n private state: \"idle\" | \"running\" | \"paused\" = \"idle\";\n private stopRequested = false;\n\n constructor(options: ZeldMinerOptions) {\n this.options = this.validateOptions(options);\n this.builder = new TransactionBuilder(options.network, options.satsPerVbyte);\n this.listeners = {\n progress: new Set(),\n found: new Set(),\n error: new Set(),\n stopped: new Set(),\n };\n }\n\n on<K extends ZeldMinerEvent>(\n event: K,\n handler: (payload: ZeldMinerEventMap[K]) => void\n ): void {\n this.listeners[event].add(handler);\n }\n\n off<K extends ZeldMinerEvent>(\n event: K,\n handler: (payload: ZeldMinerEventMap[K]) => void\n ): void {\n this.listeners[event].delete(handler);\n }\n\n private emit<K extends ZeldMinerEvent>(\n event: K,\n payload: ZeldMinerEventMap[K]\n ): void {\n this.listeners[event].forEach((handler) => handler(payload));\n }\n\n private validateOptions(options: ZeldMinerOptions): ZeldMinerOptions {\n if (!Number.isInteger(options.batchSize) || options.batchSize <= 0) {\n throw createMinerError(\n ZeldMinerErrorCode.INVALID_INPUT,\n \"batchSize must be a positive integer\",\n { field: \"batchSize\" }\n );\n }\n if (!Number.isInteger(options.workerThreads) || options.workerThreads <= 0) {\n throw createMinerError(\n ZeldMinerErrorCode.INVALID_INPUT,\n \"workerThreads must be a positive integer\",\n { field: \"workerThreads\" }\n );\n }\n if (!Number.isFinite(options.satsPerVbyte) || options.satsPerVbyte <= 0) {\n throw createMinerError(\n ZeldMinerErrorCode.INVALID_INPUT,\n \"satsPerVbyte must be a positive number\",\n { field: \"satsPerVbyte\" }\n );\n }\n return options;\n }\n\n private async selectBackend(): Promise<WorkerMode> {\n if (!this.options.useWebGPU) return \"cpu\";\n if (typeof navigator === \"undefined\") return \"cpu\";\n\n const gpu = (navigator as Navigator & { gpu?: { requestAdapter?: () => Promise<unknown> } }).gpu;\n const adapter =\n typeof gpu?.requestAdapter === \"function\" ? await gpu.requestAdapter() : null;\n return adapter ? \"gpu\" : \"cpu\";\n }\n\n private clearCoordinatorHandlers(\n coordinator: MiningCoordinator,\n handlers: Partial<Record<ZeldMinerEvent, (...args: any[]) => void>>\n ): void {\n if (handlers.progress) {\n coordinator.off(\"progress\", handlers.progress as (payload: ProgressStats) => void);\n }\n if (handlers.found) {\n coordinator.off(\"found\", handlers.found as (payload: MineResult) => void);\n }\n if (handlers.error) {\n coordinator.off(\"error\", handlers.error as (payload: Error) => void);\n }\n if (handlers.stopped) {\n coordinator.off(\"stopped\", handlers.stopped as (payload: void) => void);\n }\n }\n\n async mineTransaction(params: MineParams): Promise<MineResult> {\n if (this.state === \"running\") {\n throw createMinerError(\n ZeldMinerErrorCode.INVALID_INPUT,\n \"Mining is already in progress\"\n );\n }\n\n if (\n !Number.isInteger(params.targetZeros) ||\n params.targetZeros < MIN_TARGET_ZEROS ||\n params.targetZeros > MAX_TARGET_ZEROS\n ) {\n throw createMinerError(\n ZeldMinerErrorCode.INVALID_INPUT,\n `targetZeros must be an integer between ${MIN_TARGET_ZEROS} and ${MAX_TARGET_ZEROS}`,\n { targetZeros: params.targetZeros }\n );\n }\n\n const startNonce = params.startNonce ?? 0n;\n const batchSize = params.batchSize ?? this.options.batchSize;\n\n if (!Number.isInteger(batchSize) || batchSize <= 0) {\n throw createMinerError(\n ZeldMinerErrorCode.INVALID_INPUT,\n \"batchSize must be a positive integer\",\n { batchSize }\n );\n }\n\n if (params.signal?.aborted) {\n throw createMinerError(\n ZeldMinerErrorCode.MINING_ABORTED,\n \"Abort signal already triggered\"\n );\n }\n\n const distribution = params.distribution;\n const useCborNonce = Boolean(distribution && distribution.length > 0);\n let firstSegmentSize = batchSize;\n try {\n const [firstSegment] = useCborNonce\n ? splitNonceSegmentsCbor(startNonce, batchSize)\n : splitNonceSegments(startNonce, batchSize);\n if (!firstSegment) {\n throw createMinerError(\n ZeldMinerErrorCode.INVALID_INPUT,\n \"Failed to compute nonce segments\",\n { startNonce, batchSize }\n );\n }\n firstSegmentSize = firstSegment.size;\n } catch (err) {\n throw createMinerError(\n ZeldMinerErrorCode.INVALID_INPUT,\n err instanceof Error ? err.message : String(err),\n { startNonce, batchSize }\n );\n }\n\n const mode = await this.selectBackend();\n const template = await this.builder.buildMiningTemplate({\n inputs: params.inputs,\n outputs: params.outputs,\n startNonce,\n batchSize: firstSegmentSize,\n distribution,\n });\n\n const network =\n this.options.network === \"signet\" ? \"testnet\" : this.options.network;\n const coordinator = new MiningCoordinator({\n mode,\n batchSize,\n workerThreads: this.options.workerThreads,\n });\n\n this.coordinator = coordinator;\n this.state = \"running\";\n this.stopRequested = false;\n\n return new Promise<MineResult>((resolve, reject) => {\n let settled = false;\n\n const cleanup = (): void => {\n this.clearCoordinatorHandlers(coordinator, handlers);\n if (this.coordinator === coordinator) {\n this.coordinator = null;\n }\n this.state = \"idle\";\n this.stopRequested = false;\n };\n\n const resolveOnce = (value: MineResult): void => {\n if (settled) return;\n settled = true;\n cleanup();\n resolve(value);\n };\n\n const rejectOnce = (err: unknown): void => {\n if (settled) return;\n settled = true;\n const error = toZeldMinerError(err);\n cleanup();\n this.emit(\"error\", error);\n reject(error);\n };\n\n const handlers: Partial<Record<ZeldMinerEvent, (...args: any[]) => void>> = {};\n\n handlers.progress = (stats: ProgressStats): void => {\n this.emit(\"progress\", stats);\n };\n\n handlers.found = async (result: MineResult): Promise<void> => {\n try {\n const psbt = await this.builder.buildPsbt({\n inputs: params.inputs,\n outputs: params.outputs,\n nonce: result.nonce,\n distribution,\n });\n\n const finalResult: MineResult = { ...result, psbt };\n this.emit(\"found\", finalResult);\n resolveOnce(finalResult);\n } catch (err) {\n rejectOnce(err);\n }\n };\n\n handlers.error = (err: Error): void => {\n rejectOnce(err);\n };\n\n handlers.stopped = (): void => {\n if (settled) return;\n const reason = this.stopRequested\n ? createMinerError(\n ZeldMinerErrorCode.MINING_ABORTED,\n \"Mining stopped by caller\"\n )\n : createMinerError(\n ZeldMinerErrorCode.MINING_ABORTED,\n \"Mining halted\"\n );\n this.emit(\"stopped\", undefined as unknown as void);\n rejectOnce(reason);\n };\n\n coordinator.on(\"progress\", handlers.progress);\n coordinator.on(\"found\", handlers.found);\n coordinator.on(\"error\", handlers.error);\n coordinator.on(\"stopped\", handlers.stopped);\n\n coordinator\n .start({\n inputs: params.inputs,\n outputs: params.outputs,\n network,\n satsPerVbyte: this.options.satsPerVbyte,\n template,\n targetZeros: params.targetZeros,\n startNonce,\n signal: params.signal,\n distribution,\n })\n .catch(rejectOnce);\n });\n }\n\n stop(): void {\n if (!this.coordinator) return;\n this.stopRequested = true;\n this.coordinator.stop();\n }\n\n pause(): void {\n if (this.state !== \"running\" || !this.coordinator) return;\n this.state = \"paused\";\n this.coordinator.pause();\n }\n\n async resume(): Promise<void> {\n if (this.state !== \"paused\" || !this.coordinator) return;\n this.state = \"running\";\n await this.coordinator.resume();\n }\n}\n\nexport type {\n ZeldMinerOptions,\n MineParams,\n MineResult,\n ProgressStats,\n Network,\n TxInput,\n TxOutput,\n} from \"./types\";\nexport { ZeldMinerErrorCode } from \"./types\";\nexport { TransactionBuilder } from \"./builder\";\nexport { MiningCoordinator } from \"./coordinator\";\nexport {\n ZeldMinerError,\n createMinerError,\n toZeldMinerError,\n} from \"./errors\";\n\n"],"names":["formatUnknownError","err","ZeldMinerError","code","message","details","toZeldMinerError","fallbackCode","ZeldMinerErrorCode","createMinerError","safeBigIntToNumber","value","max","MiningCoordinator","options","event","handler","payload","origin","readySignals","workerUrl","i","worker","state","resolve","handleReady","initMessage","attempts","total","duration","hashRate","adjustedResult","hashesProcessed","elapsedMs","progress","lastNonce","batchAdvance","params","inputs","outputs","network","satsPerVbyte","template","targetZeros","distribution","stride","idx","workerStart","TXID_REGEX","HEX_REGEX","MAX_U64","MAX_U32","dustLimitForAddress","addressType","formatError","includesAny","haystack","needles","needle","mapWasmError","context","normalized","normalizeNetwork","TransactionBuilder","loadWasm","startNonce","batchSize","useCborNonce","lengthFn","cborNonceLength","nonceLength","startLen","lastLen","boundaryLabel","input","output","validation","reason","changeCount","o","wasm","net","dustLimit","validatedDistribution","templateUsesCbor","nonce","MIN_TARGET_ZEROS","MAX_TARGET_ZEROS","ZeldMiner","gpu","coordinator","handlers","firstSegmentSize","firstSegment","splitNonceSegmentsCbor","splitNonceSegments","mode","reject","settled","cleanup","resolveOnce","rejectOnce","error","stats","result","psbt","finalResult"],"mappings":"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"}
package/dist/nonce.js ADDED
@@ -0,0 +1,158 @@
1
+ var S = /* @__PURE__ */ ((e) => (e.INVALID_ADDRESS = "INVALID_ADDRESS", e.UNSUPPORTED_ADDRESS_TYPE = "UNSUPPORTED_ADDRESS_TYPE", e.INSUFFICIENT_FUNDS = "INSUFFICIENT_FUNDS", e.NO_CHANGE_OUTPUT = "NO_CHANGE_OUTPUT", e.MULTIPLE_CHANGE_OUTPUTS = "MULTIPLE_CHANGE_OUTPUTS", e.INVALID_INPUT = "INVALID_INPUT", e.WEBGPU_NOT_AVAILABLE = "WEBGPU_NOT_AVAILABLE", e.WORKER_ERROR = "WORKER_ERROR", e.MINING_ABORTED = "MINING_ABORTED", e.DUST_OUTPUT = "DUST_OUTPUT", e))(S || {});
2
+ const A = {};
3
+ if (typeof globalThis.__ZELDMINER_WASM_BASE__ > "u")
4
+ try {
5
+ const e = typeof window < "u" && window.location?.origin ? window.location.origin : typeof self < "u" && self.location?.origin ? self.location.origin : "http://localhost";
6
+ globalThis.__ZELDMINER_WASM_BASE__ = new URL("/wasm/", e).href;
7
+ } catch {
8
+ globalThis.__ZELDMINER_WASM_BASE__ = "/wasm/";
9
+ }
10
+ let m = null, f = null, w = !1;
11
+ const I = () => {
12
+ if (w) return;
13
+ w = !0;
14
+ const e = globalThis.GPUAdapter?.prototype, t = e?.requestDevice;
15
+ !e || typeof t != "function" || (e.requestDevice = function(r) {
16
+ if (r?.requiredLimits && typeof this.limits == "object") {
17
+ const i = r.requiredLimits, o = this.limits;
18
+ for (const s of Object.keys(i))
19
+ (!(s in o) || o[s] === void 0) && delete i[s];
20
+ }
21
+ return t.call(this, r);
22
+ });
23
+ }, l = (e) => e.endsWith("/") ? e : `${e}/`, u = (e) => {
24
+ const t = e.trim();
25
+ return t && (typeof window < "u" && typeof window.location?.origin == "string" ? l(new URL(t, window.location.origin).href) : l(new URL(t, import.meta.url).href));
26
+ }, N = () => {
27
+ const e = globalThis.__ZELDMINER_WASM_BASE__;
28
+ if (typeof e == "string" && e.trim())
29
+ return u(e);
30
+ const t = A?.VITE_ZELDMINER_WASM_BASE;
31
+ if (typeof t == "string" && t.trim())
32
+ return u(t);
33
+ const n = "./";
34
+ return n.trim() ? u(`${l(n.trim())}wasm/`) : u("/wasm/");
35
+ }, E = N(), h = `${E}zeldhash_miner_wasm.js`, b = `${E}zeldhash_miner_wasm_bg.wasm`, T = async (e) => (0, eval)("s => import(s)")(e), g = (e) => e instanceof Error ? e.message : String(e), L = async () => {
36
+ I();
37
+ let e;
38
+ try {
39
+ e = await T(
40
+ /* @vite-ignore */
41
+ h
42
+ );
43
+ } catch (r) {
44
+ throw new Error(
45
+ `Failed to import WASM bundle (${h}). Did you run ./scripts/build-wasm.sh? (${g(r)})`
46
+ );
47
+ }
48
+ const t = e.default;
49
+ if (typeof t != "function")
50
+ throw new Error("WASM init function is missing from the bundle.");
51
+ try {
52
+ const r = new URL(b, import.meta.url);
53
+ await t({ module_or_path: r });
54
+ } catch (r) {
55
+ throw new Error(
56
+ `Failed to initialize WASM bundle: ${g(r)}`
57
+ );
58
+ }
59
+ const n = e;
60
+ try {
61
+ n.init_panic_hook?.();
62
+ } catch {
63
+ }
64
+ return n;
65
+ }, D = async () => m || (f || (f = L().then((e) => (m = e, e)).catch((e) => {
66
+ throw f = null, e;
67
+ })), f), _ = (1n << 64n) - 1n, U = (e) => {
68
+ if (e < 0n)
69
+ throw new Error("nonce must be non-negative");
70
+ if (e === 0n) return 1;
71
+ let t = 0, n = e;
72
+ for (; n > 0n; )
73
+ t += 1, n >>= 8n;
74
+ return t;
75
+ }, d = (e) => {
76
+ if (e < 0n)
77
+ throw new Error("nonce must be non-negative");
78
+ if (e <= 23n) return 1;
79
+ if (e <= 0xffn) return 2;
80
+ if (e <= 0xffffn) return 3;
81
+ if (e <= 0xffffffffn) return 5;
82
+ if (e <= _) return 9;
83
+ throw new Error("nonce range exceeds u64");
84
+ }, p = (e) => {
85
+ if (!Number.isInteger(e) || e <= 0 || e > 8)
86
+ throw new Error("nonceLength must be between 1 and 8");
87
+ return (1n << BigInt(e * 8)) - 1n;
88
+ }, R = (e) => {
89
+ switch (e) {
90
+ case 1:
91
+ return 23n;
92
+ case 2:
93
+ return 0xffn;
94
+ case 3:
95
+ return 0xffffn;
96
+ case 5:
97
+ return 0xffffffffn;
98
+ case 9:
99
+ return _;
100
+ default:
101
+ throw new Error("cbor nonceLength must be one of 1, 2, 3, 5, 9");
102
+ }
103
+ }, B = (e, t) => {
104
+ if (e < 0n)
105
+ throw new Error("startNonce must be non-negative");
106
+ if (!Number.isInteger(t) || t <= 0)
107
+ throw new Error("batchSize must be a positive integer");
108
+ const n = e + BigInt(t - 1);
109
+ if (n > _)
110
+ throw new Error("nonce range exceeds u64");
111
+ const r = [];
112
+ let i = e;
113
+ for (; i <= n; ) {
114
+ const o = U(i), s = p(o), a = n < s ? n : s, c = a - i + 1n;
115
+ if (c > BigInt(Number.MAX_SAFE_INTEGER))
116
+ throw new Error("segment size exceeds safe integer range");
117
+ if (r.push({
118
+ start: i,
119
+ size: Number(c),
120
+ nonceLength: o
121
+ }), a === n)
122
+ break;
123
+ i = a + 1n;
124
+ }
125
+ return r;
126
+ }, y = (e, t) => {
127
+ if (e < 0n)
128
+ throw new Error("startNonce must be non-negative");
129
+ if (!Number.isInteger(t) || t <= 0)
130
+ throw new Error("batchSize must be a positive integer");
131
+ const n = e + BigInt(t - 1);
132
+ if (n > _)
133
+ throw new Error("nonce range exceeds u64");
134
+ const r = [];
135
+ let i = e;
136
+ for (; i <= n; ) {
137
+ const o = d(i), s = R(o), a = n < s ? n : s, c = a - i + 1n;
138
+ if (c > BigInt(Number.MAX_SAFE_INTEGER))
139
+ throw new Error("segment size exceeds safe integer range");
140
+ if (r.push({
141
+ start: i,
142
+ size: Number(c),
143
+ nonceLength: o
144
+ }), a === n)
145
+ break;
146
+ i = a + 1n;
147
+ }
148
+ return r;
149
+ };
150
+ export {
151
+ S as Z,
152
+ B as a,
153
+ d as c,
154
+ D as l,
155
+ U as n,
156
+ y as s
157
+ };
158
+ //# sourceMappingURL=nonce.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"nonce.js","sources":["../src/types.ts","../src/wasm.ts","../src/nonce.ts"],"sourcesContent":["export enum ZeldMinerErrorCode {\n INVALID_ADDRESS = \"INVALID_ADDRESS\",\n UNSUPPORTED_ADDRESS_TYPE = \"UNSUPPORTED_ADDRESS_TYPE\",\n INSUFFICIENT_FUNDS = \"INSUFFICIENT_FUNDS\",\n NO_CHANGE_OUTPUT = \"NO_CHANGE_OUTPUT\",\n MULTIPLE_CHANGE_OUTPUTS = \"MULTIPLE_CHANGE_OUTPUTS\",\n INVALID_INPUT = \"INVALID_INPUT\",\n WEBGPU_NOT_AVAILABLE = \"WEBGPU_NOT_AVAILABLE\",\n WORKER_ERROR = \"WORKER_ERROR\",\n MINING_ABORTED = \"MINING_ABORTED\",\n DUST_OUTPUT = \"DUST_OUTPUT\",\n}\n\nexport type ZeldMinerErrorDetails = Record<string, unknown>;\n\nexport type Network = \"mainnet\" | \"testnet\" | \"signet\" | \"regtest\";\n\nexport interface ZeldMinerOptions {\n network: Network;\n batchSize: number;\n useWebGPU: boolean;\n workerThreads: number;\n satsPerVbyte: number;\n}\n\nexport interface MiningCoordinatorOptions {\n mode: WorkerMode;\n batchSize: number;\n workerThreads: number;\n}\n\nexport interface TxInput {\n txid: string;\n vout: number;\n scriptPubKey: string;\n amount: number;\n}\n\nexport interface TxOutput {\n address: string;\n amount?: number;\n change: boolean;\n}\n\nexport interface MineResult {\n psbt: string;\n txid: string;\n nonce: bigint;\n attempts: bigint;\n duration: number;\n hashRate: number;\n}\n\nexport interface ProgressEvent {\n hashesProcessed: bigint;\n hashRate: number;\n elapsedMs?: number;\n lastNonce?: bigint;\n workerId?: string;\n}\n\nexport type WorkerMode = \"cpu\" | \"gpu\";\n\nexport type WorkerMessage =\n | { type: \"init\"; mode: WorkerMode }\n | {\n type: \"mine\";\n inputs: TxInput[];\n outputs: TxOutput[];\n network: Network;\n satsPerVbyte: number;\n template: WorkerTemplate;\n startNonce: bigint;\n batchSize: number;\n targetZeros: number;\n nonceStep?: bigint;\n distribution?: bigint[];\n }\n | { type: \"stop\" };\n\nexport type WorkerResponse =\n | { type: \"ready\"; workerId?: string }\n | {\n type: \"progress\";\n hashesProcessed: bigint;\n hashRate: number;\n lastNonce?: bigint;\n workerId?: string;\n }\n | {\n type: \"found\";\n result: MineResult;\n hashesProcessed?: bigint;\n hashRate?: number;\n lastNonce?: bigint;\n workerId?: string;\n }\n | { type: \"batch_complete\"; lastNonce: bigint; workerId?: string }\n | {\n type: \"error\";\n message: string;\n code?: ZeldMinerErrorCode;\n details?: ZeldMinerErrorDetails;\n workerId?: string;\n };\n\nexport interface MiningState {\n mode: WorkerMode;\n running: boolean;\n startNonce: bigint;\n nextNonce: bigint;\n batchSize: number;\n targetZeros: number;\n hashesProcessed: bigint;\n lastHashRate?: number;\n}\n\nexport interface MiningTemplate {\n prefix: Uint8Array;\n suffix: Uint8Array;\n useCborNonce?: boolean;\n}\n\nexport interface WorkerTemplate extends MiningTemplate {\n nonceLength: number;\n useCborNonce?: boolean;\n}\n\nexport interface ValidationResult {\n ok: boolean;\n error?: string;\n addressType?: \"p2wpkh\" | \"p2tr\";\n network?: Network;\n}\n\nexport interface WasmExports {\n init_panic_hook: () => void;\n mine_batch_wasm: (\n txPrefix: Uint8Array,\n txSuffix: Uint8Array,\n startNonce: bigint,\n batchSize: number,\n targetZeros: number,\n useCborNonce?: boolean\n ) => ValidationResult | MineResult | null;\n mine_batch_gpu?: (\n txPrefix: Uint8Array,\n txSuffix: Uint8Array,\n startNonce: bigint,\n batchSize: number,\n targetZeros: number,\n useCborNonce?: boolean\n ) => Promise<ValidationResult | MineResult | null>;\n mine_range_wasm: (\n inputs: TxInput[],\n outputs: TxOutput[],\n network: Network,\n satsPerVbyte: bigint,\n startNonce: bigint,\n batchSize: number,\n targetZeros: number,\n distribution?: bigint[] | null\n ) => ValidationResult | MineResult | null;\n mine_range_gpu?: (\n inputs: TxInput[],\n outputs: TxOutput[],\n network: Network,\n satsPerVbyte: bigint,\n startNonce: bigint,\n batchSize: number,\n targetZeros: number,\n distribution?: bigint[] | null\n ) => Promise<ValidationResult | MineResult | null>;\n validate_address: (addr: string, network: Network) => ValidationResult;\n build_psbt: (\n inputs: TxInput[],\n outputs: TxOutput[],\n network: Network,\n satsPerVbyte: bigint,\n nonce: bigint,\n distribution?: bigint[] | null\n ) => string;\n build_mining_template: (\n inputs: TxInput[],\n outputs: TxOutput[],\n network: Network,\n satsPerVbyte: bigint,\n startNonce: bigint,\n batchSize: number,\n distribution?: bigint[] | null\n ) => MiningTemplate;\n compute_txid: (txBytes: Uint8Array) => string;\n init_gpu?: () => Promise<unknown>;\n calibrate_batch_size?: () => Promise<number>;\n}\n\nexport type ProgressStats = ProgressEvent;\n\nexport interface MineParams {\n inputs: TxInput[];\n outputs: TxOutput[];\n targetZeros: number;\n startNonce?: bigint;\n batchSize?: number;\n signal?: AbortSignal;\n distribution?: bigint[];\n}\n\n","import type { WasmExports } from \"./types\";\n\n// Bootstrap: define a sensible default for WASM base path so consumers don't\n// need to patch the library. In Vite/dev the import.meta.url points inside\n// node_modules which is not served, so we derive a public path from the window\n// origin when available.\nif (typeof (globalThis as { __ZELDMINER_WASM_BASE__?: unknown }).__ZELDMINER_WASM_BASE__ === \"undefined\") {\n try {\n const origin =\n typeof window !== \"undefined\" && window.location?.origin\n ? window.location.origin\n : typeof self !== \"undefined\" && (self as { location?: { origin?: string } }).location?.origin\n ? (self as { location?: { origin?: string } }).location!.origin\n : \"http://localhost\";\n (globalThis as { __ZELDMINER_WASM_BASE__?: string }).__ZELDMINER_WASM_BASE__ = new URL(\"/wasm/\", origin!).href;\n } catch {\n (globalThis as { __ZELDMINER_WASM_BASE__?: string }).__ZELDMINER_WASM_BASE__ = \"/wasm/\";\n }\n}\n\nlet wasmModule: WasmExports | null = null;\nlet wasmInitPromise: Promise<WasmExports> | null = null;\n\n// Chrome/Dawn can reject requestDevice when optional limits (e.g.\n// maxInterStageShaderComponents) are present but unsupported. Strip any\n// limit keys the adapter doesn't expose to keep WebGPU initialization\n// portable across browser versions.\nlet webGpuLimitShimInstalled = false;\nconst installWebGpuLimitShim = (): void => {\n if (webGpuLimitShimInstalled) return;\n webGpuLimitShimInstalled = true;\n\n const adapterProto = (globalThis as typeof globalThis & { GPUAdapter?: { prototype?: unknown } })\n .GPUAdapter?.prototype as GPUAdapter | undefined;\n const requestDevice = adapterProto?.requestDevice;\n if (!adapterProto || typeof requestDevice !== \"function\") return;\n\n adapterProto.requestDevice = function patchedRequestDevice(\n this: GPUAdapter,\n descriptor?: GPUDeviceDescriptor\n ): Promise<GPUDevice> {\n if (descriptor?.requiredLimits && typeof this.limits === \"object\") {\n const limits = descriptor.requiredLimits as Record<string, unknown>;\n const supported = this.limits as unknown as Record<string, unknown>;\n for (const key of Object.keys(limits)) {\n if (!(key in supported) || supported[key] === undefined) {\n delete limits[key];\n }\n }\n }\n return requestDevice.call(this, descriptor);\n };\n};\n\nconst ensureTrailingSlash = (value: string): string =>\n value.endsWith(\"/\") ? value : `${value}/`;\n\nconst toAbsoluteBase = (base: string): string => {\n const trimmed = base.trim();\n if (!trimmed) return trimmed;\n // Use window origin when available so Vite treats it as an external URL and does not try to transform public assets.\n if (typeof window !== \"undefined\" && typeof window.location?.origin === \"string\") {\n return ensureTrailingSlash(new URL(trimmed, window.location.origin).href);\n }\n return ensureTrailingSlash(new URL(trimmed, import.meta.url).href);\n};\n\nconst resolveWasmBase = (): string => {\n const globalBase = (globalThis as { __ZELDMINER_WASM_BASE__?: unknown })\n .__ZELDMINER_WASM_BASE__;\n if (typeof globalBase === \"string\" && globalBase.trim()) {\n return toAbsoluteBase(globalBase);\n }\n\n const envBase = (import.meta as ImportMeta & { env?: Record<string, unknown> })\n .env?.VITE_ZELDMINER_WASM_BASE;\n if (typeof envBase === \"string\" && envBase.trim()) {\n return toAbsoluteBase(envBase);\n }\n\n const viteBase = (import.meta as ImportMeta & { env?: Record<string, unknown> }).env?.BASE_URL;\n if (typeof viteBase === \"string\" && viteBase.trim()) {\n return toAbsoluteBase(`${ensureTrailingSlash(viteBase.trim())}wasm/`);\n }\n\n return toAbsoluteBase(\"/wasm/\");\n};\n\nconst WASM_BASE_URL = resolveWasmBase();\nconst WASM_JS_PATH = `${WASM_BASE_URL}zeldhash_miner_wasm.js`;\nconst WASM_BINARY_PATH = `${WASM_BASE_URL}zeldhash_miner_wasm_bg.wasm`;\n\nconst dynamicImport = async <T>(specifier: string): Promise<T> => {\n // Use eval to avoid bundlers rewriting the import path when we intentionally\n // want to fetch from the public /wasm/ directory.\n const importer = (0, eval)(\"s => import(s)\") as (s: string) => Promise<T>;\n return importer(specifier);\n};\n\nconst formatError = (err: unknown): string =>\n err instanceof Error ? err.message : String(err);\n\nconst loadModule = async (): Promise<WasmExports> => {\n installWebGpuLimitShim();\n\n let bindings: unknown;\n try {\n bindings = await dynamicImport<WasmExports>(/* @vite-ignore */ WASM_JS_PATH);\n } catch (err) {\n throw new Error(\n `Failed to import WASM bundle (${WASM_JS_PATH}). ` +\n `Did you run ./scripts/build-wasm.sh? (${formatError(err)})`\n );\n }\n\n const init = (bindings as { default?: unknown }).default;\n if (typeof init !== \"function\") {\n throw new Error(\"WASM init function is missing from the bundle.\");\n }\n\n try {\n const wasmUrl = new URL(WASM_BINARY_PATH, import.meta.url);\n await init({ module_or_path: wasmUrl });\n } catch (err) {\n throw new Error(\n `Failed to initialize WASM bundle: ${formatError(err)}`\n );\n }\n\n const typedBindings = bindings as WasmExports;\n try {\n typedBindings.init_panic_hook?.();\n } catch {\n /* ignore optional panic hook failures */\n }\n\n return typedBindings;\n};\n\nexport const loadWasm = async (): Promise<WasmExports> => {\n if (wasmModule) return wasmModule;\n if (!wasmInitPromise) {\n wasmInitPromise = loadModule()\n .then((mod) => {\n wasmModule = mod;\n return mod;\n })\n .catch((err) => {\n wasmInitPromise = null;\n throw err;\n });\n }\n return wasmInitPromise;\n};\n\nexport const getWasm = (): WasmExports | null => wasmModule;\n\nexport const resetWasm = (): void => {\n wasmModule = null;\n wasmInitPromise = null;\n};\n\n","const MAX_U64 = (1n << 64n) - 1n;\n\nexport interface NonceSegment {\n start: bigint;\n size: number;\n nonceLength: number;\n}\n\nexport const nonceLength = (nonce: bigint): number => {\n if (nonce < 0n) {\n throw new Error(\"nonce must be non-negative\");\n }\n\n if (nonce === 0n) return 1;\n\n let len = 0;\n let value = nonce;\n while (value > 0n) {\n len += 1;\n value >>= 8n;\n }\n return len;\n};\n\nexport const cborNonceLength = (nonce: bigint): number => {\n if (nonce < 0n) {\n throw new Error(\"nonce must be non-negative\");\n }\n\n if (nonce <= 23n) return 1;\n if (nonce <= 0xffn) return 2;\n if (nonce <= 0xffffn) return 3;\n if (nonce <= 0xffff_ffffn) return 5;\n if (nonce <= MAX_U64) return 9;\n\n throw new Error(\"nonce range exceeds u64\");\n};\n\nconst maxValueForLength = (len: number): bigint => {\n if (!Number.isInteger(len) || len <= 0 || len > 8) {\n throw new Error(\"nonceLength must be between 1 and 8\");\n }\n return (1n << BigInt(len * 8)) - 1n;\n};\n\nconst maxValueForCborLength = (len: number): bigint => {\n switch (len) {\n case 1:\n return 23n;\n case 2:\n return 0xffn;\n case 3:\n return 0xffffn;\n case 5:\n return 0xffff_ffffn;\n case 9:\n return MAX_U64;\n default:\n throw new Error(\"cbor nonceLength must be one of 1, 2, 3, 5, 9\");\n }\n};\n\nexport const splitNonceSegments = (startNonce: bigint, span: number): NonceSegment[] => {\n if (startNonce < 0n) {\n throw new Error(\"startNonce must be non-negative\");\n }\n if (!Number.isInteger(span) || span <= 0) {\n throw new Error(\"batchSize must be a positive integer\");\n }\n\n const end = startNonce + BigInt(span - 1);\n if (end > MAX_U64) {\n throw new Error(\"nonce range exceeds u64\");\n }\n\n const segments: NonceSegment[] = [];\n let current = startNonce;\n\n while (current <= end) {\n const len = nonceLength(current);\n const maxForLen = maxValueForLength(len);\n const segmentEnd = end < maxForLen ? end : maxForLen;\n const segmentSize = segmentEnd - current + 1n;\n\n if (segmentSize > BigInt(Number.MAX_SAFE_INTEGER)) {\n throw new Error(\"segment size exceeds safe integer range\");\n }\n\n segments.push({\n start: current,\n size: Number(segmentSize),\n nonceLength: len,\n });\n\n if (segmentEnd === end) {\n break;\n }\n\n current = segmentEnd + 1n;\n }\n\n return segments;\n};\n\nexport const maxNonceForLength = (len: number): bigint => maxValueForLength(len);\n\nexport const splitNonceSegmentsCbor = (startNonce: bigint, span: number): NonceSegment[] => {\n if (startNonce < 0n) {\n throw new Error(\"startNonce must be non-negative\");\n }\n if (!Number.isInteger(span) || span <= 0) {\n throw new Error(\"batchSize must be a positive integer\");\n }\n\n const end = startNonce + BigInt(span - 1);\n if (end > MAX_U64) {\n throw new Error(\"nonce range exceeds u64\");\n }\n\n const segments: NonceSegment[] = [];\n let current = startNonce;\n\n while (current <= end) {\n const len = cborNonceLength(current);\n const maxForLen = maxValueForCborLength(len);\n const segmentEnd = end < maxForLen ? end : maxForLen;\n const segmentSize = segmentEnd - current + 1n;\n\n if (segmentSize > BigInt(Number.MAX_SAFE_INTEGER)) {\n throw new Error(\"segment size exceeds safe integer range\");\n }\n\n segments.push({\n start: current,\n size: Number(segmentSize),\n nonceLength: len,\n });\n\n if (segmentEnd === end) {\n break;\n }\n\n current = segmentEnd + 1n;\n }\n\n return segments;\n};\n\n\n"],"names":["ZeldMinerErrorCode","origin","wasmModule","wasmInitPromise","webGpuLimitShimInstalled","installWebGpuLimitShim","adapterProto","requestDevice","descriptor","limits","supported","key","ensureTrailingSlash","value","toAbsoluteBase","base","trimmed","resolveWasmBase","globalBase","envBase","__vite_import_meta_env__","viteBase","WASM_BASE_URL","WASM_JS_PATH","WASM_BINARY_PATH","dynamicImport","specifier","formatError","err","loadModule","bindings","init","wasmUrl","typedBindings","loadWasm","mod","MAX_U64","nonceLength","nonce","len","cborNonceLength","maxValueForLength","maxValueForCborLength","splitNonceSegments","startNonce","span","end","segments","current","maxForLen","segmentEnd","segmentSize","splitNonceSegmentsCbor"],"mappings":"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"}
@@ -0,0 +1 @@
1
+ export { }