mirror of
				https://github.com/microsoft/playwright.git
				synced 2025-06-26 21:40:17 +00:00 
			
		
		
		
	This is a worker number between `0` and `workers - 1` that does not change after worker process restart.
		
			
				
	
	
		
			222 lines
		
	
	
		
			7.1 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			222 lines
		
	
	
		
			7.1 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
/**
 | 
						|
 * Copyright (c) Microsoft Corporation.
 | 
						|
 *
 | 
						|
 * Licensed under the Apache License, Version 2.0 (the "License");
 | 
						|
 * you may not use this file except in compliance with the License.
 | 
						|
 * You may obtain a copy of the License at
 | 
						|
 *
 | 
						|
 * http://www.apache.org/licenses/LICENSE-2.0
 | 
						|
 *
 | 
						|
 * Unless required by applicable law or agreed to in writing, software
 | 
						|
 * distributed under the License is distributed on an "AS IS" BASIS,
 | 
						|
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | 
						|
 * See the License for the specific language governing permissions and
 | 
						|
 * limitations under the License.
 | 
						|
 */
 | 
						|
 | 
						|
import { test, expect } from './playwright-test-fixtures';
 | 
						|
 | 
						|
test('should run in parallel', async ({ runInlineTest }) => {
 | 
						|
  const result = await runInlineTest({
 | 
						|
    '1.spec.ts': `
 | 
						|
      import * as fs from 'fs';
 | 
						|
      import * as path from 'path';
 | 
						|
      const { test } = pwt;
 | 
						|
      test('succeeds', async ({}, testInfo) => {
 | 
						|
        expect(testInfo.workerIndex).toBe(0);
 | 
						|
        expect(testInfo.parallelIndex).toBe(0);
 | 
						|
        // First test waits for the second to start to work around the race.
 | 
						|
        while (true) {
 | 
						|
          if (fs.existsSync(path.join(testInfo.project.outputDir, 'parallel-index.txt')))
 | 
						|
            break;
 | 
						|
          await new Promise(f => setTimeout(f, 100));
 | 
						|
        }
 | 
						|
      });
 | 
						|
    `,
 | 
						|
    '2.spec.ts': `
 | 
						|
      import * as fs from 'fs';
 | 
						|
      import * as path from 'path';
 | 
						|
      const { test } = pwt;
 | 
						|
      test('succeeds', async ({}, testInfo) => {
 | 
						|
        // First test waits for the second to start to work around the race.
 | 
						|
        fs.mkdirSync(testInfo.project.outputDir, { recursive: true });
 | 
						|
        fs.writeFileSync(path.join(testInfo.project.outputDir, 'parallel-index.txt'), 'TRUE');
 | 
						|
        expect(testInfo.workerIndex).toBe(1);
 | 
						|
        expect(testInfo.parallelIndex).toBe(1);
 | 
						|
      });
 | 
						|
    `,
 | 
						|
  });
 | 
						|
  expect(result.passed).toBe(2);
 | 
						|
  expect(result.exitCode).toBe(0);
 | 
						|
});
 | 
						|
 | 
						|
test('should reuse worker for multiple tests', async ({ runInlineTest }) => {
 | 
						|
  const result = await runInlineTest({
 | 
						|
    'a.test.js': `
 | 
						|
      const { test } = pwt;
 | 
						|
      test('succeeds 1', async ({}, testInfo) => {
 | 
						|
        expect(testInfo.workerIndex).toBe(0);
 | 
						|
        expect(testInfo.parallelIndex).toBe(0);
 | 
						|
      });
 | 
						|
 | 
						|
      test('succeeds 2', async ({}, testInfo) => {
 | 
						|
        expect(testInfo.workerIndex).toBe(0);
 | 
						|
        expect(testInfo.parallelIndex).toBe(0);
 | 
						|
      });
 | 
						|
 | 
						|
      test('succeeds 3', async ({}, testInfo) => {
 | 
						|
        expect(testInfo.workerIndex).toBe(0);
 | 
						|
        expect(testInfo.parallelIndex).toBe(0);
 | 
						|
      });
 | 
						|
    `,
 | 
						|
  });
 | 
						|
  expect(result.passed).toBe(3);
 | 
						|
  expect(result.exitCode).toBe(0);
 | 
						|
});
 | 
						|
 | 
						|
