| // Copyright (c) 2024, the Dart project authors. Please see the AUTHORS file |
| // for details. All rights reserved. Use of this source code is governed by a |
| // BSD-style license that can be found in the LICENSE file. |
| |
| import 'dart:async'; |
| import 'dart:collection'; |
| import 'dart:convert'; |
| import 'dart:io'; |
| import 'dart:math'; |
| |
| import 'package:_fe_analyzer_shared/src/util/relativize.dart' as fe_shared; |
| import 'package:args/args.dart'; |
| import 'package:collection/collection.dart'; |
| import 'package:dev_compiler/dev_compiler.dart' as ddc_names |
| show libraryUriToJsIdentifier; |
| import 'package:front_end/src/api_unstable/ddc.dart' as fe; |
| import 'package:path/path.dart' as p; |
| import 'package:reload_test/ddc_helpers.dart' as ddc_helpers; |
| import 'package:reload_test/frontend_server_controller.dart'; |
| import 'package:reload_test/hot_reload_memory_filesystem.dart'; |
| import 'package:reload_test/hot_reload_receipt.dart'; |
| import 'package:reload_test/test_helpers.dart'; |
| |
| final buildRootUri = fe.computePlatformBinariesLocation(forceBuildDir: true); |
| final sdkRoot = Platform.script.resolve('../../../'); |
| |
| /// SDK test directory containing hot reload tests. |
| final allTestsUri = sdkRoot.resolve('tests/hot_reload/'); |
| |
| /// The separator between a test file and its inlined diff. |
| /// |
| /// All contents after this separator are considered are diff comments. |
| final testDiffSeparator = '/** DIFF **/'; |
| |
| Future<void> main(List<String> args) async { |
| final options = Options.parse(args); |
| if (options.help) { |
| print(options.usage); |
| return; |
| } |
| final runner = switch (options.runtime) { |
| RuntimePlatforms.chrome => ChromeSuiteRunner(options), |
| RuntimePlatforms.d8 => D8SuiteRunner(options), |
| RuntimePlatforms.vm => VMSuiteRunner(options), |
| }; |
| await runner.runSuite(options); |
| } |
| |
| /// Command line options for the hot reload test suite. |
| class Options { |
| final bool help; |
| final RuntimePlatforms runtime; |
| final String namedConfiguration; |
| final Uri? testResultsOutputDir; |
| final RegExp testNameFilter; |
| final DiffMode diffMode; |
| final bool debug; |
| final bool verbose; |
| |
| Options._({ |
| required this.help, |
| required this.runtime, |
| required this.namedConfiguration, |
| required this.testResultsOutputDir, |
| required this.testNameFilter, |
| required this.diffMode, |
| required this.debug, |
| required this.verbose, |
| }); |
| |
| static final _parser = ArgParser() |
| ..addFlag('help', |
| abbr: 'h', |
| help: 'Display this message.', |
| negatable: false, |
| defaultsTo: false) |
| ..addOption('runtime', |
| abbr: 'r', |
| defaultsTo: 'd8', |
| allowed: RuntimePlatforms.values.map((v) => v.text), |
| help: 'runtime platform used to run tests.') |
| ..addOption('named-configuration', |
| abbr: 'n', |
| defaultsTo: 'no-configuration', |
| help: 'configuration name to use for emitting test result files.') |
| ..addOption('output-directory', |
| help: 'directory to emit test results files.') |
| ..addOption('filter', |
| abbr: 'f', defaultsTo: r'.*', help: 'regexp filter over tests to run.') |
| ..addOption('diff', |
| allowed: ['check', 'write', 'ignore'], |
| allowedHelp: { |
| 'check': 'validate that reload test diffs are generated and correct.', |
| 'write': 'write diffs for reload tests.', |
| 'ignore': 'ignore reload diffs.', |
| }, |
| defaultsTo: 'check', |
| help: 'selects whether test diffs should be checked, written, or ' |
| 'ignored.') |
| ..addFlag('debug', |
| abbr: 'd', |
| defaultsTo: false, |
| negatable: true, |
| help: 'enables additional debug behavior and logging.') |
| ..addFlag('verbose', |
| abbr: 'v', |
| defaultsTo: true, |
| negatable: true, |
| help: 'enables verbose logging.'); |
| |
| /// Usage description for these command line options. |
| String get usage => _parser.usage; |
| |
| factory Options.parse(List<String> args) { |
| final results = _parser.parse(args); |
| return Options._( |
| help: results.flag('help'), |
| runtime: RuntimePlatforms.values.byName(results.option('runtime')!), |
| namedConfiguration: results.option('named-configuration')!, |
| testResultsOutputDir: results.wasParsed('output-directory') |
| ? Uri.directory(results.option('output-directory')!) |
| : null, |
| testNameFilter: RegExp(results.option('filter')!), |
| diffMode: switch (results.option('diff')!) { |
| 'check' => DiffMode.check, |
| 'write' => DiffMode.write, |
| 'ignore' => DiffMode.ignore, |
| _ => throw Exception('Invalid diff mode: ${results.option('diff')}'), |
| }, |
| debug: results.flag('debug'), |
| verbose: results.flag('verbose'), |
| ); |
| } |
| } |
| |
| /// Modes for running diff check tests on the hot reload suite. |
| enum DiffMode { check, write, ignore } |
| |
| /// A single test for the hot reload test suite. |
| /// |
| /// A hot reload test is made of a collection of one or more Dart files over a |
| /// number of generational edits that represent interactions with source files |
| /// over the life of a running test program. |
| /// |
| /// Tests in this suite also define a config.json file with further information |
| /// describing how the test runs. |
| class HotReloadTest { |
| /// Root [Directory] containing the source files for this test. |
| final Directory directory; |
| |
| /// Test name used in results. |
| /// |
| /// By convention this matches the name of the [directory]. |
| final String name; |
| |
| /// The files that make up this test. |
| final List<TestFile> files; |
| |
| /// The number of generations in this test (one-based). |
| final int generationCount; |
| |
| /// Platforms that should not run this test. |
| final Set<RuntimePlatforms> excludedPlatforms; |
| |
| /// Edit rejection error messages expected from this test when a hot reload is |
| /// triggered by the generation in which the error is expected. |
| /// |
| /// Specified in the `config.json` file. |
| // TODO(nshahan): Support multiple expected errors for a single generation. |
| final Map<int, String> expectedErrors; |
| |
| /// Map of generation number to whether a hot restart was triggered before the |
| /// generation. |
| final Map<int, bool> isHotRestart; |
| |
| HotReloadTest(this.directory, this.name, this.generationCount, this.files, |
| ReloadTestConfiguration config, this.isHotRestart) |
| : excludedPlatforms = config.excludedPlatforms, |
| expectedErrors = config.expectedErrors; |
| |
| /// The files edited in the provided [generation] (zero-based). |
| List<TestFile> filesEditedInGeneration(int generation) => [ |
| for (final file in files) |
| if (file._editsByGeneration.containsKey(generation)) file |
| ]; |
| } |
| |
| /// An individual test file for a hot reload test across all generations. |
| class TestFile { |
| /// The reconstructed name of the file after removing the generation tag. |
| /// |
| /// For example given the files foo.0.dart, foo.1.dart, and foo.2.dart the |
| /// [baseName] would be 'foo.dart'. |
| final String baseName; |
| |
| /// The individual edits of this file by their generations (zero-based). |
| /// |
| /// By convention, iterating the entries produces them in generational order |
| /// but there could be gaps in the generation numbers. |
| // TODO(nshahan): Move the creation of this data structure to this class to |
| // ensure the iteration order. Alternatively, update the representation to |
| // require there are no gaps in the generations so we don't have to handle |
| // them and can just use a list. |
| final LinkedHashMap<int, TestFileEdit> _editsByGeneration; |
| |
| TestFile(this.baseName, this._editsByGeneration); |
| |
| /// Returns the [TestFileEdit] for the given [generation] (zero-based) for |
| /// this file. |
| TestFileEdit editForGeneration(int generation) { |
| final edit = _editsByGeneration[generation]; |
| if (edit == null) { |
| throw Exception('File: $baseName has no generation: $generation.'); |
| } |
| return edit; |
| } |
| |
| /// All edits for this file in order by generation. |
| List<TestFileEdit> get edits => _editsByGeneration.values.toList(); |
| } |
| |
| /// A single version of a test file at a specific generation. |
| class TestFileEdit { |
| /// The generation this edit belongs to. |
| final int generation; |
| |
| /// The location of this file on disk. |
| final Uri fileUri; |
| |
| TestFileEdit(this.generation, this.fileUri); |
| } |
| |
| abstract class HotReloadSuiteRunner { |
| Options options; |
| |
| /// The root directory containing generated code for all tests. |
| late final Directory generatedCodeDir = Directory.systemTemp.createTempSync(); |
| |
| /// The directory containing files emitted from Frontend Server compiles and |
| /// recompiles. |
| late final Directory frontendServerEmittedFilesDir = |
| Directory.fromUri(generatedCodeDir.uri.resolve('.fes/'))..createSync(); |
| |
| /// The output location for .dill file created by the front end server. |
| late final Uri outputDillUri = |
| frontendServerEmittedFilesDir.uri.resolve('output.dill'); |
| |
| /// The output location for the incremental .dill file created by the front |
| /// end server. |
| late final Uri outputIncrementalDillUri = |
| frontendServerEmittedFilesDir.uri.resolve('output_incremental.dill'); |
| |
| /// All test results that are reported after running the entire test suite. |
| final List<TestResultOutcome> testOutcomes = []; |
| |
| /// The directory used as a temporary staging area to construct a compile-able |
| /// test app across reload/restart generations. |
| late final Directory snapshotDir = |
| Directory.fromUri(generatedCodeDir.uri.resolve('.snapshot/')) |
| ..createSync(); |
| |
| // TODO(markzipan): Support custom entrypoints. |
| late final Uri snapshotEntrypointUri = snapshotDir.uri.resolve('main.dart'); |
| late final String snapshotEntrypointWithScheme = () { |
| final snapshotEntrypointLibraryName = fe_shared.relativizeUri( |
| snapshotDir.uri, snapshotEntrypointUri, fe_shared.isWindows); |
| return '$filesystemScheme:///$snapshotEntrypointLibraryName'; |
| }(); |
| |
| HotReloadMemoryFilesystem? filesystem; |
| final stopwatch = Stopwatch(); |
| |
| final filesystemScheme = 'hot-reload-test'; |
| |
| HotReloadSuiteRunner(this.options); |
| |
| Future<void> runSuite(Options options) async { |
| // TODO(nshahan): report time for collecting and validating test sources. |
| final testSuite = collectTestSources(options); |
| _debugPrint( |
| 'See generated hot reload framework code in ${generatedCodeDir.uri}'); |
| final controller = createFrontEndServer(); |
| for (final test in testSuite) { |
| stopwatch |
| ..start() |
| ..reset(); |
| diffCheck(test); |
| final tempDirectory = |
| Directory.fromUri(generatedCodeDir.uri.resolve('${test.name}/')) |
| ..createSync(); |
| if (options.runtime.emitsJS) { |
| filesystem = HotReloadMemoryFilesystem(tempDirectory.uri); |
| } |
| var compileSuccess = false; |
| _print('Generating test assets.', label: test.name); |
| // TODO(markzipan): replace this with a test-configurable main entrypoint. |
| final mainDartFilePath = |
| test.directory.uri.resolve('main.dart').toFilePath(); |
| _debugPrint('Test entrypoint: $mainDartFilePath', label: test.name); |
| _print('Generating code over ${test.generationCount} generations.', |
| label: test.name); |
| stopwatch |
| ..start() |
| ..reset(); |
| for (var generation = 0; |
| generation < test.generationCount; |
| generation++) { |
| final updatedFiles = copyGenerationSources(test, generation); |
| compileSuccess = await compileGeneration( |
| test, generation, tempDirectory, updatedFiles, controller); |
| if (!compileSuccess) break; |
| } |
| if (!compileSuccess) { |
| _print('Did not emit all assets due to compilation error.', |
| label: test.name); |
| // Skip to the next test and avoid execution if there is an unexpected |
| // compilation error. |
| continue; |
| } |
| _print('Finished emitting assets.', label: test.name); |
| final testOutputStreamController = StreamController<List<int>>(); |
| final testOutputBuffer = StringBuffer(); |
| testOutputStreamController.stream |
| .transform(utf8.decoder) |
| .listen(testOutputBuffer.write); |
| final testPassed = await runTest( |
| test, tempDirectory, IOSink(testOutputStreamController.sink)); |
| await reportTestOutcome( |
| test.name, testOutputBuffer.toString(), testPassed); |
| } |
| await shutdown(controller); |
| await reportAllResults(); |
| } |
| |
| /// Custom command line arguments passed to the Front End Server on startup. |
| List<String> get platformFrontEndServerArgs; |
| |
| /// Returns a controller for a freshly started front end server instance to |
| /// handle compile and recompile requests for a hot reload test. |
| HotReloadFrontendServerController createFrontEndServer() { |
| _print('Initializing the Frontend Server.'); |
| final packageConfigUri = sdkRoot.resolve('.dart_tool/package_config.json'); |
| final fesArgs = [ |
| '--incremental', |
| '--filesystem-root=${snapshotDir.uri.toFilePath()}', |
| '--filesystem-scheme=$filesystemScheme', |
| '--output-dill=${outputDillUri.toFilePath()}', |
| '--output-incremental-dill=${outputIncrementalDillUri.toFilePath()}', |
| '--packages=${packageConfigUri.toFilePath()}', |
| '--sdk-root=${sdkRoot.toFilePath()}', |
| '--verbosity=${options.verbose ? 'all' : 'info'}', |
| ...platformFrontEndServerArgs, |
| ]; |
| return HotReloadFrontendServerController(fesArgs)..start(); |
| } |
| |
| Future<void> shutdown(HotReloadFrontendServerController controller) async { |
| // Persist the temp directory for debugging. |
| await controller.stop(); |
| _print('Frontend Server has shut down.'); |
| if (!options.debug) { |
| _print('Deleting temporary directory: ${generatedCodeDir.path}.'); |
| generatedCodeDir.deleteSync(recursive: true); |
| } |
| } |
| |
| /// Returns a suite of hot reload tests discovered in the directory |
| /// [allTestsUri]. |
| /// |
| /// Assumes all files that makeup a hot reload test are named like |
| /// '$name.$integer.dart', where 0 is the first generation. |
| /// |
| /// Count the number of generations and ensure they're capped. |
| // TODO(markzipan): Account for subdirectories. |
| List<HotReloadTest> collectTestSources(Options options) { |
| // Set an arbitrary cap on generations. |
| final globalMaxGenerations = 100; |
| final validTestSourceName = RegExp( |
| // Begins with 1 or more word characters. |
| r'^(?<name>[\w,-]+)' |
| // Followed by a dot and 1 or more digits. |
| r'\.(?<generation>\d+)' |
| // Optionally a dot and either the word 'restart' indicating a hot |
| // restart, or the word 'reject', indicating a hot reload rejection. |
| r'((?<restart>\.restart)|(?<reject>\.reject))?' |
| // Ending with a dot and the word 'dart' |
| r'\.dart$'); |
| final testSuite = <HotReloadTest>[]; |
| for (var testDir in Directory.fromUri(allTestsUri).listSync()) { |
| if (testDir is! Directory) { |
| if (testDir is File) { |
| // Ignore Dart source files, which may be imported as helpers. |
| continue; |
| } |
| throw Exception('Non-directory or file entity found in ' |
| '${allTestsUri.toFilePath()}: $testDir'); |
| } |
| final testDirParts = testDir.uri.pathSegments; |
| final testName = testDirParts[testDirParts.length - 2]; |
| |
| // Skip tests that don't match the name filter. |
| if (!options.testNameFilter.hasMatch(testName)) { |
| _print('Skipping test', label: testName); |
| continue; |
| } |
| var maxGenerations = 0; |
| final configFileUri = testDir.uri.resolve('config.json'); |
| final testConfig = File.fromUri(configFileUri).existsSync() |
| ? ReloadTestConfiguration.fromJsonFile(configFileUri) |
| : ReloadTestConfiguration(); |
| if (testConfig.excludedPlatforms.contains(options.runtime)) { |
| // Skip this test directory if this platform is excluded. |
| _print('Skipping test on platform: ${options.runtime.text}', |
| label: testName); |
| continue; |
| } |
| final isHotRestart = <int, bool>{}; |
| final expectedErrors = testConfig.expectedErrors; |
| final dartFiles = testDir |
| .listSync() |
| .where((e) => e is File && e.uri.path.endsWith('.dart')); |
| // All files in this test clustered by file name - in generation order. |
| final filesByGeneration = <String, PriorityQueue<TestFileEdit>>{}; |
| for (final file in dartFiles) { |
| final fileName = p.basename(file.uri.toFilePath()); |
| final matches = validTestSourceName.allMatches(fileName); |
| if (matches.length != 1) { |
| throw Exception('Invalid test source file name: $fileName\n' |
| 'Valid names look like "file_name.10.dart", ' |
| '"file_name.10.restart.dart" or "file_name.10.reject.dart".'); |
| } |
| final match = matches.single; |
| final name = match.namedGroup('name'); |
| final restoredName = '$name.dart'; |
| final generation = int.parse(match.namedGroup('generation')!); |
| maxGenerations = max(maxGenerations, generation); |
| final restart = match.namedGroup('restart') != null; |
| if (!isHotRestart.containsKey(generation)) { |
| isHotRestart[generation] = restart; |
| } else { |
| if (restart != isHotRestart[generation]) { |
| throw Exception('Expected all files for generation $generation to ' |
| "be consistent about having a '.restart' suffix, but $fileName " |
| 'does not match other files in the same generation.'); |
| } |
| } |
| final rejectExpected = match.namedGroup('reject') != null; |
| assert(!(rejectExpected && restart)); |
| if (rejectExpected && !expectedErrors.containsKey(generation)) { |
| throw Exception( |
| 'Expected error for generation file missing from config.json: ' |
| '$fileName'); |
| } |
| if (!rejectExpected && expectedErrors.containsKey(generation)) { |
| throw Exception( |
| 'Error for generation $generation found in config.json: ' |
| '"${expectedErrors[generation]}"\n' |
| 'Either remove the error or update the name of this file: ' |
| '$fileName -> ' |
| '$name.$generation.reject.dart'); |
| } |
| if (generation == 0 && rejectExpected) { |
| throw Exception('The first generation may not be rejected: ' |
| '$fileName'); |
| } |
| filesByGeneration |
| .putIfAbsent( |
| restoredName, |
| () => PriorityQueue((TestFileEdit a, TestFileEdit b) => |
| a.generation - b.generation)) |
| .add(TestFileEdit(generation, file.uri)); |
| } |
| if (maxGenerations > globalMaxGenerations) { |
| throw Exception('Too many generations specified in test ' |
| '(requested: $maxGenerations, max: $globalMaxGenerations).'); |
| } |
| final testFiles = <TestFile>[]; |
| for (final entry in filesByGeneration.entries) { |
| final fileName = entry.key; |
| final fileEdits = entry.value.toList(); |
| final editsByGeneration = LinkedHashMap<int, TestFileEdit>.from( |
| {for (final edit in fileEdits) edit.generation: edit}); |
| testFiles.add(TestFile(fileName, editsByGeneration)); |
| } |
| testSuite.add(HotReloadTest(testDir, testName, maxGenerations + 1, |
| testFiles, testConfig, isHotRestart)); |
| } |
| return testSuite; |
| } |
| |
| /// Report results for this test's execution. |
| Future<void> reportTestOutcome( |
| String testName, String testOutput, bool testPassed) async { |
| stopwatch.stop(); |
| final outcome = TestResultOutcome( |
| configuration: options.namedConfiguration, |
| testName: testName, |
| testOutput: testOutput, |
| ); |
| outcome.elapsedTime = stopwatch.elapsed; |
| outcome.matchedExpectations = testPassed; |
| testOutcomes.add(outcome); |
| if (testPassed) { |
| _print('PASSED with:\n $testOutput', label: testName); |
| } else { |
| _print('FAILED with:\n $testOutput', label: testName); |
| } |
| } |
| |
| /// Report results for this test's sources' diff validations. |
| void reportDiffOutcome( |
| String testName, Uri fileUri, String testOutput, bool testPassed) { |
| stopwatch.stop(); |
| final filePath = fileUri.path; |
| final relativeFilePath = p.relative(filePath, from: allTestsUri.path); |
| final outcome = TestResultOutcome( |
| configuration: options.namedConfiguration, |
| testName: '$relativeFilePath-diff', |
| testOutput: testOutput, |
| ); |
| outcome.elapsedTime = stopwatch.elapsed; |
| outcome.matchedExpectations = testPassed; |
| testOutcomes.add(outcome); |
| if (testPassed) { |
| _debugPrint('PASSED (diff on $filePath) with:\n $testOutput', |
| label: testName); |
| } else { |
| _debugPrint('FAILED (diff on $filePath) with:\n $testOutput', |
| label: testName); |
| } |
| } |
| |
| /// Performs the desired diff checks for [test] and reports the results. |
| void diffCheck(HotReloadTest test) { |
| var diffMode = options.diffMode; |
| if (fe_shared.isWindows && diffMode != DiffMode.ignore) { |
| _print("Diffing isn't supported on Windows. Defaulting to 'ignore'.", |
| label: test.name); |
| diffMode = DiffMode.ignore; |
| } |
| switch (diffMode) { |
| case DiffMode.check: |
| _print('Checking source file diffs.', label: test.name); |
| for (final file in test.files) { |
| _debugPrint('Checking source file diffs for $file.', |
| label: test.name); |
| var edits = file.edits.iterator; |
| if (!edits.moveNext()) { |
| throw Exception('Test file created with no generation edits.'); |
| } |
| var currentEdit = edits.current; |
| // Check that the first file does not have a diff. |
| var (currentCode, currentDiff) = |
| _splitTestByDiff(currentEdit.fileUri); |
| var diffCount = testDiffSeparator.allMatches(currentDiff).length; |
| if (diffCount == 0) { |
| reportDiffOutcome(test.name, currentEdit.fileUri, |
| 'First generation does not have a diff', true); |
| } else { |
| reportDiffOutcome(test.name, currentEdit.fileUri, |
| 'First generation should not have any diffs', false); |
| } |
| while (edits.moveNext()) { |
| final previousEdit = currentEdit; |
| currentEdit = edits.current; |
| final previousCode = currentCode; |
| (currentCode, currentDiff) = _splitTestByDiff(currentEdit.fileUri); |
| // Check that exactly one diff exists. |
| diffCount = testDiffSeparator.allMatches(currentDiff).length; |
| if (diffCount == 0) { |
| reportDiffOutcome(test.name, currentEdit.fileUri, |
| 'No diff found for ${currentEdit.fileUri}', false); |
| continue; |
| } else if (diffCount > 1) { |
| reportDiffOutcome( |
| test.name, |
| currentEdit.fileUri, |
| 'Too many diffs found for ${currentEdit.fileUri} ' |
| '(expected 1)', |
| false); |
| continue; |
| } |
| // Check that the diff is properly generated. |
| // 'main' is allowed to have empty diffs since the first |
| // generation must be specified. |
| if (file.baseName != 'main.dart' && previousCode == currentCode) { |
| // TODO(markzipan): Should we make this an error? |
| _print( |
| 'Extraneous file detected. ${currentEdit.fileUri} ' |
| 'is identical to ${previousEdit.fileUri} and can be removed.', |
| label: test.name); |
| } |
| final previousTempUri = generatedCodeDir.uri.resolve('__previous'); |
| final currentTempUri = generatedCodeDir.uri.resolve('__current'); |
| // Avoid 'No newline at end of file' messages in the output by |
| // appending a newline to the trimmed source code strings. |
| File.fromUri(previousTempUri).writeAsStringSync('$previousCode\n'); |
| File.fromUri(currentTempUri).writeAsStringSync('$currentCode\n'); |
| final diffOutput = _diffWithFileUris( |
| previousTempUri, currentTempUri, |
| label: test.name); |
| File.fromUri(previousTempUri).deleteSync(); |
| File.fromUri(currentTempUri).deleteSync(); |
| if (diffOutput != currentDiff) { |
| reportDiffOutcome( |
| test.name, |
| currentEdit.fileUri, |
| 'Unexpected diff found for ${currentEdit.fileUri}:\n' |
| '-- Expected --\n$diffOutput\n' |
| '-- Actual --\n$currentDiff', |
| false); |
| } else { |
| reportDiffOutcome(test.name, currentEdit.fileUri, |
| 'Correct diff found for ${currentEdit.fileUri}', true); |
| } |
| } |
| } |
| case DiffMode.write: |
| _print('Generating source file diffs.', label: test.name); |
| for (final file in test.files) { |
| _debugPrint('Generating source file diffs for ${file.edits}.', |
| label: test.name); |
| var edits = file.edits.iterator; |
| if (!edits.moveNext()) { |
| throw Exception('Test file created with no generation edits.'); |
| } |
| var currentEdit = edits.current; |
| var (currentCode, currentDiff) = |
| _splitTestByDiff(currentEdit.fileUri); |
| // Don't generate a diff for the first file of any generation, |
| // and delete any diffs encountered. |
| if (currentDiff.isNotEmpty) { |
| _print('Removing extraneous diff from ${currentEdit.fileUri}', |
| label: test.name); |
| File.fromUri(currentEdit.fileUri).writeAsStringSync(currentCode); |
| } |
| while (edits.moveNext()) { |
| currentEdit = edits.current; |
| final previousCode = currentCode; |
| (currentCode, currentDiff) = _splitTestByDiff(currentEdit.fileUri); |
| final previousTempUri = generatedCodeDir.uri.resolve('__previous'); |
| final currentTempUri = generatedCodeDir.uri.resolve('__current'); |
| // Avoid 'No newline at end of file' messages in the output by |
| // appending a newline to the trimmed source code strings. |
| File.fromUri(previousTempUri).writeAsStringSync('$previousCode\n'); |
| File.fromUri(currentTempUri).writeAsStringSync('$currentCode\n'); |
| final diffOutput = _diffWithFileUris( |
| previousTempUri, currentTempUri, |
| label: test.name); |
| File.fromUri(previousTempUri).deleteSync(); |
| File.fromUri(currentTempUri).deleteSync(); |
| // Write an empty line between the code and the diff comment to |
| // agree with the dart formatter. |
| final newCurrentText = '$currentCode\n\n$diffOutput\n'; |
| File.fromUri(currentEdit.fileUri).writeAsStringSync(newCurrentText); |
| _print('Writing updated diff to $currentEdit.fileUri', |
| label: test.name); |
| _debugPrint('Updated diff:\n$diffOutput', label: test.name); |
| reportDiffOutcome(test.name, currentEdit.fileUri, |
| 'diff updated for $currentEdit.fileUri', true); |
| } |
| } |
| case DiffMode.ignore: |
| _print('Ignoring source file diffs.', label: test.name); |
| for (final file in test.files) { |
| for (final edit in file.edits) { |
| var uri = edit.fileUri; |
| _debugPrint('Ignoring source file diffs for $uri.', |
| label: test.name); |
| reportDiffOutcome(test.name, uri, 'Ignoring diff for $uri', true); |
| } |
| } |
| } |
| } |
| |
| /// Attempts to extract a reload receipt from [line] and if found passes it as |
| /// a [HotReloadReceipt] to [onReloadReceipt]. |
| /// |
| /// If no reload receipt is found the line is passed to [orElse]. [test] is |
| /// only used to label debug logs. |
| void parseReloadReceipt(HotReloadTest test, String line, |
| Function(HotReloadReceipt) onReloadReceipt, Function(String) orElse) { |
| if (line.startsWith(HotReloadReceipt.hotReloadReceiptTag)) { |
| // Reload utils write reload receipts as output lines with a leading tag |
| // so the lines can be extracted here. |
| final reloadReceipt = HotReloadReceipt.fromJson(jsonDecode( |
| line.substring(HotReloadReceipt.hotReloadReceiptTag.length)) |
| as Map<String, dynamic>); |
| onReloadReceipt(reloadReceipt); |
| _debugPrint( |
| [ |
| 'Generation ${reloadReceipt.generation} ' |
| 'was ${reloadReceipt.status.name}', |
| if (reloadReceipt.status == Status.rejected) |
| ': "${reloadReceipt.rejectionMessage}"' |
| else |
| '.', |
| ].join(), |
| label: test.name); |
| } else { |
| orElse(line); |
| } |
| } |
| |
| /// Validates all reloads/restarts and returns `true` if they were performed |
| /// as expected during the test run. |
| /// |
| /// This serves as a sanity check to ensure that just because no errors were |
| /// reported, the test still ran through all expected generations with the |
| /// expected accept/reject/restart status. |
| bool reloadReceiptCheck( |
| HotReloadTest test, List<HotReloadReceipt> reloadReceipts) { |
| // Check number of reloads. |
| // No reload receipt will appear for generation 0. |
| final expectedReloadCount = test.generationCount - 1; |
| if (reloadReceipts.length != expectedReloadCount) { |
| _print( |
| 'Unexpected number of reloads/restarts were performed. ' |
| 'Expected: $expectedReloadCount Actual: ${reloadReceipts.length}\n' |
| '${reloadReceipts.join('\n')}', |
| label: test.name); |
| return false; |
| } |
| var expectedGeneration = 0; |
| for (final reloadReceipt in reloadReceipts) { |
| expectedGeneration++; |
| // Validate order of reloads. |
| if (reloadReceipt.generation != expectedGeneration) { |
| _print( |
| 'Generation reload order mismatch. ' |
| 'Expected: $expectedGeneration ' |
| 'Actual: ${reloadReceipt.generation}\n' |
| '${reloadReceipts.join('\n')}', |
| label: test.name); |
| return false; |
| } |
| final expectedError = test.expectedErrors[reloadReceipt.generation]; |
| // Check the reloads match the expected accept/reject. |
| if (reloadReceipt.status == Status.accepted && expectedError != null) { |
| _print( |
| 'Generation ${reloadReceipt.generation} was not rejected. ' |
| 'Expected: $expectedError', |
| label: test.name); |
| return false; |
| } |
| if (reloadReceipt.status == Status.rejected) { |
| if (expectedError == null) { |
| _print( |
| 'Generation ${reloadReceipt.generation} was unexpectedly ' |
| 'rejected: ${reloadReceipt.rejectionMessage}', |
| label: test.name); |
| return false; |
| } |
| final rejectionMessage = reloadReceipt.rejectionMessage; |
| if (rejectionMessage == null || |
| !rejectionMessage.contains(expectedError)) { |
| _print( |
| 'Generation ${reloadReceipt.generation} was rejected but error ' |
| 'was unexpected. Expected: "$expectedError" Actual: ' |
| '${reloadReceipt.rejectionMessage}', |
| label: test.name); |
| return false; |
| } |
| } |
| } |
| _debugPrint( |
| 'Generation reloads matched expected outcomes:\n' |
| ' ${reloadReceipts.join('\n ')}', |
| label: test.name); |
| return true; |
| } |
| |
| /// Copy all files in [test] for the given [generation] into the snapshot |
| /// directory and returns uris of all the files copied. |
| /// |
| /// The uris describe the copy destination in the form of the hot reload file |
| /// system scheme. |
| List<String> copyGenerationSources(HotReloadTest test, int generation) { |
| _debugPrint('Entering generation $generation', label: test.name); |
| final updatedFilesInCurrentGeneration = <String>[]; |
| // Copy all files in this generation to the snapshot directory with their |
| // names restored (e.g., path/to/main' from 'path/to/main.0.dart). |
| // TODO(markzipan): support subdirectories. |
| _debugPrint( |
| 'Copying Dart files to snapshot directory: ' |
| '${snapshotDir.uri.toFilePath()}', |
| label: test.name); |
| for (final file in test.filesEditedInGeneration(generation)) { |
| final fileSnapshotUri = snapshotDir.uri.resolve(file.baseName); |
| final editUri = file.editForGeneration(generation).fileUri; |
| File.fromUri(editUri).copySync(fileSnapshotUri.toFilePath()); |
| final relativeSnapshotPath = fe_shared.relativizeUri( |
| snapshotDir.uri, fileSnapshotUri, fe_shared.isWindows); |
| final snapshotPathWithScheme = |
| '$filesystemScheme:///$relativeSnapshotPath'; |
| updatedFilesInCurrentGeneration.add(snapshotPathWithScheme); |
| } |
| _print( |
| 'Updated files in generation $generation: ' |
| '$updatedFilesInCurrentGeneration', |
| label: test.name); |
| return updatedFilesInCurrentGeneration; |
| } |
| |
| /// Compiles all [updatedFiles] in [test] for the given [generation] with the |
| /// front end server [controller], copies all outputs to [outputDirectory], |
| /// and returns whether the compilation was successful. |
| /// |
| /// Reports test failures on compile time errors. |
| Future<bool> compileGeneration( |
| HotReloadTest test, |
| int generation, |
| Directory outputDirectory, |
| List<String> updatedFiles, |
| HotReloadFrontendServerController controller); |
| |
| /// Runs [test] from compiled and generated assets in [tempDirectory] and |
| /// returns `true` if it passes. |
| /// |
| /// All output (standard and errors) from running the test is written to |
| /// [outputSink]. |
| Future<bool> runTest( |
| HotReloadTest test, Directory tempDirectory, IOSink outputSink); |
| |
| /// Reports test results to standard out as well as the output .json file if |
| /// requested. |
| Future<void> reportAllResults() async { |
| if (options.testResultsOutputDir != null) { |
| // Used to communicate individual test failures to our test bots. |
| final testOutcomeResults = testOutcomes.map((o) => o.toRecordJson()); |
| final testOutcomeLogs = testOutcomes.map((o) => o.toLogJson()); |
| final testResultsOutputDir = options.testResultsOutputDir!; |
| _print('Saving test results to ${testResultsOutputDir.toFilePath()}.'); |
| |
| // Test outputs must have one JSON blob per line and be |
| // newline-terminated. |
| final testResultsUri = testResultsOutputDir.resolve('results.json'); |
| final testResultsSink = File.fromUri(testResultsUri).openWrite(); |
| testOutcomeResults.forEach(testResultsSink.writeln); |
| await testResultsSink.flush(); |
| await testResultsSink.close(); |
| |
| final testLogsUri = testResultsOutputDir.resolve('logs.json'); |
| if (Platform.isWindows) { |
| // TODO(55297): Logs are disabled on windows until this but is fixed. |
| _print('Logs are not written on Windows. ' |
| 'See: https://github.com/dart-lang/sdk/issues/55297'); |
| } else { |
| final testLogsSink = File.fromUri(testLogsUri).openWrite(); |
| testOutcomeLogs.forEach(testLogsSink.writeln); |
| await testLogsSink.flush(); |
| await testLogsSink.close(); |
| } |
| _print('Emitted logs to ${testResultsUri.toFilePath()} ' |
| 'and ${testLogsUri.toFilePath()}.'); |
| } |
| if (testOutcomes.isEmpty) { |
| print('No tests ran: no sub-directories in ${allTestsUri.toFilePath()} ' |
| 'match the provided filter:\n' |
| '${options.testNameFilter}'); |
| exit(0); |
| } |
| // Report failed tests. |
| var failedTests = |
| testOutcomes.where((outcome) => !outcome.matchedExpectations); |
| if (failedTests.isNotEmpty) { |
| print('Some tests failed:'); |
| failedTests.forEach((outcome) { |
| print(outcome.testName); |
| }); |
| // Exit cleanly after writing test results. |
| exit(0); |
| } |
| } |
| |
| /// Runs the [command] with [args] in [environment]. |
| /// |
| /// Will echo the commands to the console before running them when running in |
| /// `verbose` mode. |
| Future<Process> startProcess(String name, String command, List<String> args, |
| {Map<String, String> environment = const {}, |
| ProcessStartMode mode = ProcessStartMode.normal}) { |
| if (options.verbose) { |
| print('Running $name:\n$command ${args.join(' ')}\n'); |
| if (environment.isNotEmpty) { |
| var environmentVariables = |
| environment.entries.map((e) => '${e.key}: ${e.value}').join('\n'); |
| print('With environment:\n$environmentVariables\n'); |
| } |
| } |
| return Process.start(command, args, mode: mode, environment: environment); |
| } |
| |
| /// Prints messages if 'verbose' mode is enabled. |
| void _print(String message, {String? label}) { |
| if (options.verbose) { |
| final labelText = label == null ? '' : '($label)'; |
| print('hot_reload_test$labelText: $message'); |
| } |
| } |
| |
| /// Prints messages if 'debug' mode is enabled. |
| void _debugPrint(String message, {String? label}) { |
| if (options.debug) { |
| final labelText = label == null ? '' : '($label)'; |
| print('DEBUG$labelText: $message'); |
| } |
| } |
| |
| /// Returns the diff'd output between two files. |
| /// |
| /// These diffs are appended at the end of updated file generations for better |
| /// test readability. |
| /// |
| /// If [commented] is set, the output will be wrapped in multiline comments |
| /// and the diff separator. |
| /// |
| /// If [trimHeaders] is set, the leading '+++' and '---' file headers will be |
| /// removed. |
| String _diffWithFileUris(Uri file1, Uri file2, |
| {String label = '', bool commented = true, bool trimHeaders = true}) { |
| final file1Path = file1.toFilePath(); |
| final file2Path = file2.toFilePath(); |
| final diffArgs = ['diff', '-u', file1Path, file2Path]; |
| _debugPrint("Running diff with 'git diff ${diffArgs.join(' ')}'.", |
| label: label); |
| final diffProcess = Process.runSync('git', diffArgs); |
| final errOutput = diffProcess.stderr as String; |
| if (errOutput.isNotEmpty) { |
| throw Exception('git diff failed with:\n$errOutput'); |
| } |
| var output = diffProcess.stdout as String; |
| if (trimHeaders) { |
| // Skip the diff header. 'git diff' has 5 lines in its header. |
| // TODO(markzipan): Add support for Windows-style line endings. |
| output = output.split('\n').skip(5).join('\n'); |
| } |
| return commented ? '$testDiffSeparator\n/*\n$output*/' : output; |
| } |
| |
| /// Returns the code and diff portions of [file] with all leading and trailing |
| /// whitespace trimmed. |
| (String, String) _splitTestByDiff(Uri file) { |
| final text = File.fromUri(file).readAsStringSync(); |
| final diffIndex = text.indexOf(testDiffSeparator); |
| final diffSplitIndex = diffIndex == -1 ? text.length - 1 : diffIndex; |
| final codeText = text.substring(0, diffSplitIndex).trim(); |
| final diffText = text.substring(diffSplitIndex, text.length - 1).trim(); |
| return (codeText, diffText); |
| } |
| } |
| |
| /// Hot reload test suite runner for DDC specific behavior that is agnostic to |
| /// the environment where the compiled code is eventually run. |
| abstract class DdcSuiteRunner extends HotReloadSuiteRunner { |
| late final String entrypointLibraryExportName = |
| ddc_names.libraryUriToJsIdentifier(snapshotEntrypointUri); |
| final Uri dartSdkJSUri = |
| buildRootUri.resolve('gen/utils/ddc/canary/sdk/ddc/dart_sdk.js'); |
| final Uri ddcModuleLoaderJSUri = |
| sdkRoot.resolve('pkg/dev_compiler/lib/js/ddc/ddc_module_loader.js'); |
| final String ddcPlatformDillFromSdkRoot = fe_shared.relativizeUri( |
| sdkRoot, buildRootUri.resolve('ddc_outline.dill'), fe_shared.isWindows); |
| final String entrypointModuleName = 'hot-reload-test:///main.dart'; |
| |
| DdcSuiteRunner(super.options); |
| |
| @override |
| List<String> get platformFrontEndServerArgs => [ |
| '--dartdevc-module-format=ddc', |
| '--dartdevc-canary', |
| '--platform=$ddcPlatformDillFromSdkRoot', |
| '--target=dartdevc', |
| ]; |
| |
| @override |
| Future<bool> compileGeneration( |
| HotReloadTest test, |
| int generation, |
| Directory outputDirectory, |
| List<String> updatedFiles, |
| HotReloadFrontendServerController controller) async { |
| // The first generation calls `compile`, but subsequent ones call |
| // `recompile`. |
| // Likewise, use the incremental output file for `recompile` calls. |
| // TODO(nshahan): Sending compile/recompile instructions is likely |
| // the same across backends and should be shared code. |
| String outputDillPath; |
| _print('Compiling generation $generation with the Frontend Server.', |
| label: test.name); |
| CompilerOutput compilerOutput; |
| if (generation == 0) { |
| _debugPrint( |
| 'Compiling snapshot entrypoint: $snapshotEntrypointWithScheme', |
| label: test.name); |
| outputDillPath = outputDillUri.toFilePath(); |
| compilerOutput = |
| await controller.sendCompile(snapshotEntrypointWithScheme); |
| } else { |
| _debugPrint( |
| 'Recompiling snapshot entrypoint: $snapshotEntrypointWithScheme', |
| label: test.name); |
| outputDillPath = outputIncrementalDillUri.toFilePath(); |
| compilerOutput = await controller.sendRecompile( |
| snapshotEntrypointWithScheme, |
| invalidatedFiles: updatedFiles, |
| recompileRestart: test.isHotRestart[generation]!); |
| } |
| final expectedError = test.expectedErrors[generation]; |
| if (compilerOutput.errorCount > 0) { |
| // Frontend Server reported compile errors. |
| await controller.sendReject(); |
| if (expectedError != null && |
| compilerOutput.outputText.contains(expectedError)) { |
| // If the failure was an expected rejection it is OK to continue |
| // compiling generations and run the test. |
| _debugPrint( |
| 'DDC rejected generation $generation: ' |
| '"${compilerOutput.outputText}"', |
| label: test.name); |
| // Remove the expected error from this test to avoid expecting it to |
| // appear as a rejection at runtime. |
| test.expectedErrors[generation] = |
| HotReloadReceipt.compileTimeErrorMessage; |
| return true; |
| } else { |
| // Fail if the error was unexpected. |
| await reportTestOutcome( |
| test.name, |
| 'Test failed with compile error: ${compilerOutput.outputText}', |
| false); |
| return false; |
| } |
| } else { |
| // No errors were reported. |
| controller.sendAccept(); |
| } |
| if (expectedError != null) { |
| // A rejection error was expected but not seen. |
| await reportTestOutcome( |
| test.name, |
| 'Missing rejection for generation $generation. ' |
| 'Expected: "$expectedError"', |
| false); |
| return false; |
| } |
| _debugPrint( |
| 'Frontend Server successfully compiled outputs to: ' |
| '$outputDillPath', |
| label: test.name); |
| _debugPrint('Emitting JS code to ${outputDirectory.path}.', |
| label: test.name); |
| // Update the memory filesystem with the newly-created JS files. |
| _print('Loading generation $generation files into the memory filesystem.', |
| label: test.name); |
| final codeFile = File('$outputDillPath.sources'); |
| final manifestFile = File('$outputDillPath.json'); |
| final sourcemapFile = File('$outputDillPath.map'); |
| filesystem!.update(codeFile, manifestFile, sourcemapFile, |
| generation: '$generation'); |
| // Write JS files and sourcemaps to their respective generation. |
| _print('Writing generation $generation assets.', label: test.name); |
| _debugPrint('Writing JS assets to ${outputDirectory.path}', |
| label: test.name); |
| filesystem!.writeToDisk(outputDirectory.uri, generation: '$generation'); |
| return true; |
| } |
| } |
| |
| /// Hot reload test suite runner for behavior specific to DDC compiled code |
| /// running in D8. |
| class D8SuiteRunner extends DdcSuiteRunner { |
| D8SuiteRunner(super.options); |
| |
| @override |
| Future<bool> runTest( |
| HotReloadTest test, Directory tempDirectory, IOSink outputSink) async { |
| _print('Creating D8 hot reload test suite.', label: test.name); |
| final bootstrapJSUri = |
| tempDirectory.uri.resolve('generation0/bootstrap.js'); |
| _print('Preparing to run D8 test.', label: test.name); |
| final d8BootstrapJS = ddc_helpers.generateD8Bootstrapper( |
| ddcModuleLoaderJsPath: escapedString(ddcModuleLoaderJSUri.toFilePath()), |
| dartSdkJsPath: escapedString(dartSdkJSUri.toFilePath()), |
| entrypointModuleName: escapedString(entrypointModuleName), |
| entrypointLibraryExportName: escapedString(entrypointLibraryExportName), |
| scriptDescriptors: filesystem!.scriptDescriptorForBootstrap, |
| modifiedFilesPerGeneration: filesystem!.generationsToModifiedFilePaths, |
| ); |
| _debugPrint('Writing D8 bootstrapper: $bootstrapJSUri', label: test.name); |
| final bootstrapJSFile = File.fromUri(bootstrapJSUri) |
| ..writeAsStringSync(d8BootstrapJS); |
| _debugPrint('Running test in D8.', label: test.name); |
| final reloadReceipts = <HotReloadReceipt>[]; |
| final config = ddc_helpers.D8Configuration(sdkRoot); |
| final process = await startProcess('D8', config.binary.toFilePath(), [ |
| config.sealNativeObjectScript.toFilePath(), |
| config.preamblesScript.toFilePath(), |
| bootstrapJSFile.path |
| ]); |
| process.stdout.transform(utf8.decoder).transform(LineSplitter()).listen( |
| (line) => parseReloadReceipt( |
| test, line, reloadReceipts.add, outputSink.writeln)); |
| return await process.exitCode == 0 && |
| reloadReceiptCheck(test, reloadReceipts); |
| } |
| } |
| |
| /// Hot reload test suite runner for behavior specific to DDC compiled code |
| /// running in Chrome. |
| class ChromeSuiteRunner extends DdcSuiteRunner { |
| ChromeSuiteRunner(super.options); |
| |
| @override |
| Future<bool> runTest( |
| HotReloadTest test, Directory tempDirectory, IOSink outputSink) async { |
| // TODO(markzipan): Chrome tests are currently only configured for |
| // debugging a single test instance. This is due to: |
| // 1) Our tests not capturing test success/failure signals. These must be |
| // determined programmatically since Chrome console errors are unrelated |
| // to the Chrome process's stderr. |
| // 2) Chrome not closing after a test. We need to add logic to detect when |
| // to either shut down Chrome or load the next test (reusing instances). |
| _print('Creating Chrome hot reload test suite.', label: test.name); |
| final mainEntrypointJSUri = |
| tempDirectory.uri.resolve('generation0/main_module.bootstrap.js'); |
| final bootstrapJSUri = |
| tempDirectory.uri.resolve('generation0/bootstrap.js'); |
| final bootstrapHtmlUri = |
| tempDirectory.uri.resolve('generation0/index.html'); |
| _print('Preparing to run Chrome test.', label: test.name); |
| var bootstrapHtml = ''' |
| <html> |
| <head> |
| <base href="/"> |
| </head> |
| <body> |
| <script src="$bootstrapJSUri"></script> |
| </body> |
| </html> |
| '''; |
| final entrypointLibraryExportName = |
| ddc_names.libraryUriToJsIdentifier(snapshotEntrypointUri); |
| final (chromeMainEntrypointJS, chromeBootstrapJS) = |
| ddc_helpers.generateChromeBootstrapperFiles( |
| ddcModuleLoaderJsPath: escapedString(ddcModuleLoaderJSUri.toFilePath()), |
| dartSdkJsPath: escapedString(dartSdkJSUri.toFilePath()), |
| entrypointModuleName: escapedString(entrypointModuleName), |
| mainModuleEntrypointJsPath: |
| escapedString(mainEntrypointJSUri.toFilePath()), |
| entrypointLibraryExportName: escapedString(entrypointLibraryExportName), |
| scriptDescriptors: filesystem!.scriptDescriptorForBootstrap, |
| modifiedFilesPerGeneration: filesystem!.generationsToModifiedFilePaths, |
| ); |
| _debugPrint( |
| 'Writing Chrome bootstrap files: ' |
| '$mainEntrypointJSUri, $bootstrapJSUri, $bootstrapHtmlUri', |
| label: test.name); |
| File.fromUri(mainEntrypointJSUri).writeAsStringSync(chromeMainEntrypointJS); |
| File.fromUri(bootstrapJSUri).writeAsStringSync(chromeBootstrapJS); |
| final bootstrapHtmlFile = File.fromUri(bootstrapHtmlUri) |
| ..writeAsStringSync(bootstrapHtml); |
| _debugPrint('Running test in Chrome.', label: test.name); |
| final reloadReceipts = <HotReloadReceipt>[]; |
| final config = ddc_helpers.ChromeConfiguration(sdkRoot); |
| // Specifying '--user-data-dir' forces Chrome to not reuse an instance. |
| final chromeDataDir = Directory.systemTemp.createTempSync(); |
| final process = await startProcess('Chrome', config.binary.toFilePath(), [ |
| '--no-first-run', |
| '--no-default-browser-check', |
| '--allow-file-access-from-files', |
| '--user-data-dir=${chromeDataDir.path}', |
| '--disable-default-apps', |
| '--disable-translate', |
| // These two flags are used to get the Chrome process to output messages |
| // to stderr so we can read the console.log messages. |
| // TODO(nshahan): Update if there is an easier way to get console.log |
| // messages. |
| '--enable-logging=stderr', |
| '--v=1', |
| bootstrapHtmlFile.path, |
| ]).then((process) { |
| StreamSubscription stdoutSubscription; |
| StreamSubscription stderrSubscription; |
| // The console.log messages in the output are prefixed with a header like: |
| // [42029:259:1126/154323.385793:INFO:CONSOLE(27547)] |
| final chromeConsoleLog = RegExp( |
| // Line starts with digits separated by ":", "." or "\"." |
| r'^\[[\d:\.\/]+' |
| // Followed by a console tag then digits in parenthesis and a space. |
| r':INFO:CONSOLE\(\d+\)\] ' |
| // Followed by the logged message in quotes. |
| r'"(?<consoleLog>.+)"' |
| // Followed by a comma and the source location. |
| r', source:.+$'); |
| |
| var stdoutDone = Completer<void>(); |
| var stderrDone = Completer<void>(); |
| |
| void closeStdout([_]) { |
| if (!stdoutDone.isCompleted) stdoutDone.complete(); |
| } |
| |
| void closeStderr([_]) { |
| if (!stderrDone.isCompleted) stderrDone.complete(); |
| } |
| |
| stdoutSubscription = process.stdout |
| .listen((data) => outputSink.addStream, onDone: closeStderr); |
| |
| stderrSubscription = process.stderr |
| .transform(utf8.decoder) |
| .transform(LineSplitter()) |
| .listen((rawLine) { |
| final matches = chromeConsoleLog.allMatches(rawLine); |
| // Only considering lines that match the chrome console log format. |
| // All other output is discarded here because the logging is very |
| // chatty. |
| if (matches.isNotEmpty) { |
| final line = matches.single.namedGroup('consoleLog')!; |
| parseReloadReceipt( |
| test, line, reloadReceipts.add, outputSink.writeln); |
| } |
| }, onDone: closeStderr); |
| |
| process.exitCode.then((exitCode) { |
| stdoutSubscription.cancel(); |
| stderrSubscription.cancel(); |
| closeStdout(); |
| closeStderr(); |
| }); |
| |
| Future.wait([stdoutDone.future, stderrDone.future]).then((_) { |
| _debugPrint('Chrome process successfully shut down.', label: test.name); |
| }); |
| |
| return process; |
| }); |
| |
| return await process.exitCode == 0 && |
| reloadReceiptCheck(test, reloadReceipts); |
| } |
| |
| @override |
| Future<void> runSuite(Options options) async { |
| // Only allow Chrome when debugging a single test. |
| // TODO(markzipan): Add support for full Chrome testing. |
| if (options.runtime == RuntimePlatforms.chrome) { |
| var matchingTests = |
| Directory.fromUri(allTestsUri).listSync().where((testDir) { |
| if (testDir is! Directory) return false; |
| final testDirParts = testDir.uri.pathSegments; |
| final testName = testDirParts[testDirParts.length - 2]; |
| return options.testNameFilter.hasMatch(testName); |
| }); |
| |
| if (matchingTests.length > 1) { |
| throw Exception('Chrome is only supported when debugging a single test.' |
| "Please filter on a single test with '-f'."); |
| } |
| } |
| await super.runSuite(options); |
| } |
| } |
| |
| /// Hot reload test suite runner for behavior specific to the VM. |
| class VMSuiteRunner extends HotReloadSuiteRunner { |
| final String vmPlatformDillFromSdkRoot = fe_shared.relativizeUri(sdkRoot, |
| buildRootUri.resolve('vm_platform_strong.dill'), fe_shared.isWindows); |
| |
| VMSuiteRunner(super.options); |
| |
| @override |
| List<String> get platformFrontEndServerArgs => [ |
| '--platform=$vmPlatformDillFromSdkRoot', |
| '--target=vm', |
| ]; |
| |
| @override |
| Future<bool> compileGeneration( |
| HotReloadTest test, |
| int generation, |
| Directory outputDirectory, |
| List<String> updatedFiles, |
| HotReloadFrontendServerController controller) async { |
| // The first generation calls `compile`, but subsequent ones call |
| // `recompile`. |
| // Likewise, use the incremental output directory for `recompile` calls. |
| // TODO(nshahan): Sending compile/recompile instructions is likely |
| // the same across backends and should be shared code. |
| String outputDillPath; |
| _print('Compiling generation $generation with the Frontend Server.', |
| label: test.name); |
| CompilerOutput compilerOutput; |
| if (generation == 0) { |
| _debugPrint( |
| 'Compiling snapshot entrypoint: $snapshotEntrypointWithScheme', |
| label: test.name); |
| outputDillPath = outputDillUri.toFilePath(); |
| compilerOutput = |
| await controller.sendCompile(snapshotEntrypointWithScheme); |
| } else { |
| _debugPrint( |
| 'Recompiling snapshot entrypoint: $snapshotEntrypointWithScheme', |
| label: test.name); |
| outputDillPath = outputIncrementalDillUri.toFilePath(); |
| // TODO(markzipan): Add logic to reject bad compiles. |
| compilerOutput = await controller.sendRecompile( |
| snapshotEntrypointWithScheme, |
| invalidatedFiles: updatedFiles, |
| // The VM never uses the `recompile-restart` instruction as it does |
| // not recompile during a hot restart. |
| recompileRestart: false); |
| } |
| var hasExpectedCompileError = false; |
| final expectedError = test.expectedErrors[generation]; |
| // Frontend Server reported compile errors. Fail if they weren't |
| // expected, and do not run tests. |
| if (compilerOutput.errorCount > 0) { |
| await controller.sendReject(); |
| if (expectedError != null && |
| compilerOutput.outputText.contains(expectedError)) { |
| hasExpectedCompileError = true; |
| _debugPrint( |
| 'VM rejected generation $generation: ' |
| '"${compilerOutput.outputText}"', |
| label: test.name); |
| // Remove the expected error from this test to avoid expecting it to |
| // appear as a rejection at runtime. |
| test.expectedErrors[generation] = |
| HotReloadReceipt.compileTimeErrorMessage; |
| } else { |
| await reportTestOutcome( |
| test.name, |
| 'Test failed with compile error: ${compilerOutput.outputText}', |
| false); |
| return false; |
| } |
| } else if (test.expectedErrors.containsKey(generation)) { |
| // Automatically reject generations that are expected to be rejected so |
| // the front end server can update it's internal state correctly. This |
| // ensures the next delta will always be calculated from against the last |
| // accepted generation. The actual rejections will be validated when the |
| // test runs on the VM. |
| await controller.sendReject(); |
| _debugPrint('VM compile automatically rejected generation: $generation.', |
| label: test.name); |
| } else { |
| controller.sendAccept(); |
| } |
| _debugPrint( |
| 'Frontend Server successfully compiled outputs to: ' |
| '$outputDillPath', |
| label: test.name); |
| final dillOutputDir = |
| Directory.fromUri(outputDirectory.uri.resolve('generation$generation')); |
| dillOutputDir.createSync(); |
| // Write an .error.dill file as a signal to the runtime utils that this |
| // generation contains compile time errors and should not be reloaded. |
| final dillOutputUri = hasExpectedCompileError |
| ? dillOutputDir.uri.resolve('${test.name}.error.dill') |
| : dillOutputDir.uri.resolve('${test.name}.dill'); |
| // Write dills to their respective generation. |
| _print('Writing generation $generation assets.', label: test.name); |
| _debugPrint('Writing dill to ${dillOutputUri.toFilePath()}', |
| label: test.name); |
| File(outputDillPath).copySync(dillOutputUri.toFilePath()); |
| return true; |
| } |
| |
| @override |
| Future<bool> runTest( |
| HotReloadTest test, Directory tempDirectory, IOSink outputSink) async { |
| final firstGenerationDillUri = |
| tempDirectory.uri.resolve('generation0/${test.name}.dill'); |
| // Start the VM at generation 0. |
| final vmArgs = [ |
| '--enable-vm-service=0', // 0 avoids port collisions. |
| '--disable-service-auth-codes', |
| '--disable-dart-dev', |
| firstGenerationDillUri.toFilePath(), |
| ]; |
| _debugPrint( |
| 'Starting VM with command: ' |
| '${Platform.executable} ${vmArgs.join(" ")}', |
| label: test.name); |
| final reloadReceipts = <HotReloadReceipt>[]; |
| final vm = await Process.start(Platform.executable, vmArgs); |
| vm.stdout |
| .transform(utf8.decoder) |
| .transform(LineSplitter()) |
| .listen((line) => parseReloadReceipt( |
| test, |
| line, |
| reloadReceipts.add, |
| (line) { |
| _debugPrint('VM stdout: $line', label: test.name); |
| outputSink.writeln(line); |
| }, |
| )); |
| vm.stderr |
| .transform(utf8.decoder) |
| .transform(LineSplitter()) |
| .listen((String err) { |
| _debugPrint('VM stderr: $err', label: test.name); |
| outputSink.writeln(err); |
| }); |
| _print('Executing VM test.', label: test.name); |
| final testTimeoutSeconds = 10; |
| final vmExitCode = await vm.exitCode |
| .timeout(Duration(seconds: testTimeoutSeconds), onTimeout: () { |
| final timeoutText = 'Test timed out after $testTimeoutSeconds seconds.'; |
| _print(timeoutText, label: test.name); |
| outputSink.writeln(timeoutText); |
| vm.kill(); |
| return 1; |
| }); |
| return vmExitCode == 0 && reloadReceiptCheck(test, reloadReceipts); |
| } |
| } |