public-api: simplify and improve output

pull/691/head
Tamir Duberstein 2 years ago
parent 06d6a3e7a6
commit 321bda7539
No known key found for this signature in database

@ -19,6 +19,5 @@ quote = { workspace = true }
rustdoc-json = { workspace = true } rustdoc-json = { workspace = true }
rustup-toolchain = { workspace = true } rustup-toolchain = { workspace = true }
syn = { workspace = true } syn = { workspace = true }
thiserror = { workspace = true }
tempfile = { workspace = true } tempfile = { workspace = true }
which = { workspace = true } which = { workspace = true }

@ -1,15 +1,15 @@
use std::{ use std::{
fmt::Write as _, fmt::Write as _,
fs::{read_to_string, write}, fs::{read_to_string, File},
io::Write as _,
path::Path, path::Path,
}; };
use anyhow::{bail, Context as _}; use anyhow::{bail, Context as _, Result};
use cargo_metadata::{Metadata, Package}; use cargo_metadata::{Metadata, Package};
use clap::Parser; use clap::Parser;
use dialoguer::{theme::ColorfulTheme, Confirm}; use dialoguer::{theme::ColorfulTheme, Confirm};
use diff::{lines, Result as Diff}; use diff::{lines, Result as Diff};
use thiserror::Error;
#[derive(Debug, Parser)] #[derive(Debug, Parser)]
pub struct Options { pub struct Options {
@ -18,18 +18,7 @@ pub struct Options {
pub bless: bool, pub bless: bool,
} }
#[derive(Error, Debug)] pub fn public_api(options: Options, metadata: Metadata) -> Result<()> {
enum PublicApiError {
#[error("error checking public api for {package}\n{source}\n")]
Error {
package: String,
source: anyhow::Error,
},
#[error("public api for {package} changed:\n{diff}\n")]
Changed { package: String, diff: String },
}
pub fn public_api(options: Options, metadata: Metadata) -> anyhow::Result<()> {
let toolchain = "nightly"; let toolchain = "nightly";
let Options { bless } = options; let Options { bless } = options;
@ -50,18 +39,30 @@ pub fn public_api(options: Options, metadata: Metadata) -> anyhow::Result<()> {
.. ..
} = &metadata; } = &metadata;
let mut buf = String::new(); let errors = packages
packages.iter().for_each(|Package { name, publish, .. }| { .iter()
if matches!(publish, Some(publish) if publish.is_empty()) { .fold(String::new(), |mut buf, Package { name, publish, .. }| {
return; if !matches!(publish, Some(publish) if publish.is_empty()) {
match check_package_api(name, toolchain, bless, workspace_root.as_std_path()) {
Ok(diff) => {
if !diff.is_empty() {
writeln!(
&mut buf,
"{name} public API changed; re-run with --bless. diff:\n{diff}"
)
.unwrap();
}
}
Err(err) => {
writeln!(&mut buf, "{name} failed to check public API: {err}").unwrap();
}
} }
if let Err(e) = check_package_api(name, toolchain, bless, workspace_root.as_std_path()) {
write!(&mut buf, "{}", e).unwrap();
} }
buf
}); });
if !buf.is_empty() { if !errors.is_empty() {
bail!("public api may have changed in one or more packages.\nplease bless by re-running this command with --bless\nErrors:\n{buf}"); bail!("public API errors:\n{errors}");
} }
Ok(()) Ok(())
} }
@ -71,7 +72,7 @@ fn check_package_api(
toolchain: &str, toolchain: &str,
bless: bool, bless: bool,
workspace_root: &Path, workspace_root: &Path,
) -> Result<(), PublicApiError> { ) -> Result<String> {
let path = workspace_root let path = workspace_root
.join("xtask") .join("xtask")
.join("public-api") .join("public-api")
@ -83,47 +84,29 @@ fn check_package_api(
.package(package) .package(package)
.all_features(true) .all_features(true)
.build() .build()
.map_err(|source| PublicApiError::Error { .context("rustdoc_json::Builder::build")?;
package: package.to_string(),
source: source.into(),
})?;
let public_api = public_api::Builder::from_rustdoc_json(rustdoc_json) let public_api = public_api::Builder::from_rustdoc_json(rustdoc_json)
.build() .build()
.map_err(|source| PublicApiError::Error { .context("public_api::Builder::build")?;
package: package.to_string(),
source: source.into(),
})?;
if bless { if bless {
write(&path, public_api.to_string().as_bytes()).map_err(|source| { let mut output =
PublicApiError::Error { File::create(&path).with_context(|| format!("error creating {}", path.display()))?;
package: package.to_string(), write!(&mut output, "{}", public_api)
source: source.into(), .with_context(|| format!("error writing {}", path.display()))?;
} }
})?; let current_api =
} read_to_string(&path).with_context(|| format!("error reading {}", path.display()))?;
let current_api = read_to_string(&path)
.with_context(|| format!("error reading {}", &path.display()))
.map_err(|source| PublicApiError::Error {
package: package.to_string(),
source,
})?;
let mut buf = String::new(); Ok(lines(&current_api, &public_api.to_string())
lines(&current_api, &public_api.to_string())
.into_iter() .into_iter()
.for_each(|diff| match diff { .fold(String::new(), |mut buf, diff| {
match diff {
Diff::Both(..) => (), Diff::Both(..) => (),
Diff::Right(line) => writeln!(&mut buf, "-{}", line).unwrap(), Diff::Right(line) => writeln!(&mut buf, "-{}", line).unwrap(),
Diff::Left(line) => writeln!(&mut buf, "+{}", line).unwrap(), Diff::Left(line) => writeln!(&mut buf, "+{}", line).unwrap(),
});
if !buf.is_empty() {
return Err(PublicApiError::Changed {
package: package.to_string(),
diff: buf,
});
}; };
Ok(()) buf
}))
} }

Loading…
Cancel
Save