blob: 0f62d7a4c199d7f0f5bd65dbf02904913efb9020 [file] [log] [blame]
// 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.md file.
library test.kernel.closures.suite;
import 'dart:async' show Future;
import 'dart:io' show File;
import 'package:front_end/physical_file_system.dart';
import 'package:testing/testing.dart'
show Chain, ChainContext, Result, Step, TestDescription, runMe;
import 'package:kernel/ast.dart' show Program, Library;
import 'package:front_end/src/fasta/testing/kernel_chain.dart' show runDiff;
import 'package:front_end/src/fasta/ticker.dart' show Ticker;
import 'package:front_end/src/fasta/dill/dill_target.dart' show DillTarget;
import 'package:front_end/src/fasta/kernel/kernel_target.dart'
show KernelTarget;
import 'package:front_end/src/fasta/translate_uri.dart' show TranslateUri;
import 'package:front_end/src/fasta/errors.dart' show InputError;
import 'package:front_end/src/fasta/testing/patched_sdk_location.dart';
import 'package:kernel/kernel.dart' show loadProgramFromBinary;
import 'package:kernel/interpreter/interpreter.dart';
const String STRONG_MODE = " strong mode ";
class InterpreterContext extends ChainContext {
final bool strongMode;
final TranslateUri uriTranslator;
final List<Step> steps;
Future<Program> platform;
InterpreterContext(this.strongMode, this.uriTranslator)
: steps = <Step>[
const FastaCompile(),
const Interpret(),
const MatchLogExpectation(".expect"),
];
Future<Program> loadPlatform() async {
Uri sdk = await computePatchedSdk();
return loadProgramFromBinary(sdk.resolve('platform.dill').toFilePath());
}
static Future<InterpreterContext> create(
Chain suite, Map<String, String> environment) async {
Uri packages = Uri.base.resolve(".packages");
bool strongMode = environment.containsKey(STRONG_MODE);
TranslateUri uriTranslator =
await TranslateUri.parse(PhysicalFileSystem.instance, packages);
return new InterpreterContext(strongMode, uriTranslator);
}
}
class FastaCompile extends Step<TestDescription, Program, InterpreterContext> {
const FastaCompile();
String get name => "fasta compile";
Future<Result<Program>> run(
TestDescription description, InterpreterContext context) async {
Program platform = await context.loadPlatform();
Ticker ticker = new Ticker();
DillTarget dillTarget = new DillTarget(ticker, context.uriTranslator);
platform.unbindCanonicalNames();
dillTarget.loader.appendLibraries(platform);
KernelTarget sourceTarget = new KernelTarget(PhysicalFileSystem.instance,
dillTarget, context.uriTranslator, context.strongMode);
Program p;
try {
sourceTarget.read(description.uri);
await dillTarget.buildOutlines();
await sourceTarget.buildOutlines();
p = await sourceTarget.buildProgram();
} on InputError catch (e, s) {
return fail(null, e.error, s);
}
return pass(p);
}
}
class Interpret extends Step<Program, EvaluationLog, InterpreterContext> {
const Interpret();
String get name => "interpret";
Future<Result<EvaluationLog>> run(Program program, _) async {
Library library = program.libraries
.firstWhere((Library library) => library.importUri.scheme != "dart");
Uri uri = library.importUri;
StringBuffer buffer = new StringBuffer();
log.onRecord.listen((LogRecord rec) => buffer.write(rec.message));
try {
new Interpreter(program).run();
} catch (e, s) {
return crash(e, s);
}
return pass(new EvaluationLog(uri, "$buffer"));
}
}
class MatchLogExpectation extends Step<EvaluationLog, int, InterpreterContext> {
final String suffix;
String get name => "match log expectation";
const MatchLogExpectation(this.suffix);
Future<Result<int>> run(EvaluationLog result, _) async {
Uri uri = result.uri;
File expectedFile = new File("${uri.toFilePath()}$suffix");
if (await expectedFile.exists()) {
String expected = await expectedFile.readAsString();
if (expected.trim() != result.log.trim()) {
String diff = await runDiff(expectedFile.uri, result.log);
return fail(null, "$uri doesn't match ${expectedFile.uri}\n$diff");
} else {
return pass(0);
}
}
return fail(
null,
"""Please create file ${expectedFile.path} with this content:
${result.log}""");
}
}
class EvaluationLog {
/// Evaluated program uri.
final Uri uri;
/// Evaluated program log.
final String log;
EvaluationLog(this.uri, this.log);
}
main(List<String> arguments) =>
runMe(arguments, InterpreterContext.create, "testing.json");