Compare commits

..

3 commits
main ... rc

13 changed files with 179 additions and 195 deletions

View file

@ -1,6 +1,6 @@
[package]
name = "remux"
version = "0.4.0"
version = "0.3.6"
edition = "2021"
authors = [ "Valerie Wolfe <sleeplessval@gmail.com>" ]
description = "A friendly command shortener for tmux"

View file

@ -92,16 +92,10 @@ using an AUR package manager such as <a href="https://github.com/Morganamilo/par
Install the package using Cargo with the command <code>cargo install tmux-remux</code>.
</details>
### Supplemental
### Man Page
<details>
<summary>Bash Completions</summary>
Copy <code>bash-completion/remux</code> to the appropriate directory, typically
<code>/usr/share/bash-completion</code>.
</details>
<details>
<summary>Man Page: Section 1</summary>
<summary>Section 1</summary>
Copy <code>man/remux.1</code> into <code>/usr/share/man/man1/</code>.
</details>

View file

@ -1,19 +0,0 @@
_remux() {
local word
COMPREPLY=()
word="${COMP_WORDS[COMP_CWORD]}"
case $COMP_CWORD in
1)
COMPREPLY=( `compgen -W 'attach detach has help list new path switch title' -- "$word"` )
;;
2)
COMPREPLY=( `compgen -W "$(remux l -q $word)"` )
;;
esac
return 0
}
complete -F _remux remux

View file

@ -1,5 +1,6 @@
.Dd $Mdocdate$
.Dt REMUX 1
.Os
.Sh NAME
.Nm remux
.Nd a command shortener for
@ -7,7 +8,6 @@
.Sh SYNOPSIS
.Nm remux
.Op Fl dhnqrtv
.Op Fl D Ar path
.Op Ar command
.Op args...
.Sh DESCRIPTION
@ -35,8 +35,6 @@ Attaches to an existing session.
.Bl -tag -width Ds -compact
.It Fl d , Fl -detach
Detach all other connections to the session.
.It Fl D , Fl -dir Ar path
Sets the working directory for the given command.
.It Fl n , Fl -nest
Allow nesting (attaching a session from inside another session).
.It Fl r , Fl -read-only
@ -103,8 +101,8 @@ aliases: p
Prints the session path.
.Ed
.It Xo Ic switch
.Op Fl rd
.Op Ar title
.Op Fl r , Fl -read-only
.Ar title
.Xc
.Bd -literal -compact
aliases: s
@ -112,12 +110,10 @@ Switches from the current session to the target.
.Ed
.Pp
.Bl -tag -width Ds -compact
.It Fl d , Fl -detach
Detaches other clients from the target session.
.It Fl r , Fl -read-only
Switch to the target session in read-only mode.
.It Ar title
The title of the session to switch to. If blank, the previous session will be used.
The title of the session to switch to.
.El
.It Ic title
.Bd -literal -compact
@ -139,14 +135,6 @@ Default: '>'
.It Ev REMUX_NEW_WINDOW
Provides a default windows name when creating a new session. Unused if empty.
Default: (unset)
.It Ev REMUX_PREVIOUS_SYMBOL
Changes the symbol displayed for the previous session in the
.Ic list
command.
Default: '-'
.It Ev REMUX_REPO_FILE
The filename to match on when trying to find the root of a repository.
Default: '.git'
.El
.Sh EXIT STATUS
.Bl -tag -Width Ds

View file

