Chromium Code Reviews

Unified Diff: sdk/lib/_internal/pub/lib/src/solver/version_solver.dart

Issue 15347004: Gracefully handle pubspecs with dependencies using unknown sources. (Closed) Base URL: https://dart.googlecode.com/svn/branches/bleeding_edge/dart
Patch Set: Handle bad sources in dev deps. Created 7 years, 7 months ago
Use n/p to move between diff chunks; N/P to move between comments.
Jump to:
View side-by-side diff with in-line comments
Index: sdk/lib/_internal/pub/lib/src/solver/version_solver.dart
diff --git a/sdk/lib/_internal/pub/lib/src/solver/version_solver.dart b/sdk/lib/_internal/pub/lib/src/solver/version_solver.dart
index 885acdc55659b5529140d2cdece9e102d6219af5..7034aeeb5374c46a5e136eb7cc62bb9f79b9321f 100644
--- a/sdk/lib/_internal/pub/lib/src/solver/version_solver.dart
+++ b/sdk/lib/_internal/pub/lib/src/solver/version_solver.dart
@@ -14,6 +14,7 @@ import '../pubspec.dart';
import '../source.dart';
import '../source_registry.dart';
import '../version.dart';
+import '../utils.dart';
import 'backtracking_solver.dart';
/// Attempts to select the best concrete versions for all of the transitive
@@ -70,9 +71,23 @@ class SolveResult {
/// Used to avoid requesting the same pubspec from the server repeatedly.
class PubspecCache {
final SourceRegistry _sources;
+
+ /// The already requested cached version lists.
nweiz 2013/05/17 23:55:55 "already-requested" (also below)
Bob Nystrom 2013/05/20 20:06:00 Done.
final _versions = new Map<PackageRef, List<PackageId>>();
+
+ /// The already requested cached pubspecs.
final _pubspecs = new Map<PackageId, Pubspec>();
+ /// Pending version list requests that have not completed yet. If multiple
+ /// requests for the same package are made concurrently, this avoids
+ /// duplicate requests.
+ final _pendingVersions = new Map<PackageRef, Future<List<PackageId>>>();
+
+ /// Pending pubspec requests that have not completed yet. If multiple
+ /// requests for the same package are made concurrently, this avoids
+ /// duplicate requests.
+ final _pendingPubspecs = new Map<PackageId, Future<Pubspec>>();
nweiz 2013/05/17 23:55:55 I don't understand why this change is part of this
Bob Nystrom 2013/05/20 20:06:00 The behavior here was always the intended behavior
+
/// The number of times a version list was requested and it wasn't cached and
/// had to be requested from the source.
int versionCacheMisses = 0;
@@ -104,11 +119,22 @@ class PubspecCache {
return new Future<Pubspec>.value(_pubspecs[id]);
}
+ // If we're already waiting on a request, use the same future.
+ if (_pendingPubspecs.containsKey(id)) {
+ pubspecCacheHits++;
+ return _pendingPubspecs[id];
+ }
+
pubspecCacheMisses++;
- return id.describe().then((pubspec) {
+
+ var source = _sources[id.source];
+ var future = source.describe(id);
+ _pendingPubspecs[id] = future;
+ return future.then((pubspec) {
log.solver('requested $id pubspec');
// Cache it.
+ _pendingPubspecs.remove(id);
_pubspecs[id] = pubspec;
return pubspec;
});
@@ -119,21 +145,38 @@ class PubspecCache {
Pubspec getCachedPubspec(PackageId id) => _pubspecs[id];
/// Gets the list of versions for [package] in descending order.
- Future<List<PackageId>> getVersions(PackageRef package) {
+ Future<List<PackageId>> getVersions(PackageDep package) {
nweiz 2013/05/17 23:55:55 Why is this taking a PackageDep now? Why is the ve
Bob Nystrom 2013/05/20 20:06:00 That was related to another change I ended up aban
+ if (package.isRoot) {
+ throw new StateError("Cannot get versions for root package $package.");
+ }
+
// See if we have it cached.
- var versions = _versions[package];
+ var ref = package.toRef();
+ var versions = _versions[ref];
if (versions != null) {
versionCacheHits++;
return new Future.value(versions);
}
+ // If we're already waiting on a request, use the same future.
+ if (_pendingVersions.containsKey(ref)) {
+ pubspecCacheHits++;
+ return _pendingVersions[ref];
+ }
+
versionCacheMisses++;
- return package.getVersions().then((ids) {
+
+ var source = _sources[package.source];
+ var future = source.getVersions(package.name, package.description);
+ _pendingVersions[ref] = future;
+ return future.then((versions) {
// Sort by descending version so we try newer versions first.
- ids.sort((a, b) => b.version.compareTo(a.version));
+ versions.sort((a, b) => b.compareTo(a));
- log.solver('requested $package version list');
- _versions[package] = ids;
+ var ids = versions.map((version) => ref.atVersion(version)).toList();
+
+ _pendingVersions.remove(ref);
+ _versions[ref] = ids;
return ids;
nweiz 2013/05/17 23:55:55 Right now you're returning `ids` for the first fut
Bob Nystrom 2013/05/20 20:06:00 Yeah, this code is clearly wrong. Removed.
});
}
@@ -153,7 +196,7 @@ class Dependency {
}
/// Base class for all failures that can occur while trying to resolve versions.
-class SolveFailure implements Exception {
+abstract class SolveFailure implements ApplicationException {
/// The name of the package whose version could not be solved. Will be `null`
/// if the failure is not specific to one package.
final String package;
@@ -270,7 +313,6 @@ class DisjointConstraintException extends SolveFailure {
/// Exception thrown when two packages with the same name but different sources
/// are depended upon.
class SourceMismatchException extends SolveFailure {
-
SourceMismatchException(String package, Iterable<Dependency> dependencies)
: super(package, dependencies);
@@ -280,6 +322,18 @@ class SourceMismatchException extends SolveFailure {
"depends on it from source ${dep.source}";
}
+/// Exception thrown when a dependency on an unknown source name is found.
+class UnknownSourceException extends SolveFailure {
+ UnknownSourceException(String package, Iterable<Dependency> dependencies)
+ : super(package, dependencies);
+
+ String toString() {
+ var dep = only(dependencies);
+ return "Package '${dep.depender}' depends on '${dep.dep.name}' from "
+ "unknown source '${dep.dep.source}'.";
+ }
+}
+
/// Exception thrown when two packages with the same name and source but
/// different descriptions are depended upon.
class DescriptionMismatchException extends SolveFailure {

Powered by Google App Engine