diff --git a/Makefile b/Makefile index 7d6e25de16a45d1b3d42db2165ea81d193debc2b..ca0569e6821405936d2c34d24ef9e22c52e7bd42 100644 --- a/Makefile +++ b/Makefile @@ -1,4 +1,5 @@ -DESTDIR ?= /usr/local/bin/ +prefix ?= /usr/local +BINARY = $(shell echo $(prefix)/bin/ion | sed 's/\/\//\//g') all: cargo build --release @@ -10,9 +11,20 @@ tests: cargo test --manifest-path members/ranges/Cargo.toml bash examples/run_examples.sh -install: - strip -s target/release/ion - install -Dm0755 target/release/ion $(DESTDIR) +install: update-shells + install -Dm0755 target/release/ion $(DESTDIR)/$(BINARY) uninstall: - rm $(DESTDIR)/ion + rm $(DESTDIR)/$(BINARY) + +update-shells: + if ! grep ion /etc/shells >/dev/null; then \ + echo $(BINARY) >> /etc/shells; \ + else \ + shell=$(shell grep ion /etc/shells); \ + if [ $$shell != $(BINARY) ]; then \ + before=$$(echo $$shell | sed 's/\//\\\//g'); \ + after=$$(echo $(BINARY) | sed 's/\//\\\//g'); \ + sed -i -e "s/$$before/$$after/g" /etc/shells; \ + fi \ + fi diff --git a/README.md b/README.md index ac3d16c6f83639932947f1aeaf8c8facfb43a6a2..4275a2b7a22e53e5ae2b7d7c9eaac8cb00703a84 100644 --- a/README.md +++ b/README.md @@ -46,7 +46,7 @@ cargo +nightly install --git https://gitlab.redox-os.org/redox-os/ion/ ```sh git clone https://gitlab.redox-os.org/redox-os/ion/ -cd ion && cargo +nightly build --release +cd ion && make && sudo make install prefix=/usr ``` # Vim/NeoVim Syntax Highlighting Plugin diff --git a/src/lib/builtins/mod.rs b/src/lib/builtins/mod.rs index 4a6da3f7e9c4fdd561434bc1f677b880a493a592..a482b9d8d0762c47dcf1b0611ff7e97c541abac5 100644 --- a/src/lib/builtins/mod.rs +++ b/src/lib/builtins/mod.rs @@ -19,9 +19,7 @@ use self::{ status::status, test::test, variables::{alias, drop_alias, drop_array, drop_variable}, }; -use std::{ - env, error::Error, io::{self, Write}, -}; +use std::{error::Error, io::{self, Write}}; use parser::Terminator; use shell::{ @@ -156,20 +154,7 @@ pub fn builtin_cd(args: &[String], shell: &mut Shell) -> i32 { match shell.directory_stack.cd(args, &shell.variables) { Ok(()) => { - match env::current_dir() { - Ok(cwd) => { - let pwd = env::var("PWD").ok().unwrap_or_else(String::default); - let pwd = &pwd; - let current_dir = cwd.to_str().unwrap_or("?"); - - if pwd != current_dir { - env::set_var("OLDPWD", pwd); - env::set_var("PWD", current_dir); - } - fork_function(shell, "CD_CHANGE", &["ion"]); - } - Err(_) => env::set_var("PWD", "?"), - }; + fork_function(shell, "CD_CHANGE", &["ion"]); SUCCESS } Err(why) => { diff --git a/src/lib/shell/directory_stack.rs b/src/lib/shell/directory_stack.rs index dbaca704f0453606ffff8f25a3dac9faca31116e..d9564e94d2ab201e1ee1623ba9a4061f52e613ce 100644 --- a/src/lib/shell/directory_stack.rs +++ b/src/lib/shell/directory_stack.rs @@ -3,10 +3,25 @@ use super::{ variables::Variables, }; use std::{ - borrow::Cow, collections::VecDeque, env::{self, current_dir, home_dir, set_current_dir}, - path::PathBuf, + borrow::Cow, collections::VecDeque, env::{self, home_dir, set_current_dir}, + path::{Component, Path, PathBuf}, }; +fn set_current_dir_ion(dir: &Path) -> Result<(), Cow<'static, str>> { + set_current_dir(dir) + .map_err(|why| Cow::Owned(format!("{}", why)))?; + + env::set_var( + "OLDPWD", + env::var("PWD").ok() + .and_then(|pwd| if pwd.is_empty() { None } else { Some(pwd) }) + .unwrap_or_else(|| "?".into()) + ); + + env::set_var("PWD", dir.to_str().unwrap_or_else(|| "?".into())); + Ok(()) +} + #[derive(Debug)] pub struct DirectoryStack { dirs: VecDeque<PathBuf>, // The top is always the current directory @@ -14,7 +29,6 @@ pub struct DirectoryStack { impl DirectoryStack { fn normalize_path(&mut self, dir: &str) -> PathBuf { - use std::path::{Component, Path}; // Create a clone of the current directory. let mut new_dir = match self.dirs.front() { Some(cur_dir) => cur_dir.clone(), @@ -60,13 +74,12 @@ impl DirectoryStack { ) -> Result<(), Cow<'static, str>> { let dir = self.dirs.get(index).ok_or_else(|| { Cow::Owned(format!( - "ion: {}: {}: directory stack out of range\n", + "ion: {}: {}: directory stack out of range", caller, index )) })?; - set_current_dir(dir) - .map_err(|_| Cow::Owned(format!("ion: {}: Failed setting current dir\n", caller))) + set_current_dir_ion(dir) } fn print_dirs(&self) { @@ -170,16 +183,13 @@ impl DirectoryStack { variables: &Variables, ) -> Result<(), Cow<'static, str>> { let new_dir = self.normalize_path(dir); - - // Try to change into the new directory - match set_current_dir(&new_dir) { + match set_current_dir_ion(&new_dir) { Ok(()) => { - // Push the new current directory onto the directory stack. self.push_dir(new_dir, variables); Ok(()) } Err(err) => Err(Cow::Owned(format!( - "ion: failed to set current dir to {}: {}\n", + "ion: failed to set current dir to {}: {}", new_dir.to_string_lossy(), err ))), @@ -187,14 +197,9 @@ impl DirectoryStack { } fn get_previous_dir(&self) -> Option<String> { - match env::var("OLDPWD") { - Ok(previous_pwd) => if previous_pwd == "?" || previous_pwd == "" { - None - } else { - Some(previous_pwd) - }, - Err(_) => None, - } + env::var("OLDPWD").ok().and_then(|pwd| + if pwd.is_empty() || pwd == "?" { None } else { Some(pwd) } + ) } fn switch_to_previous_directory( @@ -225,23 +230,6 @@ impl DirectoryStack { ) } - fn update_env_variables(&mut self) { - // Update $OLDPWD - if let Ok(old_pwd) = env::var("PWD") { - if old_pwd.is_empty() { - env::set_var("OLDPWD", "?"); - } else { - env::set_var("OLDPWD", &old_pwd); - } - } - - // Update $PWD - match current_dir() { - Ok(current_dir) => env::set_var("PWD", current_dir.to_str().unwrap_or("?")), - Err(_) => env::set_var("PWD", "?"), - } - } - pub(crate) fn cd<I: IntoIterator>( &mut self, args: I, @@ -295,7 +283,7 @@ impl DirectoryStack { None => Action::Push(PathBuf::from(arg)), // no numeric arg => `dir`-parameter }; } else { - return Err(Cow::Borrowed("ion: pushd: too many arguments\n")); + return Err(Cow::Borrowed("ion: pushd: too many arguments")); } } @@ -303,7 +291,7 @@ impl DirectoryStack { match action { Action::Switch => { if len < 2 { - return Err(Cow::Borrowed("ion: pushd: no other directory\n")); + return Err(Cow::Borrowed("ion: pushd: no other directory")); } if !keep_front { self.set_current_dir_by_index(1, "pushd")?; @@ -326,7 +314,6 @@ impl DirectoryStack { } }; - self.update_env_variables(); self.print_dirs(); Ok(()) } @@ -353,7 +340,7 @@ impl DirectoryStack { } None => { return Err(Cow::Owned(format!( - "ion: popd: {}: invalid argument\n", + "ion: popd: {}: invalid argument", arg ))) } @@ -363,14 +350,14 @@ impl DirectoryStack { let len: usize = self.dirs.len(); if len <= 1 { - return Err(Cow::Borrowed("ion: popd: directory stack empty\n")); + return Err(Cow::Borrowed("ion: popd: directory stack empty")); } let mut index: usize = if count_from_front { num } else { (len - 1).checked_sub(num).ok_or_else(|| { - Cow::Owned("ion: popd: negative directory stack index out of range\n".to_owned()) + Cow::Owned("ion: popd: negative directory stack index out of range".to_owned()) })? }; @@ -387,12 +374,11 @@ impl DirectoryStack { // pop element if self.dirs.remove(index).is_none() { return Err(Cow::Owned(format!( - "ion: popd: {}: directory stack index out of range\n", + "ion: popd: {}: directory stack index out of range", index ))); } - self.update_env_variables(); self.print_dirs(); Ok(()) } @@ -413,13 +399,15 @@ impl DirectoryStack { /// if available. pub(crate) fn new() -> DirectoryStack { let mut dirs: VecDeque<PathBuf> = VecDeque::new(); - match current_dir() { + match env::current_dir() { Ok(curr_dir) => { + env::set_var("PWD", curr_dir.to_str().unwrap_or_else(|| "?")); dirs.push_front(curr_dir); DirectoryStack { dirs } } Err(_) => { eprintln!("ion: failed to get current directory when building directory stack"); + env::set_var("PWD", "?"); DirectoryStack { dirs } } } diff --git a/src/lib/shell/variables/mod.rs b/src/lib/shell/variables/mod.rs index f71a32141bcdc16c32655eb19b32d346b05250f9..a866562546fa432fb8ff0a39acc304899cf5203d 100644 --- a/src/lib/shell/variables/mod.rs +++ b/src/lib/shell/variables/mod.rs @@ -198,12 +198,6 @@ impl Default for Variables { map.insert("HISTORY_IGNORE".into(), VariableType::Array(array!["no_such_command", "whitespace", "duplicates"])); - // Initialize the PWD (Present Working Directory) variable - env::current_dir().ok().map_or_else( - || env::set_var("PWD", "?"), - |path| env::set_var("PWD", path.to_str().unwrap_or("?")), - ); - // Initialize the HOME variable env::home_dir().map_or_else( || env::set_var("HOME", "?"),