Skip to content

Commit f09ef68

Browse files
authored
Auto merge of #2857 - alexcrichton:redirect-sources, r=brson
Add support local mirrors of registries, take 2 This series of commits culminates in first class support in Cargo for local mirrors of registries. This is implemented through a number of other more generic mechanisms, and extra support was added along the way. The highlights of this PR, however, are: New `.cargo/config` keys have been added to enable *replacing one source with another*. This functionality is intended to be used for mirrors of the main registry or otherwise one to one source correspondences. The support looks like: ```toml [source.crates-io] replace-with = 'my-awesome-registry' [source.my-awesome-registry] registry = 'https://github.com/my-awesome/registry-index' ``` This configuration means that instead of using `crates-io` (e.g. `https://github.com/rust-lang/crates.io-index`), Cargo will query the `my-awesome-registry` source instead (configured to a different index here). This alternate source **must be the exact same as the crates.io index**. Cargo assumes that replacement sources are exact 1:1 mirrors in this respect, and the following support is designed around that assumption. When generating a lock file for crate using a replacement registry, the *original registry* will be encoded into the lock file. For example in the configuration above, all lock files will still mention crates.io as the registry that packages originated from. This semantically represents how crates.io is the source of truth for all crates, and this is upheld because all replacements have a 1:1 correspondance. Overall, this means that no matter what replacement source you're working with, you can ship your lock file to anyone else and you'll all still have verifiably reproducible builds! With the above support for custom registries, it's now possible for a project to be downloading crates from any number of sources. One of Cargo's core goals is reproducible builds, and with all these new sources of information it may be easy for a few situations to arise: 1. A local replacement of crates.io could be corrupt 2. A local replacement of crates.io could have made subtle changes to crates In both of these cases, Cargo would today simply give non-reproducible builds. To help assuage these concerns, Cargo will now track the sha256 checksum of all crates from registries in the lock file. Whenever a `Cargo.lock` is generated from now on it will contain a `[metadata]` section which lists the sha256 checksum of all crates in the lock file (or `<none>` if the sha256 checksum isn't known). Cargo already checks registry checksums against what's actually downloaded, and Cargo will now verify between iterations of the lock file that checksums remain the same as well. This means that if a local replacement registry is **not** in a 1:1 correspondance with crates.io, the lock file will prevent the build from progressing until the discrepancy is resolved. In addition to the support above, there is now a new kind of source in Cargo, a "local registry", which is intended to be a subset of the crates.io ecosystem purposed for a local build for any particular project here or there. The way to enable this looks like: ```toml [source.crates-io] replace-with = 'my-awesome-registry' [source.my-awesome-registry] local-registry = 'path/to/my/local/registry' ``` This local registry is expected to have two components: 1. A directory called `index` which matches the same structure as the crates.io index. The `config.json` file is not required here. 2. Inside the registry directory are any number of `.crate` files (downloaded from crates.io). Each crate file has the name `<package>-<version>.crate`. This local registry must currently be managed manually, but I plan on publishing and maintaining a Cargo subcommand to manage a local registry. It will have options to do things like: 1. Sync a local registry with a `Cargo.lock` 2. Add a registry package to a local registry 3. Remove a package from a local registry In addition to local registries, Cargo also supports a "directory source" like so ```toml [source.crates-io] replace-with = 'my-awesome-registry' [source.my-awesome-registry] directory = 'path/to/some/sources' ``` A directory source is similar to a local registry above, except that all the crates are unpacked and visible as vendored source. This format is suitable for checking into source trees, like Gecko's. Unlike local registries above we don't have a tarball to verify the crates.io checksum with, but each vendored dependency has metadata containing what it *would* have been. To further prevent modifications by accident, the metadata contains the checksum of each file which should prevent accidental local modifications and steer towards `[replace]` as the mechanism to edit dependencies if necessary. This is quite a bit of new features! What's all this meant to do? Some example scenarios that this is envisioned to solve are: 1. Supporting mirrors for crates.io in a first class fashion. Once we have the ability to spin up your own local registry, it should be easy to locally select a new mirror. 2. Supporting round-robin mirrors, this provides an easy vector for configuration of "instead of crates.io hit the first source in this list that works" 3. Build environments where network access is not an option. Preparing a local registry ahead-of-time (from a known good lock file) will be a vector to ensure that all Rust dependencies are locally available. * Note this is intended to include use cases like Debian and Gecko Even with the new goodies here, there's some more vectors through which this can be expanded: * Support for running your own mirror of crates.io needs to be implemented to be "easy to do". There should for example be a `cargo install foo` available to have everything "Just Work". * Replacing a source with a list of sources (attempted in round robin fashion) needs to be implemented * Eventually this support will be extended to the `Cargo.toml` file itself. For example: * packages should be downloadable from multiple registries * replacement sources should be encodable into `Cargo.toml` (note that these replacements, unlike the ones above, would be encoded into `Cargo.lock`) * adding multiple mirrors to a `Cargo.toml` should be supported * Implementing the subcommand above to manage local registries needs to happen (I will attend to this shortly)
2 parents 01bdd66 + 63ac9e1 commit f09ef68

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

54 files changed

+3242
-912
lines changed

Makefile.in

+1-1
Original file line numberDiff line numberDiff line change
@@ -136,7 +136,7 @@ clean:
136136
# === Documentation
137137

138138
DOCS := index faq config guide manifest build-script pkgid-spec crates-io \
139-
environment-variables specifying-dependencies
139+
environment-variables specifying-dependencies source-replacement
140140
DOC_DIR := target/doc
141141
DOC_OPTS := --markdown-no-toc \
142142
--markdown-css stylesheets/normalize.css \

src/bin/git_checkout.rs

+3-9
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
use cargo::core::source::{Source, SourceId, GitReference};
22
use cargo::sources::git::{GitSource};
3-
use cargo::util::{Config, CliResult, CliError, human, ToUrl};
3+
use cargo::util::{Config, CliResult, ToUrl};
44

55
#[derive(RustcDecodable)]
66
pub struct Options {
@@ -37,20 +37,14 @@ pub fn execute(options: Options, config: &Config) -> CliResult<Option<()>> {
3737
options.flag_locked));
3838
let Options { flag_url: url, flag_reference: reference, .. } = options;
3939

40-
let url = try!(url.to_url().map_err(|e| {
41-
human(format!("The URL `{}` you passed was \
42-
not a valid URL: {}", url, e))
43-
})
44-
.map_err(|e| CliError::new(e, 1)));
40+
let url = try!(url.to_url());
4541

4642
let reference = GitReference::Branch(reference.clone());
4743
let source_id = SourceId::for_git(&url, reference);
4844

4945
let mut source = GitSource::new(&source_id, config);
5046

51-
try!(source.update().map_err(|e| {
52-
CliError::new(human(format!("Couldn't update {:?}: {:?}", source, e)), 1)
53-
}));
47+
try!(source.update());
5448

5549
Ok(None)
5650
}

src/bin/install.rs

+3-3
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
use cargo::ops;
22
use cargo::core::{SourceId, GitReference};
3-
use cargo::util::{CliResult, Config, ToUrl, human};
3+
use cargo::util::{CliResult, Config, ToUrl};
44

55
#[derive(RustcDecodable)]
66
pub struct Options {
@@ -116,7 +116,7 @@ pub fn execute(options: Options, config: &Config) -> CliResult<Option<()>> {
116116
};
117117

118118
let source = if let Some(url) = options.flag_git {
119-
let url = try!(url.to_url().map_err(human));
119+
let url = try!(url.to_url());
120120
let gitref = if let Some(branch) = options.flag_branch {
121121
GitReference::Branch(branch)
122122
} else if let Some(tag) = options.flag_tag {
@@ -132,7 +132,7 @@ pub fn execute(options: Options, config: &Config) -> CliResult<Option<()>> {
132132
} else if options.arg_crate == None {
133133
try!(SourceId::for_path(&config.cwd()))
134134
} else {
135-
try!(SourceId::for_central(config))
135+
try!(SourceId::crates_io(config))
136136
};
137137

138138
let krate = options.arg_crate.as_ref().map(|s| &s[..]);

src/bin/login.rs

+3-3
Original file line numberDiff line numberDiff line change
@@ -43,10 +43,10 @@ pub fn execute(options: Options, config: &Config) -> CliResult<Option<()>> {
4343
let token = match options.arg_token.clone() {
4444
Some(token) => token,
4545
None => {
46-
let src = try!(SourceId::for_central(config));
47-
let mut src = RegistrySource::new(&src, config);
46+
let src = try!(SourceId::crates_io(config));
47+
let mut src = RegistrySource::remote(&src, config);
4848
try!(src.update());
49-
let config = try!(src.config());
49+
let config = try!(src.config()).unwrap();
5050
let host = options.flag_host.clone().unwrap_or(config.api);
5151
println!("please visit {}me and paste the API Token below", host);
5252
let mut line = String::new();

src/cargo/core/dependency.rs

+12
Original file line numberDiff line numberDiff line change
@@ -249,6 +249,7 @@ impl Dependency {
249249
pub fn is_transitive(&self) -> bool { self.inner.is_transitive() }
250250
pub fn is_build(&self) -> bool { self.inner.is_build() }
251251
pub fn is_optional(&self) -> bool { self.inner.is_optional() }
252+
252253
/// Returns true if the default features of the dependency are requested.
253254
pub fn uses_default_features(&self) -> bool {
254255
self.inner.uses_default_features()
@@ -263,6 +264,17 @@ impl Dependency {
263264
pub fn matches_id(&self, id: &PackageId) -> bool {
264265
self.inner.matches_id(id)
265266
}
267+
268+
pub fn map_source(self, to_replace: &SourceId, replace_with: &SourceId)
269+
-> Dependency {
270+
if self.source_id() != to_replace {
271+
self
272+
} else {
273+
Rc::try_unwrap(self.inner).unwrap_or_else(|r| (*r).clone())
274+
.set_source_id(replace_with.clone())
275+
.into_dependency()
276+
}
277+
}
266278
}
267279

268280
impl Platform {

src/cargo/core/manifest.rs

+10-1
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,8 @@ use std::path::{PathBuf, Path};
44
use semver::Version;
55
use rustc_serialize::{Encoder, Encodable};
66

7-
use core::{Dependency, PackageId, PackageIdSpec, Summary, WorkspaceConfig};
7+
use core::{Dependency, PackageId, Summary, SourceId, PackageIdSpec};
8+
use core::WorkspaceConfig;
89
use core::package_id::Metadata;
910

1011
pub enum EitherManifest {
@@ -232,6 +233,14 @@ impl Manifest {
232233
pub fn set_summary(&mut self, summary: Summary) {
233234
self.summary = summary;
234235
}
236+
237+
pub fn map_source(self, to_replace: &SourceId, replace_with: &SourceId)
238+
-> Manifest {
239+
Manifest {
240+
summary: self.summary.map_source(to_replace, replace_with),
241+
..self
242+
}
243+
}
235244
}
236245

237246
impl VirtualManifest {

src/cargo/core/package.rs

+8
Original file line numberDiff line numberDiff line change
@@ -99,6 +99,14 @@ impl Package {
9999
.filter(|&(d, _)| d < 4);
100100
matches.min_by_key(|t| t.0).map(|t| t.1)
101101
}
102+
103+
pub fn map_source(self, to_replace: &SourceId, replace_with: &SourceId)
104+
-> Package {
105+
Package {
106+
manifest: self.manifest.map_source(to_replace, replace_with),
107+
manifest_path: self.manifest_path,
108+
}
109+
}
102110
}
103111

104112
impl fmt::Display for Package {

src/cargo/core/package_id.rs

+33-7
Original file line numberDiff line numberDiff line change
@@ -13,12 +13,12 @@ use util::{CargoResult, CargoError, short_hash, ToSemver};
1313
use core::source::SourceId;
1414

1515
/// Identifier for a specific version of a package in a specific source.
16-
#[derive(Clone, Debug)]
16+
#[derive(Clone)]
1717
pub struct PackageId {
1818
inner: Arc<PackageIdInner>,
1919
}
2020

21-
#[derive(PartialEq, PartialOrd, Eq, Ord, Debug)]
21+
#[derive(PartialEq, PartialOrd, Eq, Ord)]
2222
struct PackageIdInner {
2323
name: String,
2424
version: semver::Version,
@@ -38,13 +38,19 @@ impl Decodable for PackageId {
3838
fn decode<D: Decoder>(d: &mut D) -> Result<PackageId, D::Error> {
3939
let string: String = try!(Decodable::decode(d));
4040
let regex = Regex::new(r"^([^ ]+) ([^ ]+) \(([^\)]+)\)$").unwrap();
41-
let captures = regex.captures(&string).expect("invalid serialized PackageId");
41+
let captures = try!(regex.captures(&string).ok_or_else(|| {
42+
d.error("invalid serialized PackageId")
43+
}));
4244

4345
let name = captures.at(1).unwrap();
4446
let version = captures.at(2).unwrap();
4547
let url = captures.at(3).unwrap();
46-
let version = semver::Version::parse(version).ok().expect("invalid version");
47-
let source_id = SourceId::from_url(url);
48+
let version = try!(semver::Version::parse(version).map_err(|_| {
49+
d.error("invalid version")
50+
}));
51+
let source_id = try!(SourceId::from_url(url).map_err(|e| {
52+
d.error(&e.to_string())
53+
}));
4854

4955
Ok(PackageId {
5056
inner: Arc::new(PackageIdInner {
@@ -151,6 +157,16 @@ impl PackageId {
151157
}),
152158
}
153159
}
160+
161+
pub fn with_source_id(&self, source: &SourceId) -> PackageId {
162+
PackageId {
163+
inner: Arc::new(PackageIdInner {
164+
name: self.inner.name.to_string(),
165+
version: self.inner.version.clone(),
166+
source_id: source.clone(),
167+
}),
168+
}
169+
}
154170
}
155171

156172
impl Metadata {
@@ -173,16 +189,26 @@ impl fmt::Display for PackageId {
173189
}
174190
}
175191

192+
impl fmt::Debug for PackageId {
193+
fn fmt(&self, f: &mut Formatter) -> fmt::Result {
194+
f.debug_struct("PackageId")
195+
.field("name", &self.inner.name)
196+
.field("version", &self.inner.version.to_string())
197+
.field("source", &self.inner.source_id.to_string())
198+
.finish()
199+
}
200+
}
201+
176202
#[cfg(test)]
177203
mod tests {
178204
use super::PackageId;
179205
use core::source::SourceId;
180-
use sources::RegistrySource;
206+
use sources::CRATES_IO;
181207
use util::ToUrl;
182208

183209
#[test]
184210
fn invalid_version_handled_nicely() {
185-
let loc = RegistrySource::default_url().to_url().unwrap();
211+
let loc = CRATES_IO.to_url().unwrap();
186212
let repo = SourceId::for_registry(&loc);
187213

188214
assert!(PackageId::new("foo", "1.0", &repo).is_err());

src/cargo/core/registry.rs

+23-7
Original file line numberDiff line numberDiff line change
@@ -3,13 +3,22 @@ use std::collections::{HashSet, HashMap};
33
use core::{Source, SourceId, SourceMap, Summary, Dependency, PackageId, Package};
44
use core::PackageSet;
55
use util::{CargoResult, ChainError, Config, human, profile};
6+
use sources::config::SourceConfigMap;
67

78
/// Source of information about a group of packages.
89
///
910
/// See also `core::Source`.
1011
pub trait Registry {
1112
/// Attempt to find the packages that match a dependency request.
1213
fn query(&mut self, name: &Dependency) -> CargoResult<Vec<Summary>>;
14+
15+
/// Returns whether or not this registry will return summaries with
16+
/// checksums listed.
17+
///
18+
/// By default, registries do not support checksums.
19+
fn supports_checksums(&self) -> bool {
20+
false
21+
}
1322
}
1423

1524
impl Registry for Vec<Summary> {
@@ -26,6 +35,12 @@ impl Registry for Vec<Package> {
2635
}
2736
}
2837

38+
impl<'a, T: ?Sized + Registry + 'a> Registry for Box<T> {
39+
fn query(&mut self, name: &Dependency) -> CargoResult<Vec<Summary>> {
40+
(**self).query(name)
41+
}
42+
}
43+
2944
/// This structure represents a registry of known packages. It internally
3045
/// contains a number of `Box<Source>` instances which are used to load a
3146
/// `Package` from.
@@ -41,7 +56,6 @@ impl Registry for Vec<Package> {
4156
/// operations if necessary) and is ready to be queried for packages.
4257
pub struct PackageRegistry<'cfg> {
4358
sources: SourceMap<'cfg>,
44-
config: &'cfg Config,
4559

4660
// A list of sources which are considered "overrides" which take precedent
4761
// when querying for packages.
@@ -65,6 +79,7 @@ pub struct PackageRegistry<'cfg> {
6579
source_ids: HashMap<SourceId, (SourceId, Kind)>,
6680

6781
locked: HashMap<SourceId, HashMap<String, Vec<(PackageId, Vec<PackageId>)>>>,
82+
source_config: SourceConfigMap<'cfg>,
6883
}
6984

7085
#[derive(PartialEq, Eq, Clone, Copy)]
@@ -75,14 +90,15 @@ enum Kind {
7590
}
7691

7792
impl<'cfg> PackageRegistry<'cfg> {
78-
pub fn new(config: &'cfg Config) -> PackageRegistry<'cfg> {
79-
PackageRegistry {
93+
pub fn new(config: &'cfg Config) -> CargoResult<PackageRegistry<'cfg>> {
94+
let source_config = try!(SourceConfigMap::new(config));
95+
Ok(PackageRegistry {
8096
sources: SourceMap::new(),
8197
source_ids: HashMap::new(),
8298
overrides: Vec::new(),
83-
config: config,
99+
source_config: source_config,
84100
locked: HashMap::new(),
85-
}
101+
})
86102
}
87103

88104
pub fn get(self, package_ids: &[PackageId]) -> PackageSet<'cfg> {
@@ -158,8 +174,8 @@ impl<'cfg> PackageRegistry<'cfg> {
158174

159175
fn load(&mut self, source_id: &SourceId, kind: Kind) -> CargoResult<()> {
160176
(|| {
161-
// Save off the source
162-
let source = source_id.load(self.config);
177+
let source = try!(self.source_config.load(source_id));
178+
163179
if kind == Kind::Override {
164180
self.overrides.push(source_id.clone());
165181
}

0 commit comments

Comments
 (0)