use clap::Args;
use serde::Serialize;
use homeboy::deploy::{self, ComponentDeployResult, DeployConfig, DeploySummary};
use homeboy::resolve::{infer_project_for_components, resolve_project_components};
use super::CmdResult;
#[derive(Args)]
pub struct DeployArgs {
pub project_id: String,
pub component_ids: Vec<String>,
#[arg(long, short = 'p')]
pub project: Option<String>,
#[arg(long, short = 'c')]
pub component: Option<Vec<String>>,
#[arg(long)]
pub json: Option<String>,
#[arg(long)]
pub all: bool,
#[arg(long)]
pub outdated: bool,
#[arg(long)]
pub dry_run: bool,
#[arg(long, visible_alias = "status")]
pub check: bool,
#[arg(long)]
pub force: bool,
#[arg(long, value_delimiter = ',')]
pub projects: Option<Vec<String>>,
}
#[derive(Serialize)]
pub struct DeployOutput {
pub command: String,
pub project_id: String,
pub all: bool,
pub outdated: bool,
pub dry_run: bool,
pub check: bool,
pub force: bool,
pub results: Vec<ComponentDeployResult>,
pub summary: DeploySummary,
}
#[derive(Serialize)]
pub struct MultiProjectDeployOutput {
pub command: String,
pub component_ids: Vec<String>,
pub projects: Vec<ProjectDeployResult>,
pub summary: MultiProjectDeploySummary,
pub dry_run: bool,
pub check: bool,
pub force: bool,
}
#[derive(Serialize)]
pub struct ProjectDeployResult {
pub project_id: String,
pub status: String,
#[serde(skip_serializing_if = "Option::is_none")]
pub error: Option<String>,
pub results: Vec<ComponentDeployResult>,
pub summary: DeploySummary,
}
#[derive(Serialize)]
pub struct MultiProjectDeploySummary {
pub total_projects: u32,
pub succeeded: u32,
pub failed: u32,
}
#[derive(Serialize)]
#[serde(untagged)]
pub enum DeployCommandOutput {
Single(DeployOutput),
Multi(MultiProjectDeployOutput),
}
pub fn run(
mut args: DeployArgs,
_global: &crate::commands::GlobalArgs,
) -> CmdResult<DeployCommandOutput> {
if let Some(ref project_ids) = args.projects {
return run_multi_project(&args, project_ids);
}
let (project_id, component_ids) = match (&args.project, &args.component) {
(Some(ref proj), Some(ref comps)) => (proj.clone(), comps.clone()),
(Some(ref proj), None) => {
let mut comps = vec![args.project_id.clone()];
comps.extend(args.component_ids.clone());
(proj.clone(), comps)
}
(None, Some(ref comps)) => {
let projects = homeboy::project::list_ids().unwrap_or_default();
if projects.contains(&args.project_id) {
(args.project_id.clone(), comps.clone())
} else {
match infer_project_for_components(comps) {
Some(proj) => (proj, comps.clone()),
None => {
return Err(homeboy::Error::validation_invalid_argument(
"project_id",
"Could not infer project. Use --project flag or provide project as first argument.",
None,
None,
)
.into())
}
}
}
}
(None, None) => resolve_project_components(&args.project_id, &args.component_ids)?,
};
args.project_id = project_id.clone();
args.component_ids = component_ids;
if let Some(ref spec) = args.json {
args.component_ids = deploy::parse_bulk_component_ids(spec)?;
}
let config = DeployConfig {
component_ids: args.component_ids.clone(),
all: args.all,
outdated: args.outdated,
dry_run: args.dry_run,
check: args.check,
force: args.force,
skip_build: false,
};
let result = deploy::run(&project_id, &config).map_err(|e| {
if e.message.contains("No components configured for project") {
e.with_hint(format!(
"Run 'homeboy project components add {} <component-id>' to add components",
project_id
))
.with_hint("Run 'homeboy init' to see project context and available components")
} else {
e
}
})?;
let exit_code = if result.summary.failed > 0 { 1 } else { 0 };
Ok((
DeployCommandOutput::Single(DeployOutput {
command: "deploy.run".to_string(),
project_id: project_id.clone(),
all: args.all,
outdated: args.outdated,
dry_run: args.dry_run,
check: args.check,
force: args.force,
results: result.results,
summary: result.summary,
}),
exit_code,
))
}
fn run_multi_project(
args: &DeployArgs,
project_ids: &[String],
) -> CmdResult<DeployCommandOutput> {
let mut component_ids = vec![args.project_id.clone()];
component_ids.extend(args.component_ids.clone());
let component_ids: Vec<String> = component_ids.into_iter().filter(|s| !s.is_empty()).collect();
if component_ids.is_empty() {
return Err(homeboy::Error::validation_invalid_argument(
"component_ids",
"At least one component ID is required when using --projects",
None,
None,
)
.into());
}
let known_projects = homeboy::project::list_ids().unwrap_or_default();
for project_id in project_ids {
if !known_projects.contains(project_id) {
return Err(homeboy::Error::validation_invalid_argument(
"projects",
&format!("Unknown project: '{}'", project_id),
None,
None,
)
.into());
}
}
eprintln!(
"[deploy] Deploying {:?} to {} project(s)...",
component_ids,
project_ids.len()
);
let mut project_results = Vec::new();
let mut succeeded: u32 = 0;
let mut failed: u32 = 0;
let mut first_project = true;
for project_id in project_ids {
eprintln!("[deploy] Deploying to project '{}'...", project_id);
let config = DeployConfig {
component_ids: component_ids.clone(),
all: args.all,
outdated: args.outdated,
dry_run: args.dry_run,
check: args.check,
force: args.force,
skip_build: !first_project, };
match deploy::run(project_id, &config) {
Ok(result) => {
let deploy_failed = result.summary.failed > 0;
if deploy_failed {
let error_msg = result
.results
.iter()
.find_map(|r| r.error.clone())
.unwrap_or_else(|| "Deployment failed".to_string());
project_results.push(ProjectDeployResult {
project_id: project_id.clone(),
status: "failed".to_string(),
error: Some(error_msg),
results: result.results,
summary: result.summary,
});
failed += 1;
} else {
project_results.push(ProjectDeployResult {
project_id: project_id.clone(),
status: "deployed".to_string(),
error: None,
results: result.results,
summary: result.summary,
});
succeeded += 1;
}
}
Err(e) => {
project_results.push(ProjectDeployResult {
project_id: project_id.clone(),
status: "failed".to_string(),
error: Some(e.to_string()),
results: vec![],
summary: DeploySummary {
total: 0,
succeeded: 0,
skipped: 0,
failed: 1,
},
});
failed += 1;
}
}
first_project = false;
}
let total = project_results.len() as u32;
let exit_code = if failed > 0 { 1 } else { 0 };
Ok((
DeployCommandOutput::Multi(MultiProjectDeployOutput {
command: "deploy.run_multi".to_string(),
component_ids,
projects: project_results,
summary: MultiProjectDeploySummary {
total_projects: total,
succeeded,
failed,
},
dry_run: args.dry_run,
check: args.check,
force: args.force,
}),
exit_code,
))
}