use std::collections::HashMap;
use std::path::PathBuf;
use serde::{Deserialize, Serialize};
#[derive(Debug, Clone, Serialize, Deserialize)]
#[serde(rename_all = "camelCase")]
pub struct SessionState {
pub name: String,
pub backend_type: String,
pub prompt: String,
#[serde(skip_serializing_if = "Option::is_none")]
pub model: Option<String>,
#[serde(skip_serializing_if = "Option::is_none")]
pub cwd: Option<PathBuf>,
#[serde(skip_serializing_if = "Option::is_none")]
pub max_turns: Option<i32>,
#[serde(default, skip_serializing_if = "Vec::is_empty")]
pub allowed_tools: Vec<String>,
#[serde(skip_serializing_if = "Option::is_none")]
pub permission_mode: Option<String>,
#[serde(skip_serializing_if = "Option::is_none")]
pub reasoning_effort: Option<String>,
#[serde(default, skip_serializing_if = "HashMap::is_empty")]
pub env: HashMap<String, String>,
#[serde(default, skip_serializing_if = "Option::is_none")]
pub memory_config: Option<crate::memory::MemoryConfig>,
#[serde(default, skip_serializing_if = "HashMap::is_empty")]
pub metadata: HashMap<String, String>,
pub created_at: chrono::DateTime<chrono::Utc>,
}
impl SessionState {
pub fn from_config(
config: &crate::backend::SpawnConfig,
backend_type: &crate::backend::BackendType,
) -> Self {
Self {
name: config.name.clone(),
backend_type: backend_type.to_string(),
prompt: config.prompt.clone(),
model: config.model.clone(),
cwd: config.cwd.clone(),
max_turns: config.max_turns,
allowed_tools: config.allowed_tools.clone(),
permission_mode: config.permission_mode.clone(),
reasoning_effort: config.reasoning_effort.clone(),
env: config.env.clone(),
memory_config: config.memory_config.clone(),
metadata: HashMap::new(),
created_at: chrono::Utc::now(),
}
}
pub fn to_spawn_config(&self) -> crate::backend::SpawnConfig {
crate::backend::SpawnConfig {
name: self.name.clone(),
prompt: self.prompt.clone(),
model: self.model.clone(),
cwd: self.cwd.clone(),
max_turns: self.max_turns,
allowed_tools: self.allowed_tools.clone(),
permission_mode: self.permission_mode.clone(),
reasoning_effort: self.reasoning_effort.clone(),
env: self.env.clone(),
memory_config: self.memory_config.clone(),
delegations: Vec::new(),
}
}
pub fn parse_backend_type(&self) -> Option<crate::backend::BackendType> {
self.backend_type.parse().ok()
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn session_state_round_trip() {
let state = SessionState {
name: "test-agent".into(),
backend_type: "codex".into(),
prompt: "You are a test assistant.".into(),
model: Some("gpt-4.1".into()),
cwd: Some(PathBuf::from("/tmp")),
max_turns: None,
allowed_tools: vec![],
permission_mode: None,
reasoning_effort: Some("medium".into()),
env: HashMap::new(),
memory_config: None,
metadata: {
let mut m = HashMap::new();
m.insert("thread_id".into(), "abc-123".into());
m
},
created_at: chrono::Utc::now(),
};
let json = serde_json::to_string_pretty(&state).unwrap();
let parsed: SessionState = serde_json::from_str(&json).unwrap();
assert_eq!(parsed.name, "test-agent");
assert_eq!(parsed.backend_type, "codex");
assert_eq!(parsed.model.as_deref(), Some("gpt-4.1"));
assert_eq!(parsed.metadata.get("thread_id").unwrap(), "abc-123");
}
#[test]
fn from_config_and_back() {
let config = crate::backend::SpawnConfig::new("reviewer", "You review code.");
let state = SessionState::from_config(&config, &crate::backend::BackendType::GeminiCli);
assert_eq!(state.name, "reviewer");
assert_eq!(state.backend_type, "gemini-cli");
assert_eq!(state.parse_backend_type(), Some(crate::backend::BackendType::GeminiCli));
let restored = state.to_spawn_config();
assert_eq!(restored.name, "reviewer");
assert_eq!(restored.prompt, "You review code.");
}
}