@ -1,5 +1,6 @@
//! commands accessible from within a session
use termion::{ color, style };
use tmux_interface::{
Tmux,
commands
@ -7,19 +8,44 @@ use tmux_interface::{
use crate::{
error,
script,
state::State,
util::{
self,
message,
MSG_PREVIOUS, MSG_SESSION_PATH, NULL
}
util::{ self, NULL }
};
pub fn path(state: &mut State) {
state.session_enforce("path");
if let Some(message) = message(MSG_SESSION_PATH) {
println!("{message}");
let message = commands::DisplayMessage::new().print().message("#{session_path}");
let result = Tmux::new().add_command(message).output().unwrap();
let text = String::from_utf8(result.0.stdout);
if let Ok(output) = text {
// trim the trailing line break
let target = output.len() - 1;
println!("{}", &output[0..target]);
}
}
pub fn source(state: &mut State) {
if let Some(scripts) = script::list() {
if let Some(name) = state.target() {
if let Some(path) = scripts.get(&name) {
let rc = commands::SourceFile::new()
.path(path);
Tmux::new().add_command(rc).output().ok();
}
} else {
for (name, path) in scripts.into_iter() {
println!(
"{bold}{name}{reset} ({yellow}{path:?}{reset})",
bold = style::Bold,
yellow = color::Fg(color::LightYellow),
reset = style::Reset
);
}
}
}
}
@ -30,33 +56,21 @@ pub fn switch(state: &mut State) {
// consume optional flags
let read_only = state.flags.read_only;
let detach_other = state.flags.detached;
//TODO: -d flag handling needs to be done manually
let args = state.args.clone().finish();
let target: String = match if let Some(inner) = args.get(0) { inner.to_str() } else { None } {
None |
Some("-") => if let Some(prev) = message(MSG_PREVIOUS) { prev }
else { error::missing_target() },
Some(inner) => inner.to_owned()
};
if args.len() < 1 { error::missing_target(); }
let target = args.get(0).unwrap().to_string_lossy().to_string();
let exists = util::session_exists(target.clone());
if !exists { error::no_target(target.clone()); }
let mut tmux = Tmux::new();
if detach_other {
let detach = commands::DetachClient::new()
.target_session(&target);
tmux = tmux.add_command(detach);
}
let mut switch = commands::SwitchClient::new();
switch = switch.target_session(&target);
switch = switch.target_session(target);
if read_only { switch.read_only = true; }
tmux.add_command(switch)
Tmux::new()
.add_command(switch)
.stderr(NULL).output().ok();
}

View file

@ -9,19 +9,12 @@ use tmux_interface::{
};
use crate::{
env::{
self,
env_var,
SYMBOL_ATTACH, SYMBOL_CURRENT, SYMBOL_PREV
},
env::{ self, env_var },
error,
flag,
script,
state::State,
util::{
self,
message,
MSG_PREVIOUS, NULL
}
util::{ self, NULL }
};
pub fn attach(state: &mut State) {
@ -64,7 +57,7 @@ pub fn attach(state: &mut State) {
state.nest_deinit();
}
pub fn context_action(state: &mut State) {
pub fn context_action(state: &State) {
if !state.session {
if let Some(repository) = &state.repository {
let target = repository.name.clone();
@ -80,7 +73,7 @@ pub fn context_action(state: &mut State) {
}
}
// fallback behavior is list
list(state);
list(&state);
}
pub fn detach(state: &mut State) {
@ -122,55 +115,40 @@ pub fn has(state: &mut State) {
exit( if success { 0 } else { 1 });
}
pub fn list(state: &mut State) {
pub fn list(state: &State) {
// get session list
let sessions = util::get_sessions().unwrap_or(Vec::new());
let search = state.target();
let previous = message(MSG_PREVIOUS);
// handle empty case
if sessions.len() == 0 {
if !state.flags.quiet { println!("no sessions"); }
println!("no sessions");
return;
}
// get attached session symbol
let attach_symbol = env_var(SYMBOL_ATTACH);
let current_symbol = env_var(SYMBOL_CURRENT);
let prev_symbol = env_var(SYMBOL_PREV);
let attach_symbol = env_var(env::ATTACH_SYMBOL);
let current_symbol = env_var(env::CURRENT_SYMBOL);
// pretty print session list
if !state.flags.quiet { println!("sessions:"); }
for session in sessions {
println!("sessions:");
for session in sessions.into_iter() {
let name = session.name.unwrap_or("[untitled]".to_string());
if search.is_some() && !name.starts_with(search.as_ref().unwrap()) { continue; }
if !state.flags.quiet {
let id = session.id.unwrap();
let attached = session.attached.unwrap_or(0) > 0;
let compare = Some(name.clone());
let marker =
if compare == state.title { current_symbol.clone() }
else if state.session && compare == previous { prev_symbol.clone() }
else { " ".to_string() };
let current = Some(name.clone()) == state.title;
println!(
" {marker} {name}{reset} ({bold}{blue}{id}{reset}) {bold}{green}{attach}{reset}",
" {current} {name}{reset} ({bold}{blue}{id}{reset}) {bold}{green}{attach}{reset}",
// values
attach = if attached { attach_symbol.clone() } else { "".to_string() },
current = if current { current_symbol.clone() } else { " ".to_string() },
// formatting
bold = style::Bold,
blue = color::Fg(color::Blue),
green = color::Fg(color::LightGreen),
reset = style::Reset,
);
} else {
print!("{name} ");
}
}
}
@ -204,11 +182,22 @@ pub fn new(state: &mut State) {
let mut tmux = Tmux::new().add_command(new);
// rename window if var not empty
// rename window & check for script if var not empty
if !window_name.is_empty() {
let auto_name = commands::RenameWindow::new()
.new_name(window_name);
.new_name(&window_name);
tmux = tmux.add_command(auto_name);
let rc_var = env_var(env::NEW_RC);
if !(rc_var.is_empty() || rc_var == "0") {
if let Some(mut scripts) = script::list() {
if let Some(path) = scripts.remove(&window_name) {
let rc = commands::SourceFile::new()
.path(path);
tmux = tmux.add_command(rc);
}
}
}
}
tmux.stderr(NULL).output().ok();

View file

@ -2,13 +2,12 @@ use std::env::var;
pub type EnvVar = (&'static str, &'static str);
pub const NEW_WINDOW_NAME: EnvVar = ("REMUX_NEW_WINDOW", "");
pub const REPO_FILE: EnvVar = ("REMUX_REPO_FILE", ".git");
pub const SYMBOL_ATTACH: EnvVar = ("REMUX_ATTACH_SYMBOL", "*");
pub const SYMBOL_CURRENT: EnvVar = ("REMUX_CURRENT_SYMBOL", ">");
pub const SYMBOL_PREV: EnvVar = ("REMUX_PREVIOUS_SYMBOL", "-");
pub static ATTACH_SYMBOL: EnvVar = ("REMUX_ATTACH_SYMBOL", "*");
pub static CURRENT_SYMBOL: EnvVar = ("REMUX_CURRENT_SYMBOL", ">");
pub static NEW_WINDOW_NAME: EnvVar = ("REMUX_NEW_WINDOW", "");
pub static NEW_RC: EnvVar = ("REMUX_NEW_RC", "1");
pub const TMUX: &str = "TMUX";
pub static TMUX: &str = "TMUX";
/// get or default an environment variable
pub fn env_var(envvar: EnvVar) -> String {

View file

@ -1,84 +1,77 @@
use std::process::exit;
/// no subcommand that matches user input; code 1
pub fn no_subcommand(subcommand: String) -> ! {
eprintln!("remux: no command match for \"{subcommand}\"");
pub fn no_subcommand(subcommand: String) {
println!("remux: no command match for \"{subcommand}\"");
exit(1);
}
/// target session not found; code 2
pub fn no_target<S: Into<String>>(target: S) -> ! {
pub fn no_target<S: Into<String>>(target: S) {
let target = target.into();
eprintln!("remux: no session \"{target}\" exists");
println!("remux: no session \"{target}\" exists");
exit(2);
}
/// help topic doesn't exist; code 3
pub fn no_help(topic: String) -> ! {
eprintln!("remux: no help for \"{topic}\"");
pub fn no_help(topic: String) {
println!("remux: no help for \"{topic}\"");
exit(3);
}
/// user provided no target; code 4
pub fn missing_target() -> ! {
eprintln!("remux: no target provided");
pub fn missing_target() {
println!("remux: no target provided");
exit(4);
}
/// refuse to attach to current session; code 4
pub fn same_session() -> ! {
eprintln!("remux: cannot attach to same session");
pub fn same_session() {
println!("remux: cannot attach to same session");
exit(4);
}
/// a session with the target name already exists; code 4
pub fn target_exists<S: Into<String>>(target: S) -> ! {
pub fn target_exists<S: Into<String>>(target: S) {
let target = target.into();
eprintln!("remux: session \"{target}\" already exists");
println!("remux: session \"{target}\" already exists");
exit(4);
}
/// non-terminal environment prevention; code 5
pub fn not_terminal() -> ! {
eprintln!("remux: not running from a terminal");
pub fn not_terminal() {
println!("remux: not running from a terminal");
exit(5);
}
/// tried to nest while not in a session; code 6
pub fn not_nesting() -> ! {
eprintln!("remux: inappropriate nesting flag (-n); not in a session");
pub fn not_nesting() {
println!("remux: inappropriate nesting flag (-n); not in a session");
exit(6);
}
/// operation requires nesting flag; code 6
pub fn prevent_nest() -> ! {
eprintln!("remux: the nesting flag (-n) is required for nesting operation");
pub fn prevent_nest() {
println!("remux: the nesting flag (-n) is required for nesting operation");
exit(6);
}
/// operation conflicts with nesting flag; code 6
pub fn conflict_nest(reason: Option<&'static str>) -> ! {
if let Some(reason) = reason { eprintln!("remux: inappropriate nesting flag (-n): {reason}"); }
else { eprintln!("remux: nesting flag (-n) is inappropriate for this operation."); }
pub fn conflict_nest(reason: Option<&'static str>) {
if let Some(reason) = reason { println!("remux: inappropriate nesting flag (-n): {reason}"); }
else { println!("remux: nesting flag (-n) is inappropriate for this operation."); }
exit(6);
}
/// tried to run a session command outside a session; code 7
pub fn not_in_session(cmd: &'static str) -> ! {
eprintln!("remux: '{cmd}' must be run from within a session");
pub fn not_in_session(cmd: &'static str) {
println!("remux: '{cmd}' must be run from within a session");
exit(7);
}
/// failed to set working directory; code 8
pub fn working_dir_fail(working_dir: &str) -> ! {
eprintln!("remux: failed to set working directory to '{working_dir}'");
exit(8);
}

View file

@ -3,14 +3,13 @@ use pico_args::Arguments;
type Flag = [&'static str;2];
pub static DETACH: Flag = [ "-d", "--detach" ];
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" ];
pub static WORKING_DIR: Flag = [ "-D", "--dir" ];
pub static DETACH: Flag = ["-d", "--detach"];
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"];
pub struct Flags {
pub detached: bool,
@ -18,7 +17,6 @@ pub struct Flags {
pub quiet: bool,
pub read_only: bool,
pub target: Option<String>,
pub working_dir: Option<String>
}
impl Flags {
@ -29,15 +27,13 @@ impl Flags {
let quiet = args.contains(QUIET);
let read_only = args.contains(READ_ONLY);
let target = args.value_from_str(TARGET).ok();
let working_dir = args.value_from_str(WORKING_DIR).ok();
Flags {
detached,
nested,
quiet,
read_only,
target,
working_dir
target
}
}
@ -47,8 +43,7 @@ impl Flags {
nested: self.nested,
quiet: self.quiet,
read_only: self.read_only,
target: self.target.clone(),
working_dir: self.working_dir.clone()
target: None
}
}

View file

@ -6,6 +6,7 @@ mod env;
mod error;
mod flag;
mod help;
mod script;
mod state;
mod util;
@ -38,7 +39,7 @@ fn main() {
Some("help")
=> help(&mut args),
None
=> command::share::context_action(&mut state),
=> command::share::context_action(&state),
Some("a" | "attach")
=> command::share::attach(&mut state),
@ -50,7 +51,7 @@ fn main() {
=> command::share::has(&mut state),
Some("l" | "ls" | "list")
=> command::share::list(&mut state),
=> command::share::list(&state),
Some("n" | "new")
=> command::share::new(&mut state),
@ -64,6 +65,9 @@ fn main() {
Some("t" | "title" | "which")
=> command::session::title(state),
Some("x" | "source")
=> command::session::source(&mut state),
_
=> error::no_subcommand(target.unwrap())
}

40
src/script.rs Normal file
View file

@ -0,0 +1,40 @@
use std::{
collections::HashMap,
env,
fs::read_dir,
path::{ Path, PathBuf }
};
use crate::util::find;
pub type Scripts<'a> = HashMap<String, String>;
/// get a hashmap of scripts
pub fn list<'a>() -> Option<Scripts<'a>> {
let global: Option<PathBuf> =
if let Ok(home) = env::var("HOME") { Some(Path::new(&(home + "/.config/remux")).to_path_buf()) }
else { None };
let local = find(".remux", env::current_dir().unwrap());
if global.is_none() && local.is_none() { return None; }
let mut output = Scripts::new();
if let Some(global) = global { read(&global, &mut output); }
if let Some(local) = local { read(&local, &mut output); }
Some(output)
}
/// populate the given hashmap with scripts from the given path
fn read(path: &PathBuf, map: &mut Scripts) {
if let Ok(children) = read_dir(path) {
for child in children {
if let Ok(child) = child {
let path = child.path();
if let Some(name) = path.file_stem() {
map.insert(name.to_string_lossy().into(), path.to_string_lossy().into());
}
}
}
}
}

View file

@ -6,10 +6,10 @@ use std::{
use pico_args::Arguments;
use crate::{
env::{ env_var, REPO_FILE, TMUX },
env::TMUX,
error,
flag::Flags,
util::{ find, message, MSG_SESSION_NAME }
util::{ find, session_name }
};
pub struct State<'a> {
@ -29,10 +29,7 @@ impl State<'_> {
let flags = Flags::from(args);
let tmux_var = env::var(TMUX).ok();
let session = tmux_var.is_some();
if let Some(ref path) = flags.working_dir { State::set_working_dir(&path); }
let title = if session { message(MSG_SESSION_NAME) } else { None };
let title = if session { session_name() } else { None };
let repository = Repository::find();
State {
@ -65,19 +62,15 @@ impl State<'_> {
if !self.session { error::not_in_session(cmd); }
}
fn set_working_dir(path: &str) {
let result = env::set_current_dir(path);
if result.is_err() {
error::working_dir_fail(path);
}
}
pub fn target(&mut self) -> Option<String> { self.args.subcommand().unwrap_or(None) }
pub fn target_title(&mut self) -> Option<String> {
let from_args = self.target();
if from_args.is_some() { return from_args; }
else if let Some(repository) = &self.repository { Some(repository.name.clone()) }
else { error::missing_target() }
else {
error::missing_target();
None
}
}
}
@ -89,7 +82,7 @@ pub struct Repository {
impl Repository {
pub fn find() -> Option<Repository> {
let path = find(&env_var(REPO_FILE), env::current_dir().unwrap());
let path = find(".git", env::current_dir().unwrap());
if let Some(path) = path {
let name = path.file_name().unwrap().to_string_lossy().to_string();
let inner = Repository {

View file

@ -14,20 +14,14 @@ use crate::error;
pub const NULL: Option<StdIO> = Some(StdIO::Null);
pub const MSG_PREVIOUS: &str = "#{client_last_session}";
pub const MSG_SESSION_NAME: &str = "#S";
pub const MSG_SESSION_PATH: &str = "#{session_path}";
pub const MSG_WINDOW_NAME: &str = "#{window_name}";
pub fn message(fstr: &str) -> Option<String> {
let message = commands::DisplayMessage::new().print().message(fstr);
pub fn session_name() -> Option<String> {
let message = commands::DisplayMessage::new().print().message("#{session_name}");
let result = Tmux::new().add_command(message).output();
if let Ok(output) = result {
let text = String::from_utf8(output.0.stdout);
if let Ok(title) = text {
if title.len() > 0 { Some(title[0..title.len() - 1].to_owned()) }
else { None }
Some(title[0..title.len() - 1].to_owned())
} else { None }
} else { None }
}
@ -57,7 +51,7 @@ pub fn terminal_enforce() {
}
/// recursively propagate up directories to find a child
pub fn find(target: &str, path: PathBuf) -> Option<PathBuf> {
pub fn find(target: &'static str, path: PathBuf) -> Option<PathBuf> {
if path.join(target).exists() { return Some(path); }
let parent = path.parent();