From aba802e77d094936b2bee8a453cd2266da0fa9e1 Mon Sep 17 00:00:00 2001 From: Valerie Date: Wed, 6 Mar 2024 15:54:41 -0500 Subject: [PATCH] upgraded tmux_interface to 0.3.2 and restructured command module --- Cargo.toml | 4 +- src/command.rs | 170 ------------------------------------------- src/command/mod.rs | 4 + src/command/share.rs | 170 +++++++++++++++++++++++++++++++++++++++++++ src/flag.rs | 11 +++ src/main.rs | 17 +++-- src/util.rs | 25 ++++--- 7 files changed, 209 insertions(+), 192 deletions(-) delete mode 100644 src/command.rs create mode 100644 src/command/mod.rs create mode 100644 src/command/share.rs create mode 100644 src/flag.rs diff --git a/Cargo.toml b/Cargo.toml index baafae3..6d24041 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -1,6 +1,6 @@ [package] name = "remux" -version = "0.2.0" +version = "0.3.0" edition = "2021" authors = [ "Valerie Wolfe " ] description = "A friendly command shortener for tmux" @@ -15,7 +15,7 @@ path = "src/main.rs" [dependencies] pico-args = { version = "0.5.0", features = [ "combined-flags", "eq-separator" ] } termion = "2.0.1" -tmux_interface = "0.2.1" +tmux_interface = "0.3.2" [profile.release] opt-level = 's' diff --git a/src/command.rs b/src/command.rs deleted file mode 100644 index 984f857..0000000 --- a/src/command.rs +++ /dev/null @@ -1,170 +0,0 @@ -use std::{ - env::current_dir, - ffi::OsString, - process::exit -}; - -use pico_args::Arguments; -use termion::{ color, style }; -use tmux_interface::TmuxCommand; - -use crate::error; -use crate::util; - -pub fn attach(pargs: &mut Arguments) { - util::prevent_nest(); - - // get optional flags - let read_only = pargs.contains(["-r", "--readonly"]); - let detach_other = pargs.contains(["-d", "--detach"]); - - // collect target and window arguments - let args = pargs.clone().finish(); - let target: String; - let window: Option<&OsString>; - if args.len() < 1 { - // missing name will attempt to fall back to repository - target = util::repo_fallback(); - if !util::session_exists(target.clone()) { error::missing_target(); } - window = None; - } else { - target = args.get(0).unwrap().to_string_lossy().to_string(); - window = args.get(1); - } - - // focus window if provided - if window.is_some() { - let target = window.unwrap().to_string_lossy(); - TmuxCommand::new() - .select_window() - .target_window(target) - .output().ok(); - } - - // make sure the target session exists - let exists = util::session_exists(target.clone()); - if !exists { error::no_target(target.clone()); } - - // build command - let mut attach = TmuxCommand::new().attach_session(); - - // handle optional flags - if read_only { attach.read_only(); } - if detach_other { attach.detach_other(); } - - // run command - attach - .target_session(target) - .output().ok(); -} - -pub fn detach(pargs: &mut Arguments) { - // get target and error out if not provided - let args = pargs.clone().finish(); - if args.len() < 1 { error::missing_target(); } - let target = args.get(0).unwrap().to_string_lossy(); - - // make sure the target session exists - let exists = util::session_exists(target.clone()); - if !exists { error::no_target(target.clone()); } - - // build command and run - TmuxCommand::new() - .detach_client() - .target_session(target) - .output().ok(); -} - -pub fn has(pargs: &mut Arguments) { - // get optional flag - let quiet = pargs.contains(["-q", "--quiet"]); - - // collect target argument - let args = pargs.clone().finish(); - let target: String; - if args.len() < 1 { - // missing name will attempt to fall back to repository - target = util::repo_fallback(); - } else { - target = args.get(0).unwrap().to_string_lossy().to_string(); - } - - // run command - let success = util::session_exists(target.clone()); - - // handle optional flag - // inverted; print text if NOT quiet - if !quiet { println!("session \"{target}\" {}.", if success { "exists" } else { "does not exist" }); } - - // exit codes for scripts to use - exit( if success { 0 } else { 1 }); -} - -pub fn list() { - // get session list - let sessions = util::get_sessions().unwrap_or(Vec::new()); - - // handle empty case - if sessions.len() == 0 { - println!("no sessions"); - return; - } - - // iterate over pretty print - println!("sessions:"); - for session in sessions.into_iter() { - let group = session.group.unwrap_or("[untitled]".to_string()); - let id = session.id.unwrap(); - let attached = session.attached.unwrap_or(0) > 0; - - println!( - " {group} ({bold}{blue}{id}{reset}) {bold}{green}{attach_sym}{reset}", - // values - attach_sym = if attached { "󰌹" } else {""}, - // formatting - bold = style::Bold, - blue = color::Fg(color::Blue), - green = color::Fg(color::LightGreen), - reset = style::Reset - ); - } -} - -pub fn new(pargs: &mut Arguments) { - use pico_args::Error; - - // show nest message - util::prevent_nest(); - - // get optional flag - let target_dir: Result = pargs.value_from_str(["-t", "--target"]); - - // get target and error out if not provided - let args = pargs.clone().finish(); - - // collect name and command arguments - let title: String; - let command: Option<&OsString>; - if args.len() < 1 { - // missing name will attempt to fall back to repository - title = util::repo_fallback(); - command = None; - } else { - title = args.get(0).unwrap().to_string_lossy().to_string(); - command = args.get(1); - } - - // build command - let mut new = TmuxCommand::new().new_session(); - - // handle shell command argument - if command.is_some() { new.shell_command(command.unwrap().to_string_lossy()); } - - // run command - new - .group_name(title) - .attach() - .start_directory(target_dir.unwrap_or(current_dir().unwrap().to_string_lossy().to_string())) - .output().ok(); -} - diff --git a/src/command/mod.rs b/src/command/mod.rs new file mode 100644 index 0000000..8b5d2e6 --- /dev/null +++ b/src/command/mod.rs @@ -0,0 +1,4 @@ + +pub mod share; +//pub mod session; + diff --git a/src/command/share.rs b/src/command/share.rs new file mode 100644 index 0000000..0de817b --- /dev/null +++ b/src/command/share.rs @@ -0,0 +1,170 @@ +//! globally available tmux commands. +use std::{ + ffi::OsString, + process::exit +}; + +use pico_args::{ Arguments, Error }; +use termion::{ color, style }; +use tmux_interface::{ + Tmux, + commands +}; + +use crate::{ error, flag, util }; + +pub fn attach(pargs: &mut Arguments) { + // don't allow unflagged nests + util::prevent_nest(); + + // consume optional flags + let read_only = pargs.contains(flag::READ_ONLY); + let detach_other = pargs.contains(flag::DETACHED); + + let args = pargs.clone().finish(); + let target: String; + let window: Option<&OsString>; + if args.len() < 1 { + // missing name will attempt to fall back to repository + target = util::repo_fallback(); + if !util::session_exists(target.clone()) { error::missing_target(); } + window = None; + } else { + target = args.get(0).unwrap().to_string_lossy().to_string(); + window = args.get(1); + } + + // make sure the session exists + let exists = util::session_exists(target.clone()); + if !exists { error::no_target(target.clone()); } + + // build attach command + let mut attach = commands::AttachSession::new(); + attach = attach.target_session(target); + if read_only { attach.read_only = true; } + if detach_other { attach.detach_other = true; } + + let select_window: Option; + if let Some(window) = window { + let mut command = commands::SelectWindow::new(); + command.target_window = Some(window.to_string_lossy()); + select_window = Some(command); + } else { select_window = None; } + + // build dispatch + let mut tmux = Tmux::new().add_command(attach); + if let Some(select_window) = select_window { tmux = tmux.add_command(select_window); } + tmux.output().ok(); +} + +pub fn detach(pargs: &mut Arguments) { + // get target or fallback + let args = pargs.clone().finish(); + let target: String; + if args.len() < 1 { + target = util::repo_fallback(); + } else { + target = args.get(0).unwrap().to_string_lossy().to_string(); + } + + // make sure the session exists + let exists = util::session_exists(target.clone()); + if !exists { error::no_target(target.clone()); } + + // build and dispatch + let detach = commands::DetachClient::new() + .target_session(target); + Tmux::new() + .add_command(detach) + .output().ok(); +} + +pub fn has(pargs: &mut Arguments) { + // consume optional flags + let quiet = pargs.contains(flag::QUIET); + + // get target or fallback + let args = pargs.clone().finish(); + let target: String; + if args.len() < 1 { + target = util::repo_fallback(); + } else { + target = args.get(0).unwrap().to_string_lossy().to_string(); + } + + // run command + let success = util::session_exists(target.clone()); + + // print if not quiet + if !quiet { + println!("session \"{target}\" {}.", + if success { "exists" } + else { "does not exist" } + ); + } + + // emit exit code + exit( if success { 0 } else { 1 }); +} + +pub fn list() { + // get session list + let sessions = util::get_sessions().unwrap_or(Vec::new()); + + // handle empty case + if sessions.len() == 0 { + println!("no sessions"); + return; + } + + // pretty print session list + println!("sessions:"); + for session in sessions.into_iter() { + let group = session.group.unwrap_or("[untitled]".to_string()); + let id = session.id.unwrap(); + let attached = session.attached.unwrap_or(0) > 0; + + println!( + " {group} ({bold}{blue}{id}{reset}) {bold}{green}{attach_sym}{reset}", + // value + attach_sym = if attached { "\u{F0339}" } else { "" }, + // formatting + bold = style::Bold, + blue = color::Fg(color::Blue), + green = color::Fg(color::LightGreen), + reset = style::Reset + ); + } +} + +pub fn new(pargs: &mut Arguments) { + // don't allow unflagged nesting + util::prevent_nest(); + + // get optional flag + let target_dir: Result = pargs.value_from_str(flag::TARGET); + + // get target or fallback + let args = pargs.clone().finish(); + let title: String; + let command: Option<&OsString>; + if args.len() < 1 { + // attempt repo fallback + title = util::repo_fallback(); + command = None; + } else { + title = args.get(0).unwrap().to_string_lossy().to_string(); + command = args.get(1); + } + + let mut new = commands::NewSession::new(); + new = new.group_name(title); + if let Some(command) = command { new.shell_command = Some(command.to_string_lossy()); } + if let Ok(target_dir) = target_dir { new = new.start_directory(target_dir); } + + Tmux::new() + .add_command(new) + + .output().ok(); +} + diff --git a/src/flag.rs b/src/flag.rs new file mode 100644 index 0000000..92056f3 --- /dev/null +++ b/src/flag.rs @@ -0,0 +1,11 @@ + +type Flag = [&'static str;2]; + +pub static DETACHED: Flag = ["-d", "--detached"]; +pub static HELP: Flag = ["-h", "--help"]; +pub static NEST: Flag = ["-n", "--nest"]; +pub static QUIET: Flag = ["-q", "--quiet"]; +pub static READ_ONLY: Flag = ["-r", "--read-only"]; +pub static TARGET: Flag = ["-t", "--target"]; +pub static VERSION: Flag = ["-v", "--version"]; + diff --git a/src/main.rs b/src/main.rs index 4757ee0..4276941 100644 --- a/src/main.rs +++ b/src/main.rs @@ -7,6 +7,7 @@ use pico_args::Arguments; mod command; mod error; +mod flag; mod help; mod util; @@ -19,17 +20,17 @@ fn main() { let mut args = Arguments::from_env(); // consume flags - if args.contains(["-h", "--help"]) { + if args.contains(flag::HELP) { help(&mut args); return; } - if args.contains(["-v", "--version"]) { + if args.contains(flag::VERSION) { version(); return; } - let nesting = args.contains(["-n", "--nest"]); + let nesting = args.contains(flag::NEST); let tmux_var = var("TMUX").ok(); if nesting { if tmux_var.is_none() { @@ -48,20 +49,20 @@ fn main() { => help(&mut args), Some("a" | "attach") - => command::attach(&mut args), + => command::share::attach(&mut args), Some("d" | "detach") - => command::detach(&mut args), + => command::share::detach(&mut args), Some("has") - => command::has(&mut args), + => command::share::has(&mut args), None | Some("l" | "ls" | "list") - => command::list(), + => command::share::list(), Some("n" | "new") - => command::new(&mut args), + => command::share::new(&mut args), _ => error::no_subcommand(subcommand.unwrap()) diff --git a/src/util.rs b/src/util.rs index e5ced2f..c99302b 100644 --- a/src/util.rs +++ b/src/util.rs @@ -5,20 +5,20 @@ use std::{ }; use tmux_interface::{ - Session, Sessions, TmuxCommand, - variables::session::session::SESSION_ALL + Session, Tmux, + + commands, + variables::session::SessionsCtl }; use crate::error; /// return a Vec of all sessions or None pub fn get_sessions() -> Option> { - let i_sessions = Sessions::get(SESSION_ALL); - if i_sessions.is_err() { return None; } - let sessions = i_sessions.ok(); - if sessions.is_none() { return None; } - - Some(sessions.unwrap().0) + let sessions = SessionsCtl::new().get_all(); + if let Ok(sessions) = sessions { + return Some(sessions.0); + } else { return None; } } /// show the tmux nest text if env var is not unset @@ -32,10 +32,11 @@ pub fn prevent_nest() { /// check whether a target session exists pub fn session_exists>(target: S) -> bool { - TmuxCommand::new() - .has_session() - .target_session(target.into()) - .output().unwrap() + let has_session = commands::HasSession::new() + .target_session(target.into()); + Tmux::new().add_command(has_session) + .status() + .unwrap() .success() }