| Index: utils/pub/cache.dart
|
| diff --git a/utils/pub/cache.dart b/utils/pub/cache.dart
|
| new file mode 100644
|
| index 0000000000000000000000000000000000000000..4c599e02c5800976b444b3d4778bcbed9576e42a
|
| --- /dev/null
|
| +++ b/utils/pub/cache.dart
|
| @@ -0,0 +1,104 @@
|
| +// Copyright (c) 2012, 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.
|
| +
|
| +/**
|
| + * The local cache of previously installed packages.
|
| + */
|
| +class PackageCache {
|
| + /**
|
| + * The root directory where this package cache is located.
|
| + */
|
| + final String rootDir;
|
| +
|
| + // TODO(rnystrom): When packages are versioned, String here and elsewhere will
|
| + // become a name/version/(source?) tuple.
|
| + final Map<String, Package> _loadedPackages;
|
| +
|
| + /**
|
| + * Packages which are currently being asynchronously loaded.
|
| + */
|
| + final Map<String, Future<Package>> _pendingPackages;
|
| +
|
| + /**
|
| + * Creates a new package cache which is backed by the given directory on the
|
| + * user's file system.
|
| + */
|
| + PackageCache(this.rootDir)
|
| + : _loadedPackages = <Package>{},
|
| + _pendingPackages = <Future<Package>>{};
|
| +
|
| + /**
|
| + * Loads all of the packages in the cache and returns them.
|
| + */
|
| + Future<List<Package>> listAll() {
|
| + return listDir(rootDir).chain((paths) {
|
| + final packages = paths.map((path) => find(basename(path)));
|
| + return Futures.wait(packages);
|
| + });
|
| + }
|
| +
|
| + /**
|
| + * Loads the package named [name] from this cache, if present.
|
| + */
|
| + // TODO(rnystrom): What happens if the package isn't cached?
|
| + Future<Package> find(String name) {
|
| + // Use the previously loaded one.
|
| + final package = _loadedPackages[name];
|
| + if (package != null) {
|
| + return new Future.immediate(package);
|
| + }
|
| +
|
| + // If we are already in-progress loading it, re-use that one.
|
| + final pending = _pendingPackages[name];
|
| + if (pending != null) {
|
| + return pending;
|
| + }
|
| +
|
| + return _loadPackage(name);
|
| + }
|
| +
|
| + /**
|
| + * Start loading the package.
|
| + */
|
| + Future<Package> _loadPackage(String name) {
|
| + final future = _parsePubspec(name).transform((dependencies) {
|
| + final package = new Package._(this, name, dependencies);
|
| +
|
| + _pendingPackages.remove(name);
|
| + _loadedPackages[name] = package;
|
| + return package;
|
| + });
|
| +
|
| + _pendingPackages[name] = future;
|
| + return future;
|
| + }
|
| +
|
| + Future<List<String>> _parsePubspec(String name) {
|
| + final completer = new Completer<List<String>>();
|
| + final pubspecPath = join(rootDir, name, 'pubspec');
|
| +
|
| + // TODO(rnystrom): Handle the directory not existing.
|
| + // TODO(rnystrom): Error-handling.
|
| + final readFuture = readTextFile(pubspecPath);
|
| + readFuture.handleException((error) {
|
| + // If there is no pubspec, we implicitly treat that as a package with no
|
| + // dependencies.
|
| + // TODO(rnystrom): Distinguish file not found from other real errors.
|
| + completer.complete(<String>[]);
|
| + return true;
|
| + });
|
| +
|
| + readFuture.then((pubspec) {
|
| + // TODO(rnystrom): Use YAML parser when ready. For now, it's just a flat
|
| + // list of newline-separated strings.
|
| + final dependencyNames = pubspec.split('\n').
|
| + map((name) => name.trim()).
|
| + filter((name) => (name != null) && (name != ''));
|
| +
|
| + completer.complete(dependencyNames);
|
| + });
|
| +
|
| + return completer.future;
|
| + }
|
| +}
|
|
|