#![expect(clippy::unwrap_used)]
use wiremock::matchers::{method, path};
use wiremock::{Mock, ResponseTemplate};
use crate::cli::UserAction;
use crate::test_helpers::{capture_stdout, setup_test_env};
use crate::types::OutputFormat;
#[test]
fn resolve_login_denied_text_disable_with_custom_text() {
assert_eq!(
super::resolve_login_denied_text(Some(true), Some("Go away")),
Some("Go away".into())
);
}
#[test]
fn resolve_login_denied_text_disable_without_custom_text_uses_default() {
assert_eq!(
super::resolve_login_denied_text(Some(true), None),
Some("Account disabled".into())
);
}
#[test]
fn resolve_login_denied_text_enable_clears_to_empty_string() {
assert_eq!(
super::resolve_login_denied_text(Some(false), None),
Some(String::new())
);
assert_eq!(
super::resolve_login_denied_text(Some(false), Some("ignored")),
Some(String::new())
);
}
#[test]
fn resolve_login_denied_text_unset_returns_none() {
assert_eq!(super::resolve_login_denied_text(None, None), None);
assert_eq!(
super::resolve_login_denied_text(None, Some("ignored")),
None
);
}
#[tokio::test]
async fn user_search_returns_results() {
let (_lock, mock, _tmp) = setup_test_env().await;
Mock::given(method("GET"))
.and(path("/rest/user"))
.respond_with(ResponseTemplate::new(200).set_body_json(serde_json::json!({
"users": [{
"id": 1,
"name": "alice@test.com",
"real_name": "Alice"
}]
})))
.mount(&mock)
.await;
let action = UserAction::Search {
query: "alice".to_string(),
details: false,
};
let (result, output) =
capture_stdout(super::execute(&action, None, OutputFormat::Json, None)).await;
assert!(result.is_ok());
let parsed: serde_json::Value = crate::test_helpers::extract_json(&output);
assert_eq!(parsed[0]["id"], 1);
assert_eq!(parsed[0]["name"], "alice@test.com");
}
#[tokio::test]
async fn update_user_disable_login_sends_denied_text() {
let (_lock, mock, _tmp) = setup_test_env().await;
Mock::given(method("PUT"))
.and(path("/rest/user/alice%40test%2Ecom"))
.respond_with(ResponseTemplate::new(200).set_body_json(serde_json::json!({})))
.expect(1)
.mount(&mock)
.await;
let action = UserAction::Update {
user: "alice@test.com".to_string(),
real_name: None,
email: None,
disable_login: Some(true),
login_denied_text: Some("Go away".to_string()),
};
let result = super::execute(&action, None, OutputFormat::Json, None).await;
assert!(
result.is_ok(),
"update with disable_login failed: {result:?}"
);
}
#[tokio::test]
async fn update_user_enable_login_sends_empty_denied_text() {
let (_lock, mock, _tmp) = setup_test_env().await;
Mock::given(method("PUT"))
.and(path("/rest/user/bob%40test%2Ecom"))
.respond_with(ResponseTemplate::new(200).set_body_json(serde_json::json!({})))
.expect(1)
.mount(&mock)
.await;
let action = UserAction::Update {
user: "bob@test.com".to_string(),
real_name: None,
email: None,
disable_login: Some(false),
login_denied_text: None,
};
let result = super::execute(&action, None, OutputFormat::Json, None).await;
assert!(
result.is_ok(),
"update with enable_login failed: {result:?}"
);
}
#[tokio::test]
async fn user_create_sends_post() {
let (_lock, mock, _tmp) = setup_test_env().await;
Mock::given(method("POST"))
.and(path("/rest/user"))
.respond_with(ResponseTemplate::new(201).set_body_json(serde_json::json!({"id": 99})))
.expect(1)
.mount(&mock)
.await;
let action = UserAction::Create {
email: "new@test.com".into(),
login: None,
full_name: Some("New User".into()),
password: None,
};
let (result, output) =
capture_stdout(super::execute(&action, None, OutputFormat::Json, None)).await;
assert!(result.is_ok(), "user create failed: {result:?}");
let parsed: serde_json::Value = crate::test_helpers::extract_json(&output);
assert_eq!(parsed["action"], "created");
assert_eq!(parsed["id"], 99);
}
#[tokio::test]
async fn user_search_http_500_returns_error() {
let (_lock, mock, _tmp) = setup_test_env().await;
Mock::given(method("GET"))
.and(path("/rest/user"))
.respond_with(ResponseTemplate::new(500).set_body_string("Internal Server Error"))
.mount(&mock)
.await;
let action = UserAction::Search {
query: "alice".to_string(),
details: false,
};
let result = super::execute(&action, None, OutputFormat::Json, None).await;
assert!(result.is_err());
let err = result.unwrap_err().to_string();
assert!(
err.contains("500") || err.contains("Internal Server Error"),
"expected HTTP 500 error, got: {err}"
);
}
#[tokio::test]
async fn user_search_malformed_json_returns_error() {
let (_lock, mock, _tmp) = setup_test_env().await;
Mock::given(method("GET"))
.and(path("/rest/user"))
.respond_with(ResponseTemplate::new(200).set_body_string("not valid json"))
.mount(&mock)
.await;
let action = UserAction::Search {
query: "alice".to_string(),
details: false,
};
let result = super::execute(&action, None, OutputFormat::Json, None).await;
assert!(result.is_err());
}