| // Copyright (c) 2015, 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:io'; |
| import 'package:_fe_analyzer_shared/src/util/filenames.dart'; |
| import 'package:expect/async_helper.dart'; |
| import 'package:expect/expect.dart'; |
| import 'package:compiler/src/commandline_options.dart'; |
| import 'package:compiler/src/io/source_information.dart'; |
| import 'package:compiler/src/js/js_debug.dart'; |
| import 'package:js_ast/js_ast.dart'; |
| import '../helpers/sourcemap_helper.dart'; |
| |
| typedef CodePointWhiteListFunction WhiteListFunction( |
| String configuration, |
| String file, |
| ); |
| |
| typedef bool CodePointWhiteListFunction(CodePoint codePoint); |
| |
| CodePointWhiteListFunction emptyWhiteListFunction(String config, String file) { |
| return emptyWhiteList; |
| } |
| |
| bool emptyWhiteList(CodePoint codePoint) => false; |
| |
| main(List<String> arguments) { |
| test(arguments); |
| } |
| |
| void test( |
| List<String> arguments, { |
| WhiteListFunction whiteListFunction = emptyWhiteListFunction, |
| }) { |
| Set<String> configurations = Set<String>(); |
| Map<String, Uri> tests = <String, Uri>{}; |
| if (!parseArguments(arguments, configurations, tests)) { |
| return; |
| } |
| |
| asyncTest(() async { |
| bool errorsFound = false; |
| for (String file in tests.keys) { |
| print('==$file========================================================='); |
| for (String config in configurations) { |
| List<String> options = TEST_CONFIGURATIONS[config]!; |
| print('---$config----------------------------------------------------'); |
| Uri uri = tests[file]!; |
| TestResult result = await runTests(config, file, uri, options); |
| if (result.missingCodePointsMap.isNotEmpty) { |
| errorsFound = result.printMissingCodePoints( |
| whiteListFunction(config, file), |
| ); |
| } |
| if (result.multipleNodesMap.isNotEmpty) { |
| result.printMultipleNodes(); |
| errorsFound = true; |
| } |
| if (result.multipleOffsetsMap.isNotEmpty) { |
| result.printMultipleOffsets(); |
| errorsFound = true; |
| } |
| } |
| } |
| Expect.isFalse( |
| errorsFound, |
| "Errors found. " |
| "Run the test with a URI option, " |
| "`source_mapping_test_viewer [--out=<uri>] [configs] [tests]`, to " |
| "create a html visualization of the missing code points.", |
| ); |
| }); |
| } |
| |
| bool parseArguments( |
| List<String> arguments, |
| Set<String> configurations, |
| Map<String, Uri> tests, { |
| bool measure = false, |
| }) { |
| Set<String>? extra = arguments.contains('--file') ? Set<String>() : null; |
| |
| for (String argument in arguments) { |
| if (!parseArgument(argument, configurations, tests, extra)) { |
| return false; |
| } |
| } |
| |
| if (configurations.isEmpty) { |
| configurations.addAll(TEST_CONFIGURATIONS.keys); |
| if (!measure) { |
| configurations.remove('old'); |
| } |
| } |
| if (extra != null) { |
| for (String file in extra) { |
| Uri uri = Uri.base.resolve(nativeToUriPath(file)); |
| tests[uri.pathSegments.last] = uri; |
| } |
| } |
| if (tests.isEmpty) { |
| tests.addAll(TEST_FILES); |
| } |
| if (arguments.contains('--exclude')) { |
| List<String> filesToRemove = List<String>.from(tests.keys); |
| tests.clear(); |
| tests.addAll(TEST_FILES); |
| filesToRemove.forEach(tests.remove); |
| } |
| return true; |
| } |
| |
| /// Parse [argument] for a valid configuration or test-file option. |
| /// |
| /// On success, the configuration name is added to [configurations] or the |
| /// test-file name is added to [testFiles], and `true` is returned. |
| /// On failure, a message is printed and `false` is returned. |
| /// |
| /// Unmatching arguments are added to [files] is provided. |
| bool parseArgument( |
| String argument, |
| Set<String> configurations, |
| Map<String, Uri> tests, |
| Set<String>? extra, |
| ) { |
| if (argument.startsWith('-')) { |
| // Skip options. |
| return true; |
| } else if (TEST_CONFIGURATIONS.containsKey(argument)) { |
| configurations.add(argument); |
| } else if (TEST_FILES.containsKey(argument)) { |
| tests[argument] = TEST_FILES[argument]!; |
| } else if (extra != null) { |
| extra.add(argument); |
| } else { |
| print( |
| "Unknown configuration or test file '$argument'. " |
| "Must be one of '${TEST_CONFIGURATIONS.keys.join("', '")}' or " |
| "'${TEST_FILES.keys.join("', '")}'.", |
| ); |
| return false; |
| } |
| return true; |
| } |
| |
| const Map<String, List<String>> TEST_CONFIGURATIONS = const { |
| 'kernel': const [], |
| }; |
| |
| final Map<String, Uri> TEST_FILES = _computeTestFiles(); |
| |
| Map<String, Uri> _computeTestFiles() { |
| Map<String, Uri> map = <String, Uri>{}; |
| Directory dataDir = Directory.fromUri( |
| Uri.base.resolve('pkg/compiler/test/sourcemaps/data/'), |
| ); |
| for (FileSystemEntity file in dataDir.listSync()) { |
| Uri uri = file.uri; |
| map[uri.pathSegments.last] = uri; |
| } |
| return map; |
| } |
| |
| Future<TestResult> runTests( |
| String config, |
| String filename, |
| Uri uri, |
| List<String> options, { |
| bool verbose = true, |
| }) async { |
| SourceMapProcessor processor = SourceMapProcessor(uri); |
| SourceMaps sourceMaps = await processor.process([ |
| '--csp', |
| Flags.disableInlining, |
| ...options, |
| ], verbose: verbose); |
| TestResult result = TestResult(config, filename, processor); |
| for (SourceMapInfo info in sourceMaps.elementSourceMapInfos.values) { |
| if (info.element!.library.canonicalUri.isScheme('dart')) continue; |
| result.userInfoList.add(info); |
| Iterable<CodePoint> missingCodePoints = info.codePoints.where( |
| (c) => c.isMissing, |
| ); |
| if (missingCodePoints.isNotEmpty) { |
| result.missingCodePointsMap[info] = missingCodePoints; |
| } |
| Map<int, Set<SourceLocation>> offsetToLocationsMap = |
| <int, Set<SourceLocation>>{}; |
| for (Node node in info.nodeMap.nodes) { |
| info.nodeMap[node]!.forEach(( |
| int targetOffset, |
| List<SourceLocation> sourceLocations, |
| ) { |
| if (sourceLocations.length > 1) { |
| Map<Node, List<SourceLocation>> multipleMap = result.multipleNodesMap |
| .putIfAbsent(info, () => <Node, List<SourceLocation>>{}); |
| multipleMap[node] = sourceLocations; |
| } else { |
| offsetToLocationsMap |
| .putIfAbsent(targetOffset, () => Set<SourceLocation>()) |
| .addAll(sourceLocations); |
| } |
| }); |
| } |
| offsetToLocationsMap.forEach(( |
| int targetOffset, |
| Set<SourceLocation> sourceLocations, |
| ) { |
| if (sourceLocations.length > 1) { |
| Map<int, Set<SourceLocation>> multipleMap = result.multipleOffsetsMap |
| .putIfAbsent(info, () => <int, Set<SourceLocation>>{}); |
| multipleMap[targetOffset] = sourceLocations; |
| } |
| }); |
| } |
| return result; |
| } |
| |
| class TestResult { |
| final String config; |
| final String file; |
| final SourceMapProcessor processor; |
| List<SourceMapInfo> userInfoList = <SourceMapInfo>[]; |
| Map<SourceMapInfo, Iterable<CodePoint>> missingCodePointsMap = |
| <SourceMapInfo, Iterable<CodePoint>>{}; |
| |
| /// For each [SourceMapInfo] a map from JS node to multiple source locations |
| /// associated with the node. |
| Map<SourceMapInfo, Map<Node, List<SourceLocation>>> multipleNodesMap = |
| <SourceMapInfo, Map<Node, List<SourceLocation>>>{}; |
| |
| /// For each [SourceMapInfo] a map from JS offset to multiple source locations |
| /// associated with the offset. |
| Map<SourceMapInfo, Map<int, Set<SourceLocation>>> multipleOffsetsMap = |
| <SourceMapInfo, Map<int, Set<SourceLocation>>>{}; |
| |
| TestResult(this.config, this.file, this.processor); |
| |
| bool printMissingCodePoints([ |
| CodePointWhiteListFunction codePointWhiteList = emptyWhiteList, |
| ]) { |
| bool allWhiteListed = true; |
| missingCodePointsMap.forEach((info, missingCodePoints) { |
| print( |
| "Missing code points for ${info.element} in '$file' " |
| "in config '$config':", |
| ); |
| for (CodePoint codePoint in missingCodePoints) { |
| if (codePointWhiteList(codePoint)) { |
| print(" $codePoint (white-listed)"); |
| } else { |
| print(" $codePoint"); |
| allWhiteListed = false; |
| } |
| } |
| }); |
| return !allWhiteListed; |
| } |
| |
| void printMultipleNodes() { |
| multipleNodesMap.forEach((info, multipleMap) { |
| multipleMap.forEach((node, sourceLocations) { |
| print( |
| 'Multiple source locations:\n ${sourceLocations.join('\n ')}\n' |
| 'for `${nodeToString(node)}` in ${info.element} in ' |
| '$file.', |
| ); |
| }); |
| }); |
| } |
| |
| void printMultipleOffsets() { |
| multipleOffsetsMap.forEach((info, multipleMap) { |
| multipleMap.forEach((targetOffset, sourceLocations) { |
| print( |
| 'Multiple source locations:\n ${sourceLocations.join('\n ')}\n' |
| 'for offset $targetOffset in ${info.element} in $file.', |
| ); |
| }); |
| }); |
| } |
| } |