Newer
Older
rnystrom@google.com
committed
// Copyright (c) 2012, the Dart project authors. Please see the AUTHORS file
// for details. All rights reserved. Use of this source code is governed by a
// BSD-style license that can be found in the LICENSE file.
import 'package:pathos/path.dart' as path;
import 'io.dart';
import 'pubspec.dart';
import 'source.dart';
import 'source_registry.dart';
import 'version.dart';
final _README_REGEXP = new RegExp(r"^README($|\.)", caseSensitive: false);
/// A named, versioned, unit of code and resource reuse.
class Package {
/// The path to the directory containing the package.
if (pubspec.name != null) return pubspec.name;
if (dir != null) return path.basename(dir);
rnystrom@google.com
committed
Version get version => pubspec.version;
/// The parsed pubspec associated with this package.
final Pubspec pubspec;
/// The ids of the packages that this package depends on. This is what is
/// specified in the pubspec when this package depends on another.
List<PackageRef> get dependencies => pubspec.dependencies;
rnystrom@google.com
committed
/// Returns the path to the README file at the root of the entrypoint, or null
/// if no README file is found. If multiple READMEs are found, this uses the
/// same conventions as pub.dartlang.org for choosing the primary one: the
/// README with the fewest extensions that is lexically ordered first is
/// chosen.
Future<String> get readmePath {
return listDir(dir).then((entries) {
var readmes = entries.map(path.basename).
where((entry) => entry.contains(_README_REGEXP));
if (readmes.isEmpty) return;
return path.join(dir, readmes.min((readme1, readme2) {
var extensions1 = ".".allMatches(readme1).length;
var extensions2 = ".".allMatches(readme2).length;
var comparison = extensions1.compareTo(extensions2);
if (comparison != 0) return comparison;
return readme1.compareTo(readme2);
/// Loads the package whose root directory is [packageDir]. [name] is the
/// expected name of that package (e.g. the name given in the dependency), or
/// `null` if the package being loaded is the entrypoint package.
Package.load(String name, String packageDir, SourceRegistry sources)
: dir = packageDir,
pubspec = new Pubspec.load(name, packageDir, sources);
/// Constructs a package with the given pubspec. The package will have no
/// directory associated with it.
Package.inMemory(this.pubspec)
/// Constructs a package. This should not be called directly. Instead, acquire
/// packages from [load()].
Package._(this.dir, this.pubspec);
rnystrom@google.com
committed
/// Returns a debug string for the package.
String toString() => '$name $version ($dir)';
/// An unambiguous resolved reference to a package. A package ID contains enough
/// information to correctly install the package.
///
/// Note that it's possible for multiple distinct package IDs to point to
/// different directories that happen to contain identical packages. For
/// example, the same package may be available from multiple sources. As far as
/// Pub is concerned, those packages are different.
class PackageId implements Comparable<PackageId> {
/// The name of the package being identified.
final String name;
/// The [Source] used to look up this package given its [description]. If
/// this is a root package ID, this will be `null`.
final Source source;
final Version version;
/// The metadata used by the package's [source] to identify and locate it. It
/// contains whatever [Source]-specific data it needs to be able to install
/// the package. For example, the description of a git sourced package might
/// by the URL "git://github.com/dart/uilib.git".
final description;
PackageId(this.name, this.source, this.version, this.description);
/// Whether this ID identifies the root package.
bool get isRoot => source == null;
int get hashCode => name.hashCode ^ source.hashCode ^ version.hashCode;
/// Gets the directory where this package is or would be found in the
/// [SystemCache].
Future<String> get systemCacheDirectory => source.systemCacheDirectory(this);
bool operator ==(other) {
if (other is! PackageId) return false;
// TODO(rnystrom): We're assuming here the name/version/source tuple is
// enough to uniquely identify the package and that we don't need to delve
// into the description.
return other.name == name &&
other.version == version;
}
String toString() {
if (isRoot) return "$name $version (root)";
if (source.isDefault) return "$name $version";
return "$name $version from $source";
}
var sourceComp = source.name.compareTo(other.source.name);
if (sourceComp != 0) return sourceComp;
var nameComp = name.compareTo(other.name);
if (nameComp != 0) return nameComp;
return version.compareTo(other.version);
rnystrom@google.com
committed
}
/// Returns the pubspec for this package.
Future<Pubspec> describe() => source.describe(this);
/// Returns a future that completes to the resovled [PackageId] for this id.
Future<PackageId> get resolved => source.resolveId(this);
rnystrom@google.com
committed
}
/// A reference to a package. Unlike a [PackageId], a PackageRef may not
/// unambiguously refer to a single package. It may describe a range of allowed
/// packages.
class PackageRef {
/// The name of the package being identified.
final String name;
/// The [Source] used to look up the package. If this refers to a root
/// package, this will be `null`.
/// The allowed package versions.
final VersionConstraint constraint;
/// The metadata used to identify the package being referenced. The
/// interpretation of this will vary based on the [source].
final description;
PackageRef(this.name, this.source, this.constraint, this.description);
/// Creates a reference to the given root package.
PackageRef.root(Package package)
: name = package.name,
source = null,
constraint = package.version,
description = package.name;
/// Whether this refers to the root package.
bool get isRoot => source == null;
String toString() {
if (isRoot) return "$name $constraint (root)";
return "$name $constraint from $source ($description)";
}
/// Returns a [PackageId] generated from this [PackageRef] with the given
/// concrete version.
PackageId atVersion(Version version) =>
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
new PackageId(name, source, version, description);
}
class PubspecNotFoundException implements Exception {
final String name;
PubspecNotFoundException(this.name);
String toString() => 'Package "$name" doesn\'t have a pubspec.yaml file.';
}
class PubspecHasNoNameException implements Exception {
final String name;
PubspecHasNoNameException(this.name);
String toString() => 'Package "$name"\'s pubspec.yaml file is missing the '
'required "name" field (e.g. "name: $name").';
}
class PubspecNameMismatchException implements Exception {
final String expectedName;
final String actualName;
PubspecNameMismatchException(this.expectedName, this.actualName);
String toString() => 'The name you specified for your dependency, '
'"$expectedName", doesn\'t match the name "$actualName" in its pubspec.';