vitest-pool-assemblyscript 0.10.1 → 0.12.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (39) hide show
  1. package/README.md +3 -3
  2. package/assembly/compare.ts +44 -38
  3. package/assembly/expect.ts +13 -3
  4. package/assembly/utils.ts +4 -1
  5. package/dist/{compile-runner-BGHM_85g.mjs → compile-runner-BGYyiUEf.mjs} +3 -3
  6. package/dist/{compile-runner-BGHM_85g.mjs.map → compile-runner-BGYyiUEf.mjs.map} +1 -1
  7. package/dist/config/index-v3.d.mts +1 -1
  8. package/dist/config/index.d.mts +2 -2
  9. package/dist/config/index.mjs +2 -2
  10. package/dist/coverage-provider/index.mjs.map +1 -1
  11. package/dist/index-internal.d.mts +1 -1
  12. package/dist/index-v3.mjs +1 -1
  13. package/dist/index.d.mts +1 -1
  14. package/dist/index.mjs +2 -2
  15. package/dist/{load-user-imports-Bcx9NOt9.mjs → load-user-imports-iAQUMKAi.mjs} +2 -2
  16. package/dist/{load-user-imports-Bcx9NOt9.mjs.map → load-user-imports-iAQUMKAi.mjs.map} +1 -1
  17. package/dist/{pool-runner-init-CCvnKt5o.d.mts → pool-runner-init-CPQL6pVL.d.mts} +2 -2
  18. package/dist/pool-runner-init-CPQL6pVL.d.mts.map +1 -0
  19. package/dist/{pool-runner-init-DjRCbiX-.mjs → pool-runner-init-CSJt3dHv.mjs} +2 -2
  20. package/dist/{pool-runner-init-DjRCbiX-.mjs.map → pool-runner-init-CSJt3dHv.mjs.map} +1 -1
  21. package/dist/pool-thread/compile-worker-thread.d.mts +1 -1
  22. package/dist/pool-thread/compile-worker-thread.mjs +3 -3
  23. package/dist/pool-thread/test-worker-thread.d.mts +1 -1
  24. package/dist/pool-thread/test-worker-thread.mjs +3 -3
  25. package/dist/pool-thread/v3-tinypool-thread.d.mts +1 -1
  26. package/dist/pool-thread/v3-tinypool-thread.mjs +4 -4
  27. package/dist/{test-runner-BeP8ClnE.mjs → test-runner-BDcm8vg0.mjs} +3 -3
  28. package/dist/{test-runner-BeP8ClnE.mjs.map → test-runner-BDcm8vg0.mjs.map} +1 -1
  29. package/dist/{types-CoroKYxB.d.mts → types-CRbjiCC7.d.mts} +2 -2
  30. package/dist/{types-CoroKYxB.d.mts.map → types-CRbjiCC7.d.mts.map} +1 -1
  31. package/dist/{vitest-file-tasks-vvZzigcF.mjs → vitest-file-tasks-BBsZ_wS6.mjs} +4 -2
  32. package/dist/vitest-file-tasks-BBsZ_wS6.mjs.map +1 -0
  33. package/package.json +17 -10
  34. package/prebuilds/darwin-arm64/vitest-pool-assemblyscript.glibc.node +0 -0
  35. package/prebuilds/darwin-x64/vitest-pool-assemblyscript.glibc.node +0 -0
  36. package/prebuilds/win32-arm64/vitest-pool-assemblyscript.glibc.node +0 -0
  37. package/prebuilds/win32-x64/vitest-pool-assemblyscript.glibc.node +0 -0
  38. package/dist/pool-runner-init-CCvnKt5o.d.mts.map +0 -1
  39. package/dist/vitest-file-tasks-vvZzigcF.mjs.map +0 -1
package/README.md CHANGED
@@ -95,7 +95,7 @@ npm install -D vitest vitest-pool-assemblyscript assemblyscript
95
95
 
96
96
  Create or update `vitest.config.ts`. See the [Configuration Guide](docs/configuration-guide.md) for all supported vitest options, pool options, coverage configuration, and multi-project setups.
97
97
 
