|  | // Copyright (c) 2017, 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:convert" show utf8; | 
|  | import 'dart:io' show File, Platform; | 
|  | import "dart:typed_data" show Uint8List; | 
|  |  | 
|  | import 'package:_fe_analyzer_shared/src/messages/diagnostic_message.dart' | 
|  | show DiagnosticMessage, getMessageCodeObject; | 
|  | import 'package:_fe_analyzer_shared/src/messages/severity.dart' | 
|  | show Severity, severityEnumValues; | 
|  | import 'package:front_end/src/api_prototype/compiler_options.dart' | 
|  | show CompilerOptions, parseExperimentalArguments, parseExperimentalFlags; | 
|  | import 'package:front_end/src/api_prototype/experimental_flags.dart' | 
|  | show ExperimentalFlag, defaultExperimentalFlags; | 
|  | import 'package:front_end/src/api_prototype/memory_file_system.dart' | 
|  | show MemoryFileSystem; | 
|  | import 'package:front_end/src/base/command_line_reporting.dart' | 
|  | as command_line_reporting; | 
|  | import 'package:front_end/src/base/hybrid_file_system.dart' | 
|  | show HybridFileSystem; | 
|  | import 'package:front_end/src/compute_platform_binaries_location.dart' | 
|  | show computePlatformBinariesLocation; | 
|  | import 'package:kernel/ast.dart' show Location, Source; | 
|  | import "package:kernel/target/targets.dart" show TargetFlags; | 
|  | import "package:testing/testing.dart" | 
|  | show Chain, ChainContext, Expectation, Result, Step, TestDescription; | 
|  | import "package:vm/modular/target/vm.dart" show VmTarget; | 
|  | import "package:yaml/yaml.dart" show YamlList, YamlMap, YamlNode, loadYamlNode; | 
|  |  | 
|  | import "../tool/entry_points.dart" show BatchCompiler; | 
|  | import 'spell_checking_utils.dart' as spell; | 
|  | import 'utils/suite_utils.dart' show internalMain; | 
|  |  | 
|  | class MessageTestDescription extends TestDescription { | 
|  | @override | 
|  | final Uri uri; | 
|  |  | 
|  | @override | 
|  | final String shortName; | 
|  |  | 
|  | final String name; | 
|  |  | 
|  | final YamlMap data; | 
|  |  | 
|  | final Example? example; | 
|  |  | 
|  | final String? problem; | 
|  |  | 
|  | MessageTestDescription(this.uri, this.shortName, this.name, this.data, | 
|  | this.example, this.problem); | 
|  | } | 
|  |  | 
|  | class MessageTestSuite extends ChainContext { | 
|  | @override | 
|  | final List<Step> steps = const <Step>[ | 
|  | const Validate(), | 
|  | const Compile(), | 
|  | ]; | 
|  |  | 
|  | final MemoryFileSystem fileSystem; | 
|  |  | 
|  | final BatchCompiler compiler; | 
|  |  | 
|  | final bool fastOnly; | 
|  | final bool interactive; | 
|  |  | 
|  | final Map<String, List<String>?> reportedWordsAndAlternatives = {}; | 
|  | final Set<String> reportedWordsDenylisted = {}; | 
|  |  | 
|  | @override | 
|  | Future<void> postRun() { | 
|  | String dartPath = Platform.resolvedExecutable; | 
|  | Uri suiteUri = | 
|  | spell.repoDir.resolve("pkg/front_end/test/messages_suite.dart"); | 
|  | File suiteFile = new File.fromUri(suiteUri).absolute; | 
|  | if (!suiteFile.existsSync()) { | 
|  | throw "Specified suite path is invalid."; | 
|  | } | 
|  | String suitePath = suiteFile.path; | 
|  | spell.spellSummarizeAndInteractiveMode( | 
|  | reportedWordsAndAlternatives, | 
|  | reportedWordsDenylisted, | 
|  | [spell.Dictionaries.cfeMessages], | 
|  | interactive, | 
|  | '"$dartPath" "$suitePath" -DfastOnly=true -Dinteractive=true'); | 
|  | return new Future.value(); | 
|  | } | 
|  |  | 
|  | MessageTestSuite(this.fastOnly, this.interactive) | 
|  | : fileSystem = new MemoryFileSystem(Uri.parse("org-dartlang-cfe:///")), | 
|  | compiler = new BatchCompiler(null); | 
|  |  | 
|  | @override | 
|  | Set<Expectation> processExpectedOutcomes( | 
|  | Set<Expectation> outcomes, TestDescription description) { | 
|  | if (description.shortName.contains("/spelling")) { | 
|  | return {Expectation.pass}; | 
|  | } | 
|  | return outcomes; | 
|  | } | 
|  |  | 
|  | /// Convert all the examples found in `messages.yaml` to a test | 
|  | /// description. In addition, create a test description for each kind of | 
|  | /// problem that a message can have. This problem will then be reported as a | 
|  | /// failure by the [Validate] step that can be suppressed via the status | 
|  | /// file. | 
|  | @override | 
|  | Future<List<MessageTestDescription>> list(Chain suite) { | 
|  | List<MessageTestDescription> result = []; | 
|  | Uri uri = suite.root.resolve("messages.yaml"); | 
|  | File file = new File.fromUri(uri); | 
|  | String fileContent = file.readAsStringSync(); | 
|  | YamlMap messages = loadYamlNode(fileContent, sourceUrl: uri) as YamlMap; | 
|  | for (String name in messages.keys) { | 
|  | YamlMap messageNode = messages.nodes[name] as YamlMap; | 
|  | dynamic message = messageNode.value; | 
|  | if (message is String) continue; | 
|  |  | 
|  | List<String> unknownKeys = <String>[]; | 
|  | bool exampleAllowMoreCodes = false; | 
|  | List<Example> examples = <Example>[]; | 
|  | String? externalTest; | 
|  | bool frontendInternal = false; | 
|  | List<String>? analyzerCodes; | 
|  | Severity? severity; | 
|  | YamlNode? badSeverity; | 
|  | YamlNode? unnecessarySeverity; | 
|  | List<String> badHasPublishedDocsValue = <String>[]; | 
|  | List<String>? spellingMessages; | 
|  | const String spellingPostMessage = "\nIf the word(s) look okay, update " | 
|  | "'spell_checking_list_messages.txt' or " | 
|  | "'spell_checking_list_common.txt'."; | 
|  | Map<ExperimentalFlag, bool>? experimentalFlags; | 
|  |  | 
|  | Source? source; | 
|  | List<String> formatSpellingMistakes(spell.SpellingResult spellResult, | 
|  | int offset, String message, String messageForDenyListed) { | 
|  | if (source == null) { | 
|  | Uint8List bytes = file.readAsBytesSync(); | 
|  | List<int> lineStarts = <int>[]; | 
|  | int indexOf = 0; | 
|  | while (indexOf >= 0) { | 
|  | lineStarts.add(indexOf); | 
|  | indexOf = bytes.indexOf(10, indexOf + 1); | 
|  | } | 
|  | lineStarts.add(bytes.length); | 
|  | source = new Source(lineStarts, bytes, uri, uri); | 
|  | } | 
|  | List<String> result = <String>[]; | 
|  | for (int i = 0; i < spellResult.misspelledWords!.length; i++) { | 
|  | Location location = source! | 
|  | .getLocation(uri, offset + spellResult.misspelledWordsOffset![i]); | 
|  | bool denylisted = spellResult.misspelledWordsDenylisted![i]; | 
|  | String messageToUse = message; | 
|  | if (denylisted) { | 
|  | messageToUse = messageForDenyListed; | 
|  | reportedWordsDenylisted.add(spellResult.misspelledWords![i]); | 
|  | } else { | 
|  | reportedWordsAndAlternatives[spellResult.misspelledWords![i]] = | 
|  | spellResult.misspelledWordsAlternatives![i]; | 
|  | } | 
|  | result.add(command_line_reporting.formatErrorMessage( | 
|  | source!.getTextLine(location.line), | 
|  | location, | 
|  | spellResult.misspelledWords![i].length, | 
|  | relativize(uri), | 
|  | "$messageToUse: '${spellResult.misspelledWords![i]}'.")); | 
|  | } | 
|  | return result; | 
|  | } | 
|  |  | 
|  | for (String key in message.keys) { | 
|  | YamlNode node = message.nodes[key]; | 
|  | var value = node.value; | 
|  | // When positions matter, use node.span.text. | 
|  | // When using node.span.text, replace r"\n" with "\n\n" to replace two | 
|  | // characters with two characters without actually having the string | 
|  | // "backslash n". | 
|  | switch (key) { | 
|  | case "problemMessage": | 
|  | spell.SpellingResult spellingResult = spell.spellcheckString( | 
|  | node.span.text.replaceAll(r"\n", "\n\n"), | 
|  | dictionaries: const [ | 
|  | spell.Dictionaries.common, | 
|  | spell.Dictionaries.cfeMessages | 
|  | ]); | 
|  | if (spellingResult.misspelledWords != null) { | 
|  | spellingMessages ??= <String>[]; | 
|  | spellingMessages.addAll(formatSpellingMistakes( | 
|  | spellingResult, | 
|  | node.span.start.offset, | 
|  | "problemMessage has the following word that is " | 
|  | "not in our dictionary", | 
|  | "problemMessage has the following word that is " | 
|  | "on our deny-list")); | 
|  | } | 
|  | break; | 
|  |  | 
|  | case "correctionMessage": | 
|  | spell.SpellingResult spellingResult = spell.spellcheckString( | 
|  | node.span.text.replaceAll(r"\n", "\n\n"), | 
|  | dictionaries: const [ | 
|  | spell.Dictionaries.common, | 
|  | spell.Dictionaries.cfeMessages | 
|  | ]); | 
|  | if (spellingResult.misspelledWords != null) { | 
|  | spellingMessages ??= <String>[]; | 
|  | spellingMessages.addAll(formatSpellingMistakes( | 
|  | spellingResult, | 
|  | node.span.start.offset, | 
|  | "correctionMessage has the following word that is " | 
|  | "not in our dictionary", | 
|  | "correctionMessage has the following word that is " | 
|  | "on our deny-list")); | 
|  | } | 
|  | break; | 
|  |  | 
|  | case "severity": | 
|  | severity = severityEnumValues[value]; | 
|  | if (severity == null) { | 
|  | badSeverity = node; | 
|  | } else if (severity == Severity.error) { | 
|  | unnecessarySeverity = node; | 
|  | } | 
|  | break; | 
|  |  | 
|  | case "frontendInternal": | 
|  | frontendInternal = value; | 
|  | break; | 
|  |  | 
|  | case "analyzerCode": | 
|  | analyzerCodes = value is String | 
|  | ? <String>[value] | 
|  | : new List<String>.from(value); | 
|  | break; | 
|  |  | 
|  | case "sharedName": | 
|  | if (value is! String) { | 
|  | throw new ArgumentError('sharedName should be a string: $value.'); | 
|  | } | 
|  | break; | 
|  |  | 
|  | case "exampleAllowMoreCodes": | 
|  | exampleAllowMoreCodes = value; | 
|  | break; | 
|  |  | 
|  | case "bytes": | 
|  | YamlList list = node as YamlList; | 
|  | if (list.first is List) { | 
|  | for (YamlNode bytes in list.nodes) { | 
|  | int i = 0; | 
|  | examples.add( | 
|  | new BytesExample("bytes${++i}", name, bytes as YamlList)); | 
|  | } | 
|  | } else { | 
|  | examples.add(new BytesExample("bytes", name, list)); | 
|  | } | 
|  | break; | 
|  |  | 
|  | case "declaration": | 
|  | if (node is YamlList) { | 
|  | int i = 0; | 
|  | for (YamlNode declaration in node.nodes) { | 
|  | examples.add(new DeclarationExample( | 
|  | "declaration${++i}", name, declaration)); | 
|  | } | 
|  | } else { | 
|  | examples.add(new DeclarationExample("declaration", name, node)); | 
|  | } | 
|  | break; | 
|  |  | 
|  | case "expression": | 
|  | if (node is YamlList) { | 
|  | int i = 0; | 
|  | for (YamlNode expression in node.nodes) { | 
|  | examples.add(new ExpressionExample( | 
|  | "expression${++i}", name, expression)); | 
|  | } | 
|  | } else { | 
|  | examples.add(new ExpressionExample("expression", name, node)); | 
|  | } | 
|  | break; | 
|  |  | 
|  | case "script": | 
|  | if (node is YamlList) { | 
|  | int i = 0; | 
|  | for (YamlNode script in node.nodes) { | 
|  | examples | 
|  | .add(new ScriptExample("script${++i}", name, script, this)); | 
|  | } | 
|  | } else { | 
|  | examples.add(new ScriptExample("script", name, node, this)); | 
|  | } | 
|  | break; | 
|  |  | 
|  | case "statement": | 
|  | if (node is YamlList) { | 
|  | int i = 0; | 
|  | for (YamlNode statement in node.nodes) { | 
|  | examples.add( | 
|  | new StatementExample("statement${++i}", name, statement)); | 
|  | } | 
|  | } else { | 
|  | examples.add(new StatementExample("statement", name, node)); | 
|  | } | 
|  | break; | 
|  |  | 
|  | case "external": | 
|  | externalTest = node.value; | 
|  | break; | 
|  |  | 
|  | case "index": | 
|  | // index is validated during generation | 
|  | break; | 
|  |  | 
|  | case "hasPublishedDocs": | 
|  | if (value != true) { | 
|  | badHasPublishedDocsValue.add(name); | 
|  | } | 
|  | break; | 
|  |  | 
|  | case "experiments": | 
|  | if (value is String) { | 
|  | experimentalFlags = parseExperimentalFlags( | 
|  | parseExperimentalArguments(value.split(',')), | 
|  | onError: (message) => throw new ArgumentError(message)); | 
|  | } else { | 
|  | throw new ArgumentError("Unknown experiments value: $value."); | 
|  | } | 
|  | break; | 
|  |  | 
|  | case "documentation": | 
|  | if (value is! String) { | 
|  | throw new ArgumentError( | 
|  | 'documentation should be a string: $value.'); | 
|  | } | 
|  | break; | 
|  |  | 
|  | case "comment": | 
|  | if (value is! String) { | 
|  | throw new ArgumentError('comment should be a string: $value.'); | 
|  | } | 
|  | break; | 
|  |  | 
|  | default: | 
|  | unknownKeys.add(key); | 
|  | } | 
|  | } | 
|  |  | 
|  | if (exampleAllowMoreCodes) { | 
|  | // Update all examples. | 
|  | for (Example example in examples) { | 
|  | example.allowMoreCodes = exampleAllowMoreCodes; | 
|  | } | 
|  | } | 
|  | for (Example example in examples) { | 
|  | example.experimentalFlags = | 
|  | experimentalFlags ?? defaultExperimentalFlags; | 
|  | } | 
|  |  | 
|  | MessageTestDescription createDescription( | 
|  | String subName, Example? example, String? problem, | 
|  | {location}) { | 
|  | String shortName = "$name/$subName"; | 
|  | if (problem != null) { | 
|  | String filename = relativize(uri); | 
|  | location ??= message.span.start; | 
|  | int line = location.line + 1; | 
|  | int column = location.column; | 
|  | problem = "$filename:$line:$column: error:\n$problem"; | 
|  | } | 
|  | return new MessageTestDescription(uri.resolve("#$shortName"), shortName, | 
|  | name, messageNode, example, problem); | 
|  | } | 
|  |  | 
|  | if (!fastOnly) { | 
|  | for (Example example in examples) { | 
|  | result.add(createDescription(example.name, example, null)); | 
|  | } | 
|  | // "Wrap" example as a part. | 
|  | for (Example example in examples) { | 
|  | result.add(createDescription( | 
|  | "part_wrapped_${example.name}", | 
|  | new PartWrapExample("part_wrapped_${example.name}", name, | 
|  | exampleAllowMoreCodes, example), | 
|  | null)); | 
|  | } | 
|  | } | 
|  |  | 
|  | result.add(createDescription( | 
|  | "knownKeys", | 
|  | null, | 
|  | unknownKeys.isNotEmpty | 
|  | ? "Unknown keys: ${unknownKeys.join(' ')}." | 
|  | : null)); | 
|  |  | 
|  | result.add(createDescription( | 
|  | 'hasPublishedDocs', | 
|  | null, | 
|  | badHasPublishedDocsValue.isNotEmpty | 
|  | ? "Bad hasPublishedDocs value (only 'true' supported) in:" | 
|  | " ${badHasPublishedDocsValue.join(', ')}" | 
|  | : null)); | 
|  |  | 
|  | result.add(createDescription( | 
|  | "severity", | 
|  | null, | 
|  | badSeverity != null | 
|  | ? "Unknown severity: '${badSeverity.value}'." | 
|  | : null, | 
|  | location: badSeverity?.span.start)); | 
|  |  | 
|  | result.add(createDescription( | 
|  | "unnecessarySeverity", | 
|  | null, | 
|  | unnecessarySeverity != null | 
|  | ? "The 'ERROR' severity is the default and not necessary." | 
|  | : null, | 
|  | location: unnecessarySeverity?.span.start)); | 
|  |  | 
|  | result.add(createDescription( | 
|  | "spelling", | 
|  | null, | 
|  | spellingMessages != null | 
|  | ? spellingMessages.join("\n") + spellingPostMessage | 
|  | : null)); | 
|  |  | 
|  | bool exampleAndAnalyzerCodeRequired = severity != Severity.context && | 
|  | severity != Severity.internalProblem && | 
|  | severity != Severity.ignored; | 
|  |  | 
|  | result.add(createDescription( | 
|  | "externalExample", | 
|  | null, | 
|  | exampleAndAnalyzerCodeRequired && | 
|  | externalTest != null && | 
|  | !(new File.fromUri(suite.root.resolve(externalTest)) | 
|  | .existsSync()) | 
|  | ? "Given external example for $name points to a nonexisting file " | 
|  | "(${suite.root.resolve(externalTest)})." | 
|  | : null)); | 
|  |  | 
|  | result.add(createDescription( | 
|  | "example", | 
|  | null, | 
|  | exampleAndAnalyzerCodeRequired && | 
|  | examples.isEmpty && | 
|  | externalTest == null | 
|  | ? "No example for $name, please add at least one example." | 
|  | : null)); | 
|  |  | 
|  | result.add(createDescription( | 
|  | "analyzerCode", | 
|  | null, | 
|  | exampleAndAnalyzerCodeRequired && | 
|  | !frontendInternal && | 
|  | analyzerCodes == null | 
|  | ? "No analyzer code for $name." | 
|  | "\nTry running" | 
|  | " <BUILDDIR>/dart-sdk/bin/dartanalyzer --format=machine" | 
|  | " on an example to find the code." | 
|  | " The code is printed just before the file name." | 
|  | : null)); | 
|  | } | 
|  | return Future.value(result); | 
|  | } | 
|  |  | 
|  | String formatProblems( | 
|  | String message, Example example, List<DiagnosticMessage> messages) { | 
|  | var span = example.node.span; | 
|  | StringBuffer buffer = new StringBuffer(); | 
|  | buffer | 
|  | ..write(relativize(span.sourceUrl!)) | 
|  | ..write(":") | 
|  | ..write(span.start.line + 1) | 
|  | ..write(":") | 
|  | ..write(span.start.column) | 
|  | ..write(": error: ") | 
|  | ..write(message); | 
|  | buffer.write("\n${span.text}"); | 
|  | for (DiagnosticMessage message in messages) { | 
|  | buffer.write("\nCode: ${getMessageCodeObject(message)!.name}"); | 
|  | buffer.write("\n  > "); | 
|  | buffer.write( | 
|  | message.plainTextFormatted.join("\n").replaceAll("\n", "\n  > ")); | 
|  | } | 
|  |  | 
|  | return "$buffer"; | 
|  | } | 
|  | } | 
|  |  | 
|  | abstract class Example { | 
|  | final String name; | 
|  |  | 
|  | final String expectedCode; | 
|  |  | 
|  | bool allowMoreCodes = false; | 
|  |  | 
|  | Map<ExperimentalFlag, bool>? experimentalFlags; | 
|  |  | 
|  | Example(this.name, this.expectedCode); | 
|  |  | 
|  | YamlNode get node; | 
|  |  | 
|  | Map<String, Script> get scripts; | 
|  |  | 
|  | String get mainFilename => "main.dart"; | 
|  | } | 
|  |  | 
|  | class BytesExample extends Example { | 
|  | @override | 
|  | final YamlList node; | 
|  |  | 
|  | final Uint8List bytes; | 
|  |  | 
|  | BytesExample(String name, String code, this.node) | 
|  | : bytes = new Uint8List.fromList(node.cast<int>()), | 
|  | super(name, code); | 
|  |  | 
|  | @override | 
|  | Map<String, Script> get scripts { | 
|  | return {mainFilename: new Script(bytes, '', null)}; | 
|  | } | 
|  | } | 
|  |  | 
|  | class DeclarationExample extends Example { | 
|  | @override | 
|  | final YamlNode node; | 
|  |  | 
|  | final String declaration; | 
|  |  | 
|  | DeclarationExample(String name, String code, this.node) | 
|  | : declaration = node.value, | 
|  | super(name, code); | 
|  |  | 
|  | @override | 
|  | Map<String, Script> get scripts { | 
|  | return { | 
|  | mainFilename: new Script.fromSource(""" | 
|  | $declaration | 
|  |  | 
|  | main() { | 
|  | } | 
|  | """) | 
|  | }; | 
|  | } | 
|  | } | 
|  |  | 
|  | class StatementExample extends Example { | 
|  | @override | 
|  | final YamlNode node; | 
|  |  | 
|  | final String statement; | 
|  |  | 
|  | StatementExample(String name, String code, this.node) | 
|  | : statement = node.value, | 
|  | super(name, code); | 
|  |  | 
|  | @override | 
|  | Map<String, Script> get scripts { | 
|  | return { | 
|  | mainFilename: new Script.fromSource(""" | 
|  | main() { | 
|  | $statement | 
|  | } | 
|  | """) | 
|  | }; | 
|  | } | 
|  | } | 
|  |  | 
|  | class ExpressionExample extends Example { | 
|  | @override | 
|  | final YamlNode node; | 
|  |  | 
|  | final String expression; | 
|  |  | 
|  | ExpressionExample(String name, String code, this.node) | 
|  | : expression = node.value, | 
|  | super(name, code); | 
|  |  | 
|  | @override | 
|  | Map<String, Script> get scripts { | 
|  | return { | 
|  | mainFilename: new Script.fromSource(""" | 
|  | main() { | 
|  | $expression; | 
|  | } | 
|  | """) | 
|  | }; | 
|  | } | 
|  | } | 
|  |  | 
|  | class ScriptExample extends Example { | 
|  | @override | 
|  | final YamlNode node; | 
|  |  | 
|  | final Object script; | 
|  |  | 
|  | ScriptExample(String name, String code, this.node, MessageTestSuite suite) | 
|  | : script = node.value, | 
|  | super(name, code) { | 
|  | if (script is! String && script is! Map) { | 
|  | throw suite.formatProblems( | 
|  | "A script must be either a String or a Map in $code:", | 
|  | this, <DiagnosticMessage>[]); | 
|  | } | 
|  | } | 
|  |  | 
|  | @override | 
|  | Map<String, Script> get scripts { | 
|  | Object script = this.script; | 
|  | if (script is Map) { | 
|  | Map<String, Script> scriptFiles = <String, Script>{}; | 
|  | script.forEach((fileName, value) { | 
|  | scriptFiles[fileName] = new Script.fromSource(value); | 
|  | print("$fileName => $value\n\n======\n\n"); | 
|  | }); | 
|  | return scriptFiles; | 
|  | } else { | 
|  | return {mainFilename: new Script.fromSource(script as String)}; | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | class PartWrapExample extends Example { | 
|  | final Example example; | 
|  | @override | 
|  | final bool allowMoreCodes; | 
|  |  | 
|  | PartWrapExample(String name, String code, this.allowMoreCodes, this.example) | 
|  | : super(name, code) { | 
|  | experimentalFlags = example.experimentalFlags; | 
|  | } | 
|  |  | 
|  | @override | 
|  | String get mainFilename => "main_wrapped.dart"; | 
|  |  | 
|  | @override | 
|  | Map<String, Script> get scripts { | 
|  | Map<String, Script> wrapped = example.scripts; | 
|  |  | 
|  | Map<String, Script> scriptFiles = <String, Script>{}; | 
|  | scriptFiles.addAll(wrapped); | 
|  |  | 
|  | // Create a new main file | 
|  | // TODO: Technically we should find a un-used name. | 
|  | if (scriptFiles.containsKey(mainFilename)) { | 
|  | throw "Framework failure: " | 
|  | "Wanted to create wrapper file, but the file already exists!"; | 
|  | } | 
|  | Script originalMainScript = scriptFiles[example.mainFilename]!; | 
|  | String preamble = originalMainScript.preamble; | 
|  | scriptFiles[mainFilename] = new Script.fromSource(""" | 
|  | ${preamble}part "${example.mainFilename}"; | 
|  | """); | 
|  |  | 
|  | // Modify the original main file to be part of the wrapper and add lots of | 
|  | // gunk so every actual position in the file is not a valid position in the | 
|  | // wrapper. | 
|  | String? originalMainSource = originalMainScript.sourceWithoutPreamble; | 
|  | String partPrefix = """ | 
|  | ${preamble}part of "${mainFilename}"; | 
|  | // La la la la la la la la la la la la la. | 
|  | // La la la la la la la la la la la la la. | 
|  | // La la la la la la la la la la la la la. | 
|  | // La la la la la la la la la la la la la. | 
|  | // La la la la la la la la la la la la la. | 
|  |  | 
|  | """; | 
|  | if (originalMainSource != null) { | 
|  | scriptFiles[example.mainFilename] = | 
|  | new Script.fromSource('$partPrefix$originalMainSource'); | 
|  | } else { | 
|  | scriptFiles[example.mainFilename] = new Script( | 
|  | new Uint8List.fromList( | 
|  | utf8.encode(partPrefix) + originalMainScript.bytes), | 
|  | '', | 
|  | null); | 
|  | } | 
|  |  | 
|  | return scriptFiles; | 
|  | } | 
|  |  | 
|  | @override | 
|  | YamlNode get node => example.node; | 
|  | } | 
|  |  | 
|  | class Validate | 
|  | extends Step<MessageTestDescription, Example?, MessageTestSuite> { | 
|  | const Validate(); | 
|  |  | 
|  | @override | 
|  | String get name => "validate"; | 
|  |  | 
|  | @override | 
|  | Future<Result<Example?>> run( | 
|  | MessageTestDescription description, MessageTestSuite suite) { | 
|  | if (description.problem != null) { | 
|  | return new Future.value(fail(null, description.problem)); | 
|  | } else { | 
|  | return new Future.value(pass(description.example)); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | class Compile extends Step<Example?, Null, MessageTestSuite> { | 
|  | const Compile(); | 
|  |  | 
|  | @override | 
|  | String get name => "compile"; | 
|  |  | 
|  | @override | 
|  | Future<Result<Null>> run(Example? example, MessageTestSuite suite) async { | 
|  | if (example == null) return pass(null); | 
|  | String dir = "${example.expectedCode}/${example.name}"; | 
|  | example.scripts.forEach((String fileName, Script script) { | 
|  | Uri uri = suite.fileSystem.currentDirectory.resolve("$dir/$fileName"); | 
|  | suite.fileSystem.entityForUri(uri).writeAsBytesSync(script.bytes); | 
|  | }); | 
|  | Uri main = suite.fileSystem.currentDirectory | 
|  | .resolve("$dir/${example.mainFilename}"); | 
|  | Uri output = | 
|  | suite.fileSystem.currentDirectory.resolve("$dir/main.dart.dill"); | 
|  |  | 
|  | // Setup .dart_tool/package_config.json if it doesn't exist. | 
|  | Uri packageConfigUri = suite.fileSystem.currentDirectory | 
|  | .resolve("$dir/.dart_tool/package_config.json"); | 
|  | if (!await suite.fileSystem.entityForUri(packageConfigUri).exists()) { | 
|  | suite.fileSystem | 
|  | .entityForUri(packageConfigUri) | 
|  | .writeAsStringSync('{"configVersion": 2, "packages": []}'); | 
|  | } | 
|  |  | 
|  | print("Compiling $main"); | 
|  | List<DiagnosticMessage> messages = <DiagnosticMessage>[]; | 
|  |  | 
|  | await suite.compiler.batchCompile( | 
|  | new CompilerOptions() | 
|  | ..sdkSummary = computePlatformBinariesLocation(forceBuildDir: true) | 
|  | .resolve("vm_platform_strong.dill") | 
|  | ..explicitExperimentalFlags = example.experimentalFlags ?? {} | 
|  | ..target = new VmTarget(new TargetFlags()) | 
|  | ..fileSystem = new HybridFileSystem(suite.fileSystem) | 
|  | ..packagesFileUri = packageConfigUri | 
|  | ..onDiagnostic = messages.add | 
|  | ..environmentDefines = const {} | 
|  | ..omitPlatform = true, | 
|  | main, | 
|  | output); | 
|  |  | 
|  | List<DiagnosticMessage> unexpectedMessages = <DiagnosticMessage>[]; | 
|  | if (example.allowMoreCodes) { | 
|  | List<DiagnosticMessage> messagesFiltered = <DiagnosticMessage>[]; | 
|  | for (DiagnosticMessage message in messages) { | 
|  | if (getMessageCodeObject(message)!.name == example.expectedCode) { | 
|  | messagesFiltered.add(message); | 
|  | } | 
|  | } | 
|  | messages = messagesFiltered; | 
|  | } | 
|  | for (DiagnosticMessage message in messages) { | 
|  | if (getMessageCodeObject(message)!.name != example.expectedCode) { | 
|  | unexpectedMessages.add(message); | 
|  | } | 
|  | } | 
|  | if (unexpectedMessages.isEmpty) { | 
|  | switch (messages.length) { | 
|  | case 0: | 
|  | return fail( | 
|  | null, | 
|  | suite.formatProblems("No message reported in ${example.name}:", | 
|  | example, messages)); | 
|  | case 1: | 
|  | return pass(null); | 
|  | default: | 
|  | return fail( | 
|  | null, | 
|  | suite.formatProblems( | 
|  | "Message reported multiple times in ${example.name}:", | 
|  | example, | 
|  | messages)); | 
|  | } | 
|  | } | 
|  | return fail( | 
|  | null, | 
|  | suite.formatProblems( | 
|  | "Too many or unexpected messages (${messages.length}) reported " | 
|  | "in ${example.name}:", | 
|  | example, | 
|  | messages)); | 
|  | } | 
|  | } | 
|  |  | 
|  | Future<MessageTestSuite> createContext( | 
|  | Chain suite, Map<String, String> environment) { | 
|  | final bool fastOnly = environment["fastOnly"] == "true"; | 
|  | final bool interactive = environment["interactive"] == "true"; | 
|  | return new Future.value(new MessageTestSuite(fastOnly, interactive)); | 
|  | } | 
|  |  | 
|  | String relativize(Uri uri) { | 
|  | String base = "${Uri.base}"; | 
|  | String filename = "$uri"; | 
|  | if (filename.startsWith(base)) { | 
|  | return filename.substring(base.length); | 
|  | } else { | 
|  | return filename; | 
|  | } | 
|  | } | 
|  |  | 
|  | class Script { | 
|  | final Uint8List bytes; | 
|  | final String preamble; | 
|  | final String? sourceWithoutPreamble; | 
|  |  | 
|  | Script(this.bytes, this.preamble, this.sourceWithoutPreamble); | 
|  |  | 
|  | factory Script.fromSource(String source) { | 
|  | List<String> lines = source.split('\n'); | 
|  | String firstLine = lines.first; | 
|  | String preamble; | 
|  | String sourceWithoutPreamble; | 
|  | if (firstLine.trim().startsWith('//') && firstLine.contains('@dart=')) { | 
|  | preamble = '$firstLine\n'; | 
|  | sourceWithoutPreamble = lines.skip(1).join('\n'); | 
|  | } else { | 
|  | preamble = ''; | 
|  | sourceWithoutPreamble = source; | 
|  | } | 
|  | return new Script(utf8.encode(source), preamble, sourceWithoutPreamble); | 
|  | } | 
|  | } | 
|  |  | 
|  | Future<void> main([List<String> arguments = const []]) async { | 
|  | await internalMain( | 
|  | createContext, | 
|  | arguments: arguments, | 
|  | displayName: "messages suite", | 
|  | configurationPath: "../testing.json", | 
|  | ); | 
|  | } |