mirror of
https://github.com/dart-lang/sdk
synced 2024-11-02 10:49:00 +00:00
Move the status file parser into its own package.
This required munging a bit of test.dart code too to tease out some dependencies, but the changes are minor. I considered moving all of test.dart out into a package and making the status file library a public one that other packages in the repo could import but this seemed like the less intrusive change. R=bkonyi@google.com Review-Url: https://codereview.chromium.org/2984203002 .
This commit is contained in:
parent
64785f0614
commit
796eb21071
21 changed files with 192 additions and 107 deletions
|
@ -88,6 +88,7 @@ source_map_stack_trace:third_party/pkg/source_map_stack_trace/lib
|
|||
source_maps:third_party/pkg/source_maps/lib
|
||||
source_span:third_party/pkg/source_span/lib
|
||||
stack_trace:third_party/pkg/stack_trace/lib
|
||||
status_file:pkg/status_file/lib
|
||||
stream_channel:third_party/pkg/stream_channel/lib
|
||||
string_scanner:third_party/pkg/string_scanner/lib
|
||||
telemetry:pkg/telemetry/lib
|
||||
|
|
|
@ -226,3 +226,6 @@ analyzer/test/src/task/strong/checker_test: Pass, Slow
|
|||
|
||||
[ $browser ]
|
||||
testing/test/analyze_test: SkipByDesign
|
||||
|
||||
[ $browser || $runtime == flutter ]
|
||||
status_file/*: SkipByDesign # Only meant to run on the standalone VM.
|
||||
|
|
4
pkg/status_file/.packages
Normal file
4
pkg/status_file/.packages
Normal file
|
@ -0,0 +1,4 @@
|
|||
# Generated by pub on 2017-07-24 16:32:37.651832.
|
||||
expect:../expect/lib/
|
||||
path:../../third_party/pkg/path/lib/
|
||||
status_file:lib/
|
16
pkg/status_file/lib/environment.dart
Normal file
16
pkg/status_file/lib/environment.dart
Normal file
|
@ -0,0 +1,16 @@
|
|||
// 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.
|
||||
|
||||
/// An implementation of this defines the variables that are available for use
|
||||
/// inside a status file section header.
|
||||
abstract class Environment {
|
||||
/// Validates that the variable with [name] exists and can be compared
|
||||
/// against [value].
|
||||
///
|
||||
/// If any errors are found, adds them to [errors].
|
||||
void validate(String name, String value, List<String> errors);
|
||||
|
||||
/// Looks up the value of the variable with [name].
|
||||
String lookUp(String name);
|
||||
}
|
|
@ -1,8 +1,8 @@
|
|||
// Copyright (c) 2011, the Dart project authors. Please see the AUTHORS file
|
||||
// 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 'environment.dart';
|
||||
import '../environment.dart';
|
||||
|
||||
/// A parsed Boolean expression AST.
|
||||
abstract class Expression {
|
||||
|
@ -29,7 +29,7 @@ abstract class Expression {
|
|||
/// Ensures that any variable names are known and that any literal values are
|
||||
/// allowed for their corresponding variable. If an invalid variable or value
|
||||
/// is found, adds appropriate error messages to [errors].
|
||||
void validate(List<String> errors);
|
||||
void validate(Environment environment, List<String> errors);
|
||||
|
||||
/// Evaluates the expression where all variables are defined by the given
|
||||
/// [environment].
|
||||
|
@ -83,8 +83,8 @@ class _ComparisonExpression implements Expression {
|
|||
|
||||
_ComparisonExpression(this.left, this.right, this.negate);
|
||||
|
||||
void validate(List<String> errors) {
|
||||
Environment.validate(left.name, right, errors);
|
||||
void validate(Environment environment, List<String> errors) {
|
||||
environment.validate(left.name, right, errors);
|
||||
}
|
||||
|
||||
bool evaluate(Environment environment) {
|
||||
|
@ -117,9 +117,9 @@ class _VariableExpression implements Expression {
|
|||
|
||||
_VariableExpression(this.variable, {this.negate = false});
|
||||
|
||||
void validate(List<String> errors) {
|
||||
void validate(Environment environment, List<String> errors) {
|
||||
// It must be a Boolean, so it should allow either Boolean value.
|
||||
Environment.validate(variable.name, "true", errors);
|
||||
environment.validate(variable.name, "true", errors);
|
||||
}
|
||||
|
||||
bool evaluate(Environment environment) =>
|
||||
|
@ -138,9 +138,9 @@ class _LogicExpression implements Expression {
|
|||
|
||||
_LogicExpression(this.op, this.left, this.right);
|
||||
|
||||
void validate(List<String> errors) {
|
||||
left.validate(errors);
|
||||
right.validate(errors);
|
||||
void validate(Environment environment, List<String> errors) {
|
||||
left.validate(environment, errors);
|
||||
right.validate(environment, errors);
|
||||
}
|
||||
|
||||
bool evaluate(Environment environment) => (op == _Token.and)
|
|
@ -4,10 +4,11 @@
|
|||
|
||||
import 'dart:io';
|
||||
|
||||
import 'package:path/path.dart' as p;
|
||||
|
||||
import 'environment.dart';
|
||||
import 'expectation.dart';
|
||||
import 'path.dart';
|
||||
import 'status_expression.dart';
|
||||
import 'src/expression.dart';
|
||||
|
||||
/// Matches the header that begins a new section, like:
|
||||
///
|
||||
|
@ -46,6 +47,8 @@ class StatusFile {
|
|||
final List<StatusSection> sections = [];
|
||||
|
||||
/// Parses the status file at [_path].
|
||||
///
|
||||
/// Throws a [SyntaxError] if the file could not be parsed.
|
||||
StatusFile.read(this._path) {
|
||||
var lines = new File(_path).readAsLinesSync();
|
||||
|
||||
|
@ -58,14 +61,7 @@ class StatusFile {
|
|||
lineNumber++;
|
||||
|
||||
fail(String message, [List<String> errors]) {
|
||||
print('$message in "$_shortPath" line $lineNumber:\n$line');
|
||||
|
||||
if (errors != null) {
|
||||
for (var error in errors) {
|
||||
print("- ${error.replaceAll('\n', '\n ')}");
|
||||
}
|
||||
}
|
||||
exit(1);
|
||||
throw new SyntaxError(_shortPath, lineNumber, line, message, errors);
|
||||
}
|
||||
|
||||
// Strip off the comment and whitespace.
|
||||
|
@ -85,17 +81,7 @@ class StatusFile {
|
|||
var match = _sectionPattern.firstMatch(source);
|
||||
if (match != null) {
|
||||
try {
|
||||
var condition = Expression.parse(match[1].trim());
|
||||
|
||||
var errors = <String>[];
|
||||
condition.validate(errors);
|
||||
|
||||
if (errors.isNotEmpty) {
|
||||
var s = errors.length > 1 ? "s" : "";
|
||||
fail('Validation error$s', errors);
|
||||
}
|
||||
|
||||
section = new StatusSection(condition);
|
||||
section = new StatusSection(Expression.parse(match[1].trim()));
|
||||
sections.add(section);
|
||||
} on FormatException {
|
||||
fail("Status expression syntax error");
|
||||
|
@ -135,12 +121,31 @@ class StatusFile {
|
|||
}
|
||||
}
|
||||
|
||||
/// Validates that the variables and values used in all of the section
|
||||
/// condition expressions are defined in [environment].
|
||||
///
|
||||
/// Throws a [SyntaxError] on the first found error.
|
||||
void validate(Environment environment) {
|
||||
// TODO(rnystrom): It would be more useful if it reported all of the errors
|
||||
// instead of stopping on the first.
|
||||
for (var section in sections) {
|
||||
if (section._condition == null) continue;
|
||||
|
||||
var errors = <String>[];
|
||||
section._condition.validate(environment, errors);
|
||||
|
||||
if (errors.isNotEmpty) {
|
||||
var s = errors.length > 1 ? "s" : "";
|
||||
throw new SyntaxError(_shortPath, section.lineNumber,
|
||||
"[ ${section._condition} ]", 'Validation error$s', errors);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/// Gets the path to this status file relative to the Dart repo root.
|
||||
String get _shortPath {
|
||||
var repoRoot = new Path(Platform.script
|
||||
.toFilePath(windows: Platform.operatingSystem == "windows"))
|
||||
.join(new Path("../../../../"));
|
||||
return new Path(_path).relativeTo(repoRoot).toString();
|
||||
var repoRoot = p.join(p.dirname(p.fromUri(Platform.script)), "../../../");
|
||||
return p.normalize(p.relative(_path, from: repoRoot));
|
||||
}
|
||||
|
||||
/// Returns the issue number embedded in [comment] or `null` if there is none.
|
||||
|
@ -196,3 +201,28 @@ class StatusEntry {
|
|||
|
||||
StatusEntry(this.path, this.expectations, this.issue);
|
||||
}
|
||||
|
||||
/// Error thrown when a parse or validation error occurs in a [StatusFile].
|
||||
class SyntaxError implements Exception {
|
||||
final String file;
|
||||
final int lineNumber;
|
||||
final String line;
|
||||
final String message;
|
||||
final List<String> errors;
|
||||
|
||||
SyntaxError(this.file, this.lineNumber, this.line, this.message, this.errors);
|
||||
|
||||
String toString() {
|
||||
var buffer = new StringBuffer();
|
||||
buffer.writeln('$message in "$file" line $lineNumber:');
|
||||
buffer.writeln(line);
|
||||
|
||||
if (errors != null) {
|
||||
for (var error in errors) {
|
||||
buffer.writeln("- ${error.replaceAll('\n', '\n ')}");
|
||||
}
|
||||
}
|
||||
|
||||
return buffer.toString().trimRight();
|
||||
}
|
||||
}
|
12
pkg/status_file/pubspec.yaml
Normal file
12
pkg/status_file/pubspec.yaml
Normal file
|
@ -0,0 +1,12 @@
|
|||
name: status_file
|
||||
author: Dart Team <misc@dartlang.org>
|
||||
description: Parses status files.
|
||||
homepage: http://www.dartlang.org
|
||||
#environment:
|
||||
# sdk: '>=1.0.0 <2.0.0'
|
||||
dependencies:
|
||||
path:
|
||||
path: ../../third_party/pkg/path
|
||||
dev_dependencies:
|
||||
expect:
|
||||
path: ../expect
|
36
pkg/status_file/test/repo_status_files_test.dart
Normal file
36
pkg/status_file/test/repo_status_files_test.dart
Normal file
|
@ -0,0 +1,36 @@
|
|||
// 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.
|
||||
|
||||
/// Tests that every .status file in the Dart repository can be successfully
|
||||
/// parsed.
|
||||
import 'dart:io';
|
||||
|
||||
import 'package:expect/expect.dart';
|
||||
import 'package:path/path.dart' as p;
|
||||
import 'package:status_file/status_file.dart';
|
||||
|
||||
final repoRoot =
|
||||
p.normalize(p.join(p.dirname(p.fromUri(Platform.script)), "../../../"));
|
||||
|
||||
void main() {
|
||||
// Parse every status file in the repository.
|
||||
for (var directory in ["tests", p.join("runtime", "tests")]) {
|
||||
for (var entry in new Directory(p.join(repoRoot, directory))
|
||||
.listSync(recursive: true)) {
|
||||
if (!entry.path.endsWith(".status")) continue;
|
||||
|
||||
// Inside the co19 repository, there is a status file that doesn't appear
|
||||
// to be valid and looks more like some kind of template or help document.
|
||||
// Ignore it.
|
||||
if (entry.path.endsWith(p.join("co19", "src", "co19.status"))) continue;
|
||||
|
||||
try {
|
||||
new StatusFile.read(entry.path);
|
||||
} catch (err) {
|
||||
var path = p.relative(entry.path, from: repoRoot);
|
||||
Expect.fail("Could not parse '$path'.\n$err");
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
|
@ -1,17 +1,21 @@
|
|||
// Copyright (c) 2011, the Dart project authors. Please see the AUTHORS file
|
||||
// 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 "package:expect/expect.dart";
|
||||
|
||||
import "../../tools/testing/dart/environment.dart";
|
||||
import "../../tools/testing/dart/status_expression.dart";
|
||||
import "package:status_file/environment.dart";
|
||||
import "package:status_file/src/expression.dart";
|
||||
|
||||
class TestEnvironment implements Environment {
|
||||
final Map<String, String> _values;
|
||||
|
||||
TestEnvironment(this._values);
|
||||
|
||||
void validate(String name, String value, List<String> errors) {
|
||||
throw new UnimplementedError();
|
||||
}
|
||||
|
||||
/// Looks up the value of the variable with [name].
|
||||
String lookUp(String name) => _values[name];
|
||||
|
|
@ -1,40 +0,0 @@
|
|||
// Copyright (c) 2013, 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.
|
||||
|
||||
library StatusFileParserTest;
|
||||
|
||||
import "package:expect/expect.dart";
|
||||
import "dart:io";
|
||||
import "../../../tools/testing/dart/path.dart";
|
||||
import "../../../tools/testing/dart/status_file.dart";
|
||||
import "../../../tools/testing/dart/utils.dart";
|
||||
|
||||
void main() {
|
||||
testReadStatusFile("runtime/tests/vm/vm.status");
|
||||
testReadStatusFile("samples/tests/samples/samples.status");
|
||||
testReadStatusFile("tests/co19/co19-compiler.status");
|
||||
testReadStatusFile("tests/co19/co19-runtime.status");
|
||||
testReadStatusFile("tests/corelib/corelib.status");
|
||||
testReadStatusFile("tests/dom/dom.status");
|
||||
testReadStatusFile("tests/html/html.status");
|
||||
testReadStatusFile("tests/isolate/isolate.status");
|
||||
testReadStatusFile("tests/language/language.status");
|
||||
testReadStatusFile("tests/standalone/standalone.status");
|
||||
}
|
||||
|
||||
String fixFilePath(String filePath) {
|
||||
if (new File(filePath).existsSync()) {
|
||||
return filePath;
|
||||
} else {
|
||||
return "../${filePath}";
|
||||
}
|
||||
}
|
||||
|
||||
void testReadStatusFile(String filePath) {
|
||||
var file = new File(fixFilePath(filePath));
|
||||
if (!file.existsSync()) return;
|
||||
|
||||
var statusFile = new StatusFile.read(file.path);
|
||||
Expect.isTrue(statusFile.sections.length > 0);
|
||||
}
|
|
@ -4,9 +4,11 @@
|
|||
|
||||
import "dart:io";
|
||||
import "dart:async";
|
||||
|
||||
import "package:status_file/expectation.dart";
|
||||
|
||||
import "../../../tools/testing/dart/command.dart";
|
||||
import "../../../tools/testing/dart/configuration.dart";
|
||||
import "../../../tools/testing/dart/expectation.dart";
|
||||
import "../../../tools/testing/dart/options.dart";
|
||||
import "../../../tools/testing/dart/test_runner.dart";
|
||||
import "../../../tools/testing/dart/test_suite.dart";
|
||||
|
|
|
@ -471,7 +471,5 @@ io/process_run_output_test: Fail # Unable to parse package files Flutter Issue 9
|
|||
|
||||
io/dependency_graph_test: CompileTimeError # Imports dart:mirrors
|
||||
io/skipping_dart2js_compilations_test: CompileTimeError # Uses mirrors
|
||||
io/status_file_parser_test: CompileTimeError # Uses mirrors
|
||||
io/test_harness_analyzer_test: CompileTimeError # Uses mirrors
|
||||
io/test_runner_test: CompileTimeError # Uses mirrors
|
||||
status_expression_test: CompileTimeError # Uses mirrors
|
||||
|
|
|
@ -7,9 +7,10 @@ import 'dart:async';
|
|||
// CommandOutput.exitCode in subclasses of CommandOutput.
|
||||
import 'dart:io' as io;
|
||||
|
||||
import 'package:status_file/expectation.dart';
|
||||
|
||||
import 'command_output.dart';
|
||||
import 'configuration.dart';
|
||||
import 'expectation.dart';
|
||||
import 'path.dart';
|
||||
import 'utils.dart';
|
||||
|
||||
|
|
|
@ -7,10 +7,11 @@ import 'dart:convert';
|
|||
// CommandOutput.exitCode in subclasses of CommandOutput.
|
||||
import 'dart:io' as io;
|
||||
|
||||
import 'package:status_file/expectation.dart';
|
||||
|
||||
import 'browser_controller.dart';
|
||||
import 'command.dart';
|
||||
import 'configuration.dart';
|
||||
import 'expectation.dart';
|
||||
import 'test_runner.dart';
|
||||
import 'utils.dart';
|
||||
|
||||
|
|
|
@ -2,6 +2,8 @@
|
|||
// 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 'package:status_file/environment.dart';
|
||||
|
||||
import 'configuration.dart';
|
||||
|
||||
typedef String _LookUpFunction(Configuration configuration);
|
||||
|
@ -51,12 +53,17 @@ String _runtimeName(Configuration configuration) {
|
|||
///
|
||||
/// These mostly map to command line arguments with the same name, though this
|
||||
/// is only a subset of the full set of command line arguments.
|
||||
class Environment {
|
||||
class ConfigurationEnvironment implements Environment {
|
||||
/// The configuration where variable data is found.
|
||||
final Configuration _configuration;
|
||||
|
||||
ConfigurationEnvironment(this._configuration);
|
||||
|
||||
/// Validates that the variable with [name] exists and can be compared
|
||||
/// against [value].
|
||||
///
|
||||
/// If any errors are found, adds them to [errors].
|
||||
static void validate(String name, String value, List<String> errors) {
|
||||
void validate(String name, String value, List<String> errors) {
|
||||
var variable = _variables[name];
|
||||
if (variable == null) {
|
||||
errors.add('Unknown variable "$name".');
|
||||
|
@ -74,11 +81,6 @@ class Environment {
|
|||
}
|
||||
}
|
||||
|
||||
/// The configuration where variable data is found.
|
||||
final Configuration _configuration;
|
||||
|
||||
Environment(this._configuration);
|
||||
|
||||
/// Looks up the value of the variable with [name].
|
||||
String lookUp(String name) {
|
||||
var variable = _variables[name];
|
||||
|
|
|
@ -2,10 +2,13 @@
|
|||
// 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:status_file/expectation.dart';
|
||||
import 'package:status_file/status_file.dart';
|
||||
|
||||
import 'configuration.dart';
|
||||
import 'environment.dart';
|
||||
import 'expectation.dart';
|
||||
import 'status_file.dart';
|
||||
|
||||
/// Tracks the [Expectation]s associated with a set of file paths.
|
||||
///
|
||||
|
@ -18,10 +21,12 @@ class ExpectationSet {
|
|||
/// when in [configuration].
|
||||
static ExpectationSet read(
|
||||
List<String> statusFilePaths, Configuration configuration) {
|
||||
var environment = new Environment(configuration);
|
||||
try {
|
||||
var environment = new ConfigurationEnvironment(configuration);
|
||||
var expectations = new ExpectationSet._();
|
||||
for (var path in statusFilePaths) {
|
||||
var file = new StatusFile.read(path);
|
||||
file.validate(environment);
|
||||
for (var section in file.sections) {
|
||||
if (section.isEnabled(environment)) {
|
||||
for (var entry in section.entries) {
|
||||
|
@ -32,6 +37,12 @@ class ExpectationSet {
|
|||
}
|
||||
|
||||
return expectations;
|
||||
} on SyntaxError catch (error) {
|
||||
stderr.writeln(error.toString());
|
||||
exit(1);
|
||||
|
||||
throw "unreachable";
|
||||
}
|
||||
}
|
||||
|
||||
// Only create one copy of each Set<Expectation>.
|
||||
|
|
|
@ -4,7 +4,8 @@
|
|||
|
||||
library summary_report;
|
||||
|
||||
import "expectation.dart";
|
||||
import "package:status_file/expectation.dart";
|
||||
|
||||
import "test_runner.dart";
|
||||
|
||||
final summaryReport = new SummaryReport();
|
||||
|
|
|
@ -5,10 +5,11 @@
|
|||
import 'dart:convert' show JSON;
|
||||
import 'dart:io';
|
||||
|
||||
import "package:status_file/expectation.dart";
|
||||
|
||||
import 'command.dart';
|
||||
import 'command_output.dart';
|
||||
import 'configuration.dart';
|
||||
import 'expectation.dart';
|
||||
import 'path.dart';
|
||||
import 'summary_report.dart';
|
||||
import 'test_runner.dart';
|
||||
|
|
|
@ -17,13 +17,14 @@ import 'dart:convert';
|
|||
import 'dart:io' as io;
|
||||
import 'dart:math' as math;
|
||||
|
||||
import "package:status_file/expectation.dart";
|
||||
|
||||
import 'android.dart';
|
||||
import 'browser_controller.dart';
|
||||
import 'command.dart';
|
||||
import 'command_output.dart';
|
||||
import 'configuration.dart';
|
||||
import 'dependency_graph.dart';
|
||||
import 'expectation.dart';
|
||||
import 'runtime_configuration.dart';
|
||||
import 'test_progress.dart';
|
||||
import 'test_suite.dart';
|
||||
|
|
|
@ -15,11 +15,12 @@
|
|||
import 'dart:async';
|
||||
import 'dart:io';
|
||||
|
||||
import "package:status_file/expectation.dart";
|
||||
|
||||
import 'browser_test.dart';
|
||||
import 'command.dart';
|
||||
import 'compiler_configuration.dart';
|
||||
import 'configuration.dart';
|
||||
import 'expectation.dart';
|
||||
import 'expectation_set.dart';
|
||||
import 'html_test.dart' as html_test;
|
||||
import 'http_server.dart';
|
||||
|
|
Loading…
Reference in a new issue