98
- **vitest 4.x:**
98
+ **vitest 5.x & 4.x:**
99
99
  ```typescript
100
100
  import { defineConfig } from 'vitest/config';
101
101
  import { createAssemblyScriptPool } from 'vitest-pool-assemblyscript/config';
@@ -161,7 +161,7 @@ npx vitest run
161
161
  - Works with Vitest UI, reporters, and coverage tools
162
162
  - Project (workspace) config allows coexisting AssemblyScript pools and JavaScript pools
163
163
  - Hybrid Coverage Provider unifies test reports (`html`, `lcov`, `json`, etc) from multiple pools (delegates to v8 provider for JS coverage)
164
- - Supports vitest 3.x and 4.x
164
+ - Supports vitest 3.2.x, 4.x, and 5.x
165
165
 
166
166
  ### Per-Test WASM Isolation
167
167
  - Each AssemblyScript test **file** is compiled to a WASM binary
@@ -277,7 +277,7 @@ The overall goal is tight vitest experience integration - most CLI commands, rep
277
277
  | Dependency | Supported Versions |
278
278
  |---|---|
279
279
  | Node.js | (20*), 22, 24+ |
280
- | Vitest | 3.2.x, 4.x.x |
280
+ | Vitest | 3.2.x, 4.x.x, 5.x.x |
281
281
  | AssemblyScript | 0.28.9+ ([more?](#frequently-asked-questions)) |
282
282
 
283
283
  >ℹ️ ***Node 20 Support:** If you don't need code coverage, Node 20 should continue to work for test execution.
@@ -1,4 +1,4 @@
1
- import { stringifyValue } from './utils';
1
+ import { isNull, stringifyValue } from './utils';
2
2
 
3
3
  /**
4
4
  * Byte offset from an object pointer to the rtId field in the AS managed object header.
@@ -325,19 +325,15 @@ function arrayBufferEquals<T, U>(actual: T, expected: U): __vitest_assemblyscrip
325
325
  * (or same reference type) for provided values.
326
326
  */
327
327
  export function identical<T, U>(actual: T, expected: U): bool {
328
+ const actualIsNull = isNull(actual);
329
+ const expectedIsNull = isNull(expected);
330
+
328
331
  // both refs
329
332
  if (isReference<T>() && isReference<U>()) {
330
- const actualIsNullable = isNullable<T>();
331
- // Use changetype pointer check instead of `== null` to avoid invoking
332
- // user-defined @operator("==") overloads, which reject null arguments
333
- const actualIsNull = actualIsNullable && changetype<usize>(actual) == 0;
334
- const expectedIsNullable = isNullable<U>();
335
- const expectedIsNull = expectedIsNullable && changetype<usize>(expected) == 0;
336
-
337
333
  // null refs
338
334
  if (actualIsNull && expectedIsNull) {
339
335
  return true;
340
- } else if ( (actualIsNull && !expectedIsNull) || (!actualIsNull && expectedIsNull) ) {
336
+ } else if ( actualIsNull != expectedIsNull ) {
341
337
  return false;
342
338
  }
343
339
 
@@ -348,27 +344,26 @@ export function identical<T, U>(actual: T, expected: U): bool {
348
344
  // object refs
349
345
  return changetype<usize>(actual) == changetype<usize>(expected);
350
346
  }
351
- } else if (isReference<T>() && !isReference<U>()) {
352
- // Both null/zero: null reference matches bare null (usize(0))
353
- if (changetype<usize>(actual) == 0 && expected == usize(0)) {
347
+ } else if (
348
+ (isReference<T>() && !isReference<U>())
349
+ || (!isReference<T>() && isReference<U>())
350
+ ) {
351
+ if (actualIsNull && expectedIsNull) {
354
352
  return true;
353
+ } else if ( actualIsNull != expectedIsNull ) {
354
+ return false;
355
355
  }
356
+
356
357
  // Non-null reference vs value type: fundamentally incomparable
357
358
  throw new Error(
358
359
  "Cannot compare " + nameof<T>() + " with " + nameof<U>()
359
360
  + equalsPathAtSuffix() + ": reference and value types are not comparable."
360
361
  );
361
- } else if (!isReference<T>() && isReference<U>()) {
362
- // Both null/zero: bare null (usize(0)) matches null reference
363
- if (actual == usize(0) && changetype<usize>(expected) == 0) {
364
- return true;
365
- }
366
- // Value type vs non-null reference: fundamentally incomparable
367
- throw new Error(
368
- "Cannot compare " + nameof<T>() + " with " + nameof<U>()
369
- + equalsPathAtSuffix() + ": reference and value types are not comparable."
370
- );
371
362
  } else { // both primitives
363
+ if ( actualIsNull != expectedIsNull ) {
364
+ return false;
365
+ }
366
+
372
367
  if ( (isBoolean<T>() && !isBoolean<U>()) || (!isBoolean<T>() && isBoolean<U>())
373
368
  ) {
374
369
  // when one is boolean and the other is not, they are not identical
@@ -484,27 +479,39 @@ export function equals<T, U>(actual: T, expected: U): __vitest_assemblyscript_Eq
484
479
  exactMatch = identical(actual, expected);
485
480
  }
486
481
 
487
- if (!isReference<T>() || isString<T>() || isVector<T>()) {
488
- // non-bool primitives or strings: return result of comparing
482
+ const actualIsNull = isNull(actual);
483
+ const expectedIsNull = isNull(expected);
484
+
485
+ if (
486
+ (!isReference<T>() && !isReference<U>())
487
+ || (isString<T>() && isString<U>())
488
+ || (isVector<T>() && isVector<U>())
489
+ ) {
490
+ // non-bool primitives or strings or vectors: return result of identity compare
489
491
  return exactMatch ? __vitest_assemblyscript_EqualityResult.Equal : __vitest_assemblyscript_EqualityResult.NotEqual;
490
492
  } else if (exactMatch) {
491
493
  // primitive / reference comparison passed already
492
494
  return __vitest_assemblyscript_EqualityResult.Equal;
493
- }
494
-
495
- if (isNullable<T>()) {
496
- // Use changetype pointer checks instead of `== null` / `!= null` to avoid
497
- // invoking user-defined @operator("==") overloads, which reject null arguments
498
- const actualIsNull = changetype<usize>(actual) == 0;
499
- const expectedIsNull = changetype<usize>(expected) == 0;
500
-
501
- if (actualIsNull && expectedIsNull) {
502
- return __vitest_assemblyscript_EqualityResult.Equal;
495
+ } else if (
496
+ (isReference<T>() && !isReference<U>())
497
+ || (!isReference<T>() && isReference<U>())
498
+ ) {
499
+ if (actualIsNull || expectedIsNull) {
500
+ return exactMatch ? __vitest_assemblyscript_EqualityResult.Equal : __vitest_assemblyscript_EqualityResult.NotEqual;
503
501
  }
504
502
 
505
- if (actualIsNull != expectedIsNull) {
506
- return __vitest_assemblyscript_EqualityResult.NotEqual;
507
- }
503
+ // Non-null reference vs value type: fundamentally incomparable
504
+ throw new Error(
505
+ "Cannot compare " + nameof<T>() + " with " + nameof<U>()
506
+ + equalsPathAtSuffix() + ": reference and value types are not comparable."
507
+ );
508
+ }
509
+
510
+ if (actualIsNull && expectedIsNull) {
511
+ return __vitest_assemblyscript_EqualityResult.Equal;
512
+ }
513
+ if (actualIsNull != expectedIsNull) {
514
+ return __vitest_assemblyscript_EqualityResult.NotEqual;
508
515
  }
509
516
 
510
517
  // Cycle detection: if this reference pair is already being compared further up
@@ -784,4 +791,3 @@ export function compareInequality<T, U>(actual: T, compareTo: U, expectedOperati
784
791
  export function truthyOrFalsey<T>(actual: T, expected: bool): bool {
785
792
  return actual ? expected == true : expected == false;
786
793
  }
787
-
@@ -77,10 +77,15 @@ abstract class BaseExpectMatcher<T> {
77
77
  * involving a float and allows all numeric types because it can still produce accurate
78
78
  * results in precision-loss casting edge cases.
79
79
  *
80
+ * Null comparison: a null reference, the bare `null` literal, and `usize(0)` are all treated
81
+ * as null and compare as identical to one another. Zero-valued primitives (`0`, `false`, `0.0`) are
82
+ * **not** identical to null. Because the bare `null` literal is `usize(0)` in AssemblyScript,
83
+ * `usize(0)` will not compare as identical to a `0` of another numeric type.
84
+ *
80
85
  * @throws When comparing float/integer types where the float's mantissa cannot losslessly
81
86
  * represent the integer type's range (e.g. `f32` vs `i32`, `f64` vs `i64`).
82
87
  * @throws When comparing fundamentally incompatible types: reference vs value type
83
- * (e.g. `string` vs `i32`, unless one is null/zero), or `v128` vs non-vector type.
88
+ * (e.g. `string` vs `i32`, unless one side is null, or `v128` vs non-vector type).
84
89
  *
85
90
  * @example
86
91
  * expect(1 + 1).toBe(2);
@@ -278,12 +283,17 @@ abstract class BaseExpectMatcher<T> {
278
283
  * are not included, as deep equality injection is scoped to user source files only
279
284
  *
280
285
  * SIMD vectors use WASM's native `==` comparison, which compares at the bit level,
281
- * ignoring lane type.
286
+ * ignoring lane type.
287
+ *
288
+ * Null comparison: a null reference, the bare `null` literal, and `usize(0)` are all treated
289
+ * as null and compare equal to one another. Zero-valued primitives (`0`, `false`, `0.0`) are
290
+ * **not** equal to null. Because the bare `null` literal is `usize(0)` in AssemblyScript,
291
+ * `usize(0)` will not compare as identical to a `0` of another numeric type.
282
292
  *
283
293
  * @throws When comparing float/integer types where the float's mantissa cannot losslessly
284
294
  * represent the integer type's range (e.g. `f32` vs `i32`, `f64` vs `i64`).
285
295
  * @throws When comparing fundamentally incompatible types: reference vs value type
286
- * (e.g. `string` vs `i32`, unless one is null/zero), or `v128` vs non-vector type.
296
+ * (e.g. `string` vs `i32`, unless one side is null, or `v128` vs non-vector type).
287
297
  * @throws When comparing containers with incompatible element types (e.g. `Array<string>`
288
298
  * vs `Array<i32>`), or precision-loss numeric combinations (e.g. `Array<f32>` vs `Array<i32>`).
289
299
  *
package/assembly/utils.ts CHANGED
@@ -1,7 +1,9 @@
1
1
  export function isNull<T>(value: T): bool {
2
2
  if (isReference<T>()) {
3
3
  if (isNullable<T>()) {
4
- return value == null;
4
+ // Use changetype pointer checks instead of `== null` / `!= null` to avoid
5
+ // invoking user-defined @operator("==") overloads, which reject null arguments
6
+ return changetype<usize>(value) == 0;
5
7
  } else {
6
8
  return false;
7
9
  }
@@ -11,6 +13,7 @@ export function isNull<T>(value: T): bool {
11
13
  } else if (isVector<T>()) {
12
14
  return false;
13
15
  } else {
16
+ // handles bare nulls
14
17
  return nameof<T>(value) == 'usize' && value == 0;
15
18
  }
16
19
  }
@@ -1,7 +1,7 @@
1
1
  import { INTERNAL_PATH_LIB_PREFIX, POOL_INTERNAL_PATHS } from "./constants-Bq5KNxXJ.mjs";
2
- import { failFile, getFullTaskHierarchy, getTaskLogLabel, getTaskLogPrefix, prepareFileTaskForCollection } from "./vitest-file-tasks-vvZzigcF.mjs";
2
+ import { failFile, getFullTaskHierarchy, getTaskLogLabel, getTaskLogPrefix, prepareFileTaskForCollection } from "./vitest-file-tasks-BBsZ_wS6.mjs";
3
3
  import { buildEnhancedFileError, debug, toForwardSlash } from "./pool-errors-Bn6YaguR.mjs";
4
- import { executeWASMDiscovery, flushRpcUpdates, reportFileCollected, reportFileError, reportFileQueued, reportUserConsoleLogs } from "./load-user-imports-Bcx9NOt9.mjs";
4
+ import { executeWASMDiscovery, flushRpcUpdates, reportFileCollected, reportFileError, reportFileQueued, reportUserConsoleLogs } from "./load-user-imports-iAQUMKAi.mjs";
5
5
  import { compileAssemblyScript } from "./compiler-CXR5UJId.mjs";
6
6
  import { basename, relative } from "node:path";
7
7
 
@@ -79,4 +79,4 @@ async function runCompileAndDiscover(file, logModule, rpc, asPoolOptions, projec
79
79
 
80
80
  //#endregion
81
81
  export { runCompileAndDiscover };
82
- //# sourceMappingURL=compile-runner-BGHM_85g.mjs.map
82
+ //# sourceMappingURL=compile-runner-BGYyiUEf.mjs.map
@@ -1 +1 @@
1
- {"version":3,"file":"compile-runner-BGHM_85g.mjs","names":[],"sources":["../src/pool-thread/runner/compile-runner.ts"],"sourcesContent":["/**\n * Worker thread test runner logic for AssemblyScript Pool\n */\n\nimport { basename, relative } from 'node:path';\nimport type { File } from '@vitest/runner/types';\nimport type { SerializedDiffOptions } from '@vitest/utils/diff';\n\nimport type {\n AssemblyScriptCompilerOptions,\n AssemblyScriptConsoleLog,\n AssemblyScriptConsoleLogHandler,\n InstrumentationOptions,\n ResolvedAssemblyScriptPoolOptions,\n ThreadImports,\n WASMCompilation,\n WorkerRPC,\n} from '../../types/types.js';\nimport {\n AS_POOL_WASM_COVERAGE_MEM_IMPORT_NAME,\n AS_POOL_WASM_IMPORTS_MODULE_NAME,\n ASSEMBLYSCRIPT_LIB_PREFIX,\n INTERNAL_FUNCTION_NAME_SUBSTRING,\n INTERNAL_PATH_LIB_PREFIX,\n POOL_INTERNAL_PATHS,\n} from '../../types/constants.js';\nimport { executeWASMDiscovery } from '../../wasm-executor/index.js';\nimport { debug } from '../../util/debug.js';\nimport { toForwardSlash } from '../../util/path-utils.js';\nimport {\n reportFileQueued,\n reportFileCollected,\n reportUserConsoleLogs,\n flushRpcUpdates,\n reportFileError,\n} from '../rpc-reporter.js';\nimport { compileAssemblyScript } from '../../compiler/index.js';\nimport { getTaskLogLabel, getTaskLogPrefix } from '../../util/vitest-tasks.js';\nimport {\n failFile,\n getFullTaskHierarchy,\n prepareFileTaskForCollection,\n} from '../../util/vitest-file-tasks.js';\nimport { buildEnhancedFileError } from '../../util/pool-errors.js';\n\nlet threadCompilationCount: number = 0;\n\nexport async function runCompileAndDiscover(\n file: File,\n logModule: string,\n rpc: WorkerRPC,\n asPoolOptions: ResolvedAssemblyScriptPoolOptions,\n projectRoot: string,\n collectCoverage: boolean,\n relativeUserCoverageExclusions: string[],\n threadImports: ThreadImports,\n diffOptions?: SerializedDiffOptions,\n testNamePattern?: RegExp,\n allowOnly?: boolean,\n): Promise<WASMCompilation | undefined> {\n const base = basename(file.filepath);\n const fileLogPrefix = getTaskLogPrefix(logModule, base, file);\n const fileLogLabel = getTaskLogLabel(base, file);\n\n debug(`${fileLogPrefix} - Beginning runCompileAndDiscover for \"${file.filepath}\" at ${Date.now()}`);\n\n let compilation: WASMCompilation | undefined;\n let reportedQueued: boolean = false;\n \n const runStartPerf = performance.now();\n\n try {\n await reportFileQueued(rpc, file, logModule, fileLogLabel);\n reportedQueued = true;\n\n const relativeTestFilePath = toForwardSlash(relative(projectRoot, file.filepath));\n const instrumentationOptions: InstrumentationOptions = {\n projectRoot,\n relativeExcludedFiles: [\n relativeTestFilePath,\n ...(asPoolOptions._instrumentPoolInternals ? [] : POOL_INTERNAL_PATHS),\n ...relativeUserCoverageExclusions,\n ],\n excludedLibraryFilePrefix: ASSEMBLYSCRIPT_LIB_PREFIX,\n excludedLibraryFileOverridePrefix: asPoolOptions._instrumentPoolInternals ? INTERNAL_PATH_LIB_PREFIX : undefined,\n excludedInternalFunctionSubstring: INTERNAL_FUNCTION_NAME_SUBSTRING,\n coverageMemoryPagesMin: asPoolOptions.coverageMemoryPagesInitial ?? 1,\n coverageMemoryPagesMax: asPoolOptions.coverageMemoryPagesMax,\n debug: asPoolOptions.debugNative,\n coverageMemoryModule: AS_POOL_WASM_IMPORTS_MODULE_NAME,\n coverageMemoryName: AS_POOL_WASM_COVERAGE_MEM_IMPORT_NAME\n };\n const compilerOptions: AssemblyScriptCompilerOptions = {\n stripInline: asPoolOptions.stripInline,\n projectRoot,\n shouldInstrument: collectCoverage,\n instrumentationOptions,\n extraFlags: asPoolOptions.extraCompilerFlags\n };\n\n compilation = await compileAssemblyScript(\n file.filepath,\n compilerOptions,\n logModule,\n fileLogLabel\n );\n file.setupDuration = compilation.compileTiming;\n threadCompilationCount++;\n\n debug(`${fileLogPrefix} - TIMING compileAssemblyScript total `\n + `(thread comp # ${threadCompilationCount}): ${compilation.compileTiming.toFixed(2)} ms`\n );\n \n const logMessages: AssemblyScriptConsoleLog[] = [];\n const handleLog: AssemblyScriptConsoleLogHandler = (msg: string, isError: boolean = false): void => {\n logMessages.push({ msg, time: Date.now(), isError });\n };\n \n const discoverStartPerf = performance.now();\n\n await executeWASMDiscovery(\n compilation,\n asPoolOptions,\n collectCoverage,\n handleLog,\n file,\n logModule,\n threadImports,\n );\n\n // set skips when using only and/or user test name pattern, skip file task if all tests skipped\n prepareFileTaskForCollection(file, testNamePattern, allowOnly);\n\n file.collectDuration = performance.now() - discoverStartPerf;\n debug(`${fileLogPrefix} - TIMING Discovery Phase: ${file.collectDuration.toFixed(2)} ms`);\n\n // vitest collect - report discovery results\n await Promise.all([\n // Report user console logs\n reportUserConsoleLogs(rpc, logMessages, logModule, base, file),\n\n // Report onCollected with collected and filtered tasks\n reportFileCollected(rpc, file, logModule, fileLogLabel),\n ]);\n\n debug(() => `${fileLogPrefix} - Collected Test Suite Hierarchy:\\n${getFullTaskHierarchy(file)}`);\n\n const totalTime = performance.now() - runStartPerf;\n debug(`${fileLogPrefix} - TIMING Compilation and Discovery: ${totalTime.toFixed(2)} ms`);\n } catch (error) {\n const testError = await buildEnhancedFileError(\n error,\n file,\n compilation?.sourceMap,\n fileLogPrefix,\n projectRoot,\n 'compile runner',\n diffOptions\n );\n\n failFile(file, testError, runStartPerf);\n\n if (!reportedQueued) {\n await reportFileQueued(rpc, file, logModule, fileLogLabel);\n }\n await reportFileError(rpc, file, logModule, fileLogLabel);\n\n debug(`${fileLogPrefix} - runCompileAndDiscover - Reported file error:`, testError);\n } finally {\n await flushRpcUpdates(rpc);\n debug(`${fileLogPrefix} - runCompileAndDiscover Completed`);\n }\n\n return compilation;\n}\n"],"mappings":";;;;;;;;;;;AA6CA,IAAI,yBAAiC;AAErC,eAAsB,sBACpB,MACA,WACA,KACA,eACA,aACA,iBACA,gCACA,eACA,aACA,iBACA,WACsC;CACtC,MAAM,OAAO,SAAS,KAAK,SAAS;CACpC,MAAM,gBAAgB,iBAAiB,WAAW,MAAM,KAAK;CAC7D,MAAM,eAAe,gBAAgB,MAAM,KAAK;AAEhD,OAAM,GAAG,cAAc,0CAA0C,KAAK,SAAS,OAAO,KAAK,KAAK,GAAG;CAEnG,IAAI;CACJ,IAAI,iBAA0B;CAE9B,MAAM,eAAe,YAAY,KAAK;AAEtC,KAAI;AACF,QAAM,iBAAiB,KAAK,MAAM,WAAW,aAAa;AAC1D,mBAAiB;EAGjB,MAAM,yBAAiD;GACrD;GACA,uBAAuB;IAHI,eAAe,SAAS,aAAa,KAAK,SAAS,CAAC;IAK7E,GAAI,cAAc,2BAA2B,EAAE,GAAG;IAClD,GAAG;IACJ;GACD;GACA,mCAAmC,cAAc,2BAA2B,2BAA2B;GACvG;GACA,wBAAwB,cAAc,8BAA8B;GACpE,wBAAwB,cAAc;GACtC,OAAO,cAAc;GACrB;GACA;GACD;EACD,MAAM,kBAAiD;GACrD,aAAa,cAAc;GAC3B;GACA,kBAAkB;GAClB;GACA,YAAY,cAAc;GAC3B;AAED,gBAAc,MAAM,sBAClB,KAAK,UACL,iBACA,WACA,aACD;AACD,OAAK,gBAAgB,YAAY;AACjC;AAEA,QAAM,GAAG,cAAc,uDACD,uBAAuB,KAAK,YAAY,cAAc,QAAQ,EAAE,CAAC,KACtF;EAED,MAAM,cAA0C,EAAE;EAClD,MAAM,aAA8C,KAAa,UAAmB,UAAgB;AAClG,eAAY,KAAK;IAAE;IAAK,MAAM,KAAK,KAAK;IAAE;IAAS,CAAC;;EAGtD,MAAM,oBAAoB,YAAY,KAAK;AAE3C,QAAM,qBACJ,aACA,eACA,iBACA,WACA,MACA,WACA,cACD;AAGD,+BAA6B,MAAM,iBAAiB,UAAU;AAE9D,OAAK,kBAAkB,YAAY,KAAK,GAAG;AAC3C,QAAM,GAAG,cAAc,6BAA6B,KAAK,gBAAgB,QAAQ,EAAE,CAAC,KAAK;AAGzF,QAAM,QAAQ,IAAI,CAEhB,sBAAsB,KAAK,aAAa,WAAW,MAAM,KAAK,EAG9D,oBAAoB,KAAK,MAAM,WAAW,aAAa,CACxD,CAAC;AAEF,cAAY,GAAG,cAAc,sCAAsC,qBAAqB,KAAK,GAAG;AAGhG,QAAM,GAAG,cAAc,wCADL,YAAY,KAAK,GAAG,cACkC,QAAQ,EAAE,CAAC,KAAK;UACjF,OAAO;EACd,MAAM,YAAY,MAAM,uBACtB,OACA,MACA,aAAa,WACb,eACA,aACA,kBACA,YACD;AAED,WAAS,MAAM,WAAW,aAAa;AAEvC,MAAI,CAAC,eACH,OAAM,iBAAiB,KAAK,MAAM,WAAW,aAAa;AAE5D,QAAM,gBAAgB,KAAK,MAAM,WAAW,aAAa;AAEzD,QAAM,GAAG,cAAc,kDAAkD,UAAU;WAC3E;AACR,QAAM,gBAAgB,IAAI;AAC1B,QAAM,GAAG,cAAc,oCAAoC;;AAG7D,QAAO"}
1
+ {"version":3,"file":"compile-runner-BGYyiUEf.mjs","names":[],"sources":["../src/pool-thread/runner/compile-runner.ts"],"sourcesContent":["/**\n * Worker thread test runner logic for AssemblyScript Pool\n */\n\nimport { basename, relative } from 'node:path';\nimport type { File } from '@vitest/runner/types';\nimport type { SerializedDiffOptions } from '@vitest/utils/diff';\n\nimport type {\n AssemblyScriptCompilerOptions,\n AssemblyScriptConsoleLog,\n AssemblyScriptConsoleLogHandler,\n InstrumentationOptions,\n ResolvedAssemblyScriptPoolOptions,\n ThreadImports,\n WASMCompilation,\n WorkerRPC,\n} from '../../types/types.js';\nimport {\n AS_POOL_WASM_COVERAGE_MEM_IMPORT_NAME,\n AS_POOL_WASM_IMPORTS_MODULE_NAME,\n ASSEMBLYSCRIPT_LIB_PREFIX,\n INTERNAL_FUNCTION_NAME_SUBSTRING,\n INTERNAL_PATH_LIB_PREFIX,\n POOL_INTERNAL_PATHS,\n} from '../../types/constants.js';\nimport { executeWASMDiscovery } from '../../wasm-executor/index.js';\nimport { debug } from '../../util/debug.js';\nimport { toForwardSlash } from '../../util/path-utils.js';\nimport {\n reportFileQueued,\n reportFileCollected,\n reportUserConsoleLogs,\n flushRpcUpdates,\n reportFileError,\n} from '../rpc-reporter.js';\nimport { compileAssemblyScript } from '../../compiler/index.js';\nimport { getTaskLogLabel, getTaskLogPrefix } from '../../util/vitest-tasks.js';\nimport {\n failFile,\n getFullTaskHierarchy,\n prepareFileTaskForCollection,\n} from '../../util/vitest-file-tasks.js';\nimport { buildEnhancedFileError } from '../../util/pool-errors.js';\n\nlet threadCompilationCount: number = 0;\n\nexport async function runCompileAndDiscover(\n file: File,\n logModule: string,\n rpc: WorkerRPC,\n asPoolOptions: ResolvedAssemblyScriptPoolOptions,\n projectRoot: string,\n collectCoverage: boolean,\n relativeUserCoverageExclusions: string[],\n threadImports: ThreadImports,\n diffOptions?: SerializedDiffOptions,\n testNamePattern?: RegExp,\n allowOnly?: boolean,\n): Promise<WASMCompilation | undefined> {\n const base = basename(file.filepath);\n const fileLogPrefix = getTaskLogPrefix(logModule, base, file);\n const fileLogLabel = getTaskLogLabel(base, file);\n\n debug(`${fileLogPrefix} - Beginning runCompileAndDiscover for \"${file.filepath}\" at ${Date.now()}`);\n\n let compilation: WASMCompilation | undefined;\n let reportedQueued: boolean = false;\n \n const runStartPerf = performance.now();\n\n try {\n await reportFileQueued(rpc, file, logModule, fileLogLabel);\n reportedQueued = true;\n\n const relativeTestFilePath = toForwardSlash(relative(projectRoot, file.filepath));\n const instrumentationOptions: InstrumentationOptions = {\n projectRoot,\n relativeExcludedFiles: [\n relativeTestFilePath,\n ...(asPoolOptions._instrumentPoolInternals ? [] : POOL_INTERNAL_PATHS),\n ...relativeUserCoverageExclusions,\n ],\n excludedLibraryFilePrefix: ASSEMBLYSCRIPT_LIB_PREFIX,\n excludedLibraryFileOverridePrefix: asPoolOptions._instrumentPoolInternals ? INTERNAL_PATH_LIB_PREFIX : undefined,\n excludedInternalFunctionSubstring: INTERNAL_FUNCTION_NAME_SUBSTRING,\n coverageMemoryPagesMin: asPoolOptions.coverageMemoryPagesInitial ?? 1,\n coverageMemoryPagesMax: asPoolOptions.coverageMemoryPagesMax,\n debug: asPoolOptions.debugNative,\n coverageMemoryModule: AS_POOL_WASM_IMPORTS_MODULE_NAME,\n coverageMemoryName: AS_POOL_WASM_COVERAGE_MEM_IMPORT_NAME\n };\n const compilerOptions: AssemblyScriptCompilerOptions = {\n stripInline: asPoolOptions.stripInline,\n projectRoot,\n shouldInstrument: collectCoverage,\n instrumentationOptions,\n extraFlags: asPoolOptions.extraCompilerFlags\n };\n\n compilation = await compileAssemblyScript(\n file.filepath,\n compilerOptions,\n logModule,\n fileLogLabel\n );\n file.setupDuration = compilation.compileTiming;\n threadCompilationCount++;\n\n debug(`${fileLogPrefix} - TIMING compileAssemblyScript total `\n + `(thread comp # ${threadCompilationCount}): ${compilation.compileTiming.toFixed(2)} ms`\n );\n \n const logMessages: AssemblyScriptConsoleLog[] = [];\n const handleLog: AssemblyScriptConsoleLogHandler = (msg: string, isError: boolean = false): void => {\n logMessages.push({ msg, time: Date.now(), isError });\n };\n \n const discoverStartPerf = performance.now();\n\n await executeWASMDiscovery(\n compilation,\n asPoolOptions,\n collectCoverage,\n handleLog,\n file,\n logModule,\n threadImports,\n );\n\n // set skips when using only and/or user test name pattern, skip file task if all tests skipped\n prepareFileTaskForCollection(file, testNamePattern, allowOnly);\n\n file.collectDuration = performance.now() - discoverStartPerf;\n debug(`${fileLogPrefix} - TIMING Discovery Phase: ${file.collectDuration.toFixed(2)} ms`);\n\n // vitest collect - report discovery results\n await Promise.all([\n // Report user console logs\n reportUserConsoleLogs(rpc, logMessages, logModule, base, file),\n\n // Report onCollected with collected and filtered tasks\n reportFileCollected(rpc, file, logModule, fileLogLabel),\n ]);\n\n debug(() => `${fileLogPrefix} - Collected Test Suite Hierarchy:\\n${getFullTaskHierarchy(file)}`);\n\n const totalTime = performance.now() - runStartPerf;\n debug(`${fileLogPrefix} - TIMING Compilation and Discovery: ${totalTime.toFixed(2)} ms`);\n } catch (error) {\n const testError = await buildEnhancedFileError(\n error,\n file,\n compilation?.sourceMap,\n fileLogPrefix,\n projectRoot,\n 'compile runner',\n diffOptions\n );\n\n failFile(file, testError, runStartPerf);\n\n if (!reportedQueued) {\n await reportFileQueued(rpc, file, logModule, fileLogLabel);\n }\n await reportFileError(rpc, file, logModule, fileLogLabel);\n\n debug(`${fileLogPrefix} - runCompileAndDiscover - Reported file error:`, testError);\n } finally {\n await flushRpcUpdates(rpc);\n debug(`${fileLogPrefix} - runCompileAndDiscover Completed`);\n }\n\n return compilation;\n}\n"],"mappings":";;;;;;;;;;;AA6CA,IAAI,yBAAiC;AAErC,eAAsB,sBACpB,MACA,WACA,KACA,eACA,aACA,iBACA,gCACA,eACA,aACA,iBACA,WACsC;CACtC,MAAM,OAAO,SAAS,KAAK,SAAS;CACpC,MAAM,gBAAgB,iBAAiB,WAAW,MAAM,KAAK;CAC7D,MAAM,eAAe,gBAAgB,MAAM,KAAK;AAEhD,OAAM,GAAG,cAAc,0CAA0C,KAAK,SAAS,OAAO,KAAK,KAAK,GAAG;CAEnG,IAAI;CACJ,IAAI,iBAA0B;CAE9B,MAAM,eAAe,YAAY,KAAK;AAEtC,KAAI;AACF,QAAM,iBAAiB,KAAK,MAAM,WAAW,aAAa;AAC1D,mBAAiB;EAGjB,MAAM,yBAAiD;GACrD;GACA,uBAAuB;IAHI,eAAe,SAAS,aAAa,KAAK,SAAS,CAAC;IAK7E,GAAI,cAAc,2BAA2B,EAAE,GAAG;IAClD,GAAG;IACJ;GACD;GACA,mCAAmC,cAAc,2BAA2B,2BAA2B;GACvG;GACA,wBAAwB,cAAc,8BAA8B;GACpE,wBAAwB,cAAc;GACtC,OAAO,cAAc;GACrB;GACA;GACD;EACD,MAAM,kBAAiD;GACrD,aAAa,cAAc;GAC3B;GACA,kBAAkB;GAClB;GACA,YAAY,cAAc;GAC3B;AAED,gBAAc,MAAM,sBAClB,KAAK,UACL,iBACA,WACA,aACD;AACD,OAAK,gBAAgB,YAAY;AACjC;AAEA,QAAM,GAAG,cAAc,uDACD,uBAAuB,KAAK,YAAY,cAAc,QAAQ,EAAE,CAAC,KACtF;EAED,MAAM,cAA0C,EAAE;EAClD,MAAM,aAA8C,KAAa,UAAmB,UAAgB;AAClG,eAAY,KAAK;IAAE;IAAK,MAAM,KAAK,KAAK;IAAE;IAAS,CAAC;;EAGtD,MAAM,oBAAoB,YAAY,KAAK;AAE3C,QAAM,qBACJ,aACA,eACA,iBACA,WACA,MACA,WACA,cACD;AAGD,+BAA6B,MAAM,iBAAiB,UAAU;AAE9D,OAAK,kBAAkB,YAAY,KAAK,GAAG;AAC3C,QAAM,GAAG,cAAc,6BAA6B,KAAK,gBAAgB,QAAQ,EAAE,CAAC,KAAK;AAGzF,QAAM,QAAQ,IAAI,CAEhB,sBAAsB,KAAK,aAAa,WAAW,MAAM,KAAK,EAG9D,oBAAoB,KAAK,MAAM,WAAW,aAAa,CACxD,CAAC;AAEF,cAAY,GAAG,cAAc,sCAAsC,qBAAqB,KAAK,GAAG;AAGhG,QAAM,GAAG,cAAc,wCADL,YAAY,KAAK,GAAG,cACkC,QAAQ,EAAE,CAAC,KAAK;UACjF,OAAO;EACd,MAAM,YAAY,MAAM,uBACtB,OACA,MACA,aAAa,WACb,eACA,aACA,kBACA,YACD;AAED,WAAS,MAAM,WAAW,aAAa;AAEvC,MAAI,CAAC,eACH,OAAM,iBAAiB,KAAK,MAAM,WAAW,aAAa;AAE5D,QAAM,gBAAgB,KAAK,MAAM,WAAW,aAAa;AAEzD,QAAM,GAAG,cAAc,kDAAkD,UAAU;WAC3E;AACR,QAAM,gBAAgB,IAAI;AAC1B,QAAM,GAAG,cAAc,oCAAoC;;AAG7D,QAAO"}
@@ -1,4 +1,4 @@
1
- import { AssemblyScriptPoolOptions, HybridProviderOptions, WasmImportsFactory, WasmImportsFactoryInfo } from "../types-CoroKYxB.mjs";
1
+ import { AssemblyScriptPoolOptions, HybridProviderOptions, WasmImportsFactory, WasmImportsFactoryInfo } from "../types-CRbjiCC7.mjs";
2
2
  import { CoverageV8Options } from "vitest/node";
