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

Side by Side Diff: utils/pub/version_solver.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 | « utils/pub/source.dart ('k') | utils/tests/pub/lock_file_test.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 /** 5 /**
6 * Attempts to resolve a set of version constraints for a package dependency 6 * Attempts to resolve a set of version constraints for a package dependency
7 * graph and select an appropriate set of best specific versions for all 7 * graph and select an appropriate set of best specific versions for all
8 * dependent packages. It works iteratively and tries to reach a stable 8 * dependent packages. It works iteratively and tries to reach a stable
9 * solution where the constraints of all dependencies are met. If it fails to 9 * solution where the constraints of all dependencies are met. If it fails to
10 * reach a solution after a certain number of iterations, it assumes the 10 * reach a solution after a certain number of iterations, it assumes the
(...skipping 20 matching lines...) Expand all
31 * of the constraints that its depending packages place on it. If that overall 31 * of the constraints that its depending packages place on it. If that overall
32 * constraint changes (say from "<3.0.0" to "<2.5.0"), then the currently 32 * constraint changes (say from "<3.0.0" to "<2.5.0"), then the currently
33 * picked version for that package may fall outside of the new constraint. If 33 * picked version for that package may fall outside of the new constraint. If
34 * that happens, we find the new best version that meets the updated constraint 34 * that happens, we find the new best version that meets the updated constraint
35 * and then the change the package to use that version. That cycles back up to 35 * and then the change the package to use that version. That cycles back up to
36 * the beginning again. 36 * the beginning again.
37 */ 37 */
38 #library('version_solver'); 38 #library('version_solver');
39 39
40 #import('dart:json'); 40 #import('dart:json');
41 #import('lock_file.dart');
42 #import('package.dart'); 41 #import('package.dart');
43 #import('pubspec.dart'); 42 #import('pubspec.dart');
44 #import('root_source.dart'); 43 #import('root_source.dart');
45 #import('source.dart'); 44 #import('source.dart');
46 #import('source_registry.dart'); 45 #import('source_registry.dart');
47 #import('utils.dart'); 46 #import('utils.dart');
48 #import('version.dart'); 47 #import('version.dart');
49 48
50 /** 49 /**
51 * Attempts to select the best concrete versions for all of the transitive 50 * Attempts to select the best concrete versions for all of the transitive
52 * dependencies of [root] taking into account all of the [VersionConstraint]s 51 * dependencies of [root] taking into account all of the [VersionConstraint]s
53 * that those dependencies place on each other and the requirements imposed by 52 * that those dependencies place on each other. If successful, completes to a
54 * [lockFile]. If successful, completes to a [Map] that maps package names to 53 * [Map] that maps package names to the selected version for that package. If
55 * the selected version for that package. If it fails, the future will complete 54 * it fails, the future will complete with a [NoVersionException],
56 * with a [NoVersionException], [DisjointConstraintException], or 55 * [DisjointConstraintException], or [CouldNotSolveException].
57 * [CouldNotSolveException].
58 */ 56 */
59 Future<List<PackageId>> resolveVersions(SourceRegistry sources, Package root, 57 Future<List<PackageId>> resolveVersions(SourceRegistry sources, Package root) {
60 LockFile lockFile) { 58 return new VersionSolver(sources, root).solve();
61 return new VersionSolver(sources, root, lockFile).solve();
62 } 59 }
63 60
64 class VersionSolver { 61 class VersionSolver {
65 final SourceRegistry _sources; 62 final SourceRegistry _sources;
66 final Package _root; 63 final Package _root;
67 final LockFile lockFile;
68 final PubspecCache _pubspecs; 64 final PubspecCache _pubspecs;
69 final Map<String, Dependency> _packages; 65 final Map<String, Dependency> _packages;
70 final Queue<WorkItem> _work; 66 final Queue<WorkItem> _work;
71 int _numIterations = 0; 67 int _numIterations = 0;
72 68
73 VersionSolver(SourceRegistry sources, this._root, this.lockFile) 69 VersionSolver(SourceRegistry sources, Package root)
74 : _sources = sources, 70 : _sources = sources,
71 _root = root,
75 _pubspecs = new PubspecCache(sources), 72 _pubspecs = new PubspecCache(sources),
76 _packages = <Dependency>{}, 73 _packages = <Dependency>{},
77 _work = new Queue<WorkItem>(); 74 _work = new Queue<WorkItem>();
78 75
79 Future<List<PackageId>> solve() { 76 Future<List<PackageId>> solve() {
80 // Kick off the work by adding the root package at its concrete version to 77 // Kick off the work by adding the root package at its concrete version to
81 // the dependency graph. 78 // the dependency graph.
82 var ref = new PackageRef(new RootSource(_root), _root.version, _root.name); 79 var ref = new PackageRef(new RootSource(_root), _root.version, _root.name);
83 enqueue(new AddConstraint('(entrypoint)', ref)); 80 enqueue(new AddConstraint('(entrypoint)', ref));
84 _pubspecs.cache(ref.atVersion(_root.version), _root.pubspec); 81 _pubspecs.cache(ref.atVersion(_root.version), _root.pubspec);
(...skipping 40 matching lines...) Expand 10 before | Expand all | Expand 10 after
125 } 122 }
126 123
127 /** 124 /**
128 * Sets the best selected version of [package] to [version]. 125 * Sets the best selected version of [package] to [version].
129 */ 126 */
130 void setVersion(String package, Version version) { 127 void setVersion(String package, Version version) {
131 _packages[package].version = version; 128 _packages[package].version = version;
132 } 129 }
133 130
134 List<PackageId> buildResults() { 131 List<PackageId> buildResults() {
135 return _packages.getValues().filter((dep) => dep.isDependedOn).map((dep) { 132 return _packages.getValues()
136 var description = dep.description; 133 .filter((dep) => dep.isDependedOn)
137 134 .map((dep) => new PackageId(dep.source, dep.version, dep.description));
138 // If the lockfile contains a fully-resolved description for the package,
139 // use that. This allows e.g. Git to ensure that the same commit is used.
140 var lockedPackage = lockFile.packages[dep.name];
141 if (lockedPackage != null && lockedPackage.version == dep.version &&
142 lockedPackage.source.name == dep.source.name &&
143 dep.source.descriptionsEqual(
144 description, lockedPackage.description)) {
145 description = lockedPackage.description;
146 }
147
148 return new PackageId(dep.source, dep.version, description);
149 });
150 } 135 }
151 } 136 }
152 137
153 /** 138 /**
154 * The constraint solver works by iteratively processing a queue of work items. 139 * The constraint solver works by iteratively processing a queue of work items.
155 * Each item is a single atomic change to the dependency graph. Handling them 140 * Each item is a single atomic change to the dependency graph. Handling them
156 * in a queue lets us handle asynchrony (resolving versions requires information 141 * in a queue lets us handle asynchrony (resolving versions requires information
157 * from servers) as well as avoid deeply nested recursion. 142 * from servers) as well as avoid deeply nested recursion.
158 */ 143 */
159 interface WorkItem { 144 interface WorkItem {
(...skipping 123 matching lines...) Expand 10 before | Expand all | Expand 10 after
283 } 268 }
284 269
285 // If the dependency is on the root package, then we don't need to do 270 // If the dependency is on the root package, then we don't need to do
286 // anything since it's already at the best version. 271 // anything since it's already at the best version.
287 if (name == solver._root.name) { 272 if (name == solver._root.name) {
288 solver.enqueue(new ChangeVersion( 273 solver.enqueue(new ChangeVersion(
289 source, description, solver._root.version)); 274 source, description, solver._root.version));
290 return null; 275 return null;
291 } 276 }
292 277
293 // If the dependency is on a package in the lockfile, use the lockfile's
294 // version for that package if it's valid given the other constraints.
295 var lockedPackage = solver.lockFile.packages[name];
296 if (lockedPackage != null) {
297 var lockedVersion = lockedPackage.version;
298 if (newConstraint.allows(lockedVersion)) {
299 solver.enqueue(new ChangeVersion(source, description, lockedVersion));
300 return null;
301 }
302 }
303
304 // The constraint has changed, so see what the best version of the package 278 // The constraint has changed, so see what the best version of the package
305 // that meets the new constraint is. 279 // that meets the new constraint is.
306 return source.getVersions(description).transform((versions) { 280 return source.getVersions(description).transform((versions) {
307 var best = null; 281 var best = null;
308 for (var version in versions) { 282 for (var version in versions) {
309 if (newConstraint.allows(version)) { 283 if (newConstraint.allows(version)) {
310 if (best == null || version > best) best = version; 284 if (best == null || version > best) best = version;
311 } 285 }
312 } 286 }
313 287
(...skipping 258 matching lines...) Expand 10 before | Expand all | Expand 10 after
572 final description1; 546 final description1;
573 final description2; 547 final description2;
574 548
575 DescriptionMismatchException(this.package, this.description1, 549 DescriptionMismatchException(this.package, this.description1,
576 this.description2); 550 this.description2);
577 551
578 // TODO(nweiz): Dump to YAML when that's supported 552 // TODO(nweiz): Dump to YAML when that's supported
579 String toString() => "Package '$package' has conflicting descriptions " 553 String toString() => "Package '$package' has conflicting descriptions "
580 "'${JSON.stringify(description1)}' and '${JSON.stringify(description2)}'"; 554 "'${JSON.stringify(description1)}' and '${JSON.stringify(description2)}'";
581 } 555 }
OLDNEW
« no previous file with comments | « utils/pub/source.dart ('k') | utils/tests/pub/lock_file_test.dart » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698