use std::collections::HashMap;
use crate::{
controllers::{
deployment::{stream_build_logs, stream_deploy_logs},
environment::get_matched_environment,
project::{ensure_project_and_environment_exist, get_project},
},
util::logs::format_attr_log,
};
use anyhow::bail;
use serde_json::Value;
use super::{
queries::deployments::{DeploymentListInput, DeploymentStatus},
*,
};
#[derive(Parser)]
pub struct Args {
#[clap(short, long)]
service: Option<String>,
#[clap(short, long)]
environment: Option<String>,
#[clap(short, long, group = "log_type")]
deployment: bool,
#[clap(short, long, group = "log_type")]
build: bool,
deployment_id: Option<String>,
}
pub async fn command(args: Args, json: bool) -> Result<()> {
let configs = Configs::new()?;
let client = GQLClient::new_authorized(&configs)?;
let linked_project = configs.get_linked_project().await?;
ensure_project_and_environment_exist(&client, &configs, &linked_project).await?;
let project = get_project(&client, &configs, linked_project.project.clone()).await?;
let environment = args
.environment
.clone()
.unwrap_or(linked_project.environment.clone());
let services = project.services.edges.iter().collect::<Vec<_>>();
let environment_id = get_matched_environment(&project, environment)?.id;
let service = match (args.service, linked_project.service) {
(Some(service_arg), _) => services
.iter()
.find(|service| service.node.name == service_arg || service.node.id == service_arg)
.with_context(|| format!("Service '{service_arg}' not found"))?
.node
.id
.to_owned(),
(_, Some(linked_service)) => linked_service,
_ => bail!("No service could be found. Please either link one with `railway service` or specify one via the `--service` flag."),
};
let vars = queries::deployments::Variables {
input: DeploymentListInput {
project_id: Some(linked_project.project.clone()),
environment_id: Some(environment_id),
service_id: Some(service),
include_deleted: None,
status: None,
},
};
let deployments =
post_graphql::<queries::Deployments, _>(&client, configs.get_backboard(), vars)
.await?
.deployments;
let mut deployments: Vec<_> = deployments
.edges
.into_iter()
.filter_map(|deployment| {
(deployment.node.status == DeploymentStatus::SUCCESS).then_some(deployment.node)
})
.collect();
let deployment;
if let Some(deployment_id) = args.deployment_id {
deployment = deployments
.iter()
.find(|deployment| deployment.id == deployment_id)
.context("Deployment id does not exist")?;
} else {
deployments.sort_by(|a, b| b.created_at.cmp(&a.created_at));
deployment = deployments.first().context("No deployments found")?;
};
if (args.build || deployment.status == DeploymentStatus::FAILED) && !args.deployment {
stream_build_logs(deployment.id.clone(), |log| {
if json {
println!("{}", serde_json::to_string(&log).unwrap());
} else {
println!("{}", log.message);
}
})
.await?;
} else {
stream_deploy_logs(
deployment.id.clone(),
|log: subscriptions::deployment_logs::LogFields| {
if json {
let mut map: HashMap<String, Value> = HashMap::new();
map.insert(
"message".to_string(),
serde_json::to_value(log.message.clone()).unwrap(),
);
map.insert(
"timestamp".to_string(),
serde_json::to_value(log.timestamp.clone()).unwrap(),
);
for attribute in log.attributes {
let value = match attribute.value.trim_matches('"').parse::<Value>() {
Ok(value) => value,
Err(_) => {
serde_json::to_value(attribute.value.trim_matches('"')).unwrap()
}
};
map.insert(attribute.key, value);
}
let json_string = serde_json::to_string(&map).unwrap();
println!("{}", json_string);
} else {
format_attr_log(log);
}
},
)
.await?;
}
Ok(())
}