3
3
  import { ConfigEnv, UserWorkspaceConfig, ViteUserConfig } from "vitest/config";
4
4
 
@@ -1,5 +1,5 @@
1
- import { AssemblyScriptPoolOptions, HybridProviderOptions, WasmImportsFactory, WasmImportsFactoryInfo } from "../types-CoroKYxB.mjs";
2
- import { createAssemblyScriptPool } from "../pool-runner-init-CCvnKt5o.mjs";
1
+ import { AssemblyScriptPoolOptions, HybridProviderOptions, WasmImportsFactory, WasmImportsFactoryInfo } from "../types-CRbjiCC7.mjs";
2
+ import { createAssemblyScriptPool } from "../pool-runner-init-CPQL6pVL.mjs";
3
3
 
4
4
  //#region src/config/coverage-options.d.ts
5
5
  /**
@@ -1,7 +1,7 @@
1
1
  import "../constants-Bq5KNxXJ.mjs";
2
- import "../vitest-file-tasks-vvZzigcF.mjs";
2
+ import "../vitest-file-tasks-BBsZ_wS6.mjs";
3
3
  import "../pool-errors-Bn6YaguR.mjs";
4
4
  import "../worker-rpc-channel-CvCrc8aa.mjs";
5
- import { createAssemblyScriptPool } from "../pool-runner-init-DjRCbiX-.mjs";
5
+ import { createAssemblyScriptPool } from "../pool-runner-init-CSJt3dHv.mjs";
6
6
 
7
7
  export { createAssemblyScriptPool };
@@ -1 +1 @@
1
- {"version":3,"file":"index.mjs","names":["parsePath","AssemblyScriptParser"],"sources":["../../src/coverage-provider/containment-matcher.ts","../../src/coverage-provider/istanbul-converter.ts","../../src/coverage-provider/ast-parser.ts","../../src/coverage-provider/glob-utils.ts","../../src/coverage-provider/hybrid-coverage-provider.ts","../../src/coverage-provider/index.ts"],"sourcesContent":["/**\n * Containment Matcher\n *\n * Provides containment-based matching for coverage mapping.\n * Binary debug info provides points (representativeLocation), source parsing provides ranges.\n * We find which source range contains each binary point.\n *\n * Usage by version:\n * - v1: Function containment matching (binary representativeLocation → source function range)\n * - v2: Function containment (same) + statement containment + branch path containment\n *\n * Why containment matching (not direct position lookup):\n * AS compiler generates source map entries differently by statement type:\n * - Variable declarations: source map points to statement start (keyword)\n * - Control flow (if/switch/for): source map points to condition EXPRESSION, not keyword\n *\n * Example: `if (n < 0)` → binary reports column 7 ('n'), not column 3 ('i' of 'if')\n *\n * Containment matching is more robust:\n * - Binary gives us a position (representativeLocation) somewhere in the function\n * - Source gives us function ranges (start/end line/column)\n * - We find which source function range contains the binary position\n * - Handles nested functions with \"tightest fit\" (innermost function wins)\n */\n\nimport type { ParsedSourceFunctionInfo, SourceRange } from '../types/types.js';\n\n/**\n * Functions indexed by file path, then by start line.\n * Multiple functions can start on the same line (e.g., nested arrow functions).\n */\nexport type FunctionsByFileAndStartLine = Record<string, Record<number, ParsedSourceFunctionInfo[]>>;\n\n/**\n * Find the source function whose range contains the given position.\n *\n * For nested functions, uses \"tightest fit\" - returns the innermost function\n * (the one with the largest start position among all containing functions).\n *\n * @param functionsByStartLine - Functions for a single file, indexed by start line\n * @param line - Target line number (1-based)\n * @param column - Target column number (1-based)\n * @returns The containing function, or undefined if no match\n */\nexport function findFunctionContainingPosition(\n functionsByStartLine: Record<number, ParsedSourceFunctionInfo[]>,\n line: number,\n column: number\n): ParsedSourceFunctionInfo | undefined {\n let bestMatch: ParsedSourceFunctionInfo | undefined;\n let bestStartLine = -1;\n let bestStartColumn = -1;\n\n // Check functions starting on lines <= target line\n for (const [startLineStr, functions] of Object.entries(functionsByStartLine)) {\n const startLine = Number(startLineStr);\n if (startLine > line) continue;\n\n for (const func of functions) {\n const { range } = func;\n\n // Check if position is within this function's range\n if (!isPositionInRange(line, column, range)) continue;\n\n // Tightest fit: prefer function with largest start position (innermost)\n if (startLine > bestStartLine ||\n (startLine === bestStartLine && range.startColumn > bestStartColumn)) {\n bestMatch = func;\n bestStartLine = startLine;\n bestStartColumn = range.startColumn;\n }\n }\n }\n\n return bestMatch;\n}\n\n/**\n * Check if a position (line, column) falls within a source range.\n *\n * @param line - Target line number (1-based)\n * @param column - Target column number (1-based)\n * @param range - Source range to check against\n * @returns true if position is within range (inclusive)\n */\nexport function isPositionInRange(\n line: number,\n column: number,\n range: SourceRange\n): boolean {\n // Outside line range entirely\n if (line < range.startLine || line > range.endLine) return false;\n\n // On start line but before start column\n if (line === range.startLine && column < range.startColumn) return false;\n\n // On end line but after end column\n if (line === range.endLine && column > range.endColumn) return false;\n\n return true;\n}\n","/**\n * Istanbul Format Converter\n *\n * Converts AssemblyScript coverage data to Istanbul's FileCoverageData format.\n * This enables integration with Vitest's coverage reporting system and standard\n * coverage tools like Codecov, Coveralls, etc.\n *\n * Current Implementation: Function-level coverage only\n * - Uses containment matching: binary positions → source function ranges\n * - Each function maps to both a function entry AND a statement entry\n * - Statement coverage matches function coverage (function-level granularity)\n * - Branch coverage is 0% (no branches tracked yet)\n * - Line coverage derived from statement coverage\n *\n * Future Enhancement (v2): Block-level statement and branch coverage\n */\n\nimport type { FileCoverageData, Range, FunctionMapping, BranchMapping } from 'istanbul-lib-coverage';\nimport type { ParsedSourceFunctionInfo } from '../types/types.js';\nimport { findFunctionContainingPosition } from './containment-matcher.js';\nimport { debugOverride } from '../util/debug.js';\n\n/**\n * Convert AssemblyScript coverage data to Istanbul format for a single file\n *\n * Algorithm (containment matching):\n * 1. For each hit position in fileHitCountsByPosition:\n * - Use containment matcher to find which source function contains this position\n * - Record the hit count for that function\n * 2. For each function in fileFunctionsByStartLine:\n * - Add function mapping to fnMap\n * - Add function hit count to f (from matched hits, or 0 if not hit)\n * - Add corresponding statement mapping to statementMap\n * - Add same hit count to s (statement coverage matches function coverage)\n *\n * @param fileFunctionsByStartLine - Functions for this file, keyed by start line (from AST parser)\n * @param fileHitCountsByPosition - Hit counts for this file, keyed by position \"line:column\" (from accumulated coverage)\n * @param absoluteFilePath - Absolute path to the source file (for Istanbul output)\n * @returns Istanbul FileCoverage object\n */\nexport async function convertToIstanbulFormat(\n fileFunctionsByStartLine: Record<number, ParsedSourceFunctionInfo[]>,\n fileHitCountsByPosition: Record<string, number>,\n absoluteFilePath: string,\n istanbulDebugEnabled: boolean\n): Promise<FileCoverageData> {\n const startMatch = performance.now();\n\n function istanbulDebug(...args: any[]): void {\n if (istanbulDebugEnabled) {\n debugOverride(...args);\n }\n };\n\n istanbulDebug(() => {\n const sourceFunctionCount = Object.values(fileFunctionsByStartLine).reduce((sum, funcs) => sum + funcs.length, 0);\n const uniqueHitPosCount = Object.keys(fileHitCountsByPosition).length;\n const coverageEstimate = sourceFunctionCount === 0 ? Infinity : ((uniqueHitPosCount * 100) / sourceFunctionCount).toFixed(2);\n\n return `[IstanbulConverter] Processing source file: \"${absoluteFilePath}\"\\n`\n + `[IstanbulConverter] Source: ${sourceFunctionCount} total functions, Coverage: ${uniqueHitPosCount} unique hit positions\\n`\n + `[IstanbulConverter] Sanity Check - AS File Coverage Estimate: ${coverageEstimate}%\\n`\n + `[IstanbulConverter] Containment matching functions: coverage hit positions to source functions by range`;\n });\n\n // Build a map of function → hit count using containment matching\n // Key: function source identity (qualifiedName), Value: hit count\n const hitCountsBySourceFunctionName = new Map<ParsedSourceFunctionInfo, number>();\n\n // For each hit position, find which function contains it\n for (const [positionKey, hitCount] of Object.entries(fileHitCountsByPosition)) {\n // Position key format is \"line:column\"\n const parts = positionKey.split(':');\n const lineStr = parts[0];\n const columnStr = parts[1];\n\n if (lineStr && columnStr) {\n const line = parseInt(lineStr, 10);\n const column = parseInt(columnStr, 10);\n\n const containingFunction = findFunctionContainingPosition(fileFunctionsByStartLine, line, column);\n if (containingFunction) {\n // Accumulate hits (in case multiple positions map to same function)\n const existingHits = hitCountsBySourceFunctionName.get(containingFunction);\n const existingHitsCount = existingHits ?? 0;\n const max = Math.max(existingHitsCount, hitCount);\n hitCountsBySourceFunctionName.set(containingFunction, max); // <-- TODO: Explain this max logic\n\n if (existingHits !== undefined) {\n istanbulDebug(`[IstanbulConverter] Position ${positionKey} → function \"${containingFunction.shortName}\" EXISTING HITS: ${existingHits} NEW COUNT: ${max}`);\n } else {\n istanbulDebug(`[IstanbulConverter] Position ${positionKey} → function \"${containingFunction.shortName}\" (hits: ${hitCount})`);\n }\n } else {\n istanbulDebug(`[IstanbulConverter] Position ${positionKey} has no containing function`);\n }\n }\n }\n\n const startConvert = performance.now();\n istanbulDebug(`[IstanbulConverter] Matching Complete - Converting to Istanbul format`);\n\n // Initialize Istanbul data structures\n const fnMap: { [key: string]: FunctionMapping } = {};\n const f: { [key: string]: number } = {};\n const statementMap: { [key: string]: Range } = {};\n const s: { [key: string]: number } = {};\n const branchMap: { [key: string]: BranchMapping } = {};\n const b: { [key: string]: number[] } = {};\n\n // Convert function coverage to Istanbul format\n // Iterate all functions from parsed source (ensures 0-hit functions are included)\n let funcIdx = 0;\n for (const functions of Object.values(fileFunctionsByStartLine)) {\n for (const funcInfo of functions) {\n const { range, shortName } = funcInfo;\n\n // Defensive: skip functions with invalid metadata (shouldn't happen - AST parser filters these)\n if (range.startLine === 0) {\n continue;\n }\n\n // Get hit count from containment matching (or 0 if not hit)\n const hitCount = hitCountsBySourceFunctionName.get(funcInfo) ?? 0;\n const displayShortName = shortName && shortName !== '' ? shortName : '<anonymous>';\n istanbulDebug(\n `[IstanbulConverter] Istanbul function index ${funcIdx}: \"${displayShortName}\"`\n + ` (source ${range.startLine}:${range.startColumn} - ${range.endLine}:${range.endColumn}), hits: ${hitCount}`\n );\n\n // Create function mapping\n // Both 'decl' (declaration) and 'loc' (location) use the same range\n // Internal ParsedSourceFunctionInfo uses 1-based columns, Istanbul expects 0-based\n const istanbulRange: Range = {\n start: { line: range.startLine, column: range.startColumn - 1 },\n end: { line: range.endLine, column: range.endColumn - 1 }\n };\n\n const idxStr = funcIdx.toString();\n fnMap[idxStr] = {\n name: shortName,\n decl: istanbulRange,\n loc: istanbulRange,\n line: range.startLine\n };\n f[idxStr] = hitCount;\n\n // Create corresponding statement mapping\n // For function-level coverage, each function is one \"statement\"\n // The statement range matches the function range\n // This gives us statement coverage at function granularity\n statementMap[idxStr] = istanbulRange;\n s[idxStr] = hitCount;\n\n funcIdx++;\n }\n }\n\n const done = performance.now();\n const matchingMs = (startConvert - startMatch).toFixed(2);\n const convertMs = (done - startConvert).toFixed(2);\n const totalMs = (done - startMatch).toFixed(2);\n\n istanbulDebug(\n `[IstanbulConverter] Coverage Coversion Complete: ${Object.keys(fnMap).length} functions,` \n + ` ${totalMs} ms total (matching: ${matchingMs} ms, convert: ${convertMs} ms)`\n );\n\n return {\n path: absoluteFilePath,\n fnMap,\n f,\n statementMap,\n s,\n branchMap,\n b\n };\n}\n","/**\n * AST Parser for AssemblyScript Source Files\n *\n * Parses AS source files to extract function metadata for coverage.\n * Used by generateCoverage to build empty coverage map from all source files.\n *\n * Source AST is the source of truth for what SHOULD be covered.\n * Binary instrumentation tells us what we CAN measure (hit counts).\n *\n * Functions are grouped by start line for efficient containment matching.\n *\n * Architecture:\n * - Uses shared ASTVisitor for complete NodeKind coverage\n * - Overrides hooks to extract function information during traversal\n */\n\nimport { readFile } from 'node:fs/promises';\nimport { parse as parsePath } from 'node:path';\nimport {\n Parser as AssemblyScriptParser,\n Source,\n BlockStatement,\n Node,\n FunctionDeclaration,\n MethodDeclaration,\n ClassDeclaration,\n NamespaceDeclaration,\n VariableDeclaration,\n FunctionExpression,\n} from 'assemblyscript';\n\nimport type { ParsedSourceFunctionInfo, SourceRange } from '../types/types.js';\nimport { ASCommonFlags, ASNodeKind } from '../types/constants.js';\nimport { ASTVisitor } from '../util/ast-visitor.js';\n\n/**\n * Visitor that extracts function information from AST nodes\n */\nclass FunctionExtractorVisitor extends ASTVisitor {\n /** Source file being parsed */\n private source: Source;\n /** Module path for building qualified names */\n private modulePath: string;\n /** Absolute file path */\n private filePath: string;\n /** Accumulated function records, keyed by start line */\n readonly functions: Record<number, ParsedSourceFunctionInfo[]> = {};\n /** Namespace context stack (supports nested namespaces) */\n private namespaceStack: string[] = [];\n /** Current class name (when inside a class) */\n private currentClassName: string | null = null;\n\n constructor(source: Source, modulePath: string, filePath: string) {\n super();\n this.source = source;\n this.modulePath = modulePath;\n this.filePath = filePath;\n }\n\n /**\n * Track namespace context when entering a namespace.\n * Supports nesting — push onto the stack.\n */\n protected onNamespaceEnter(node: NamespaceDeclaration): void {\n this.namespaceStack.push(node.name?.text ?? 'Anonymous');\n }\n\n /**\n * Restore namespace context when exiting a namespace.\n */\n protected onNamespaceExit(_node: NamespaceDeclaration): void {\n this.namespaceStack.pop();\n }\n\n /**\n * Track class context when entering a class.\n * Includes namespace prefix when inside a namespace, matching binary naming convention\n * (e.g. \"NS_A.Item\" for class Item inside namespace NS_A).\n */\n protected onClassEnter(node: ClassDeclaration): void {\n const className = node.name?.text ?? 'Anonymous';\n if (this.namespaceStack.length > 0) {\n this.currentClassName = `${this.namespaceStack.join('.')}.${className}`;\n } else {\n this.currentClassName = className;\n }\n }\n\n /**\n * Restore class context when exiting a class\n */\n protected onClassExit(_node: ClassDeclaration): void {\n this.currentClassName = null;\n }\n\n /**\n * Extract function info from function declarations\n */\n protected onFunctionDeclaration(node: FunctionDeclaration): boolean {\n if (node.body && this.hasBodyStatements(node.body)) {\n const funcName = node.name?.text ?? '~anonymous';\n const shortName = this.namespaceStack.length > 0\n ? `${this.namespaceStack.join('.')}.${funcName}`\n : funcName;\n const qualifiedName = `${this.modulePath}/${shortName}`;\n const range = this.buildRange(node, node.name ?? null);\n this.addFunction(qualifiedName, shortName, range);\n }\n return true; // Continue recursion into body\n }\n\n /**\n * Extract function info from method declarations\n */\n protected onMethodDeclaration(node: MethodDeclaration): boolean {\n if (node.body && this.hasBodyStatements(node.body)) {\n const methodName = node.name?.text ?? 'constructor';\n const className = this.currentClassName ?? 'Unknown';\n const flags = node.flags;\n\n // Determine method type from flags\n const isStatic = (flags & ASCommonFlags.Static) !== 0;\n const isGetter = (flags & ASCommonFlags.Get) !== 0;\n const isSetter = (flags & ASCommonFlags.Set) !== 0;\n\n // Build short name to match binary naming convention:\n // - Static: ClassName.methodName\n // - Getter: ClassName#get:propertyName\n // - Setter: ClassName#set:propertyName\n // - Instance: ClassName#methodName\n let shortName: string;\n if (isStatic) {\n shortName = `${className}.${methodName}`;\n } else if (isGetter) {\n shortName = `${className}#get:${methodName}`;\n } else if (isSetter) {\n shortName = `${className}#set:${methodName}`;\n } else {\n shortName = `${className}#${methodName}`;\n }\n\n const qualifiedName = `${this.modulePath}/${shortName}`;\n const range = this.buildRange(node, node.name ?? null);\n this.addFunction(qualifiedName, shortName, range);\n }\n return true; // Continue recursion into body\n }\n\n /**\n * Extract function info from variable declarations (arrow functions)\n */\n protected onVariableDeclaration(node: VariableDeclaration): boolean {\n if (node.initializer && node.initializer.kind === ASNodeKind.Function) {\n const funcExpr = node.initializer as FunctionExpression;\n const funcDecl = funcExpr.declaration;\n\n if (funcDecl.body && this.hasBodyStatements(funcDecl.body)) {\n // Use variable name for the function\n const shortName = node.name.text;\n const qualifiedName = `${this.modulePath}/${shortName}`;\n\n // Use the variable declaration's range\n const range: SourceRange = {\n filePath: this.filePath,\n startLine: this.source.lineAt(node.range.start),\n startColumn: this.source.columnAt(),\n endLine: this.source.lineAt(node.range.end),\n endColumn: this.source.columnAt(),\n };\n\n this.addFunction(qualifiedName, shortName, range);\n }\n\n // Visit the function body manually since we're handling this specially\n if (funcDecl.body) {\n this.visitNode(funcDecl.body);\n }\n return false; // Don't recurse again - we handled it\n }\n return true; // Continue recursion for non-function initializers\n }\n\n /**\n * Check if a function body has statements (non-empty body)\n */\n private hasBodyStatements(body: Node): boolean {\n if (body.kind === ASNodeKind.Block) {\n const blockBody = body as BlockStatement;\n return blockBody.statements.length > 0;\n }\n // Expression body (braceless arrow) - always has the expression\n return true;\n }\n\n /**\n * Add a function to the functions record, keyed by start line\n */\n private addFunction(qualifiedName: string, shortName: string, range: SourceRange): void {\n const startLine = range.startLine;\n if (!this.functions[startLine]) {\n this.functions[startLine] = [];\n }\n this.functions[startLine].push({ qualifiedName, shortName, range });\n }\n\n /**\n * Build a SourceRange for a node, using name.range.start to skip decorators\n */\n private buildRange(node: Node, nameNode: Node | null): SourceRange {\n const startNode = nameNode ?? node;\n return {\n filePath: this.filePath,\n startLine: this.source.lineAt(startNode.range.start),\n startColumn: this.source.columnAt(),\n endLine: this.source.lineAt(node.range.end),\n endColumn: this.source.columnAt(),\n };\n }\n}\n\n/**\n * Parse functions from a single AS source file\n *\n * @param absoluteSourceFilePath - Absolute path to AS source file\n * @param relativeSourceFilePath - Relative path to AS source file (derived once in caller and used several places)\n * @returns Record of start line to array of ParsedSourceFunctionInfo (multiple functions can start on same line)\n */\nexport async function parseFunctionsFromFile(\n absoluteSourceFilePath: string,\n relativeSourceFilePath: string,\n): Promise<Record<number, ParsedSourceFunctionInfo[]>> {\n const sourceCode = await readFile(absoluteSourceFilePath, 'utf8');\n\n // Build the module path (strip any extension, use forward slashes)\n const parsed = parsePath(relativeSourceFilePath);\n const modulePath = parsed.dir ? `${parsed.dir}/${parsed.name}` : parsed.name;\n\n // Parse with AssemblyScript parser\n const asParser = new AssemblyScriptParser();\n asParser.parseFile(sourceCode, relativeSourceFilePath, true);\n\n const source = asParser.currentSource;\n if (!source) {\n return {};\n }\n\n // Create visitor and traverse\n const visitor = new FunctionExtractorVisitor(source, modulePath, absoluteSourceFilePath);\n visitor.visitSource(source);\n\n return visitor.functions || {};\n}\n","/**\n * Glob Utilities for Coverage\n *\n * Uses tinyglobby to glob AssemblyScript files matching coverage include/exclude patterns.\n * Supports ../ path traversals for projects that import source from outside the project root.\n *\n * Note: tinyglobby docs warn of a performance cost when globbing outside cwd (e.g. ../ patterns)\n * due to recalculating relative paths. Acceptable here since this runs once at startup.\n */\n\nimport { relative } from 'node:path';\nimport { globSync } from 'tinyglobby';\n\nimport { GlobResult } from '../types/types.js';\nimport { toForwardSlash } from '../util/path-utils.js';\n\nconst NODE_MODULES_GLOB = '**/node_modules/**';\n\n/**\n * Glob files matching coverage include/exclude patterns\n *\n * @param include - Include patterns (e.g., ['assembly/**\\/*.ts'])\n * @param exclude - Exclude patterns (e.g., ['**\\/*.test.ts'])\n * @param projectRoot - Project root directory\n * @returns Array of glob results with absolute and project-root-relative paths\n */\nexport function globFiles(\n include: string[],\n exclude: string[],\n projectRoot: string\n): GlobResult[] {\n if (include.length === 0) {\n return [];\n }\n\n const absolutePaths = globSync(include, {\n cwd: projectRoot,\n ignore: [NODE_MODULES_GLOB, ...exclude],\n absolute: true,\n });\n\n return absolutePaths.map((absolute) => ({\n absolute: toForwardSlash(absolute),\n projectRootRelative: toForwardSlash(relative(projectRoot, absolute)),\n }));\n}\n","/**\n * Hybrid Coverage Provider\n *\n * This provider handles both AssemblyScript and JavaScript and coverage\n * - Converts AS coverage to Istanbul format\n * - Delegates JS coverage to Vitest's v8 provider\n * - Merges both into a unified coverage report\n */\n\nimport { basename, relative } from 'node:path';\nimport type {\n CoverageProvider,\n Vitest,\n ReportContext,\n ResolvedCoverageOptions,\n} from 'vitest/node';\nimport type { AfterSuiteRunMeta } from 'vitest';\nimport v8CoverageModule from '@vitest/coverage-v8';\nimport type { CoverageMap } from 'istanbul-lib-coverage';\nimport istanbulCoverage from 'istanbul-lib-coverage';\nconst { createCoverageMap } = istanbulCoverage;\n\nimport { convertToIstanbulFormat } from './istanbul-converter.js';\nimport { parseFunctionsFromFile } from './ast-parser.js';\nimport { globFiles } from './glob-utils.js';\nimport { mergeCoverageData } from './coverage-merge.js';\nimport { debug } from '../util/debug.js';\nimport { createPoolError } from '../util/pool-errors.js';\nimport type {\n AssemblyScriptCoveragePayload,\n CoverageData,\n GlobResult,\n ResolvedHybridProviderOptions,\n} from '../types/types.js';\nimport {\n POOL_ERROR_NAMES,\n COVERAGE_PAYLOAD_FORMATS\n} from '../types/constants.js';\nimport { warnIfASCoverageNotSupportedByNode, warnIfNativeBuildFailed } from '../util/feature-check.js';\n\nexport class HybridCoverageProvider implements CoverageProvider {\n name = 'hybrid-assemblyscript-v8' as const;\n\n private v8Provider: CoverageProvider | undefined;\n private accumulatedCoverageData: CoverageData = { hitCountsByFileAndPosition: {} };\n private projectRoot: string = '';\n private definedCoverageOptions: ResolvedCoverageOptions = {} as ResolvedCoverageOptions;\n private resolvedProviderOptions: ResolvedHybridProviderOptions = {} as ResolvedHybridProviderOptions;\n\n /**\n * Initialize the provider and get reference to v8 provider\n */\n async initialize(ctx: Vitest): Promise<void> {\n this.projectRoot = ctx.config.root;\n this.definedCoverageOptions = ctx.config.coverage;\n\n debug('[HybridCoverageProvider] Initializing Provider');\n\n // Get v8 provider from the coverage module\n this.v8Provider = await v8CoverageModule.getProvider();\n\n if (!this.v8Provider) {\n throw createPoolError(\n POOL_ERROR_NAMES.HybridCoverageProviderError,\n 'initialize failed to get delegated v8 provider',\n );\n }\n\n await this.v8Provider.initialize(ctx);\n this.v8Provider.name = 'hybrid-assemblyscript-v8 (delegated v8 reporter)' as const;\n debug('[HybridCoverageProvider] Initialized with delegated v8 provider');\n\n warnIfASCoverageNotSupportedByNode();\n await warnIfNativeBuildFailed();\n }\n\n /**\n * Handle suite completion - delegate based on coverage format marker\n */\n async onAfterSuiteRun(meta: AfterSuiteRunMeta): Promise<void> {\n const start = performance.now();\n const format: string | undefined = (meta?.coverage as any)?.__format;\n let suiteLogLabel = meta.testFiles.length > 0 ? basename(meta.testFiles[0]!) : '';\n\n // Check for AssemblyScript format marker\n if (format === COVERAGE_PAYLOAD_FORMATS.AssemblyScript) {\n const payload = meta.coverage as AssemblyScriptCoveragePayload;\n const { coverageData, suiteLogLabel: label } = payload;\n suiteLogLabel = label;\n\n debug(() => {\n const fileCount = Object.keys(coverageData.hitCountsByFileAndPosition).length;\n const positionCount = Object.values(coverageData.hitCountsByFileAndPosition)\n .reduce((sum, positions) => sum + Object.keys(positions).length, 0);\n return `[HybridCoverageProvider] ${suiteLogLabel} - onAfterSuiteRun - Suite payload: ${positionCount} unique positions over ${fileCount} source files`;\n });\n\n // Merge incoming coverage data into accumulated (by position, summing hit counts)\n mergeCoverageData(this.accumulatedCoverageData, coverageData);\n\n debug(() => {\n const fileCount = Object.keys(this.accumulatedCoverageData.hitCountsByFileAndPosition).length;\n const positionCount = Object.values(this.accumulatedCoverageData.hitCountsByFileAndPosition)\n .reduce((sum, positions) => sum + Object.keys(positions).length, 0);\n return `[HybridCoverageProvider] ${suiteLogLabel} - onAfterSuiteRun - Accumulated coverage: ${positionCount} unique positions over ${fileCount} source files`;\n });\n } else {\n // Delegate to v8 provider for all other formats (JS, etc.)\n if (!this.v8Provider) {\n throw createPoolError(\n POOL_ERROR_NAMES.HybridCoverageProviderError,\n 'onAfterSuiteRun failed to delegate to v8 provider',\n );\n }\n\n debug(`[HybridCoverageProvider] ${suiteLogLabel} - Delegating to v8 provider`);\n await this.v8Provider.onAfterSuiteRun(meta);\n }\n\n debug(() => {\n const files = meta.testFiles.map(tf => relative(this.projectRoot, tf)).join(',');\n return `[HybridCoverageProvider] ${suiteLogLabel} - onAfterSuiteRun complete - TIMING ${(performance.now() - start).toFixed(2)} ms | testFiles: \"${files}\"`;\n });\n }\n\n /**\n * Generate unified coverage map (merging JS and AS coverage)\n *\n * Flow:\n * 1. Parse included AS source files to get sourceDebugInfo (source of truth for line numbers)\n * 1. Build merged CoverageData (all source functions + accumulated hit counts)\n * 4. Convert merged CoverageData to Istanbul format\n * 5. Get JS coverage from v8 provider\n * 6. Merge AS coverage into JS coverage\n */\n async generateCoverage(context: ReportContext): Promise<unknown> {\n const start = performance.now();\n\n debug('[HybridCoverageProvider] Generating coverage for test run');\n\n if (!this.v8Provider) {\n throw createPoolError(\n POOL_ERROR_NAMES.HybridCoverageProviderError,\n 'generateCoverage failed to delegate to v8 provider',\n );\n }\n\n // Build AS coverage map\n let asCoverageMap = createCoverageMap();\n\n if (this.resolvedProviderOptions.globbedAssemblyScriptInclude.length > 0) {\n debug(`[HybridCoverageProvider] Building AS coverage map with ${this.resolvedProviderOptions.globbedAssemblyScriptInclude.length} source files `);\n debug(() => {\n const accumulatedPositionCount = Object.values(this.accumulatedCoverageData.hitCountsByFileAndPosition)\n .reduce((sum, positions) => sum + Object.keys(positions)?.length, 0);\n const files = Object.keys(this.accumulatedCoverageData.hitCountsByFileAndPosition).length;\n return `[HybridCoverageProvider] Accumulated coverage data has ${accumulatedPositionCount} unique positions hit across ${files} debug source files`;\n });\n\n // parse source files with AST parser, then match to hits and convert to istanbul format\n const fileProcessingPromises = this.resolvedProviderOptions.globbedAssemblyScriptInclude.map(async (include: GlobResult) => {\n debug(`[HybridCoverageProvider] Parsing AS source for expected coverage: \"${include.absolute}\"`);\n \n const functionsByStartLine = await parseFunctionsFromFile(include.absolute, include.projectRootRelative) || {};\n debug(`[HybridCoverageProvider] Parsed ${Object.keys(functionsByStartLine).length} AS source functions in \"${include.absolute}\"`);\n\n const fileHitCountsByPosition = this.accumulatedCoverageData.hitCountsByFileAndPosition[include.absolute] ?? {};\n debug(`[HybridCoverageProvider] Accumulated AS coverage has ${Object.keys(fileHitCountsByPosition).length} positions for \"${include.absolute}\"`);\n\n // Containment matching (binary hit position → source) is performed during istanbul conversion\n return convertToIstanbulFormat(functionsByStartLine, fileHitCountsByPosition, include.absolute, this.resolvedProviderOptions.debugIstanbul);\n });\n\n // Wait for all files to complete\n const istanbulResults = await Promise.all(fileProcessingPromises);\n\n // Add all results to coverage map\n for (const istanbulData of istanbulResults) {\n asCoverageMap.addFileCoverage(istanbulData);\n }\n\n debug(`[HybridCoverageProvider] Built AS coverage map with ${Object.keys(asCoverageMap.data).length} files`);\n } else {\n debug('[HybridCoverageProvider] WARNING: No assemblyScriptInclude patterns yieldled files - Coverage Map will be empty!');\n }\n\n const asGenerateEnd = performance.now();\n debug(`[HybridCoverageProvider] TIMING AS generateCoverage: ${(asGenerateEnd - start).toFixed(2)} ms`);\n\n // Get JS coverage from v8 provider\n debug('[HybridCoverageProvider] Getting JS coverage from v8 provider');\n const jsCoverage = await this.v8Provider.generateCoverage(context) as CoverageMap;\n debug(`[HybridCoverageProvider] JS coverage has ${Object.keys(jsCoverage.data).length} files`);\n debug(`[HybridCoverageProvider] TIMING JS generateCoverage: ${(performance.now() - asGenerateEnd).toFixed(2)} ms`);\n\n // Merge AS coverage into JS coverage.\n // Use toJSON() to pass plain data instead of the CoverageMap instance. This avoids\n // instanceof failures when istanbul-lib-coverage is loaded from different module paths\n // (e.g. the pool workers resolve one copy while the coverage provider resolves another).\n if (asCoverageMap.files().length) {\n debug('[HybridCoverageProvider] Merging AS coverage into JS coverage');\n jsCoverage.merge(asCoverageMap.toJSON());\n } else {\n debug('[HybridCoverageProvider] AS coverage map empty, not merging into JS coverage');\n }\n debug(`[HybridCoverageProvider] Final merged coverage has ${Object.keys(jsCoverage.data).length} files`);\n\n debug(`[HybridCoverageProvider] TIMING Total generateCoverage: ${(performance.now() - start).toFixed(2)} ms`);\n\n return jsCoverage;\n }\n\n /**\n * Report coverage - delegate to v8 provider\n */\n async reportCoverage(coverageMap: unknown, context: ReportContext): Promise<void> {\n if (!this.v8Provider) {\n throw createPoolError(\n POOL_ERROR_NAMES.HybridCoverageProviderError,\n 'reportCoverage failed to delegate to v8 provider',\n );\n }\n\n debug(`[HybridCoverageProvider] Reporting coverage (allTestsRun=${context.allTestsRun})`);\n await this.v8Provider.reportCoverage(coverageMap, context);\n }\n\n /**\n * Resolve options\n */\n resolveOptions(): ResolvedHybridProviderOptions {\n if (!this.v8Provider) {\n throw createPoolError(\n POOL_ERROR_NAMES.HybridCoverageProviderError,\n 'resolveOptions failed to delegate to v8 provider',\n );\n }\n \n debug(`[HybridCoverageProvider] Resolving Coverage Options`);\n \n const resolvedV8Options = this.v8Provider.resolveOptions() as ResolvedCoverageOptions;\n\n // For some reason the v8 provider builds its `excludes` values to include a null byte.\n // Remove null bytes for logging purposes so tools like grep won't complain about binary content.\n const sanitizedV8Options: ResolvedCoverageOptions = {\n ...resolvedV8Options,\n include: resolvedV8Options.include?.map(i => i.replace(/\\0/g, '')) || undefined,\n exclude: resolvedV8Options.exclude?.map(i => i.replace(/\\0/g, '')) || undefined\n };\n\n debug(`[HybridCoverageProvider] AS include: ${(this.definedCoverageOptions.assemblyScriptInclude || []).join(', ') || '(none)'}`);\n debug(`[HybridCoverageProvider] AS exclude: ${(this.definedCoverageOptions.assemblyScriptExclude || []).join(', ') || '(none)'}`);\n debug(`[HybridCoverageProvider] JS include: ${(sanitizedV8Options.include || []).join(', ') || '(none)'}`);\n debug(`[HybridCoverageProvider] JS exclude: ${(sanitizedV8Options.exclude || []).join(', ') || '(none)'}`);\n\n debug(`[HybridCoverageProvider] Globbing AS source files to include for coverage map basis`);\n const globbedAssemblyScriptInclude = globFiles(\n this.definedCoverageOptions.assemblyScriptInclude || [],\n this.definedCoverageOptions.assemblyScriptExclude || [],\n this.projectRoot\n );\n debug(`[HybridCoverageProvider] Including ${globbedAssemblyScriptInclude.length} AS files in coverage map`);\n \n const globbedAssemblyScriptExcludeOnly = globFiles(\n this.definedCoverageOptions.assemblyScriptExclude || [],\n [],\n this.projectRoot\n );\n debug(`[HybridCoverageProvider] Excluding ${globbedAssemblyScriptExcludeOnly.length} AS files from coverage map & instrumentation`);\n \n const resolved: ResolvedHybridProviderOptions = {\n ...resolvedV8Options,\n provider: 'custom',\n customProviderModule: this.definedCoverageOptions.customProviderModule || '',\n debugIstanbul: this.definedCoverageOptions.debugIstanbul ?? false,\n assemblyScriptInclude: this.definedCoverageOptions.assemblyScriptInclude ?? [],\n assemblyScriptExclude: this.definedCoverageOptions.assemblyScriptExclude ?? [],\n globbedAssemblyScriptInclude,\n globbedAssemblyScriptProjectRelativeExcludeOnly : globbedAssemblyScriptExcludeOnly.map(gr => gr.projectRootRelative),\n isResolved: true\n }; \n\n this.resolvedProviderOptions = resolved;\n return resolved;\n }\n\n async clean(clean: boolean = true): Promise<void> {\n debug('[HybridCoverageProvider] Clean coverage data - clean:', clean);\n if (clean) {\n this.accumulatedCoverageData = { hitCountsByFileAndPosition: {} };\n debug('[HybridCoverageProvider] Cleaned all internal coverage data');\n }\n\n if (this.v8Provider) {\n await this.v8Provider.clean(clean);\n debug(`[HybridCoverageProvider] V8 provider finished clean(${clean})`);\n }\n }\n}\n","/**\n * Coverage Provider Module Export\n *\n * This module exports the hybrid coverage provider for Vitest to load.\n * Users configure this via:\n * coverage.provider = 'custom'\n * coverage.customProviderModule = 'vitest-pool-assemblyscript/coverage'\n */\n\nimport { CoverageProviderModule } from 'vitest/node';\nimport v8CoverageModule from '@vitest/coverage-v8';\n\nimport { POOL_ERROR_NAMES } from '../types/constants.js';\nimport { createPoolError } from '../util/pool-errors.js';\nimport { HybridCoverageProvider } from './hybrid-coverage-provider.js';\n\n/**\n * Hybrid Coverage Provider\n *\n * This provider handles both AssemblyScript and JavaScript and coverage\n * - Converts AS coverage to Istanbul format\n * - Delegates JS coverage to Vitest's v8 provider\n * - Merges both into a unified coverage report\n */\nconst hybridProviderModule: CoverageProviderModule = {\n getProvider: (): HybridCoverageProvider => new HybridCoverageProvider(),\n\n startCoverage: async (runtimeOptions: {\n isolate: boolean\n }): Promise<unknown> => {\n if (v8CoverageModule.startCoverage) {\n return await v8CoverageModule.startCoverage(runtimeOptions);\n } else {\n throw createPoolError(\n POOL_ERROR_NAMES.HybridCoverageProviderError,\n 'HybridCoverageProvider - v8 coverage module does not provide `startCoverage`',\n );\n }\n },\n \n takeCoverage: async (runtimeOptions?: any): Promise<unknown> => {\n if (v8CoverageModule.takeCoverage) {\n return await v8CoverageModule.takeCoverage(runtimeOptions);\n } else {\n throw createPoolError(\n POOL_ERROR_NAMES.HybridCoverageProviderError,\n 'HybridCoverageProvider - v8 coverage module does not provide `takeCoverage`',\n );\n }\n },\n \n stopCoverage: async (runtimeOptions: {\n isolate: boolean\n }): Promise<unknown> => {\n if (v8CoverageModule.stopCoverage) {\n return await v8CoverageModule.stopCoverage(runtimeOptions);\n } else {\n throw createPoolError(\n POOL_ERROR_NAMES.HybridCoverageProviderError,\n 'HybridCoverageProvider - v8 coverage module does not provide `stopCoverage`',\n );\n }\n },\n};\n\nexport default hybridProviderModule;\n"],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;AA4CA,SAAgB,+BACd,sBACA,MACA,QACsC;CACtC,IAAI;CACJ,IAAI,gBAAgB;CACpB,IAAI,kBAAkB;AAGtB,MAAK,MAAM,CAAC,cAAc,cAAc,OAAO,QAAQ,qBAAqB,EAAE;EAC5E,MAAM,YAAY,OAAO,aAAa;AACtC,MAAI,YAAY,KAAM;AAEtB,OAAK,MAAM,QAAQ,WAAW;GAC5B,MAAM,EAAE,UAAU;AAGlB,OAAI,CAAC,kBAAkB,MAAM,QAAQ,MAAM,CAAE;AAG7C,OAAI,YAAY,iBACX,cAAc,iBAAiB,MAAM,cAAc,iBAAkB;AACxE,gBAAY;AACZ,oBAAgB;AAChB,sBAAkB,MAAM;;;;AAK9B,QAAO;;;;;;;;;;AAWT,SAAgB,kBACd,MACA,QACA,OACS;AAET,KAAI,OAAO,MAAM,aAAa,OAAO,MAAM,QAAS,QAAO;AAG3D,KAAI,SAAS,MAAM,aAAa,SAAS,MAAM,YAAa,QAAO;AAGnE,KAAI,SAAS,MAAM,WAAW,SAAS,MAAM,UAAW,QAAO;AAE/D,QAAO;;;;;;;;;;;;;;;;;;;;;;;AC3DT,eAAsB,wBACpB,0BACA,yBACA,kBACA,sBAC2B;CAC3B,MAAM,aAAa,YAAY,KAAK;CAEpC,SAAS,cAAc,GAAG,MAAmB;AAC3C,MAAI,qBACF,eAAc,GAAG,KAAK;;AAI1B,qBAAoB;EAClB,MAAM,sBAAsB,OAAO,OAAO,yBAAyB,CAAC,QAAQ,KAAK,UAAU,MAAM,MAAM,QAAQ,EAAE;EACjH,MAAM,oBAAoB,OAAO,KAAK,wBAAwB,CAAC;AAG/D,SAAO,kDAAkD,iBAAiB,mCACvC,oBAAoB,8BAA8B,kBAAkB,yFAH9E,wBAAwB,IAAI,YAAa,oBAAoB,MAAO,qBAAqB,QAAQ,EAAE,CAItC;GAEtF;CAIF,MAAM,gDAAgC,IAAI,KAAuC;AAGjF,MAAK,MAAM,CAAC,aAAa,aAAa,OAAO,QAAQ,wBAAwB,EAAE;EAE7E,MAAM,QAAQ,YAAY,MAAM,IAAI;EACpC,MAAM,UAAU,MAAM;EACtB,MAAM,YAAY,MAAM;AAExB,MAAI,WAAW,WAAW;GAIxB,MAAM,qBAAqB,+BAA+B,0BAH7C,SAAS,SAAS,GAAG,EACnB,SAAS,WAAW,GAAG,CAE2D;AACjG,OAAI,oBAAoB;IAEtB,MAAM,eAAe,8BAA8B,IAAI,mBAAmB;IAC1E,MAAM,oBAAoB,gBAAgB;IAC1C,MAAM,MAAM,KAAK,IAAI,mBAAmB,SAAS;AACjD,kCAA8B,IAAI,oBAAoB,IAAI;AAE1D,QAAI,iBAAiB,OACnB,eAAc,oCAAoC,YAAY,eAAe,mBAAmB,UAAU,mBAAmB,aAAa,cAAc,MAAM;QAE9J,eAAc,oCAAoC,YAAY,eAAe,mBAAmB,UAAU,WAAW,SAAS,GAAG;SAGnI,eAAc,oCAAoC,YAAY,6BAA6B;;;CAKjG,MAAM,eAAe,YAAY,KAAK;AACtC,eAAc,0EAA0E;CAGxF,MAAM,QAA4C,EAAE;CACpD,MAAM,IAA+B,EAAE;CACvC,MAAM,eAAyC,EAAE;CACjD,MAAM,IAA+B,EAAE;CACvC,MAAM,YAA8C,EAAE;CACtD,MAAM,IAAiC,EAAE;CAIzC,IAAI,UAAU;AACd,MAAK,MAAM,aAAa,OAAO,OAAO,yBAAyB,CAC7D,MAAK,MAAM,YAAY,WAAW;EAChC,MAAM,EAAE,OAAO,cAAc;AAG7B,MAAI,MAAM,cAAc,EACtB;EAIF,MAAM,WAAW,8BAA8B,IAAI,SAAS,IAAI;AAEhE,gBACE,mDAAmD,QAAQ,KAFpC,aAAa,cAAc,KAAK,YAAY,cAEc,YACnE,MAAM,UAAU,GAAG,MAAM,YAAY,KAAK,MAAM,QAAQ,GAAG,MAAM,UAAU,WAAW,WACrG;EAKD,MAAM,gBAAuB;GAC3B,OAAO;IAAE,MAAM,MAAM;IAAW,QAAQ,MAAM,cAAc;IAAG;GAC/D,KAAK;IAAE,MAAM,MAAM;IAAS,QAAQ,MAAM,YAAY;IAAG;GAC1D;EAED,MAAM,SAAS,QAAQ,UAAU;AACjC,QAAM,UAAU;GACd,MAAM;GACN,MAAM;GACN,KAAK;GACL,MAAM,MAAM;GACb;AACD,IAAE,UAAU;AAMZ,eAAa,UAAU;AACvB,IAAE,UAAU;AAEZ;;CAIJ,MAAM,OAAO,YAAY,KAAK;CAC9B,MAAM,cAAc,eAAe,YAAY,QAAQ,EAAE;CACzD,MAAM,aAAa,OAAO,cAAc,QAAQ,EAAE;CAClD,MAAM,WAAW,OAAO,YAAY,QAAQ,EAAE;AAE9C,eACE,sDAAsD,OAAO,KAAK,MAAM,CAAC,OAAO,cAC1E,QAAQ,uBAAuB,WAAW,gBAAgB,UAAU,MAC3E;AAED,QAAO;EACL,MAAM;EACN;EACA;EACA;EACA;EACA;EACA;EACD;;;;;;;;;;;;;;;;;;;;;;;AC1IH,IAAM,2BAAN,cAAuC,WAAW;;CAEhD,AAAQ;;CAER,AAAQ;;CAER,AAAQ;;CAER,AAAS,YAAwD,EAAE;;CAEnE,AAAQ,iBAA2B,EAAE;;CAErC,AAAQ,mBAAkC;CAE1C,YAAY,QAAgB,YAAoB,UAAkB;AAChE,SAAO;AACP,OAAK,SAAS;AACd,OAAK,aAAa;AAClB,OAAK,WAAW;;;;;;CAOlB,AAAU,iBAAiB,MAAkC;AAC3D,OAAK,eAAe,KAAK,KAAK,MAAM,QAAQ,YAAY;;;;;CAM1D,AAAU,gBAAgB,OAAmC;AAC3D,OAAK,eAAe,KAAK;;;;;;;CAQ3B,AAAU,aAAa,MAA8B;EACnD,MAAM,YAAY,KAAK,MAAM,QAAQ;AACrC,MAAI,KAAK,eAAe,SAAS,EAC/B,MAAK,mBAAmB,GAAG,KAAK,eAAe,KAAK,IAAI,CAAC,GAAG;MAE5D,MAAK,mBAAmB;;;;;CAO5B,AAAU,YAAY,OAA+B;AACnD,OAAK,mBAAmB;;;;;CAM1B,AAAU,sBAAsB,MAAoC;AAClE,MAAI,KAAK,QAAQ,KAAK,kBAAkB,KAAK,KAAK,EAAE;GAClD,MAAM,WAAW,KAAK,MAAM,QAAQ;GACpC,MAAM,YAAY,KAAK,eAAe,SAAS,IAC3C,GAAG,KAAK,eAAe,KAAK,IAAI,CAAC,GAAG,aACpC;GACJ,MAAM,gBAAgB,GAAG,KAAK,WAAW,GAAG;GAC5C,MAAM,QAAQ,KAAK,WAAW,MAAM,KAAK,QAAQ,KAAK;AACtD,QAAK,YAAY,eAAe,WAAW,MAAM;;AAEnD,SAAO;;;;;CAMT,AAAU,oBAAoB,MAAkC;AAC9D,MAAI,KAAK,QAAQ,KAAK,kBAAkB,KAAK,KAAK,EAAE;GAClD,MAAM,aAAa,KAAK,MAAM,QAAQ;GACtC,MAAM,YAAY,KAAK,oBAAoB;GAC3C,MAAM,QAAQ,KAAK;GAGnB,MAAM,YAAY,QAAQ,cAAc,YAAY;GACpD,MAAM,YAAY,QAAQ,cAAc,SAAS;GACjD,MAAM,YAAY,QAAQ,cAAc,SAAS;GAOjD,IAAI;AACJ,OAAI,SACF,aAAY,GAAG,UAAU,GAAG;YACnB,SACT,aAAY,GAAG,UAAU,OAAO;YACvB,SACT,aAAY,GAAG,UAAU,OAAO;OAEhC,aAAY,GAAG,UAAU,GAAG;GAG9B,MAAM,gBAAgB,GAAG,KAAK,WAAW,GAAG;GAC5C,MAAM,QAAQ,KAAK,WAAW,MAAM,KAAK,QAAQ,KAAK;AACtD,QAAK,YAAY,eAAe,WAAW,MAAM;;AAEnD,SAAO;;;;;CAMT,AAAU,sBAAsB,MAAoC;AAClE,MAAI,KAAK,eAAe,KAAK,YAAY,SAAS,WAAW,UAAU;GAErE,MAAM,WADW,KAAK,YACI;AAE1B,OAAI,SAAS,QAAQ,KAAK,kBAAkB,SAAS,KAAK,EAAE;IAE1D,MAAM,YAAY,KAAK,KAAK;IAC5B,MAAM,gBAAgB,GAAG,KAAK,WAAW,GAAG;IAG5C,MAAM,QAAqB;KACzB,UAAU,KAAK;KACf,WAAW,KAAK,OAAO,OAAO,KAAK,MAAM,MAAM;KAC/C,aAAa,KAAK,OAAO,UAAU;KACnC,SAAS,KAAK,OAAO,OAAO,KAAK,MAAM,IAAI;KAC3C,WAAW,KAAK,OAAO,UAAU;KAClC;AAED,SAAK,YAAY,eAAe,WAAW,MAAM;;AAInD,OAAI,SAAS,KACX,MAAK,UAAU,SAAS,KAAK;AAE/B,UAAO;;AAET,SAAO;;;;;CAMT,AAAQ,kBAAkB,MAAqB;AAC7C,MAAI,KAAK,SAAS,WAAW,MAE3B,QADkB,KACD,WAAW,SAAS;AAGvC,SAAO;;;;;CAMT,AAAQ,YAAY,eAAuB,WAAmB,OAA0B;EACtF,MAAM,YAAY,MAAM;AACxB,MAAI,CAAC,KAAK,UAAU,WAClB,MAAK,UAAU,aAAa,EAAE;AAEhC,OAAK,UAAU,WAAW,KAAK;GAAE;GAAe;GAAW;GAAO,CAAC;;;;;CAMrE,AAAQ,WAAW,MAAY,UAAoC;EACjE,MAAM,YAAY,YAAY;AAC9B,SAAO;GACL,UAAU,KAAK;GACf,WAAW,KAAK,OAAO,OAAO,UAAU,MAAM,MAAM;GACpD,aAAa,KAAK,OAAO,UAAU;GACnC,SAAS,KAAK,OAAO,OAAO,KAAK,MAAM,IAAI;GAC3C,WAAW,KAAK,OAAO,UAAU;GAClC;;;;;;;;;;AAWL,eAAsB,uBACpB,wBACA,wBACqD;CACrD,MAAM,aAAa,MAAM,SAAS,wBAAwB,OAAO;CAGjE,MAAM,SAASA,MAAU,uBAAuB;CAChD,MAAM,aAAa,OAAO,MAAM,GAAG,OAAO,IAAI,GAAG,OAAO,SAAS,OAAO;CAGxE,MAAM,WAAW,IAAIC,QAAsB;AAC3C,UAAS,UAAU,YAAY,wBAAwB,KAAK;CAE5D,MAAM,SAAS,SAAS;AACxB,KAAI,CAAC,OACH,QAAO,EAAE;CAIX,MAAM,UAAU,IAAI,yBAAyB,QAAQ,YAAY,uBAAuB;AACxF,SAAQ,YAAY,OAAO;AAE3B,QAAO,QAAQ,aAAa,EAAE;;;;;;;;;;;;;;AC1OhC,MAAM,oBAAoB;;;;;;;;;AAU1B,SAAgB,UACd,SACA,SACA,aACc;AACd,KAAI,QAAQ,WAAW,EACrB,QAAO,EAAE;AASX,QANsB,SAAS,SAAS;EACtC,KAAK;EACL,QAAQ,CAAC,mBAAmB,GAAG,QAAQ;EACvC,UAAU;EACX,CAAC,CAEmB,KAAK,cAAc;EACtC,UAAU,eAAe,SAAS;EAClC,qBAAqB,eAAe,SAAS,aAAa,SAAS,CAAC;EACrE,EAAE;;;;;;;;;;;;;ACxBL,MAAM,EAAE,sBAAsB;AAoB9B,IAAa,yBAAb,MAAgE;CAC9D,OAAO;CAEP,AAAQ;CACR,AAAQ,0BAAwC,EAAE,4BAA4B,EAAE,EAAE;CAClF,AAAQ,cAAsB;CAC9B,AAAQ,yBAAkD,EAAE;CAC5D,AAAQ,0BAAyD,EAAE;;;;CAKnE,MAAM,WAAW,KAA4B;AAC3C,OAAK,cAAc,IAAI,OAAO;AAC9B,OAAK,yBAAyB,IAAI,OAAO;AAEzC,QAAM,iDAAiD;AAGvD,OAAK,aAAa,MAAM,iBAAiB,aAAa;AAEtD,MAAI,CAAC,KAAK,WACR,OAAM,gBACJ,iBAAiB,6BACjB,iDACD;AAGH,QAAM,KAAK,WAAW,WAAW,IAAI;AACrC,OAAK,WAAW,OAAO;AACvB,QAAM,kEAAkE;AAExE,sCAAoC;AACpC,QAAM,yBAAyB;;;;;CAMjC,MAAM,gBAAgB,MAAwC;EAC5D,MAAM,QAAQ,YAAY,KAAK;EAC/B,MAAM,UAA8B,MAAM,WAAkB;EAC5D,IAAI,gBAAgB,KAAK,UAAU,SAAS,IAAI,SAAS,KAAK,UAAU,GAAI,GAAG;AAG/E,MAAI,WAAW,yBAAyB,gBAAgB;GAEtD,MAAM,EAAE,cAAc,eAAe,UADrB,KAAK;AAErB,mBAAgB;AAEhB,eAAY;IACV,MAAM,YAAY,OAAO,KAAK,aAAa,2BAA2B,CAAC;IACvE,MAAM,gBAAgB,OAAO,OAAO,aAAa,2BAA2B,CACzE,QAAQ,KAAK,cAAc,MAAM,OAAO,KAAK,UAAU,CAAC,QAAQ,EAAE;AACrE,WAAO,4BAA4B,cAAc,sCAAsC,cAAc,yBAAyB,UAAU;KACxI;AAGF,qBAAkB,KAAK,yBAAyB,aAAa;AAE7D,eAAY;IACV,MAAM,YAAY,OAAO,KAAK,KAAK,wBAAwB,2BAA2B,CAAC;IACvF,MAAM,gBAAgB,OAAO,OAAO,KAAK,wBAAwB,2BAA2B,CACzF,QAAQ,KAAK,cAAc,MAAM,OAAO,KAAK,UAAU,CAAC,QAAQ,EAAE;AACrE,WAAO,4BAA4B,cAAc,6CAA6C,cAAc,yBAAyB,UAAU;KAC/I;SACG;AAEL,OAAI,CAAC,KAAK,WACR,OAAM,gBACJ,iBAAiB,6BACjB,oDACD;AAGH,SAAM,4BAA4B,cAAc,8BAA8B;AAC9E,SAAM,KAAK,WAAW,gBAAgB,KAAK;;AAG7C,cAAY;GACV,MAAM,QAAQ,KAAK,UAAU,KAAI,OAAM,SAAS,KAAK,aAAa,GAAG,CAAC,CAAC,KAAK,IAAI;AAChF,UAAO,4BAA4B,cAAc,wCAAwC,YAAY,KAAK,GAAG,OAAO,QAAQ,EAAE,CAAC,oBAAoB,MAAM;IACzJ;;;;;;;;;;;;CAaJ,MAAM,iBAAiB,SAA0C;EAC/D,MAAM,QAAQ,YAAY,KAAK;AAE/B,QAAM,4DAA4D;AAElE,MAAI,CAAC,KAAK,WACR,OAAM,gBACJ,iBAAiB,6BACjB,qDACD;EAIH,IAAI,gBAAgB,mBAAmB;AAEvC,MAAI,KAAK,wBAAwB,6BAA6B,SAAS,GAAG;AACxE,SAAM,0DAA0D,KAAK,wBAAwB,6BAA6B,OAAO,gBAAgB;AACjJ,eAAY;AAIV,WAAO,0DAH0B,OAAO,OAAO,KAAK,wBAAwB,2BAA2B,CACpG,QAAQ,KAAK,cAAc,MAAM,OAAO,KAAK,UAAU,EAAE,QAAQ,EAAE,CAEoB,+BAD5E,OAAO,KAAK,KAAK,wBAAwB,2BAA2B,CAAC,OAC4C;KAC/H;GAGF,MAAM,yBAAyB,KAAK,wBAAwB,6BAA6B,IAAI,OAAO,YAAwB;AAC1H,UAAM,sEAAsE,QAAQ,SAAS,GAAG;IAEhG,MAAM,uBAAuB,MAAM,uBAAuB,QAAQ,UAAU,QAAQ,oBAAoB,IAAI,EAAE;AAC9G,UAAM,mCAAmC,OAAO,KAAK,qBAAqB,CAAC,OAAO,2BAA2B,QAAQ,SAAS,GAAG;IAEjI,MAAM,0BAA0B,KAAK,wBAAwB,2BAA2B,QAAQ,aAAa,EAAE;AAC/G,UAAM,wDAAwD,OAAO,KAAK,wBAAwB,CAAC,OAAO,kBAAkB,QAAQ,SAAS,GAAG;AAGhJ,WAAO,wBAAwB,sBAAsB,yBAAyB,QAAQ,UAAU,KAAK,wBAAwB,cAAc;KAC3I;GAGF,MAAM,kBAAkB,MAAM,QAAQ,IAAI,uBAAuB;AAGjE,QAAK,MAAM,gBAAgB,gBACzB,eAAc,gBAAgB,aAAa;AAG7C,SAAM,uDAAuD,OAAO,KAAK,cAAc,KAAK,CAAC,OAAO,QAAQ;QAE5G,OAAM,mHAAmH;EAG3H,MAAM,gBAAgB,YAAY,KAAK;AACvC,QAAM,yDAAyD,gBAAgB,OAAO,QAAQ,EAAE,CAAC,KAAK;AAGtG,QAAM,gEAAgE;EACtE,MAAM,aAAa,MAAM,KAAK,WAAW,iBAAiB,QAAQ;AAClE,QAAM,4CAA4C,OAAO,KAAK,WAAW,KAAK,CAAC,OAAO,QAAQ;AAC9F,QAAM,yDAAyD,YAAY,KAAK,GAAG,eAAe,QAAQ,EAAE,CAAC,KAAK;AAMlH,MAAI,cAAc,OAAO,CAAC,QAAQ;AAChC,SAAM,gEAAgE;AACtE,cAAW,MAAM,cAAc,QAAQ,CAAC;QAExC,OAAM,+EAA+E;AAEvF,QAAM,sDAAsD,OAAO,KAAK,WAAW,KAAK,CAAC,OAAO,QAAQ;AAExG,QAAM,4DAA4D,YAAY,KAAK,GAAG,OAAO,QAAQ,EAAE,CAAC,KAAK;AAE7G,SAAO;;;;;CAMT,MAAM,eAAe,aAAsB,SAAuC;AAChF,MAAI,CAAC,KAAK,WACR,OAAM,gBACJ,iBAAiB,6BACjB,mDACD;AAGH,QAAM,4DAA4D,QAAQ,YAAY,GAAG;AACzF,QAAM,KAAK,WAAW,eAAe,aAAa,QAAQ;;;;;CAM5D,iBAAgD;AAC9C,MAAI,CAAC,KAAK,WACR,OAAM,gBACJ,iBAAiB,6BACjB,mDACD;AAGH,QAAM,sDAAsD;EAE5D,MAAM,oBAAoB,KAAK,WAAW,gBAAgB;EAI1D,MAAM,qBAA8C;GAClD,GAAG;GACH,SAAS,kBAAkB,SAAS,KAAI,MAAK,EAAE,QAAQ,OAAO,GAAG,CAAC,IAAI;GACtE,SAAS,kBAAkB,SAAS,KAAI,MAAK,EAAE,QAAQ,OAAO,GAAG,CAAC,IAAI;GACvE;AAED,QAAM,yCAAyC,KAAK,uBAAuB,yBAAyB,EAAE,EAAE,KAAK,KAAK,IAAI,WAAW;AACjI,QAAM,yCAAyC,KAAK,uBAAuB,yBAAyB,EAAE,EAAE,KAAK,KAAK,IAAI,WAAW;AACjI,QAAM,yCAAyC,mBAAmB,WAAW,EAAE,EAAE,KAAK,KAAK,IAAI,WAAW;AAC1G,QAAM,yCAAyC,mBAAmB,WAAW,EAAE,EAAE,KAAK,KAAK,IAAI,WAAW;AAE1G,QAAM,sFAAsF;EAC5F,MAAM,+BAA+B,UACnC,KAAK,uBAAuB,yBAAyB,EAAE,EACvD,KAAK,uBAAuB,yBAAyB,EAAE,EACvD,KAAK,YACN;AACD,QAAM,sCAAsC,6BAA6B,OAAO,2BAA2B;EAE3G,MAAM,mCAAmC,UACvC,KAAK,uBAAuB,yBAAyB,EAAE,EACvD,EAAE,EACF,KAAK,YACN;AACD,QAAM,sCAAsC,iCAAiC,OAAO,+CAA+C;EAEnI,MAAM,WAA0C;GAC9C,GAAG;GACH,UAAU;GACV,sBAAsB,KAAK,uBAAuB,wBAAwB;GAC1E,eAAe,KAAK,uBAAuB,iBAAiB;GAC5D,uBAAuB,KAAK,uBAAuB,yBAAyB,EAAE;GAC9E,uBAAuB,KAAK,uBAAuB,yBAAyB,EAAE;GAC9E;GACA,iDAAkD,iCAAiC,KAAI,OAAM,GAAG,oBAAoB;GACpH,YAAY;GACb;AAED,OAAK,0BAA0B;AAC/B,SAAO;;CAGT,MAAM,MAAM,QAAiB,MAAqB;AAChD,QAAM,yDAAyD,MAAM;AACrE,MAAI,OAAO;AACT,QAAK,0BAA0B,EAAE,4BAA4B,EAAE,EAAE;AACjE,SAAM,8DAA8D;;AAGtE,MAAI,KAAK,YAAY;AACnB,SAAM,KAAK,WAAW,MAAM,MAAM;AAClC,SAAM,uDAAuD,MAAM,GAAG;;;;;;;;;;;;;;;AC/Q5E,MAAM,uBAA+C;CACnD,mBAA2C,IAAI,wBAAwB;CAEvE,eAAe,OAAO,mBAEE;AACtB,MAAI,iBAAiB,cACnB,QAAO,MAAM,iBAAiB,cAAc,eAAe;MAE3D,OAAM,gBACJ,iBAAiB,6BACjB,+EACD;;CAIL,cAAc,OAAO,mBAA2C;AAC9D,MAAI,iBAAiB,aACnB,QAAO,MAAM,iBAAiB,aAAa,eAAe;MAE1D,OAAM,gBACJ,iBAAiB,6BACjB,8EACD;;CAIL,cAAc,OAAO,mBAEG;AACtB,MAAI,iBAAiB,aACnB,QAAO,MAAM,iBAAiB,aAAa,eAAe;MAE1D,OAAM,gBACJ,iBAAiB,6BACjB,8EACD;;CAGN"}
1
+ {"version":3,"file":"index.mjs","names":["parsePath","AssemblyScriptParser"],"sources":["../../src/coverage-provider/containment-matcher.ts","../../src/coverage-provider/istanbul-converter.ts","../../src/coverage-provider/ast-parser.ts","../../src/coverage-provider/glob-utils.ts","../../src/coverage-provider/hybrid-coverage-provider.ts","../../src/coverage-provider/index.ts"],"sourcesContent":["/**\n * Containment Matcher\n *\n * Provides containment-based matching for coverage mapping.\n * Binary debug info provides points (representativeLocation), source parsing provides ranges.\n * We find which source range contains each binary point.\n *\n * Usage by version:\n * - v1: Function containment matching (binary representativeLocation → source function range)\n * - v2: Function containment (same) + statement containment + branch path containment\n *\n * Why containment matching (not direct position lookup):\n * AS compiler generates source map entries differently by statement type:\n * - Variable declarations: source map points to statement start (keyword)\n * - Control flow (if/switch/for): source map points to condition EXPRESSION, not keyword\n *\n * Example: `if (n < 0)` → binary reports column 7 ('n'), not column 3 ('i' of 'if')\n *\n * Containment matching is more robust:\n * - Binary gives us a position (representativeLocation) somewhere in the function\n * - Source gives us function ranges (start/end line/column)\n * - We find which source function range contains the binary position\n * - Handles nested functions with \"tightest fit\" (innermost function wins)\n */\n\nimport type { ParsedSourceFunctionInfo, SourceRange } from '../types/types.js';\n\n/**\n * Functions indexed by file path, then by start line.\n * Multiple functions can start on the same line (e.g., nested arrow functions).\n */\nexport type FunctionsByFileAndStartLine = Record<string, Record<number, ParsedSourceFunctionInfo[]>>;\n\n/**\n * Find the source function whose range contains the given position.\n *\n * For nested functions, uses \"tightest fit\" - returns the innermost function\n * (the one with the largest start position among all containing functions).\n *\n * @param functionsByStartLine - Functions for a single file, indexed by start line\n * @param line - Target line number (1-based)\n * @param column - Target column number (1-based)\n * @returns The containing function, or undefined if no match\n */\nexport function findFunctionContainingPosition(\n functionsByStartLine: Record<number, ParsedSourceFunctionInfo[]>,\n line: number,\n column: number\n): ParsedSourceFunctionInfo | undefined {\n let bestMatch: ParsedSourceFunctionInfo | undefined;\n let bestStartLine = -1;\n let bestStartColumn = -1;\n\n // Check functions starting on lines <= target line\n for (const [startLineStr, functions] of Object.entries(functionsByStartLine)) {\n const startLine = Number(startLineStr);\n if (startLine > line) continue;\n\n for (const func of functions) {\n const { range } = func;\n\n // Check if position is within this function's range\n if (!isPositionInRange(line, column, range)) continue;\n\n // Tightest fit: prefer function with largest start position (innermost)\n if (startLine > bestStartLine ||\n (startLine === bestStartLine && range.startColumn > bestStartColumn)) {\n bestMatch = func;\n bestStartLine = startLine;\n bestStartColumn = range.startColumn;\n }\n }\n }\n\n return bestMatch;\n}\n\n/**\n * Check if a position (line, column) falls within a source range.\n *\n * @param line - Target line number (1-based)\n * @param column - Target column number (1-based)\n * @param range - Source range to check against\n * @returns true if position is within range (inclusive)\n */\nexport function isPositionInRange(\n line: number,\n column: number,\n range: SourceRange\n): boolean {\n // Outside line range entirely\n if (line < range.startLine || line > range.endLine) return false;\n\n // On start line but before start column\n if (line === range.startLine && column < range.startColumn) return false;\n\n // On end line but after end column\n if (line === range.endLine && column > range.endColumn) return false;\n\n return true;\n}\n","/**\n * Istanbul Format Converter\n *\n * Converts AssemblyScript coverage data to Istanbul's FileCoverageData format.\n * This enables integration with Vitest's coverage reporting system and standard\n * coverage tools like Codecov, Coveralls, etc.\n *\n * Current Implementation: Function-level coverage only\n * - Uses containment matching: binary positions → source function ranges\n * - Each function maps to both a function entry AND a statement entry\n * - Statement coverage matches function coverage (function-level granularity)\n * - Branch coverage is 0% (no branches tracked yet)\n * - Line coverage derived from statement coverage\n *\n * Future Enhancement (v2): Block-level statement and branch coverage\n */\n\nimport type { FileCoverageData, Range, FunctionMapping, BranchMapping } from 'istanbul-lib-coverage';\nimport type { ParsedSourceFunctionInfo } from '../types/types.js';\nimport { findFunctionContainingPosition } from './containment-matcher.js';\nimport { debugOverride } from '../util/debug.js';\n\n/**\n * Convert AssemblyScript coverage data to Istanbul format for a single file\n *\n * Algorithm (containment matching):\n * 1. For each hit position in fileHitCountsByPosition:\n * - Use containment matcher to find which source function contains this position\n * - Record the hit count for that function\n * 2. For each function in fileFunctionsByStartLine:\n * - Add function mapping to fnMap\n * - Add function hit count to f (from matched hits, or 0 if not hit)\n * - Add corresponding statement mapping to statementMap\n * - Add same hit count to s (statement coverage matches function coverage)\n *\n * @param fileFunctionsByStartLine - Functions for this file, keyed by start line (from AST parser)\n * @param fileHitCountsByPosition - Hit counts for this file, keyed by position \"line:column\" (from accumulated coverage)\n * @param absoluteFilePath - Absolute path to the source file (for Istanbul output)\n * @returns Istanbul FileCoverage object\n */\nexport async function convertToIstanbulFormat(\n fileFunctionsByStartLine: Record<number, ParsedSourceFunctionInfo[]>,\n fileHitCountsByPosition: Record<string, number>,\n absoluteFilePath: string,\n istanbulDebugEnabled: boolean\n): Promise<FileCoverageData> {\n const startMatch = performance.now();\n\n function istanbulDebug(...args: any[]): void {\n if (istanbulDebugEnabled) {\n debugOverride(...args);\n }\n };\n\n istanbulDebug(() => {\n const sourceFunctionCount = Object.values(fileFunctionsByStartLine).reduce((sum, funcs) => sum + funcs.length, 0);\n const uniqueHitPosCount = Object.keys(fileHitCountsByPosition).length;\n const coverageEstimate = sourceFunctionCount === 0 ? Infinity : ((uniqueHitPosCount * 100) / sourceFunctionCount).toFixed(2);\n\n return `[IstanbulConverter] Processing source file: \"${absoluteFilePath}\"\\n`\n + `[IstanbulConverter] Source: ${sourceFunctionCount} total functions, Coverage: ${uniqueHitPosCount} unique hit positions\\n`\n + `[IstanbulConverter] Sanity Check - AS File Coverage Estimate: ${coverageEstimate}%\\n`\n + `[IstanbulConverter] Containment matching functions: coverage hit positions to source functions by range`;\n });\n\n // Build a map of function → hit count using containment matching\n // Key: function source identity (qualifiedName), Value: hit count\n const hitCountsBySourceFunctionName = new Map<ParsedSourceFunctionInfo, number>();\n\n // For each hit position, find which function contains it\n for (const [positionKey, hitCount] of Object.entries(fileHitCountsByPosition)) {\n // Position key format is \"line:column\"\n const parts = positionKey.split(':');\n const lineStr = parts[0];\n const columnStr = parts[1];\n\n if (lineStr && columnStr) {\n const line = parseInt(lineStr, 10);\n const column = parseInt(columnStr, 10);\n\n const containingFunction = findFunctionContainingPosition(fileFunctionsByStartLine, line, column);\n if (containingFunction) {\n // Accumulate hits (in case multiple positions map to same function)\n const existingHits = hitCountsBySourceFunctionName.get(containingFunction);\n const existingHitsCount = existingHits ?? 0;\n const max = Math.max(existingHitsCount, hitCount);\n hitCountsBySourceFunctionName.set(containingFunction, max); // <-- TODO: Explain this max logic\n\n if (existingHits !== undefined) {\n istanbulDebug(`[IstanbulConverter] Position ${positionKey} → function \"${containingFunction.shortName}\" EXISTING HITS: ${existingHits} NEW COUNT: ${max}`);\n } else {\n istanbulDebug(`[IstanbulConverter] Position ${positionKey} → function \"${containingFunction.shortName}\" (hits: ${hitCount})`);\n }\n } else {\n istanbulDebug(`[IstanbulConverter] Position ${positionKey} has no containing function`);\n }\n }\n }\n\n const startConvert = performance.now();\n istanbulDebug(`[IstanbulConverter] Matching Complete - Converting to Istanbul format`);\n\n // Initialize Istanbul data structures\n const fnMap: { [key: string]: FunctionMapping } = {};\n const f: { [key: string]: number } = {};\n const statementMap: { [key: string]: Range } = {};\n const s: { [key: string]: number } = {};\n const branchMap: { [key: string]: BranchMapping } = {};\n const b: { [key: string]: number[] } = {};\n\n // Convert function coverage to Istanbul format\n // Iterate all functions from parsed source (ensures 0-hit functions are included)\n let funcIdx = 0;\n for (const functions of Object.values(fileFunctionsByStartLine)) {\n for (const funcInfo of functions) {\n const { range, shortName } = funcInfo;\n\n // Defensive: skip functions with invalid metadata (shouldn't happen - AST parser filters these)\n if (range.startLine === 0) {\n continue;\n }\n\n // Get hit count from containment matching (or 0 if not hit)\n const hitCount = hitCountsBySourceFunctionName.get(funcInfo) ?? 0;\n const displayShortName = shortName && shortName !== '' ? shortName : '<anonymous>';\n istanbulDebug(\n `[IstanbulConverter] Istanbul function index ${funcIdx}: \"${displayShortName}\"`\n + ` (source ${range.startLine}:${range.startColumn} - ${range.endLine}:${range.endColumn}), hits: ${hitCount}`\n );\n\n // Create function mapping\n // Both 'decl' (declaration) and 'loc' (location) use the same range\n // Internal ParsedSourceFunctionInfo uses 1-based columns, Istanbul expects 0-based\n const istanbulRange: Range = {\n start: { line: range.startLine, column: range.startColumn - 1 },\n end: { line: range.endLine, column: range.endColumn - 1 }\n };\n\n const idxStr = funcIdx.toString();\n fnMap[idxStr] = {\n name: shortName,\n decl: istanbulRange,\n loc: istanbulRange,\n line: range.startLine\n };\n f[idxStr] = hitCount;\n\n // Create corresponding statement mapping\n // For function-level coverage, each function is one \"statement\"\n // The statement range matches the function range\n // This gives us statement coverage at function granularity\n statementMap[idxStr] = istanbulRange;\n s[idxStr] = hitCount;\n\n funcIdx++;\n }\n }\n\n const done = performance.now();\n const matchingMs = (startConvert - startMatch).toFixed(2);\n const convertMs = (done - startConvert).toFixed(2);\n const totalMs = (done - startMatch).toFixed(2);\n\n istanbulDebug(\n `[IstanbulConverter] Coverage Coversion Complete: ${Object.keys(fnMap).length} functions,` \n + ` ${totalMs} ms total (matching: ${matchingMs} ms, convert: ${convertMs} ms)`\n );\n\n return {\n path: absoluteFilePath,\n fnMap,\n f,\n statementMap,\n s,\n branchMap,\n b\n };\n}\n","/**\n * AST Parser for AssemblyScript Source Files\n *\n * Parses AS source files to extract function metadata for coverage.\n * Used by generateCoverage to build empty coverage map from all source files.\n *\n * Source AST is the source of truth for what SHOULD be covered.\n * Binary instrumentation tells us what we CAN measure (hit counts).\n *\n * Functions are grouped by start line for efficient containment matching.\n *\n * Architecture:\n * - Uses shared ASTVisitor for complete NodeKind coverage\n * - Overrides hooks to extract function information during traversal\n */\n\nimport { readFile } from 'node:fs/promises';\nimport { parse as parsePath } from 'node:path';\nimport {\n Parser as AssemblyScriptParser,\n Source,\n BlockStatement,\n Node,\n FunctionDeclaration,\n MethodDeclaration,\n ClassDeclaration,\n NamespaceDeclaration,\n VariableDeclaration,\n FunctionExpression,\n} from 'assemblyscript';\n\nimport type { ParsedSourceFunctionInfo, SourceRange } from '../types/types.js';\nimport { ASCommonFlags, ASNodeKind } from '../types/constants.js';\nimport { ASTVisitor } from '../util/ast-visitor.js';\n\n/**\n * Visitor that extracts function information from AST nodes\n */\nclass FunctionExtractorVisitor extends ASTVisitor {\n /** Source file being parsed */\n private source: Source;\n /** Module path for building qualified names */\n private modulePath: string;\n /** Absolute file path */\n private filePath: string;\n /** Accumulated function records, keyed by start line */\n readonly functions: Record<number, ParsedSourceFunctionInfo[]> = {};\n /** Namespace context stack (supports nested namespaces) */\n private namespaceStack: string[] = [];\n /** Current class name (when inside a class) */\n private currentClassName: string | null = null;\n\n constructor(source: Source, modulePath: string, filePath: string) {\n super();\n this.source = source;\n this.modulePath = modulePath;\n this.filePath = filePath;\n }\n\n /**\n * Track namespace context when entering a namespace.\n * Supports nesting — push onto the stack.\n */\n protected onNamespaceEnter(node: NamespaceDeclaration): void {\n this.namespaceStack.push(node.name?.text ?? 'Anonymous');\n }\n\n /**\n * Restore namespace context when exiting a namespace.\n */\n protected onNamespaceExit(_node: NamespaceDeclaration): void {\n this.namespaceStack.pop();\n }\n\n /**\n * Track class context when entering a class.\n * Includes namespace prefix when inside a namespace, matching binary naming convention\n * (e.g. \"NS_A.Item\" for class Item inside namespace NS_A).\n */\n protected onClassEnter(node: ClassDeclaration): void {\n const className = node.name?.text ?? 'Anonymous';\n if (this.namespaceStack.length > 0) {\n this.currentClassName = `${this.namespaceStack.join('.')}.${className}`;\n } else {\n this.currentClassName = className;\n }\n }\n\n /**\n * Restore class context when exiting a class\n */\n protected onClassExit(_node: ClassDeclaration): void {\n this.currentClassName = null;\n }\n\n /**\n * Extract function info from function declarations\n */\n protected onFunctionDeclaration(node: FunctionDeclaration): boolean {\n if (node.body && this.hasBodyStatements(node.body)) {\n const funcName = node.name?.text ?? '~anonymous';\n const shortName = this.namespaceStack.length > 0\n ? `${this.namespaceStack.join('.')}.${funcName}`\n : funcName;\n const qualifiedName = `${this.modulePath}/${shortName}`;\n const range = this.buildRange(node, node.name ?? null);\n this.addFunction(qualifiedName, shortName, range);\n }\n return true; // Continue recursion into body\n }\n\n /**\n * Extract function info from method declarations\n */\n protected onMethodDeclaration(node: MethodDeclaration): boolean {\n if (node.body && this.hasBodyStatements(node.body)) {\n const methodName = node.name?.text ?? 'constructor';\n const className = this.currentClassName ?? 'Unknown';\n const flags = node.flags;\n\n // Determine method type from flags\n const isStatic = (flags & ASCommonFlags.Static) !== 0;\n const isGetter = (flags & ASCommonFlags.Get) !== 0;\n const isSetter = (flags & ASCommonFlags.Set) !== 0;\n\n // Build short name to match binary naming convention:\n // - Static: ClassName.methodName\n // - Getter: ClassName#get:propertyName\n // - Setter: ClassName#set:propertyName\n // - Instance: ClassName#methodName\n let shortName: string;\n if (isStatic) {\n shortName = `${className}.${methodName}`;\n } else if (isGetter) {\n shortName = `${className}#get:${methodName}`;\n } else if (isSetter) {\n shortName = `${className}#set:${methodName}`;\n } else {\n shortName = `${className}#${methodName}`;\n }\n\n const qualifiedName = `${this.modulePath}/${shortName}`;\n const range = this.buildRange(node, node.name ?? null);\n this.addFunction(qualifiedName, shortName, range);\n }\n return true; // Continue recursion into body\n }\n\n /**\n * Extract function info from variable declarations (arrow functions)\n */\n protected onVariableDeclaration(node: VariableDeclaration): boolean {\n if (node.initializer && node.initializer.kind === ASNodeKind.Function) {\n const funcExpr = node.initializer as FunctionExpression;\n const funcDecl = funcExpr.declaration;\n\n if (funcDecl.body && this.hasBodyStatements(funcDecl.body)) {\n // Use variable name for the function\n const shortName = node.name.text;\n const qualifiedName = `${this.modulePath}/${shortName}`;\n\n // Use the variable declaration's range\n const range: SourceRange = {\n filePath: this.filePath,\n startLine: this.source.lineAt(node.range.start),\n startColumn: this.source.columnAt(),\n endLine: this.source.lineAt(node.range.end),\n endColumn: this.source.columnAt(),\n };\n\n this.addFunction(qualifiedName, shortName, range);\n }\n\n // Visit the function body manually since we're handling this specially\n if (funcDecl.body) {\n this.visitNode(funcDecl.body);\n }\n return false; // Don't recurse again - we handled it\n }\n return true; // Continue recursion for non-function initializers\n }\n\n /**\n * Check if a function body has statements (non-empty body)\n */\n private hasBodyStatements(body: Node): boolean {\n if (body.kind === ASNodeKind.Block) {\n const blockBody = body as BlockStatement;\n return blockBody.statements.length > 0;\n }\n // Expression body (braceless arrow) - always has the expression\n return true;\n }\n\n /**\n * Add a function to the functions record, keyed by start line\n */\n private addFunction(qualifiedName: string, shortName: string, range: SourceRange): void {\n const startLine = range.startLine;\n if (!this.functions[startLine]) {\n this.functions[startLine] = [];\n }\n this.functions[startLine].push({ qualifiedName, shortName, range });\n }\n\n /**\n * Build a SourceRange for a node, using name.range.start to skip decorators\n */\n private buildRange(node: Node, nameNode: Node | null): SourceRange {\n const startNode = nameNode ?? node;\n return {\n filePath: this.filePath,\n startLine: this.source.lineAt(startNode.range.start),\n startColumn: this.source.columnAt(),\n endLine: this.source.lineAt(node.range.end),\n endColumn: this.source.columnAt(),\n };\n }\n}\n\n/**\n * Parse functions from a single AS source file\n *\n * @param absoluteSourceFilePath - Absolute path to AS source file\n * @param relativeSourceFilePath - Relative path to AS source file (derived once in caller and used several places)\n * @returns Record of start line to array of ParsedSourceFunctionInfo (multiple functions can start on same line)\n */\nexport async function parseFunctionsFromFile(\n absoluteSourceFilePath: string,\n relativeSourceFilePath: string,\n): Promise<Record<number, ParsedSourceFunctionInfo[]>> {\n const sourceCode = await readFile(absoluteSourceFilePath, 'utf8');\n\n // Build the module path (strip any extension, use forward slashes)\n const parsed = parsePath(relativeSourceFilePath);\n const modulePath = parsed.dir ? `${parsed.dir}/${parsed.name}` : parsed.name;\n\n // Parse with AssemblyScript parser\n const asParser = new AssemblyScriptParser();\n asParser.parseFile(sourceCode, relativeSourceFilePath, true);\n\n const source = asParser.currentSource;\n if (!source) {\n return {};\n }\n\n // Create visitor and traverse\n const visitor = new FunctionExtractorVisitor(source, modulePath, absoluteSourceFilePath);\n visitor.visitSource(source);\n\n return visitor.functions || {};\n}\n","/**\n * Glob Utilities for Coverage\n *\n * Uses tinyglobby to glob AssemblyScript files matching coverage include/exclude patterns.\n * Supports ../ path traversals for projects that import source from outside the project root.\n *\n * Note: tinyglobby docs warn of a performance cost when globbing outside cwd (e.g. ../ patterns)\n * due to recalculating relative paths. Acceptable here since this runs once at startup.\n */\n\nimport { relative } from 'node:path';\nimport { globSync } from 'tinyglobby';\n\nimport { GlobResult } from '../types/types.js';\nimport { toForwardSlash } from '../util/path-utils.js';\n\nconst NODE_MODULES_GLOB = '**/node_modules/**';\n\n/**\n * Glob files matching coverage include/exclude patterns\n *\n * @param include - Include patterns (e.g., ['assembly/**\\/*.ts'])\n * @param exclude - Exclude patterns (e.g., ['**\\/*.test.ts'])\n * @param projectRoot - Project root directory\n * @returns Array of glob results with absolute and project-root-relative paths\n */\nexport function globFiles(\n include: string[],\n exclude: string[],\n projectRoot: string\n): GlobResult[] {\n if (include.length === 0) {\n return [];\n }\n\n const absolutePaths = globSync(include, {\n cwd: projectRoot,\n ignore: [NODE_MODULES_GLOB, ...exclude],\n absolute: true,\n });\n\n return absolutePaths.map((absolute) => ({\n absolute: toForwardSlash(absolute),\n projectRootRelative: toForwardSlash(relative(projectRoot, absolute)),\n }));\n}\n","/**\n * Hybrid Coverage Provider\n *\n * This provider handles both AssemblyScript and JavaScript and coverage\n * - Converts AS coverage to Istanbul format\n * - Delegates JS coverage to Vitest's v8 provider\n * - Merges both into a unified coverage report\n */\n\nimport { basename, relative } from 'node:path';\nimport type {\n CoverageProvider,\n Vitest,\n ReportContext,\n ResolvedCoverageOptions,\n} from 'vitest/node';\nimport type { AfterSuiteRunMeta } from 'vitest';\nimport v8CoverageModule from '@vitest/coverage-v8';\nimport type { CoverageMap } from 'istanbul-lib-coverage';\nimport istanbulCoverage from 'istanbul-lib-coverage';\nconst { createCoverageMap } = istanbulCoverage;\n\nimport { convertToIstanbulFormat } from './istanbul-converter.js';\nimport { parseFunctionsFromFile } from './ast-parser.js';\nimport { globFiles } from './glob-utils.js';\nimport { mergeCoverageData } from './coverage-merge.js';\nimport { debug } from '../util/debug.js';\nimport { createPoolError } from '../util/pool-errors.js';\nimport type {\n AssemblyScriptCoveragePayload,\n CoverageData,\n GlobResult,\n ResolvedHybridProviderOptions,\n} from '../types/types.js';\nimport {\n POOL_ERROR_NAMES,\n COVERAGE_PAYLOAD_FORMATS\n} from '../types/constants.js';\nimport { warnIfASCoverageNotSupportedByNode, warnIfNativeBuildFailed } from '../util/feature-check.js';\n\nexport class HybridCoverageProvider implements CoverageProvider {\n name = 'hybrid-assemblyscript-v8' as const;\n\n private v8Provider: CoverageProvider | undefined;\n private accumulatedCoverageData: CoverageData = { hitCountsByFileAndPosition: {} };\n private projectRoot: string = '';\n private definedCoverageOptions: ResolvedCoverageOptions = {} as ResolvedCoverageOptions;\n private resolvedProviderOptions: ResolvedHybridProviderOptions = {} as ResolvedHybridProviderOptions;\n\n /**\n * Initialize the provider and get reference to v8 provider\n */\n async initialize(ctx: Vitest): Promise<void> {\n this.projectRoot = ctx.config.root;\n this.definedCoverageOptions = ctx.config.coverage;\n\n debug('[HybridCoverageProvider] Initializing Provider');\n\n // Get v8 provider from the coverage module\n this.v8Provider = await v8CoverageModule.getProvider();\n\n if (!this.v8Provider) {\n throw createPoolError(\n POOL_ERROR_NAMES.HybridCoverageProviderError,\n 'initialize failed to get delegated v8 provider',\n );\n }\n\n await this.v8Provider.initialize(ctx);\n this.v8Provider.name = 'hybrid-assemblyscript-v8 (delegated v8 reporter)' as const;\n debug('[HybridCoverageProvider] Initialized with delegated v8 provider');\n\n warnIfASCoverageNotSupportedByNode();\n await warnIfNativeBuildFailed();\n }\n\n /**\n * Handle suite completion - delegate based on coverage format marker\n */\n async onAfterSuiteRun(meta: AfterSuiteRunMeta): Promise<void> {\n const start = performance.now();\n const format: string | undefined = (meta?.coverage as any)?.__format;\n let suiteLogLabel = meta.testFiles.length > 0 ? basename(meta.testFiles[0]!) : '';\n\n // Check for AssemblyScript format marker\n if (format === COVERAGE_PAYLOAD_FORMATS.AssemblyScript) {\n const payload = meta.coverage as AssemblyScriptCoveragePayload;\n const { coverageData, suiteLogLabel: label } = payload;\n suiteLogLabel = label;\n\n debug(() => {\n const fileCount = Object.keys(coverageData.hitCountsByFileAndPosition).length;\n const positionCount = Object.values(coverageData.hitCountsByFileAndPosition)\n .reduce((sum, positions) => sum + Object.keys(positions).length, 0);\n return `[HybridCoverageProvider] ${suiteLogLabel} - onAfterSuiteRun - Suite payload: ${positionCount} unique positions over ${fileCount} source files`;\n });\n\n // Merge incoming coverage data into accumulated (by position, summing hit counts)\n mergeCoverageData(this.accumulatedCoverageData, coverageData);\n\n debug(() => {\n const fileCount = Object.keys(this.accumulatedCoverageData.hitCountsByFileAndPosition).length;\n const positionCount = Object.values(this.accumulatedCoverageData.hitCountsByFileAndPosition)\n .reduce((sum, positions) => sum + Object.keys(positions).length, 0);\n return `[HybridCoverageProvider] ${suiteLogLabel} - onAfterSuiteRun - Accumulated coverage: ${positionCount} unique positions over ${fileCount} source files`;\n });\n } else {\n // Delegate to v8 provider for all other formats (JS, etc.)\n if (!this.v8Provider) {\n throw createPoolError(\n POOL_ERROR_NAMES.HybridCoverageProviderError,\n 'onAfterSuiteRun failed to delegate to v8 provider',\n );\n }\n\n debug(`[HybridCoverageProvider] ${suiteLogLabel} - Delegating to v8 provider`);\n await this.v8Provider.onAfterSuiteRun(meta);\n }\n\n debug(() => {\n const files = meta.testFiles.map(tf => relative(this.projectRoot, tf)).join(',');\n return `[HybridCoverageProvider] ${suiteLogLabel} - onAfterSuiteRun complete - TIMING ${(performance.now() - start).toFixed(2)} ms | testFiles: \"${files}\"`;\n });\n }\n\n /**\n * Generate unified coverage map (merging JS and AS coverage)\n *\n * Flow:\n * 1. Parse included AS source files to get sourceDebugInfo (source of truth for line numbers)\n * 1. Build merged CoverageData (all source functions + accumulated hit counts)\n * 4. Convert merged CoverageData to Istanbul format\n * 5. Get JS coverage from v8 provider\n * 6. Merge AS coverage into JS coverage\n */\n async generateCoverage(context: ReportContext): Promise<unknown> {\n const start = performance.now();\n\n debug('[HybridCoverageProvider] Generating coverage for test run');\n\n if (!this.v8Provider) {\n throw createPoolError(\n POOL_ERROR_NAMES.HybridCoverageProviderError,\n 'generateCoverage failed to delegate to v8 provider',\n );\n }\n\n // Build AS coverage map\n let asCoverageMap = createCoverageMap();\n\n if (this.resolvedProviderOptions.globbedAssemblyScriptInclude.length > 0) {\n debug(`[HybridCoverageProvider] Building AS coverage map with ${this.resolvedProviderOptions.globbedAssemblyScriptInclude.length} source files `);\n debug(() => {\n const accumulatedPositionCount = Object.values(this.accumulatedCoverageData.hitCountsByFileAndPosition)\n .reduce((sum, positions) => sum + Object.keys(positions)?.length, 0);\n const files = Object.keys(this.accumulatedCoverageData.hitCountsByFileAndPosition).length;\n return `[HybridCoverageProvider] Accumulated coverage data has ${accumulatedPositionCount} unique positions hit across ${files} debug source files`;\n });\n\n // parse source files with AST parser, then match to hits and convert to istanbul format\n const fileProcessingPromises = this.resolvedProviderOptions.globbedAssemblyScriptInclude.map(async (include: GlobResult) => {\n debug(`[HybridCoverageProvider] Parsing AS source for expected coverage: \"${include.absolute}\"`);\n \n const functionsByStartLine = await parseFunctionsFromFile(include.absolute, include.projectRootRelative) || {};\n debug(`[HybridCoverageProvider] Parsed ${Object.keys(functionsByStartLine).length} AS source functions in \"${include.absolute}\"`);\n\n const fileHitCountsByPosition = this.accumulatedCoverageData.hitCountsByFileAndPosition[include.absolute] ?? {};\n debug(`[HybridCoverageProvider] Accumulated AS coverage has ${Object.keys(fileHitCountsByPosition).length} positions for \"${include.absolute}\"`);\n\n // Containment matching (binary hit position → source) is performed during istanbul conversion\n return convertToIstanbulFormat(functionsByStartLine, fileHitCountsByPosition, include.absolute, this.resolvedProviderOptions.debugIstanbul);\n });\n\n // Wait for all files to complete\n const istanbulResults = await Promise.all(fileProcessingPromises);\n\n // Add all results to coverage map\n for (const istanbulData of istanbulResults) {\n asCoverageMap.addFileCoverage(istanbulData);\n }\n\n debug(`[HybridCoverageProvider] Built AS coverage map with ${Object.keys(asCoverageMap.data).length} files`);\n } else {\n debug('[HybridCoverageProvider] WARNING: No assemblyScriptInclude patterns yieldled files - Coverage Map will be empty!');\n }\n\n const asGenerateEnd = performance.now();\n debug(`[HybridCoverageProvider] TIMING AS generateCoverage: ${(asGenerateEnd - start).toFixed(2)} ms`);\n\n // Get JS coverage from v8 provider\n debug('[HybridCoverageProvider] Getting JS coverage from v8 provider');\n const jsCoverage = await this.v8Provider.generateCoverage(context) as CoverageMap;\n debug(`[HybridCoverageProvider] JS coverage has ${Object.keys(jsCoverage.data).length} files`);\n debug(`[HybridCoverageProvider] TIMING JS generateCoverage: ${(performance.now() - asGenerateEnd).toFixed(2)} ms`);\n\n // Merge AS coverage into JS coverage.\n // Use toJSON() to pass plain data instead of the CoverageMap instance. This avoids\n // instanceof failures when istanbul-lib-coverage is loaded from different module paths\n // (e.g. the pool workers resolve one copy while the coverage provider resolves another).\n if (asCoverageMap.files().length) {\n debug('[HybridCoverageProvider] Merging AS coverage into JS coverage');\n jsCoverage.merge(asCoverageMap.toJSON());\n } else {\n debug('[HybridCoverageProvider] AS coverage map empty, not merging into JS coverage');\n }\n debug(`[HybridCoverageProvider] Final merged coverage has ${Object.keys(jsCoverage.data).length} files`);\n\n debug(`[HybridCoverageProvider] TIMING Total generateCoverage: ${(performance.now() - start).toFixed(2)} ms`);\n\n return jsCoverage;\n }\n\n /**\n * Report coverage - delegate to v8 provider\n */\n async reportCoverage(coverageMap: unknown, context: ReportContext): Promise<void> {\n if (!this.v8Provider) {\n throw createPoolError(\n POOL_ERROR_NAMES.HybridCoverageProviderError,\n 'reportCoverage failed to delegate to v8 provider',\n );\n }\n\n debug(`[HybridCoverageProvider] Reporting coverage (allTestsRun=${context.allTestsRun})`);\n await this.v8Provider.reportCoverage(coverageMap, context);\n }\n\n /**\n * Resolve options\n */\n resolveOptions(): ResolvedHybridProviderOptions {\n if (!this.v8Provider) {\n throw createPoolError(\n POOL_ERROR_NAMES.HybridCoverageProviderError,\n 'resolveOptions failed to delegate to v8 provider',\n );\n }\n \n debug(`[HybridCoverageProvider] Resolving Coverage Options`);\n \n const resolvedV8Options = this.v8Provider.resolveOptions() as ResolvedCoverageOptions;\n\n // For some reason the v8 provider builds its `excludes` values to include a null byte.\n // Remove null bytes for logging purposes so tools like grep won't complain about binary content.\n const sanitizedV8Options: ResolvedCoverageOptions = {\n ...resolvedV8Options,\n include: resolvedV8Options.include?.map(i => i.replace(/\\0/g, '')) || undefined,\n exclude: resolvedV8Options.exclude?.map(i => i.replace(/\\0/g, '')) || undefined\n };\n\n debug(`[HybridCoverageProvider] AS include: ${(this.definedCoverageOptions.assemblyScriptInclude || []).join(', ') || '(none)'}`);\n debug(`[HybridCoverageProvider] AS exclude: ${(this.definedCoverageOptions.assemblyScriptExclude || []).join(', ') || '(none)'}`);\n debug(`[HybridCoverageProvider] JS include: ${(sanitizedV8Options.include || []).join(', ') || '(none)'}`);\n debug(`[HybridCoverageProvider] JS exclude: ${(sanitizedV8Options.exclude || []).join(', ') || '(none)'}`);\n\n debug(`[HybridCoverageProvider] Globbing AS source files to include for coverage map basis`);\n const globbedAssemblyScriptInclude = globFiles(\n this.definedCoverageOptions.assemblyScriptInclude || [],\n this.definedCoverageOptions.assemblyScriptExclude || [],\n this.projectRoot\n );\n debug(`[HybridCoverageProvider] Including ${globbedAssemblyScriptInclude.length} AS files in coverage map`);\n \n const globbedAssemblyScriptExcludeOnly = globFiles(\n this.definedCoverageOptions.assemblyScriptExclude || [],\n [],\n this.projectRoot\n );\n debug(`[HybridCoverageProvider] Excluding ${globbedAssemblyScriptExcludeOnly.length} AS files from coverage map & instrumentation`);\n \n const resolved: ResolvedHybridProviderOptions = {\n ...resolvedV8Options,\n provider: 'custom',\n customProviderModule: this.definedCoverageOptions.customProviderModule || '',\n debugIstanbul: this.definedCoverageOptions.debugIstanbul ?? false,\n assemblyScriptInclude: this.definedCoverageOptions.assemblyScriptInclude ?? [],\n assemblyScriptExclude: this.definedCoverageOptions.assemblyScriptExclude ?? [],\n globbedAssemblyScriptInclude,\n globbedAssemblyScriptProjectRelativeExcludeOnly : globbedAssemblyScriptExcludeOnly.map(gr => gr.projectRootRelative),\n isResolved: true\n }; \n\n this.resolvedProviderOptions = resolved;\n return resolved;\n }\n\n async clean(clean: boolean = true): Promise<void> {\n debug('[HybridCoverageProvider] Clean coverage data - clean:', clean);\n if (clean) {\n this.accumulatedCoverageData = { hitCountsByFileAndPosition: {} };\n debug('[HybridCoverageProvider] Cleaned all internal coverage data');\n }\n\n if (this.v8Provider) {\n await this.v8Provider.clean(clean);\n debug(`[HybridCoverageProvider] V8 provider finished clean(${clean})`);\n }\n }\n}\n","/**\n * Coverage Provider Module Export\n *\n * This module exports the hybrid coverage provider for Vitest to load.\n * Users configure this via:\n * coverage.provider = 'custom'\n * coverage.customProviderModule = 'vitest-pool-assemblyscript/coverage'\n */\n\nimport { CoverageProviderModule } from 'vitest/node';\nimport v8CoverageModule from '@vitest/coverage-v8';\n\nimport { POOL_ERROR_NAMES } from '../types/constants.js';\nimport { createPoolError } from '../util/pool-errors.js';\nimport { HybridCoverageProvider } from './hybrid-coverage-provider.js';\n\n/**\n * Hybrid Coverage Provider\n *\n * This provider handles both AssemblyScript and JavaScript and coverage\n * - Converts AS coverage to Istanbul format\n * - Delegates JS coverage to Vitest's v8 provider\n * - Merges both into a unified coverage report\n */\nconst hybridProviderModule: CoverageProviderModule = {\n getProvider: (): HybridCoverageProvider => new HybridCoverageProvider(),\n\n startCoverage: async (runtimeOptions: {\n isolate: boolean,\n autoAttachSubprocess: boolean,\n reportsDirectory: string\n }): Promise<unknown> => {\n if (v8CoverageModule.startCoverage) {\n return await v8CoverageModule.startCoverage(runtimeOptions);\n } else {\n throw createPoolError(\n POOL_ERROR_NAMES.HybridCoverageProviderError,\n 'HybridCoverageProvider - v8 coverage module does not provide `startCoverage`',\n );\n }\n },\n \n takeCoverage: async (runtimeOptions?: any): Promise<unknown> => {\n if (v8CoverageModule.takeCoverage) {\n return await v8CoverageModule.takeCoverage(runtimeOptions);\n } else {\n throw createPoolError(\n POOL_ERROR_NAMES.HybridCoverageProviderError,\n 'HybridCoverageProvider - v8 coverage module does not provide `takeCoverage`',\n );\n }\n },\n \n stopCoverage: async (runtimeOptions: {\n isolate: boolean\n }): Promise<unknown> => {\n if (v8CoverageModule.stopCoverage) {\n return await v8CoverageModule.stopCoverage(runtimeOptions);\n } else {\n throw createPoolError(\n POOL_ERROR_NAMES.HybridCoverageProviderError,\n 'HybridCoverageProvider - v8 coverage module does not provide `stopCoverage`',\n );\n }\n },\n};\n\nexport default hybridProviderModule;\n"],"mappings":"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"}
@@ -1,4 +1,4 @@
1
- import { AssemblyScriptCompilerOptions, AssemblyScriptPoolOptions, WASMCompilation } from "./types-CoroKYxB.mjs";
1
+ import { AssemblyScriptCompilerOptions, AssemblyScriptPoolOptions, WASMCompilation } from "./types-CRbjiCC7.mjs";
2
2
 
3
3
  //#region src/compiler/index.d.ts
4
4
  /**
package/dist/index-v3.mjs CHANGED
@@ -1,5 +1,5 @@
1
1
  import { POOL_ERROR_NAMES } from "./constants-Bq5KNxXJ.mjs";
2
- import { createInitialFileTask, failTestWithTimeoutError, flagTestTerminated, getCompatConfig, getConfigs } from "./vitest-file-tasks-vvZzigcF.mjs";
2
+ import { createInitialFileTask, failTestWithTimeoutError, flagTestTerminated, getCompatConfig, getConfigs } from "./vitest-file-tasks-BBsZ_wS6.mjs";
3
3
  import { debug, isAbortError, setGlobalDebugMode } from "./pool-errors-Bn6YaguR.mjs";
4
4
  import { createWorkerRPCChannel } from "./worker-rpc-channel-CvCrc8aa.mjs";
5
5
  import { basename, resolve } from "node:path";
package/dist/index.d.mts CHANGED
@@ -1,2 +1,2 @@
1
- import { createAssemblyScriptPool } from "./pool-runner-init-CCvnKt5o.mjs";
1
+ import { createAssemblyScriptPool } from "./pool-runner-init-CPQL6pVL.mjs";
2
2
  export { createAssemblyScriptPool };
package/dist/index.mjs CHANGED
@@ -1,7 +1,7 @@
1
1
  import "./constants-Bq5KNxXJ.mjs";
2
- import "./vitest-file-tasks-vvZzigcF.mjs";
2
+ import "./vitest-file-tasks-BBsZ_wS6.mjs";
3
3
  import "./pool-errors-Bn6YaguR.mjs";
4
4
  import "./worker-rpc-channel-CvCrc8aa.mjs";
5
- import { createAssemblyScriptPool } from "./pool-runner-init-DjRCbiX-.mjs";
5
+ import { createAssemblyScriptPool } from "./pool-runner-init-CSJt3dHv.mjs";
6
6
 
7
7
  export { createAssemblyScriptPool };
@@ -1,5 +1,5 @@
1
1
  import { COVERAGE_PAYLOAD_FORMATS, POOL_ERROR_NAMES, TEST_ERROR_NAMES } from "./constants-Bq5KNxXJ.mjs";
2
- import { createAfterSuiteRunMeta, createSuiteTask, createTestTask, failTestAssertionError, failTestRuntimeError, getTaskLogLabel, isSuiteOwnFile } from "./vitest-file-tasks-vvZzigcF.mjs";
2
+ import { createAfterSuiteRunMeta, createSuiteTask, createTestTask, failTestAssertionError, failTestRuntimeError, getTaskLogLabel, isSuiteOwnFile } from "./vitest-file-tasks-BBsZ_wS6.mjs";
3
3
  import { abortWASMExecution, abortWASMExecutionOnSuccess, createPoolError, debug, debugOverride, enhanceTestError, extractCallStack, getExpectedMessageOrAny, wrapPoolError } from "./pool-errors-Bn6YaguR.mjs";
4
4
  import { createMemory, decodeAbortInfo, liftString } from "./wasm-memory-C8Nkl2Sz.mjs";
5
5
  import { basename, resolve } from "node:path";
@@ -566,4 +566,4 @@ async function loadUserWasmImportsFactory(relativePath, projectRoot, logModule)
566
566
 
567
567
  //#endregion
568
568
  export { createRpcClient, executeWASMDiscovery, executeWASMTest, flushRpcUpdates, loadUserWasmImportsFactory, reportFileCollected, reportFileError, reportFileQueued, reportSuiteFinished, reportSuitePrepare, reportTestFinished, reportTestPrepare, reportTestRetried, reportUserConsoleLogs };
569
- //# sourceMappingURL=load-user-imports-Bcx9NOt9.mjs.map
569
+ //# sourceMappingURL=load-user-imports-iAQUMKAi.mjs.map