Compare commits
19 commits
Author | SHA1 | Date | |
---|---|---|---|
4f76590930 | |||
d3a408ad33 | |||
8449ae00d6 | |||
ff30bc1052 | |||
8e2826b110 | |||
0fe3906578 | |||
fdf3114c04 | |||
8ad16ad825 | |||
a9a73314af | |||
af33e82415 | |||
eaf72847b1 | |||
beb880ed43 | |||
449c460bbb | |||
1b51633d4f | |||
b7b893d55c | |||
cdc68986fa | |||
1acc7aeb7b | |||
449921656c | |||
638fce2331 |
12 changed files with 206 additions and 100 deletions
|
@ -1,6 +1,6 @@
|
||||||
[package]
|
[package]
|
||||||
name = "remux"
|
name = "remux"
|
||||||
version = "0.3.4"
|
version = "0.4.0"
|
||||||
edition = "2021"
|
edition = "2021"
|
||||||
authors = [ "Valerie Wolfe <sleeplessval@gmail.com>" ]
|
authors = [ "Valerie Wolfe <sleeplessval@gmail.com>" ]
|
||||||
description = "A friendly command shortener for tmux"
|
description = "A friendly command shortener for tmux"
|
||||||
|
|
13
README.md
13
README.md
|
@ -92,6 +92,19 @@ 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>.
|
Install the package using Cargo with the command <code>cargo install tmux-remux</code>.
|
||||||
</details>
|
</details>
|
||||||
|
|
||||||
|
### Supplemental
|
||||||
|
|
||||||
|
<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>
|
||||||
|
Copy <code>man/remux.1</code> into <code>/usr/share/man/man1/</code>.
|
||||||
|
</details>
|
||||||
|
|
||||||
## Configuration
|
## Configuration
|
||||||
|
|
||||||
The pretty-print attached symbol (default: `*`) can be set manually by setting `REMUX_ATTACH_SYMBOL`.
|
The pretty-print attached symbol (default: `*`) can be set manually by setting `REMUX_ATTACH_SYMBOL`.
|
||||||
|
|
19
bash-completion/remux
Normal file
19
bash-completion/remux
Normal file
|
@ -0,0 +1,19 @@
|
||||||
|
_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
|
||||||
|
|
22
man/remux.1
22
man/remux.1
|
@ -1,6 +1,5 @@
|
||||||
.Dd $Mdocdate$
|
.Dd $Mdocdate$
|
||||||
.Dt REMUX 1
|
.Dt REMUX 1
|
||||||
.Os
|
|
||||||
.Sh NAME
|
.Sh NAME
|
||||||
.Nm remux
|
.Nm remux
|
||||||
.Nd a command shortener for
|
.Nd a command shortener for
|
||||||
|
@ -8,7 +7,8 @@
|
||||||
.Sh SYNOPSIS
|
.Sh SYNOPSIS
|
||||||
.Nm remux
|
.Nm remux
|
||||||
.Op Fl dhnqrtv
|
.Op Fl dhnqrtv
|
||||||
.Op command
|
.Op Fl D Ar path
|
||||||
|
.Op Ar command
|
||||||
.Op args...
|
.Op args...
|
||||||
.Sh DESCRIPTION
|
.Sh DESCRIPTION
|
||||||
.Nm
|
.Nm
|
||||||
|
@ -35,6 +35,8 @@ Attaches to an existing session.
|
||||||
.Bl -tag -width Ds -compact
|
.Bl -tag -width Ds -compact
|
||||||
.It Fl d , Fl -detach
|
.It Fl d , Fl -detach
|
||||||
Detach all other connections to the session.
|
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
|
.It Fl n , Fl -nest
|
||||||
Allow nesting (attaching a session from inside another session).
|
Allow nesting (attaching a session from inside another session).
|
||||||
.It Fl r , Fl -read-only
|
.It Fl r , Fl -read-only
|
||||||
|
@ -101,8 +103,8 @@ aliases: p
|
||||||
Prints the session path.
|
Prints the session path.
|
||||||
.Ed
|
.Ed
|
||||||
.It Xo Ic switch
|
.It Xo Ic switch
|
||||||
.Op Fl r , Fl -read-only
|
.Op Fl rd
|
||||||
.Ar title
|
.Op Ar title
|
||||||
.Xc
|
.Xc
|
||||||
.Bd -literal -compact
|
.Bd -literal -compact
|
||||||
aliases: s
|
aliases: s
|
||||||
|
@ -110,10 +112,12 @@ Switches from the current session to the target.
|
||||||
.Ed
|
.Ed
|
||||||
.Pp
|
.Pp
|
||||||
.Bl -tag -width Ds -compact
|
.Bl -tag -width Ds -compact
|
||||||
|
.It Fl d , Fl -detach
|
||||||
|
Detaches other clients from the target session.
|
||||||
.It Fl r , Fl -read-only
|
.It Fl r , Fl -read-only
|
||||||
Switch to the target session in read-only mode.
|
Switch to the target session in read-only mode.
|
||||||
.It Ar title
|
.It Ar title
|
||||||
The title of the session to switch to.
|
The title of the session to switch to. If blank, the previous session will be used.
|
||||||
.El
|
.El
|
||||||
.It Ic title
|
.It Ic title
|
||||||
.Bd -literal -compact
|
.Bd -literal -compact
|
||||||
|
@ -135,6 +139,14 @@ Default: '>'
|
||||||
.It Ev REMUX_NEW_WINDOW
|
.It Ev REMUX_NEW_WINDOW
|
||||||
Provides a default windows name when creating a new session. Unused if empty.
|
Provides a default windows name when creating a new session. Unused if empty.
|
||||||
Default: (unset)
|
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
|
.El
|
||||||
.Sh EXIT STATUS
|
.Sh EXIT STATUS
|
||||||
.Bl -tag -Width Ds
|
.Bl -tag -Width Ds
|
||||||
|
|
|
@ -8,21 +8,18 @@ use tmux_interface::{
|
||||||
use crate::{
|
use crate::{
|
||||||
error,
|
error,
|
||||||
state::State,
|
state::State,
|
||||||
util
|
util::{
|
||||||
|
self,
|
||||||
|
message,
|
||||||
|
MSG_PREVIOUS, MSG_SESSION_PATH, NULL
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
pub fn path(state: &mut State) {
|
pub fn path(state: &mut State) {
|
||||||
state.session_enforce("path");
|
state.session_enforce("path");
|
||||||
|
|
||||||
let message = commands::DisplayMessage::new().print().message("#{session_path}");
|
if let Some(message) = message(MSG_SESSION_PATH) {
|
||||||
|
println!("{message}");
|
||||||
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]);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -33,22 +30,34 @@ pub fn switch(state: &mut State) {
|
||||||
|
|
||||||
// consume optional flags
|
// consume optional flags
|
||||||
let read_only = state.flags.read_only;
|
let read_only = state.flags.read_only;
|
||||||
//TODO: -d flag handling needs to be done manually
|
let detach_other = state.flags.detached;
|
||||||
|
|
||||||
let args = state.args.clone().finish();
|
let args = state.args.clone().finish();
|
||||||
if args.len() < 1 { error::missing_target(); }
|
let target: String = match if let Some(inner) = args.get(0) { inner.to_str() } else { None } {
|
||||||
let target = args.get(0).unwrap().to_string_lossy().to_string();
|
None |
|
||||||
|
Some("-") => if let Some(prev) = message(MSG_PREVIOUS) { prev }
|
||||||
|
else { error::missing_target() },
|
||||||
|
|
||||||
|
Some(inner) => inner.to_owned()
|
||||||
|
};
|
||||||
|
|
||||||
let exists = util::session_exists(target.clone());
|
let exists = util::session_exists(target.clone());
|
||||||
if !exists { error::no_target(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();
|
let mut switch = commands::SwitchClient::new();
|
||||||
switch = switch.target_session(target);
|
switch = switch.target_session(&target);
|
||||||
if read_only { switch.read_only = true; }
|
if read_only { switch.read_only = true; }
|
||||||
|
|
||||||
Tmux::new()
|
tmux.add_command(switch)
|
||||||
.add_command(switch)
|
.stderr(NULL).output().ok();
|
||||||
.output().ok();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn title(state: State) {
|
pub fn title(state: State) {
|
||||||
|
|
|
@ -9,11 +9,19 @@ use tmux_interface::{
|
||||||
};
|
};
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
env::{ self, env_var },
|
env::{
|
||||||
|
self,
|
||||||
|
env_var,
|
||||||
|
SYMBOL_ATTACH, SYMBOL_CURRENT, SYMBOL_PREV
|
||||||
|
},
|
||||||
error,
|
error,
|
||||||
flag,
|
flag,
|
||||||
state::State,
|
state::State,
|
||||||
util
|
util::{
|
||||||
|
self,
|
||||||
|
message,
|
||||||
|
MSG_PREVIOUS, NULL
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
pub fn attach(state: &mut State) {
|
pub fn attach(state: &mut State) {
|
||||||
|
@ -51,12 +59,12 @@ pub fn attach(state: &mut State) {
|
||||||
// build dispatch
|
// build dispatch
|
||||||
let mut tmux = Tmux::new().add_command(attach);
|
let mut tmux = Tmux::new().add_command(attach);
|
||||||
if let Some(select_window) = select_window { tmux = tmux.add_command(select_window); }
|
if let Some(select_window) = select_window { tmux = tmux.add_command(select_window); }
|
||||||
tmux.output().ok();
|
tmux.stderr(NULL).output().ok();
|
||||||
|
|
||||||
state.nest_deinit();
|
state.nest_deinit();
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn context_action(state: &State) {
|
pub fn context_action(state: &mut State) {
|
||||||
if !state.session {
|
if !state.session {
|
||||||
if let Some(repository) = &state.repository {
|
if let Some(repository) = &state.repository {
|
||||||
let target = repository.name.clone();
|
let target = repository.name.clone();
|
||||||
|
@ -72,7 +80,7 @@ pub fn context_action(state: &State) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// fallback behavior is list
|
// fallback behavior is list
|
||||||
list(&state);
|
list(state);
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn detach(state: &mut State) {
|
pub fn detach(state: &mut State) {
|
||||||
|
@ -114,40 +122,55 @@ pub fn has(state: &mut State) {
|
||||||
exit( if success { 0 } else { 1 });
|
exit( if success { 0 } else { 1 });
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn list(state: &State) {
|
pub fn list(state: &mut State) {
|
||||||
// get session list
|
// get session list
|
||||||
let sessions = util::get_sessions().unwrap_or(Vec::new());
|
let sessions = util::get_sessions().unwrap_or(Vec::new());
|
||||||
|
|
||||||
|
let search = state.target();
|
||||||
|
let previous = message(MSG_PREVIOUS);
|
||||||
|
|
||||||
// handle empty case
|
// handle empty case
|
||||||
if sessions.len() == 0 {
|
if sessions.len() == 0 {
|
||||||
println!("no sessions");
|
if !state.flags.quiet { println!("no sessions"); }
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// get attached session symbol
|
// get attached session symbol
|
||||||
let attach_symbol = env_var(env::ATTACH_SYMBOL);
|
let attach_symbol = env_var(SYMBOL_ATTACH);
|
||||||
let current_symbol = env_var(env::CURRENT_SYMBOL);
|
let current_symbol = env_var(SYMBOL_CURRENT);
|
||||||
|
let prev_symbol = env_var(SYMBOL_PREV);
|
||||||
|
|
||||||
// pretty print session list
|
// pretty print session list
|
||||||
println!("sessions:");
|
if !state.flags.quiet { println!("sessions:"); }
|
||||||
for session in sessions.into_iter() {
|
for session in sessions {
|
||||||
let name = session.name.unwrap_or("[untitled]".to_string());
|
let name = session.name.unwrap_or("[untitled]".to_string());
|
||||||
let id = session.id.unwrap();
|
|
||||||
|
|
||||||
let attached = session.attached.unwrap_or(0) > 0;
|
if search.is_some() && !name.starts_with(search.as_ref().unwrap()) { continue; }
|
||||||
let current = Some(name.clone()) == state.title;
|
|
||||||
|
|
||||||
println!(
|
if !state.flags.quiet {
|
||||||
" {current} {name}{reset} ({bold}{blue}{id}{reset}) {bold}{green}{attach}{reset}",
|
let id = session.id.unwrap();
|
||||||
// values
|
|
||||||
attach = if attached { attach_symbol.clone() } else { "".to_string() },
|
let attached = session.attached.unwrap_or(0) > 0;
|
||||||
current = if current { current_symbol.clone() } else { " ".to_string() },
|
|
||||||
// formatting
|
let compare = Some(name.clone());
|
||||||
bold = style::Bold,
|
let marker =
|
||||||
blue = color::Fg(color::Blue),
|
if compare == state.title { current_symbol.clone() }
|
||||||
green = color::Fg(color::LightGreen),
|
else if state.session && compare == previous { prev_symbol.clone() }
|
||||||
reset = style::Reset,
|
else { " ".to_string() };
|
||||||
);
|
|
||||||
|
println!(
|
||||||
|
" {marker} {name}{reset} ({bold}{blue}{id}{reset}) {bold}{green}{attach}{reset}",
|
||||||
|
// values
|
||||||
|
attach = if attached { attach_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} ");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -188,6 +211,6 @@ pub fn new(state: &mut State) {
|
||||||
tmux = tmux.add_command(auto_name);
|
tmux = tmux.add_command(auto_name);
|
||||||
}
|
}
|
||||||
|
|
||||||
tmux.output().ok();
|
tmux.stderr(NULL).output().ok();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
10
src/env.rs
10
src/env.rs
|
@ -2,11 +2,13 @@ use std::env::var;
|
||||||
|
|
||||||
pub type EnvVar = (&'static str, &'static str);
|
pub type EnvVar = (&'static str, &'static str);
|
||||||
|
|
||||||
pub static ATTACH_SYMBOL: EnvVar = ("REMUX_ATTACH_SYMBOL", "*");
|
pub const NEW_WINDOW_NAME: EnvVar = ("REMUX_NEW_WINDOW", "");
|
||||||
pub static CURRENT_SYMBOL: EnvVar = ("REMUX_CURRENT_SYMBOL", ">");
|
pub const REPO_FILE: EnvVar = ("REMUX_REPO_FILE", ".git");
|
||||||
pub static NEW_WINDOW_NAME: EnvVar = ("REMUX_NEW_WINDOW", "");
|
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 TMUX: &str = "TMUX";
|
pub const TMUX: &str = "TMUX";
|
||||||
|
|
||||||
/// get or default an environment variable
|
/// get or default an environment variable
|
||||||
pub fn env_var(envvar: EnvVar) -> String {
|
pub fn env_var(envvar: EnvVar) -> String {
|
||||||
|
|
53
src/error.rs
53
src/error.rs
|
@ -1,77 +1,84 @@
|
||||||
use std::process::exit;
|
use std::process::exit;
|
||||||
|
|
||||||
/// no subcommand that matches user input; code 1
|
/// no subcommand that matches user input; code 1
|
||||||
pub fn no_subcommand(subcommand: String) {
|
pub fn no_subcommand(subcommand: String) -> ! {
|
||||||
println!("remux: no command match for \"{subcommand}\"");
|
eprintln!("remux: no command match for \"{subcommand}\"");
|
||||||
exit(1);
|
exit(1);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/// target session not found; code 2
|
/// 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();
|
let target = target.into();
|
||||||
println!("remux: no session \"{target}\" exists");
|
eprintln!("remux: no session \"{target}\" exists");
|
||||||
exit(2);
|
exit(2);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/// help topic doesn't exist; code 3
|
/// help topic doesn't exist; code 3
|
||||||
pub fn no_help(topic: String) {
|
pub fn no_help(topic: String) -> ! {
|
||||||
println!("remux: no help for \"{topic}\"");
|
eprintln!("remux: no help for \"{topic}\"");
|
||||||
exit(3);
|
exit(3);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/// user provided no target; code 4
|
/// user provided no target; code 4
|
||||||
pub fn missing_target() {
|
pub fn missing_target() -> ! {
|
||||||
println!("remux: no target provided");
|
eprintln!("remux: no target provided");
|
||||||
exit(4);
|
exit(4);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// refuse to attach to current session; code 4
|
/// refuse to attach to current session; code 4
|
||||||
pub fn same_session() {
|
pub fn same_session() -> ! {
|
||||||
println!("remux: cannot attach to same session");
|
eprintln!("remux: cannot attach to same session");
|
||||||
exit(4);
|
exit(4);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// a session with the target name already exists; code 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();
|
let target = target.into();
|
||||||
println!("remux: session \"{target}\" already exists");
|
eprintln!("remux: session \"{target}\" already exists");
|
||||||
exit(4);
|
exit(4);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/// non-terminal environment prevention; code 5
|
/// non-terminal environment prevention; code 5
|
||||||
pub fn not_terminal() {
|
pub fn not_terminal() -> ! {
|
||||||
println!("remux: not running from a terminal");
|
eprintln!("remux: not running from a terminal");
|
||||||
exit(5);
|
exit(5);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/// tried to nest while not in a session; code 6
|
/// tried to nest while not in a session; code 6
|
||||||
pub fn not_nesting() {
|
pub fn not_nesting() -> ! {
|
||||||
println!("remux: inappropriate nesting flag (-n); not in a session");
|
eprintln!("remux: inappropriate nesting flag (-n); not in a session");
|
||||||
exit(6);
|
exit(6);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// operation requires nesting flag; code 6
|
/// operation requires nesting flag; code 6
|
||||||
pub fn prevent_nest() {
|
pub fn prevent_nest() -> ! {
|
||||||
println!("remux: the nesting flag (-n) is required for nesting operation");
|
eprintln!("remux: the nesting flag (-n) is required for nesting operation");
|
||||||
exit(6);
|
exit(6);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// operation conflicts with nesting flag; code 6
|
/// operation conflicts with nesting flag; code 6
|
||||||
pub fn conflict_nest(reason: Option<&'static str>) {
|
pub fn conflict_nest(reason: Option<&'static str>) -> ! {
|
||||||
if let Some(reason) = reason { println!("remux: inappropriate nesting flag (-n): {reason}"); }
|
if let Some(reason) = reason { eprintln!("remux: inappropriate nesting flag (-n): {reason}"); }
|
||||||
else { println!("remux: nesting flag (-n) is inappropriate for this operation."); }
|
else { eprintln!("remux: nesting flag (-n) is inappropriate for this operation."); }
|
||||||
exit(6);
|
exit(6);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/// tried to run a session command outside a session; code 7
|
/// tried to run a session command outside a session; code 7
|
||||||
pub fn not_in_session(cmd: &'static str) {
|
pub fn not_in_session(cmd: &'static str) -> ! {
|
||||||
println!("remux: '{cmd}' must be run from within a session");
|
eprintln!("remux: '{cmd}' must be run from within a session");
|
||||||
exit(7);
|
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);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
23
src/flag.rs
23
src/flag.rs
|
@ -3,13 +3,14 @@ use pico_args::Arguments;
|
||||||
|
|
||||||
type Flag = [&'static str;2];
|
type Flag = [&'static str;2];
|
||||||
|
|
||||||
pub static DETACH: Flag = ["-d", "--detach"];
|
pub static DETACH: Flag = [ "-d", "--detach" ];
|
||||||
pub static HELP: Flag = ["-h", "--help"];
|
pub static HELP: Flag = [ "-h", "--help" ];
|
||||||
pub static NEST: Flag = ["-n", "--nest"];
|
pub static NEST: Flag = [ "-n", "--nest" ];
|
||||||
pub static QUIET: Flag = ["-q", "--quiet"];
|
pub static QUIET: Flag = [ "-q", "--quiet" ];
|
||||||
pub static READ_ONLY: Flag = ["-r", "--read-only"];
|
pub static READ_ONLY: Flag = [ "-r", "--read-only" ];
|
||||||
pub static TARGET: Flag = ["-t", "--target"];
|
pub static TARGET: Flag = [ "-t", "--target" ];
|
||||||
pub static VERSION: Flag = ["-v", "--version"];
|
pub static VERSION: Flag = [ "-v", "--version" ];
|
||||||
|
pub static WORKING_DIR: Flag = [ "-D", "--dir" ];
|
||||||
|
|
||||||
pub struct Flags {
|
pub struct Flags {
|
||||||
pub detached: bool,
|
pub detached: bool,
|
||||||
|
@ -17,6 +18,7 @@ pub struct Flags {
|
||||||
pub quiet: bool,
|
pub quiet: bool,
|
||||||
pub read_only: bool,
|
pub read_only: bool,
|
||||||
pub target: Option<String>,
|
pub target: Option<String>,
|
||||||
|
pub working_dir: Option<String>
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Flags {
|
impl Flags {
|
||||||
|
@ -27,13 +29,15 @@ impl Flags {
|
||||||
let quiet = args.contains(QUIET);
|
let quiet = args.contains(QUIET);
|
||||||
let read_only = args.contains(READ_ONLY);
|
let read_only = args.contains(READ_ONLY);
|
||||||
let target = args.value_from_str(TARGET).ok();
|
let target = args.value_from_str(TARGET).ok();
|
||||||
|
let working_dir = args.value_from_str(WORKING_DIR).ok();
|
||||||
|
|
||||||
Flags {
|
Flags {
|
||||||
detached,
|
detached,
|
||||||
nested,
|
nested,
|
||||||
quiet,
|
quiet,
|
||||||
read_only,
|
read_only,
|
||||||
target
|
target,
|
||||||
|
working_dir
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -43,7 +47,8 @@ impl Flags {
|
||||||
nested: self.nested,
|
nested: self.nested,
|
||||||
quiet: self.quiet,
|
quiet: self.quiet,
|
||||||
read_only: self.read_only,
|
read_only: self.read_only,
|
||||||
target: None
|
target: self.target.clone(),
|
||||||
|
working_dir: self.working_dir.clone()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -38,7 +38,7 @@ fn main() {
|
||||||
Some("help")
|
Some("help")
|
||||||
=> help(&mut args),
|
=> help(&mut args),
|
||||||
None
|
None
|
||||||
=> command::share::context_action(&state),
|
=> command::share::context_action(&mut state),
|
||||||
|
|
||||||
Some("a" | "attach")
|
Some("a" | "attach")
|
||||||
=> command::share::attach(&mut state),
|
=> command::share::attach(&mut state),
|
||||||
|
@ -50,7 +50,7 @@ fn main() {
|
||||||
=> command::share::has(&mut state),
|
=> command::share::has(&mut state),
|
||||||
|
|
||||||
Some("l" | "ls" | "list")
|
Some("l" | "ls" | "list")
|
||||||
=> command::share::list(&state),
|
=> command::share::list(&mut state),
|
||||||
|
|
||||||
Some("n" | "new")
|
Some("n" | "new")
|
||||||
=> command::share::new(&mut state),
|
=> command::share::new(&mut state),
|
||||||
|
|
23
src/state.rs
23
src/state.rs
|
@ -6,10 +6,10 @@ use std::{
|
||||||
use pico_args::Arguments;
|
use pico_args::Arguments;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
env::TMUX,
|
env::{ env_var, REPO_FILE, TMUX },
|
||||||
error,
|
error,
|
||||||
flag::Flags,
|
flag::Flags,
|
||||||
util::{ find, session_name }
|
util::{ find, message, MSG_SESSION_NAME }
|
||||||
};
|
};
|
||||||
|
|
||||||
pub struct State<'a> {
|
pub struct State<'a> {
|
||||||
|
@ -29,7 +29,10 @@ impl State<'_> {
|
||||||
let flags = Flags::from(args);
|
let flags = Flags::from(args);
|
||||||
let tmux_var = env::var(TMUX).ok();
|
let tmux_var = env::var(TMUX).ok();
|
||||||
let session = tmux_var.is_some();
|
let session = tmux_var.is_some();
|
||||||
let title = if session { session_name() } else { None };
|
|
||||||
|
if let Some(ref path) = flags.working_dir { State::set_working_dir(&path); }
|
||||||
|
|
||||||
|
let title = if session { message(MSG_SESSION_NAME) } else { None };
|
||||||
let repository = Repository::find();
|
let repository = Repository::find();
|
||||||
|
|
||||||
State {
|
State {
|
||||||
|
@ -62,15 +65,19 @@ impl State<'_> {
|
||||||
if !self.session { error::not_in_session(cmd); }
|
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(&mut self) -> Option<String> { self.args.subcommand().unwrap_or(None) }
|
||||||
pub fn target_title(&mut self) -> Option<String> {
|
pub fn target_title(&mut self) -> Option<String> {
|
||||||
let from_args = self.target();
|
let from_args = self.target();
|
||||||
if from_args.is_some() { return from_args; }
|
if from_args.is_some() { return from_args; }
|
||||||
else if let Some(repository) = &self.repository { Some(repository.name.clone()) }
|
else if let Some(repository) = &self.repository { Some(repository.name.clone()) }
|
||||||
else {
|
else { error::missing_target() }
|
||||||
error::missing_target();
|
|
||||||
None
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -82,7 +89,7 @@ pub struct Repository {
|
||||||
|
|
||||||
impl Repository {
|
impl Repository {
|
||||||
pub fn find() -> Option<Repository> {
|
pub fn find() -> Option<Repository> {
|
||||||
let path = find(".git", env::current_dir().unwrap());
|
let path = find(&env_var(REPO_FILE), env::current_dir().unwrap());
|
||||||
if let Some(path) = path {
|
if let Some(path) = path {
|
||||||
let name = path.file_name().unwrap().to_string_lossy().to_string();
|
let name = path.file_name().unwrap().to_string_lossy().to_string();
|
||||||
let inner = Repository {
|
let inner = Repository {
|
||||||
|
|
19
src/util.rs
19
src/util.rs
|
@ -4,7 +4,7 @@ use std::{
|
||||||
};
|
};
|
||||||
|
|
||||||
use tmux_interface::{
|
use tmux_interface::{
|
||||||
Session, Tmux,
|
Session, StdIO, Tmux,
|
||||||
|
|
||||||
commands,
|
commands,
|
||||||
variables::session::SessionsCtl
|
variables::session::SessionsCtl
|
||||||
|
@ -12,14 +12,22 @@ use tmux_interface::{
|
||||||
|
|
||||||
use crate::error;
|
use crate::error;
|
||||||
|
|
||||||
pub fn session_name() -> Option<String> {
|
pub const NULL: Option<StdIO> = Some(StdIO::Null);
|
||||||
let message = commands::DisplayMessage::new().print().message("#{session_name}");
|
|
||||||
|
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);
|
||||||
|
|
||||||
let result = Tmux::new().add_command(message).output();
|
let result = Tmux::new().add_command(message).output();
|
||||||
if let Ok(output) = result {
|
if let Ok(output) = result {
|
||||||
let text = String::from_utf8(output.0.stdout);
|
let text = String::from_utf8(output.0.stdout);
|
||||||
if let Ok(title) = text {
|
if let Ok(title) = text {
|
||||||
Some(title[0..title.len() - 1].to_owned())
|
if title.len() > 0 { Some(title[0..title.len() - 1].to_owned()) }
|
||||||
|
else { None }
|
||||||
} else { None }
|
} else { None }
|
||||||
} else { None }
|
} else { None }
|
||||||
}
|
}
|
||||||
|
@ -37,6 +45,7 @@ pub fn session_exists<S: Into<String>>(target: S) -> bool {
|
||||||
let has_session = commands::HasSession::new()
|
let has_session = commands::HasSession::new()
|
||||||
.target_session(target.into());
|
.target_session(target.into());
|
||||||
Tmux::new().add_command(has_session)
|
Tmux::new().add_command(has_session)
|
||||||
|
.stderr(NULL)
|
||||||
.status()
|
.status()
|
||||||
.unwrap()
|
.unwrap()
|
||||||
.success()
|
.success()
|
||||||
|
@ -48,7 +57,7 @@ pub fn terminal_enforce() {
|
||||||
}
|
}
|
||||||
|
|
||||||
/// recursively propagate up directories to find a child
|
/// recursively propagate up directories to find a child
|
||||||
pub fn find(target: &'static str, path: PathBuf) -> Option<PathBuf> {
|
pub fn find(target: &str, path: PathBuf) -> Option<PathBuf> {
|
||||||
if path.join(target).exists() { return Some(path); }
|
if path.join(target).exists() { return Some(path); }
|
||||||
|
|
||||||
let parent = path.parent();
|
let parent = path.parent();
|
||||||
|
|
Loading…
Reference in a new issue