Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(5)

Side by Side Diff: utils/pub/entrypoint.dart

Issue 10803042: Reverting 9767. (Closed) Base URL: https://dart.googlecode.com/svn/branches/bleeding_edge/dart
Patch Set: Created 8 years, 5 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch | Annotate | Revision Log
« no previous file with comments | « no previous file | utils/pub/git_source.dart » ('j') | no next file with comments »
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
OLDNEW
1 // Copyright (c) 2012, the Dart project authors. Please see the AUTHORS file 1 // Copyright (c) 2012, the Dart project authors. Please see the AUTHORS file
2 // for details. All rights reserved. Use of this source code is governed by a 2 // for details. All rights reserved. Use of this source code is governed by a
3 // BSD-style license that can be found in the LICENSE file. 3 // BSD-style license that can be found in the LICENSE file.
4 4
5 #library('entrypoint'); 5 #library('entrypoint');
6 6
7 #import('io.dart'); 7 #import('io.dart');
8 #import('lock_file.dart');
9 #import('package.dart'); 8 #import('package.dart');
10 #import('root_source.dart'); 9 #import('root_source.dart');
11 #import('system_cache.dart'); 10 #import('system_cache.dart');
12 #import('version.dart'); 11 #import('version.dart');
13 #import('version_solver.dart'); 12 #import('version_solver.dart');
14 #import('utils.dart'); 13 #import('utils.dart');
15 14
16 /** 15 /**
17 * Pub operates over a directed graph of dependencies that starts at a root 16 * Pub operates over a directed graph of dependencies that starts at a root
18 * "entrypoint" package. This is typically the package where the current 17 * "entrypoint" package. This is typically the package where the current
(...skipping 17 matching lines...) Expand all
36 */ 35 */
37 final Package root; 36 final Package root;
38 37
39 /** 38 /**
40 * The system-wide cache which caches packages that need to be fetched over 39 * The system-wide cache which caches packages that need to be fetched over
41 * the network. 40 * the network.
42 */ 41 */
43 final SystemCache cache; 42 final SystemCache cache;
44 43
45 /** 44 /**
46 * Packages which are either currently being asynchronously installed to the 45 * Packages which have already been loaded into memory.
47 * directory, or have already been installed.
48 */ 46 */
49 final Map<PackageId, Future<PackageId>> _installs; 47 final Map<PackageId, Package> _loadedPackages;
48
49 /**
50 * Packages which are currently being asynchronously installed to the
51 * directory.
52 */
53 final Map<PackageId, Future<Package>> _pendingInstalls;
50 54
51 Entrypoint(this.root, this.cache) 55 Entrypoint(this.root, this.cache)
52 : _installs = new Map<PackageId, Future<PackageId>>(); 56 : _loadedPackages = new Map<PackageId, Package>(),
57 _pendingInstalls = new Map<PackageId, Future<Package>>();
53 58
54 /** 59 /**
55 * The path to this "packages" directory. 60 * The path to this "packages" directory.
56 */ 61 */
57 // TODO(rnystrom): Make this path configurable. 62 // TODO(rnystrom): Make this path configurable.
58 String get path() => join(root.dir, 'packages'); 63 String get path() => join(root.dir, 'packages');
59 64
60 /** 65 /**
61 * Ensures that the package identified by [id] is installed to the directory. 66 * Ensures that the package identified by [id] is installed to the directory,
62 * Returns the resolved [PackageId]. 67 * loads it, and returns it.
63 * 68 *
64 * If this completes successfully, the package is guaranteed to be importable 69 * If this completes successfully, the package is guaranteed to be importable
65 * using the `package:` scheme. 70 * using the `package:` scheme.
66 * 71 *
67 * This will automatically install the package to the system-wide cache as 72 * This will automatically install the package to the system-wide cache as
68 * well if it requires network access to retrieve (specifically, if 73 * well if it requires network access to retrieve (specifically, if
69 * `id.source.shouldCache` is true). 74 * `id.source.shouldCache` is true).
70 * 75 *
71 * See also [installDependencies]. 76 * See also [installTransitively].
72 */ 77 */
73 Future<PackageId> install(PackageId id) { 78 Future<Package> install(PackageId id) {
74 var pendingOrCompleted = _installs[id]; 79 var package = _loadedPackages[id];
75 if (pendingOrCompleted != null) return pendingOrCompleted; 80 if (package != null) return new Future<Package>.immediate(package);
81
82 var pending = _pendingInstalls[id];
83 if (pending != null) return new Future<Package>.immediate(package);
76 84
77 var packageDir = join(path, id.name); 85 var packageDir = join(path, id.name);
78 var future = ensureDir(dirname(packageDir)).chain((_) { 86 var future = ensureDir(dirname(packageDir)).chain((_) {
79 return exists(packageDir); 87 return exists(packageDir);
80 }).chain((exists) { 88 }).chain((exists) {
81 if (!exists) return new Future.immediate(null); 89 // If the package already exists in the directory, no need to re-install.
82 // TODO(nweiz): figure out when to actually delete the directory, and when 90 if (exists) return new Future.immediate(null);
83 // we can just re-use the existing symlink. 91
84 return deleteDir(packageDir);
85 }).chain((_) {
86 if (id.source.shouldCache) { 92 if (id.source.shouldCache) {
87 return cache.install(id).chain( 93 return cache.install(id).chain(
88 (pkg) => createSymlink(pkg.dir, packageDir)); 94 (pkg) => createSymlink(pkg.dir, packageDir));
89 } else { 95 } else {
90 return id.source.install(id, packageDir).transform((found) { 96 return id.source.install(id, packageDir).transform((found) {
91 if (found) return null; 97 if (found) return null;
92 // TODO(nweiz): More robust error-handling. 98 // TODO(nweiz): More robust error-handling.
93 throw 'Package ${id.name} not found in source "${id.source.name}".'; 99 throw 'Package ${id.name} not found in source "${id.source.name}".';
94 }); 100 });
95 } 101 }
96 }).chain((_) => id.resolved); 102 }).chain((_) => Package.load(packageDir, cache.sources));
97 103
98 _installs[id] = future; 104 future.then((pkg) => _loadedPackages[id] = pkg);
105 always(future, () => _pendingInstalls.remove(id));
106 _pendingInstalls[id] = future;
99 107
100 return future; 108 return future;
101 } 109 }
102 110
103 /** 111 /**
104 * Installs all dependencies of the [root] package to its "packages" 112 * Installs all dependencies of the [root] package to its "packages"
105 * directory. Returns a [Future] that completes when all dependencies are 113 * directory. Returns a [Future] that completes when all dependencies are
106 * installed. 114 * installed.
107 */ 115 */
108 Future installDependencies() { 116 Future installDependencies() {
109 return _getResolvedDependencies().chain((packageVersions) { 117 return resolveVersions(cache.sources, root).chain((packageVersions) {
118 // TODO(nweiz): persist packageVersions to a lockfile.
110 return Futures.wait(packageVersions.map((id) { 119 return Futures.wait(packageVersions.map((id) {
111 if (id.source is RootSource) return new Future.immediate(id); 120 if (id.source is RootSource) return new Future.immediate(null);
112 return install(id); 121 return install(id);
113 })); 122 }));
114 }).chain(_saveLockFile);
115 }
116
117 /**
118 * Gets a list of all the [PackageId]s that this entrypoint transitively
119 * depends on. The concrete versions of these ids are given by the
120 * [VersionSolver] and the `pubspec.lock` file, if it exists.
121 */
122 Future<List<PackageId>> _getResolvedDependencies() {
123 return _loadLockFile().chain((lockFile) =>
124 resolveVersions(cache.sources, root, lockFile));
125 }
126
127 /**
128 * Loads the list of concrete package versions from the `pubspec.lock`, if it
129 * exists. If it doesn't, this completes to an empty [LockFile].
130 *
131 * If there's an error reading the `pubspec.lock` file, this will print a
132 * warning message and act as though the file doesn't exist.
133 */
134 Future<LockFile> _loadLockFile() {
135 var completer = new Completer<LockFile>();
136 var lockFilePath = join(root.dir, 'pubspec.lock');
137 var future = readTextFile(lockFilePath);
138
139 future.handleException((_) {
140 completer.complete(new LockFile.empty());
141
142 // If we failed to load the lockfile but it does exist, something's
143 // probably wrong and we should notify the user.
144 fileExists(lockFilePath).then((exists) {
145 if (!exists) return;
146 printError("Error reading pubspec.lock: ${future.exception}");
147 });
148
149 return true;
150 }); 123 });
151
152 future.then((text) =>
153 completer.complete(new LockFile.parse(text, cache.sources)));
154 return completer.future;
155 }
156
157 /**
158 * Saves a list of concrete package versions to the `pubspec.lock` file.
159 */
160 Future _saveLockFile(List<PackageId> packageIds) {
161 var lockFile = new LockFile.empty();
162 for (var id in packageIds) {
163 if (id.source is! RootSource) lockFile.packages[id.name] = id;
164 }
165
166 return writeTextFile(join(root.dir, 'pubspec.lock'), lockFile.serialize());
167 } 124 }
168 } 125 }
OLDNEW
« no previous file with comments | « no previous file | utils/pub/git_source.dart » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698