diff --git a/src/cargo/ops/cargo_package.rs b/src/cargo/ops/cargo_package.rs index 7a13c849704e..9e0c1e8e8bd9 100644 --- a/src/cargo/ops/cargo_package.rs +++ b/src/cargo/ops/cargo_package.rs @@ -93,30 +93,6 @@ struct GitVcsInfo { dirty: bool, } -/// Packages a single package in a workspace, returning the resulting tar file. -/// -/// # Panics -/// Panics if `opts.list` is true. In that case you probably don't want to -/// actually build the package tarball; you should just make and print the list -/// of files. (We don't currently provide a public API for that, but see how -/// [`package`] does it.) -pub fn package_one( - ws: &Workspace<'_>, - pkg: &Package, - opts: &PackageOpts<'_>, -) -> CargoResult { - assert!(!opts.list); - - let ar_files = prepare_archive(ws, pkg, opts)?; - let tarball = create_package(ws, pkg, ar_files, None)?; - - if opts.verify { - run_verify(ws, pkg, &tarball, None, opts)?; - } - - Ok(tarball) -} - // Builds a tarball and places it in the output directory. fn create_package( ws: &Workspace<'_>, @@ -179,6 +155,29 @@ fn create_package( /// Returns the generated package files. If `opts.list` is true, skips /// generating package files and returns an empty list. pub fn package(ws: &Workspace<'_>, opts: &PackageOpts<'_>) -> CargoResult> { + Ok(do_package(ws, opts)?.into_iter().map(|x| x.2).collect()) +} + +/// Packages an entire workspace. +/// +/// Returns the generated package files and the dependencies between them. If +/// `opts.list` is true, skips generating package files and returns an empty +/// list. +pub(crate) fn package_with_dep_graph( + ws: &Workspace<'_>, + opts: &PackageOpts<'_>, +) -> CargoResult> { + let output = do_package(ws, opts)?; + + Ok(local_deps(output.into_iter().map( + |(pkg, opts, tarball)| (pkg, (opts.cli_features, tarball)), + ))) +} + +fn do_package<'a>( + ws: &Workspace<'_>, + opts: &PackageOpts<'a>, +) -> CargoResult, FileLock)>> { let specs = &opts.to_package.to_package_id_specs(ws)?; // If -p is used, we should check spec is matched with the members (See #13719) if let ops::Packages::Packages(_) = opts.to_package { @@ -264,7 +263,7 @@ pub fn package(ws: &Workspace<'_>, opts: &PackageOpts<'_>) -> CargoResult, - graph: Graph, +pub(crate) struct LocalDependencies { + pub packages: HashMap, + pub graph: Graph, } -impl LocalDependencies { - fn sort(&self) -> Vec<(Package, CliFeatures)> { +impl LocalDependencies { + pub fn sort(&self) -> Vec<(Package, T)> { self.graph .sort() .into_iter() @@ -335,9 +333,10 @@ impl LocalDependencies { /// ignoring dev dependencies. /// /// We assume that the packages all belong to this workspace. -fn local_deps(packages: impl Iterator) -> LocalDependencies { - let packages: HashMap = - packages.map(|pkg| (pkg.0.package_id(), pkg)).collect(); +fn local_deps(packages: impl Iterator) -> LocalDependencies { + let packages: HashMap = packages + .map(|(pkg, payload)| (pkg.package_id(), (pkg, payload))) + .collect(); // Dependencies have source ids but not package ids. We draw an edge // whenever a dependency's source id matches one of our packages. This is @@ -349,7 +348,7 @@ fn local_deps(packages: impl Iterator) -> LocalDe .collect(); let mut graph = Graph::new(); - for (pkg, _features) in packages.values() { + for (pkg, _payload) in packages.values() { graph.add(pkg.package_id()); for dep in pkg.dependencies() { // Ignore local dev-dependencies because they aren't needed for intra-workspace diff --git a/src/cargo/ops/mod.rs b/src/cargo/ops/mod.rs index 0cbf8f922030..2075c9ccc4c6 100644 --- a/src/cargo/ops/mod.rs +++ b/src/cargo/ops/mod.rs @@ -10,7 +10,7 @@ pub use self::cargo_fetch::{fetch, FetchOptions}; pub use self::cargo_install::{install, install_list}; pub use self::cargo_new::{init, new, NewOptions, NewProjectKind, VersionControl}; pub use self::cargo_output_metadata::{output_metadata, ExportInfo, OutputMetadataOptions}; -pub use self::cargo_package::{check_yanked, package, package_one, PackageOpts}; +pub use self::cargo_package::{check_yanked, package, PackageOpts}; pub use self::cargo_pkgid::pkgid; pub use self::cargo_read_manifest::read_package; pub use self::cargo_run::run; diff --git a/src/cargo/ops/registry/publish.rs b/src/cargo/ops/registry/publish.rs index 7fd3acfbec78..83108ee2fa8e 100644 --- a/src/cargo/ops/registry/publish.rs +++ b/src/cargo/ops/registry/publish.rs @@ -3,8 +3,11 @@ //! [1]: https://doc.rust-lang.org/nightly/cargo/reference/registry-web-api.html#publish use std::collections::BTreeMap; +use std::collections::HashMap; use std::collections::HashSet; use std::fs::File; +use std::io::Seek; +use std::io::SeekFrom; use std::time::Duration; use anyhow::bail; @@ -15,32 +18,36 @@ use cargo_util::paths; use crates_io::NewCrate; use crates_io::NewCrateDependency; use crates_io::Registry; +use itertools::Itertools; use crate::core::dependency::DepKind; use crate::core::manifest::ManifestMetadata; use crate::core::resolver::CliFeatures; use crate::core::Dependency; use crate::core::Package; +use crate::core::PackageId; use crate::core::PackageIdSpecQuery; use crate::core::SourceId; use crate::core::Workspace; use crate::ops; use crate::ops::PackageOpts; use crate::ops::Packages; +use crate::ops::RegistryOrIndex; use crate::sources::source::QueryKind; +use crate::sources::source::Source; use crate::sources::SourceConfigMap; use crate::sources::CRATES_IO_REGISTRY; use crate::util::auth; use crate::util::cache_lock::CacheLockMode; use crate::util::context::JobsConfig; use crate::util::toml::prepare_for_publish; +use crate::util::Graph; use crate::util::Progress; use crate::util::ProgressStyle; use crate::CargoResult; use crate::GlobalContext; use super::super::check_dep_has_version; -use super::RegistryOrIndex; pub struct PublishOpts<'gctx> { pub gctx: &'gctx GlobalContext, @@ -57,16 +64,23 @@ pub struct PublishOpts<'gctx> { } pub fn publish(ws: &Workspace<'_>, opts: &PublishOpts<'_>) -> CargoResult<()> { + let multi_package_mode = ws.gctx().cli_unstable().package_workspace; let specs = opts.to_publish.to_package_id_specs(ws)?; - if specs.len() > 1 { - bail!("the `-p` argument must be specified to select a single package to publish") + + if !multi_package_mode { + if specs.len() > 1 { + bail!("the `-p` argument must be specified to select a single package to publish") + } + if Packages::Default == opts.to_publish && ws.is_virtual() { + bail!("the `-p` argument must be specified in the root of a virtual workspace") + } } - if Packages::Default == opts.to_publish && ws.is_virtual() { - bail!("the `-p` argument must be specified in the root of a virtual workspace") + + let member_ids: Vec<_> = ws.members().map(|p| p.package_id()).collect(); + // Check that the specs match members. + for spec in &specs { + spec.query(member_ids.clone())?; } - let member_ids = ws.members().map(|p| p.package_id()); - // Check that the spec matches exactly one member. - specs[0].query(member_ids)?; let mut pkgs = ws.members_with_features(&specs, &opts.cli_features)?; // In `members_with_features_old`, it will add "current" package (determined by the cwd) // So we need filter @@ -74,60 +88,31 @@ pub fn publish(ws: &Workspace<'_>, opts: &PublishOpts<'_>) -> CargoResult<()> { .into_iter() .filter(|(m, _)| specs.iter().any(|spec| spec.matches(m.package_id()))) .collect(); - // Double check. It is safe theoretically, unless logic has updated. - assert_eq!(pkgs.len(), 1); - let (pkg, cli_features) = pkgs.pop().unwrap(); - - let mut publish_registry = match opts.reg_or_index.as_ref() { - Some(RegistryOrIndex::Registry(registry)) => Some(registry.clone()), - _ => None, - }; - if let Some(ref allowed_registries) = *pkg.publish() { - if publish_registry.is_none() && allowed_registries.len() == 1 { - // If there is only one allowed registry, push to that one directly, - // even though there is no registry specified in the command. - let default_registry = &allowed_registries[0]; - if default_registry != CRATES_IO_REGISTRY { - // Don't change the registry for crates.io and don't warn the user. - // crates.io will be defaulted even without this. - opts.gctx.shell().note(&format!( - "found `{}` as only allowed registry. Publishing to it automatically.", - default_registry - ))?; - publish_registry = Some(default_registry.clone()); + let just_pkgs: Vec<_> = pkgs.iter().map(|p| p.0).collect(); + let reg_or_index = match opts.reg_or_index.clone() { + Some(r) => { + validate_registry(&just_pkgs, Some(&r))?; + Some(r) + } + None => { + let reg = super::infer_registry(&just_pkgs)?; + validate_registry(&just_pkgs, reg.as_ref())?; + if let Some(RegistryOrIndex::Registry(ref registry)) = ® { + if registry != CRATES_IO_REGISTRY { + // Don't warn for crates.io. + opts.gctx.shell().note(&format!( + "found `{}` as only allowed registry. Publishing to it automatically.", + registry + ))?; + } } + reg } + }; - let reg_name = publish_registry - .clone() - .unwrap_or_else(|| CRATES_IO_REGISTRY.to_string()); - if allowed_registries.is_empty() { - bail!( - "`{}` cannot be published.\n\ - `package.publish` must be set to `true` or a non-empty list in Cargo.toml to publish.", - pkg.name(), - ); - } else if !allowed_registries.contains(®_name) { - bail!( - "`{}` cannot be published.\n\ - The registry `{}` is not listed in the `package.publish` value in Cargo.toml.", - pkg.name(), - reg_name - ); - } - } // This is only used to confirm that we can create a token before we build the package. // This causes the credential provider to be called an extra time, but keeps the same order of errors. - let ver = pkg.version().to_string(); - let operation = Operation::Read; - - let reg_or_index = match opts.reg_or_index.clone() { - Some(RegistryOrIndex::Registry(_)) | None => { - publish_registry.map(RegistryOrIndex::Registry) - } - val => val, - }; let source_ids = super::get_source_id(opts.gctx, reg_or_index.as_ref())?; let mut registry = super::registry( opts.gctx, @@ -135,73 +120,136 @@ pub fn publish(ws: &Workspace<'_>, opts: &PublishOpts<'_>) -> CargoResult<()> { opts.token.as_ref().map(Secret::as_deref), reg_or_index.as_ref(), true, - Some(operation).filter(|_| !opts.dry_run), + Some(Operation::Read).filter(|_| !opts.dry_run), )?; - verify_dependencies(pkg, ®istry, source_ids.original)?; - // Prepare a tarball, with a non-suppressible warning if metadata - // is missing since this is being put online. - let tarball = ops::package_one( + // Validate all the packages before publishing any of them. + for (pkg, _) in &pkgs { + verify_dependencies(pkg, ®istry, source_ids.original)?; + } + + let specs = Packages::from_flags( + false, + vec![], + just_pkgs.iter().map(|p| p.name().to_string()).collect(), + )?; + let pkg_dep_graph = ops::cargo_package::package_with_dep_graph( ws, - pkg, &PackageOpts { gctx: opts.gctx, verify: opts.verify, list: false, check_metadata: true, allow_dirty: opts.allow_dirty, - to_package: Packages::Default, + to_package: specs, targets: opts.targets.clone(), jobs: opts.jobs.clone(), keep_going: opts.keep_going, - cli_features, - reg_or_index, + cli_features: opts.cli_features.clone(), + reg_or_index: reg_or_index.clone(), }, )?; - if !opts.dry_run { - let hash = cargo_util::Sha256::new() - .update_file(tarball.file())? - .finish_hex(); - let operation = Operation::Publish { - name: pkg.name().as_str(), - vers: &ver, - cksum: &hash, - }; - registry.set_token(Some(auth::auth_token( - &opts.gctx, - &source_ids.original, - None, - operation, - vec![], - false, - )?)); - } + let (mut order, mut ready) = PublishOrder::new(&pkg_dep_graph.graph); + let mut to_confirm: HashSet<_> = ready.iter().copied().collect(); + let mut remaining: HashSet<_> = just_pkgs.iter().map(|pkg| pkg.package_id()).collect(); - opts.gctx - .shell() - .status("Uploading", pkg.package_id().to_string())?; - transmit( - opts.gctx, - ws, - pkg, - tarball.file(), - &mut registry, - source_ids.original, - opts.dry_run, - )?; - if !opts.dry_run { - const DEFAULT_TIMEOUT: u64 = 60; - let timeout = if opts.gctx.cli_unstable().publish_timeout { - let timeout: Option = opts.gctx.get("publish.timeout")?; - timeout.unwrap_or(DEFAULT_TIMEOUT) + while !to_confirm.is_empty() { + if !ready.is_empty() { + opts.gctx.shell().status( + "Uploading", + ready.iter().map(PackageId::to_string).sorted().join(", "), + )?; + } + + for pkg_id in &ready { + let (pkg, (_features, tarball)) = &pkg_dep_graph.packages[pkg_id]; + + if !opts.dry_run { + let ver = pkg.version().to_string(); + + tarball.file().seek(SeekFrom::Start(0))?; + let hash = cargo_util::Sha256::new() + .update_file(tarball.file())? + .finish_hex(); + let operation = Operation::Publish { + name: pkg.name().as_str(), + vers: &ver, + cksum: &hash, + }; + registry.set_token(Some(auth::auth_token( + &opts.gctx, + &source_ids.original, + None, + operation, + vec![], + false, + )?)); + } + + transmit( + opts.gctx, + ws, + pkg, + tarball.file(), + &mut registry, + source_ids.original, + opts.dry_run, + )?; + + if !opts.dry_run { + // Short does not include the registry name. + let short_pkg_description = format!("{} v{}", pkg.name(), pkg.version()); + let source_description = source_ids.original.to_string(); + ws.gctx().shell().status( + "Uploaded", + format!("{short_pkg_description} to {source_description}"), + )?; + } + } + + let finished = if opts.dry_run { + ready.clone() } else { - DEFAULT_TIMEOUT + const DEFAULT_TIMEOUT: u64 = 60; + let timeout = if opts.gctx.cli_unstable().publish_timeout { + let timeout: Option = opts.gctx.get("publish.timeout")?; + timeout.unwrap_or(DEFAULT_TIMEOUT) + } else { + DEFAULT_TIMEOUT + }; + if 0 < timeout { + let timeout = Duration::from_secs(timeout); + wait_for_publish(opts.gctx, source_ids.original, &to_confirm, timeout)? + } else { + Vec::new() + } }; - if 0 < timeout { - let timeout = Duration::from_secs(timeout); - wait_for_publish(opts.gctx, source_ids.original, pkg, timeout)?; + + for id in &finished { + to_confirm.remove(id); + remaining.remove(id); + } + if finished.is_empty() { + // If nothing finished, it means we timed out while waiting for confirmation. + // We're going to exit, but first we need to check: if we pretend that the unconfirmed + // uploads are confirmed, did we finish everything? + for id in &ready { + remaining.remove(id); + } + + if remaining.is_empty() { + // It's ok that we timed out, because nothing was waiting on dependencies to + // be confirmed. + break; + } else { + let failed_list = package_list(remaining, "and"); + bail!("failed to publish {failed_list} because we timed out waiting for dependencies."); + } } + + ready = order.mark_published(finished); + to_confirm.extend(ready.iter().copied()); } Ok(()) @@ -210,34 +258,28 @@ pub fn publish(ws: &Workspace<'_>, opts: &PublishOpts<'_>) -> CargoResult<()> { fn wait_for_publish( gctx: &GlobalContext, registry_src: SourceId, - pkg: &Package, + pkgs: &HashSet, timeout: Duration, -) -> CargoResult<()> { - let version_req = format!("={}", pkg.version()); +) -> CargoResult> { let mut source = SourceConfigMap::empty(gctx)?.load(registry_src, &HashSet::new())?; // Disable the source's built-in progress bars. Repeatedly showing a bunch // of independent progress bars can be a little confusing. There is an // overall progress bar managed here. source.set_quiet(true); let source_description = source.source_id().to_string(); - let query = Dependency::parse(pkg.name(), Some(&version_req), registry_src)?; let now = std::time::Instant::now(); let sleep_time = Duration::from_secs(1); let max = timeout.as_secs() as usize; // Short does not include the registry name. - let short_pkg_description = format!("{} v{}", pkg.name(), pkg.version()); - gctx.shell().status( - "Uploaded", - format!("{short_pkg_description} to {source_description}"), - )?; + let short_pkg_descriptions = package_list(pkgs.iter().copied(), "or"); gctx.shell().note(format!( - "waiting for `{short_pkg_description}` to be available at {source_description}.\n\ + "waiting for {short_pkg_descriptions} to be available at {source_description}.\n\ You may press ctrl-c to skip waiting; the crate should be available shortly." ))?; let mut progress = Progress::with_style("Waiting", ProgressStyle::Ratio, gctx); progress.tick_now(0, max, "")?; - let is_available = loop { + let available = loop { { let _lock = gctx.acquire_package_cache_lock(CacheLockMode::DownloadExclusive)?; // Force re-fetching the source @@ -247,43 +289,67 @@ fn wait_for_publish( // multiple times gctx.updated_sources().remove(&source.replaced_source_id()); source.invalidate_cache(); - let summaries = loop { - // Exact to avoid returning all for path/git - match source.query_vec(&query, QueryKind::Exact) { - std::task::Poll::Ready(res) => { - break res?; - } - std::task::Poll::Pending => source.block_until_ready()?, + let mut available = Vec::new(); + for pkg in pkgs { + if poll_one_package(registry_src, pkg, &mut source)? { + available.push(*pkg); } - }; - if !summaries.is_empty() { - break true; + } + + // As soon as any package is available, break this loop so we can see if another + // one can be uploaded. + if !available.is_empty() { + break available; } } let elapsed = now.elapsed(); if timeout < elapsed { gctx.shell().warn(format!( - "timed out waiting for `{short_pkg_description}` to be available in {source_description}", + "timed out waiting for {short_pkg_descriptions} to be available in {source_description}", ))?; gctx.shell().note( "the registry may have a backlog that is delaying making the \ crate available. The crate should be available soon.", )?; - break false; + break Vec::new(); } progress.tick_now(elapsed.as_secs() as usize, max, "")?; std::thread::sleep(sleep_time); }; - if is_available { + if !available.is_empty() { + let short_pkg_description = available + .iter() + .map(|pkg| format!("{} v{}", pkg.name(), pkg.version())) + .sorted() + .join(", "); gctx.shell().status( "Published", format!("{short_pkg_description} at {source_description}"), )?; } - Ok(()) + Ok(available) +} + +fn poll_one_package( + registry_src: SourceId, + pkg_id: &PackageId, + source: &mut dyn Source, +) -> CargoResult { + let version_req = format!("={}", pkg_id.version()); + let query = Dependency::parse(pkg_id.name(), Some(&version_req), registry_src)?; + let summaries = loop { + // Exact to avoid returning all for path/git + match source.query_vec(&query, QueryKind::Exact) { + std::task::Poll::Ready(res) => { + break res?; + } + std::task::Poll::Pending => source.block_until_ready()?, + } + }; + Ok(!summaries.is_empty()) } fn verify_dependencies( @@ -498,3 +564,152 @@ fn transmit( Ok(()) } + +// State for tracking dependencies during upload. +struct PublishOrder { + // The reverse-dependency graph. It has an edge p -> q if package q depends on package p. + rev_graph: Graph, + // The weight of a package is the number of unpublished dependencies it has. + weights: HashMap, +} + +impl PublishOrder { + // Given a package dependency graph, creates a `PublishOrder` for tracking state and also + // a list of packages that have no dependencies (i.e. can be published immediately). + fn new(graph: &Graph) -> (Self, Vec) { + let rev_graph = graph.reversed(); + + let weights: HashMap<_, _> = rev_graph + .iter() + .map(|id| (*id, graph.edges(id).count())) + .collect(); + let ready = weights + .iter() + .filter_map(|(id, weight)| (*weight == 0).then_some(*id)) + .collect(); + (Self { rev_graph, weights }, ready) + } + + // Marks a list of packages as having been published, and returns a list of packages + // that are newly ready for publishing. + fn mark_published(&mut self, published: Vec) -> Vec { + let mut ret = Vec::new(); + for id in published { + for (dependent_id, _) in self.rev_graph.edges(&id) { + if let Some(weight) = self.weights.get_mut(dependent_id) { + *weight = weight.saturating_sub(1); + if *weight == 0 { + ret.push(*dependent_id); + } + } + } + } + ret + } +} + +// Format a collection of packages as a list, like "foo v0.1.0, bar v0.2.0, and baz v0.3.0". +// The final separator (i.e. "and" in the previous example) can be chosen. +fn package_list(pkgs: impl IntoIterator, final_sep: &str) -> String { + let mut names: Vec<_> = pkgs + .into_iter() + .map(|pkg| format!("`{} v{}`", pkg.name(), pkg.version())) + .collect(); + names.sort(); + + match &names[..] { + [] => String::new(), + [a] => a.clone(), + [a, b] => format!("{a} {final_sep} {b}"), + [names @ .., last] => { + format!("{}, {final_sep} {last}", names.join(", ")) + } + } +} + +fn validate_registry(pkgs: &[&Package], reg_or_index: Option<&RegistryOrIndex>) -> CargoResult<()> { + for pkg in pkgs { + if pkg.publish() == &Some(Vec::new()) { + bail!( + "`{}` cannot be published.\n\ + `package.publish` must be set to `true` or a non-empty list in Cargo.toml to publish.", + pkg.name(), + ); + } + } + + let reg_name = match reg_or_index { + Some(RegistryOrIndex::Registry(r)) => Some(r.as_str()), + None => Some(CRATES_IO_REGISTRY), + Some(RegistryOrIndex::Index(_)) => None, + }; + if let Some(reg_name) = reg_name { + for pkg in pkgs { + if let Some(allowed) = pkg.publish().as_ref() { + if !allowed.iter().any(|a| a == reg_name) { + bail!( + "`{}` cannot be published.\n\ + The registry `{}` is not listed in the `package.publish` value in Cargo.toml.", + pkg.name(), + reg_name + ); + } + } + } + } + + Ok(()) +} + +#[cfg(test)] +mod tests { + use crate::{ + core::{PackageId, SourceId}, + sources::CRATES_IO_INDEX, + util::{Graph, IntoUrl}, + }; + + use super::PublishOrder; + + fn pkg_id(name: &str) -> PackageId { + let loc = CRATES_IO_INDEX.into_url().unwrap(); + PackageId::try_new(name, "1.0.0", SourceId::for_registry(&loc).unwrap()).unwrap() + } + + #[test] + fn parallel_schedule() { + let mut graph: Graph = Graph::new(); + let a = pkg_id("a"); + let b = pkg_id("b"); + let c = pkg_id("c"); + let d = pkg_id("d"); + let e = pkg_id("e"); + + graph.add(a); + graph.add(b); + graph.add(c); + graph.add(d); + graph.add(e); + graph.link(a, c); + graph.link(b, c); + graph.link(c, d); + graph.link(c, e); + + let (mut order, mut ready) = PublishOrder::new(&graph); + ready.sort(); + assert_eq!(ready, vec![d, e]); + + let ready = order.mark_published(vec![d]); + assert!(ready.is_empty()); + + let ready = order.mark_published(vec![e]); + assert_eq!(ready, vec![c]); + + let mut ready = order.mark_published(vec![c]); + ready.sort(); + assert_eq!(ready, vec![a, b]); + + let ready = order.mark_published(vec![a, b]); + assert!(ready.is_empty()); + } +} diff --git a/src/cargo/util/graph.rs b/src/cargo/util/graph.rs index 4ae005944304..9e8ba143f401 100644 --- a/src/cargo/util/graph.rs +++ b/src/cargo/util/graph.rs @@ -25,6 +25,20 @@ impl Graph { .or_insert_with(Default::default) } + /// Returns the graph obtained by reversing all edges. + pub fn reversed(&self) -> Graph { + let mut ret = Graph::new(); + + for n in self.iter() { + ret.add(n.clone()); + for (m, e) in self.edges(n) { + *ret.link(m.clone(), n.clone()) = e.clone(); + } + } + + ret + } + pub fn contains(&self, k: &Q) -> bool where N: Borrow, @@ -206,6 +220,19 @@ fn path_to_self() { assert_eq!(new.path_to_bottom(&0), vec![(&0, Some(&()))]); } +#[test] +fn reverse() { + let mut new: Graph = Graph::new(); + new.link(0, 1); + new.link(0, 2); + + let mut expected: Graph = Graph::new(); + expected.add(0); + expected.link(1, 0); + expected.link(2, 0); + assert_eq!(new.reversed(), expected); +} + impl Default for Graph { fn default() -> Graph { Graph::new() diff --git a/tests/testsuite/credential_process.rs b/tests/testsuite/credential_process.rs index a07006d549b3..0ef7fd03f729 100644 --- a/tests/testsuite/credential_process.rs +++ b/tests/testsuite/credential_process.rs @@ -76,8 +76,8 @@ fn publish() { {"v":1,"registry":{"index-url":"[..]","name":"alternative","headers":[..]},"kind":"get","operation":"read"} [PACKAGING] foo v0.1.0 ([ROOT]/foo) [PACKAGED] 3 files, [FILE_SIZE]B ([FILE_SIZE]B compressed) -{"v":1,"registry":{"index-url":"[..]","name":"alternative"},"kind":"get","operation":"publish","name":"foo","vers":"0.1.0","cksum":"[..]"} [UPLOADING] foo v0.1.0 ([ROOT]/foo) +{"v":1,"registry":{"index-url":"[..]","name":"alternative"},"kind":"get","operation":"publish","name":"foo","vers":"0.1.0","cksum":"[..]"} [UPLOADED] foo v0.1.0 to registry `alternative` [NOTE] waiting for `foo v0.1.0` to be available at registry `alternative`. You may press ctrl-c [..] @@ -529,8 +529,8 @@ fn token_caching() { {"v":1,"registry":{"index-url":"[..]","name":"alternative"},"kind":"get","operation":"read"} [PACKAGING] foo v0.1.0 ([ROOT]/foo) [PACKAGED] 3 files, [FILE_SIZE]B ([FILE_SIZE]B compressed) -{"v":1,"registry":{"index-url":"[..]","name":"alternative"},"kind":"get","operation":"publish","name":"foo","vers":"0.1.0","cksum":"[..]"} [UPLOADING] foo v0.1.0 ([ROOT]/foo) +{"v":1,"registry":{"index-url":"[..]","name":"alternative"},"kind":"get","operation":"publish","name":"foo","vers":"0.1.0","cksum":"[..]"} [UPLOADED] foo v0.1.0 to registry `alternative` [NOTE] waiting [..] You may press ctrl-c [..] diff --git a/tests/testsuite/open_namespaces.rs b/tests/testsuite/open_namespaces.rs index bc4308d13401..f294fddfb197 100644 --- a/tests/testsuite/open_namespaces.rs +++ b/tests/testsuite/open_namespaces.rs @@ -351,19 +351,15 @@ fn publish_namespaced() { .file("src/lib.rs", "fn main() {}") .build(); + // We'd like to have an error about namespaced packages not being publishable, + // but instead it fails early while trying to parse `foo::bar` as a package spec. p.cargo("publish") .masquerade_as_nightly_cargo(&["script", "open-namespaces"]) .replace_crates_io(registry.index_url()) .with_status(101) .with_stderr_data(str![[r#" [UPDATING] crates.io index -[WARNING] manifest has no documentation, homepage or repository. -See https://doc.rust-lang.org/cargo/reference/manifest.html#package-metadata for more info. -[PACKAGING] foo::bar v0.0.1 ([ROOT]/foo) -[ERROR] failed to prepare local package for uploading - -Caused by: - cannot publish with `open-namespaces` +[ERROR] expected a version like "1.32" "#]]) .run(); diff --git a/tests/testsuite/publish.rs b/tests/testsuite/publish.rs index 9a764bbd0ee7..b53d1f915753 100644 --- a/tests/testsuite/publish.rs +++ b/tests/testsuite/publish.rs @@ -961,7 +961,6 @@ fn publish_failed_with_index_and_only_allowed_registry() { .arg(registry.index_url().as_str()) .with_status(101) .with_stderr_data(str![[r#" -[NOTE] found `alternative` as only allowed registry. Publishing to it automatically. [ERROR] command-line argument --index requires --token to be specified "#]]) @@ -995,8 +994,7 @@ fn publish_fail_with_no_registry_specified() { p.cargo("publish") .with_status(101) .with_stderr_data(str![[r#" -[ERROR] `foo` cannot be published. -The registry `crates-io` is not listed in the `package.publish` value in Cargo.toml. +[ERROR] --registry is required to disambiguate between "alternative" or "test" registries "#]]) .run(); @@ -3175,6 +3173,7 @@ See https://doc.rust-lang.org/cargo/reference/manifest.html#package-metadata for [PACKAGING] foo v0.0.1 ([ROOT]/foo) [PACKAGED] 4 files, [FILE_SIZE]B ([FILE_SIZE]B compressed) [UPLOADING] foo v0.0.1 ([ROOT]/foo) +[UPLOADED] foo v0.0.1 to registry `crates-io` "#]]) .run(); @@ -3307,7 +3306,26 @@ fn timeout_waiting_for_dependency_publish() { .masquerade_as_nightly_cargo(&["publish-timeout", "package-workspace"]) .with_status(101) .with_stderr_data(str![[r#" -[ERROR] the `-p` argument must be specified to select a single package to publish +[UPDATING] crates.io index +[WARNING] manifest has no documentation, homepage or repository. +See https://doc.rust-lang.org/cargo/reference/manifest.html#package-metadata for more info. +[PACKAGING] dep v0.0.1 ([ROOT]/foo/dep) +[PACKAGED] 3 files, [FILE_SIZE]B ([FILE_SIZE]B compressed) +[WARNING] manifest has no documentation, homepage or repository. +See https://doc.rust-lang.org/cargo/reference/manifest.html#package-metadata for more info. +[PACKAGING] main v0.0.1 ([ROOT]/foo/main) +[PACKAGED] 4 files, [FILE_SIZE]B ([FILE_SIZE]B compressed) +[WARNING] manifest has no documentation, homepage or repository. +See https://doc.rust-lang.org/cargo/reference/manifest.html#package-metadata for more info. +[PACKAGING] other v0.0.1 ([ROOT]/foo/other) +[PACKAGED] 4 files, [FILE_SIZE]B ([FILE_SIZE]B compressed) +[UPLOADING] dep v0.0.1 ([ROOT]/foo/dep) +[UPLOADED] dep v0.0.1 to registry `crates-io` +[NOTE] waiting for `dep v0.0.1` to be available at registry `crates-io`. +You may press ctrl-c to skip waiting; the crate should be available shortly. +[WARNING] timed out waiting for `dep v0.0.1` to be available in registry `crates-io` +[NOTE] the registry may have a backlog that is delaying making the crate available. The crate should be available soon. +[ERROR] failed to publish `main v0.0.1` and `other v0.0.1` because we timed out waiting for dependencies. "#]]) .run(); @@ -3590,10 +3608,47 @@ fn workspace_with_local_deps_nightly() { p.cargo("publish -Zpackage-workspace") .masquerade_as_nightly_cargo(&["package-workspace"]) - .with_status(101) .replace_crates_io(registry.index_url()) .with_stderr_data(str![[r#" -[ERROR] the `-p` argument must be specified to select a single package to publish +[UPDATING] crates.io index +[PACKAGING] level3 v0.0.1 ([ROOT]/foo/level3) +[PACKAGED] 3 files, [FILE_SIZE]B ([FILE_SIZE]B compressed) +[PACKAGING] level2 v0.0.1 ([ROOT]/foo/level2) +[PACKAGED] 3 files, [FILE_SIZE]B ([FILE_SIZE]B compressed) +[PACKAGING] level1 v0.0.1 ([ROOT]/foo/level1) +[UPDATING] crates.io index +[PACKAGED] 4 files, [FILE_SIZE]B ([FILE_SIZE]B compressed) +[VERIFYING] level3 v0.0.1 ([ROOT]/foo/level3) +[COMPILING] level3 v0.0.1 ([ROOT]/foo/target/package/level3-0.0.1) +[FINISHED] `dev` profile [unoptimized + debuginfo] target(s) in [ELAPSED]s +[VERIFYING] level2 v0.0.1 ([ROOT]/foo/level2) +[UPDATING] crates.io index +[UNPACKING] level3 v0.0.1 (registry `[ROOT]/foo/target/package/tmp-registry`) +[COMPILING] level3 v0.0.1 +[COMPILING] level2 v0.0.1 ([ROOT]/foo/target/package/level2-0.0.1) +[FINISHED] `dev` profile [unoptimized + debuginfo] target(s) in [ELAPSED]s +[VERIFYING] level1 v0.0.1 ([ROOT]/foo/level1) +[UPDATING] crates.io index +[UNPACKING] level2 v0.0.1 (registry `[ROOT]/foo/target/package/tmp-registry`) +[COMPILING] level3 v0.0.1 +[COMPILING] level2 v0.0.1 +[COMPILING] level1 v0.0.1 ([ROOT]/foo/target/package/level1-0.0.1) +[FINISHED] `dev` profile [unoptimized + debuginfo] target(s) in [ELAPSED]s +[UPLOADING] level3 v0.0.1 ([ROOT]/foo/level3) +[UPLOADED] level3 v0.0.1 to registry `crates-io` +[NOTE] waiting for `level3 v0.0.1` to be available at registry `crates-io`. +You may press ctrl-c to skip waiting; the crate should be available shortly. +[PUBLISHED] level3 v0.0.1 at registry `crates-io` +[UPLOADING] level2 v0.0.1 ([ROOT]/foo/level2) +[UPLOADED] level2 v0.0.1 to registry `crates-io` +[NOTE] waiting for `level2 v0.0.1` to be available at registry `crates-io`. +You may press ctrl-c to skip waiting; the crate should be available shortly. +[PUBLISHED] level2 v0.0.1 at registry `crates-io` +[UPLOADING] level1 v0.0.1 ([ROOT]/foo/level1) +[UPLOADED] level1 v0.0.1 to registry `crates-io` +[NOTE] waiting for `level1 v0.0.1` to be available at registry `crates-io`. +You may press ctrl-c to skip waiting; the crate should be available shortly. +[PUBLISHED] level1 v0.0.1 at registry `crates-io` "#]]) .run(); @@ -3661,10 +3716,40 @@ fn workspace_parallel() { p.cargo("publish -Zpackage-workspace") .masquerade_as_nightly_cargo(&["package-workspace"]) .replace_crates_io(registry.index_url()) - .with_status(101) .with_stderr_data( str![[r#" -[ERROR] the `-p` argument must be specified to select a single package to publish +[UPDATING] crates.io index +[PACKAGING] a v0.0.1 ([ROOT]/foo/a) +[PACKAGED] 3 files, [FILE_SIZE]B ([FILE_SIZE]B compressed) +[PACKAGING] b v0.0.1 ([ROOT]/foo/b) +[PACKAGED] 3 files, [FILE_SIZE]B ([FILE_SIZE]B compressed) +[PACKAGING] c v0.0.1 ([ROOT]/foo/c) +[PACKAGED] 3 files, [FILE_SIZE]B ([FILE_SIZE]B compressed) +[VERIFYING] a v0.0.1 ([ROOT]/foo/a) +[COMPILING] a v0.0.1 ([ROOT]/foo/target/package/a-0.0.1) +[FINISHED] `dev` profile [unoptimized + debuginfo] target(s) in [ELAPSED]s +[VERIFYING] b v0.0.1 ([ROOT]/foo/b) +[COMPILING] b v0.0.1 ([ROOT]/foo/target/package/b-0.0.1) +[FINISHED] `dev` profile [unoptimized + debuginfo] target(s) in [ELAPSED]s +[VERIFYING] c v0.0.1 ([ROOT]/foo/c) +[UPDATING] crates.io index +[UNPACKING] a v0.0.1 (registry `[ROOT]/foo/target/package/tmp-registry`) +[UNPACKING] b v0.0.1 (registry `[ROOT]/foo/target/package/tmp-registry`) +[COMPILING] a v0.0.1 +[COMPILING] b v0.0.1 +[COMPILING] c v0.0.1 ([ROOT]/foo/target/package/c-0.0.1) +[FINISHED] `dev` profile [unoptimized + debuginfo] target(s) in [ELAPSED]s +[UPLOADING] a v0.0.1 ([ROOT]/foo/a), b v0.0.1 ([ROOT]/foo/b) +[UPLOADED] b v0.0.1 to registry `crates-io` +[UPLOADED] a v0.0.1 to registry `crates-io` +[NOTE] waiting for `a v0.0.1` or `b v0.0.1` to be available at registry `crates-io`. +You may press ctrl-c to skip waiting; the crate should be available shortly. +[PUBLISHED] a v0.0.1, b v0.0.1 at registry `crates-io` +[UPLOADING] c v0.0.1 ([ROOT]/foo/c) +[UPLOADED] c v0.0.1 to registry `crates-io` +[NOTE] waiting for `c v0.0.1` to be available at registry `crates-io`. +You may press ctrl-c to skip waiting; the crate should be available shortly. +[PUBLISHED] c v0.0.1 at registry `crates-io` "#]] .unordered(), @@ -3726,9 +3811,12 @@ fn workspace_missing_dependency() { [PACKAGED] 3 files, [FILE_SIZE]B ([FILE_SIZE]B compressed) [VERIFYING] b v0.0.1 ([ROOT]/foo/b) [UPDATING] crates.io index -[ERROR] no matching package named `a` found -location searched: registry `crates-io` -required by package `b v0.0.1 ([ROOT]/foo/target/package/b-0.0.1)` +[ERROR] failed to verify package tarball + +Caused by: + no matching package named `a` found + location searched: registry `crates-io` + required by package `b v0.0.1 ([ROOT]/foo/target/package/b-0.0.1)` "#]]) .run(); @@ -3758,7 +3846,23 @@ You may press ctrl-c to skip waiting; the crate should be available shortly. .replace_crates_io(registry.index_url()) .with_status(101) .with_stderr_data(str![[r#" -[ERROR] the `-p` argument must be specified to select a single package to publish +[UPDATING] crates.io index +[PACKAGING] a v0.0.1 ([ROOT]/foo/a) +[PACKAGED] 3 files, [FILE_SIZE]B ([FILE_SIZE]B compressed) +[PACKAGING] b v0.0.1 ([ROOT]/foo/b) +[PACKAGED] 3 files, [FILE_SIZE]B ([FILE_SIZE]B compressed) +[VERIFYING] a v0.0.1 ([ROOT]/foo/a) +[COMPILING] a v0.0.1 ([ROOT]/foo/target/package/a-0.0.1) +[FINISHED] `dev` profile [unoptimized + debuginfo] target(s) in [ELAPSED]s +[VERIFYING] b v0.0.1 ([ROOT]/foo/b) +[UPDATING] crates.io index +[ERROR] failed to verify package tarball + +Caused by: + failed to get `a` as a dependency of package `b v0.0.1 ([ROOT]/foo/target/package/b-0.0.1)` + +Caused by: + found a package in the remote registry and the local overlay: a@0.0.1 "#]]) .run();