101 lines
		
	
	
		
			4.4 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
		
		
			
		
	
	
			101 lines
		
	
	
		
			4.4 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| 
								 | 
							
								/**
							 | 
						||
| 
								 | 
							
								 * @license
							 | 
						||
| 
								 | 
							
								 * Copyright Google Inc. All Rights Reserved.
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * Use of this source code is governed by an MIT-style license that can be
							 | 
						||
| 
								 | 
							
								 * found in the LICENSE file at https://angular.io/license
							 | 
						||
| 
								 | 
							
								 */
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/*
							 | 
						||
| 
								 | 
							
								 * Script that determines the sharded tests for the current CircleCI container. CircleCI starts
							 | 
						||
| 
								 | 
							
								 * multiple containers if the "parallelism" option has been specified and this script splits up
							 | 
						||
| 
								 | 
							
								 * the integration tests into shards based on the amount of parallelism.
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * It's also possible to manually specify tests which should run on a container because some
							 | 
						||
| 
								 | 
							
								 * integration tests are more complex and take up more time. In order to properly balance the
							 | 
						||
| 
								 | 
							
								 * duration of each container, we allow manual test shards to be specified.
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * The output of this script can then be used to only run the tests which are assigned to the
							 | 
						||
| 
								 | 
							
								 * current CircleCI container.
							 | 
						||
| 
								 | 
							
								 */
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								const fs = require('fs');
							 | 
						||
| 
								 | 
							
								const path = require('path');
							 | 
						||
| 
								 | 
							
								const minimist = require('minimist');
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								// Parsed command line arguments.
							 | 
						||
| 
								 | 
							
								const {shardIndex, maxShards} = minimist(process.argv.slice(2));
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								// Ensure that all CLI options are set properly.
							 | 
						||
| 
								 | 
							
								if (shardIndex == null) {
							 | 
						||
| 
								 | 
							
								  throw new Error('The "--shardIndex" option has not been specified.')
							 | 
						||
| 
								 | 
							
								} else if (maxShards == null) {
							 | 
						||
| 
								 | 
							
								  throw new Error('The "--maxShards" option has not been specified.');
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								// List of all integration tests that are available.
							 | 
						||
| 
								 | 
							
								const integrationTests = fs.readdirSync(__dirname).filter(
							 | 
						||
| 
								 | 
							
								    testName => fs.statSync(path.join(__dirname, testName)).isDirectory());
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								// Manual test shards which aren't computed automatically. This is helpful when a specific
							 | 
						||
| 
								 | 
							
								// set of integration test takes up *way* more time than all other tests, and we want to
							 | 
						||
| 
								 | 
							
								// balance out the duration for all specific shards.
							 | 
						||
| 
								 | 
							
								const manualTestShards = [
							 | 
						||
| 
								 | 
							
								  // The first shard should only run the bazel integration tests because these take up
							 | 
						||
| 
								 | 
							
								  // a lot of time and shouldn't be split up automatically.
							 | 
						||
| 
								 | 
							
								  ['bazel', 'bazel-schematics']
							 | 
						||
| 
								 | 
							
								];
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								// Tests which haven't been assigned manually to a shard. These tests will be automatically
							 | 
						||
| 
								 | 
							
								// split across the remaining available shards.
							 | 
						||
| 
								 | 
							
								const unassignedTests = stripManualOverrides(integrationTests, manualTestShards);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								if (manualTestShards.length === maxShards && unassignedTests.length) {
							 | 
						||
| 
								 | 
							
								  throw new Error(
							 | 
						||
| 
								 | 
							
								      `Tests have been specified manually for all available shards, but there were ` +
							 | 
						||
| 
								 | 
							
								      `integration tests which haven't been specified and won't run right now. Missing ` +
							 | 
						||
| 
								 | 
							
								      `tests: ${unassignedTests.join(', ')}`)
							 | 
						||
| 
								 | 
							
								} else if (manualTestShards.length > maxShards) {
							 | 
						||
| 
								 | 
							
								  throw new Error(
							 | 
						||
| 
								 | 
							
								      `Too many manual shards have been specified. Increase the amount of maximum shards.`);
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								// In case the shard for the current index has been specified manually, we just output
							 | 
						||
| 
								 | 
							
								// the tests for the manual shard.
							 | 
						||
| 
								 | 
							
								if (manualTestShards[shardIndex]) {
							 | 
						||
| 
								 | 
							
								  printTestNames(manualTestShards[shardIndex]);
							 | 
						||
| 
								 | 
							
								} else {
							 | 
						||
| 
								 | 
							
								  const amountManualShards = manualTestShards.length;
							 | 
						||
| 
								 | 
							
								  // In case there isn't a manual shard specified for this shard index, we just compute the
							 | 
						||
| 
								 | 
							
								  // tests for this shard. Note that we need to subtract the amount of manual shards because
							 | 
						||
| 
								 | 
							
								  // we need to split up the unassigned tests across the remaining available shards.
							 | 
						||
| 
								 | 
							
								  printTestNames(getTestsForShardIndex(
							 | 
						||
| 
								 | 
							
								      unassignedTests, shardIndex - amountManualShards, maxShards - amountManualShards));
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/**
							 | 
						||
| 
								 | 
							
								 * Splits the specified tests into a limited amount of shards and returns the tests that should
							 | 
						||
| 
								 | 
							
								 * run on the given shard. The shards of tests are being created deterministically and therefore
							 | 
						||
| 
								 | 
							
								 * we get reproducible tests when executing the same script multiple times.
							 | 
						||
| 
								 | 
							
								 */
							 | 
						||
| 
								 | 
							
								function getTestsForShardIndex(tests, shardIndex, maxShards) {
							 | 
						||
| 
								 | 
							
								  return tests.filter((n, index) => index % maxShards === shardIndex);
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/**
							 | 
						||
| 
								 | 
							
								 * Strips all manual tests from the list of integration tests. This is necessary because
							 | 
						||
| 
								 | 
							
								 * when computing the shards automatically we don't want to include manual tests again. This
							 | 
						||
| 
								 | 
							
								 * would mean that CircleCI runs some integration tests multiple times.
							 | 
						||
| 
								 | 
							
								 */
							 | 
						||
| 
								 | 
							
								function stripManualOverrides(integrationTests, manualShards) {
							 | 
						||
| 
								 | 
							
								  const allManualTests = manualShards.reduce((res, manualTests) => res.concat(manualTests), []);
							 | 
						||
| 
								 | 
							
								  return integrationTests.filter(testName => !allManualTests.includes(testName))
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/** Prints the specified test names to the stdout. */
							 | 
						||
| 
								 | 
							
								function printTestNames(testNames) {
							 | 
						||
| 
								 | 
							
								  // Print the test names joined with spaces because this allows Bash to easily convert the output
							 | 
						||
| 
								 | 
							
								  // of this script into an array.
							 | 
						||
| 
								 | 
							
								  process.stdout.write(testNames.join(' '));
							 | 
						||
| 
								 | 
							
								}
							 |