use super::helpers;
use crate::pipeline;
use anodizer_core::context::{Context, ContextOptions};
use anodizer_core::log::{StageLogger, Verbosity};
use anyhow::Result;
use std::path::PathBuf;
pub struct ContinueOpts {
pub dist: Option<PathBuf>,
pub dry_run: bool,
pub skip: Vec<String>,
pub token: Option<String>,
pub config_override: Option<PathBuf>,
pub verbose: bool,
pub debug: bool,
pub quiet: bool,
pub merge: bool,
}
pub fn run(opts: ContinueOpts) -> Result<()> {
let log = StageLogger::new(
"continue",
Verbosity::from_flags(opts.quiet, opts.verbose, opts.debug),
);
let ctx_opts = ContextOptions {
dry_run: opts.dry_run,
quiet: opts.quiet,
verbose: opts.verbose,
debug: opts.debug,
skip_stages: opts.skip,
token: opts.token,
merge: opts.merge,
..Default::default()
};
if opts.merge {
let config_path =
pipeline::find_config_with_logger(opts.config_override.as_deref(), Some(&log))?;
let mut config = pipeline::load_config(&config_path)?;
helpers::infer_project_name(&mut config, &log);
helpers::auto_detect_github(&mut config, &log);
let mut ctx = Context::new(config.clone(), ctx_opts);
helpers::setup_context(&mut ctx, &config, &log)?;
ctx.populate_metadata_var()?;
return super::release::run_merge(
&mut ctx,
&config,
&log,
opts.dry_run,
opts.dist.as_deref(),
);
}
let (_config, mut ctx, _dist) = helpers::init_publish_stage_ctx(
opts.config_override.as_deref(),
ctx_opts,
opts.dist.as_deref(),
true,
&log,
)?;
ctx.populate_metadata_var()?;
let p = pipeline::build_publish_pipeline();
p.run(&mut ctx, &log)
}
#[cfg(test)]
mod tests {
use super::*;
use serial_test::serial;
use std::fs;
fn write_minimal_config(dir: &std::path::Path) {
fs::write(
dir.join(".anodizer.yaml"),
r#"project_name: test
crates:
- name: test
path: "."
tag_template: "v{{ .Version }}"
"#,
)
.unwrap();
}
#[test]
#[serial]
fn no_merge_missing_dist_or_git_bails() {
let tmp = tempfile::tempdir().unwrap();
write_minimal_config(tmp.path());
let dist = tmp.path().join("dist-empty");
fs::create_dir_all(&dist).unwrap();
let result = run(ContinueOpts {
dist: Some(dist),
dry_run: true,
skip: vec![],
token: None,
config_override: Some(tmp.path().join(".anodizer.yaml")),
verbose: false,
debug: false,
quiet: true,
merge: false,
});
assert!(result.is_err(), "must fail with no manifest / no git");
}
#[test]
#[serial]
fn merge_missing_config_bails() {
let tmp = tempfile::tempdir().unwrap();
let err = run(ContinueOpts {
dist: None,
dry_run: true,
skip: vec![],
token: None,
config_override: Some(tmp.path().join("nope.yaml")),
verbose: false,
debug: false,
quiet: true,
merge: true,
})
.unwrap_err()
.to_string();
assert!(err.contains("config file not found"), "{err}");
}
#[test]
fn continue_opts_struct_round_trips() {
let opts = ContinueOpts {
dist: Some(std::path::PathBuf::from("/tmp/x")),
dry_run: true,
skip: vec!["docker".into()],
token: Some("t".into()),
config_override: None,
verbose: false,
debug: false,
quiet: true,
merge: false,
};
assert!(opts.dry_run);
assert_eq!(opts.skip, vec!["docker".to_string()]);
assert_eq!(opts.token.as_deref(), Some("t"));
assert!(!opts.merge);
}
}