Index: lib/src/source/hosted.dart |
diff --git a/lib/src/source/hosted.dart b/lib/src/source/hosted.dart |
index 0808105a2c4ff294ab7464401aad11980b0cb1e0..ea282c5938e70c4ecf456dfbd3590918da6daf92 100644 |
--- a/lib/src/source/hosted.dart |
+++ b/lib/src/source/hosted.dart |
@@ -16,29 +16,47 @@ import '../io.dart'; |
import '../log.dart' as log; |
import '../package.dart'; |
import '../pubspec.dart'; |
+import '../source.dart'; |
+import '../system_cache.dart'; |
import '../utils.dart'; |
import 'cached.dart'; |
/// A package source that gets packages from a package hosting site that uses |
/// the same API as pub.dartlang.org. |
-class HostedSource extends CachedSource { |
+class HostedSource extends Source { |
+ final name = "hosted"; |
+ final hasMultipleVersions = true; |
+ |
+ BoundHostedSource bind(SystemCache systemCache, {bool isOffline: false}) => |
+ isOffline |
+ ? new _OfflineHostedSource(this, systemCache) |
+ : new BoundHostedSource(this, systemCache); |
+ |
+ /// Gets the default URL for the package server for hosted dependencies. |
+ String get defaultUrl { |
+ var url = io.Platform.environment["PUB_HOSTED_URL"]; |
+ if (url != null) return url; |
+ |
+ return "https://pub.dartlang.org"; |
+ } |
+ |
/// Returns a reference to a hosted package named [name]. |
/// |
/// If [url] is passed, it's the URL of the pub server from which the package |
/// should be downloaded. It can be a [Uri] or a [String]. |
- static PackageRef refFor(String name, {url}) => |
+ PackageRef refFor(String name, {url}) => |
new PackageRef(name, 'hosted', _descriptionFor(name, url)); |
/// Returns an ID for a hosted package named [name] at [version]. |
/// |
/// If [url] is passed, it's the URL of the pub server from which the package |
/// should be downloaded. It can be a [Uri] or a [String]. |
- static PackageId idFor(String name, Version version, {url}) => |
+ PackageId idFor(String name, Version version, {url}) => |
new PackageId(name, 'hosted', version, _descriptionFor(name, url)); |
/// Returns the description for a hosted package named [name] with the |
/// given package server [url]. |
- static _descriptionFor(String name, [url]) { |
+ _descriptionFor(String name, [url]) { |
if (url == null) return name; |
if (url is! String && url is! Uri) { |
@@ -48,17 +66,60 @@ class HostedSource extends CachedSource { |
return {'name': name, 'url': url.toString()}; |
} |
- final name = "hosted"; |
- final hasMultipleVersions = true; |
+ bool descriptionsEqual(description1, description2) => |
+ _parseDescription(description1) == _parseDescription(description2); |
- /// Gets the default URL for the package server for hosted dependencies. |
- static String get defaultUrl { |
- var url = io.Platform.environment["PUB_HOSTED_URL"]; |
- if (url != null) return url; |
+ /// Ensures that [description] is a valid hosted package description. |
+ /// |
+ /// There are two valid formats. A plain string refers to a package with the |
+ /// given name from the default host, while a map with keys "name" and "url" |
+ /// refers to a package with the given name from the host at the given URL. |
+ PackageRef parseRef(String name, description, {String containingPath}) { |
+ _parseDescription(description); |
+ return new PackageRef(name, this.name, description); |
+ } |
- return "https://pub.dartlang.org"; |
+ PackageId parseId(String name, Version version, description) { |
+ _parseDescription(description); |
+ return new PackageId(name, this.name, version, description); |
} |
+ /// Parses the description for a package. |
+ /// |
+ /// If the package parses correctly, this returns a (name, url) pair. If not, |
+ /// this throws a descriptive FormatException. |
+ Pair<String, String> _parseDescription(description) { |
+ if (description is String) { |
+ return new Pair<String, String>(description, defaultUrl); |
+ } |
+ |
+ if (description is! Map) { |
+ throw new FormatException( |
+ "The description must be a package name or map."); |
+ } |
+ |
+ if (!description.containsKey("name")) { |
+ throw new FormatException( |
+ "The description map must contain a 'name' key."); |
+ } |
+ |
+ var name = description["name"]; |
+ if (name is! String) { |
+ throw new FormatException("The 'name' key must have a string value."); |
+ } |
+ |
+ return new Pair<String, String>(name, description["url"] ?? defaultUrl); |
+ } |
+} |
+ |
+/// The [BoundSource] for [HostedSource]. |
+class BoundHostedSource extends CachedSource { |
+ final HostedSource source; |
+ |
+ final SystemCache systemCache; |
+ |
+ BoundHostedSource(this.source, this.systemCache); |
+ |
/// Downloads a list of all versions of a package that are available from the |
/// site. |
Future<List<PackageId>> doGetVersions(PackageRef ref) async { |
@@ -71,7 +132,7 @@ class HostedSource extends CachedSource { |
try { |
body = await httpClient.read(url, headers: PUB_API_HEADERS); |
} catch (error, stackTrace) { |
- var parsed = _parseDescription(ref.description); |
+ var parsed = source._parseDescription(ref.description); |
_throwFriendlyError(error, stackTrace, parsed.first, parsed.last); |
} |
@@ -80,7 +141,7 @@ class HostedSource extends CachedSource { |
var pubspec = new Pubspec.fromMap( |
map['pubspec'], systemCache.sources, |
expectedName: ref.name, location: url); |
- var id = idFor(ref.name, pubspec.version, |
+ var id = source.idFor(ref.name, pubspec.version, |
url: _serverFor(ref.description)); |
memoizePubspec(id, pubspec); |
@@ -88,6 +149,17 @@ class HostedSource extends CachedSource { |
}).toList(); |
} |
+ /// Parses [description] into its server and package name components, then |
+ /// converts that to a Uri given [pattern]. |
+ /// |
+ /// Ensures the package name is properly URL encoded. |
+ Uri _makeUrl(description, String pattern(String server, String package)) { |
+ var parsed = source._parseDescription(description); |
+ var server = parsed.last; |
+ var package = Uri.encodeComponent(parsed.first); |
+ return Uri.parse(pattern(server, package)); |
+ } |
+ |
/// Downloads and parses the pubspec for a specific version of a package that |
/// is available from the site. |
Future<Pubspec> describeUncached(PackageId id) async { |
@@ -101,7 +173,7 @@ class HostedSource extends CachedSource { |
version = JSON.decode( |
await httpClient.read(url, headers: PUB_API_HEADERS)); |
} catch (error, stackTrace) { |
- var parsed = _parseDescription(id.description); |
+ var parsed = source._parseDescription(id.description); |
_throwFriendlyError(error, stackTrace, id.name, parsed.last); |
} |
@@ -115,7 +187,7 @@ class HostedSource extends CachedSource { |
if (!isInSystemCache(id)) { |
var packageDir = getDirectory(id); |
ensureDir(p.dirname(packageDir)); |
- var parsed = _parseDescription(id.description); |
+ var parsed = source._parseDescription(id.description); |
await _download(parsed.last, parsed.first, id.version, packageDir); |
} |
@@ -128,31 +200,11 @@ class HostedSource extends CachedSource { |
/// Each of these subdirectories then contains a subdirectory for each |
/// package downloaded from that site. |
String getDirectory(PackageId id) { |
- var parsed = _parseDescription(id.description); |
+ var parsed = source._parseDescription(id.description); |
var dir = _urlToDirectory(parsed.last); |
return p.join(systemCacheRoot, dir, "${parsed.first}-${id.version}"); |
} |
- String packageName(description) => _parseDescription(description).first; |
- |
- bool descriptionsEqual(description1, description2) => |
- _parseDescription(description1) == _parseDescription(description2); |
- |
- /// Ensures that [description] is a valid hosted package description. |
- /// |
- /// There are two valid formats. A plain string refers to a package with the |
- /// given name from the default host, while a map with keys "name" and "url" |
- /// refers to a package with the given name from the host at the given URL. |
- PackageRef parseRef(String name, description, {String containingPath}) { |
- _parseDescription(description); |
- return new PackageRef(name, this.name, description); |
- } |
- |
- PackageId parseId(String name, Version version, description) { |
- _parseDescription(description); |
- return new PackageId(name, this.name, version, description); |
- } |
- |
/// Re-downloads all packages that have been previously downloaded into the |
/// system cache from any server. |
Future<Pair<List<PackageId>, List<PackageId>>> repairCachedPackages() async { |
@@ -167,7 +219,7 @@ class HostedSource extends CachedSource { |
packages.sort(Package.orderByNameAndVersion); |
for (var package in packages) { |
- var id = idFor(package.name, package.version, url: url); |
+ var id = source.idFor(package.name, package.version, url: url); |
try { |
await _download(url, package.name, package.version, package.dir); |
@@ -176,7 +228,7 @@ class HostedSource extends CachedSource { |
failures.add(id); |
var message = "Failed to repair ${log.bold(package.name)} " |
"${package.version}"; |
- if (url != defaultUrl) message += " from $url"; |
+ if (url != source.defaultUrl) message += " from $url"; |
log.error("$message. Error:\n$error"); |
log.fine(stackTrace); |
@@ -190,9 +242,8 @@ class HostedSource extends CachedSource { |
/// Gets all of the packages that have been downloaded into the system cache |
/// from the default server. |
- List<Package> getCachedPackages() { |
- return _getCachedPackagesInDirectory(_urlToDirectory(defaultUrl)); |
- } |
+ List<Package> getCachedPackages() => |
+ _getCachedPackagesInDirectory(_urlToDirectory(source.defaultUrl)); |
/// Gets all of the packages that have been downloaded into the system cache |
/// into [dir]. |
@@ -248,6 +299,77 @@ class HostedSource extends CachedSource { |
// Otherwise re-throw the original exception. |
throw error; |
} |
+ |
+ /// Given a URL, returns a "normalized" string to be used as a directory name |
+ /// for packages downloaded from the server at that URL. |
+ /// |
+ /// This normalization strips off the scheme (which is presumed to be HTTP or |
+ /// HTTPS) and *sort of* URL-encodes it. I say "sort of" because it does it |
+ /// incorrectly: it uses the character's *decimal* ASCII value instead of hex. |
+ /// |
+ /// This could cause an ambiguity since some characters get encoded as three |
+ /// digits and others two. It's possible for one to be a prefix of the other. |
+ /// In practice, the set of characters that are encoded don't happen to have |
+ /// any collisions, so the encoding is reversible. |
+ /// |
+ /// This behavior is a bug, but is being preserved for compatibility. |
+ String _urlToDirectory(String url) { |
+ // Normalize all loopback URLs to "localhost". |
+ url = url.replaceAllMapped( |
+ new RegExp(r"^(https?://)(127\.0\.0\.1|\[::1\]|localhost)?"), |
+ (match) { |
+ // Don't include the scheme for HTTPS URLs. This makes the directory names |
+ // nice for the default and most recommended scheme. We also don't include |
+ // it for localhost URLs, since they're always known to be HTTP. |
+ var localhost = match[2] == null ? '' : 'localhost'; |
+ var scheme = |
+ match[1] == 'https://' || localhost.isNotEmpty ? '' : match[1]; |
+ return "$scheme$localhost"; |
+ }); |
+ return replace(url, new RegExp(r'[<>:"\\/|?*%]'), |
+ (match) => '%${match[0].codeUnitAt(0)}'); |
+ } |
+ |
+ /// Given a directory name in the system cache, returns the URL of the server |
+ /// whose packages it contains. |
+ /// |
+ /// See [_urlToDirectory] for details on the mapping. Note that because the |
+ /// directory name does not preserve the scheme, this has to guess at it. It |
+ /// chooses "http" for loopback URLs (mainly to support the pub tests) and |
+ /// "https" for all others. |
+ String _directoryToUrl(String url) { |
+ // Decode the pseudo-URL-encoded characters. |
+ var chars = '<>:"\\/|?*%'; |
+ for (var i = 0; i < chars.length; i++) { |
+ var c = chars.substring(i, i + 1); |
+ url = url.replaceAll("%${c.codeUnitAt(0)}", c); |
+ } |
+ |
+ // If the URL has an explicit scheme, use that. |
+ if (url.contains("://")) return url; |
+ |
+ // Otherwise, default to http for localhost and https for everything else. |
+ var scheme = |
+ isLoopback(url.replaceAll(new RegExp(":.*"), "")) ? "http" : "https"; |
+ return "$scheme://$url"; |
+ } |
+ |
+ /// Returns the server URL for [description]. |
+ Uri _serverFor(description) => |
+ Uri.parse(source._parseDescription(description).last); |
+ |
+ /// Parses [id] into its server, package name, and version components, then |
+ /// converts that to a Uri given [pattern]. |
+ /// |
+ /// Ensures the package name is properly URL encoded. |
+ Uri _makeVersionUrl(PackageId id, |
+ String pattern(String server, String package, String version)) { |
+ var parsed = source._parseDescription(id.description); |
+ var server = parsed.last; |
+ var package = Uri.encodeComponent(parsed.first); |
+ var version = Uri.encodeComponent(id.version.toString()); |
+ return Uri.parse(pattern(server, package, version)); |
+ } |
} |
/// This is the modified hosted source used when pub get or upgrade are run |
@@ -255,10 +377,13 @@ class HostedSource extends CachedSource { |
/// |
/// This uses the system cache to get the list of available packages and does |
/// no network access. |
-class OfflineHostedSource extends HostedSource { |
+class _OfflineHostedSource extends BoundHostedSource { |
+ _OfflineHostedSource(HostedSource source, SystemCache systemCache) |
+ : super(source, systemCache); |
+ |
/// Gets the list of all versions of [ref] that are in the system cache. |
Future<List<PackageId>> doGetVersions(PackageRef ref) async { |
- var parsed = _parseDescription(ref.description); |
+ var parsed = source._parseDescription(ref.description); |
var server = parsed.last; |
log.io("Finding versions of ${ref.name} in " |
"$systemCacheRoot/${_urlToDirectory(server)}"); |
@@ -270,7 +395,7 @@ class OfflineHostedSource extends HostedSource { |
versions = await listDir(dir).map((entry) { |
var components = p.basename(entry).split("-"); |
if (components.first != ref.name) return null; |
- return HostedSource.idFor( |
+ return source.idFor( |
ref.name, new Version.parse(components.skip(1).join("-")), |
url: _serverFor(ref.description)); |
}).where((id) => id != null).toList(); |
@@ -299,113 +424,3 @@ class OfflineHostedSource extends HostedSource { |
"${id.name} ${id.version} is not available in your system cache."); |
} |
} |
- |
-/// Given a URL, returns a "normalized" string to be used as a directory name |
-/// for packages downloaded from the server at that URL. |
-/// |
-/// This normalization strips off the scheme (which is presumed to be HTTP or |
-/// HTTPS) and *sort of* URL-encodes it. I say "sort of" because it does it |
-/// incorrectly: it uses the character's *decimal* ASCII value instead of hex. |
-/// |
-/// This could cause an ambiguity since some characters get encoded as three |
-/// digits and others two. It's possible for one to be a prefix of the other. |
-/// In practice, the set of characters that are encoded don't happen to have |
-/// any collisions, so the encoding is reversible. |
-/// |
-/// This behavior is a bug, but is being preserved for compatibility. |
-String _urlToDirectory(String url) { |
- // Normalize all loopback URLs to "localhost". |
- url = url.replaceAllMapped( |
- new RegExp(r"^(https?://)(127\.0\.0\.1|\[::1\]|localhost)?"), |
- (match) { |
- // Don't include the scheme for HTTPS URLs. This makes the directory names |
- // nice for the default and most recommended scheme. We also don't include |
- // it for localhost URLs, since they're always known to be HTTP. |
- var localhost = match[2] == null ? '' : 'localhost'; |
- var scheme = match[1] == 'https://' || localhost.isNotEmpty ? '' : match[1]; |
- return "$scheme$localhost"; |
- }); |
- return replace(url, new RegExp(r'[<>:"\\/|?*%]'), |
- (match) => '%${match[0].codeUnitAt(0)}'); |
-} |
- |
-/// Given a directory name in the system cache, returns the URL of the server |
-/// whose packages it contains. |
-/// |
-/// See [_urlToDirectory] for details on the mapping. Note that because the |
-/// directory name does not preserve the scheme, this has to guess at it. It |
-/// chooses "http" for loopback URLs (mainly to support the pub tests) and |
-/// "https" for all others. |
-String _directoryToUrl(String url) { |
- // Decode the pseudo-URL-encoded characters. |
- var chars = '<>:"\\/|?*%'; |
- for (var i = 0; i < chars.length; i++) { |
- var c = chars.substring(i, i + 1); |
- url = url.replaceAll("%${c.codeUnitAt(0)}", c); |
- } |
- |
- // If the URL has an explicit scheme, use that. |
- if (url.contains("://")) return url; |
- |
- // Otherwise, default to http for localhost and https for everything else. |
- var scheme = |
- isLoopback(url.replaceAll(new RegExp(":.*"), "")) ? "http" : "https"; |
- return "$scheme://$url"; |
-} |
- |
-/// Parses [description] into its server and package name components, then |
-/// converts that to a Uri given [pattern]. |
-/// |
-/// Ensures the package name is properly URL encoded. |
-Uri _makeUrl(description, String pattern(String server, String package)) { |
- var parsed = _parseDescription(description); |
- var server = parsed.last; |
- var package = Uri.encodeComponent(parsed.first); |
- return Uri.parse(pattern(server, package)); |
-} |
- |
-/// Returns the server URL for [description]. |
-Uri _serverFor(description) => Uri.parse(_parseDescription(description).last); |
- |
-/// Parses [id] into its server, package name, and version components, then |
-/// converts that to a Uri given [pattern]. |
-/// |
-/// Ensures the package name is properly URL encoded. |
-Uri _makeVersionUrl(PackageId id, |
- String pattern(String server, String package, String version)) { |
- var parsed = _parseDescription(id.description); |
- var server = parsed.last; |
- var package = Uri.encodeComponent(parsed.first); |
- var version = Uri.encodeComponent(id.version.toString()); |
- return Uri.parse(pattern(server, package, version)); |
-} |
- |
-/// Parses the description for a package. |
-/// |
-/// If the package parses correctly, this returns a (name, url) pair. If not, |
-/// this throws a descriptive FormatException. |
-Pair<String, String> _parseDescription(description) { |
- if (description is String) { |
- return new Pair<String, String>(description, HostedSource.defaultUrl); |
- } |
- |
- if (description is! Map) { |
- throw new FormatException( |
- "The description must be a package name or map."); |
- } |
- |
- if (!description.containsKey("name")) { |
- throw new FormatException( |
- "The description map must contain a 'name' key."); |
- } |
- |
- var name = description["name"]; |
- if (name is! String) { |
- throw new FormatException("The 'name' key must have a string value."); |
- } |
- |
- var url = description["url"]; |
- if (url == null) url = HostedSource.defaultUrl; |
- |
- return new Pair<String, String>(name, url); |
-} |