use std::collections::HashMap;
use std::fs;
use std::io::Write as _;
use std::path::PathBuf;
use serde::{Deserialize, Serialize};
use crate::error::{BzrError, Result};
use crate::types::{ApiMode, AuthMethod, BugTemplate, SavedQuery};
#[derive(Debug, Serialize, Deserialize, Default)]
#[non_exhaustive]
pub struct Config {
pub default_server: Option<String>,
#[serde(default)]
pub servers: HashMap<String, ServerConfig>,
#[serde(default, skip_serializing_if = "HashMap::is_empty")]
pub templates: HashMap<String, BugTemplate>,
#[serde(default, skip_serializing_if = "HashMap::is_empty")]
pub queries: HashMap<String, SavedQuery>,
}
#[derive(Debug, Clone, Serialize, Deserialize)]
#[non_exhaustive]
pub struct ServerConfig {
pub url: String,
#[serde(default, skip_serializing_if = "Option::is_none")]
pub api_key: Option<String>,
#[serde(default, skip_serializing_if = "Option::is_none")]
pub api_key_env: Option<String>,
#[serde(default, skip_serializing_if = "Option::is_none")]
pub api_key_keyring: Option<KeyringRef>,
#[serde(default, skip_serializing_if = "Option::is_none")]
pub email: Option<String>,
#[serde(default, skip_serializing_if = "Option::is_none")]
pub auth_method: Option<AuthMethod>,
#[serde(default, skip_serializing_if = "Option::is_none")]
pub api_mode: Option<ApiMode>,
#[serde(default, skip_serializing_if = "Option::is_none")]
pub server_version: Option<String>,
#[serde(default, skip_serializing_if = "std::ops::Not::not")]
pub tls_insecure: bool,
#[serde(default, skip_serializing_if = "Option::is_none")]
pub tls_ca_cert: Option<PathBuf>,
#[serde(default, skip_serializing_if = "Option::is_none")]
pub tls_pin_sha256: Option<String>,
#[serde(default, skip_serializing_if = "Option::is_none")]
pub tls_pin_issuer: Option<String>,
#[serde(default, skip_serializing_if = "Option::is_none")]
pub tls_pin_issuer_der: Option<String>,
}
#[derive(Debug, Clone, Serialize, Deserialize, Default)]
#[non_exhaustive]
pub struct KeyringRef {
#[serde(default, skip_serializing_if = "Option::is_none")]
pub service: Option<String>,
#[serde(default, skip_serializing_if = "Option::is_none")]
pub account: Option<String>,
}
impl KeyringRef {
pub fn service_or_default(&self) -> &str {
self.service.as_deref().unwrap_or("bzr")
}
pub fn account_or_default<'a>(&'a self, server_name: &'a str) -> &'a str {
self.account.as_deref().unwrap_or(server_name)
}
}
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
pub enum CredentialSourceKind {
Inline,
Env,
Keyring,
}
#[derive(Debug)]
pub enum CredentialSource<'a> {
Inline(&'a str),
EnvVar(&'a str),
Keyring { service: &'a str, account: &'a str },
}
impl CredentialSource<'_> {
pub fn kind(&self) -> CredentialSourceKind {
match self {
CredentialSource::Inline(_) => CredentialSourceKind::Inline,
CredentialSource::EnvVar(_) => CredentialSourceKind::Env,
CredentialSource::Keyring { .. } => CredentialSourceKind::Keyring,
}
}
}
impl CredentialSourceKind {
pub fn as_str(self) -> &'static str {
match self {
CredentialSourceKind::Inline => "inline",
CredentialSourceKind::Env => "env",
CredentialSourceKind::Keyring => "keyring",
}
}
}
impl ServerConfig {
pub fn tls_config(&self, server_name: &str) -> crate::tls::TlsConfig {
crate::tls::TlsConfig {
insecure: self.tls_insecure,
ca_cert_path: self.tls_ca_cert.clone(),
pin_sha256: self.tls_pin_sha256.clone(),
pin_issuer_der: self.tls_pin_issuer_der.clone(),
server_name: Some(server_name.to_string()),
}
}
pub fn validate(&self, server_name: &str) -> Result<()> {
self.credential_source()
.map(|_| ())
.map_err(|err| BzrError::config(format!("server '{server_name}': {err}")))?;
self.validate_tls(server_name)
}
pub fn credential_source(&self) -> Result<CredentialSource<'_>> {
let count = usize::from(self.api_key.is_some())
+ usize::from(self.api_key_env.is_some())
+ usize::from(self.api_key_keyring.is_some());
match count {
0 => Err(BzrError::config(
"server config must define one of 'api_key', 'api_key_env', or 'api_key_keyring'",
)),
1 => {
if let Some(api_key) = self.api_key.as_deref() {
Ok(CredentialSource::Inline(api_key))
} else if let Some(var_name) = self.api_key_env.as_deref() {
Ok(CredentialSource::EnvVar(var_name))
} else {
let r = self.api_key_keyring.as_ref().ok_or_else(|| {
BzrError::config("internal: keyring credential unexpectedly missing")
})?;
Ok(CredentialSource::Keyring {
service: r.service_or_default(),
account: r.account.as_deref().unwrap_or(""),
})
}
}
_ => Err(BzrError::config(
"server config cannot define multiple API key sources \
(api_key, api_key_env, api_key_keyring)",
)),
}
}
pub fn credential_source_kind(&self) -> Result<CredentialSourceKind> {
Ok(self.credential_source()?.kind())
}
pub fn resolve_api_key(&self, server_name: &str) -> Result<String> {
match self.credential_source()? {
CredentialSource::Inline(api_key) => Ok(api_key.to_string()),
CredentialSource::EnvVar(var_name) => {
let value = std::env::var(var_name).map_err(|_| {
BzrError::config(format!(
"server '{server_name}' uses API key env var '{var_name}', but it is not set"
))
})?;
if value.is_empty() {
return Err(BzrError::config(format!(
"server '{server_name}' uses API key env var '{var_name}', but it is empty"
)));
}
Ok(value)
}
CredentialSource::Keyring { service, account } => {
let account = if account.is_empty() {
server_name
} else {
account
};
crate::credentials::keyring::retrieve(service, account)
}
}
}
pub fn validate_tls(&self, server_name: &str) -> Result<()> {
let ctx = |msg: &str| BzrError::config(format!("server '{server_name}': {msg}"));
if self.tls_insecure && self.tls_ca_cert.is_some() {
return Err(ctx("tls_insecure and tls_ca_cert are mutually exclusive"));
}
if self.tls_insecure && self.tls_pin_sha256.is_some() {
return Err(ctx(
"tls_insecure and tls_pin_sha256 are mutually exclusive",
));
}
if self.tls_ca_cert.is_some() && self.tls_pin_sha256.is_some() {
return Err(ctx("tls_ca_cert and tls_pin_sha256 are mutually exclusive"));
}
if let Some(path) = &self.tls_ca_cert {
if !path.exists() {
return Err(BzrError::config(format!(
"server '{server_name}': tls_ca_cert file not found: {}",
path.display()
)));
}
}
if let Some(pin) = &self.tls_pin_sha256 {
crate::tls::fingerprint::parse_pin(pin)
.map_err(|e| ctx(&format!("invalid tls_pin_sha256: {e}")))?;
}
Ok(())
}
}
impl Config {
pub fn path() -> Result<PathBuf> {
let config_dir = std::env::var_os("XDG_CONFIG_HOME")
.map(PathBuf::from)
.filter(|p| p.is_absolute())
.or_else(dirs::config_dir)
.ok_or_else(|| BzrError::config("cannot determine config directory"))?;
Ok(config_dir.join("bzr").join("config.toml"))
}
pub fn load() -> Result<Config> {
let path = Self::path()?;
match fs::read_to_string(&path) {
Ok(content) => {
Self::warn_on_insecure_permissions(&path);
let config: Config = toml::from_str(&content)?;
config.validate()?;
Ok(config)
}
Err(e) if e.kind() == std::io::ErrorKind::NotFound => Ok(Config::default()),
Err(e) => Err(e.into()),
}
}
pub fn save(&self) -> Result<()> {
self.validate()?;
self.write_to_disk()
}
pub fn save_without_validation(&self) -> Result<()> {
self.write_to_disk()
}
fn write_to_disk(&self) -> Result<()> {
let path = Self::path()?;
if let Some(parent) = path.parent() {
let parent_exists = parent.exists();
fs::create_dir_all(parent)?;
if !parent_exists {
set_private_directory_permissions(parent)?;
}
}
let content = toml::to_string_pretty(self)?;
let file_exists = path.exists();
write_private_file(&path, &content)?;
if !file_exists {
set_private_file_permissions(&path)?;
}
Self::warn_on_insecure_permissions(&path);
Ok(())
}
pub fn resolve_server<'a>(
&'a self,
server_name: Option<&'a str>,
) -> Result<(&'a str, &'a ServerConfig)> {
let name = self.resolve_server_name_only(server_name)?;
let srv = self
.servers
.get(name)
.ok_or_else(|| BzrError::config(format!("server '{name}' not found in config")))?;
Ok((name, srv))
}
pub fn resolve_server_name_only<'a>(&'a self, server_name: Option<&'a str>) -> Result<&'a str> {
server_name
.or(self.default_server.as_deref())
.ok_or_else(|| {
BzrError::config(
"no server configured. Run `bzr config set-server <name> --url <url> --api-key-env <env-var>` first",
)
})
}
fn warn_on_insecure_permissions(path: &std::path::Path) {
#[cfg(unix)]
{
if let Some(parent) = path.parent() {
warn_if_path_permissions_too_open(parent, 0o077, "config directory");
}
if path.exists() {
warn_if_path_permissions_too_open(path, 0o077, "config file");
}
}
}
fn validate(&self) -> Result<()> {
for (name, server) in &self.servers {
server.validate(name)?;
}
Ok(())
}
}
#[cfg(unix)]
fn write_private_file(path: &std::path::Path, content: &str) -> Result<()> {
use std::fs::OpenOptions;
use std::os::unix::fs::OpenOptionsExt;
let mut file = OpenOptions::new()
.create(true)
.truncate(true)
.write(true)
.mode(0o600)
.open(path)?;
file.write_all(content.as_bytes())?;
Ok(())
}
#[cfg(not(unix))]
fn write_private_file(path: &std::path::Path, content: &str) -> Result<()> {
fs::write(path, content)?;
Ok(())
}
#[cfg(unix)]
fn set_private_directory_permissions(path: &std::path::Path) -> Result<()> {
use std::os::unix::fs::PermissionsExt;
fs::set_permissions(path, fs::Permissions::from_mode(0o700))?;
Ok(())
}
#[cfg(not(unix))]
fn set_private_directory_permissions(_path: &std::path::Path) -> Result<()> {
Ok(())
}
#[cfg(unix)]
fn set_private_file_permissions(path: &std::path::Path) -> Result<()> {
use std::os::unix::fs::PermissionsExt;
fs::set_permissions(path, fs::Permissions::from_mode(0o600))?;
Ok(())
}
#[cfg(not(unix))]
fn set_private_file_permissions(_path: &std::path::Path) -> Result<()> {
Ok(())
}
#[cfg(unix)]
fn warn_if_path_permissions_too_open(path: &std::path::Path, mask: u32, kind: &str) {
use std::os::unix::fs::PermissionsExt;
let Ok(metadata) = fs::metadata(path) else {
return;
};
let mode = metadata.permissions().mode();
if mode & mask == 0 {
return;
}
warn_security(&format!(
"{kind} '{}' has overly broad permissions ({:o}); expected owner-only access. Fix with `chmod {}` '{}'",
path.display(),
mode & 0o777,
if kind == "config directory" { "700" } else { "600" },
path.display()
));
}
#[expect(clippy::print_stderr)]
fn warn_security(message: &str) {
eprintln!("warning: {message}");
}
#[cfg(test)]
#[path = "config_tests.rs"]
mod tests;