| // Copyright (c) 2014, 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:io'; |
| |
| import 'package:dart_style/dart_style.dart'; |
| import 'package:dart_style/src/constants.dart'; |
| import 'package:dart_style/src/testing/benchmark.dart'; |
| import 'package:dart_style/src/testing/test_file.dart'; |
| import 'package:path/path.dart' as p; |
| import 'package:test/test.dart'; |
| import 'package:test_descriptor/test_descriptor.dart' as d; |
| import 'package:test_process/test_process.dart'; |
| |
| const unformattedSource = 'void main() => print("hello") ;'; |
| const formattedSource = 'void main() => print("hello");\n'; |
| |
| /// The same as formatted source but without a trailing newline because |
| /// [TestProcess] filters those when it strips command line output into lines. |
| const formattedOutput = 'void main() => print("hello");'; |
| |
| /// If tool/command_shell.dart has been compiled to a snapshot, this is the path |
| /// to it. |
| String? _commandExecutablePath; |
| |
| /// If bin/format.dart has been compiled to a snapshot, this is the path to it. |
| String? _formatterExecutablePath; |
| |
| /// Compiles format.dart to a native executable for tests to use. |
| /// |
| /// Calls [setupAll()] and [tearDownAll()] to coordinate this when the |
| /// subsequent tests and to clean up the executable. |
| void compileFormatterExecutable() { |
| setUpAll(() async { |
| _formatterExecutablePath = await _compileSnapshot('bin/format.dart'); |
| }); |
| |
| tearDownAll(() async { |
| await _deleteSnapshot(_formatterExecutablePath!); |
| _formatterExecutablePath = null; |
| }); |
| } |
| |
| /// Compiles command_shell.dart to a native executable for tests to use. |
| /// |
| /// Calls [setupAll()] and [tearDownAll()] to coordinate this when the |
| /// subsequent tests and to clean up the executable. |
| void compileCommandExecutable() { |
| setUpAll(() async { |
| _commandExecutablePath = await _compileSnapshot('tool/command_shell.dart'); |
| }); |
| |
| tearDownAll(() async { |
| await _deleteSnapshot(_commandExecutablePath!); |
| _commandExecutablePath = null; |
| }); |
| } |
| |
| /// Compile the Dart [script] to an app-JIT snapshot. |
| /// |
| /// We do this instead of spawning the script from source each time because it's |
| /// much faster when the same script needs to be run several times. |
| Future<String> _compileSnapshot(String script) async { |
| var scriptName = p.basename(script); |
| var tempDir = |
| await Directory.systemTemp.createTemp(p.withoutExtension(scriptName)); |
| var snapshot = p.join(tempDir.path, '$scriptName.snapshot'); |
| var scriptPath = p.normalize(p.join(await findTestDirectory(), '..', script)); |
| |
| var compileResult = await Process.run(Platform.resolvedExecutable, [ |
| '--snapshot-kind=app-jit', |
| '--snapshot=$snapshot', |
| scriptPath, |
| '--help' |
| ]); |
| |
| if (compileResult.exitCode != 0) { |
| fail('Could not compile $scriptName to a snapshot (exit code ' |
| '${compileResult.exitCode}):\n${compileResult.stdout}\n\n' |
| '${compileResult.stderr}'); |
| } |
| |
| return snapshot; |
| } |
| |
| /// Attempts to delete to temporary directory created for [snapshot] by |
| /// [_compileSnapshot()]. |
| Future<void> _deleteSnapshot(String snapshot) async { |
| try { |
| await Directory(p.dirname(snapshot)).delete(recursive: true); |
| } on IOException { |
| // Do nothing if we failed to delete it. The OS will eventually clean it |
| // up. |
| } |
| } |
| |
| /// Runs the command line formatter, passing it [args]. |
| Future<TestProcess> runFormatter([List<String>? args]) { |
| if (_formatterExecutablePath == null) { |
| fail('Must call createFormatterExecutable() before running commands.'); |
| } |
| |
| return TestProcess.start( |
| Platform.resolvedExecutable, [_formatterExecutablePath!, ...?args], |
| workingDirectory: d.sandbox); |
| } |
| |
| /// Runs the command line formatter, passing it the test directory followed by |
| /// [args]. |
| Future<TestProcess> runFormatterOnDir([List<String>? args]) { |
| return runFormatter(['.', ...?args]); |
| } |
| |
| /// Runs the test shell for the [Command]-based formatter, passing it [args]. |
| Future<TestProcess> runCommand([List<String>? args]) { |
| if (_commandExecutablePath == null) { |
| fail('Must call createCommandExecutable() before running commands.'); |
| } |
| |
| return TestProcess.start(Platform.resolvedExecutable, |
| [_commandExecutablePath!, 'format', ...?args], |
| workingDirectory: d.sandbox); |
| } |
| |
| /// Runs the test shell for the [Command]-based formatter, passing it the test |
| /// directory followed by [args]. |
| Future<TestProcess> runCommandOnDir([List<String>? args]) { |
| return runCommand(['.', ...?args]); |
| } |
| |
| /// Run tests defined in "*.unit" and "*.stmt" files inside directory [path]. |
| Future<void> testDirectory(String path, {Iterable<StyleFix>? fixes}) async { |
| for (var test in await TestFile.listDirectory(path)) { |
| _testFile(test, fixes); |
| } |
| } |
| |
| Future<void> testFile(String path, {Iterable<StyleFix>? fixes}) async { |
| _testFile(await TestFile.read(path), fixes); |
| } |
| |
| /// Format all of the benchmarks and ensure they produce their expected outputs. |
| Future<void> testBenchmarks({required bool useTallStyle}) async { |
| var benchmarks = Benchmark.findAll(await findPackageDirectory()); |
| |
| group('Benchmarks', () { |
| for (var benchmark in benchmarks) { |
| test(benchmark.name, () { |
| var formatter = DartFormatter( |
| pageWidth: benchmark.pageWidth, |
| experimentFlags: useTallStyle |
| ? const ['inline-class', 'macros', tallStyleExperimentFlag] |
| : const ['inline-class', 'macros']); |
| |
| var actual = formatter.formatSource(SourceCode(benchmark.input)); |
| |
| // The test files always put a newline at the end of the expectation. |
| // Statements from the formatter (correctly) don't have that, so add |
| // one to line up with the expected result. |
| var actualText = actual.text; |
| if (!benchmark.isCompilationUnit) actualText += '\n'; |
| |
| var expected = |
| useTallStyle ? benchmark.tallOutput : benchmark.shortOutput; |
| |
| // Fail with an explicit message because it's easier to read than |
| // the matcher output. |
| if (actualText != expected) { |
| fail('Formatting did not match expectation. Expected:\n' |
| '$expected\nActual:\n$actualText'); |
| } |
| }); |
| } |
| }); |
| } |
| |
| void _testFile(TestFile testFile, Iterable<StyleFix>? baseFixes) { |
| var useTallStyle = |
| testFile.path.startsWith('tall/') || testFile.path.startsWith('tall\\'); |
| |
| group(testFile.path, () { |
| for (var formatTest in testFile.tests) { |
| test(formatTest.label, () { |
| var fixes = [...?baseFixes, ...formatTest.fixes]; |
| |
| if (useTallStyle && fixes.isNotEmpty) { |
| fail('Test error: Tall style does not support applying fixes.'); |
| } |
| |
| var formatter = DartFormatter( |
| pageWidth: testFile.pageWidth, |
| indent: formatTest.leadingIndent, |
| fixes: fixes, |
| experimentFlags: useTallStyle |
| ? const ['inline-class', 'macros', tallStyleExperimentFlag] |
| : const ['inline-class', 'macros']); |
| |
| var actual = formatter.formatSource(formatTest.input); |
| |
| // The test files always put a newline at the end of the expectation. |
| // Statements from the formatter (correctly) don't have that, so add |
| // one to line up with the expected result. |
| var actualText = actual.text; |
| if (!testFile.isCompilationUnit) actualText += '\n'; |
| |
| // Fail with an explicit message because it's easier to read than |
| // the matcher output. |
| if (actualText != formatTest.output.text) { |
| fail('Formatting did not match expectation. Expected:\n' |
| '${formatTest.output.text}\nActual:\n$actualText'); |
| } else if (actual.selectionStart != formatTest.output.selectionStart || |
| actual.selectionLength != formatTest.output.selectionLength) { |
| fail('Selection did not match expectation. Expected:\n' |
| '${formatTest.output.textWithSelectionMarkers}\n' |
| 'Actual:\n${actual.textWithSelectionMarkers}'); |
| } |
| |
| expect(actual.selectionStart, equals(formatTest.output.selectionStart)); |
| expect( |
| actual.selectionLength, equals(formatTest.output.selectionLength)); |
| }); |
| } |
| }); |
| } |
| |
| /// Create a test `.dart_tool` directory with a package config for a package |
| /// with [packageName] and language version [major].[minor]. |
| d.DirectoryDescriptor packageConfig(String packageName, int major, int minor) { |
| var config = ''' |
| { |
| "configVersion": 2, |
| "packages": [ |
| { |
| "name": "$packageName", |
| "rootUri": "../", |
| "packageUri": "lib/", |
| "languageVersion": "$major.$minor" |
| } |
| ] |
| }'''; |
| |
| return d.dir('.dart_tool', [d.file('package_config.json', config)]); |
| } |