use std::process::Command;
use assert_cmd::prelude::*;
use jira_cli::output::exit_codes;
use jira_cli::test_support::{EnvVarGuard, ProcessEnvLock, set_config_dir_env, write_config};
use tempfile::TempDir;
use wiremock::matchers::{body_partial_json, method, path};
use wiremock::{Mock, MockServer, ResponseTemplate};
fn config_fixture() -> &'static str {
r#"
[default]
host = "work.atlassian.net"
email = "me@example.com"
token = "secret-token"
"#
}
#[test]
fn config_show_auto_json_when_piped() {
let _env = ProcessEnvLock::acquire().unwrap();
let dir = TempDir::new().unwrap();
let config_path = write_config(dir.path(), config_fixture()).unwrap();
let _config_dir = set_config_dir_env(dir.path());
let _host = EnvVarGuard::unset("JIRA_HOST");
let _email = EnvVarGuard::unset("JIRA_EMAIL");
let _token = EnvVarGuard::unset("JIRA_TOKEN");
let _profile = EnvVarGuard::unset("JIRA_PROFILE");
let output = Command::cargo_bin("jira")
.unwrap()
.args(["config", "show"])
.output()
.unwrap();
assert!(output.status.success());
assert!(output.stderr.is_empty());
let json: serde_json::Value = serde_json::from_slice(&output.stdout).unwrap();
assert_eq!(json["configPath"], config_path.display().to_string());
assert_eq!(json["host"], "work.atlassian.net");
assert_eq!(json["email"], "me@example.com");
assert_eq!(json["tokenMasked"], "***oken");
}
#[test]
fn config_init_auto_json_when_piped() {
let _env = ProcessEnvLock::acquire().unwrap();
let dir = TempDir::new().unwrap();
let _config_dir = set_config_dir_env(dir.path());
let output = Command::cargo_bin("jira")
.unwrap()
.args(["config", "init"])
.output()
.unwrap();
assert!(output.status.success());
assert!(output.stderr.is_empty());
let json: serde_json::Value = serde_json::from_slice(&output.stdout).unwrap();
assert_eq!(
json["configPath"],
dir.path()
.join("jira")
.join("config.toml")
.display()
.to_string()
);
assert_eq!(
json["tokenInstructions"],
"https://id.atlassian.com/manage-profile/security/api-tokens"
);
assert_eq!(
json["example"]["default"]["host"],
"mycompany.atlassian.net"
);
assert!(json["pathResolution"].as_str().is_some());
assert!(json["recommendedPermissions"].as_str().is_some());
assert_eq!(json["configExists"], false);
}
#[test]
fn init_alias_matches_config_init_json_contract() {
let _env = ProcessEnvLock::acquire().unwrap();
let dir = TempDir::new().unwrap();
let _config_dir = set_config_dir_env(dir.path());
let output = Command::cargo_bin("jira")
.unwrap()
.args(["init"])
.output()
.unwrap();
assert!(output.status.success());
assert!(output.stderr.is_empty());
let json: serde_json::Value = serde_json::from_slice(&output.stdout).unwrap();
assert_eq!(
json["configPath"],
dir.path()
.join("jira")
.join("config.toml")
.display()
.to_string()
);
assert_eq!(
json["tokenInstructions"],
"https://id.atlassian.com/manage-profile/security/api-tokens"
);
}
#[test]
fn config_show_invalid_config_returns_input_exit_code() {
let _env = ProcessEnvLock::acquire().unwrap();
let dir = TempDir::new().unwrap();
let _config_dir = set_config_dir_env(dir.path());
let _host = EnvVarGuard::unset("JIRA_HOST");
let _email = EnvVarGuard::unset("JIRA_EMAIL");
let _token = EnvVarGuard::unset("JIRA_TOKEN");
let _profile = EnvVarGuard::unset("JIRA_PROFILE");
let output = Command::cargo_bin("jira")
.unwrap()
.args(["config", "show"])
.output()
.unwrap();
assert_eq!(output.status.code(), Some(exit_codes::INPUT_ERROR));
assert!(output.stdout.is_empty());
let stderr = String::from_utf8(output.stderr).unwrap();
assert!(stderr.contains("No Jira host configured"));
}
#[test]
fn completions_install_powershell_returns_input_error() {
let output = Command::cargo_bin("jira")
.unwrap()
.args(["completions", "powershell", "--install"])
.output()
.unwrap();
assert_eq!(output.status.code(), Some(exit_codes::INPUT_ERROR));
assert!(output.stdout.is_empty());
let stderr = String::from_utf8(output.stderr).unwrap();
assert!(stderr.contains("not supported"));
assert!(stderr.to_lowercase().contains("redirect"));
}
fn run_jira_against(server: &MockServer, args: &[&str]) -> std::process::Output {
let _env = ProcessEnvLock::acquire().unwrap();
let dir = TempDir::new().unwrap();
let _config_dir = set_config_dir_env(dir.path());
let host = server.uri();
let _host = EnvVarGuard::set("JIRA_HOST", &host);
let _email = EnvVarGuard::set("JIRA_EMAIL", "test@example.com");
let _token = EnvVarGuard::set("JIRA_TOKEN", "test-token");
let _profile = EnvVarGuard::unset("JIRA_PROFILE");
Command::cargo_bin("jira")
.unwrap()
.args(args)
.env("NO_COLOR", "1")
.output()
.unwrap()
}
#[tokio::test]
async fn issues_update_dispatch_assignee_me_calls_myself_then_puts_account_id() {
let server = MockServer::start().await;
Mock::given(method("GET"))
.and(path("/rest/api/3/myself"))
.respond_with(ResponseTemplate::new(200).set_body_json(serde_json::json!({
"accountId": "abc-self-123",
"displayName": "Test User",
"emailAddress": "test@example.com",
})))
.expect(1)
.mount(&server)
.await;
Mock::given(method("PUT"))
.and(path("/rest/api/3/issue/PROJ-1"))
.and(body_partial_json(serde_json::json!({
"fields": { "assignee": { "accountId": "abc-self-123" } }
})))
.respond_with(ResponseTemplate::new(204))
.expect(1)
.mount(&server)
.await;
let output = run_jira_against(&server, &["issues", "update", "PROJ-1", "--assignee", "me"]);
assert!(
output.status.success(),
"stderr: {}",
String::from_utf8_lossy(&output.stderr)
);
}
#[tokio::test]
async fn issues_update_dispatch_assignee_none_sends_null_in_single_put() {
let server = MockServer::start().await;
Mock::given(method("PUT"))
.and(path("/rest/api/3/issue/PROJ-1"))
.and(body_partial_json(serde_json::json!({
"fields": { "assignee": null }
})))
.respond_with(ResponseTemplate::new(204))
.expect(1)
.mount(&server)
.await;
let output = run_jira_against(
&server,
&["issues", "update", "PROJ-1", "--assignee", "none"],
);
assert!(
output.status.success(),
"stderr: {}",
String::from_utf8_lossy(&output.stderr)
);
}
#[tokio::test]
async fn issues_update_dispatch_fix_versions_none_sends_empty_array() {
let server = MockServer::start().await;
Mock::given(method("PUT"))
.and(path("/rest/api/3/issue/PROJ-1"))
.and(body_partial_json(serde_json::json!({
"fields": { "fixVersions": [] }
})))
.respond_with(ResponseTemplate::new(204))
.expect(1)
.mount(&server)
.await;
let output = run_jira_against(
&server,
&["issues", "update", "PROJ-1", "--fix-versions", "none"],
);
assert!(
output.status.success(),
"stderr: {}",
String::from_utf8_lossy(&output.stderr)
);
}
#[tokio::test]
async fn issues_update_dispatch_labels_passthrough() {
let server = MockServer::start().await;
Mock::given(method("PUT"))
.and(path("/rest/api/3/issue/PROJ-1"))
.and(body_partial_json(serde_json::json!({
"fields": { "labels": ["backend", "urgent"] }
})))
.respond_with(ResponseTemplate::new(204))
.expect(1)
.mount(&server)
.await;
let output = run_jira_against(
&server,
&[
"issues", "update", "PROJ-1", "--labels", "backend", "--labels", "urgent",
],
);
assert!(
output.status.success(),
"stderr: {}",
String::from_utf8_lossy(&output.stderr)
);
}
#[tokio::test]
async fn issues_update_dispatch_combined_flags_send_one_put() {
let server = MockServer::start().await;
Mock::given(method("GET"))
.and(path("/rest/api/3/myself"))
.respond_with(ResponseTemplate::new(200).set_body_json(serde_json::json!({
"accountId": "abc-self-123",
"displayName": "Test User",
"emailAddress": "test@example.com",
})))
.expect(1)
.mount(&server)
.await;
Mock::given(method("PUT"))
.and(path("/rest/api/3/issue/PROJ-1"))
.and(body_partial_json(serde_json::json!({
"fields": {
"summary": "Updated summary",
"fixVersions": [{ "name": "1.2.0" }],
"labels": ["backend"],
"assignee": { "accountId": "abc-self-123" }
}
})))
.respond_with(ResponseTemplate::new(204))
.expect(1)
.mount(&server)
.await;
let output = run_jira_against(
&server,
&[
"issues",
"update",
"PROJ-1",
"--summary",
"Updated summary",
"--fix-versions",
"1.2.0",
"--labels",
"backend",
"--assignee",
"me",
],
);
assert!(
output.status.success(),
"stderr: {}",
String::from_utf8_lossy(&output.stderr)
);
}