test('should reuse worker after test.fixme()', async ({ runInlineTest }) => {
 | 
						|
  const result = await runInlineTest({
 | 
						|
    'a.test.js': `
 | 
						|
      const { test } = pwt;
 | 
						|
      test('succeeds 1', async ({}, testInfo) => {
 | 
						|
        expect(testInfo.workerIndex).toBe(0);
 | 
						|
        expect(testInfo.parallelIndex).toBe(0);
 | 
						|
      });
 | 
						|
 | 
						|
      test('fixme 1', async ({}, testInfo) => {
 | 
						|
        test.fixme();
 | 
						|
        expect(testInfo.workerIndex).toBe(0);
 | 
						|
        expect(testInfo.parallelIndex).toBe(0);
 | 
						|
      });
 | 
						|
 | 
						|
      test('succeeds 2', async ({}, testInfo) => {
 | 
						|
        expect(testInfo.workerIndex).toBe(0);
 | 
						|
        expect(testInfo.parallelIndex).toBe(0);
 | 
						|
      });
 | 
						|
    `,
 | 
						|
  });
 | 
						|
  expect(result.passed).toBe(2);
 | 
						|
  expect(result.skipped).toBe(1);
 | 
						|
  expect(result.exitCode).toBe(0);
 | 
						|
});
 | 
						|
 | 
						|
test('should reuse worker after test.skip()', async ({ runInlineTest }) => {
 | 
						|
  const result = await runInlineTest({
 | 
						|
    'a.test.js': `
 | 
						|
      const { test } = pwt;
 | 
						|
      test('succeeds 1', async ({}, testInfo) => {
 | 
						|
        expect(testInfo.workerIndex).toBe(0);
 | 
						|
        expect(testInfo.parallelIndex).toBe(0);
 | 
						|
      });
 | 
						|
 | 
						|
      test('skip 1', async ({}, testInfo) => {
 | 
						|
        test.skip();
 | 
						|
        expect(testInfo.workerIndex).toBe(0);
 | 
						|
        expect(testInfo.parallelIndex).toBe(0);
 | 
						|
      });
 | 
						|
 | 
						|
      test('succeeds 2', async ({}, testInfo) => {
 | 
						|
        expect(testInfo.workerIndex).toBe(0);
 | 
						|
        expect(testInfo.parallelIndex).toBe(0);
 | 
						|
      });
 | 
						|
    `,
 | 
						|
  });
 | 
						|
  expect(result.passed).toBe(2);
 | 
						|
  expect(result.skipped).toBe(1);
 | 
						|
  expect(result.exitCode).toBe(0);
 | 
						|
});
 | 
						|
 | 
						|
test('should not use new worker after test.fail()', async ({ runInlineTest }) => {
 | 
						|
  const result = await runInlineTest({
 | 
						|
    'a.test.js': `
 | 
						|
      const { test } = pwt;
 | 
						|
      test('succeeds 1', async ({}, testInfo) => {
 | 
						|
        expect(testInfo.workerIndex).toBe(0);
 | 
						|
        expect(testInfo.parallelIndex).toBe(0);
 | 
						|
      });
 | 
						|
 | 
						|
      test('fail 1', async ({}, testInfo) => {
 | 
						|
        test.fail();
 | 
						|
        expect(1).toBe(0);
 | 
						|
      });
 | 
						|
 | 
						|
      test('succeeds 2', async ({}, testInfo) => {
 | 
						|
        expect(testInfo.workerIndex).toBe(0);
 | 
						|
        expect(testInfo.parallelIndex).toBe(0);
 | 
						|
      });
 | 
						|
    `,
 | 
						|
  });
 | 
						|
  expect(result.passed).toBe(3);
 | 
						|
  expect(result.failed).toBe(0);
 | 
						|
  expect(result.exitCode).toBe(0);
 | 
						|
});
 | 
						|
 | 
						|
