blob: f5ecfdd538cac83983f61f80c07a62053c8cc2dc [file] [log] [blame]
// Copyright (c) 2016, 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.md file.
// TODO(ahe): Copied from closure_conversion branch of kernel, remove this file
// when closure_conversion is merged with master.
library kernel.testing.kernel_chain;
import 'dart:async' show
Future;
import 'dart:io' show
Directory,
File,
IOSink,
Platform;
import 'dart:typed_data' show
Uint8List;
import 'package:kernel/kernel.dart' show
Repository,
loadProgramFromBinary;
import 'package:kernel/text/ast_to_text.dart' show
Printer;
import 'package:testing/testing.dart' show
Result,
StdioProcess,
Step;
import 'package:kernel/ast.dart' show
Library,
Program;
import 'package:kernel/verifier.dart' show
VerifyingVisitor;
import 'package:kernel/binary/ast_to_binary.dart' show
BinaryPrinter;
import 'package:kernel/binary/ast_from_binary.dart' show
BinaryBuilder;
import 'package:kernel/binary/loader.dart' show
BinaryLoader;
import 'package:analyzer/src/generated/sdk.dart' show
DartSdk;
import 'package:kernel/analyzer/loader.dart' show
DartLoader,
DartOptions,
createDartSdk;
import 'package:kernel/target/targets.dart' show
Target,
TargetFlags,
getTarget;
import 'package:kernel/repository.dart' show
Repository;
import 'package:testing/testing.dart' show
Chain,
ChainContext,
Result,
StdioProcess,
Step,
TestDescription;
import 'package:kernel/ast.dart' show
Program;
import 'package:package_config/discovery.dart' show
loadPackagesFile;
typedef Future<TestContext> TestContextConstructor(
Chain suite, Map<String, String> environment, String sdk, Uri vm,
Uri packages, bool strongMode, DartSdk dartSdk, bool updateExpectations);
Future<bool> fileExists(Uri base, String path) async {
return await new File.fromUri(base.resolve(path)).exists();
}
abstract class TestContext extends ChainContext {
final Uri vm;
final Uri packages;
final DartOptions options;
final DartSdk dartSdk;
TestContext(String sdk, this.vm, Uri packages, bool strongMode, this.dartSdk)
: packages = packages,
options = new DartOptions(strongMode: strongMode, sdk: sdk,
packagePath: packages.toFilePath());
Future<DartLoader> createLoader() async {
Repository repository = new Repository();
return new DartLoader(repository, options, await loadPackagesFile(packages),
ignoreRedirectingFactories: false, dartSdk: dartSdk);
}
static Future<TestContext> create(Chain suite,
Map<String, String> environment,
TestContextConstructor constructor) async {
const String suggestion =
"Try checking the value of environment variable 'DART_AOT_SDK', "
"it should point to a patched SDK.";
String sdk = await getEnvironmentVariable(
"DART_AOT_SDK", Environment.directory,
"Please define environment variable 'DART_AOT_SDK' to point to a "
"patched SDK.",
(String n) => "Couldn't locate '$n'. $suggestion");
Uri sdkUri = Uri.base.resolve("$sdk/");
const String asyncDart = "lib/async/async.dart";
if (!await fileExists(sdkUri, asyncDart)) {
throw "Couldn't find '$asyncDart' in '$sdk'. $suggestion";
}
const String asyncSources = "lib/async/async_sources.gypi";
if (await fileExists(sdkUri, asyncSources)) {
throw "Found '$asyncSources' in '$sdk', so it isn't a patched SDK. "
"$suggestion";
}
String vmPath = await getEnvironmentVariable(
"DART_AOT_VM", Environment.file,
"Please define environment variable 'DART_AOT_VM' to point to a "
"Dart VM that reads .dill files.",
(String n) => "Couldn't locate '$n'. Please check the value of "
"environment variable 'DART_AOT_VM', it should point to a "
"Dart VM that reads .dill files.");
Uri vm = Uri.base.resolve(vmPath);
Uri packages = Uri.base.resolve(".packages");
bool strongMode = false;
bool updateExpectations = environment["updateExpectations"] != "false";
return constructor(suite, environment, sdk, vm, packages, strongMode,
createDartSdk(sdk, strongMode: strongMode), updateExpectations);
}
}
enum Environment {
directory,
file,
}
Future<String> getEnvironmentVariable(
String name, Environment kind, String undefined, notFound(String n)) async {
String result = Platform.environment[name];
if (result == null) {
throw undefined;
}
switch (kind) {
case Environment.directory:
if (!await new Directory(result).exists()) throw notFound(result);
break;
case Environment.file:
if (!await new File(result).exists()) throw notFound(result);
break;
}
return result;
}
class Kernel extends Step<TestDescription, Program, TestContext> {
const Kernel();
String get name => "kernel";
Future<Result<Program>> run(
TestDescription description, TestContext testContext) async {
try {
DartLoader loader = await testContext.createLoader();
Target target = getTarget(
"vm", new TargetFlags(strongMode: testContext.options.strongMode));
Program program =
loader.loadProgram(description.uri, target: target);
for (var error in loader.errors) {
return fail(program, "$error");
}
target.transformProgram(program);
return pass(program);
} catch (e, s) {
return crash(e, s);
}
}
}
class Print extends Step<Program, Program, dynamic> {
const Print();
String get name => "print";
Future<Result<Program>> run(Program program, _) async {
StringBuffer sb = new StringBuffer();
for (Library library in program.libraries) {
Printer printer = new Printer(sb);
if (library.importUri.scheme != "dart" &&
library.importUri.scheme != "package") {
printer.writeLibraryFile(library);
}
}
print("$sb");
return pass(program);
}
}
class Verify extends Step<Program, Program, dynamic> {
final bool fullCompile;
const Verify(this.fullCompile);
String get name => "verify";
Future<Result<Program>> run(Program program, TestContext testContext) async {
try {
program.accept(new VerifyingVisitor()..isOutline = !fullCompile);
return pass(program);
} catch (e, s) {
return new Result<Program>(
null, testContext.expectationSet["VerificationError"], e, s);
}
}
}
class MatchExpectation extends Step<Program, Program, dynamic> {
final String suffix;
// TODO(ahe): This is true by default which doesn't match well with the class
// name.
final bool updateExpectations;
const MatchExpectation(this.suffix, {this.updateExpectations: true});
String get name => "match expectations";
Future<Result<Program>> run(Program program, _) async {
Library library = program.libraries.firstWhere(
(Library library) => library.importUri.scheme != "dart");
Uri uri = library.importUri;
StringBuffer buffer = new StringBuffer();
new Printer(buffer).writeLibraryFile(library);
bool updateExpectations = this.updateExpectations;
if (uri.path.contains("/test/rasta/")) {
// TODO(ahe): Remove this. Short term, we don't want to automatically
// update rasta expectations, as we have too many failures.
updateExpectations = false;
}
File expectedFile = new File("${uri.toFilePath()}$suffix");
if (await expectedFile.exists()) {
String expected = await expectedFile.readAsString();
if (expected.trim() != "$buffer".trim()) {
if (!updateExpectations) {
String diff = await runDiff(expectedFile.uri, "$buffer");
return fail(null, "$uri doesn't match ${expectedFile.uri}\n$diff");
}
} else {
return pass(program);
}
}
if (updateExpectations) {
await openWrite(expectedFile.uri, (IOSink sink) {
sink.writeln("$buffer".trim());
});
return pass(program);
} else {
return fail(program, """
Please create file ${expectedFile.path} with this content:
$buffer""");
}
}
}
class WriteDill extends Step<Program, Uri, dynamic> {
const WriteDill();
String get name => "write .dill";
Future<Result<Uri>> run(Program program, _) async {
Directory tmp = await Directory.systemTemp.createTemp();
Uri uri = tmp.uri.resolve("generated.dill");
File generated = new File.fromUri(uri);
IOSink sink = generated.openWrite();
try {
new BinaryPrinter(sink).writeProgramFile(program);
} catch (e, s) {
return fail(uri, e, s);
} finally {
print("Wrote `${generated.path}`");
await sink.close();
}
return pass(uri);
}
}
class ReadDill extends Step<Uri, Uri, dynamic> {
const ReadDill();
String get name => "read .dill";
Future<Result<Uri>> run(Uri uri, _) async {
try {
loadProgramFromBinary(uri.toFilePath());
} catch (e, s) {
return fail(uri, e, s);
}
return pass(uri);
}
}
class Copy extends Step<Program, Program, dynamic> {
const Copy();
String get name => "copy program";
Future<Result<Program>> run(Program program, _) async {
BytesCollector sink = new BytesCollector();
new BinaryPrinter(sink).writeProgramFile(program);
Uint8List bytes = sink.collect();
BinaryLoader loader = new BinaryLoader(new Repository());
return pass(new BinaryBuilder(loader, bytes).readProgramFile());
}
}
class Run extends Step<Uri, int, TestContext> {
const Run();
String get name => "run";
bool get isAsync => true;
bool get isRuntime => true;
Future<Result<int>> run(Uri uri, TestContext context) async {
File generated = new File.fromUri(uri);
StdioProcess process;
try {
process = await StdioProcess.run(
context.vm.toFilePath(), [generated.path, "Hello, World!"]);
print(process.output);
} finally {
generated.parent.delete(recursive: true);
}
return process.toResult();
}
}
class BytesCollector implements Sink<List<int>> {
final List<List<int>> lists = <List<int>>[];
int length = 0;
void add(List<int> data) {
lists.add(data);
length += data.length;
}
Uint8List collect() {
Uint8List result = new Uint8List(length);
int offset = 0;
for (List<int> list in lists) {
result.setRange(offset, offset += list.length, list);
}
lists.clear();
length = 0;
return result;
}
void close() {}
}
Future<String> runDiff(Uri expected, String actual) async {
// TODO(ahe): Implement this for Windows.
StdioProcess process = await StdioProcess.run(
"diff", <String>["-u", expected.toFilePath(), "-"], input: actual);
return process.output;
}
Future openWrite(Uri uri, f(IOSink sink)) async {
IOSink sink = new File.fromUri(uri).openWrite();
try {
await f(sink);
} finally {
await sink.close();
}
print("Wrote $uri");
}