use std::collections::HashMap;
use std::fs;
use anyhow::{Context, Result};
use serde::Serialize;
use crate::atlassian::error::AtlassianError;
use crate::utils::settings::Settings;
pub const ATLASSIAN_INSTANCE_URL: &str = "ATLASSIAN_INSTANCE_URL";
pub const ATLASSIAN_EMAIL: &str = "ATLASSIAN_EMAIL";
pub const ATLASSIAN_API_TOKEN: &str = "ATLASSIAN_API_TOKEN";
#[derive(Debug, Clone)]
pub struct AtlassianCredentials {
pub instance_url: String,
pub email: String,
pub api_token: String,
}
pub fn load_credentials() -> Result<AtlassianCredentials> {
let settings = Settings::load().unwrap_or(Settings {
env: HashMap::new(),
});
let instance_url = settings
.get_env_var(ATLASSIAN_INSTANCE_URL)
.ok_or(AtlassianError::CredentialsNotFound)?;
let email = settings
.get_env_var(ATLASSIAN_EMAIL)
.ok_or(AtlassianError::CredentialsNotFound)?;
let api_token = settings
.get_env_var(ATLASSIAN_API_TOKEN)
.ok_or(AtlassianError::CredentialsNotFound)?;
let instance_url = instance_url.trim_end_matches('/').to_string();
Ok(AtlassianCredentials {
instance_url,
email,
api_token,
})
}
#[derive(Debug, Clone, Serialize, PartialEq, Eq)]
pub struct AtlassianScopeStatus {
pub name: String,
pub has_email: bool,
pub has_token: bool,
#[serde(skip_serializing_if = "Option::is_none")]
pub instance_url: Option<String>,
}
#[derive(Debug, Clone, Serialize, PartialEq, Eq)]
pub struct AuthStatus {
pub scopes: Vec<AtlassianScopeStatus>,
}
pub fn status() -> AuthStatus {
let settings = Settings::load().unwrap_or(Settings {
env: HashMap::new(),
});
let instance_url = settings
.get_env_var(ATLASSIAN_INSTANCE_URL)
.map(|v| v.trim_end_matches('/').to_string());
let has_email = settings.get_env_var(ATLASSIAN_EMAIL).is_some();
let has_token = settings.get_env_var(ATLASSIAN_API_TOKEN).is_some();
AuthStatus {
scopes: vec![AtlassianScopeStatus {
name: "default".to_string(),
has_email,
has_token,
instance_url,
}],
}
}
pub fn save_credentials(credentials: &AtlassianCredentials) -> Result<()> {
let settings_path = Settings::get_settings_path()?;
let mut settings_value: serde_json::Value = if settings_path.exists() {
let content = fs::read_to_string(&settings_path)
.with_context(|| format!("Failed to read {}", settings_path.display()))?;
serde_json::from_str(&content)
.with_context(|| format!("Failed to parse {}", settings_path.display()))?
} else {
serde_json::json!({})
};
if !settings_value
.get("env")
.is_some_and(serde_json::Value::is_object)
{
settings_value["env"] = serde_json::json!({});
}
let Some(env) = settings_value["env"].as_object_mut() else {
anyhow::bail!("Internal error: env key is not an object after initialization");
};
env.insert(
ATLASSIAN_INSTANCE_URL.to_string(),
serde_json::Value::String(credentials.instance_url.clone()),
);
env.insert(
ATLASSIAN_EMAIL.to_string(),
serde_json::Value::String(credentials.email.clone()),
);
env.insert(
ATLASSIAN_API_TOKEN.to_string(),
serde_json::Value::String(credentials.api_token.clone()),
);
if let Some(parent) = settings_path.parent() {
fs::create_dir_all(parent)
.with_context(|| format!("Failed to create directory {}", parent.display()))?;
}
let formatted = serde_json::to_string_pretty(&settings_value)
.context("Failed to serialize settings JSON")?;
fs::write(&settings_path, formatted)
.with_context(|| format!("Failed to write {}", settings_path.display()))?;
Ok(())
}
#[cfg(test)]
#[allow(clippy::unwrap_used, clippy::expect_used)]
mod tests {
use super::*;
#[test]
fn save_and_read_credentials() {
let temp_dir = {
std::fs::create_dir_all("tmp").ok();
tempfile::TempDir::new_in("tmp").unwrap()
};
let settings_path = temp_dir.path().join("settings.json");
let existing = r#"{"env": {"SOME_KEY": "value"}}"#;
fs::write(&settings_path, existing).unwrap();
let content = fs::read_to_string(&settings_path).unwrap();
let mut val: serde_json::Value = serde_json::from_str(&content).unwrap();
val["env"]["ATLASSIAN_INSTANCE_URL"] =
serde_json::Value::String("https://test.atlassian.net".to_string());
val["env"]["ATLASSIAN_EMAIL"] = serde_json::Value::String("user@example.com".to_string());
val["env"]["ATLASSIAN_API_TOKEN"] = serde_json::Value::String("secret-token".to_string());
let formatted = serde_json::to_string_pretty(&val).unwrap();
fs::write(&settings_path, formatted).unwrap();
let content = fs::read_to_string(&settings_path).unwrap();
let val: serde_json::Value = serde_json::from_str(&content).unwrap();
assert_eq!(val["env"]["SOME_KEY"], "value");
assert_eq!(
val["env"]["ATLASSIAN_INSTANCE_URL"],
"https://test.atlassian.net"
);
assert_eq!(val["env"]["ATLASSIAN_EMAIL"], "user@example.com");
assert_eq!(val["env"]["ATLASSIAN_API_TOKEN"], "secret-token");
}
#[test]
fn load_credentials_normalizes_trailing_slash() {
let url = "https://env.atlassian.net/";
let normalized = url.trim_end_matches('/').to_string();
assert_eq!(normalized, "https://env.atlassian.net");
}
#[test]
fn constant_key_names() {
assert_eq!(ATLASSIAN_INSTANCE_URL, "ATLASSIAN_INSTANCE_URL");
assert_eq!(ATLASSIAN_EMAIL, "ATLASSIAN_EMAIL");
assert_eq!(ATLASSIAN_API_TOKEN, "ATLASSIAN_API_TOKEN");
}
#[test]
fn credentials_struct_clone_and_debug() {
let creds = AtlassianCredentials {
instance_url: "https://org.atlassian.net".to_string(),
email: "user@test.com".to_string(),
api_token: "token".to_string(),
};
let cloned = creds.clone();
assert_eq!(cloned.instance_url, creds.instance_url);
assert_eq!(cloned.email, creds.email);
assert_eq!(cloned.api_token, creds.api_token);
let debug = format!("{creds:?}");
assert!(debug.contains("AtlassianCredentials"));
}
static AUTH_ENV_MUTEX: std::sync::Mutex<()> = std::sync::Mutex::new(());
struct EnvGuard {
_lock: std::sync::MutexGuard<'static, ()>,
snapshot: Vec<(&'static str, Option<String>)>,
}
impl EnvGuard {
fn take() -> Self {
let lock = AUTH_ENV_MUTEX
.lock()
.unwrap_or_else(std::sync::PoisonError::into_inner);
let keys = [
"HOME",
ATLASSIAN_INSTANCE_URL,
ATLASSIAN_EMAIL,
ATLASSIAN_API_TOKEN,
];
let snapshot = keys
.into_iter()
.map(|k| (k, std::env::var(k).ok()))
.collect();
Self {
_lock: lock,
snapshot,
}
}
}
impl Drop for EnvGuard {
fn drop(&mut self) {
for (k, v) in &self.snapshot {
match v {
Some(val) => std::env::set_var(k, val),
None => std::env::remove_var(k),
}
}
}
}
fn with_empty_home(_guard: &EnvGuard) -> tempfile::TempDir {
let dir = {
std::fs::create_dir_all("tmp").ok();
tempfile::TempDir::new_in("tmp").unwrap()
};
std::env::set_var("HOME", dir.path());
std::env::remove_var(ATLASSIAN_INSTANCE_URL);
std::env::remove_var(ATLASSIAN_EMAIL);
std::env::remove_var(ATLASSIAN_API_TOKEN);
dir
}
#[test]
fn status_reports_all_false_when_nothing_configured() {
let guard = EnvGuard::take();
let _dir = with_empty_home(&guard);
let status = status();
assert_eq!(status.scopes.len(), 1);
let scope = &status.scopes[0];
assert_eq!(scope.name, "default");
assert!(!scope.has_email);
assert!(!scope.has_token);
assert_eq!(scope.instance_url, None);
}
#[test]
fn status_reports_presence_flags_from_settings_without_leaking_secrets() {
let guard = EnvGuard::take();
let dir = with_empty_home(&guard);
let omni_dir = dir.path().join(".omni-dev");
fs::create_dir_all(&omni_dir).unwrap();
fs::write(
omni_dir.join("settings.json"),
r#"{"env":{
"ATLASSIAN_INSTANCE_URL":"https://status.atlassian.net/",
"ATLASSIAN_EMAIL":"person@example.com",
"ATLASSIAN_API_TOKEN":"sekret-do-not-leak"
}}"#,
)
.unwrap();
let status = status();
assert_eq!(status.scopes.len(), 1);
let scope = &status.scopes[0];
assert!(scope.has_email);
assert!(scope.has_token);
assert_eq!(
scope.instance_url.as_deref(),
Some("https://status.atlassian.net")
);
let yaml = serde_yaml::to_string(&status).unwrap();
assert!(!yaml.contains("sekret-do-not-leak"), "leaked token: {yaml}");
assert!(!yaml.contains("person@example.com"), "leaked email: {yaml}");
}
#[test]
fn status_returns_instance_url_from_env_without_trailing_slash() {
let guard = EnvGuard::take();
let _dir = with_empty_home(&guard);
std::env::set_var(ATLASSIAN_INSTANCE_URL, "https://env.atlassian.net/");
let status = status();
let scope = &status.scopes[0];
assert_eq!(
scope.instance_url.as_deref(),
Some("https://env.atlassian.net")
);
assert!(!scope.has_email);
assert!(!scope.has_token);
}
#[test]
fn save_credentials_creates_and_preserves() {
let _guard = EnvGuard::take();
let original_home = std::env::var("HOME").ok();
{
let temp_dir = {
std::fs::create_dir_all("tmp").ok();
tempfile::TempDir::new_in("tmp").unwrap()
};
std::env::set_var("HOME", temp_dir.path());
let creds = AtlassianCredentials {
instance_url: "https://save.atlassian.net".to_string(),
email: "save@example.com".to_string(),
api_token: "save-token".to_string(),
};
save_credentials(&creds).unwrap();
let settings_path = temp_dir.path().join(".omni-dev").join("settings.json");
assert!(settings_path.exists());
let content = fs::read_to_string(&settings_path).unwrap();
let val: serde_json::Value = serde_json::from_str(&content).unwrap();
assert_eq!(
val["env"]["ATLASSIAN_INSTANCE_URL"],
"https://save.atlassian.net"
);
assert_eq!(val["env"]["ATLASSIAN_EMAIL"], "save@example.com");
assert_eq!(val["env"]["ATLASSIAN_API_TOKEN"], "save-token");
}
{
let temp_dir = {
std::fs::create_dir_all("tmp").ok();
tempfile::TempDir::new_in("tmp").unwrap()
};
let omni_dir = temp_dir.path().join(".omni-dev");
fs::create_dir_all(&omni_dir).unwrap();
let settings_path = omni_dir.join("settings.json");
fs::write(
&settings_path,
r#"{"env": {"OTHER_KEY": "keep_me"}, "extra": true}"#,
)
.unwrap();
std::env::set_var("HOME", temp_dir.path());
let creds = AtlassianCredentials {
instance_url: "https://org.atlassian.net".to_string(),
email: "user@test.com".to_string(),
api_token: "token".to_string(),
};
save_credentials(&creds).unwrap();
let content = fs::read_to_string(&settings_path).unwrap();
let val: serde_json::Value = serde_json::from_str(&content).unwrap();
assert_eq!(val["env"]["OTHER_KEY"], "keep_me");
assert_eq!(val["extra"], true);
assert_eq!(
val["env"]["ATLASSIAN_INSTANCE_URL"],
"https://org.atlassian.net"
);
}
if let Some(home) = original_home {
std::env::set_var("HOME", home);
}
}
}