| Index: pkg/front_end/test/summary_generator_test.dart
|
| diff --git a/pkg/front_end/test/summary_generator_test.dart b/pkg/front_end/test/summary_generator_test.dart
|
| new file mode 100644
|
| index 0000000000000000000000000000000000000000..74b25601f721a609bcf71ab8a26852ddacf60730
|
| --- /dev/null
|
| +++ b/pkg/front_end/test/summary_generator_test.dart
|
| @@ -0,0 +1,152 @@
|
| +// 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:front_end/front_end.dart';
|
| +import 'package:front_end/src/testing/compiler_common.dart';
|
| +import 'package:kernel/ast.dart';
|
| +import 'package:kernel/kernel.dart';
|
| +
|
| +import 'package:test/test.dart';
|
| +
|
| +main() {
|
| + test('summary has no source-info by default', () async {
|
| + var summary = await summarize(['a.dart'], allSources);
|
| + var program = loadProgramFromBytes(summary);
|
| +
|
| + // Note: the kernel representation always has an empty '' key in the map,
|
| + // but otherwise no other data is included here.
|
| + expect(program.uriToSource.keys.single, '');
|
| + });
|
| +
|
| + test('summary includes declarations, but no method bodies', () async {
|
| + var summary = await summarize(['a.dart'], allSources);
|
| + var program = loadProgramFromBytes(summary);
|
| + var aLib = findLibrary(program, 'a.dart');
|
| + expect(aLib.importUri.path, '/a/b/c/a.dart');
|
| + var classA = aLib.classes.first;
|
| + expect(classA.name, 'A');
|
| + var fooMethod = classA.procedures.first;
|
| + expect(fooMethod.name.name, 'foo');
|
| + expect(fooMethod.function.body is EmptyStatement, isTrue);
|
| + });
|
| +
|
| + test('summarized libraries are not marked external', () async {
|
| + var summary = await summarize(['a.dart'], allSources);
|
| + var program = loadProgramFromBytes(summary);
|
| + var aLib = findLibrary(program, 'a.dart');
|
| + expect(aLib.importUri.path, '/a/b/c/a.dart');
|
| + expect(aLib.isExternal, isFalse);
|
| + });
|
| +
|
| + test('sdk dependencies are marked external', () async {
|
| + // Note: by default this test is loading the SDK from summaries.
|
| + var summary = await summarize(['a.dart'], allSources);
|
| + var program = loadProgramFromBytes(summary);
|
| + var coreLib = findLibrary(program, 'core');
|
| + expect(coreLib.isExternal, isTrue);
|
| + });
|
| +
|
| + test('non-sdk dependencies are marked external', () async {
|
| + var summaryA = await summarize(['a.dart'], allSources);
|
| + var sourcesWithA = new Map.from(allSources);
|
| + sourcesWithA['a.dill'] = summaryA;
|
| + var summaryB =
|
| + await summarize(['b.dart'], sourcesWithA, inputSummaries: ['a.dill']);
|
| +
|
| + var program = loadProgramFromBytes(summaryB);
|
| + var aLib = findLibrary(program, 'a.dart');
|
| + var bLib = findLibrary(program, 'b.dart');
|
| + expect(aLib.isExternal, isTrue);
|
| + expect(bLib.isExternal, isFalse);
|
| + });
|
| +
|
| + test('dependencies can be combined without conflict', () async {
|
| + var summaryA = await summarize(['a.dart'], allSources);
|
| + var sourcesWithA = new Map.from(allSources);
|
| + sourcesWithA['a.dill'] = summaryA;
|
| +
|
| + var summaryBC = await summarize(['b.dart', 'c.dart'], sourcesWithA,
|
| + inputSummaries: ['a.dill']);
|
| +
|
| + var sourcesWithABC = new Map.from(sourcesWithA);
|
| + sourcesWithABC['bc.dill'] = summaryBC;
|
| +
|
| + // Note: a is loaded first, bc.dill have a.dart as an external reference so
|
| + // we want to ensure loading them here will not create a problem.
|
| + var summaryD = await summarize(['d.dart'], sourcesWithABC,
|
| + inputSummaries: ['a.dill', 'bc.dill']);
|
| +
|
| + checkDSummary(summaryD);
|
| + });
|
| +
|
| + test('dependencies can be combined in any order', () async {
|
| + var summaryA = await summarize(['a.dart'], allSources);
|
| + var sourcesWithA = new Map.from(allSources);
|
| + sourcesWithA['a.dill'] = summaryA;
|
| +
|
| + var summaryBC = await summarize(['b.dart', 'c.dart'], sourcesWithA,
|
| + inputSummaries: ['a.dill']);
|
| +
|
| + var sourcesWithABC = new Map.from(sourcesWithA);
|
| + sourcesWithABC['bc.dill'] = summaryBC;
|
| +
|
| + // Note: unlinke the previous test now bc.dill is loaded first and contains
|
| + // an external definition of library a.dart. Using this order also works
|
| + // because we share a CanonicalName root to resolve names across multiple
|
| + // dill files and because of how the kernel loader merges definitions.
|
| + var summaryD = await summarize(['d.dart'], sourcesWithABC,
|
| + inputSummaries: ['bc.dill', 'a.dill']);
|
| + checkDSummary(summaryD);
|
| + });
|
| +
|
| + test('summarization by default is hermetic', () async {
|
| + var errors = [];
|
| + var options = new CompilerOptions()..onError = (e) => errors.add(e);
|
| + await summarize(['b.dart'], allSources, options: options);
|
| + expect(errors.first.toString(), contains('Invalid access'));
|
| + errors.clear();
|
| +
|
| + await summarize(['a.dart', 'b.dart'], allSources, options: options);
|
| + expect(errors, isEmpty);
|
| + });
|
| +
|
| + // TODO(sigmund): test trimDependencies when it is part of the public API.
|
| +}
|
| +
|
| +var allSources = {
|
| + 'a.dart': 'class A { foo() { print("hi"); } }',
|
| + 'b.dart': 'import "a.dart"; class B extends A {}',
|
| + 'c.dart': 'class C { bar() => 1; }',
|
| + 'd.dart': '''
|
| + import "a.dart";
|
| + import "b.dart";
|
| + import "c.dart";
|
| + class D extends B with C implements A { }''',
|
| +};
|
| +
|
| +/// Helper function to check that some expectations from the summary of D.
|
| +checkDSummary(List<int> summary) {
|
| + var program = loadProgramFromBytes(summary);
|
| + var aLib = findLibrary(program, 'a.dart');
|
| + var bLib = findLibrary(program, 'b.dart');
|
| + var cLib = findLibrary(program, 'c.dart');
|
| + var dLib = findLibrary(program, 'd.dart');
|
| +
|
| + // All libraries but `d.dart` are marked external.
|
| + expect(aLib.isExternal, isTrue);
|
| + expect(bLib.isExternal, isTrue);
|
| + expect(cLib.isExternal, isTrue);
|
| + expect(dLib.isExternal, isFalse);
|
| +
|
| + // The type-hierarchy for A, B, D is visible and correct
|
| + var aClass = aLib.classes.firstWhere((c) => c.name == 'A');
|
| + var bClass = bLib.classes.firstWhere((c) => c.name == 'B');
|
| + expect(bClass.superclass, same(aClass));
|
| +
|
| + var dClass = dLib.classes.firstWhere((c) => c.name == 'D');
|
| + expect(dClass.superclass.superclass, same(bClass));
|
| +
|
| + var dInterface = dClass.implementedTypes.first.classNode;
|
| + expect(dInterface, same(aClass));
|
| +}
|
|
|