test('should use new worker after test failure', async ({ runInlineTest }) => {
 | 
						|
  const result = await runInlineTest({
 | 
						|
    'a.test.js': `
 | 
						|
      const { test } = pwt;
 | 
						|
      test('succeeds 1', async ({}, testInfo) => {
 | 
						|
        expect(testInfo.workerIndex).toBe(0);
 | 
						|
        expect(testInfo.parallelIndex).toBe(0);
 | 
						|
      });
 | 
						|
 | 
						|
      test('fail 1', async ({}, testInfo) => {
 | 
						|
        expect(1).toBe(0);
 | 
						|
      });
 | 
						|
 | 
						|
      test('succeeds 2', async ({}, testInfo) => {
 | 
						|
        expect(testInfo.workerIndex).toBe(1);
 | 
						|
        expect(testInfo.parallelIndex).toBe(0);
 | 
						|
      });
 | 
						|
    `,
 | 
						|
  }, { workers: 1 });
 | 
						|
  expect(result.passed).toBe(2);
 | 
						|
  expect(result.failed).toBe(1);
 | 
						|
  expect(result.exitCode).toBe(1);
 | 
						|
});
 | 
						|
 | 
						|
test('should not reuse worker for different suites', async ({ runInlineTest }) => {
 | 
						|
  const result = await runInlineTest({
 | 
						|
    'playwright.config.ts': `
 | 
						|
      module.exports = { projects: [{}, {}, {}] };
 | 
						|
    `,
 | 
						|
    'a.test.js': `
 | 
						|
      const { test } = pwt;
 | 
						|
      test('succeeds', async ({}, testInfo) => {
 | 
						|
        console.log('workerIndex-' + testInfo.workerIndex);
 | 
						|
        console.log('parallelIndex-' + testInfo.parallelIndex);
 | 
						|
      });
 | 
						|
    `,
 | 
						|
  }, { workers: 1 });
 | 
						|
  expect(result.passed).toBe(3);
 | 
						|
  expect(result.exitCode).toBe(0);
 | 
						|
  expect(result.results.map(r => r.workerIndex).sort()).toEqual([0, 1, 2]);
 | 
						|
  expect(result.output).toContain('workerIndex-0');
 | 
						|
  expect(result.output).toContain('workerIndex-1');
 | 
						|
  expect(result.output).toContain('workerIndex-2');
 | 
						|
  expect(result.output).toContain('parallelIndex-0');
 | 
						|
  expect(result.output).not.toContain('parallelIndex-1');
 | 
						|
});
 | 
						|
 | 
						|
test('parallelIndex should be in 0..workers-1', async ({ runInlineTest }) => {
 | 
						|
  const files = {};
 | 
						|
  for (let i = 0; i < 10; i++) {
 | 
						|
    files[`a${i}.test.js`] = `
 | 
						|
      const { test } = pwt;
 | 
						|
      test('passes-1', async ({}, testInfo) => {
 | 
						|
        await new Promise(f => setTimeout(f, 100 + 50 * ${i}));
 | 
						|
        expect(testInfo.parallelIndex >= 0).toBeTruthy();
 | 
						|
        expect(testInfo.parallelIndex < testInfo.config.workers).toBeTruthy();
 | 
						|
      });
 | 
						|
      test('passes-2', async ({}, testInfo) => {
 | 
						|
        await new Promise(f => setTimeout(f, 100 + 50 * ${i}));
 | 
						|
        expect(testInfo.parallelIndex >= 0).toBeTruthy();
 | 
						|
        expect(testInfo.parallelIndex < testInfo.config.workers).toBeTruthy();
 | 
						|
      });
 | 
						|
    `;
 | 
						|
  }
 | 
						|
  const result = await runInlineTest(files, { workers: 3 });
 | 
						|
  expect(result.exitCode).toBe(0);
 | 
						|
  expect(result.passed).toBe(20);
 | 
						|
});
 |