cargo_lambda_system/
lib.rs1use std::{collections::HashMap, path::PathBuf};
2
3use cargo_lambda_metadata::{
4 cargo::load_metadata,
5 config::{
6 Config, ConfigOptions, general_config_figment, get_config_from_all_packages,
7 load_config_without_cli_flags,
8 },
9};
10use clap::Args;
11use miette::{IntoDiagnostic, Result};
12
13use cargo_lambda_build::zig::{ZigInfo, check_installation, get_zig_info};
14use serde::{Deserialize, Serialize};
15use strum_macros::{Display, EnumString};
16use tracing::trace;
17
18#[derive(Clone, Debug, Default, Deserialize, Display, EnumString, Serialize)]
19#[strum(ascii_case_insensitive)]
20#[serde(rename_all = "lowercase")]
21pub enum OutputFormat {
22 #[default]
23 Text,
24 Json,
25}
26
27#[derive(Args, Clone, Debug)]
28#[command(
29 visible_alias = "config",
30 after_help = "Full command documentation: https://www.cargo-lambda.info/commands/system.html"
31)]
32pub struct System {
33 #[arg(long, visible_alias = "install-zig", alias = "install")]
35 setup: bool,
36
37 #[arg(short, long)]
39 manifest_path: Option<PathBuf>,
40
41 #[arg(short, long)]
43 output_format: Option<OutputFormat>,
44
45 #[arg(short, long)]
47 package: Option<String>,
48}
49
50impl System {
51 pub fn manifest_path(&self) -> PathBuf {
52 self.manifest_path
53 .clone()
54 .unwrap_or_else(|| "Cargo.toml".into())
55 }
56
57 pub fn package(&self) -> Option<String> {
58 self.package.clone()
59 }
60}
61
62#[derive(Debug, Serialize)]
63struct Info {
64 zig: ZigInfo,
65 #[serde(skip_serializing_if = "Option::is_none")]
66 config: Option<ConfigInfo>,
67}
68
69#[derive(Debug, Serialize)]
70enum ConfigInfo {
71 #[serde(rename = "package")]
72 Package(Config),
73 #[serde(rename = "global")]
74 Global {
75 workspace: Config,
76 packages: HashMap<String, Config>,
77 },
78}
79
80#[tracing::instrument(target = "cargo_lambda")]
81pub async fn run(config: &System, options: &ConfigOptions) -> Result<()> {
82 trace!("running config command");
83
84 if config.setup {
85 let zig_info = check_installation(true).await;
86 if let Ok(zig_info) = zig_info {
87 return print_config(config.output_format.as_ref(), zig_info);
88 }
89 }
90
91 let mut info = Info {
92 zig: get_zig_info()?,
93 config: None,
94 };
95 let manifest_path = config.manifest_path();
96
97 if manifest_path.exists() {
98 let metadata = load_metadata(manifest_path)?;
99
100 let config_info = if options.name.is_some() || metadata.packages.len() == 1 {
101 let config = load_config_without_cli_flags(&metadata, options)?;
102 ConfigInfo::Package(config)
103 } else {
104 let (_, _, workspace) = general_config_figment(&metadata, options)?;
105
106 let packages = get_config_from_all_packages(&metadata)?;
107
108 ConfigInfo::Global {
109 workspace: workspace.extract().into_diagnostic()?,
110 packages,
111 }
112 };
113
114 info.config = Some(config_info);
115 }
116
117 print_config(config.output_format.as_ref(), info)
118}
119
120fn print_config(format: Option<&OutputFormat>, info: impl Serialize) -> Result<()> {
121 match format {
122 Some(OutputFormat::Json) => {
123 serde_json::to_writer_pretty(std::io::stdout(), &info).into_diagnostic()
124 }
125 _ => serde_yml::to_writer(std::io::stdout(), &info).into_diagnostic(),
126 }
127}