pub(crate) mod config_command;
mod generate_completions;
mod init;
pub(crate) mod manifest_command;
mod release;
mod release_pr;
pub(crate) mod repo_command;
mod set_version;
mod update;
use std::path::Path;
use anyhow::Context;
use cargo_metadata::camino::{Utf8Path, Utf8PathBuf};
use cargo_utils::CARGO_TOML;
use clap::{
ValueEnum,
builder::{Styles, styling::AnsiColor},
};
use init::Init;
use release_plz_core::fs_utils::current_directory;
use set_version::SetVersion;
use tracing::info;
use crate::config::Config;
use self::{
generate_completions::GenerateCompletions, release::Release, release_pr::ReleasePr,
update::Update,
};
const MAIN_COLOR: AnsiColor = AnsiColor::Red;
const SECONDARY_COLOR: AnsiColor = AnsiColor::Yellow;
const HELP_STYLES: Styles = Styles::styled()
.header(MAIN_COLOR.on_default().bold())
.usage(MAIN_COLOR.on_default().bold())
.placeholder(SECONDARY_COLOR.on_default())
.literal(SECONDARY_COLOR.on_default());
#[derive(clap::Parser, Debug)]
#[command(about, version, author, styles = HELP_STYLES)]
pub struct CliArgs {
#[command(subcommand)]
pub command: Command,
#[arg(short, long, global = true)]
pub verbose: bool,
}
#[derive(clap::Subcommand, Debug)]
pub enum Command {
Update(Update),
ReleasePr(ReleasePr),
Release(Release),
GenerateCompletions(GenerateCompletions),
CheckUpdates,
GenerateSchema,
Init(Init),
SetVersion(SetVersion),
}
#[derive(ValueEnum, Clone, Copy, Debug, Eq, PartialEq)]
pub enum OutputType {
Json,
}
fn local_manifest(manifest_path: Option<&Utf8Path>) -> Utf8PathBuf {
match manifest_path {
Some(manifest) => manifest.to_path_buf(),
None => current_directory().unwrap().join(CARGO_TOML),
}
}
fn parse_config(config_path: Option<&Path>) -> anyhow::Result<Config> {
let (config, path) = if let Some(config_path) = config_path {
match fs_err::read_to_string(config_path) {
Ok(config) => (config, config_path),
Err(e) => match e.kind() {
std::io::ErrorKind::NotFound => {
anyhow::bail!("specified config does not exist at path {config_path:?}")
}
_ => anyhow::bail!("can't read {config_path:?}: {e:?}"),
},
}
} else {
let first_file = first_file_contents([
Path::new("release-plz.toml"),
Path::new(".release-plz.toml"),
])
.context("failed looking for release-plz config file")?;
match first_file {
Some((config, path)) => (config, path),
None => {
info!("release-plz config file not found, using default configuration");
return Ok(Config::default());
}
}
};
info!("using release-plz config file {}", path.display());
toml::from_str(&config).with_context(|| format!("invalid config file {config_path:?}"))
}
fn first_file_contents<'a>(
paths: impl IntoIterator<Item = &'a Path>,
) -> anyhow::Result<Option<(String, &'a Path)>> {
let paths = paths.into_iter();
for path in paths {
match fs_err::read_to_string(path) {
Ok(config) => return Ok(Some((config, path))),
Err(err) if err.kind() == std::io::ErrorKind::NotFound => {}
Err(err) => return Err(err.into()),
}
}
Ok(None)
}