OLD | NEW |
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 36 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
47 /** | 47 /** |
48 * Attempts to select the best concrete versions for all of the transitive | 48 * Attempts to select the best concrete versions for all of the transitive |
49 * dependencies of [root] taking into account all of the [VersionConstraint]s | 49 * dependencies of [root] taking into account all of the [VersionConstraint]s |
50 * that those dependencies place on each other. If successful, completes to a | 50 * that those dependencies place on each other. If successful, completes to a |
51 * [Map] that maps package names to the selected version for that package. If | 51 * [Map] that maps package names to the selected version for that package. If |
52 * it fails, the future will complete with a [NoVersionException], | 52 * it fails, the future will complete with a [NoVersionException], |
53 * [DisjointConstraintException], or [CouldNotSolveException]. | 53 * [DisjointConstraintException], or [CouldNotSolveException]. |
54 */ | 54 */ |
55 Future<Map<String, Version>> resolveVersions( | 55 Future<Map<String, Version>> resolveVersions( |
56 SourceRegistry sources, Package root) { | 56 SourceRegistry sources, Package root) { |
57 return new VersionSolver(sources).solve(root); | 57 return new VersionSolver(sources, root).solve(); |
58 } | 58 } |
59 | 59 |
60 class VersionSolver { | 60 class VersionSolver { |
61 final SourceRegistry _sources; | 61 final SourceRegistry _sources; |
| 62 final Package _root; |
62 final PubspecCache _pubspecs; | 63 final PubspecCache _pubspecs; |
63 final Map<String, Dependency> _packages; | 64 final Map<String, Dependency> _packages; |
64 final Queue<WorkItem> _work; | 65 final Queue<WorkItem> _work; |
65 int _numIterations = 0; | 66 int _numIterations = 0; |
66 | 67 |
67 VersionSolver(SourceRegistry sources) | 68 VersionSolver(SourceRegistry sources, Package root) |
68 : _sources = sources, | 69 : _sources = sources, |
| 70 _root = root, |
69 _pubspecs = new PubspecCache(sources), | 71 _pubspecs = new PubspecCache(sources), |
70 _packages = <Dependency>{}, | 72 _packages = <Dependency>{}, |
71 _work = new Queue<WorkItem>(); | 73 _work = new Queue<WorkItem>(); |
72 | 74 |
73 Future<Map<String, Version>> solve(Package root) { | 75 Future<Map<String, Version>> solve() { |
74 // Kick off the work by adding the root package at its concrete version to | 76 // Kick off the work by adding the root package at its concrete version to |
75 // the dependency graph. | 77 // the dependency graph. |
76 _pubspecs.cache(root); | 78 _pubspecs.cache(_root); |
77 enqueue(new ChangeConstraint('(entrypoint)', root.name, root.version)); | 79 enqueue(new ChangeConstraint('(entrypoint)', _root.name, _root.version)); |
78 | 80 |
79 Future processNextWorkItem(_) { | 81 Future processNextWorkItem(_) { |
80 while (true) { | 82 while (true) { |
81 // Stop if we are done. | 83 // Stop if we are done. |
82 if (_work.isEmpty()) return new Future.immediate(buildResults()); | 84 if (_work.isEmpty()) return new Future.immediate(buildResults()); |
83 | 85 |
84 // If we appear to be stuck in a loop, then we probably have an unstable | 86 // If we appear to be stuck in a loop, then we probably have an unstable |
85 // graph, bail. We guess this based on a rough heuristic that it should | 87 // graph, bail. We guess this based on a rough heuristic that it should |
86 // only take a certain number of steps to solve a graph with a given | 88 // only take a certain number of steps to solve a graph with a given |
87 // number of connections. | 89 // number of connections. |
(...skipping 174 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
262 // If this constraint change didn't cause the overall constraint on the | 264 // If this constraint change didn't cause the overall constraint on the |
263 // package to change, then we don't need to do any further work. | 265 // package to change, then we don't need to do any further work. |
264 if (oldConstraint == newConstraint) return null; | 266 if (oldConstraint == newConstraint) return null; |
265 | 267 |
266 // If the dependency has been cut free from the graph, just remove it. | 268 // If the dependency has been cut free from the graph, just remove it. |
267 if (!dependency.isDependedOn) { | 269 if (!dependency.isDependedOn) { |
268 solver.enqueue(new ChangeVersion(dependent, null)); | 270 solver.enqueue(new ChangeVersion(dependent, null)); |
269 return null; | 271 return null; |
270 } | 272 } |
271 | 273 |
| 274 // If the dependency is on the root package, then we don't need to do |
| 275 // anything since it's already at the best version. |
| 276 if (dependent == solver._root.name) { |
| 277 solver.enqueue(new ChangeVersion(dependent, solver._root.version)); |
| 278 return null; |
| 279 } |
| 280 |
272 // The constraint has changed, so see what the best version of the package | 281 // The constraint has changed, so see what the best version of the package |
273 // that meets the new constraint is. | 282 // that meets the new constraint is. |
274 // TODO(rnystrom): Should this always be the default source? | 283 // TODO(rnystrom): Should this always be the default source? |
275 var source = solver._sources.defaultSource; | 284 var source = solver._sources.defaultSource; |
276 return source.getVersions(dependent).transform((versions) { | 285 return source.getVersions(dependent).transform((versions) { |
277 var best = null; | 286 var best = null; |
278 for (var version in versions) { | 287 for (var version in versions) { |
279 if (newConstraint.allows(version)) { | 288 if (newConstraint.allows(version)) { |
280 if (best == null || version > best) best = version; | 289 if (best == null || version > best) best = version; |
281 } | 290 } |
(...skipping 134 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
416 /** | 425 /** |
417 * Exception thrown when the [VersionSolver] fails to find a solution after a | 426 * Exception thrown when the [VersionSolver] fails to find a solution after a |
418 * certain number of iterations. | 427 * certain number of iterations. |
419 */ | 428 */ |
420 class CouldNotSolveException implements Exception { | 429 class CouldNotSolveException implements Exception { |
421 CouldNotSolveException(); | 430 CouldNotSolveException(); |
422 | 431 |
423 String toString() => | 432 String toString() => |
424 "Could not find a solution that met all version constraints."; | 433 "Could not find a solution that met all version constraints."; |
425 } | 434 } |
OLD | NEW |