use std::path::PathBuf;
const TOKEN_BYTES: usize = 32;
const TOKEN_FILE: &str = "session_token";
pub fn generate_token() -> String {
let mut buf = [0u8; TOKEN_BYTES];
getrandom::fill(&mut buf).expect("getrandom failed");
bytes_to_hex(&buf)
}
fn bytes_to_hex(bytes: &[u8]) -> String {
let mut s = String::with_capacity(bytes.len() * 2);
for &b in bytes {
s.push(char::from_digit((b >> 4) as u32, 16).unwrap_or('0'));
s.push(char::from_digit((b & 0xf) as u32, 16).unwrap_or('0'));
}
s
}
pub fn resolve_proxy_token(env_var: &str) -> String {
if let Ok(val) = std::env::var(env_var) {
if !val.trim().is_empty() {
return val.trim().to_string();
}
}
let token_path = token_file_path();
if let Ok(existing) = std::fs::read_to_string(&token_path) {
let t = existing.trim().to_string();
if !t.is_empty() {
return t;
}
}
let token = generate_token();
write_token_file(&token_path, &token);
token
}
fn write_token_file(path: &PathBuf, token: &str) {
if let Some(parent) = path.parent() {
if let Err(e) = std::fs::create_dir_all(parent) {
tracing::error!("Failed to create token directory {}: {e}", parent.display());
return;
}
}
if let Err(e) = std::fs::write(path, token) {
tracing::error!("Failed to write session token to {}: {e}", path.display());
return;
}
set_restrictive_permissions(path);
}
fn token_file_path() -> PathBuf {
crate::core::data_dir::lean_ctx_data_dir()
.unwrap_or_else(|_| PathBuf::from(".lean-ctx"))
.join(TOKEN_FILE)
}
#[cfg(unix)]
fn set_restrictive_permissions(path: &PathBuf) {
use std::os::unix::fs::PermissionsExt;
let perms = std::fs::Permissions::from_mode(0o600);
let _ = std::fs::set_permissions(path, perms);
}
#[cfg(not(unix))]
fn set_restrictive_permissions(_path: &PathBuf) {
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn generated_token_is_64_hex_chars() {
let token = generate_token();
assert_eq!(token.len(), 64);
assert!(token.chars().all(|c| c.is_ascii_hexdigit()));
}
#[test]
fn generated_tokens_are_unique() {
let a = generate_token();
let b = generate_token();
assert_ne!(a, b);
}
}