Compare commits

27 Commits

Author SHA1 Message Date
fbf6be2779 drone
All checks were successful
continuous-integration/drone/push Build is passing
2023-07-14 12:15:36 +02:00
0aabd711a3 Merge pull request 'dev' (#4) from dev into main
Reviewed-on: #4
2023-05-01 13:44:36 +02:00
f333069a32 Fixed the ./ bug 2023-05-01 13:42:56 +02:00
7677adfafe Readme 2023-03-14 22:27:43 +01:00
1eefd44a89 Merge pull request 'dev' (#3) from dev into main
Reviewed-on: #3
2023-03-04 20:26:00 +01:00
09cea688b7 Added export command 2023-03-04 20:17:22 +01:00
5743678e23 Fixed bug on incorrect command, missing carriage return on cd 2023-03-04 19:56:50 +01:00
1e052f1a79 Merge pull request 'dev' (#2) from dev into main
Reviewed-on: #2
2023-03-04 14:31:17 +01:00
522b8e7037 Add doc 2023-03-04 14:30:13 +01:00
643c69623c Forget to add the config part 2023-03-04 14:28:20 +01:00
68dc23a15e Added a dirty implementation of env var setting 2023-03-04 14:19:01 +01:00
1eb1b239c2 Reformatted the command execution part 2023-03-02 14:49:34 +01:00
488c9c6e0a Started to implement CmdOutput - keep going... 2023-03-02 11:46:21 +01:00
7b0f4b1922 Merge pull request 'dev' (#1) from dev into main
Reviewed-on: #1
2023-03-01 17:41:20 +01:00
4c64bb8038 Reformatted the code to use elems 2023-03-01 17:40:03 +01:00
9e1d0dfcde Added elems to store terminal variables 2023-03-01 17:00:43 +01:00
aec1641024 Some code reformat 2023-03-01 00:37:05 +01:00
b5ef54c8f4 Update 'README.md' 2023-02-27 11:59:58 +01:00
df6aee4a49 Implement screen clearing with Ctrl+l 2023-02-24 12:17:04 +01:00
0656f163fd Added Italic and Bold to prompt, removed some newlines when autocomplete 2023-02-24 11:59:36 +01:00
eda72c00ab Update 'README.md' 2023-02-23 22:04:39 +01:00
a515c81e49 Update 'README.md' 2023-02-23 22:04:15 +01:00
b2f64ea970 Update 'README.md' 2023-02-23 12:15:30 +01:00
9979140645 Update 'README.md' 2023-02-23 12:07:23 +01:00
d22e34c8bd Now handling double quotes properly 2023-02-23 11:53:49 +01:00
6f63294857 "Ctrl + P and O more or less do the same thing as Ctrl + Right and left
Also remove a warning about a useless mut"
2023-02-22 16:03:51 +01:00
d903bb4bc8 Update 'README.md' 2023-02-22 14:56:51 +01:00
10 changed files with 842 additions and 253 deletions

View File

@ -3,8 +3,38 @@ type: docker
name: default name: default
steps: steps:
- name: build
- name: test image: rust:latest
image: rust:1.66.0-alpine3.17
commands: commands:
- cargo build --verbose - cargo build --release
---
kind: pipeline
type: docker
name: release
steps:
- name: build
image: rust:latest
commands:
- cargo build --release
when:
event:
include:
- tag
- name: create-release
image: plugins/gitea-release:latest
custom_dns: [ 9.9.9.9 ]
settings:
api_key:
from_secret: gitea_api
base_url: https://gitea.squi.fr
note: This is an automated release of sqish.
files:
- ./target/release/portnut
when:
event:
include:
- tag

7
Cargo.lock generated
View File

@ -195,6 +195,12 @@ version = "0.6.28"
source = "registry+https://github.com/rust-lang/crates.io-index" source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "456c603be3e8d448b072f410900c09faf164fbce2d480456f50eea6e25f9c848" checksum = "456c603be3e8d448b072f410900c09faf164fbce2d480456f50eea6e25f9c848"
[[package]]
name = "shell-words"
version = "1.1.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "24188a676b6ae68c3b2cb3a01be17fbf7240ce009799bb56d5b1409051e78fde"
[[package]] [[package]]
name = "sqish" name = "sqish"
version = "1.0.0" version = "1.0.0"
@ -203,6 +209,7 @@ dependencies = [
"dirs", "dirs",
"gethostname", "gethostname",
"regex", "regex",
"shell-words",
"termion", "termion",
"unicode-segmentation", "unicode-segmentation",
"users", "users",

View File

@ -17,3 +17,4 @@ which = { version = "4.4.0", features = ["regex"] }
regex = "1.7.1" regex = "1.7.1"
yaml-rust = "0.4.5" yaml-rust = "0.4.5"
ctrlc = "3.2.5" ctrlc = "3.2.5"
shell-words = "1.1.0"

View File

@ -1,12 +1,32 @@
# Sqish # Sqish
Rust Shell. This is an attempt to create a simple but usable shell in Rust, because why not. ## Sqish's goal
Squi's Shell, aka Sqish. This is an attempt to create a simple but usable shell in Rust, because why not. The goal (for me) is also to get more familiar with Rust since I read the Rust book and want to try things out. And, at last, building and using your own tools is pretty cool.
TODO: I also implemented user-configurable hotkeys (they're more akin to a form of macros really), because I don't think a lot of shells do that and I wanted something unique.
* git commit -m "message" does not work
* Allow redirecting >> to the end of a file ? ## Current state
Currently, the shell is pretty rough around the edges, but usable, at least for my day to day use. It's not really and equivalent to bash as it does not implement any form of scripting; the only supported operation is piping. My goal is to build a shell for my everyday use as a Linux sysadmin: simple, fast, with just the features that I need.
## Scripting ?
I don't like shell scripting and don't use it (I know the basics, but Ansible and Python are my bread and butter). Building a lexer is not part of the project. Bash / Zsh / Python / etc are still usable when running sqish, so that's that.
Sqish is now my default shell on my home computer, and I'll keep adding features as I come across them.
Also keep in mind I'm a beginner in Rust, so the code is pretty dirty; I will probably come around to reformating it once I have all the features I want and ironed out all the bugs.
# TODO
TODO (Reformat):
* Creating a struct "shell" that contains builder method taking an input and an output as parameters as well as a conf could be cool
TODO (features):
* Add a &&
TODO (Bugz):
* Autocomplete shits itself when typing a path starting with /
## sqishrc (Config) ## sqishrc (Config)
See the included sqishrc file included, and copy it as ~/.sqishrc See the included sqishrc file included for comments, and copy it as ~/.sqishrc for use (optionnal).
## Built-in shortcuts ## Built-in shortcuts
### As Hotkeys ### As Hotkeys
@ -14,10 +34,13 @@ Some shortcuts are built in. They may be unique to Sqish or be copied from Zsh f
* Ctrl+Q or Ctrl+D : Quits the shell immediatly * Ctrl+Q or Ctrl+D : Quits the shell immediatly
* Ctrl+U : Clears currently written line * Ctrl+U : Clears currently written line
* Alt+. : Appends the last word from previous command * Alt+. : Appends the last word from previous command
* Alt+! : Show a list of the user-configured hotkeys * Alt+! : Show a list of the user-configured hotkeys and aliases
* Ctrl+A : Jumps to the end of the line * Ctrl+A : Jumps to the end of the line
* Ctrl+E : Jumps to the start of the line * Ctrl+E : Jumps to the start of the line
* Ctrl+C : Clears the current line and starts a new one * Ctrl+C : Clears the current line and starts a new one
* Ctrl+L : Clears the screen
* Ctrl+P : Jump to the next word (kinda)
* Ctrl+O : Jump to the previous word (kinda)
### In Text: ### In Text:
* !12 : Returns command number 12 from history (Replace the number 12 with any number from history). You can add commands after it, like: "!12 | grep squirrel" * !12 : Returns command number 12 from history (Replace the number 12 with any number from history). You can add commands after it, like: "!12 | grep squirrel"

10
sqishrc
View File

@ -24,6 +24,11 @@
#RGB ! #RGB !
#RGB is defined as $RGB|red|green|blue_ #RGB is defined as $RGB|red|green|blue_
#For example $RGB|108|84|30_ gives you an ugly shade of brown #For example $RGB|108|84|30_ gives you an ugly shade of brown
#
#Styles:
#BOLD : start of a bold block
#ITA : start of an italic block
#STYLERESET : reset all styles.
prompt: "$COLORLBLACK_ !$HISTNUMBER_$COLORCYAN_[$USER_@$HOSTNAME_]$RGB|125|0|125_$DIR_> $COLORRESET_" prompt: "$COLORLBLACK_ !$HISTNUMBER_$COLORCYAN_[$USER_@$HOSTNAME_]$RGB|125|0|125_$DIR_> $COLORRESET_"
#Classic aliases, can be used in conjunction with hotkeys #Classic aliases, can be used in conjunction with hotkeys
@ -39,3 +44,8 @@ hotkeys:
#Init : A command to run on startup. Can be used for many things... #Init : A command to run on startup. Can be used for many things...
init: "echo ---Initialization over---" init: "echo ---Initialization over---"
#Env : to set environment variables at launch
env:
CAT: "meow"
PATH: /home/cat/bin:$PATH

View File

@ -1,9 +1,10 @@
#[allow(unused_must_use)] #[allow(unused_must_use)]
pub mod shell { pub mod shell {
use regex::Regex;
use std::io::Stdout; use std::io::Stdout;
use std::process::Command; use std::process::Command;
use std::io::stdin; use std::io::stdin;
use std::io::stdout; use std::io::{stdout};
use std::io::Write; use std::io::Write;
use std::path::Path; use std::path::Path;
use std::env; use std::env;
@ -14,7 +15,7 @@ pub mod shell {
use termion::input::TermRead; use termion::input::TermRead;
use termion::raw::IntoRawMode; use termion::raw::IntoRawMode;
use termion::raw::RawTerminal; use termion::raw::RawTerminal;
use termion::cursor; use termion::{color, cursor};
use termion; use termion;
use unicode_segmentation::UnicodeSegmentation; use unicode_segmentation::UnicodeSegmentation;
use std::collections::HashMap; use std::collections::HashMap;
@ -30,55 +31,135 @@ pub mod shell {
//used for home directory //used for home directory
extern crate dirs; extern crate dirs;
extern crate shell_words;
extern crate ctrlc; extern crate ctrlc;
///Stores essential terminal variables
struct TermElements {
//Stores currently typed command
command: String,
//Current cursor position in the line
cur_pos: usize,
//Max position of cursor in the line
max_pos: usize,
stdout: RawTerminal<Stdout>,
//Current history number
current_number: i32,
//Configuration
conf: SqishConf,
}
///Used to unify command output and error handling
///across functions
struct CmdOutput {
outp: String,
rc: i16,
}
impl CmdOutput {
fn from_values(output: String, rc: i16) -> CmdOutput {
let myoutp = CmdOutput {
outp: output,
rc: rc,
};
return myoutp;
}
fn new_empty() -> CmdOutput {
let outp = CmdOutput {
outp: String::new(),
rc: 0,
};
return outp;
}
}
fn export_var(command: String) -> Result<CmdOutput, CmdOutput> {
let cmd_split = command.split('=').collect::<Vec<&str>>();
if cmd_split.len() < 2 {
return Err(CmdOutput::from_values(
String::from("Please a format such as : VAR=value"),
255));
}
let key = String::from(cmd_split[0].to_uppercase());
let val = String::from(cmd_split[1]);
let mut map = HashMap::new();
map.insert(key, val);
set_envvars(&map);
return Ok(CmdOutput::new_empty());
}
fn change_dir(args: &Vec<String>, previous_command: &mut Option<Stdio>) -> Result<CmdOutput, CmdOutput> {
let homedir = dirs::home_dir().unwrap().into_os_string().into_string().unwrap();
//let new_dir = args.peekable().peek().map_or(homedir.as_str(), |x| *x);
let mut new_dir = String::new();
if args.len() > 0 {
*&mut new_dir = args.join(" ");
} else {
*&mut new_dir = String::from(homedir.as_str());
}
let root = Path::new(&new_dir);
if let Err(e) = env::set_current_dir(&root) {
return Err(CmdOutput::from_values(
format!("Error setting directory to {}, got {:?}",
new_dir, e), -1));
};
*previous_command = None;
return Ok(CmdOutput::new_empty());
}
fn print_hist(previous_command: &mut Option<Stdio>, command_next: bool) -> Result<CmdOutput, CmdOutput> {
//let stdout = Stdio::inherit();
let res = get_history();
match res {
Ok(r) => {
let filepath = dirs::home_dir().unwrap().join(".history.sqish");
let file = File::open(filepath).unwrap();
*previous_command = Some(Stdio::from(file));
//if !commands.peek().is_some() {
if !command_next {
print!("{}", r);
}
},
Err(e) => {
return Err(CmdOutput::from_values(format!("Could not print history, got {:?}", e), 255));
},
}
return Ok(CmdOutput::new_empty());
}
fn process_line(input: &str) -> String { fn process_line(input: &str) -> Result<CmdOutput, CmdOutput> {
let mut commands = input.trim().split(" | ").peekable(); let mut commands = input.trim().split("|").peekable();
let mut previous_command = None; let mut previous_command = None;
let mut resultat = String::new(); let mut resultat = String::new();
let mut rc: i16 = 0;
while let Some(command) = commands.next() { while let Some(command) = commands.next() {
let mut parts = command.trim().split_whitespace(); let parts = match shell_words::split(&command.trim()) {
let command = parts.next().unwrap(); Ok(w) => w,
let args = parts; Err(e) => {
return Err(CmdOutput::from_values(format!("Could not parse command {:?}", e), -1));
},
};
if parts.len() < 1 {
return Err(CmdOutput::from_values(String::from("Could not parse command"), -1));
}
let command = parts[0].as_str();
let args = Vec::from(&parts[1..]);
match command { match command {
"cd" => { "cd" => {
let args = args.collect::<Vec<&str>>(); change_dir(&args, &mut previous_command)?;
let homedir = dirs::home_dir().unwrap().into_os_string().into_string().unwrap(); print!("\r\n");
//let new_dir = args.peekable().peek().map_or(homedir.as_str(), |x| *x);
let mut new_dir = String::new();
if args.len() > 0 {
*&mut new_dir = args.join(" ");
} else {
*&mut new_dir = String::from(homedir.as_str());
}
let root = Path::new(&new_dir);
if let Err(e) = env::set_current_dir(&root) {
eprintln!(" Err: {}", e);
} else {
println!("");
}
previous_command = None;
}, },
"history" => { "history" => {
//let stdout = Stdio::inherit(); let next = commands.peek().is_some();
let res = get_history(); print_hist(&mut previous_command, next)?;
match res { },
Ok(r) => { "export" => {
let filepath = dirs::home_dir().unwrap().join(".history.sqish"); export_var(args.join(" "))?;
let file = File::open(filepath).unwrap(); print!("\r\n");
*&mut previous_command = Some(Stdio::from(file));
if !commands.peek().is_some() {
print!("{}", r);
}
},
Err(e) => eprintln!(" Err reading history: {}", e),
}
}, },
command => { command => {
if commands.peek().is_some() { if commands.peek().is_some() {
@ -116,7 +197,7 @@ pub mod shell {
Ok(h) => h, Ok(h) => h,
Err(_) => { Err(_) => {
let err_string = format!("Not found : {}", command); let err_string = format!("Not found : {}", command);
return String::from(err_string); return Err(CmdOutput::from_values(err_string, -1));
}, },
}; };
@ -130,12 +211,15 @@ pub mod shell {
previous_command = None; previous_command = None;
let format_res = format!("{}", let format_res = format!("{}",
status); status);
let _ = &mut resultat.push_str(&format_res); *&mut rc = format_res.parse::<i16>().unwrap();
*&mut resultat = String::from_utf8(output.stdout).unwrap();
//let _ = &mut resultat.push_str(&format_res);
} }
}, },
}; };
} }
return resultat; //return resultat;
return Ok(CmdOutput::from_values(resultat, rc));
} }
fn replace_signs(line: &String) -> String { fn replace_signs(line: &String) -> String {
@ -155,17 +239,18 @@ pub mod shell {
return ayo; return ayo;
} }
fn handle_input(line: &str) -> String { fn handle_input(line: &str) -> Result<CmdOutput, CmdOutput> {
//history callback //history callback
if (line.len() > 0) && (line.starts_with('!')) { if (line.len() > 0) && (line.starts_with('!')) {
let command_found = treat_history_callback(line); let command_found = treat_history_callback(line);
match command_found { match command_found {
Some(c) => { Some(c) => {
write_to_history(&c); write_to_history(&c);
let outp = process_line(&c); return process_line(&c);
return outp;
}, },
None => return String::from(" "), None => return Err(CmdOutput::from_values(
String::from("No such value"),
255)),
}; };
//Regular command //Regular command
} else if line.len() > 0 { } else if line.len() > 0 {
@ -174,7 +259,7 @@ pub mod shell {
//Nothing //Nothing
} else { } else {
//Command was empty, new line //Command was empty, new line
return String::from(""); return Ok(CmdOutput::new_empty());
} }
} }
@ -248,31 +333,40 @@ pub mod shell {
} }
fn run_cmd(mycommand: &mut String, fn run_cmd(mycommand: &mut String,
current_number: &mut i32, current_number: &mut i32,
conf: &mut SqishConf, conf: &mut SqishConf,
stdout: &mut RawTerminal<Stdout>) { stdout: &mut RawTerminal<Stdout>) {
//NORMAL CMD
if (mycommand != &String::from("")) && (mycommand != &String::from("exit")) { if (mycommand != &String::from("")) && (mycommand != &String::from("exit")) {
//Run the command
let comm = replace_signs(&mycommand); let comm = replace_signs(&mycommand);
RawTerminal::suspend_raw_mode(&stdout); RawTerminal::suspend_raw_mode(&stdout);
let res = handle_input(&comm); let res = handle_input(&comm);
//Display the results
match res {
Ok(_) => (),
Err(e) => {
println!("\r\n{}Got error {}, RC : {:?}{}",
color::Fg(color::Red),
e.outp,
e.rc,
color::Fg(color::Reset));
},
};
//Prepare for a new command
RawTerminal::activate_raw_mode(&stdout); RawTerminal::activate_raw_mode(&stdout);
mycommand.clear(); mycommand.clear();
for line in res.split("\r\n") {
if line != "" && line.starts_with('N'){
write!(stdout, "{}\r\n", line);
}
}
conf.update_prompt(get_curr_history_number()); conf.update_prompt(get_curr_history_number());
write!(stdout, "{}", conf.promptline).unwrap(); write!(stdout, "{}", conf.promptline).unwrap();
stdout.flush(); stdout.flush();
*current_number += 1; *current_number += 1;
//EXITTING
} else if mycommand == &String::from("exit") { } else if mycommand == &String::from("exit") {
write!(stdout, "\r\nThanks for using Sqish!\r\nSayonara \r\n"); write!(stdout, "\r\nThanks for using Sqish!\r\nSayonara \r\n");
RawTerminal::suspend_raw_mode(&stdout); RawTerminal::suspend_raw_mode(&stdout);
std::process::exit(0); std::process::exit(0);
//EMPTY LINE
} else { } else {
conf.update_prompt(get_curr_history_number()); conf.update_prompt(get_curr_history_number());
write!(stdout, "\r\n{}", conf.promptline).unwrap(); write!(stdout, "\r\n{}", conf.promptline).unwrap();
@ -314,27 +408,26 @@ pub mod shell {
}; };
} }
pub fn run_raw() { fn find_next_space(current_pos: &usize, max_pos: &usize, command: &String) -> usize {
let stdin = stdin(); let mut steps: usize = 0;
let mut stdout = stdout().into_raw_mode().unwrap(); if *current_pos == *max_pos { return steps; }
//RawTerminal::suspend_raw_mode(&stdout); for letter in command.chars().skip(*current_pos) {
let mut mycommand = String::new(); //println!("L : {:?}, S : {:?} C: {:?} M {:?}", letter, steps, current_pos, max_pos);
//Skip if we start on a space
if steps == 0 && (letter == ' ' || letter == ',' || letter == ';') {
continue;
} else if letter != ' ' && *current_pos + steps < *max_pos {
steps += 1;
} else {
return steps;
}
}
let max: usize = (0 + max_pos) - current_pos;
return max;
}
//Used in conjunction with Up arrow to go back in history fn handle_conf(stdout: &mut Stdout) -> SqishConf {
//Resetted by pressing Enter let conf = match SqishConf::from_sqishrc() {
let mut current_number = get_curr_history_number();
//Used to track the location of the cursor inside the command
let mut current_pos: usize = 0;
let mut max_pos: usize = 0;
//Handle Ctrl+C
ctrlc::set_handler(|| {
();
}).unwrap();
let mut conf = match SqishConf::from_sqishrc() {
Ok(c) => c, Ok(c) => c,
Err(e) => { Err(e) => {
let conf = SqishConf { let conf = SqishConf {
@ -343,6 +436,7 @@ pub mod shell {
aliases: HashMap::new(), aliases: HashMap::new(),
hotkeys: HashMap::new(), hotkeys: HashMap::new(),
init: String::new(), init: String::new(),
env: HashMap::new(),
}; };
let ret_line = format!("Could not build conf, got {}\ let ret_line = format!("Could not build conf, got {}\
\r\nUsing default promptline", e); \r\nUsing default promptline", e);
@ -350,222 +444,362 @@ pub mod shell {
conf conf
}, },
}; };
&conf.update_prompt(get_curr_history_number()); return conf;
}
fn set_ctrlc() {
ctrlc::set_handler(|| {
();
}).unwrap();
}
fn tab(elems: &mut TermElements) {
//Do NOT search on an empty command
if *&elems.command.len() == 0 {
return;
}
//Search
*&mut elems.command = replace_signs(&elems.command);
let (res, list) = Search::build(&elems.command).unwrap().autocomplete();
if list.len() > 1 { write!(elems.stdout, "\r\n{}", list); }
elems.command.clear();
elems.command.push_str(&res);
elems.max_pos = res.len();
elems.cur_pos = elems.max_pos;
write!(elems.stdout, "\r\n{}{}", elems.conf.promptline, res);
elems.stdout.flush();
}
fn enter(elems: &mut TermElements) {
elems.current_number = get_curr_history_number();
elems.cur_pos = 0;
elems.max_pos = 0;
*&mut elems.command = transform_alias(&elems.conf, elems.command.clone());
run_cmd(&mut elems.command,
&mut elems.current_number,
&mut elems.conf,
&mut elems.stdout);
}
fn cmd_clear(elems: &mut TermElements) {
elems.cur_pos = 0;
elems.max_pos = 0;
elems.current_number = get_curr_history_number();
elems.command.clear();
write!(elems.stdout, "\r\n--CANCEL--\r\n");
write!(elems.stdout, "{}", elems.conf.promptline);
}
fn undo_line(elems: &mut TermElements) {
clear_line(&elems.max_pos, &elems.cur_pos);
elems.max_pos = 0;
elems.cur_pos = 0;
elems.command.clear();
}
fn jmp_start(elems: &mut TermElements) {
if elems.cur_pos != 0 {
write!(elems.stdout, "{}", cursor::Left(elems.cur_pos as u16));
elems.cur_pos = 0;
}
}
fn jmp_end(elems: &mut TermElements) {
if elems.cur_pos != elems.max_pos {
let chars_left = elems.max_pos - elems.cur_pos;
write!(elems.stdout, "{}", cursor::Right(chars_left as u16));
elems.cur_pos = elems.max_pos;
}
}
fn press_del(elems: &mut TermElements) {
if elems.command.chars().count() == 1
&& elems.cur_pos == 0 {
clear_line(&elems.max_pos, &elems.cur_pos);
elems.command.clear();
elems.max_pos = 0;
elems.cur_pos = 0;
} else if elems.cur_pos < elems.max_pos {
elems.command.remove(elems.cur_pos);
if elems.cur_pos > 0 {
elems.cur_pos -= 1;
write!(elems.stdout, "{}", cursor::Left(1));
}
elems.max_pos -= 1;
write!(elems.stdout, "{}", cursor::Save);
clear_line(&elems.max_pos, &elems.cur_pos);
write!(elems.stdout, "{}", elems.command);
write!(elems.stdout, "{}", cursor::Restore);
elems.stdout.flush();
}
}
fn backspace(elems: &mut TermElements) {
if elems.cur_pos == 0 {
return;
}
if elems.command.chars().count() == 1 {
clear_line(&elems.max_pos, &elems.cur_pos);
elems.command.clear();
elems.cur_pos = 0;
elems.max_pos = 0;
} else {
elems.cur_pos -= 1;
elems.max_pos -= 1;
elems.command.remove(elems.cur_pos);
write!(elems.stdout, "{}", cursor::Left(1));
write!(elems.stdout, "{}", cursor::Save);
clear_line(&elems.max_pos, &elems.cur_pos);
write!(elems.stdout, "{}", elems.command);
write!(elems.stdout, "{}", cursor::Restore);
elems.stdout.flush();
}
}
fn uparrow(elems: &mut TermElements) {
if elems.current_number > 2 {
elems.command = delete_current_cmd(elems.command.clone());
elems.current_number -= 1;
let lastcmd = match get_hist_from_number(&elems.current_number) {
Some(c) => c,
None => return
};
elems.command = lastcmd.trim().to_string();
(elems.max_pos, elems.cur_pos) = get_cmd_curs_pos(&elems.command);
write!(elems.stdout, "{}", elems.command);
}
}
fn downarrow(elems: &mut TermElements) {
if elems.current_number < get_curr_history_number() {
elems.command = delete_current_cmd(elems.command.clone());
elems.current_number += 1;
let lastcmd = match get_hist_from_number(&elems.current_number) {
Some(c) => c,
None => return
};
elems.command = lastcmd.trim().to_string();
(elems.max_pos, elems.cur_pos) = get_cmd_curs_pos(&elems.command);
write!(elems.stdout, "{}", elems.command);
}
}
fn leftarrow(elems: &mut TermElements) {
if elems.cur_pos > 0 {
elems.cur_pos -= 1;
write!(elems.stdout, "{}", cursor::Left(1));
}
}
fn rightarrow(elems: &mut TermElements) {
if elems.cur_pos < elems.max_pos {
print!("{}", cursor::Right(1));
elems.cur_pos += 1;
}
}
fn screen_clear(elems: &mut TermElements) {
write!(elems.stdout, "{}{}", termion::clear::All, termion::cursor::Goto(1,1));
write!(elems.stdout, "{}{}", elems.conf.promptline, elems.command);
}
fn jmp_nxt_word(elems: &mut TermElements) {
let steps = find_next_space(&elems.cur_pos, &elems.max_pos, &elems.command);
//println!("steps: {:?} cur {:?} max {:?}", steps, elems.cur_pos, elems.max_pos);
if steps > 0 {
print!("{}", cursor::Right(steps as u16));
elems.cur_pos += steps;
}
}
fn jmp_prev_word(elems: &mut TermElements) {
let command_rev = elems.command.chars().rev().collect::<String>();
let curr_rev = elems.max_pos - elems.cur_pos + 1;
let steps = find_next_space(&curr_rev, &elems.max_pos, &command_rev);
if steps > 0 {
print!("{}", cursor::Left(steps as u16));
elems.cur_pos -= steps;
}
}
fn press_alt(elems: &mut TermElements, x: char) {
match x {
'a'..='z' => {
let x = String::from(x);
if elems.conf.hotkeys.contains_key(&x) {
let hotcmd = &elems.conf.hotkeys[&x].clone();
for c in hotcmd.chars() {
if c != '\n' {
write_letter(&mut elems.command,
&mut elems.cur_pos,
&mut elems.max_pos,
c);
} else {
elems.cur_pos = 0;
elems.max_pos = 0;
elems.command = transform_alias(&elems.conf, elems.command.clone());
run_cmd(&mut elems.command,
&mut elems.current_number,
&mut elems.conf,
&mut elems.stdout);
}
}
}
},
'.' => {
append_prev_arg(&mut elems.command, &mut elems.cur_pos, &mut elems.max_pos);
},
'!' => {
let mut cmd = format!("\r\n-- HOTKEYS --> \r\n{:?}\r\n", elems.conf.hotkeys);
cmd = format!("{}\r\n-- ALIASES -->\r\n{:?}\r\n", cmd, elems.conf.aliases);
cmd = cmd.replace(",", "\r\n");
cmd = cmd.replace('"', "");
write!(elems.stdout, "{}", cmd);
},
_ => (),
};
}
//I smell horrible code in here
fn set_envvars(vars: &HashMap<String, String>) {
let vars = vars.clone();
let re = Regex::new(r"\$[A-Za-z_]+").unwrap();
for (key, value) in vars {
let mut after = value.clone();
let mut nbr_of_vars = *&value.matches('$').count();
while nbr_of_vars >= 1 {
let temp = after.clone();
let caps = re.captures(&temp).unwrap();
for cap in caps.iter() {
match cap {
Some(c) => {
let myvar = String::from(c.as_str()).replace('$', "");
match env::var(myvar) {
Ok(r) => {
*&mut after = after.replace(c.as_str(), &r);
},
Err(_) => continue,
};
},
None => continue,
};
}
nbr_of_vars -= 1;
}
env::set_var(&key, &after);
}
}
//THE ENTRYPOINT FUNCTION
pub fn run_raw() {
let mut elems = TermElements {
command: String::new(),
cur_pos: 0,
max_pos: 0,
stdout: stdout().into_raw_mode().unwrap(),
current_number: get_curr_history_number(),
conf: handle_conf(&mut stdout().into_raw_mode().unwrap()),
};
let stdin = stdin();
//Handle Ctrl+C
set_ctrlc();
&elems.conf.update_prompt(get_curr_history_number());
//Initializing //Initializing
write!(stdout, "\r\n ---Sqish initializing...--- \r\n{}", conf.promptline); write!(elems.stdout, "\r\n ---Sqish initializing--- \r\n{}", elems.conf.promptline);
stdout.flush(); elems.stdout.flush();
set_envvars(&elems.conf.env);
if conf.init != String::from("") { if elems.conf.init != String::from("") {
run_cmd(&mut String::from(&conf.init), &mut current_number, &mut conf, &mut stdout); run_cmd(&mut String::from(&elems.conf.init), &mut elems.current_number, &mut elems.conf, &mut elems.stdout);
} }
for c in stdin.keys() { for c in stdin.keys() {
let k = c.unwrap(); let k = c.unwrap();
match k { match k {
Key::Char('\t') => { Key::Char('\t') => {
//Do NOT search on an empty command tab(&mut elems);
if *&mycommand.len() == 0 {
continue;
}
//Search
*&mut mycommand = replace_signs(&mycommand);
let (res, list) = Search::build(&mycommand).unwrap().autocomplete();
if list.len() > 0 { write!(stdout, "\r\n{}\r\n", list); }
mycommand.clear();
mycommand.push_str(&res);
max_pos = res.len();
current_pos = max_pos;
write!(stdout, "\r\n{}{}", conf.promptline, res);
stdout.flush();
} }
Key::Char('\n') => { Key::Char('\n') => {
current_number = get_curr_history_number(); enter(&mut elems);
current_pos = 0;
max_pos = 0;
mycommand = transform_alias(&conf, mycommand);
run_cmd(&mut mycommand,
&mut current_number,
&mut conf,
&mut stdout);
}, },
Key::Ctrl('q'|'d') => { Key::Ctrl('q'|'d') => {
RawTerminal::suspend_raw_mode(&stdout); writeln!(elems.stdout, "\r\nThanks for using Sqish !\r\nSayonara \r\n");
writeln!(stdout, "\r\nThanks for using Sqish !\r\nSayonara \r\n");
break; break;
}, },
Key::Ctrl('c') => { Key::Ctrl('c') => {
current_pos = 0; cmd_clear(&mut elems);
max_pos = 0;
current_number = get_curr_history_number();
mycommand.clear();
write!(stdout, "\r\n--CANCEL--\r\n");
write!(stdout, "{}", conf.promptline);
}, },
Key::Ctrl('u') => { Key::Ctrl('u') => {
clear_line(&max_pos, &current_pos); undo_line(&mut elems);
max_pos = 0;
current_pos = 0;
mycommand.clear();
}, },
Key::Ctrl('a') => { Key::Ctrl('a') => {
if current_pos != 0 { jmp_start(&mut elems);
write!(stdout, "{}", cursor::Left(current_pos as u16));
current_pos = 0;
}
}, },
Key::Ctrl('e') => { Key::Ctrl('e') => {
if current_pos != max_pos { jmp_end(&mut elems);
let chars_left = max_pos - current_pos;
write!(stdout, "{}", cursor::Right(chars_left as u16));
current_pos = max_pos;
}
}, },
Key::Char(c) => { Key::Char(c) => {
write_letter(&mut mycommand, &mut current_pos, &mut max_pos, c); write_letter(&mut elems.command, &mut elems.cur_pos, &mut elems.max_pos, c);
} }
Key::Delete => { Key::Delete => {
if mycommand.chars().count() == 1 press_del(&mut elems);
&& current_pos == 0 {
clear_line(&max_pos, &current_pos);
mycommand.clear();
max_pos = 0;
current_pos = 0;
} else if current_pos < max_pos {
mycommand.remove(current_pos);
if current_pos > 0 {
current_pos -= 1;
write!(stdout, "{}", cursor::Left(1));
}
max_pos -= 1;
write!(stdout, "{}", cursor::Save);
clear_line(&max_pos, &current_pos);
write!(stdout, "{}", mycommand);
write!(stdout, "{}", cursor::Restore);
stdout.flush();
}
}, },
Key::Backspace => { Key::Backspace => {
if current_pos == 0 { backspace(&mut elems);
continue;
}
if mycommand.chars().count() == 1 {
clear_line(&max_pos, &current_pos);
mycommand.clear();
current_pos = 0;
max_pos = 0;
} else {
current_pos -= 1;
max_pos -= 1;
mycommand.remove(current_pos);
write!(stdout, "{}", cursor::Left(1));
write!(stdout, "{}", cursor::Save);
clear_line(&max_pos, &current_pos);
write!(stdout, "{}", mycommand);
write!(stdout, "{}", cursor::Restore);
stdout.flush();
}
}, },
Key::Up => { Key::Up => {
if current_number > 2 { uparrow(&mut elems);
mycommand = delete_current_cmd(mycommand);
current_number -= 1;
let lastcmd = match get_hist_from_number(&current_number) {
Some(c) => c,
None => continue
};
mycommand = lastcmd.trim().to_string();
(max_pos, current_pos) = get_cmd_curs_pos(&mycommand);
write!(stdout, "{}", mycommand);
}
}, },
Key::Down => { Key::Down => {
if current_number < get_curr_history_number() { downarrow(&mut elems);
mycommand = delete_current_cmd(mycommand);
current_number += 1;
let lastcmd = match get_hist_from_number(&current_number) {
Some(c) => c,
None => continue
};
mycommand = lastcmd.trim().to_string();
(max_pos, current_pos) = get_cmd_curs_pos(&mycommand);
write!(stdout, "{}", mycommand);
}
}, },
Key::Ctrl('x') => {
clear_line(&max_pos, &current_pos);
mycommand.clear();
current_pos = 0;
max_pos = 0;
}
Key::Left => { Key::Left => {
if current_pos > 0 { leftarrow(&mut elems);
current_pos -= 1;
write!(stdout, "{}", cursor::Left(1));
}
}, },
Key::Right => { Key::Right => {
if current_pos < max_pos { rightarrow(&mut elems);
print!("{}", cursor::Right(1)); },
current_pos += 1;
} Key::Ctrl('l') => {
screen_clear(&mut elems);
},
Key::Ctrl('p') => {
jmp_nxt_word(&mut elems);
},
Key::Ctrl('o') => {
jmp_prev_word(&mut elems);
}, },
Key::Alt(x) => { Key::Alt(x) => {
match x { press_alt(&mut elems, x);
'a'..='z' => {
let x = String::from(x);
if conf.hotkeys.contains_key(&x) {
let hotcmd = &conf.hotkeys[&x].clone();
for c in hotcmd.chars() {
if c != '\n' {
write_letter(&mut mycommand,
&mut current_pos,
&mut max_pos,
c);
} else {
current_pos = 0;
max_pos = 0;
mycommand = transform_alias(&conf, mycommand);
run_cmd(&mut mycommand,
&mut current_number,
&mut conf,
&mut stdout);
}
}
}
},
'.' => {
append_prev_arg(&mut mycommand, &mut current_pos, &mut max_pos);
},
'!' => {
let mut cmd = format!("\r\n-- ALIASES --> \r\n{:?}\r\n", conf.hotkeys);
cmd = cmd.replace(",", "\r\n");
write!(stdout, "{}", cmd);
},
_ => (),
};
}, },
_ => (), _ => (),
}; };
stdout.flush().unwrap(); elems.stdout.flush().unwrap();
} }
} }

View File

@ -25,6 +25,8 @@ pub struct Search {
local_search: bool, local_search: bool,
//the kind of search //the kind of search
search_type: SearchType, search_type: SearchType,
//the original command
original: String,
} }
impl Search { impl Search {
@ -83,6 +85,7 @@ impl Search {
searchee: searchee, searchee: searchee,
local_search: local, local_search: local,
search_type: search_type, search_type: search_type,
original: String::from(input),
}; };
return Ok(s); return Ok(s);
} }
@ -141,6 +144,8 @@ fn autocomplete_file(search: &Search) -> std::io::Result<(String, String)> {
let mut last_found = String::new(); let mut last_found = String::new();
let mut results = Vec::new(); let mut results = Vec::new();
//This is pretty messy.
for file in fs::read_dir(search.search_path.clone())? { for file in fs::read_dir(search.search_path.clone())? {
let filepath = file.unwrap().path(); let filepath = file.unwrap().path();
let filename = filepath let filename = filepath
@ -192,18 +197,44 @@ fn autocomplete_file(search: &Search) -> std::io::Result<(String, String)> {
*&mut last_found = search.searchee.clone(); *&mut last_found = search.searchee.clone();
} }
//Handle the path when tabbing in a long path, zsh-like //Not looking for anything in particular, this is a special case.
//then return the value if search.searchee == "" {
let xxx = &search.search_path.clone().into_os_string().into_string().unwrap(); return Ok((String::from(&search.original), all_res));
let mut return_val = String::new();
if !&search.local_search && !&last_found.contains(&xxx.as_str()) {
*&mut return_val = format!("{}{}{}", &search.command, &search.search_path.display(), last_found);
} else {
*&mut return_val = format!("{}{}", &search.command, last_found);
} }
//Remote search... is annoying
if !search.local_search {
last_found = last_found.replace("./", "");
return Ok((format!("{}{}{}", search.command, search.search_path.display(), last_found), String::new()));
}
last_found = last_found.replace("./", "");
//Otherwise... Handling the return as gracefully as I can
let mut return_val = String::new();
if all_res.len() > 0 {
let ret_string = replace_last_word(&String::from(&search.original), last_found);
*&mut return_val = format!("{}", ret_string);
} else {
*&mut return_val = format!("{}", &search.original);
}
return Ok((return_val, all_res)); return Ok((return_val, all_res));
} }
//Matches on space or slash
fn replace_last_word(sentence: &String, replacement: String) -> String {
let lastword = sentence.split(|c| c == ' ' || c == '/').last().unwrap_or("");
let mut work_sent = sentence.clone();
work_sent.truncate(sentence.len() - lastword.len());
work_sent.push_str(&replacement);
return work_sent;
}
//fn return_last_word(sentence: &String) -> String {
// let lastword = sentence.split(|c| c == ' ' || c == '/').last().unwrap_or("");
// return String::from(lastword);
//}
fn autocomplete_cmd(input: &String, search: &Search) -> (String, String) { fn autocomplete_cmd(input: &String, search: &Search) -> (String, String) {
let found_bins = find_bin(input); let found_bins = find_bin(input);

236
src/shell/commandrun.rs Normal file
View File

@ -0,0 +1,236 @@
use std::io::Stdout;
use std::process::Command;
use std::io::stdin;
use std::io::{stdout};
use std::io::Write;
use std::path::Path;
use std::env;
use std::str;
use std::fs::File;
use std::process::Stdio;
use termion::event::Key;
use termion::input::TermRead;
use termion::raw::IntoRawMode;
use termion::raw::RawTerminal;
use termion::{color, cursor};
use termion;
use unicode_segmentation::UnicodeSegmentation;
use std::collections::HashMap;
mod history;
use crate::shell::history::*;
///Used to unify command output and error handling
///across functions
pub struct CmdOutput {
pub outp: String,
pub rc: i16,
}
impl CmdOutput {
pub fn from_values(output: String, rc: i16) -> CmdOutput {
let myoutp = CmdOutput {
outp: output,
rc: rc,
};
return myoutp;
}
pub fn new_empty() -> CmdOutput {
let outp = CmdOutput {
outp: String::new(),
rc: 0,
};
return outp;
}
}
fn change_dir(args: &Vec<String>, previous_command: &mut Option<Stdio>) -> Result<CmdOutput, CmdOutput> {
let homedir = dirs::home_dir().unwrap().into_os_string().into_string().unwrap();
//let new_dir = args.peekable().peek().map_or(homedir.as_str(), |x| *x);
let mut new_dir = String::new();
if args.len() > 0 {
*&mut new_dir = args.join(" ");
} else {
*&mut new_dir = String::from(homedir.as_str());
}
let root = Path::new(&new_dir);
if let Err(e) = env::set_current_dir(&root) {
return Err(CmdOutput::from_values(
format!("Error setting directory to {}, got {:?}",
new_dir, e), -1));
};
*previous_command = None;
return Ok(CmdOutput::new_empty());
}
fn print_hist(previous_command: &mut Option<Stdio>, command_next: bool) -> Result<CmdOutput, CmdOutput> {
//let stdout = Stdio::inherit();
let res = get_history();
match res {
Ok(r) => {
let filepath = dirs::home_dir().unwrap().join(".history.sqish");
let file = File::open(filepath).unwrap();
*previous_command = Some(Stdio::from(file));
//if !commands.peek().is_some() {
if !command_next {
print!("{}", r);
}
},
Err(e) => {
return Err(CmdOutput::from_values(format!("Could not print history, got {:?}", e), 255));
},
}
return Ok(CmdOutput::new_empty());
}
fn process_line(input: &str) -> Result<CmdOutput, CmdOutput> {
let mut commands = input.trim().split("|").peekable();
let mut previous_command = None;
let mut resultat = String::new();
let mut rc: i16 = 0;
while let Some(command) = commands.next() {
let parts = match shell_words::split(&command.trim()) {
Ok(w) => w,
Err(e) => {
return Err(CmdOutput::from_values(format!("Could not parse command {:?}", e), -1));
},
};
let command = parts[0].as_str();
let args = Vec::from(&parts[1..]);
match command {
"cd" => {
change_dir(&args, &mut previous_command)?;
},
"history" => {
let next = commands.peek().is_some();
print_hist(&mut previous_command, next)?;
},
command => {
if commands.peek().is_some() {
let stdin = match previous_command {
None => Stdio::inherit(),
Some(o) => o,
};
let stdout = Stdio::piped();
let output = Command::new(command)
.args(args)
.stdout(stdout)
.stdin(stdin)
.spawn();
match output {
Ok(o) => previous_command = Some(Stdio::from(o.stdout.unwrap())),
Err(e) => {
previous_command = None;
eprintln!(" Err: {}", e);
},
};
} else {
let stdin = match previous_command {
None => Stdio::inherit(),
Some(o) => o,
};
print!("\r\n");
let child = match Command::new(command)
.args(args)
.stdin(stdin)
.spawn() {
Ok(h) => h,
Err(_) => {
let err_string = format!("Not found : {}", command);
return Err(CmdOutput::from_values(err_string, -1));
},
};
let output = child
.wait_with_output()
.expect("Failed to wait on child");
let status = output.status
.code()
.unwrap_or(255)
.to_string();
previous_command = None;
let format_res = format!("{}",
status);
*&mut rc = format_res.parse::<i16>().unwrap();
*&mut resultat = String::from_utf8(output.stdout).unwrap();
//let _ = &mut resultat.push_str(&format_res);
}
},
};
}
//return resultat;
return Ok(CmdOutput::from_values(resultat, rc));
}
pub fn run_cmd(mycommand: &mut String,
current_number: &mut i32,
conf: &mut SqishConf,
stdout: &mut RawTerminal<Stdout>) {
//NORMAL CMD
if (mycommand != &String::from("")) && (mycommand != &String::from("exit")) {
//Run the command
let comm = replace_signs(&mycommand);
RawTerminal::suspend_raw_mode(&stdout);
let res = handle_input(&comm);
//Display the results
match res {
Ok(_) => (),
Err(e) => {
println!("\r\n{}Got error {}, RC : {:?}{}",
color::Fg(color::Red),
e.outp,
e.rc,
color::Fg(color::Reset));
},
};
//Prepare for a new command
RawTerminal::activate_raw_mode(&stdout);
mycommand.clear();
conf.update_prompt(get_curr_history_number());
write!(stdout, "{}", conf.promptline).unwrap();
stdout.flush();
*current_number += 1;
//EXITTING
} else if mycommand == &String::from("exit") {
write!(stdout, "\r\nThanks for using Sqish!\r\nSayonara \r\n");
RawTerminal::suspend_raw_mode(&stdout);
std::process::exit(0);
//EMPTY LINE
} else {
conf.update_prompt(get_curr_history_number());
write!(stdout, "\r\n{}", conf.promptline).unwrap();
}
stdout.flush();
}
pub fn handle_input(line: &str) -> Result<CmdOutput, CmdOutput> {
//history callback
if (line.len() > 0) && (line.starts_with('!')) {
let command_found = treat_history_callback(line);
match command_found {
Some(c) => {
write_to_history(&c);
return process_line(&c);
},
None => return Err(CmdOutput::from_values(
String::from("No such value"),
255)),
};
//Regular command
} else if line.len() > 0 {
write_to_history(line);
return process_line(line);
//Nothing
} else {
//Command was empty, new line
return Ok(CmdOutput::new_empty());
}
}

View File

@ -1,5 +1,5 @@
use users::{get_user_by_uid, get_current_uid}; use users::{get_user_by_uid, get_current_uid};
use termion::color; use termion::{color, style};
use std::{env, fs}; use std::{env, fs};
use std::collections::HashMap; use std::collections::HashMap;
use regex::Regex; use regex::Regex;
@ -16,6 +16,7 @@ pub struct SqishConf {
pub aliases: HashMap<String, String>, pub aliases: HashMap<String, String>,
pub hotkeys: HashMap<String, String>, pub hotkeys: HashMap<String, String>,
pub init: String, pub init: String,
pub env: HashMap<String, String>,
} }
impl SqishConf { impl SqishConf {
@ -54,12 +55,16 @@ impl SqishConf {
let hotkeys_yaml = &sqishrc["hotkeys"]; let hotkeys_yaml = &sqishrc["hotkeys"];
let hotkeys = Self::yaml_dict2hashmap(hotkeys_yaml); let hotkeys = Self::yaml_dict2hashmap(hotkeys_yaml);
let env_yaml = &sqishrc["env"];
let env = Self::yaml_dict2hashmap(env_yaml);
let mut out_conf = SqishConf { let mut out_conf = SqishConf {
promptline: out_str.clone(), promptline: out_str.clone(),
promptline_base: out_str, promptline_base: out_str,
aliases: aliases, aliases: aliases,
hotkeys: hotkeys, hotkeys: hotkeys,
init: startup, init: startup,
env: env,
}; };
out_conf.handle_rgb(); out_conf.handle_rgb();
@ -112,6 +117,7 @@ impl SqishConf {
fn handle_colors(&mut self) { fn handle_colors(&mut self) {
//Colors
let reset = format!("{}", color::Fg(color::Reset)); let reset = format!("{}", color::Fg(color::Reset));
let green = format!("{}", color::Fg(color::Green)); let green = format!("{}", color::Fg(color::Green));
let blue = format!("{}", color::Fg(color::Blue)); let blue = format!("{}", color::Fg(color::Blue));
@ -121,6 +127,12 @@ impl SqishConf {
let cyan = format!("{}", color::Fg(color::Cyan)); let cyan = format!("{}", color::Fg(color::Cyan));
let lightblack = format!("{}", color::Fg(color::LightBlack)); let lightblack = format!("{}", color::Fg(color::LightBlack));
//Styles
let ita = format!("{}", style::Italic);
let bold = format!("{}", style::Bold);
let stylereset = format!("{}", style::Reset);
//Colors replace
let mut prompt = self.promptline.replace("$COLORGREEN_", &green); let mut prompt = self.promptline.replace("$COLORGREEN_", &green);
prompt = prompt.replace("$COLORBLUE_", &blue); prompt = prompt.replace("$COLORBLUE_", &blue);
prompt = prompt.replace("$COLORRED_", &red); prompt = prompt.replace("$COLORRED_", &red);
@ -129,6 +141,11 @@ impl SqishConf {
prompt = prompt.replace("$COLORCYAN_", &cyan); prompt = prompt.replace("$COLORCYAN_", &cyan);
prompt = prompt.replace("$COLORLBLACK_", &lightblack); prompt = prompt.replace("$COLORLBLACK_", &lightblack);
prompt = prompt.replace("$COLORRESET_", &reset); prompt = prompt.replace("$COLORRESET_", &reset);
//Styles replace
prompt = prompt.replace("$ITA_", &ita);
prompt = prompt.replace("$BOLD_", &bold);
prompt = prompt.replace("$STYLERESET_", &stylereset);
let promptown = prompt.to_owned(); let promptown = prompt.to_owned();
self.promptline = promptown; self.promptline = promptown;
} }

View File

@ -94,7 +94,7 @@ pub fn get_hist_from_number(number: &i32) -> Option<String> {
} }
pub fn treat_history_callback(line: &str) -> Option<String> { pub fn treat_history_callback(line: &str) -> Option<String> {
let mut mystring = String::from(line); let mystring = String::from(line);
let temp = mystring.split_whitespace().collect::<Vec<&str>>(); let temp = mystring.split_whitespace().collect::<Vec<&str>>();
let mystring = temp.first().unwrap_or(&line); let mystring = temp.first().unwrap_or(&line);
let mut chars = mystring.chars(); let mut chars = mystring.chars();