| Index: sdk/lib/_internal/pub/lib/src/barback/transformer_cache.dart
|
| diff --git a/sdk/lib/_internal/pub/lib/src/barback/transformer_cache.dart b/sdk/lib/_internal/pub/lib/src/barback/transformer_cache.dart
|
| new file mode 100644
|
| index 0000000000000000000000000000000000000000..0fe07e3eebb6856cba845e16e88b9af3faf93e61
|
| --- /dev/null
|
| +++ b/sdk/lib/_internal/pub/lib/src/barback/transformer_cache.dart
|
| @@ -0,0 +1,153 @@
|
| +// Copyright (c) 2014, 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.barback.transformer_cache;
|
| +
|
| +import 'package:path/path.dart' as p;
|
| +
|
| +import '../io.dart';
|
| +import '../log.dart' as log;
|
| +import '../package_graph.dart';
|
| +import '../sdk.dart' as sdk;
|
| +import '../source/cached.dart';
|
| +import '../utils.dart';
|
| +import 'asset_environment.dart';
|
| +import 'transformer_id.dart';
|
| +
|
| +/// A cache for managing a snapshot of the first "stage" of transformers to
|
| +/// load.
|
| +///
|
| +/// This uses the [_stageTransformers] notion of a stage. Transformers are
|
| +/// divided into stages for loading based on which transformers are needed to
|
| +/// load one another. For example, if a transformer T1 produces a file that's
|
| +/// imported by another transformer T2, T2 must be put in a stage after T1.
|
| +///
|
| +/// We only cache the first stage because it's the only stage whose contents are
|
| +/// independent of any configuration. Since most transformers don't import the
|
| +/// output of other transformers, many packages will only have one stage.
|
| +class TransformerCache {
|
| + final PackageGraph _graph;
|
| +
|
| + /// The set of transformer ids that were previously cached.
|
| + ///
|
| + /// If there was no previous cache, this will be empty.
|
| + Set<TransformerId> _oldTransformers;
|
| +
|
| + /// The set of transformer ids that are newly cached or re-used from the
|
| + /// previous cache.
|
| + Set<TransformerId> _newTransformers;
|
| +
|
| + /// The directory in which transformers are cached.
|
| + ///
|
| + /// This may be `null` if there's no physical entrypoint directory.
|
| + String _dir;
|
| +
|
| + /// The directory of the manifest listing which transformers were cached.
|
| + String get _manifestPath => p.join(_dir, "manifest.txt");
|
| +
|
| + /// Loads the transformer cache for [environment].
|
| + ///
|
| + /// This may modify the cache.
|
| + TransformerCache.load(PackageGraph graph)
|
| + : _graph = graph,
|
| + _dir = p.join(graph.entrypoint.root.dir, ".pub/transformers") {
|
| + _oldTransformers = _parseManifest();
|
| + }
|
| +
|
| + /// Clear the cache if it depends on any package in [changedPackages].
|
| + void clearIfOutdated(Set<String> changedPackages) {
|
| + var snapshotDependencies = unionAll(_oldTransformers.map((id) {
|
| + return _graph.transitiveDependencies(id.package)
|
| + .map((package) => package.name).toSet();
|
| + }));
|
| +
|
| + // If none of the snapshot's dependencies have changed, then we can reuse
|
| + // it.
|
| + if (!changedPackages.any(snapshotDependencies.contains)) return;
|
| +
|
| + // Otherwise, delete it.
|
| + deleteEntry(_dir);
|
| + _oldTransformers = new Set();
|
| + }
|
| +
|
| + /// Returns the path for the transformer snapshot for [transformers], or
|
| + /// `null` if the transformers shouldn't be cached.
|
| + ///
|
| + /// There may or may not exist a file at the returned path. If one does exist,
|
| + /// it can safely be used to load the stage. Otherwise, a snapshot of the
|
| + /// stage should be written there.
|
| + String snapshotPath(Set<TransformerId> transformers) {
|
| + // A transformer is considered mutable if it comes from the entrypoint
|
| + // package or a path dependency.
|
| + //
|
| + // TODO(nweiz): Consider git packages with mutable (transitive) dependencies
|
| + // to be mutable themselves.
|
| + var usesMutableTransformer = transformers.any((id) {
|
| + // The entrypoint package doesn't appear in the lockfile.
|
| + var package = _graph.lockFile.packages[id.package];
|
| + if (package == null) return true;
|
| + var source = _graph.entrypoint.cache.sources[package.source];
|
| + return source is! CachedSource;
|
| + });
|
| +
|
| + var path = p.join(_dir, "transformers.snapshot");
|
| + if (usesMutableTransformer) {
|
| + log.fine("Not caching mutable transformers.");
|
| + deleteEntry(_dir);
|
| + return null;
|
| + }
|
| +
|
| + if (!_oldTransformers.containsAll(transformers)) {
|
| + log.fine("Cached transformer snapshot is out-of-date, deleting.");
|
| + deleteEntry(path);
|
| + } else {
|
| + log.fine("Using cached transformer snapshot.");
|
| + }
|
| +
|
| + _newTransformers = transformers;
|
| + return path;
|
| + }
|
| +
|
| + /// Saves the manifest to the transformer cache.
|
| + void save() {
|
| + // If we didn't write any snapshots, there's no need to write a manifest.
|
| + if (_newTransformers == null) {
|
| + if (_dir != null) deleteEntry(_dir);
|
| + return;
|
| + }
|
| +
|
| + // We only need to rewrite the manifest if we created a new snapshot.
|
| + if (_oldTransformers.containsAll(_newTransformers)) return;
|
| +
|
| + ensureDir(_dir);
|
| + writeTextFile(_manifestPath,
|
| + "${sdk.version}\n" +
|
| + ordered(_newTransformers.map((id) => id.serialize())).join(","));
|
| + }
|
| +
|
| + /// Parses the cache manifest and returns the set of previously-cached
|
| + /// transformers.
|
| + ///
|
| + /// If the manifest indicates that the SDK version is out-of-date, this
|
| + /// deletes the existing cache. Otherwise,
|
| + Set<TransformerId> _parseManifest() {
|
| + if (!fileExists(_manifestPath)) return new Set();
|
| +
|
| + var manifest = readTextFile(_manifestPath).split("\n");
|
| +
|
| + // The first line of the manifest is the SDK version. We want to clear out
|
| + // the snapshots even if they're VM-compatible, since pub's transformer
|
| + // isolate scaffolding may have changed.
|
| + if (manifest.removeAt(0) != sdk.version.toString()) {
|
| + deleteEntry(_dir);
|
| + return new Set();
|
| + }
|
| +
|
| + /// The second line of the manifest is a list of transformer ids used to
|
| + /// create the existing snapshot.
|
| + return manifest.single.split(",")
|
| + .map((id) => new TransformerId.parse(id, null))
|
| + .toSet();
|
| + }
|
| +}
|
|
|