@git.zone/tstest 2.3.8 → 2.4.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.
@@ -0,0 +1,205 @@
1
+ import * as plugins from './tstest.plugins.js';
2
+ import { coloredString as cs } from '@push.rocks/consolecolor';
3
+ import { RuntimeAdapter, } from './tstest.classes.runtime.adapter.js';
4
+ import { TapParser } from './tstest.classes.tap.parser.js';
5
+ import { TsTestLogger } from './tstest.logging.js';
6
+ /**
7
+ * Deno runtime adapter
8
+ * Executes tests using the Deno runtime
9
+ */
10
+ export class DenoRuntimeAdapter extends RuntimeAdapter {
11
+ constructor(logger, smartshellInstance, // SmartShell instance from @push.rocks/smartshell
12
+ timeoutSeconds, filterTags) {
13
+ super();
14
+ this.logger = logger;
15
+ this.smartshellInstance = smartshellInstance;
16
+ this.timeoutSeconds = timeoutSeconds;
17
+ this.filterTags = filterTags;
18
+ this.id = 'deno';
19
+ this.displayName = 'Deno';
20
+ }
21
+ /**
22
+ * Get default Deno options
23
+ */
24
+ getDefaultOptions() {
25
+ return {
26
+ ...super.getDefaultOptions(),
27
+ permissions: ['--allow-read', '--allow-env'],
28
+ };
29
+ }
30
+ /**
31
+ * Check if Deno is available
32
+ */
33
+ async checkAvailable() {
34
+ try {
35
+ const result = await this.smartshellInstance.exec('deno --version', {
36
+ cwd: process.cwd(),
37
+ onError: () => {
38
+ // Ignore error
39
+ }
40
+ });
41
+ if (result.exitCode !== 0) {
42
+ return {
43
+ available: false,
44
+ error: 'Deno not found. Install from: https://deno.land/',
45
+ };
46
+ }
47
+ // Parse Deno version from output (first line is "deno X.Y.Z")
48
+ const versionMatch = result.stdout.match(/deno (\d+\.\d+\.\d+)/);
49
+ const version = versionMatch ? versionMatch[1] : 'unknown';
50
+ return {
51
+ available: true,
52
+ version: `Deno ${version}`,
53
+ };
54
+ }
55
+ catch (error) {
56
+ return {
57
+ available: false,
58
+ error: error.message,
59
+ };
60
+ }
61
+ }
62
+ /**
63
+ * Create command configuration for Deno test execution
64
+ */
65
+ createCommand(testFile, options) {
66
+ const mergedOptions = this.mergeOptions(options);
67
+ const args = ['run'];
68
+ // Add permissions
69
+ const permissions = mergedOptions.permissions || ['--allow-read', '--allow-env'];
70
+ args.push(...permissions);
71
+ // Add config file if specified
72
+ if (mergedOptions.configPath) {
73
+ args.push('--config', mergedOptions.configPath);
74
+ }
75
+ // Add import map if specified
76
+ if (mergedOptions.importMap) {
77
+ args.push('--import-map', mergedOptions.importMap);
78
+ }
79
+ // Add extra args
80
+ if (mergedOptions.extraArgs && mergedOptions.extraArgs.length > 0) {
81
+ args.push(...mergedOptions.extraArgs);
82
+ }
83
+ // Add test file
84
+ args.push(testFile);
85
+ // Set environment variables
86
+ const env = { ...mergedOptions.env };
87
+ if (this.filterTags.length > 0) {
88
+ env.TSTEST_FILTER_TAGS = this.filterTags.join(',');
89
+ }
90
+ return {
91
+ command: 'deno',
92
+ args,
93
+ env,
94
+ cwd: mergedOptions.cwd,
95
+ };
96
+ }
97
+ /**
98
+ * Execute a test file in Deno
99
+ */
100
+ async run(testFile, index, total, options) {
101
+ this.logger.testFileStart(testFile, this.displayName, index, total);
102
+ const tapParser = new TapParser(testFile + ':deno', this.logger);
103
+ const mergedOptions = this.mergeOptions(options);
104
+ // Build Deno command
105
+ const command = this.createCommand(testFile, mergedOptions);
106
+ const fullCommand = `${command.command} ${command.args.join(' ')}`;
107
+ // Set filter tags as environment variable
108
+ if (this.filterTags.length > 0) {
109
+ process.env.TSTEST_FILTER_TAGS = this.filterTags.join(',');
110
+ }
111
+ // Check for 00init.ts file in test directory
112
+ const testDir = plugins.path.dirname(testFile);
113
+ const initFile = plugins.path.join(testDir, '00init.ts');
114
+ const initFileExists = await plugins.smartfile.fs.fileExists(initFile);
115
+ let runCommand = fullCommand;
116
+ let loaderPath = null;
117
+ // If 00init.ts exists, create a loader file
118
+ if (initFileExists) {
119
+ const absoluteInitFile = plugins.path.resolve(initFile);
120
+ const absoluteTestFile = plugins.path.resolve(testFile);
121
+ const loaderContent = `
122
+ import '${absoluteInitFile.replace(/\\/g, '/')}';
123
+ import '${absoluteTestFile.replace(/\\/g, '/')}';
124
+ `;
125
+ loaderPath = plugins.path.join(testDir, `.loader_${plugins.path.basename(testFile)}`);
126
+ await plugins.smartfile.memory.toFs(loaderContent, loaderPath);
127
+ // Rebuild command with loader file
128
+ const loaderCommand = this.createCommand(loaderPath, mergedOptions);
129
+ runCommand = `${loaderCommand.command} ${loaderCommand.args.join(' ')}`;
130
+ }
131
+ const execResultStreaming = await this.smartshellInstance.execStreamingSilent(runCommand);
132
+ // If we created a loader file, clean it up after test execution
133
+ if (loaderPath) {
134
+ const cleanup = () => {
135
+ try {
136
+ if (plugins.smartfile.fs.fileExistsSync(loaderPath)) {
137
+ plugins.smartfile.fs.removeSync(loaderPath);
138
+ }
139
+ }
140
+ catch (e) {
141
+ // Ignore cleanup errors
142
+ }
143
+ };
144
+ execResultStreaming.childProcess.on('exit', cleanup);
145
+ execResultStreaming.childProcess.on('error', cleanup);
146
+ }
147
+ // Start warning timer if no timeout was specified
148
+ let warningTimer = null;
149
+ if (this.timeoutSeconds === null) {
150
+ warningTimer = setTimeout(() => {
151
+ console.error('');
152
+ console.error(cs('⚠️ WARNING: Test file is running for more than 1 minute', 'orange'));
153
+ console.error(cs(` File: ${testFile}`, 'orange'));
154
+ console.error(cs(' Consider using --timeout option to set a timeout for test files.', 'orange'));
155
+ console.error(cs(' Example: tstest test --timeout=300 (for 5 minutes)', 'orange'));
156
+ console.error('');
157
+ }, 60000); // 1 minute
158
+ }
159
+ // Handle timeout if specified
160
+ if (this.timeoutSeconds !== null) {
161
+ const timeoutMs = this.timeoutSeconds * 1000;
162
+ let timeoutId;
163
+ const timeoutPromise = new Promise((_resolve, reject) => {
164
+ timeoutId = setTimeout(async () => {
165
+ // Use smartshell's terminate() to kill entire process tree
166
+ await execResultStreaming.terminate();
167
+ reject(new Error(`Test file timed out after ${this.timeoutSeconds} seconds`));
168
+ }, timeoutMs);
169
+ });
170
+ try {
171
+ await Promise.race([
172
+ tapParser.handleTapProcess(execResultStreaming.childProcess),
173
+ timeoutPromise
174
+ ]);
175
+ // Clear timeout if test completed successfully
176
+ clearTimeout(timeoutId);
177
+ }
178
+ catch (error) {
179
+ // Clear warning timer if it was set
180
+ if (warningTimer) {
181
+ clearTimeout(warningTimer);
182
+ }
183
+ // Handle timeout error
184
+ tapParser.handleTimeout(this.timeoutSeconds);
185
+ // Ensure entire process tree is killed if still running
186
+ try {
187
+ await execResultStreaming.kill(); // This kills the entire process tree with SIGKILL
188
+ }
189
+ catch (killError) {
190
+ // Process tree might already be dead
191
+ }
192
+ await tapParser.evaluateFinalResult();
193
+ }
194
+ }
195
+ else {
196
+ await tapParser.handleTapProcess(execResultStreaming.childProcess);
197
+ }
198
+ // Clear warning timer if it was set
199
+ if (warningTimer) {
200
+ clearTimeout(warningTimer);
201
+ }
202
+ return tapParser;
203
+ }
204
+ }
205
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tstest.classes.runtime.deno.js","sourceRoot":"","sources":["../ts/tstest.classes.runtime.deno.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,30 @@
1
+ import { RuntimeAdapter, type RuntimeOptions, type RuntimeCommand, type RuntimeAvailability } from './tstest.classes.runtime.adapter.js';
2
+ import { TapParser } from './tstest.classes.tap.parser.js';
3
+ import { TsTestLogger } from './tstest.logging.js';
4
+ import type { Runtime } from './tstest.classes.runtime.parser.js';
5
+ /**
6
+ * Node.js runtime adapter
7
+ * Executes tests using tsrun (TypeScript runner for Node.js)
8
+ */
9
+ export declare class NodeRuntimeAdapter extends RuntimeAdapter {
10
+ private logger;
11
+ private smartshellInstance;
12
+ private timeoutSeconds;
13
+ private filterTags;
14
+ readonly id: Runtime;
15
+ readonly displayName: string;
16
+ constructor(logger: TsTestLogger, smartshellInstance: any, // SmartShell instance from @push.rocks/smartshell
17
+ timeoutSeconds: number | null, filterTags: string[]);
18
+ /**
19
+ * Check if Node.js and tsrun are available
20
+ */
21
+ checkAvailable(): Promise<RuntimeAvailability>;
22
+ /**
23
+ * Create command configuration for Node.js test execution
24
+ */
25
+ createCommand(testFile: string, options?: RuntimeOptions): RuntimeCommand;
26
+ /**
27
+ * Execute a test file in Node.js
28
+ */
29
+ run(testFile: string, index: number, total: number, options?: RuntimeOptions): Promise<TapParser>;
30
+ }
@@ -0,0 +1,188 @@
1
+ import * as plugins from './tstest.plugins.js';
2
+ import { coloredString as cs } from '@push.rocks/consolecolor';
3
+ import { RuntimeAdapter, } from './tstest.classes.runtime.adapter.js';
4
+ import { TapParser } from './tstest.classes.tap.parser.js';
5
+ import { TsTestLogger } from './tstest.logging.js';
6
+ /**
7
+ * Node.js runtime adapter
8
+ * Executes tests using tsrun (TypeScript runner for Node.js)
9
+ */
10
+ export class NodeRuntimeAdapter extends RuntimeAdapter {
11
+ constructor(logger, smartshellInstance, // SmartShell instance from @push.rocks/smartshell
12
+ timeoutSeconds, filterTags) {
13
+ super();
14
+ this.logger = logger;
15
+ this.smartshellInstance = smartshellInstance;
16
+ this.timeoutSeconds = timeoutSeconds;
17
+ this.filterTags = filterTags;
18
+ this.id = 'node';
19
+ this.displayName = 'Node.js';
20
+ }
21
+ /**
22
+ * Check if Node.js and tsrun are available
23
+ */
24
+ async checkAvailable() {
25
+ try {
26
+ // Check Node.js version
27
+ const nodeVersion = process.version;
28
+ // Check if tsrun is available
29
+ const result = await this.smartshellInstance.exec('tsrun --version', {
30
+ cwd: process.cwd(),
31
+ onError: () => {
32
+ // Ignore error
33
+ }
34
+ });
35
+ if (result.exitCode !== 0) {
36
+ return {
37
+ available: false,
38
+ error: 'tsrun not found. Install with: pnpm install --save-dev @git.zone/tsrun',
39
+ };
40
+ }
41
+ return {
42
+ available: true,
43
+ version: nodeVersion,
44
+ };
45
+ }
46
+ catch (error) {
47
+ return {
48
+ available: false,
49
+ error: error.message,
50
+ };
51
+ }
52
+ }
53
+ /**
54
+ * Create command configuration for Node.js test execution
55
+ */
56
+ createCommand(testFile, options) {
57
+ const mergedOptions = this.mergeOptions(options);
58
+ // Build tsrun options
59
+ const args = [];
60
+ if (process.argv.includes('--web')) {
61
+ args.push('--web');
62
+ }
63
+ // Add any extra args
64
+ if (mergedOptions.extraArgs) {
65
+ args.push(...mergedOptions.extraArgs);
66
+ }
67
+ // Set environment variables
68
+ const env = { ...mergedOptions.env };
69
+ if (this.filterTags.length > 0) {
70
+ env.TSTEST_FILTER_TAGS = this.filterTags.join(',');
71
+ }
72
+ return {
73
+ command: 'tsrun',
74
+ args: [testFile, ...args],
75
+ env,
76
+ cwd: mergedOptions.cwd,
77
+ };
78
+ }
79
+ /**
80
+ * Execute a test file in Node.js
81
+ */
82
+ async run(testFile, index, total, options) {
83
+ this.logger.testFileStart(testFile, this.displayName, index, total);
84
+ const tapParser = new TapParser(testFile + ':node', this.logger);
85
+ const mergedOptions = this.mergeOptions(options);
86
+ // Build tsrun command
87
+ let tsrunOptions = '';
88
+ if (process.argv.includes('--web')) {
89
+ tsrunOptions += ' --web';
90
+ }
91
+ // Set filter tags as environment variable
92
+ if (this.filterTags.length > 0) {
93
+ process.env.TSTEST_FILTER_TAGS = this.filterTags.join(',');
94
+ }
95
+ // Check for 00init.ts file in test directory
96
+ const testDir = plugins.path.dirname(testFile);
97
+ const initFile = plugins.path.join(testDir, '00init.ts');
98
+ let runCommand = `tsrun ${testFile}${tsrunOptions}`;
99
+ const initFileExists = await plugins.smartfile.fs.fileExists(initFile);
100
+ // If 00init.ts exists, run it first
101
+ let loaderPath = null;
102
+ if (initFileExists) {
103
+ // Create a temporary loader file that imports both 00init.ts and the test file
104
+ const absoluteInitFile = plugins.path.resolve(initFile);
105
+ const absoluteTestFile = plugins.path.resolve(testFile);
106
+ const loaderContent = `
107
+ import '${absoluteInitFile.replace(/\\/g, '/')}';
108
+ import '${absoluteTestFile.replace(/\\/g, '/')}';
109
+ `;
110
+ loaderPath = plugins.path.join(testDir, `.loader_${plugins.path.basename(testFile)}`);
111
+ await plugins.smartfile.memory.toFs(loaderContent, loaderPath);
112
+ runCommand = `tsrun ${loaderPath}${tsrunOptions}`;
113
+ }
114
+ const execResultStreaming = await this.smartshellInstance.execStreamingSilent(runCommand);
115
+ // If we created a loader file, clean it up after test execution
116
+ if (loaderPath) {
117
+ const cleanup = () => {
118
+ try {
119
+ if (plugins.smartfile.fs.fileExistsSync(loaderPath)) {
120
+ plugins.smartfile.fs.removeSync(loaderPath);
121
+ }
122
+ }
123
+ catch (e) {
124
+ // Ignore cleanup errors
125
+ }
126
+ };
127
+ execResultStreaming.childProcess.on('exit', cleanup);
128
+ execResultStreaming.childProcess.on('error', cleanup);
129
+ }
130
+ // Start warning timer if no timeout was specified
131
+ let warningTimer = null;
132
+ if (this.timeoutSeconds === null) {
133
+ warningTimer = setTimeout(() => {
134
+ console.error('');
135
+ console.error(cs('⚠️ WARNING: Test file is running for more than 1 minute', 'orange'));
136
+ console.error(cs(` File: ${testFile}`, 'orange'));
137
+ console.error(cs(' Consider using --timeout option to set a timeout for test files.', 'orange'));
138
+ console.error(cs(' Example: tstest test --timeout=300 (for 5 minutes)', 'orange'));
139
+ console.error('');
140
+ }, 60000); // 1 minute
141
+ }
142
+ // Handle timeout if specified
143
+ if (this.timeoutSeconds !== null) {
144
+ const timeoutMs = this.timeoutSeconds * 1000;
145
+ let timeoutId;
146
+ const timeoutPromise = new Promise((_resolve, reject) => {
147
+ timeoutId = setTimeout(async () => {
148
+ // Use smartshell's terminate() to kill entire process tree
149
+ await execResultStreaming.terminate();
150
+ reject(new Error(`Test file timed out after ${this.timeoutSeconds} seconds`));
151
+ }, timeoutMs);
152
+ });
153
+ try {
154
+ await Promise.race([
155
+ tapParser.handleTapProcess(execResultStreaming.childProcess),
156
+ timeoutPromise
157
+ ]);
158
+ // Clear timeout if test completed successfully
159
+ clearTimeout(timeoutId);
160
+ }
161
+ catch (error) {
162
+ // Clear warning timer if it was set
163
+ if (warningTimer) {
164
+ clearTimeout(warningTimer);
165
+ }
166
+ // Handle timeout error
167
+ tapParser.handleTimeout(this.timeoutSeconds);
168
+ // Ensure entire process tree is killed if still running
169
+ try {
170
+ await execResultStreaming.kill(); // This kills the entire process tree with SIGKILL
171
+ }
172
+ catch (killError) {
173
+ // Process tree might already be dead
174
+ }
175
+ await tapParser.evaluateFinalResult();
176
+ }
177
+ }
178
+ else {
179
+ await tapParser.handleTapProcess(execResultStreaming.childProcess);
180
+ }
181
+ // Clear warning timer if it was set
182
+ if (warningTimer) {
183
+ clearTimeout(warningTimer);
184
+ }
185
+ return tapParser;
186
+ }
187
+ }
188
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHN0ZXN0LmNsYXNzZXMucnVudGltZS5ub2RlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vdHMvdHN0ZXN0LmNsYXNzZXMucnVudGltZS5ub2RlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sS0FBSyxPQUFPLE1BQU0scUJBQXFCLENBQUM7QUFDL0MsT0FBTyxFQUFFLGFBQWEsSUFBSSxFQUFFLEVBQUUsTUFBTSwwQkFBMEIsQ0FBQztBQUMvRCxPQUFPLEVBQ0wsY0FBYyxHQUlmLE1BQU0scUNBQXFDLENBQUM7QUFDN0MsT0FBTyxFQUFFLFNBQVMsRUFBRSxNQUFNLGdDQUFnQyxDQUFDO0FBQzNELE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSxxQkFBcUIsQ0FBQztBQUduRDs7O0dBR0c7QUFDSCxNQUFNLE9BQU8sa0JBQW1CLFNBQVEsY0FBYztJQUlwRCxZQUNVLE1BQW9CLEVBQ3BCLGtCQUF1QixFQUFFLGtEQUFrRDtJQUMzRSxjQUE2QixFQUM3QixVQUFvQjtRQUU1QixLQUFLLEVBQUUsQ0FBQztRQUxBLFdBQU0sR0FBTixNQUFNLENBQWM7UUFDcEIsdUJBQWtCLEdBQWxCLGtCQUFrQixDQUFLO1FBQ3ZCLG1CQUFjLEdBQWQsY0FBYyxDQUFlO1FBQzdCLGVBQVUsR0FBVixVQUFVLENBQVU7UUFQckIsT0FBRSxHQUFZLE1BQU0sQ0FBQztRQUNyQixnQkFBVyxHQUFXLFNBQVMsQ0FBQztJQVN6QyxDQUFDO0lBRUQ7O09BRUc7SUFDSCxLQUFLLENBQUMsY0FBYztRQUNsQixJQUFJLENBQUM7WUFDSCx3QkFBd0I7WUFDeEIsTUFBTSxXQUFXLEdBQUcsT0FBTyxDQUFDLE9BQU8sQ0FBQztZQUVwQyw4QkFBOEI7WUFDOUIsTUFBTSxNQUFNLEdBQUcsTUFBTSxJQUFJLENBQUMsa0JBQWtCLENBQUMsSUFBSSxDQUFDLGlCQUFpQixFQUFFO2dCQUNuRSxHQUFHLEVBQUUsT0FBTyxDQUFDLEdBQUcsRUFBRTtnQkFDbEIsT0FBTyxF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
@@ -0,0 +1,37 @@
1
+ /**
2
+ * Runtime parser for test file naming convention
3
+ * Supports: test.runtime1+runtime2.modifier.ts
4
+ * Examples:
5
+ * - test.node.ts
6
+ * - test.chromium.ts
7
+ * - test.node+chromium.ts
8
+ * - test.deno+bun.ts
9
+ * - test.chromium.nonci.ts
10
+ */
11
+ export type Runtime = 'node' | 'chromium' | 'deno' | 'bun';
12
+ export type Modifier = 'nonci';
13
+ export interface ParsedFilename {
14
+ baseName: string;
15
+ runtimes: Runtime[];
16
+ modifiers: Modifier[];
17
+ extension: string;
18
+ isLegacy: boolean;
19
+ original: string;
20
+ }
21
+ export interface ParserConfig {
22
+ strictUnknownRuntime?: boolean;
23
+ defaultRuntimes?: Runtime[];
24
+ }
25
+ /**
26
+ * Parse a test filename to extract runtimes, modifiers, and detect legacy patterns
27
+ * Algorithm: Right-to-left token analysis from the extension
28
+ */
29
+ export declare function parseTestFilename(filePath: string, config?: ParserConfig): ParsedFilename;
30
+ /**
31
+ * Check if a filename uses legacy naming convention
32
+ */
33
+ export declare function isLegacyFilename(fileName: string): boolean;
34
+ /**
35
+ * Get the suggested new filename for a legacy filename
36
+ */
37
+ export declare function getLegacyMigrationTarget(fileName: string): string | null;
@@ -0,0 +1,159 @@
1
+ /**
2
+ * Runtime parser for test file naming convention
3
+ * Supports: test.runtime1+runtime2.modifier.ts
4
+ * Examples:
5
+ * - test.node.ts
6
+ * - test.chromium.ts
7
+ * - test.node+chromium.ts
8
+ * - test.deno+bun.ts
9
+ * - test.chromium.nonci.ts
10
+ */
11
+ const KNOWN_RUNTIMES = new Set(['node', 'chromium', 'deno', 'bun']);
12
+ const KNOWN_MODIFIERS = new Set(['nonci']);
13
+ const VALID_EXTENSIONS = new Set(['ts', 'tsx', 'mts', 'cts']);
14
+ // Legacy mappings for backwards compatibility
15
+ const LEGACY_RUNTIME_MAP = {
16
+ browser: ['chromium'],
17
+ both: ['node', 'chromium'],
18
+ };
19
+ /**
20
+ * Parse a test filename to extract runtimes, modifiers, and detect legacy patterns
21
+ * Algorithm: Right-to-left token analysis from the extension
22
+ */
23
+ export function parseTestFilename(filePath, config = {}) {
24
+ const strictUnknownRuntime = config.strictUnknownRuntime ?? true;
25
+ const defaultRuntimes = config.defaultRuntimes ?? ['node'];
26
+ // Extract just the filename from the path
27
+ const fileName = filePath.split('/').pop() || filePath;
28
+ const original = fileName;
29
+ // Step 1: Extract and validate extension
30
+ const lastDot = fileName.lastIndexOf('.');
31
+ if (lastDot === -1) {
32
+ throw new Error(`Invalid test file: no extension found in "${fileName}"`);
33
+ }
34
+ const extension = fileName.substring(lastDot + 1);
35
+ if (!VALID_EXTENSIONS.has(extension)) {
36
+ throw new Error(`Invalid test file extension ".${extension}" in "${fileName}". ` +
37
+ `Valid extensions: ${Array.from(VALID_EXTENSIONS).join(', ')}`);
38
+ }
39
+ // Step 2: Split remaining basename by dots
40
+ const withoutExtension = fileName.substring(0, lastDot);
41
+ const tokens = withoutExtension.split('.');
42
+ if (tokens.length === 0) {
43
+ throw new Error(`Invalid test file: empty basename in "${fileName}"`);
44
+ }
45
+ // Step 3: Parse from right to left
46
+ let isLegacy = false;
47
+ const modifiers = [];
48
+ let runtimes = [];
49
+ let runtimeTokenIndex = -1;
50
+ // Scan from right to left
51
+ for (let i = tokens.length - 1; i >= 0; i--) {
52
+ const token = tokens[i];
53
+ // Check if this is a known modifier
54
+ if (KNOWN_MODIFIERS.has(token)) {
55
+ modifiers.unshift(token);
56
+ continue;
57
+ }
58
+ // Check if this is a legacy runtime token
59
+ if (LEGACY_RUNTIME_MAP[token]) {
60
+ isLegacy = true;
61
+ runtimes = LEGACY_RUNTIME_MAP[token];
62
+ runtimeTokenIndex = i;
63
+ break;
64
+ }
65
+ // Check if this is a runtime chain (may contain + separators)
66
+ if (token.includes('+')) {
67
+ const runtimeCandidates = token.split('+').map(r => r.trim()).filter(Boolean);
68
+ const validRuntimes = [];
69
+ const invalidRuntimes = [];
70
+ for (const candidate of runtimeCandidates) {
71
+ if (KNOWN_RUNTIMES.has(candidate)) {
72
+ // Dedupe: only add if not already in list
73
+ if (!validRuntimes.includes(candidate)) {
74
+ validRuntimes.push(candidate);
75
+ }
76
+ }
77
+ else {
78
+ invalidRuntimes.push(candidate);
79
+ }
80
+ }
81
+ if (invalidRuntimes.length > 0) {
82
+ if (strictUnknownRuntime) {
83
+ throw new Error(`Unknown runtime(s) in "${fileName}": ${invalidRuntimes.join(', ')}. ` +
84
+ `Valid runtimes: ${Array.from(KNOWN_RUNTIMES).join(', ')}`);
85
+ }
86
+ else {
87
+ console.warn(`⚠️ Warning: Unknown runtime(s) in "${fileName}": ${invalidRuntimes.join(', ')}. ` +
88
+ `Defaulting to: ${defaultRuntimes.join('+')}`);
89
+ runtimes = [...defaultRuntimes];
90
+ runtimeTokenIndex = i;
91
+ break;
92
+ }
93
+ }
94
+ if (validRuntimes.length > 0) {
95
+ runtimes = validRuntimes;
96
+ runtimeTokenIndex = i;
97
+ break;
98
+ }
99
+ }
100
+ // Check if this is a single runtime token
101
+ if (KNOWN_RUNTIMES.has(token)) {
102
+ runtimes = [token];
103
+ runtimeTokenIndex = i;
104
+ break;
105
+ }
106
+ // If we've scanned past modifiers and haven't found a runtime, stop looking
107
+ if (modifiers.length > 0) {
108
+ break;
109
+ }
110
+ }
111
+ // Step 4: Determine base name
112
+ // Everything before the runtime token (if found) is the base name
113
+ const baseNameTokens = runtimeTokenIndex >= 0 ? tokens.slice(0, runtimeTokenIndex) : tokens;
114
+ const baseName = baseNameTokens.join('.');
115
+ // Step 5: Apply defaults if no runtime was detected
116
+ if (runtimes.length === 0) {
117
+ runtimes = [...defaultRuntimes];
118
+ }
119
+ return {
120
+ baseName: baseName || 'test',
121
+ runtimes,
122
+ modifiers,
123
+ extension,
124
+ isLegacy,
125
+ original,
126
+ };
127
+ }
128
+ /**
129
+ * Check if a filename uses legacy naming convention
130
+ */
131
+ export function isLegacyFilename(fileName) {
132
+ const tokens = fileName.split('.');
133
+ for (const token of tokens) {
134
+ if (LEGACY_RUNTIME_MAP[token]) {
135
+ return true;
136
+ }
137
+ }
138
+ return false;
139
+ }
140
+ /**
141
+ * Get the suggested new filename for a legacy filename
142
+ */
143
+ export function getLegacyMigrationTarget(fileName) {
144
+ const parsed = parseTestFilename(fileName, { strictUnknownRuntime: false });
145
+ if (!parsed.isLegacy) {
146
+ return null;
147
+ }
148
+ // Reconstruct filename with new naming
149
+ const parts = [parsed.baseName];
150
+ if (parsed.runtimes.length > 0) {
151
+ parts.push(parsed.runtimes.join('+'));
152
+ }
153
+ if (parsed.modifiers.length > 0) {
154
+ parts.push(...parsed.modifiers);
155
+ }
156
+ parts.push(parsed.extension);
157
+ return parts.join('.');
158
+ }
159
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHN0ZXN0LmNsYXNzZXMucnVudGltZS5wYXJzZXIuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi90cy90c3Rlc3QuY2xhc3Nlcy5ydW50aW1lLnBhcnNlci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7Ozs7Ozs7O0dBU0c7QUFtQkgsTUFBTSxjQUFjLEdBQWdCLElBQUksR0FBRyxDQUFDLENBQUMsTUFBTSxFQUFFLFVBQVUsRUFBRSxNQUFNLEVBQUUsS0FBSyxDQUFDLENBQUMsQ0FBQztBQUNqRixNQUFNLGVBQWUsR0FBZ0IsSUFBSSxHQUFHLENBQUMsQ0FBQyxPQUFPLENBQUMsQ0FBQyxDQUFDO0FBQ3hELE1BQU0sZ0JBQWdCLEdBQWdCLElBQUksR0FBRyxDQUFDLENBQUMsSUFBSSxFQUFFLEtBQUssRUFBRSxLQUFLLEVBQUUsS0FBSyxDQUFDLENBQUMsQ0FBQztBQUUzRSw4Q0FBOEM7QUFDOUMsTUFBTSxrQkFBa0IsR0FBOEI7SUFDcEQsT0FBTyxFQUFFLENBQUMsVUFBVSxDQUFDO0lBQ3JCLElBQUksRUFBRSxDQUFDLE1BQU0sRUFBRSxVQUFVLENBQUM7Q0FDM0IsQ0FBQztBQUVGOzs7R0FHRztBQUNILE1BQU0sVUFBVSxpQkFBaUIsQ0FDL0IsUUFBZ0IsRUFDaEIsU0FBdUIsRUFBRTtJQUV6QixNQUFNLG9CQUFvQixHQUFHLE1BQU0sQ0FBQyxvQkFBb0IsSUFBSSxJQUFJLENBQUM7SUFDakUsTUFBTSxlQUFlLEdBQUcsTUFBTSxDQUFDLGVBQWUsSUFBSSxDQUFDLE1BQU0sQ0FBQyxDQUFDO0lBRTNELDBDQUEwQztJQUMxQyxNQUFNLFFBQVEsR0FBRyxRQUFRLENBQUMsS0FBSyxDQUFDLEdBQUcsQ0FBQyxDQUFDLEdBQUcsRUFBRSxJQUFJLFFBQVEsQ0FBQztJQUN2RCxNQUFNLFFBQVEsR0FBRyxRQUFRLENBQUM7SUFFMUIseUNBQXlDO0lBQ3pDLE1BQU0sT0FBTyxHQUFHLFFBQVEsQ0FBQyxXQUFXLENBQUMsR0FBRyxDQUFD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