mod input; mod role; mod session; pub use self::input::{Input, InputContext}; use self::role::Role; pub use self::role::{CODE_ROLE, EXPLAIN_ROLE, SHELL_ROLE}; use self::session::{Session, TEMP_SESSION_NAME}; use crate::client::{ create_client_config, list_client_types, list_models, ClientConfig, ExtraConfig, Message, Model, OpenAIClient, SendData, }; use crate::render::{MarkdownRender, RenderOptions}; use crate::utils::{get_env_name, light_theme_from_colorfgbg, now, render_prompt, set_text}; use anyhow::{anyhow, bail, Context, Result}; use inquire::{Confirm, Select, Text}; use is_terminal::IsTerminal; use parking_lot::RwLock; use serde::Deserialize; use std::collections::{HashMap, HashSet}; use std::{ env, fs::{create_dir_all, read_dir, read_to_string, remove_file, File, OpenOptions}, io::{stdout, Write}, path::{Path, PathBuf}, process::exit, sync::Arc, }; use syntect::highlighting::ThemeSet; /// Monokai Extended const DARK_THEME: &[u8] = include_bytes!("../../assets/monokai-extended.theme.bin"); const LIGHT_THEME: &[u8] = include_bytes!("../../assets/monokai-extended-light.theme.bin"); const CONFIG_FILE_NAME: &str = "config.yaml"; const ROLES_FILE_NAME: &str = "roles.yaml"; const MESSAGES_FILE_NAME: &str = "messages.md"; const SESSIONS_DIR_NAME: &str = "sessions"; const CLIENTS_FIELD: &str = "clients"; const SUMMARIZE_PROMPT: &str = "Summarize the discussion briefly in 200 words or less to use as a prompt for future context."; const SUMMARY_PROMPT: &str = "This is a summary of the chat history as a recap: "; const LEFT_PROMPT: &str = "{color.green}{?session {session}{?role /}}{role}{color.cyan}{?session )}{!session >}{color.reset} "; const RIGHT_PROMPT: &str = "{color.purple}{?session {?consume_tokens {consume_tokens}({consume_percent}%)}{!consume_tokens {consume_tokens}}}{color.reset}"; #[derive(Debug, Clone, Deserialize)] #[serde(default)] pub struct Config { #[serde(rename(serialize = "model", deserialize = "model"))] pub model_id: Option, pub temperature: Option, pub dry_run: bool, pub save: bool, pub save_session: Option, pub highlight: bool, pub light_theme: bool, pub wrap: Option, pub wrap_code: bool, pub auto_copy: bool, pub keybindings: Keybindings, pub prelude: Option, pub buffer_editor: Option, pub compress_threshold: usize, pub summarize_prompt: Option, pub summary_prompt: Option, pub left_prompt: Option, pub right_prompt: Option, pub clients: Vec, #[serde(skip)] pub roles: Vec, #[serde(skip)] pub role: Option, #[serde(skip)] pub session: Option, #[serde(skip)] pub model: Model, #[serde(skip)] pub working_mode: WorkingMode, #[serde(skip)] pub last_message: Option<(Input, String)>, } impl Default for Config { fn default() -> Self { Self { model_id: None, temperature: None, save: true, save_session: None, highlight: true, dry_run: false, light_theme: false, wrap: None, wrap_code: false, auto_copy: false, keybindings: Default::default(), prelude: None, buffer_editor: None, compress_threshold: 2000, summarize_prompt: None, summary_prompt: None, left_prompt: None, right_prompt: None, clients: vec![], roles: vec![], role: None, session: None, model: Default::default(), working_mode: WorkingMode::Command, last_message: None, } } } pub type GlobalConfig = Arc>; impl Config { pub fn init(working_mode: WorkingMode) -> Result { let config_path = Self::config_file()?; let api_key = env::var("OPENAI_API_KEY").ok(); let exist_config_path = config_path.exists(); if working_mode != WorkingMode::Command && api_key.is_none() && !exist_config_path { create_config_file(&config_path)?; } let mut config = if api_key.is_some() && !exist_config_path { Self::default() } else { Self::load_config(&config_path)? }; // Compatible with old configuration files if exist_config_path { config.compat_old_config(&config_path)?; } if let Some(wrap) = config.wrap.clone() { config.set_wrap(&wrap)?; } config.working_mode = working_mode; config.load_roles()?; config.setup_model()?; config.setup_highlight(); config.setup_light_theme()?; Ok(config) } pub fn apply_prelude(&mut self) -> Result<()> { let prelude = self.prelude.clone().unwrap_or_default(); if prelude.is_empty() { return Ok(()); } let err_msg = || format!("Invalid prelude '{}", prelude); match prelude.split_once(':') { Some(("role", name)) => { if self.role.is_none() && self.session.is_none() { self.set_role(name).with_context(err_msg)?; } } Some(("session", name)) => { if self.session.is_none() { self.start_session(Some(name)).with_context(err_msg)?; } } _ => { bail!("{}", err_msg()) } } Ok(()) } pub fn buffer_editor(&self) -> Option { self.buffer_editor .clone() .or_else(|| env::var("VISUAL").ok().or_else(|| env::var("EDITOR").ok())) } pub fn retrieve_role(&self, name: &str) -> Result { self.roles .iter() .find(|v| v.match_name(name)) .map(|v| { let mut role = v.clone(); role.complete_prompt_args(name); role }) .or_else(|| Role::find_system_role(name)) .ok_or_else(|| anyhow!("Unknown role `{name}`")) } pub fn config_dir() -> Result { let env_name = get_env_name("config_dir"); let path = if let Some(v) = env::var_os(env_name) { PathBuf::from(v) } else { let mut dir = dirs::config_dir().ok_or_else(|| anyhow!("Not found config dir"))?; dir.push(env!("CARGO_CRATE_NAME")); dir }; Ok(path) } pub fn local_path(name: &str) -> Result { let mut path = Self::config_dir()?; path.push(name); Ok(path) } pub fn save_message(&mut self, input: Input, output: &str) -> Result<()> { self.last_message = Some((input.clone(), output.to_string())); if self.dry_run { return Ok(()); } if let Some(session) = input.session_mut(&mut self.session) { session.add_message(&input, output)?; return Ok(()); } if !self.save { return Ok(()); } let mut file = self.open_message_file()?; if output.is_empty() || !self.save { return Ok(()); } let timestamp = now(); let summary = input.summary(); let input_markdown = input.render(); let output = match input.role() { None => { format!("# CHAT: {summary} [{timestamp}]\n{input_markdown}\n--------\n{output}\n--------\n\n",) } Some(v) => { format!( "# CHAT: {summary} [{timestamp}] ({})\n{input_markdown}\n--------\n{output}\n--------\n\n", v.name, ) } }; file.write_all(output.as_bytes()) .with_context(|| "Failed to save message") } pub fn maybe_copy(&self, text: &str) { if self.auto_copy { let _ = set_text(text); } } pub fn config_file() -> Result { Self::local_path(CONFIG_FILE_NAME) } pub fn roles_file() -> Result { let env_name = get_env_name("roles_file"); env::var(env_name).map_or_else( |_| Self::local_path(ROLES_FILE_NAME), |value| Ok(PathBuf::from(value)), ) } pub fn messages_file() -> Result { Self::local_path(MESSAGES_FILE_NAME) } pub fn sessions_dir() -> Result { Self::local_path(SESSIONS_DIR_NAME) } pub fn session_file(name: &str) -> Result { let mut path = Self::sessions_dir()?; path.push(&format!("{name}.yaml")); Ok(path) } pub fn set_prompt(&mut self, prompt: &str) -> Result<()> { let role = Role::temp(prompt); self.set_role_obj(role) } pub fn set_role(&mut self, name: &str) -> Result<()> { let role = self.retrieve_role(name)?; self.set_role_obj(role) } pub fn set_role_obj(&mut self, role: Role) -> Result<()> { if let Some(session) = self.session.as_mut() { session.guard_empty()?; session.set_temperature(role.temperature); } self.role = Some(role); Ok(()) } pub fn clear_role(&mut self) -> Result<()> { self.role = None; Ok(()) } pub fn get_state(&self) -> State { if let Some(session) = &self.session { if session.is_empty() { if self.role.is_some() { State::EmptySessionWithRole } else { State::EmptySession } } else { State::Session } } else if self.role.is_some() { State::Role } else { State::Normal } } pub fn set_temperature(&mut self, value: Option) { if let Some(session) = self.session.as_mut() { session.set_temperature(value); } else if let Some(role) = self.role.as_mut() { role.set_temperature(value); } else { self.temperature = value; } } pub fn set_save_session(&mut self, value: Option) { if let Some(session) = self.session.as_mut() { session.set_save_session(value); } else { self.save_session = value; } } pub fn set_compress_threshold(&mut self, value: Option) { if let Some(session) = self.session.as_mut() { session.set_compress_threshold(value); } else { self.compress_threshold = value.unwrap_or_default(); } } pub fn echo_messages(&self, input: &Input) -> String { if let Some(session) = input.session(&self.session) { session.echo_messages(input) } else if let Some(role) = input.role() { role.echo_messages(input) } else { input.render() } } pub fn build_messages(&self, input: &Input) -> Result> { let messages = if let Some(session) = input.session(&self.session) { session.build_emssages(input) } else if let Some(role) = input.role() { role.build_messages(input) } else { let message = Message::new(input); vec![message] }; Ok(messages) } pub fn set_wrap(&mut self, value: &str) -> Result<()> { if value == "no" { self.wrap = None; } else if value == "auto" { self.wrap = Some(value.into()); } else { value .parse::() .map_err(|_| anyhow!("Invalid wrap value"))?; self.wrap = Some(value.into()) } Ok(()) } pub fn set_model(&mut self, value: &str) -> Result<()> { let models = list_models(self); let model = Model::find(&models, value); match model { None => bail!("Invalid model '{}'", value), Some(model) => { if let Some(session) = self.session.as_mut() { session.set_model(model.clone())?; } self.model = model; Ok(()) } } } pub fn system_info(&self) -> Result { let display_path = |path: &Path| path.display().to_string(); let wrap = self .wrap .clone() .map_or_else(|| String::from("no"), |v| v.to_string()); let items = vec![ ("model", self.model.id()), ("temperature", format_option(&self.temperature)), ("dry_run", self.dry_run.to_string()), ("save", self.save.to_string()), ("save_session", format_option(&self.save_session)), ("highlight", self.highlight.to_string()), ("light_theme", self.light_theme.to_string()), ("wrap", wrap), ("wrap_code", self.wrap_code.to_string()), ("auto_copy", self.auto_copy.to_string()), ("keybindings", self.keybindings.stringify().into()), ("prelude", format_option(&self.prelude)), ("compress_threshold", self.compress_threshold.to_string()), ("config_file", display_path(&Self::config_file()?)), ("roles_file", display_path(&Self::roles_file()?)), ("messages_file", display_path(&Self::messages_file()?)), ("sessions_dir", display_path(&Self::sessions_dir()?)), ]; let output = items .iter() .map(|(name, value)| format!("{name:<20}{value}")) .collect::>() .join("\n"); Ok(output) } pub fn role_info(&self) -> Result { if let Some(role) = &self.role { role.export() } else { bail!("No role") } } pub fn session_info(&self) -> Result { if let Some(session) = &self.session { let render_options = self.get_render_options()?; let mut markdown_render = MarkdownRender::init(render_options)?; session.info(&mut markdown_render) } else { bail!("No session") } } pub fn info(&self) -> Result { if let Some(session) = &self.session { session.export() } else if let Some(role) = &self.role { role.export() } else { self.system_info() } } pub fn last_reply(&self) -> &str { self.last_message .as_ref() .map(|(_, reply)| reply.as_str()) .unwrap_or_default() } pub fn repl_complete(&self, cmd: &str, args: &[&str]) -> Vec { let (values, filter) = if args.len() == 1 { let values = match cmd { ".role" => self.roles.iter().map(|v| v.name.clone()).collect(), ".model" => list_models(self).into_iter().map(|v| v.id()).collect(), ".session" => self.list_sessions(), ".set" => vec![ "temperature ", "compress_threshold", "save ", "save_session ", "highlight ", "dry_run ", "auto_copy ", ] .into_iter() .map(|v| v.to_string()) .collect(), _ => vec![], }; (values, args[0]) } else if args.len() == 2 { let values = match args[0] { "save" => complete_bool(self.save), "save_session" => { let save_session = if let Some(session) = &self.session { session.save_session() } else { self.save_session }; complete_option_bool(save_session) } "highlight" => complete_bool(self.highlight), "dry_run" => complete_bool(self.dry_run), "auto_copy" => complete_bool(self.auto_copy), _ => vec![], }; (values, args[1]) } else { return vec![]; }; values .into_iter() .filter(|v| v.starts_with(filter)) .collect() } pub fn update(&mut self, data: &str) -> Result<()> { let parts: Vec<&str> = data.split_whitespace().collect(); if parts.len() != 2 { bail!("Usage: .set . If value is null, unset key."); } let key = parts[0]; let value = parts[1]; match key { "temperature" => { let value = parse_value(value)?; self.set_temperature(value); } "compress_threshold" => { let value = parse_value(value)?; self.set_compress_threshold(value); } "save" => { let value = value.parse().with_context(|| "Invalid value")?; self.save = value; } "save_session" => { let value = parse_value(value)?; self.set_save_session(value); } "highlight" => { let value = value.parse().with_context(|| "Invalid value")?; self.highlight = value; } "dry_run" => { let value = value.parse().with_context(|| "Invalid value")?; self.dry_run = value; } "auto_copy" => { let value = value.parse().with_context(|| "Invalid value")?; self.auto_copy = value; } _ => bail!("Unknown key `{key}`"), } Ok(()) } pub fn start_session(&mut self, session: Option<&str>) -> Result<()> { if self.session.is_some() { bail!( "Already in a session, please run '.exit session' first to exit the current session." ); } match session { None => { let session_file = Self::session_file(TEMP_SESSION_NAME)?; if session_file.exists() { remove_file(session_file).with_context(|| { format!("Failed to cleanup previous '{TEMP_SESSION_NAME}' session") })?; } let session = Session::new(self, TEMP_SESSION_NAME); self.session = Some(session); } Some(name) => { let session_path = Self::session_file(name)?; if !session_path.exists() { self.session = Some(Session::new(self, name)); } else { let session = Session::load(name, &session_path)?; let model_id = session.model().to_string(); self.session = Some(session); self.set_model(&model_id)?; } } } if let Some(session) = self.session.as_mut() { if session.is_empty() { if let Some((input, output)) = &self.last_message { let ans = Confirm::new( "Start a session that incorporates the last question and answer?", ) .with_default(false) .prompt()?; if ans { session.add_message(input, output)?; } } } } Ok(()) } pub fn end_session(&mut self) -> Result<()> { if let Some(mut session) = self.session.take() { self.last_message = None; let save_session = session.save_session(); if session.dirty && save_session != Some(false) { if save_session.is_none() || session.is_temp() { if self.working_mode != WorkingMode::Repl { return Ok(()); } let ans = Confirm::new("Save session?").with_default(false).prompt()?; if !ans { return Ok(()); } while session.is_temp() || session.name().is_empty() { session.name = Text::new("Session name:").prompt()?; } } Self::save_session_to_file(&mut session)?; } } Ok(()) } pub fn save_session(&mut self, name: &str) -> Result<()> { if let Some(session) = self.session.as_mut() { if !name.is_empty() { session.name = name.to_string(); } Self::save_session_to_file(session)?; } Ok(()) } pub fn has_session(&self) -> bool { self.session.is_some() } pub fn clear_session_messages(&mut self) -> Result<()> { if let Some(session) = self.session.as_mut() { session.clear_messages(); } Ok(()) } pub fn list_sessions(&self) -> Vec { let sessions_dir = match Self::sessions_dir() { Ok(dir) => dir, Err(_) => return vec![], }; match read_dir(sessions_dir) { Ok(rd) => { let mut names = vec![]; for entry in rd.flatten() { let name = entry.file_name(); if let Some(name) = name.to_string_lossy().strip_suffix(".yaml") { names.push(name.to_string()); } } names.sort_unstable(); names } Err(_) => vec![], } } pub fn should_compress_session(&mut self) -> bool { if let Some(session) = self.session.as_mut() { if session.need_compress(self.compress_threshold) { session.compressing = true; return true; } } false } pub fn compress_session(&mut self, summary: &str) { if let Some(session) = self.session.as_mut() { let summary_prompt = self.summary_prompt.as_deref().unwrap_or(SUMMARY_PROMPT); session.compress(format!("{}{}", summary_prompt, summary)); } } pub fn summarize_prompt(&self) -> &str { self.summarize_prompt.as_deref().unwrap_or(SUMMARIZE_PROMPT) } pub fn is_compressing_session(&self) -> bool { self.session .as_ref() .map(|v| v.compressing) .unwrap_or_default() } pub fn end_compressing_session(&mut self) { if let Some(session) = self.session.as_mut() { session.compressing = false; } } pub fn get_render_options(&self) -> Result { let theme = if self.highlight { let theme_mode = if self.light_theme { "light" } else { "dark" }; let theme_filename = format!("{theme_mode}.tmTheme"); let theme_path = Self::local_path(&theme_filename)?; if theme_path.exists() { let theme = ThemeSet::get_theme(&theme_path) .with_context(|| format!("Invalid theme at {}", theme_path.display()))?; Some(theme) } else { let theme = if self.light_theme { bincode::deserialize_from(LIGHT_THEME).expect("Invalid builtin light theme") } else { bincode::deserialize_from(DARK_THEME).expect("Invalid builtin dark theme") }; Some(theme) } } else { None }; let wrap = if stdout().is_terminal() { self.wrap.clone() } else { None }; let truecolor = matches!( env::var("COLORTERM").as_ref().map(|v| v.as_str()), Ok("truecolor") ); Ok(RenderOptions::new(theme, wrap, self.wrap_code, truecolor)) } pub fn render_prompt_left(&self) -> String { let variables = self.generate_prompt_context(); let left_prompt = self.left_prompt.as_deref().unwrap_or(LEFT_PROMPT); render_prompt(left_prompt, &variables) } pub fn render_prompt_right(&self) -> String { let variables = self.generate_prompt_context(); let right_prompt = self.right_prompt.as_deref().unwrap_or(RIGHT_PROMPT); render_prompt(right_prompt, &variables) } pub fn prepare_send_data(&self, input: &Input, stream: bool) -> Result { let messages = self.build_messages(input)?; let temperature = if let Some(session) = input.session(&self.session) { session.temperature() } else if let Some(role) = input.role() { role.temperature } else { self.temperature }; self.model.max_input_tokens_limit(&messages)?; Ok(SendData { messages, temperature, stream, }) } pub fn input_context(&self) -> InputContext { InputContext::new(self.role.clone(), self.has_session()) } pub fn maybe_print_send_tokens(&self, input: &Input) { if self.dry_run { if let Ok(messages) = self.build_messages(input) { let tokens = self.model.total_tokens(&messages); println!(">>> This message consumes {tokens} tokens. <<<"); } } } fn generate_prompt_context(&self) -> HashMap<&str, String> { let mut output = HashMap::new(); output.insert("model", self.model.id()); output.insert("client_name", self.model.client_name.clone()); output.insert("model_name", self.model.name.clone()); output.insert( "max_input_tokens", self.model.max_input_tokens.unwrap_or_default().to_string(), ); if let Some(temperature) = self.temperature { if temperature != 0.0 { output.insert("temperature", temperature.to_string()); } } if self.dry_run { output.insert("dry_run", "true".to_string()); } if self.save { output.insert("save", "true".to_string()); } if let Some(wrap) = &self.wrap { if wrap != "no" { output.insert("wrap", wrap.clone()); } } if self.auto_copy { output.insert("auto_copy", "true".to_string()); } if let Some(role) = &self.role { output.insert("role", role.name.clone()); } if let Some(session) = &self.session { output.insert("session", session.name().to_string()); output.insert("dirty", session.dirty.to_string()); let (tokens, percent) = session.tokens_and_percent(); output.insert("consume_tokens", tokens.to_string()); output.insert("consume_percent", percent.to_string()); output.insert("user_messages_len", session.user_messages_len().to_string()); } if self.highlight { output.insert("color.reset", "\u{1b}[0m".to_string()); output.insert("color.black", "\u{1b}[30m".to_string()); output.insert("color.dark_gray", "\u{1b}[90m".to_string()); output.insert("color.red", "\u{1b}[31m".to_string()); output.insert("color.light_red", "\u{1b}[91m".to_string()); output.insert("color.green", "\u{1b}[32m".to_string()); output.insert("color.light_green", "\u{1b}[92m".to_string()); output.insert("color.yellow", "\u{1b}[33m".to_string()); output.insert("color.light_yellow", "\u{1b}[93m".to_string()); output.insert("color.blue", "\u{1b}[34m".to_string()); output.insert("color.light_blue", "\u{1b}[94m".to_string()); output.insert("color.purple", "\u{1b}[35m".to_string()); output.insert("color.light_purple", "\u{1b}[95m".to_string()); output.insert("color.magenta", "\u{1b}[35m".to_string()); output.insert("color.light_magenta", "\u{1b}[95m".to_string()); output.insert("color.cyan", "\u{1b}[36m".to_string()); output.insert("color.light_cyan", "\u{1b}[96m".to_string()); output.insert("color.white", "\u{1b}[37m".to_string()); output.insert("color.light_gray", "\u{1b}[97m".to_string()); } output } fn open_message_file(&self) -> Result { let path = Self::messages_file()?; ensure_parent_exists(&path)?; OpenOptions::new() .create(true) .append(true) .open(&path) .with_context(|| format!("Failed to create/append {}", path.display())) } fn save_session_to_file(session: &mut Session) -> Result<()> { let session_path = Self::session_file(session.name())?; let sessions_dir = session_path .parent() .ok_or_else(|| anyhow!("Unable to save session file to {}", session_path.display()))?; if !sessions_dir.exists() { create_dir_all(sessions_dir).with_context(|| { format!("Failed to create session_dir '{}'", sessions_dir.display()) })?; } session.save(&session_path)?; Ok(()) } fn load_config(config_path: &Path) -> Result { let ctx = || format!("Failed to load config at {}", config_path.display()); let content = read_to_string(config_path).with_context(ctx)?; let config: Self = serde_yaml::from_str(&content) .map_err(|err| { let err_msg = err.to_string(); if err_msg.starts_with(&format!("{}: ", CLIENTS_FIELD)) { anyhow!("clients: invalid value") } else { anyhow!("{err_msg}") } }) .with_context(ctx)?; Ok(config) } fn load_roles(&mut self) -> Result<()> { let path = Self::roles_file()?; if !path.exists() { return Ok(()); } let content = read_to_string(&path) .with_context(|| format!("Failed to load roles at {}", path.display()))?; let roles: Vec = serde_yaml::from_str(&content).with_context(|| "Invalid roles config")?; self.roles = roles; Ok(()) } fn setup_model(&mut self) -> Result<()> { let model = match &self.model_id { Some(v) => v.clone(), None => { let models = list_models(self); if models.is_empty() { bail!("No available model"); } models[0].id() } }; self.set_model(&model)?; Ok(()) } fn setup_highlight(&mut self) { if let Ok(value) = env::var("NO_COLOR") { let mut no_color = false; set_bool(&mut no_color, &value); if no_color { self.highlight = false; } } } fn setup_light_theme(&mut self) -> Result<()> { if self.light_theme { return Ok(()); } if let Ok(value) = env::var(get_env_name("light_theme")) { set_bool(&mut self.light_theme, &value); return Ok(()); } else if let Ok(value) = env::var("COLORFGBG") { if let Some(light) = light_theme_from_colorfgbg(&value) { self.light_theme = light } }; Ok(()) } fn compat_old_config(&mut self, config_path: &PathBuf) -> Result<()> { let content = read_to_string(config_path)?; let value: serde_json::Value = serde_yaml::from_str(&content)?; if value.get(CLIENTS_FIELD).is_some() { return Ok(()); } if let Some(model_name) = value.get("model").and_then(|v| v.as_str()) { if model_name.starts_with("gpt") { self.model_id = Some(format!("{}:{}", OpenAIClient::NAME, model_name)); } } if let Some(ClientConfig::OpenAIConfig(client_config)) = self.clients.first_mut() { if let Some(api_key) = value.get("api_key").and_then(|v| v.as_str()) { client_config.api_key = Some(api_key.to_string()) } if let Some(organization_id) = value.get("organization_id").and_then(|v| v.as_str()) { client_config.organization_id = Some(organization_id.to_string()) } let mut extra_config = ExtraConfig::default(); if let Some(proxy) = value.get("proxy").and_then(|v| v.as_str()) { extra_config.proxy = Some(proxy.to_string()) } if let Some(connect_timeout) = value.get("connect_timeout").and_then(|v| v.as_i64()) { extra_config.connect_timeout = Some(connect_timeout as _) } client_config.extra = Some(extra_config); } Ok(()) } } #[derive(Debug, Clone, Deserialize, Default)] pub enum Keybindings { #[serde(rename = "emacs")] #[default] Emacs, #[serde(rename = "vi")] Vi, } impl Keybindings { pub fn is_vi(&self) -> bool { matches!(self, Keybindings::Vi) } pub fn stringify(&self) -> &str { match self { Keybindings::Emacs => "emacs", Keybindings::Vi => "vi", } } } #[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)] pub enum WorkingMode { Command, Repl, Serve, } #[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)] pub enum State { Normal, Role, EmptySession, EmptySessionWithRole, Session, } impl State { pub fn all() -> Vec { vec![ Self::Normal, Self::Role, Self::EmptySession, Self::EmptySessionWithRole, Self::Session, ] } pub fn in_session() -> Vec { vec![ Self::EmptySession, Self::EmptySessionWithRole, Self::Session, ] } pub fn not_in_session() -> Vec { let excludes: HashSet<_> = Self::in_session().into_iter().collect(); Self::all() .into_iter() .filter(|v| !excludes.contains(v)) .collect() } pub fn unable_change_role() -> Vec { vec![Self::Session] } pub fn able_change_role() -> Vec { let excludes: HashSet<_> = Self::unable_change_role().into_iter().collect(); Self::all() .into_iter() .filter(|v| !excludes.contains(v)) .collect() } pub fn in_role() -> Vec { vec![Self::Role, Self::EmptySessionWithRole] } } fn create_config_file(config_path: &Path) -> Result<()> { let ans = Confirm::new("No config file, create a new one?") .with_default(true) .prompt()?; if !ans { exit(0); } let client = Select::new("Platform:", list_client_types()).prompt()?; let mut config = serde_json::json!({}); let (model, clients_config) = create_client_config(client)?; config["model"] = model.into(); config[CLIENTS_FIELD] = clients_config; let config_data = serde_yaml::to_string(&config).with_context(|| "Failed to create config")?; ensure_parent_exists(config_path)?; std::fs::write(config_path, config_data).with_context(|| "Failed to write to config file")?; #[cfg(unix)] { use std::os::unix::prelude::PermissionsExt; let perms = std::fs::Permissions::from_mode(0o600); std::fs::set_permissions(config_path, perms)?; } println!("✨ Saved config file to {}\n", config_path.display()); Ok(()) } fn ensure_parent_exists(path: &Path) -> Result<()> { if path.exists() { return Ok(()); } let parent = path .parent() .ok_or_else(|| anyhow!("Failed to write to {}, No parent path", path.display()))?; if !parent.exists() { create_dir_all(parent).with_context(|| { format!( "Failed to write {}, Cannot create parent directory", path.display() ) })?; } Ok(()) } fn set_bool(target: &mut bool, value: &str) { match value { "1" | "true" => *target = true, "0" | "false" => *target = false, _ => {} } } fn parse_value(value: &str) -> Result> where T: std::str::FromStr, { let value = if value == "null" { None } else { let value = match value.parse() { Ok(value) => value, Err(_) => bail!("Invalid value '{}'", value), }; Some(value) }; Ok(value) } fn format_option(value: &Option) -> String where T: std::fmt::Display, { match value { Some(value) => value.to_string(), None => "-".to_string(), } } fn complete_bool(value: bool) -> Vec { vec![(!value).to_string()] } fn complete_option_bool(value: Option) -> Vec { match value { Some(true) => vec!["false".to_string(), "null".to_string()], Some(false) => vec!["true".to_string(), "null".to_string()], None => vec!["true".to_string(), "false".to_string()], } }