| Index: sdk/lib/_internal/pub/lib/src/barback/transformer_isolate.dart
|
| diff --git a/sdk/lib/_internal/pub/lib/src/barback/transformer_isolate.dart b/sdk/lib/_internal/pub/lib/src/barback/transformer_isolate.dart
|
| new file mode 100644
|
| index 0000000000000000000000000000000000000000..2cb9e6d7cf70574564c57a5d076b7731acd64391
|
| --- /dev/null
|
| +++ b/sdk/lib/_internal/pub/lib/src/barback/transformer_isolate.dart
|
| @@ -0,0 +1,117 @@
|
| +// 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 pub.transformer_isolate;
|
| +
|
| +import 'dart:async';
|
| +import 'dart:convert';
|
| +import 'dart:isolate';
|
| +
|
| +import 'package:barback/barback.dart';
|
| +
|
| +import '../../../asset/dart/serialize.dart';
|
| +import '../barback.dart';
|
| +import '../dart.dart' as dart;
|
| +import '../log.dart' as log;
|
| +import '../utils.dart';
|
| +import 'asset_environment.dart';
|
| +import 'barback_server.dart';
|
| +import 'foreign_transformer.dart';
|
| +import 'transformer_config.dart';
|
| +import 'transformer_id.dart';
|
| +
|
| +/// A wrapper for an isolate from which transformer plugins can be instantiated.
|
| +class TransformerIsolate {
|
| + /// The port used to communicate with the wrapped isolate.
|
| + final SendPort _port;
|
| +
|
| + /// A map indicating the barback server URLs for each [TransformerId] that's
|
| + /// loaded in the wrapped isolate.
|
| + ///
|
| + /// A barback server URL is the URL for the library that the given id
|
| + /// identifies. For example, the URL for "polymer/src/mirrors_remover" might
|
| + /// be "http://localhost:56234/packages/polymer/src/mirrors_remover.dart".
|
| + final Map<TransformerId, Uri> _idsToUrls;
|
| +
|
| + /// The barback mode for this run of pub.
|
| + final BarbackMode _mode;
|
| +
|
| + /// Spawns an isolate that loads all transformer libraries defined by [ids].
|
| + ///
|
| + /// This doesn't actually instantiate any transformers, since a
|
| + /// [TransformerId] doesn't define the transformers' configuration. The
|
| + /// transformers can be constructed using [create].
|
| + static Future<TransformerIsolate> spawn(AssetEnvironment environment,
|
| + BarbackServer transformerServer, List<TransformerId> ids) {
|
| + return mapFromIterableAsync(ids, value: (id) {
|
| + return id.getAssetId(environment.barback);
|
| + }).then((idsToAssetIds) {
|
| + var baseUrl = transformerServer.url;
|
| + var idsToUrls = mapMap(idsToAssetIds, value: (id, assetId) {
|
| + var path = assetId.path.replaceFirst('lib/', '');
|
| + // TODO(nweiz): load from a "package:" URI when issue 12474 is fixed.
|
| + return baseUrl.resolve('packages/${id.package}/$path');
|
| + });
|
| +
|
| + var code = new StringBuffer();
|
| + code.writeln("import 'dart:isolate';");
|
| +
|
| + for (var url in idsToUrls.values) {
|
| + code.writeln("import '$url';");
|
| + }
|
| +
|
| + code.writeln("import "
|
| + "r'$baseUrl/packages/\$pub/transformer_isolate.dart';");
|
| + code.writeln(
|
| + "void main(_, SendPort replyTo) => loadTransformers(replyTo);");
|
| +
|
| + log.fine("Loading transformers from $ids");
|
| +
|
| + var port = new ReceivePort();
|
| + return dart.runInIsolate(code.toString(), port.sendPort)
|
| + .then((_) => port.first)
|
| + .then((sendPort) {
|
| + return new TransformerIsolate._(sendPort, environment.mode, idsToUrls);
|
| + }).catchError((error, stackTrace) {
|
| + if (error is! CrossIsolateException) throw error;
|
| + if (error.type != 'IsolateSpawnException') throw error;
|
| +
|
| + // TODO(nweiz): don't parse this as a string once issues 12617 and 12689
|
| + // are fixed.
|
| + var firstErrorLine = error.message.split('\n')[1];
|
| + var missingTransformer = idsToUrls.keys.firstWhere((id) =>
|
| + firstErrorLine.startsWith("Failure getting ${idsToUrls[id]}:"),
|
| + orElse: () => throw error);
|
| + var packageUri = idToPackageUri(idsToAssetIds[missingTransformer]);
|
| +
|
| + // If there was an IsolateSpawnException and the import that actually
|
| + // failed was the one we were loading transformers from, throw an
|
| + // application exception with a more user-friendly message.
|
| + fail('Transformer library "$packageUri" not found.',
|
| + error, stackTrace);
|
| + });
|
| + });
|
| + }
|
| +
|
| + TransformerIsolate._(this._port, this._mode, this._idsToUrls);
|
| +
|
| + /// Instantiate the transformers in the [config.id] with
|
| + /// [config.configuration].
|
| + ///
|
| + /// If there are no transformers defined in the given library, this will
|
| + /// return an empty set.
|
| + Future<Set<Transformer>> create(TransformerConfig config) {
|
| + return call(_port, {
|
| + 'library': _idsToUrls[config.id].toString(),
|
| + 'mode': _mode.name,
|
| + 'configuration': JSON.encode(config.configuration)
|
| + }).then((transformers) {
|
| + transformers = transformers.map(
|
| + (transformer) => deserializeTransformerLike(transformer, config))
|
| + .toSet();
|
| + log.fine("Transformers from $config: $transformers");
|
| + return transformers;
|
| + });
|
| + }
|
| +}
|
|
|