blob: 9aabc8b4964f8c63d924740a3992996f38586ca1 [file] [log] [blame]
// 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.
import 'package:collection/collection.dart';
import 'barback/transformer_cache.dart';
import 'compiler.dart';
import 'entrypoint.dart';
import 'lock_file.dart';
import 'package.dart';
import 'solver/version_solver.dart';
import 'source/cached.dart';
/// A holistic view of the entire transitive dependency graph for an entrypoint.
class PackageGraph {
/// The entrypoint.
final Entrypoint entrypoint;
/// The entrypoint's lockfile.
///
/// This describes the sources and resolved descriptions of everything in
/// [packages].
final LockFile lockFile;
/// The transitive dependencies of the entrypoint (including itself).
///
/// This may not include all transitive dependencies of the entrypoint if the
/// creator of the package graph knows only a subset of the packages are
/// relevant in the current context.
final Map<String, Package> packages;
/// A map of transitive dependencies for each package.
Map<String, Set<Package>> _transitiveDependencies;
/// The transformer cache, if it's been loaded.
TransformerCache _transformerCache;
PackageGraph(this.entrypoint, this.lockFile, this.packages);
/// Creates a package graph using the data from [result].
///
/// This is generally faster than loading a package graph from scratch, since
/// the packages' pubspecs are already fully-parsed.
factory PackageGraph.fromSolveResult(
Entrypoint entrypoint, SolveResult result) {
var packages = new Map<String, Package>.fromIterable(result.packages,
key: (id) => id.name,
value: (id) {
if (id.name == entrypoint.root.name) return entrypoint.root;
return new Package(result.pubspecs[id.name],
entrypoint.cache.source(id.source).getDirectory(id));
});
return new PackageGraph(entrypoint, result.lockFile, packages);
}
/// Loads the transformer cache for this graph.
///
/// This may only be called if [entrypoint] represents a physical package.
/// This may modify the cache.
TransformerCache loadTransformerCache() {
if (_transformerCache == null) {
if (entrypoint.root.dir == null) {
throw new StateError("Can't load the transformer cache for virtual "
"entrypoint ${entrypoint.root.name}.");
}
_transformerCache = new TransformerCache.load(this);
}
return _transformerCache;
}
/// Returns all transitive dependencies of [package].
///
/// For the entrypoint this returns all packages in [packages], which includes
/// dev and override. For any other package, it ignores dev and override
/// dependencies.
Set<Package> transitiveDependencies(String package) {
if (package == entrypoint.root.name) return packages.values.toSet();
if (_transitiveDependencies == null) {
var closure = transitiveClosure(
mapMap<String, Package, String, Iterable<String>>(packages,
value: (_, package) =>
package.dependencies.map((dep) => dep.name)));
_transitiveDependencies =
mapMap<String, Set<String>, String, Set<Package>>(closure,
value: (depender, names) {
var set = names.map((name) => packages[name]).toSet();
set.add(packages[depender]);
return set;
});
}
return _transitiveDependencies[package];
}
/// Returns whether [package], or any of its transitive dependencies, have
/// transformers that run on any of their public assets.
///
/// This is pessimistic; if any package can't be determined to be transformed,
/// this returns `true`.
bool isPackageTransformed(String packageName) {
if (_isIndividualPackageTransformed(packages[packageName])) return true;
return transitiveDependencies(packageName)
.any(_isIndividualPackageTransformed);
}
/// Returns whether [package] itself has transformers that run on any of its
/// public assets.
bool _isIndividualPackageTransformed(Package package) {
// If the caller passed in an unknown package name to isPackageTransformed,
// the package will be null.
if (package == null) return true;
if (package.name == entrypoint.root.name) {
return package.pubspec.transformers.isNotEmpty;
}
return package.pubspec.transformers.any((phase) {
return phase.any((config) => config.canTransformPublicFiles);
});
}
/// Returns whether [package] is mutable.
///
/// A package is considered to be mutable if it or any of its dependencies
/// don't come from a cached source, since the user can change its contents
/// without modifying the pub cache. Information generated from mutable
/// packages is generally not safe to cache, since it may change frequently.
bool isPackageMutable(String package) {
var id = lockFile.packages[package];
if (id == null) return true;
if (entrypoint.cache.source(id.source) is! CachedSource) return true;
return transitiveDependencies(package).any((dep) {
var depId = lockFile.packages[dep.name];
// The entrypoint package doesn't have a lockfile entry. It's always
// mutable.
if (depId == null) return true;
return entrypoint.cache.source(depId.source) is! CachedSource;
});
}
/// Returns whether [package] is static.
///
/// A package is considered to be static if it's not transformed and it came
/// from a cached source. Static packages don't need to be fully processed by
/// barback.
///
/// If [compiler] is [Compiler.dartDevc] then no package is static because the
/// transformer will be added to all packages.
///
/// Note that a static package isn't the same as an immutable package (see
/// [isPackageMutable]).
bool isPackageStatic(String package, Compiler compiler) {
var id = lockFile.packages[package];
if (id == null) return false;
if (entrypoint.cache.source(id.source) is! CachedSource) return false;
return packages[package].pubspec.transformers.isEmpty;
}
}