Newer
Older
// 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:yaml/yaml.dart';
import 'package:path/path.dart' as path;
import 'package.dart';
import 'source.dart';
import 'source_registry.dart';
import 'utils.dart';
import 'version.dart';
/// The parsed and validated contents of a pubspec file.
class Pubspec {
final String name;
final Version version;
/// The packages this package depends on.
rnystrom@google.com
committed
final List<PackageDep> dependencies;
/// The packages this package depends on when it is the root package.
rnystrom@google.com
committed
final List<PackageDep> devDependencies;
/// The environment-related metadata.
final PubspecEnvironment environment;
/// All pubspec fields. This includes the fields from which other properties
/// are derived.
final Map<String, Object> fields;
/// Loads the pubspec for a package [name] located in [packageDir].
factory Pubspec.load(String name, String packageDir, SourceRegistry sources) {
var pubspecPath = path.join(packageDir, 'pubspec.yaml');
if (!fileExists(pubspecPath)) throw new PubspecNotFoundException(name);
try {
var pubspec = new Pubspec.parse(pubspecPath, readTextFile(pubspecPath),
sources);
if (pubspec.name == null) {
throw new PubspecHasNoNameException(name);
}
if (name != null && pubspec.name != name) {
throw new PubspecNameMismatchException(name, pubspec.name);
}
return pubspec;
} on FormatException catch (ex) {
fail('Could not parse $pubspecPath:\n${ex.message}');
Pubspec(this.name, this.version, this.dependencies, this.devDependencies,
this.environment, [Map<String, Object> fields])
: this.fields = fields == null ? {} : fields;
Pubspec.empty()
: name = null,
version = Version.none,
rnystrom@google.com
committed
dependencies = <PackageDep>[],
devDependencies = <PackageDep>[],
environment = new PubspecEnvironment(),
/// Whether or not the pubspec has no contents.
name == null && version == Version.none && dependencies.isEmpty;
/// Returns a Pubspec object for an already-parsed map representing its
/// contents.
///
/// This will validate that [contents] is a valid pubspec.
factory Pubspec.fromMap(Map contents, SourceRegistry sources) =>
_parseMap(null, contents, sources);
// TODO(rnystrom): Instead of allowing a null argument here, split this up
// into load(), parse(), and _parse() like LockFile does.
/// Parses the pubspec stored at [filePath] whose text is [contents]. If the
/// pubspec doesn't define version for itself, it defaults to [Version.none].
/// [filePath] may be `null` if the pubspec is not on the user's local
/// file system.
factory Pubspec.parse(String filePath, String contents,
SourceRegistry sources) {
if (contents.trim() == '') return new Pubspec.empty();
var parsedPubspec = loadYaml(contents);
if (parsedPubspec == null) return new Pubspec.empty();
if (parsedPubspec is! Map) {
throw new FormatException('The pubspec must be a YAML mapping.');
}
return _parseMap(filePath, parsedPubspec, sources);
}
}
/// Evaluates whether the given [url] for [field] is valid.
///
/// Throws [FormatException] on an invalid url.
void _validateFieldUrl(url, String field) {
if (url is! String) {
throw new FormatException(
'The "$field" field should be a string, but was "$url".');
}
var goodScheme = new RegExp(r'^https?:');
if (!goodScheme.hasMatch(url)) {
throw new FormatException(
'The "$field" field should be an "http:" or "https:" URL, but '
'was "$url".');
}
}
Pubspec _parseMap(String filePath, Map map, SourceRegistry sources) {
var name = null;
var version = Version.none;
if (map.containsKey('name')) {
name = map['name'];
if (name is! String) {
'The pubspec "name" field should be a string, but was "$name".');
if (map.containsKey('version')) {
version = new Version.parse(map['version']);
}
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
var dependencies = _parseDependencies(filePath, sources,
map['dependencies']);
var devDependencies = _parseDependencies(filePath, sources,
map['dev_dependencies']);
// Make sure the same package doesn't appear as both a regular and dev
// dependency.
var dependencyNames = dependencies.map((dep) => dep.name).toSet();
var collisions = dependencyNames.intersection(
devDependencies.map((dep) => dep.name).toSet());
if (!collisions.isEmpty) {
var packageNames;
if (collisions.length == 1) {
packageNames = 'Package "${collisions.first}"';
} else {
var names = collisions.toList();
names.sort();
var buffer = new StringBuffer();
buffer.write("Packages ");
for (var i = 0; i < names.length; i++) {
buffer.write('"');
buffer.write(names[i]);
buffer.write('"');
if (i == names.length - 2) {
buffer.write(", ");
} else if (i == names.length - 1) {
buffer.write(", and ");
}
}
packageNames = buffer.toString();
throw new FormatException(
'$packageNames cannot appear in both "dependencies" and '
'"dev_dependencies".');
}
var environmentYaml = map['environment'];
var sdkConstraint = VersionConstraint.any;
if (environmentYaml != null) {
if (environmentYaml is! Map) {
'The pubspec "environment" field should be a map, but was '
'"$environmentYaml".');
var sdkYaml = environmentYaml['sdk'];
if (sdkYaml is! String) {
throw new FormatException(
'The "sdk" field of "environment" should be a string, but was '
'"$sdkYaml".');
sdkConstraint = new VersionConstraint.parse(sdkYaml);
}
var environment = new PubspecEnvironment(sdkConstraint);
// Even though the pub app itself doesn't use these fields, we validate
// them here so that users find errors early before they try to upload to
// the server:
// TODO(rnystrom): We should split this validation into separate layers:
// 1. Stuff that is required in any pubspec to perform any command. Things
// like "must have a name". That should go here.
// 2. Stuff that is required to upload a package. Things like "homepage
// must use a valid scheme". That should go elsewhere. pub upload should
// call it, and we should provide a separate command to show the user,
// and also expose it to the editor in some way.
if (map.containsKey('homepage')) {
_validateFieldUrl(map['homepage'], 'homepage');
}
if (map.containsKey('documentation')) {
_validateFieldUrl(map['documentation'], 'documentation');
if (map.containsKey('author') && map['author'] is! String) {
throw new FormatException(
'The "author" field should be a string, but was '
'${map["author"]}.');
if (map.containsKey('authors')) {
var authors = map['authors'];
if (authors is List) {
// All of the elements must be strings.
if (!authors.every((author) => author is String)) {
throw new FormatException('The "authors" field should be a string '
'or a list of strings, but was "$authors".');
}
} else if (authors is! String) {
throw new FormatException('The pubspec "authors" field should be a '
'string or a list of strings, but was "$authors".');
}
if (map.containsKey('author')) {
throw new FormatException('A pubspec should not have both an "author" '
'and an "authors" field.');
}
return new Pubspec(name, version, dependencies, devDependencies,
environment, map);
rnystrom@google.com
committed
List<PackageDep> _parseDependencies(String pubspecPath, SourceRegistry sources,
rnystrom@google.com
committed
var dependencies = <PackageDep>[];
// Allow an empty dependencies key.
if (yaml == null) return dependencies;
if (yaml is! Map || yaml.keys.any((e) => e is! String)) {
throw new FormatException(
'The pubspec dependencies should be a map of package names, but '
'was ${yaml}.');
}
yaml.forEach((name, spec) {
var description;
var sourceName;
var versionConstraint = new VersionRange();
if (spec == null) {
description = name;
sourceName = sources.defaultSource.name;
} else if (spec is String) {
description = name;
sourceName = sources.defaultSource.name;
versionConstraint = new VersionConstraint.parse(spec);
} else if (spec is Map) {
if (spec.containsKey('version')) {
versionConstraint = new VersionConstraint.parse(spec.remove('version'));
}
var sourceNames = spec.keys.toList();
if (sourceNames.length > 1) {
throw new FormatException(
'Dependency $name may only have one source: $sourceNames.');
}
sourceName = only(sourceNames);
if (sourceName is! String) {
throw new FormatException(
'Source name $sourceName should be a string.');
}
description = spec[sourceName];
} else {
throw new FormatException(
'Dependency specification $spec should be a string or a mapping.');
}
// If we have a valid source, use it to process the description. Allow
// unknown sources so pub doesn't choke on old pubspecs.
if (sources.contains(sourceName)) {
description = sources[sourceName].parseDescription(
pubspecPath, description, fromLockFile: false);
}
rnystrom@google.com
committed
dependencies.add(new PackageDep(
name, sourceName, versionConstraint, description));
});
return dependencies;
/// The environment-related metadata in the pubspec. Corresponds to the data
/// under the "environment:" key in the pubspec.
class PubspecEnvironment {
/// The version constraint specifying which SDK versions this package works
/// with.
final VersionConstraint sdkVersion;
PubspecEnvironment([VersionConstraint sdk])
: sdkVersion = sdk != null ? sdk : VersionConstraint.any;
}