From e79f6746c85a483cdafa2009ff3ce166a73038f5 Mon Sep 17 00:00:00 2001 From: Jacob MacDonald <jakemac@google.com> Date: Thu, 4 May 2017 12:49:51 -0700 Subject: [PATCH] Add DartDevcBootstrapTransformer and compiler tests for dartdevc (#1582) Some additional updates based on testing with PPW (now that we have an e2e to play with): * add sourcemaps to output and update tests * use strong mode summaries, support filenames with dots, add a fake sourcemap to satisfy package test * add better error if the module reader fails to find a module for a file * stop diet parsing, remove --unsafe-force-compile * add environment constant support to the DartDevcModuleTransformer * don't output ddc resources in folders with no entrypoints * don't create sourcemaps in release mode * update relevant compiler tests to run with dartdevc --- lib/src/barback/asset_environment.dart | 27 +++-- .../dartdevc_bootstrap_transformer.dart | 101 ++++++++++++++++++ .../dartdevc/dartdevc_module_transformer.dart | 31 +++++- .../dartdevc_resource_transformer.dart | 13 +++ .../dartdevc/linked_summary_transformer.dart | 4 +- lib/src/barback/dartdevc/module_computer.dart | 1 + lib/src/barback/dartdevc/module_reader.dart | 8 ++ .../unlinked_summary_transformer.dart | 4 +- .../dartdevc_module_transformer_test.dart | 5 + .../allows_import_in_dart_code_test.dart | 21 +++- test/compiler/compiler_flag_test.dart | 47 +++++--- ...iles_entrypoints_in_root_package_test.dart | 14 ++- .../compiles_generated_dart_file_test.dart | 14 ++- ...file_from_dependency_outside_web_test.dart | 19 +++- ...s_generated_file_from_dependency_test.dart | 18 +++- ...compiles_imported_generated_file_test.dart | 15 ++- ...nverts_isolate_entrypoint_in_web_test.dart | 21 +++- test/compiler/environment_constant_test.dart | 58 +++++++--- ...ignores_entrypoint_in_dependency_test.dart | 6 +- .../ignores_entrypoints_in_lib_test.dart | 15 +-- ...gnores_non_entrypoint_dart_files_test.dart | 23 +++- .../includes_source_maps_in_debug_test.dart | 62 ++++++++--- .../omits_source_map_in_release_test.dart | 54 ++++++++-- .../reports_dart_parse_errors_test.dart | 41 ++++--- .../source_maps_are_self_contained_test.dart | 41 +++++-- ...upports_configuration_with_build_test.dart | 9 +- test/compiler/utils.dart | 20 ++++ test/serve/utils.dart | 14 ++- 28 files changed, 568 insertions(+), 138 deletions(-) create mode 100644 lib/src/barback/dartdevc/dartdevc_bootstrap_transformer.dart create mode 100644 test/compiler/utils.dart diff --git a/lib/src/barback/asset_environment.dart b/lib/src/barback/asset_environment.dart index 7c5581a0..935bb641 100644 --- a/lib/src/barback/asset_environment.dart +++ b/lib/src/barback/asset_environment.dart @@ -18,6 +18,7 @@ import '../package.dart'; import '../package_graph.dart'; import '../source/cached.dart'; import '../utils.dart'; +import 'dartdevc/dartdevc_bootstrap_transformer.dart'; import 'dartdevc/dartdevc_module_transformer.dart'; import 'dartdevc/dartdevc_resource_transformer.dart'; import 'dartdevc/linked_summary_transformer.dart'; @@ -181,21 +182,27 @@ class AssetEnvironment { /// /// Returns `null` if there are none. Iterable<Set> getBuiltInTransformers(Package package) { - var transformers = <Set>[]; + var transformers = <List>[]; var isRootPackage = package.name == rootPackage.name; switch (compiler) { case Compiler.dartDevc: + var firstPhase = <dynamic>[new ModuleConfigTransformer()]; + var lastPhase = <dynamic>[ + new DartDevcModuleTransformer(mode, + environmentConstants: environmentConstants) + ]; + if (isRootPackage) { + firstPhase.add(new DartDevcResourceTransformer()); + lastPhase.add(new DartDevcBootstrapTransformer(mode)); + } + transformers.addAll([ - [new ModuleConfigTransformer()], + firstPhase, [new UnlinkedSummaryTransformer()], [new LinkedSummaryTransformer()], - [new DartDevcModuleTransformer(mode)], - ].map((list) => list.toSet())); - - if (isRootPackage) { - transformers.first.add(new DartDevcResourceTransformer()); - } + lastPhase, + ]); break; case Compiler.dart2JS: // the dart2js transformer only runs on the root package. @@ -213,12 +220,12 @@ class AssetEnvironment { transformers.add([ new Dart2JSTransformer(this, mode), new DartForwardingTransformer(), - ].toSet()); + ]); } } } - return transformers; + return transformers.map((list) => list.toSet()); } /// Starts up the admin server on an appropriate port and returns it. diff --git a/lib/src/barback/dartdevc/dartdevc_bootstrap_transformer.dart b/lib/src/barback/dartdevc/dartdevc_bootstrap_transformer.dart new file mode 100644 index 00000000..e831c0c7 --- /dev/null +++ b/lib/src/barback/dartdevc/dartdevc_bootstrap_transformer.dart @@ -0,0 +1,101 @@ +// 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:async'; + +import 'package:analyzer/analyzer.dart'; +import 'package:barback/barback.dart'; +import 'package:path/path.dart' as p; + +import '../../dart.dart'; +import '../../io.dart'; +import 'module_reader.dart'; + +class DartDevcBootstrapTransformer extends Transformer { + final BarbackMode mode; + + DartDevcBootstrapTransformer(this.mode); + + @override + bool isPrimary(AssetId id) { + // Only `.dart` files not under `lib` or `bin` are considered candidates for + // being dartdevc application entrypoints. + if (id.extension != '.dart') return false; + var dir = topLevelDir(id.path); + return dir != 'lib' && dir != 'bin'; + } + + @override + Future apply(Transform transform) async { + var parsed = + parseCompilationUnit(await transform.primaryInput.readAsString()); + if (!isEntrypoint(parsed)) return; + await _bootstrapEntrypoint(transform.primaryInput.id, mode, transform); + } +} + +/// Bootstraps the js module for the entrypoint dart file [dartEntrypointId] +/// with two additional JS files: +/// +/// * A `$dartEntrypointId.js` file which is the main entrypoint for the app. It +/// injects a script tag whose src is `require.js` and whose `data-main` +/// attribute points at a `$dartEntrypointId.bootstrap.js` file. +/// * A `$dartEntrypointId.bootstrap.js` file which invokes the top level `main` +/// function from the entrypoint module, after performing some necessary SDK +/// setup. +Future _bootstrapEntrypoint( + AssetId dartEntrypointId, BarbackMode mode, Transform transform) async { + var moduleReader = new ModuleReader(transform.readInputAsString); + var module = await moduleReader.moduleFor(dartEntrypointId); + + // The path to the entrypoint js module as it should appear in the call to + // `require` in the bootstrap file. + var moduleDir = topLevelDir(dartEntrypointId.path); + var appModulePath = p.relative(p.join(moduleDir, module.id.name), + from: p.dirname(dartEntrypointId.path)); + + // The name of the entrypoint dart library within the entrypoint js module. + // + // This is used to invoke `main()` from within the bootstrap script. + // + // TODO(jakemac53): Sane module name creation, this only works in the most + // basic of cases. + // + // See https://github.com/dart-lang/sdk/issues/27262 for the root issue which + // will allow us to not rely on the naming schemes that dartdevc uses + // internally, but instead specify our own. + var appModuleScope = p.url + .split(p + .withoutExtension(p.relative(dartEntrypointId.path, from: moduleDir))) + .join("__") + .replaceAll('.', '\$46'); + var bootstrapContent = ''' +require(["$appModulePath", "dart_sdk"], function(app, dart_sdk) { + dart_sdk._isolate_helper.startRootIsolate(() => {}, []); + app.$appModuleScope.main(); +}); +'''; + var bootstrapId = dartEntrypointId.addExtension('.bootstrap.js'); + transform.addOutput(new Asset.fromString(bootstrapId, bootstrapContent)); + + var bootstrapModuleName = p.withoutExtension( + p.relative(bootstrapId.path, from: p.dirname(dartEntrypointId.path))); + var entrypointJsContent = ''' +var el = document.createElement("script"); +el.defer = true; +el.async = false; +el.src = "require.js"; +el.setAttribute("data-main", "$bootstrapModuleName"); +document.head.appendChild(el); +'''; + transform.addOutput(new Asset.fromString( + dartEntrypointId.addExtension('.js'), entrypointJsContent)); + + if (mode == BarbackMode.DEBUG) { + transform.addOutput(new Asset.fromString( + dartEntrypointId.addExtension('.js.map'), + '{"version":3,"sourceRoot":"","sources":[],"names":[],"mappings":"",' + '"file":""}')); + } +} diff --git a/lib/src/barback/dartdevc/dartdevc_module_transformer.dart b/lib/src/barback/dartdevc/dartdevc_module_transformer.dart index 3c2a4495..9bdc3eef 100644 --- a/lib/src/barback/dartdevc/dartdevc_module_transformer.dart +++ b/lib/src/barback/dartdevc/dartdevc_module_transformer.dart @@ -23,10 +23,11 @@ class DartDevcModuleTransformer extends Transformer { @override String get allowedExtensions => moduleConfigName; + final Map<String, String> environmentConstants; final BarbackMode mode; - DartDevcModuleTransformer(this.mode); - + DartDevcModuleTransformer(this.mode, {this.environmentConstants = const {}}); + @override Future apply(Transform transform) async { ScratchSpace scratchSpace; @@ -46,7 +47,12 @@ class DartDevcModuleTransformer extends Transformer { scratchSpace = await ScratchSpace.create(allAssetIds, transform.readInput); await Future.wait(modules.map((m) => _createDartdevcModule( - m, scratchSpace, summariesForModule[m.id], mode, transform))); + m, + scratchSpace, + summariesForModule[m.id], + environmentConstants, + mode, + transform))); } finally { scratchSpace?.delete(); } @@ -59,6 +65,7 @@ Future _createDartdevcModule( Module module, ScratchSpace scratchSpace, Set<AssetId> linkedSummaryIds, + Map<String, String> environmentConstants, BarbackMode mode, Transform transform) async { var jsOutputFile = scratchSpace.fileFor(module.id.jsId); @@ -74,13 +81,25 @@ Future _createDartdevcModule( '--module-root=${scratchSpace.tempDir.path}', '--library-root=${p.dirname(jsOutputFile.path)}', '--summary-extension=${linkedSummaryExtension.substring(1)}', + '--no-summarize', '-o', jsOutputFile.path, ]); + + if (mode == BarbackMode.RELEASE) { + request.arguments.add('--no-source-map'); + } + + // Add environment constants. + environmentConstants.forEach((key, value) { + request.arguments.add('-D$key=$value'); + }); + // Add all the linked summaries as summary inputs. for (var id in linkedSummaryIds) { request.arguments.addAll(['-s', scratchSpace.fileFor(id).path]); } + // Add URL mappings for all the package: files to tell DartDevc where to find // them. for (var id in module.assetIds) { @@ -121,5 +140,11 @@ Future _createDartdevcModule( } else { transform.addOutput( new Asset.fromBytes(module.id.jsId, jsOutputFile.readAsBytesSync())); + if (mode == BarbackMode.DEBUG) { + var sourceMapOutputId = module.id.jsId.addExtension('.map'); + var sourceMapFile = scratchSpace.fileFor(sourceMapOutputId); + transform.addOutput(new Asset.fromBytes( + sourceMapOutputId, sourceMapFile.readAsBytesSync())); + } } } diff --git a/lib/src/barback/dartdevc/dartdevc_resource_transformer.dart b/lib/src/barback/dartdevc/dartdevc_resource_transformer.dart index d9d68364..46e69cbc 100644 --- a/lib/src/barback/dartdevc/dartdevc_resource_transformer.dart +++ b/lib/src/barback/dartdevc/dartdevc_resource_transformer.dart @@ -5,10 +5,12 @@ import 'dart:async'; import 'dart:io'; +import 'package:analyzer/analyzer.dart'; import 'package:barback/barback.dart'; import 'package:cli_util/cli_util.dart' as cli_util; import 'package:path/path.dart' as p; +import '../../dart.dart'; import '../../io.dart'; /// Copies the `dart_sdk.js` and `require.js` AMD files from the SDK into each @@ -26,6 +28,17 @@ class DartDevcResourceTransformer extends AggregateTransformer @override Future apply(AggregateTransform transform) async { + // If there are no entrypoints then skip this folder. + var hasEntrypoint = false; + await for (var asset in transform.primaryInputs) { + if (isEntrypoint(parseCompilationUnit(await asset.readAsString(), + parseFunctionBodies: false))) { + hasEntrypoint = true; + break; + } + } + if (!hasEntrypoint) return; + var sdk = cli_util.getSdkDir(); // Copy the dart_sdk.js file for AMD into the output folder. diff --git a/lib/src/barback/dartdevc/linked_summary_transformer.dart b/lib/src/barback/dartdevc/linked_summary_transformer.dart index 2980536b..4ba00bf4 100644 --- a/lib/src/barback/dartdevc/linked_summary_transformer.dart +++ b/lib/src/barback/dartdevc/linked_summary_transformer.dart @@ -59,10 +59,12 @@ Future _createLinkedSummaryForModule( Transform transform) async { var summaryOutputFile = tempEnv.fileFor(module.id.linkedSummaryId); var request = new WorkRequest(); + // TODO(jakemac53): Diet parsing results in erroneous errors in later steps, + // but ideally we would do that (pass '--build-summary-only-diet'). request.arguments.addAll([ '--build-summary-only', - '--build-summary-only-diet', '--build-summary-output=${summaryOutputFile.path}', + '--strong', ]); // Add all the unlinked summaries as build summary inputs. request.arguments.addAll(unlinkedSummaryIds.map( diff --git a/lib/src/barback/dartdevc/module_computer.dart b/lib/src/barback/dartdevc/module_computer.dart index 9721866b..85b5694f 100644 --- a/lib/src/barback/dartdevc/module_computer.dart +++ b/lib/src/barback/dartdevc/module_computer.dart @@ -85,6 +85,7 @@ Future<List<Module>> computeModules( for (var asset in srcAssets) { var id = asset.id; var content = await asset.readAsString(); + // Skip errors here, dartdevc gives nicer messages. var parsed = parseCompilationUnit(content, name: id.path, parseFunctionBodies: false, suppressErrors: true); parsedAssetsById[id] = parsed; diff --git a/lib/src/barback/dartdevc/module_reader.dart b/lib/src/barback/dartdevc/module_reader.dart index 38716dbe..8e30ad55 100644 --- a/lib/src/barback/dartdevc/module_reader.dart +++ b/lib/src/barback/dartdevc/module_reader.dart @@ -58,6 +58,14 @@ class ModuleReader { Future updateDeps(Iterable<AssetId> assetDepIds) async { for (var assetDepId in assetDepIds) { var assetDepModule = await moduleFor(assetDepId); + if (assetDepModule == null) { + throw new StateError( + 'Unable to find module for asset `$assetDepId`. This indicates ' + 'that either the file doesn\'t exist or it is not imported by ' + 'any public entrypoints in its package (files not under `lib/src`' + '). Importing a file directly that lives under `lib/src` is not ' + 'supported by the dartdevc transformers.'); + } if (!result.add(assetDepModule.id)) continue; await updateDeps(assetDepModule.directDependencies); } diff --git a/lib/src/barback/dartdevc/unlinked_summary_transformer.dart b/lib/src/barback/dartdevc/unlinked_summary_transformer.dart index 2bce94d1..bb6ed440 100644 --- a/lib/src/barback/dartdevc/unlinked_summary_transformer.dart +++ b/lib/src/barback/dartdevc/unlinked_summary_transformer.dart @@ -47,11 +47,13 @@ Future _createUnlinkedSummaryForModule( Module module, ScratchSpace scratchSpace, Transform transform) async { var summaryOutputFile = scratchSpace.fileFor(module.id.unlinkedSummaryId); var request = new WorkRequest(); + // TODO(jakemac53): Diet parsing results in erroneous errors later on today, + // but ideally we would do that (pass '--build-summary-only-diet'). request.arguments.addAll([ '--build-summary-only', '--build-summary-only-unlinked', - '--build-summary-only-diet', '--build-summary-output=${summaryOutputFile.path}', + '--strong', ]); // Add all the files to include in the unlinked summary bundle. request.arguments.addAll(module.assetIds.map((id) { diff --git a/test/barback/dartdevc/dartdevc_module_transformer_test.dart b/test/barback/dartdevc/dartdevc_module_transformer_test.dart index 0bc0b205..3e3d40ab 100644 --- a/test/barback/dartdevc/dartdevc_module_transformer_test.dart +++ b/test/barback/dartdevc/dartdevc_module_transformer_test.dart @@ -52,8 +52,13 @@ void main() { // Just confirm some basic things are present indicating that the module // was compiled. The goal here is not to test dartdevc itself. requestShouldSucceed('web__main.js', contains('main')); + requestShouldSucceed('web__main.js.map', contains('web__main.js')); requestShouldSucceed('packages/myapp/lib__hello.js', contains('hello')); + requestShouldSucceed( + 'packages/myapp/lib__hello.js.map', contains('lib__hello.js')); requestShouldSucceed('packages/foo/lib__foo.js', contains('message')); + requestShouldSucceed( + 'packages/foo/lib__foo.js.map', contains('lib__foo.js')); endPubServe(); }); } diff --git a/test/compiler/allows_import_in_dart_code_test.dart b/test/compiler/allows_import_in_dart_code_test.dart index a2926ad6..0514d2bb 100644 --- a/test/compiler/allows_import_in_dart_code_test.dart +++ b/test/compiler/allows_import_in_dart_code_test.dart @@ -7,9 +7,10 @@ import 'package:scheduled_test/scheduled_test.dart'; import '../descriptor.dart' as d; import '../test_pub.dart'; import '../serve/utils.dart'; +import 'utils.dart'; main() { - integration("handles imports in the Dart code", () { + integrationWithCompiler("handles imports in the Dart code", (compiler) { d.dir("foo", [ d.libPubspec("foo", "0.0.1"), d.dir("lib", [ @@ -47,9 +48,21 @@ void main() { ]).create(); pubGet(); - pubServe(); - requestShouldSucceed("main.dart.js", contains("footext")); - requestShouldSucceed("main.dart.js", contains("libtext")); + pubServe(compiler: compiler); + switch (compiler) { + case Compiler.dart2JS: + requestShouldSucceed("main.dart.js", contains("footext")); + requestShouldSucceed("main.dart.js", contains("libtext")); + break; + case Compiler.dartDevc: + requestShouldSucceed("main.dart.js", contains("main.dart.bootstrap")); + requestShouldSucceed("main.dart.bootstrap.js", contains("web__main")); + requestShouldSucceed( + "web__main.js", allOf(contains("foo"), contains("lib"))); + requestShouldSucceed("packages/foo/lib__foo.js", contains("footext")); + requestShouldSucceed("packages/myapp/lib__lib.js", contains("libtext")); + break; + } endPubServe(); }); } diff --git a/test/compiler/compiler_flag_test.dart b/test/compiler/compiler_flag_test.dart index f6da3851..57763e6d 100644 --- a/test/compiler/compiler_flag_test.dart +++ b/test/compiler/compiler_flag_test.dart @@ -11,9 +11,10 @@ import 'package:pub/src/exit_codes.dart'; import '../descriptor.dart' as d; import '../test_pub.dart'; import '../serve/utils.dart'; +import 'utils.dart'; main() { - integration("compiler flag switches compilers", () { + integrationWithCompiler("compiler flag switches compilers", (compiler) { d.dir(appPath, [ d.appPubspec(), d.dir("lib", [ @@ -31,22 +32,36 @@ main() { ]).create(); pubGet(); - pubServe(args: ['--compiler', 'dartdevc']); - requestShouldSucceed( - 'packages/$appPath/$moduleConfigName', contains('lib__hello')); - requestShouldSucceed(moduleConfigName, contains('web__main')); - requestShouldSucceed('packages/$appPath/lib__hello.unlinked.sum', null); - requestShouldSucceed('web__main.unlinked.sum', null); - requestShouldSucceed('packages/$appPath/lib__hello.linked.sum', null); - requestShouldSucceed('web__main.linked.sum', null); - requestShouldSucceed('packages/$appPath/lib__hello.js', contains('hello')); - requestShouldSucceed('web__main.js', contains('hello')); - requestShouldSucceed('dart_sdk.js', null); - requestShouldSucceed('require.js', null); - // TODO(jakemac53): Not implemented yet, update once available. - requestShould404('main.dart.js'); + pubServe(compiler: compiler); + switch (compiler) { + case Compiler.dartDevc: + requestShouldSucceed( + 'packages/$appPath/$moduleConfigName', contains('lib__hello')); + requestShouldSucceed(moduleConfigName, contains('web__main')); + requestShouldSucceed('packages/$appPath/lib__hello.unlinked.sum', null); + requestShouldSucceed('web__main.unlinked.sum', null); + requestShouldSucceed('packages/$appPath/lib__hello.linked.sum', null); + requestShouldSucceed('web__main.linked.sum', null); + requestShouldSucceed( + 'packages/$appPath/lib__hello.js', contains('hello')); + requestShouldSucceed( + 'packages/$appPath/lib__hello.js.map', contains('lib__hello.js')); + requestShouldSucceed('web__main.js', contains('hello')); + requestShouldSucceed('web__main.js.map', contains('web__main.js')); + requestShouldSucceed('dart_sdk.js', null); + requestShouldSucceed('require.js', null); + requestShouldSucceed('main.dart.js', null); + break; + case Compiler.dart2JS: + requestShouldSucceed('main.dart.js', null); + requestShould404('web__main.js'); + break; + case Compiler.none: + requestShould404('main.dart.js'); + break; + } endPubServe(); - }); + }, compilers: Compiler.all); integration("invalid compiler flag gives an error", () { d.dir(appPath, [ diff --git a/test/compiler/compiles_entrypoints_in_root_package_test.dart b/test/compiler/compiles_entrypoints_in_root_package_test.dart index ebfcca9b..5321c3a0 100644 --- a/test/compiler/compiles_entrypoints_in_root_package_test.dart +++ b/test/compiler/compiles_entrypoints_in_root_package_test.dart @@ -9,9 +9,11 @@ import 'package:scheduled_test/scheduled_test.dart'; import '../descriptor.dart' as d; import '../test_pub.dart'; +import 'utils.dart'; main() { - integration("compiles Dart entrypoints in root package to JS", () { + integrationWithCompiler("compiles Dart entrypoints in root package to JS", + (compiler) { d.dir(appPath, [ d.appPubspec(), d.dir('benchmark', [ @@ -35,9 +37,13 @@ main() { ]).create(); pubGet(); - schedulePub( - args: ["build", "benchmark", "foo", "web"], - output: new RegExp(r'Built 6 files to "build".')); + schedulePub(args: [ + "build", + "benchmark", + "foo", + "web", + "--compiler=${compiler.name}" + ], output: new RegExp(r'Built [\d]+ files to "build".')); d.dir(appPath, [ d.dir('build', [ diff --git a/test/compiler/compiles_generated_dart_file_test.dart b/test/compiler/compiles_generated_dart_file_test.dart index 9a09be83..f18fac46 100644 --- a/test/compiler/compiles_generated_dart_file_test.dart +++ b/test/compiler/compiles_generated_dart_file_test.dart @@ -7,9 +7,10 @@ import 'package:scheduled_test/scheduled_test.dart'; import '../descriptor.dart' as d; import '../test_pub.dart'; import '../serve/utils.dart'; +import 'utils.dart'; main() { - integration("compiles a generated Dart file to JS", () { + integrationWithCompiler("compiles a generated Dart file to JS", (compiler) { serveBarback(); d.dir(appPath, [ @@ -31,8 +32,15 @@ void main() => print(TOKEN); ]).create(); pubGet(); - pubServe(); - requestShouldSucceed("main.dart.js", contains("(before, munge)")); + pubServe(compiler: compiler); + switch (compiler) { + case Compiler.dart2JS: + requestShouldSucceed("main.dart.js", contains("(before, munge)")); + break; + case Compiler.dartDevc: + requestShouldSucceed("web__main.js", contains("(before, munge)")); + break; + } endPubServe(); }); } diff --git a/test/compiler/compiles_generated_file_from_dependency_outside_web_test.dart b/test/compiler/compiles_generated_file_from_dependency_outside_web_test.dart index ab7b96ec..84cecefe 100644 --- a/test/compiler/compiles_generated_file_from_dependency_outside_web_test.dart +++ b/test/compiler/compiles_generated_file_from_dependency_outside_web_test.dart @@ -7,12 +7,13 @@ import 'package:scheduled_test/scheduled_test.dart'; import '../descriptor.dart' as d; import '../test_pub.dart'; import '../serve/utils.dart'; +import 'utils.dart'; main() { // This is a regression test for issue #17198. - integration( + integrationWithCompiler( "compiles a Dart file that imports a generated file to JS " - "outside web/", () { + "outside web/", (compiler) { serveBarback(); d.dir(appPath, [ @@ -39,9 +40,17 @@ const TOKEN = "before"; ]).create(); pubGet(); - pubServe(args: ["test"]); - requestShouldSucceed("main.dart.js", contains("(before, munge)"), - root: "test"); + pubServe(args: ["test"], compiler: compiler); + switch (compiler) { + case Compiler.dart2JS: + requestShouldSucceed("main.dart.js", contains("(before, munge)"), + root: "test"); + break; + case Compiler.dartDevc: + requestShouldSucceed("test__main.js", contains("(before, munge)"), + root: "test"); + break; + } endPubServe(); }); } diff --git a/test/compiler/compiles_generated_file_from_dependency_test.dart b/test/compiler/compiles_generated_file_from_dependency_test.dart index 5713720c..4e68478e 100644 --- a/test/compiler/compiles_generated_file_from_dependency_test.dart +++ b/test/compiler/compiles_generated_file_from_dependency_test.dart @@ -7,11 +7,12 @@ import 'package:scheduled_test/scheduled_test.dart'; import '../descriptor.dart' as d; import '../test_pub.dart'; import '../serve/utils.dart'; +import 'utils.dart'; main() { - integration( + integrationWithCompiler( "compiles a Dart file that imports a generated file in another " - "package to JS", () { + "package to JS", (compiler) { serveBarback(); d.dir("foo", [ @@ -47,8 +48,17 @@ main() => print(foo()); ]).create(); pubGet(); - pubServe(); - requestShouldSucceed("main.dart.js", contains("(before, munge)")); + pubServe(compiler: compiler); + switch (compiler) { + case Compiler.dart2JS: + requestShouldSucceed("main.dart.js", contains("(before, munge)")); + break; + case Compiler.dartDevc: + requestShouldSucceed("web__main.js", contains("foo")); + requestShouldSucceed( + "packages/foo/lib__foo.js", contains("(before, munge)")); + break; + } endPubServe(); }); } diff --git a/test/compiler/compiles_imported_generated_file_test.dart b/test/compiler/compiles_imported_generated_file_test.dart index 4b774e35..18d548ca 100644 --- a/test/compiler/compiles_imported_generated_file_test.dart +++ b/test/compiler/compiles_imported_generated_file_test.dart @@ -7,9 +7,11 @@ import 'package:scheduled_test/scheduled_test.dart'; import '../descriptor.dart' as d; import '../test_pub.dart'; import '../serve/utils.dart'; +import 'utils.dart'; main() { - integration("compiles a Dart file that imports a generated file to JS", () { + integrationWithCompiler( + "compiles a Dart file that imports a generated file to JS", (compiler) { serveBarback(); d.dir(appPath, [ @@ -36,8 +38,15 @@ const TOKEN = "before"; ]).create(); pubGet(); - pubServe(); - requestShouldSucceed("main.dart.js", contains("(before, munge)")); + pubServe(compiler: compiler); + switch (compiler) { + case Compiler.dart2JS: + requestShouldSucceed("main.dart.js", contains("(before, munge)")); + break; + case Compiler.dartDevc: + requestShouldSucceed("web__main.js", contains("(before, munge)")); + break; + } endPubServe(); }); } diff --git a/test/compiler/converts_isolate_entrypoint_in_web_test.dart b/test/compiler/converts_isolate_entrypoint_in_web_test.dart index 6c8f91f5..a6f30d39 100644 --- a/test/compiler/converts_isolate_entrypoint_in_web_test.dart +++ b/test/compiler/converts_isolate_entrypoint_in_web_test.dart @@ -10,22 +10,33 @@ import 'package:scheduled_test/scheduled_test.dart'; import '../descriptor.dart' as d; import '../test_pub.dart'; import '../serve/utils.dart'; +import 'utils.dart'; main() { - integration("converts a Dart isolate entrypoint in web to JS", () { + integrationWithCompiler("converts a Dart isolate entrypoint in web to JS", + (compiler) { d.dir(appPath, [ d.appPubspec(), d.dir("web", [ d.file( "isolate.dart", - "void main(List<String> args, SendPort " - "sendPort) => print('hello');") + """ + import 'dart:isolate'; + void main(List<String> args, SendPort sendPort) => print('hello'); + """) ]) ]).create(); pubGet(); - pubServe(); - requestShouldSucceed("isolate.dart.js", contains("hello")); + pubServe(compiler: compiler); + switch (compiler) { + case Compiler.dart2JS: + requestShouldSucceed("isolate.dart.js", contains("hello")); + break; + case Compiler.dartDevc: + requestShouldSucceed("web__isolate.js", contains("hello")); + break; + } endPubServe(); }); } diff --git a/test/compiler/environment_constant_test.dart b/test/compiler/environment_constant_test.dart index 54077704..8e2853e4 100644 --- a/test/compiler/environment_constant_test.dart +++ b/test/compiler/environment_constant_test.dart @@ -10,9 +10,10 @@ import 'package:scheduled_test/scheduled_test.dart'; import '../descriptor.dart' as d; import '../serve/utils.dart'; import '../test_pub.dart'; +import 'utils.dart'; main() { - group("passes environment constants to dart2js", () { + group("passes environment constants to", () { setUp(() { d.dir(appPath, [ d.appPubspec(), @@ -23,29 +24,48 @@ main() { ]).create(); }); - integration('from "pub build"', () { + integrationWithCompiler('from "pub build"', (compiler) { pubGet(); - schedulePub( - args: ["build", "--define", "name=fblthp"], - output: new RegExp(r'Built 1 file to "build".')); + schedulePub(args: [ + "build", + "--define", + "name=fblthp", + "--compiler=${compiler.name}" + ], output: new RegExp(r'Built [\d]+ file[s]? to "build".')); + + var expectedFile; + switch (compiler) { + case Compiler.dart2JS: + expectedFile = d.matcherFile('file.dart.js', contains('fblthp')); + break; + case Compiler.dartDevc: + expectedFile = d.matcherFile('web__file.js', contains('fblthp')); + break; + } d.dir(appPath, [ d.dir('build', [ - d.dir('web', [ - d.matcherFile('file.dart.js', contains('fblthp')), - ]) + d.dir('web', [expectedFile]) ]) ]).validate(); }); - integration('from "pub serve"', () { + integrationWithCompiler('from "pub serve"', (compiler) { pubGet(); - pubServe(args: ["--define", "name=fblthp"]); - requestShouldSucceed("file.dart.js", contains("fblthp")); + pubServe(args: ["--define", "name=fblthp"], compiler: compiler); + switch (compiler) { + case Compiler.dart2JS: + requestShouldSucceed("file.dart.js", contains("fblthp")); + break; + case Compiler.dartDevc: + requestShouldSucceed("web__file.js", contains("fblthp")); + break; + } endPubServe(); }); - integration('which takes precedence over the pubspec', () { + integrationWithCompiler('which takes precedence over the pubspec', + (compiler) { d.dir(appPath, [ d.pubspec({ "name": "myapp", @@ -60,9 +80,17 @@ main() { ]).create(); pubGet(); - pubServe(args: ["--define", "name=fblthp"]); - requestShouldSucceed("file.dart.js", - allOf([contains("fblthp"), isNot(contains("slartibartfast"))])); + pubServe(args: ["--define", "name=fblthp"], compiler: compiler); + switch (compiler) { + case Compiler.dart2JS: + requestShouldSucceed("file.dart.js", + allOf([contains("fblthp"), isNot(contains("slartibartfast"))])); + break; + case Compiler.dartDevc: + requestShouldSucceed("web__file.js", + allOf([contains("fblthp"), isNot(contains("slartibartfast"))])); + break; + } endPubServe(); }); }); diff --git a/test/compiler/ignores_entrypoint_in_dependency_test.dart b/test/compiler/ignores_entrypoint_in_dependency_test.dart index 53c075a4..8ea407b1 100644 --- a/test/compiler/ignores_entrypoint_in_dependency_test.dart +++ b/test/compiler/ignores_entrypoint_in_dependency_test.dart @@ -5,9 +5,11 @@ import '../descriptor.dart' as d; import '../test_pub.dart'; import '../serve/utils.dart'; +import 'utils.dart'; main() { - integration("ignores a Dart entrypoint in a dependency", () { + integrationWithCompiler("ignores a Dart entrypoint in a dependency", + (compiler) { d.dir("foo", [ d.libPubspec("foo", "0.0.1"), d.dir("lib", [d.file("lib.dart", "main() => print('foo');")]) @@ -20,7 +22,7 @@ main() { ]).create(); pubGet(); - pubServe(); + pubServe(compiler: compiler); requestShould404("web/packages/foo/lib.dart.js"); endPubServe(); }); diff --git a/test/compiler/ignores_entrypoints_in_lib_test.dart b/test/compiler/ignores_entrypoints_in_lib_test.dart index 23f25fc9..c5f36d37 100644 --- a/test/compiler/ignores_entrypoints_in_lib_test.dart +++ b/test/compiler/ignores_entrypoints_in_lib_test.dart @@ -7,6 +7,7 @@ import 'package:scheduled_test/scheduled_test.dart'; import '../descriptor.dart' as d; import '../test_pub.dart'; import '../serve/utils.dart'; +import 'utils.dart'; main() { setUp(() { @@ -21,21 +22,23 @@ main() { ]).create(); }); - integration("build ignores Dart entrypoints in lib", () { + integrationWithCompiler("build ignores Dart entrypoints in lib", (compiler) { pubGet(); schedulePub( - args: ["build", "--all"], - output: new RegExp(r'Built 1 file to "build".')); + args: ["build", "--all", "--compiler=${compiler.name}"], + output: new RegExp(r'Built [\d]+ files? to "build".')); d.dir(appPath, [ - d.dir('build', [d.nothing('lib')]) + d.dir('build', [ + d.nothing('lib'), + ]) ]).validate(); }); - integration("serve ignores Dart entrypoints in lib", () { + integrationWithCompiler("serve ignores Dart entrypoints in lib", (compiler) { pubGet(); pubServe(); - requestShould404("packages/myapp/main.dart.js"); + requestShould404("packages/myapp/file.dart.js"); endPubServe(); }); } diff --git a/test/compiler/ignores_non_entrypoint_dart_files_test.dart b/test/compiler/ignores_non_entrypoint_dart_files_test.dart index 04416ca4..3e678e43 100644 --- a/test/compiler/ignores_non_entrypoint_dart_files_test.dart +++ b/test/compiler/ignores_non_entrypoint_dart_files_test.dart @@ -7,6 +7,7 @@ import 'package:scheduled_test/scheduled_test.dart'; import '../descriptor.dart' as d; import '../test_pub.dart'; import '../serve/utils.dart'; +import 'utils.dart'; main() { setUp(() { @@ -21,19 +22,31 @@ main() { ]).create(); }); - integration("build ignores non-entrypoint Dart files", () { + integrationWithCompiler("build ignores non-entrypoint Dart files", + (compiler) { pubGet(); schedulePub( - args: ["build"], output: new RegExp(r'Built 0 files to "build".')); + args: ["build", "--compiler=${compiler.name}"], + output: new RegExp(r'Built [\d]+ files? to "build".')); + var expectedWebDir; + switch (compiler) { + case Compiler.dart2JS: + expectedWebDir = d.nothing('web'); + break; + case Compiler.dartDevc: + expectedWebDir = d.dir('web', [d.file('.moduleConfig', '[]')]); + break; + } d.dir(appPath, [ - d.dir('build', [d.nothing('web')]) + d.dir('build', [expectedWebDir]) ]).validate(); }); - integration("serve ignores non-entrypoint Dart files", () { + integrationWithCompiler("serve ignores non-entrypoint Dart files", + (compiler) { pubGet(); - pubServe(); + pubServe(compiler: compiler); requestShould404("file1.dart.js"); requestShould404("file2.dart.js"); requestShould404("file3.dart.js"); diff --git a/test/compiler/includes_source_maps_in_debug_test.dart b/test/compiler/includes_source_maps_in_debug_test.dart index 8977b717..c048748e 100644 --- a/test/compiler/includes_source_maps_in_debug_test.dart +++ b/test/compiler/includes_source_maps_in_debug_test.dart @@ -6,28 +6,66 @@ import 'package:scheduled_test/scheduled_test.dart'; import '../descriptor.dart' as d; import '../test_pub.dart'; +import 'utils.dart'; main() { - integration("includes source map URLs in a debug build", () { + integrationWithCompiler("includes source map URLs in a debug build", + (compiler) { d.dir(appPath, [ d.appPubspec(), - d.dir("web", [d.file("main.dart", "void main() => print('hello');")]) + d.dir("lib", [d.file("message.dart", "String get message => 'hello';")]), + d.dir("web", [ + d.file( + "main.dart", + """ + import "package:$appPath/message.dart"; + void main() => print(message); + """) + ]) ]).create(); pubGet(); schedulePub( - args: ["build", "--mode", "debug"], + args: ["build", "--mode", "debug", "--compiler=${compiler.name}"], output: new RegExp(r'Built \d+ files to "build".'), exitCode: 0); - d.dir(appPath, [ - d.dir('build', [ - d.dir('web', [ - d.matcherFile( - 'main.dart.js', contains("# sourceMappingURL=main.dart.js.map")), - d.matcherFile('main.dart.js.map', contains('"file": "main.dart.js"')) - ]) - ]) - ]).validate(); + switch (compiler) { + case Compiler.dart2JS: + d.dir(appPath, [ + d.dir('build', [ + d.dir('web', [ + d.matcherFile('main.dart.js', + contains("# sourceMappingURL=main.dart.js.map")), + d.matcherFile( + 'main.dart.js.map', contains('"file": "main.dart.js"')) + ]) + ]) + ]).validate(); + break; + case Compiler.dartDevc: + d.dir(appPath, [ + d.dir('build', [ + d.dir('web', [ + d.dir('packages', [ + d.dir(appPath, [ + d.matcherFile('lib__message.js', + contains("# sourceMappingURL=lib__message.js.map")), + d.matcherFile('lib__message.js.map', + contains('"file":"lib__message.js"')), + ]), + ]), + d.matcherFile('web__main.js', + contains("# sourceMappingURL=web__main.js.map")), + d.matcherFile( + 'web__main.js.map', contains('"file":"web__main.js"')), + // This exists to make package:test happy, but is fake (no + // original dart file to map to). + d.matcherFile('main.dart.js.map', anything), + ]) + ]) + ]).validate(); + break; + } }); } diff --git a/test/compiler/omits_source_map_in_release_test.dart b/test/compiler/omits_source_map_in_release_test.dart index a8e4502f..df4b1c5b 100644 --- a/test/compiler/omits_source_map_in_release_test.dart +++ b/test/compiler/omits_source_map_in_release_test.dart @@ -2,26 +2,62 @@ // 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:scheduled_test/scheduled_test.dart'; + import '../descriptor.dart' as d; import '../test_pub.dart'; +import 'utils.dart'; main() { - integration("omits source maps from a release build", () { + integrationWithCompiler("omits source maps from a release build", (compiler) { d.dir(appPath, [ d.appPubspec(), - d.dir("web", [d.file("main.dart", "void main() => print('hello');")]) + d.dir("lib", [d.file("message.dart", "String get message => 'hello';")]), + d.dir("web", [ + d.file( + "main.dart", + """ + import "package:$appPath/message.dart"; + void main() => print(message); + """) + ]) ]).create(); pubGet(); schedulePub( - args: ["build"], - output: new RegExp(r'Built 1 file to "build".'), + args: ["build", "--compiler=${compiler.name}"], + output: new RegExp(r'Built \d+ files? to "build".'), exitCode: 0); - d.dir(appPath, [ - d.dir('build', [ - d.dir('web', [d.nothing('main.dart.js.map')]) - ]) - ]).validate(); + switch (compiler) { + case Compiler.dart2JS: + d.dir(appPath, [ + d.dir('build', [ + d.dir('web', [ + d.nothing('main.dart.js.map'), + ]) + ]) + ]).validate(); + break; + case Compiler.dartDevc: + d.dir(appPath, [ + d.dir('build', [ + d.dir('web', [ + d.dir('packages', [ + d.dir(appPath, [ + d.matcherFile( + 'lib__message.js', + isNot( + contains("# sourceMappingURL=lib__message.js.map"))), + d.nothing('lib__message.js.map'), + ]), + ]), + d.nothing('main.dart.js.map'), + d.nothing('web__main.js.map'), + ]) + ]) + ]).validate(); + break; + } }); } diff --git a/test/compiler/reports_dart_parse_errors_test.dart b/test/compiler/reports_dart_parse_errors_test.dart index f8946462..4aabb792 100644 --- a/test/compiler/reports_dart_parse_errors_test.dart +++ b/test/compiler/reports_dart_parse_errors_test.dart @@ -12,31 +12,48 @@ import 'package:scheduled_test/scheduled_stream.dart'; import '../descriptor.dart' as d; import '../test_pub.dart'; +import 'utils.dart'; main() { - integration("reports Dart parse errors", () { + integrationWithCompiler("reports Dart parse errors", (compiler) { d.dir(appPath, [ d.appPubspec(), d.dir('web', [ d.file('file.txt', 'contents'), - d.file('file.dart', 'void void;'), - d.dir('subdir', [d.file('subfile.dart', 'void void;')]) + d.file('file.dart', 'void main() {}; void void;'), + d.dir('subdir', [d.file('subfile.dart', 'void main() {}; void void;')]) ]) ]).create(); pubGet(); - var pub = startPub(args: ["build"]); + var pub = startPub(args: ["build", "--compiler", compiler.name]); pub.stdout.expect(startsWith("Loading source assets...")); pub.stdout.expect(startsWith("Building myapp...")); - var consumeFile = consumeThrough(inOrder([ - "[Error from Dart2JS]:", - startsWith(p.join("web", "file.dart") + ":") - ])); - var consumeSubfile = consumeThrough(inOrder([ - "[Error from Dart2JS]:", - startsWith(p.join("web", "subdir", "subfile.dart") + ":") - ])); + var consumeFile; + var consumeSubfile; + switch (compiler) { + case Compiler.dart2JS: + consumeFile = consumeThrough(inOrder([ + "[Error from Dart2JS]:", + startsWith(p.join("web", "file.dart") + ":") + ])); + consumeSubfile = consumeThrough(inOrder([ + "[Error from Dart2JS]:", + startsWith(p.join("web", "subdir", "subfile.dart") + ":") + ])); + break; + case Compiler.dartDevc: + consumeFile = consumeThrough(inOrder([ + startsWith("Error compiling dartdevc module:"), + contains(p.join("web", "file.dart")) + ])); + consumeSubfile = consumeThrough(inOrder([ + startsWith("Error compiling dartdevc module:"), + contains(p.join("web", "subdir", "subfile.dart")) + ])); + break; + } // It's nondeterministic what order the dart2js transformers start running, // so we allow the error messages to be emitted in either order. diff --git a/test/compiler/source_maps_are_self_contained_test.dart b/test/compiler/source_maps_are_self_contained_test.dart index d197bccf..0a227118 100644 --- a/test/compiler/source_maps_are_self_contained_test.dart +++ b/test/compiler/source_maps_are_self_contained_test.dart @@ -6,13 +6,15 @@ import 'package:scheduled_test/scheduled_test.dart'; import '../descriptor.dart' as d; import '../test_pub.dart'; +import 'utils.dart'; main() { // This test is a bit shaky. Since dart2js is free to inline things, it's // not precise as to which source libraries will actually be referenced in // the source map. But this tries to use a type from a package and validate // that its source ends up in the source map with a valid URI. - integration("Source maps URIs for files in packages are self-contained", () { + integrationWithCompiler( + "Source maps URIs for files in packages are self-contained", (compiler) { d.dir("foo", [ d.libPubspec("foo", "0.0.1"), d.dir("lib", [ @@ -53,13 +55,14 @@ main() { pubGet(); schedulePub( - args: ["build", "--mode", "debug"], + args: ["build", "--mode", "debug", "--compiler", compiler.name], output: new RegExp(r'Built \d+ files to "build".'), exitCode: 0); - d.dir(appPath, [ - d.dir("build", [ - d.dir("web", [ + var expectedWebDir; + switch (compiler) { + case Compiler.dart2JS: + expectedWebDir = d.dir("web", [ d.matcherFile( "main.dart.js.map", // Note: we include the quotes to ensure this is the full URL path @@ -67,13 +70,31 @@ main() { contains(r'"packages/foo/foo.dart"')), d.dir("sub", [ d.matcherFile( - "main2.dart.js.map", contains(r'"../packages/foo/foo.dart"')) + "main2.dart.js.map", contains(r'"../packages/foo/foo.dart"')), ]), d.dir("packages", [ - d.dir(r"foo", [d.matcherFile("foo.dart", contains("foo() {"))]) - ]) - ]) - ]) + d.dir(r"foo", [d.matcherFile("foo.dart", contains("foo() {"))]), + ]), + ]); + break; + case Compiler.dartDevc: + expectedWebDir = d.dir("web", [ + d.dir("packages", [ + d.dir("foo", [ + d.matcherFile("foo.dart", contains("foo() {")), + d.matcherFile('lib__foo.js', contains("foo.dart")), + ]), + ]), + d.matcherFile("web__main.js.map", contains(r'"main.dart"')), + d.matcherFile( + "web__sub__main2.js.map", contains(r'"sub/main2.dart"')), + ]); + + break; + } + + d.dir(appPath, [ + d.dir("build", [expectedWebDir]) ]).validate(); }); } diff --git a/test/compiler/supports_configuration_with_build_test.dart b/test/compiler/supports_configuration_with_build_test.dart index 14e8c083..5f993a0c 100644 --- a/test/compiler/supports_configuration_with_build_test.dart +++ b/test/compiler/supports_configuration_with_build_test.dart @@ -11,11 +11,12 @@ import 'package:scheduled_test/scheduled_test.dart'; import '../descriptor.dart' as d; import '../test_pub.dart'; +import 'utils.dart'; main() { - integration( + integrationWithCompiler( "compiles dart.js and interop.js next to entrypoints when " - "dartjs is explicitly configured", () { + "dartjs is explicitly configured", (compiler) { serve([ d.dir('api', [ d.dir('packages', [ @@ -70,8 +71,8 @@ main() { pubGet(); schedulePub( - args: ["build"], - output: new RegExp(r'Built 3 files to "build".'), + args: ["build", "--compiler", compiler.name], + output: new RegExp(r'Built \d+ files? to "build".'), exitCode: 0); d.dir(appPath, [ diff --git a/test/compiler/utils.dart b/test/compiler/utils.dart new file mode 100644 index 00000000..b2bee254 --- /dev/null +++ b/test/compiler/utils.dart @@ -0,0 +1,20 @@ +// Copyright (c) 2017, the Dart project authors. Please see the AUTHORS d.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:pub/src/barback/compiler.dart'; + +import '../test_pub.dart'; + +// For convenience, otherwise we need this import in pretty much all tests. +export 'package:pub/src/barback/compiler.dart'; + +/// Runs an integration test once for each [Compiler] in [compilers], defaulting +/// to [Compiler.dart2JS] and [Compiler.dartDevc]. +void integrationWithCompiler(String name, void testFn(Compiler compiler), + {List<Compiler> compilers}) { + compilers ??= [Compiler.dart2JS, Compiler.dartDevc]; + for (var compiler in compilers) { + integration('--compiler=${compiler.name} $name', () => testFn(compiler)); + } +} diff --git a/test/serve/utils.dart b/test/serve/utils.dart index 1b37ae09..4a0411ed 100644 --- a/test/serve/utils.dart +++ b/test/serve/utils.dart @@ -7,6 +7,7 @@ import 'dart:convert'; import 'dart:io'; import 'package:http/http.dart' as http; +import 'package:pub/src/barback/compiler.dart'; import 'package:pub/src/utils.dart'; import 'package:scheduled_test/scheduled_process.dart'; import 'package:scheduled_test/scheduled_stream.dart'; @@ -147,14 +148,17 @@ class DartTransformer extends Transformer { /// /// Returns the `pub serve` process. ScheduledProcess startPubServe( - {Iterable<String> args, bool createWebDir: true}) { + {Iterable<String> args, bool createWebDir: true, Compiler compiler}) { var pubArgs = [ "serve", "--port=0", // Use port 0 to get an ephemeral port. "--force-poll", "--admin-port=0", // Use port 0 to get an ephemeral port. - "--log-admin-url" + "--log-admin-url", ]; + if (compiler != null) { + pubArgs.add("--compiler=${compiler.name}"); + } if (args != null) pubArgs.addAll(args); @@ -169,8 +173,10 @@ ScheduledProcess startPubServe( /// so pub doesn't complain about having nothing to serve. /// /// Returns the `pub serve` process. -ScheduledProcess pubServe({bool createWebDir: true, Iterable<String> args}) { - _pubServer = startPubServe(args: args, createWebDir: createWebDir); +ScheduledProcess pubServe( + {bool createWebDir: true, Iterable<String> args, Compiler compiler}) { + _pubServer = + startPubServe(args: args, createWebDir: createWebDir, compiler: compiler); _portsCompleter = new Completer(); currentSchedule.onComplete.schedule(() { -- GitLab