diff --git a/src/cargo/core/package_id.rs b/src/cargo/core/package_id.rs index c25a668e4e9..11c86e9006d 100644 --- a/src/cargo/core/package_id.rs +++ b/src/cargo/core/package_id.rs @@ -155,6 +155,14 @@ impl PackageId { PackageId::pure(self.inner.name, self.inner.version.clone(), source) } + pub fn map_source(self, to_replace: SourceId, replace_with: SourceId) -> Self { + if self.source_id() == to_replace { + self.with_source_id(replace_with) + } else { + self + } + } + pub fn stable_hash(self, workspace: &Path) -> PackageIdStableHash<'_> { PackageIdStableHash(self, workspace) } diff --git a/src/cargo/core/registry.rs b/src/cargo/core/registry.rs index 533b4cb0b72..bc1667071ae 100644 --- a/src/cargo/core/registry.rs +++ b/src/cargo/core/registry.rs @@ -1,4 +1,4 @@ -use std::collections::HashMap; +use std::collections::{HashMap, HashSet}; use log::{debug, trace}; use semver::VersionReq; @@ -71,6 +71,7 @@ pub struct PackageRegistry<'cfg> { source_ids: HashMap, locked: LockedMap, + yanked_whitelist: HashSet, source_config: SourceConfigMap<'cfg>, patches: HashMap>, @@ -97,6 +98,7 @@ impl<'cfg> PackageRegistry<'cfg> { overrides: Vec::new(), source_config, locked: HashMap::new(), + yanked_whitelist: HashSet::new(), patches: HashMap::new(), patches_locked: false, patches_available: HashMap::new(), @@ -166,6 +168,14 @@ impl<'cfg> PackageRegistry<'cfg> { self.add_source(source, Kind::Override); } + pub fn add_to_yanked_whitelist(&mut self, iter: impl Iterator) { + let pkgs = iter.collect::>(); + for (_, source) in self.sources.sources_mut() { + source.add_to_yanked_whitelist(&pkgs); + } + self.yanked_whitelist.extend(pkgs); + } + pub fn register_lock(&mut self, id: PackageId, deps: Vec) { trace!("register_lock: {}", id); for dep in deps.iter() { @@ -301,7 +311,7 @@ impl<'cfg> PackageRegistry<'cfg> { fn load(&mut self, source_id: SourceId, kind: Kind) -> CargoResult<()> { (|| { debug!("loading source {}", source_id); - let source = self.source_config.load(source_id)?; + let source = self.source_config.load(source_id, &self.yanked_whitelist)?; assert_eq!(source.source_id(), source_id); if kind == Kind::Override { diff --git a/src/cargo/core/source/mod.rs b/src/cargo/core/source/mod.rs index 62583147c99..50d1c63d869 100644 --- a/src/cargo/core/source/mod.rs +++ b/src/cargo/core/source/mod.rs @@ -83,6 +83,11 @@ pub trait Source { fn is_replaced(&self) -> bool { false } + + /// Add a number of crates that should be whitelisted for showing up during + /// queries, even if they are yanked. Currently only applies to registry + /// sources. + fn add_to_yanked_whitelist(&mut self, pkgs: &[PackageId]); } pub enum MaybePackage { @@ -152,6 +157,10 @@ impl<'a, T: Source + ?Sized + 'a> Source for Box { fn is_replaced(&self) -> bool { (**self).is_replaced() } + + fn add_to_yanked_whitelist(&mut self, pkgs: &[PackageId]) { + (**self).add_to_yanked_whitelist(pkgs); + } } impl<'a, T: Source + ?Sized + 'a> Source for &'a mut T { @@ -206,6 +215,10 @@ impl<'a, T: Source + ?Sized + 'a> Source for &'a mut T { fn is_replaced(&self) -> bool { (**self).is_replaced() } + + fn add_to_yanked_whitelist(&mut self, pkgs: &[PackageId]) { + (**self).add_to_yanked_whitelist(pkgs); + } } /// A `HashMap` of `SourceId` -> `Box` diff --git a/src/cargo/core/source/source_id.rs b/src/cargo/core/source/source_id.rs index 72f382355f0..a96ec53e629 100644 --- a/src/cargo/core/source/source_id.rs +++ b/src/cargo/core/source/source_id.rs @@ -13,6 +13,7 @@ use serde::de; use serde::ser; use url::Url; +use crate::core::PackageId; use crate::ops; use crate::sources::git; use crate::sources::DirectorySource; @@ -257,7 +258,11 @@ impl SourceId { } /// Creates an implementation of `Source` corresponding to this ID. - pub fn load<'a>(self, config: &'a Config) -> CargoResult> { + pub fn load<'a>( + self, + config: &'a Config, + yanked_whitelist: &HashSet, + ) -> CargoResult> { trace!("loading SourceId; {}", self); match self.inner.kind { Kind::Git(..) => Ok(Box::new(GitSource::new(self, config)?)), @@ -268,7 +273,11 @@ impl SourceId { }; Ok(Box::new(PathSource::new(&path, self, config))) } - Kind::Registry => Ok(Box::new(RegistrySource::remote(self, config))), + Kind::Registry => Ok(Box::new(RegistrySource::remote( + self, + yanked_whitelist, + config, + ))), Kind::LocalRegistry => { let path = match self.inner.url.to_file_path() { Ok(p) => p, diff --git a/src/cargo/ops/cargo_install.rs b/src/cargo/ops/cargo_install.rs index 4e1036abb52..3d80b7a71d9 100644 --- a/src/cargo/ops/cargo_install.rs +++ b/src/cargo/ops/cargo_install.rs @@ -1,4 +1,4 @@ -use std::collections::{BTreeMap, BTreeSet}; +use std::collections::{BTreeMap, BTreeSet, HashSet}; use std::path::{Path, PathBuf}; use std::sync::Arc; use std::{env, fs}; @@ -181,7 +181,7 @@ fn install_one( })? } else { select_pkg( - map.load(source_id)?, + map.load(source_id, &HashSet::new())?, krate, vers, config, diff --git a/src/cargo/ops/registry.rs b/src/cargo/ops/registry.rs index 04cf2886918..4447dc9699e 100644 --- a/src/cargo/ops/registry.rs +++ b/src/cargo/ops/registry.rs @@ -1,4 +1,4 @@ -use std::collections::BTreeMap; +use std::collections::{BTreeMap, HashSet}; use std::fs::{self, File}; use std::io::{self, BufRead}; use std::iter::repeat; @@ -350,7 +350,7 @@ pub fn registry( let token = token.or(token_config); let sid = get_source_id(config, index_config.or(index), registry)?; let api_host = { - let mut src = RegistrySource::remote(sid, config); + let mut src = RegistrySource::remote(sid, &HashSet::new(), config); // Only update the index if the config is not available or `force` is set. let cfg = src.config(); let cfg = if force_update || cfg.is_err() { @@ -696,7 +696,7 @@ fn get_source_id( (_, Some(i)) => SourceId::for_registry(&i.to_url()?), _ => { let map = SourceConfigMap::new(config)?; - let src = map.load(SourceId::crates_io(config)?)?; + let src = map.load(SourceId::crates_io(config)?, &HashSet::new())?; Ok(src.replaced_source_id()) } } diff --git a/src/cargo/ops/resolve.rs b/src/cargo/ops/resolve.rs index 33f2a2c038c..81ce837ad1b 100644 --- a/src/cargo/ops/resolve.rs +++ b/src/cargo/ops/resolve.rs @@ -471,6 +471,7 @@ fn register_previous_locks( // package's dependencies here as that'll be covered below to poison those // if they changed. let mut avoid_locking = HashSet::new(); + registry.add_to_yanked_whitelist(resolve.iter().filter(keep)); for node in resolve.iter() { if !keep(&node) { add_deps(resolve, node, &mut avoid_locking); diff --git a/src/cargo/sources/config.rs b/src/cargo/sources/config.rs index ccc66b693f2..e1975fc624d 100644 --- a/src/cargo/sources/config.rs +++ b/src/cargo/sources/config.rs @@ -4,13 +4,13 @@ //! structure usable by Cargo itself. Currently this is primarily used to map //! sources to one another via the `replace-with` key in `.cargo/config`. -use std::collections::HashMap; +use std::collections::{HashMap, HashSet}; use std::path::{Path, PathBuf}; use log::debug; use url::Url; -use crate::core::{GitReference, Source, SourceId}; +use crate::core::{GitReference, PackageId, Source, SourceId}; use crate::sources::{ReplacedSource, CRATES_IO_REGISTRY}; use crate::util::config::ConfigValue; use crate::util::errors::{CargoResult, CargoResultExt}; @@ -73,11 +73,15 @@ impl<'cfg> SourceConfigMap<'cfg> { self.config } - pub fn load(&self, id: SourceId) -> CargoResult> { + pub fn load( + &self, + id: SourceId, + yanked_whitelist: &HashSet, + ) -> CargoResult> { debug!("loading: {}", id); let mut name = match self.id2name.get(&id) { Some(name) => name, - None => return Ok(id.load(self.config)?), + None => return Ok(id.load(self.config, yanked_whitelist)?), }; let mut path = Path::new("/"); let orig_name = name; @@ -99,7 +103,7 @@ impl<'cfg> SourceConfigMap<'cfg> { name = s; path = p; } - None if id == cfg.id => return Ok(id.load(self.config)?), + None if id == cfg.id => return Ok(id.load(self.config, yanked_whitelist)?), None => { new_id = cfg.id.with_precise(id.precise().map(|s| s.to_string())); break; @@ -116,8 +120,15 @@ impl<'cfg> SourceConfigMap<'cfg> { ) } } - let new_src = new_id.load(self.config)?; - let old_src = id.load(self.config)?; + + let new_src = new_id.load( + self.config, + &yanked_whitelist + .iter() + .map(|p| p.map_source(id, new_id)) + .collect(), + )?; + let old_src = id.load(self.config, yanked_whitelist)?; if !new_src.supports_checksums() && old_src.supports_checksums() { failure::bail!( "\ diff --git a/src/cargo/sources/directory.rs b/src/cargo/sources/directory.rs index c1a4b4dc771..8009a149dc0 100644 --- a/src/cargo/sources/directory.rs +++ b/src/cargo/sources/directory.rs @@ -214,4 +214,6 @@ impl<'cfg> Source for DirectorySource<'cfg> { fn describe(&self) -> String { format!("directory source `{}`", self.root.display()) } + + fn add_to_yanked_whitelist(&mut self, _pkgs: &[PackageId]) {} } diff --git a/src/cargo/sources/git/source.rs b/src/cargo/sources/git/source.rs index 9b96ede0308..ff60990254f 100644 --- a/src/cargo/sources/git/source.rs +++ b/src/cargo/sources/git/source.rs @@ -238,6 +238,8 @@ impl<'cfg> Source for GitSource<'cfg> { fn describe(&self) -> String { format!("git repository {}", self.source_id) } + + fn add_to_yanked_whitelist(&mut self, _pkgs: &[PackageId]) {} } #[cfg(test)] diff --git a/src/cargo/sources/path.rs b/src/cargo/sources/path.rs index e64e0361b4f..e550efd7f46 100644 --- a/src/cargo/sources/path.rs +++ b/src/cargo/sources/path.rs @@ -569,4 +569,6 @@ impl<'cfg> Source for PathSource<'cfg> { Err(_) => self.source_id.to_string(), } } + + fn add_to_yanked_whitelist(&mut self, _pkgs: &[PackageId]) {} } diff --git a/src/cargo/sources/registry/index.rs b/src/cargo/sources/registry/index.rs index b165c4c65d1..249eb65f79a 100644 --- a/src/cargo/sources/registry/index.rs +++ b/src/cargo/sources/registry/index.rs @@ -1,4 +1,4 @@ -use std::collections::HashMap; +use std::collections::{HashMap, HashSet}; use std::path::Path; use std::str; @@ -273,6 +273,7 @@ impl<'cfg> RegistryIndex<'cfg> { &mut self, dep: &Dependency, load: &mut dyn RegistryData, + yanked_whitelist: &HashSet, f: &mut dyn FnMut(Summary), ) -> CargoResult<()> { let source_id = self.source_id; @@ -280,7 +281,13 @@ impl<'cfg> RegistryIndex<'cfg> { let summaries = self.summaries(name, load)?; let summaries = summaries .iter() - .filter(|&&(_, yanked)| dep.source_id().precise().is_some() || !yanked) + .filter(|&(summary, yanked)| { + !yanked || { + log::debug!("{:?}", yanked_whitelist); + log::debug!("{:?}", summary.package_id()); + yanked_whitelist.contains(&summary.package_id()) + } + }) .map(|s| s.0.clone()); // Handle `cargo update --precise` here. If specified, our own source diff --git a/src/cargo/sources/registry/mod.rs b/src/cargo/sources/registry/mod.rs index 474e3a884a7..33479dde648 100644 --- a/src/cargo/sources/registry/mod.rs +++ b/src/cargo/sources/registry/mod.rs @@ -160,6 +160,7 @@ use std::borrow::Cow; use std::collections::BTreeMap; +use std::collections::HashSet; use std::io::Write; use std::path::{Path, PathBuf}; @@ -192,6 +193,7 @@ pub struct RegistrySource<'cfg> { updated: bool, ops: Box, index: index::RegistryIndex<'cfg>, + yanked_whitelist: HashSet, index_locked: bool, } @@ -385,16 +387,34 @@ fn short_name(id: SourceId) -> String { } impl<'cfg> RegistrySource<'cfg> { - pub fn remote(source_id: SourceId, config: &'cfg Config) -> RegistrySource<'cfg> { + pub fn remote( + source_id: SourceId, + yanked_whitelist: &HashSet, + config: &'cfg Config, + ) -> RegistrySource<'cfg> { let name = short_name(source_id); let ops = remote::RemoteRegistry::new(source_id, config, &name); - RegistrySource::new(source_id, config, &name, Box::new(ops), true) + RegistrySource::new( + source_id, + config, + &name, + Box::new(ops), + yanked_whitelist, + true, + ) } pub fn local(source_id: SourceId, path: &Path, config: &'cfg Config) -> RegistrySource<'cfg> { let name = short_name(source_id); let ops = local::LocalRegistry::new(path, config, &name); - RegistrySource::new(source_id, config, &name, Box::new(ops), false) + RegistrySource::new( + source_id, + config, + &name, + Box::new(ops), + &HashSet::new(), + false, + ) } fn new( @@ -402,6 +422,7 @@ impl<'cfg> RegistrySource<'cfg> { config: &'cfg Config, name: &str, ops: Box, + yanked_whitelist: &HashSet, index_locked: bool, ) -> RegistrySource<'cfg> { RegistrySource { @@ -410,6 +431,7 @@ impl<'cfg> RegistrySource<'cfg> { source_id, updated: false, index: index::RegistryIndex::new(source_id, ops.index_path(), config, index_locked), + yanked_whitelist: yanked_whitelist.clone(), index_locked, ops, } @@ -431,9 +453,7 @@ impl<'cfg> RegistrySource<'cfg> { // unpacked and to lock the directory for unpacking. let mut ok = { let package_dir = format!("{}-{}", pkg.name(), pkg.version()); - let dst = self - .src_path - .join(&package_dir); + let dst = self.src_path.join(&package_dir); dst.create_dir()?; // Attempt to open a read-only copy first to avoid an exclusive write @@ -526,12 +546,13 @@ impl<'cfg> Source for RegistrySource<'cfg> { if dep.source_id().precise().is_some() && !self.updated { debug!("attempting query without update"); let mut called = false; - self.index.query_inner(dep, &mut *self.ops, &mut |s| { - if dep.matches(&s) { - called = true; - f(s); - } - })?; + self.index + .query_inner(dep, &mut *self.ops, &self.yanked_whitelist, &mut |s| { + if dep.matches(&s) { + called = true; + f(s); + } + })?; if called { return Ok(()); } else { @@ -540,15 +561,17 @@ impl<'cfg> Source for RegistrySource<'cfg> { } } - self.index.query_inner(dep, &mut *self.ops, &mut |s| { - if dep.matches(&s) { - f(s); - } - }) + self.index + .query_inner(dep, &mut *self.ops, &self.yanked_whitelist, &mut |s| { + if dep.matches(&s) { + f(s); + } + }) } fn fuzzy_query(&mut self, dep: &Dependency, f: &mut dyn FnMut(Summary)) -> CargoResult<()> { - self.index.query_inner(dep, &mut *self.ops, f) + self.index + .query_inner(dep, &mut *self.ops, &self.yanked_whitelist, f) } fn supports_checksums(&self) -> bool { @@ -602,4 +625,8 @@ impl<'cfg> Source for RegistrySource<'cfg> { fn describe(&self) -> String { self.source_id.display_registry() } + + fn add_to_yanked_whitelist(&mut self, pkgs: &[PackageId]) { + self.yanked_whitelist.extend(pkgs); + } } diff --git a/src/cargo/sources/replaced.rs b/src/cargo/sources/replaced.rs index 4a4de90a419..465b0f08de4 100644 --- a/src/cargo/sources/replaced.rs +++ b/src/cargo/sources/replaced.rs @@ -113,4 +113,10 @@ impl<'cfg> Source for ReplacedSource<'cfg> { fn is_replaced(&self) -> bool { true } + + fn add_to_yanked_whitelist(&mut self, pkgs: &[PackageId]) { + let pkgs = pkgs.iter().map(|id| id.with_source_id(self.replace_with)) + .collect::>(); + self.inner.add_to_yanked_whitelist(&pkgs); + } } diff --git a/tests/testsuite/registry.rs b/tests/testsuite/registry.rs index efc88b72c3c..d56e714109a 100644 --- a/tests/testsuite/registry.rs +++ b/tests/testsuite/registry.rs @@ -590,6 +590,104 @@ required by package `foo v0.0.1 ([..])` .run(); } +#[test] +fn yanks_in_lockfiles_are_ok_for_other_update() { + let p = project() + .file( + "Cargo.toml", + r#" + [project] + name = "foo" + version = "0.0.1" + authors = [] + + [dependencies] + bar = "*" + baz = "*" + "#, + ) + .file("src/main.rs", "fn main() {}") + .build(); + + Package::new("bar", "0.0.1").publish(); + Package::new("baz", "0.0.1").publish(); + + p.cargo("build").run(); + + registry_path().join("3").rm_rf(); + + Package::new("bar", "0.0.1").yanked(true).publish(); + Package::new("baz", "0.0.1").publish(); + + p.cargo("build").with_stdout("").run(); + + Package::new("baz", "0.0.2").publish(); + + p.cargo("update") + .with_status(101) + .with_stderr_contains( + "\ +error: no matching package named `bar` found +location searched: registry [..] +required by package `foo v0.0.1 ([..])` +", + ) + .run(); + + p.cargo("update -p baz") + .with_status(0) + .with_stderr_contains( + "\ +[UPDATING] `[..]` index +[UPDATING] baz v0.0.1 -> v0.0.2 +", + ) + .run(); +} + +#[test] +fn yanks_in_lockfiles_are_ok_with_new_dep() { + let p = project() + .file( + "Cargo.toml", + r#" + [project] + name = "foo" + version = "0.0.1" + authors = [] + + [dependencies] + bar = "*" + "#, + ) + .file("src/main.rs", "fn main() {}") + .build(); + + Package::new("bar", "0.0.1").publish(); + + p.cargo("build").run(); + + registry_path().join("3").rm_rf(); + + Package::new("bar", "0.0.1").yanked(true).publish(); + Package::new("baz", "0.0.1").publish(); + + t!(t!(File::create(p.root().join("Cargo.toml"))).write_all( + br#" + [project] + name = "foo" + version = "0.0.1" + authors = [] + + [dependencies] + bar = "*" + baz = "*" + "# + )); + + p.cargo("build").with_stdout("").run(); +} + #[test] fn update_with_lockfile_if_packages_missing() { let p = project() diff --git a/tests/testsuite/search.rs b/tests/testsuite/search.rs index f80925a2afd..e0f7f4807ec 100644 --- a/tests/testsuite/search.rs +++ b/tests/testsuite/search.rs @@ -1,3 +1,4 @@ +use std::collections::HashSet; use std::fs::{self, File}; use std::io::prelude::*; use std::path::Path; @@ -107,7 +108,7 @@ fn not_update() { let sid = SourceId::for_registry(®istry_url()).unwrap(); let cfg = Config::new(Shell::new(), paths::root(), paths::home().join(".cargo")); - let mut regsrc = RegistrySource::remote(sid, &cfg); + let mut regsrc = RegistrySource::remote(sid, &HashSet::new(), &cfg); regsrc.update().unwrap(); cargo_process("search postgres")