|
@@ -0,0 +1,675 @@
|
|
|
+use help::Help;
|
|
|
+use regex::{Captures, Regex};
|
|
|
+use std::io::Read;
|
|
|
+use std::os::unix::ffi::OsStrExt;
|
|
|
+use std::{
|
|
|
+ format,
|
|
|
+ fs::{self, File, OpenOptions},
|
|
|
+ io::Write,
|
|
|
+ path::Path,
|
|
|
+ print, println,
|
|
|
+ string::String,
|
|
|
+ vec::Vec,
|
|
|
+};
|
|
|
+
|
|
|
+use crate::shell::Shell;
|
|
|
+use crate::Env;
|
|
|
+use crate::{special_keycode::*, ROOT_PATH};
|
|
|
+
|
|
|
+mod help;
|
|
|
+
|
|
|
+enum CommandType {
|
|
|
+ InternalCommand(BuildInCmd),
|
|
|
+ ExternalCommand(String),
|
|
|
+}
|
|
|
+
|
|
|
+pub struct Command {
|
|
|
+ args: Vec<String>,
|
|
|
+ cmd_type: CommandType,
|
|
|
+}
|
|
|
+
|
|
|
+#[derive(Debug, PartialEq, Eq, Clone)]
|
|
|
+pub enum CommandError {
|
|
|
+ CommandNotFound(String),
|
|
|
+ InvalidArgument(String),
|
|
|
+ WrongArgumentCount(usize),
|
|
|
+ EnvironmentVariableNotFound(String),
|
|
|
+ PathNotFound(String),
|
|
|
+ FileNotFound(String),
|
|
|
+ DirectoryNotFound(String),
|
|
|
+ NotDirectory(String),
|
|
|
+ NotFile(String),
|
|
|
+}
|
|
|
+
|
|
|
+impl CommandError {
|
|
|
+ pub fn handle(e: CommandError) {
|
|
|
+ match e {
|
|
|
+ CommandError::CommandNotFound(command) => {
|
|
|
+ println!("cannot find command: {}", command)
|
|
|
+ }
|
|
|
+ CommandError::InvalidArgument(argument) => {
|
|
|
+ println!("invalid argument: {}", argument)
|
|
|
+ }
|
|
|
+ CommandError::WrongArgumentCount(count) => {
|
|
|
+ println!("argument count incorrect: {}", count)
|
|
|
+ }
|
|
|
+ CommandError::EnvironmentVariableNotFound(env) => {
|
|
|
+ println!("environment variable not found: {}", env);
|
|
|
+ }
|
|
|
+ CommandError::PathNotFound(path) => {
|
|
|
+ println!("cannot found file or dirctory: {}", path)
|
|
|
+ }
|
|
|
+ CommandError::FileNotFound(path) => {
|
|
|
+ println!("cannot found file: {}", path)
|
|
|
+ }
|
|
|
+ CommandError::DirectoryNotFound(path) => {
|
|
|
+ println!("cannot found dirctory: {}", path)
|
|
|
+ }
|
|
|
+ CommandError::NotDirectory(path) => {
|
|
|
+ println!("path is not a dirctory: {}", path)
|
|
|
+ }
|
|
|
+ CommandError::NotFile(path) => {
|
|
|
+ println!("path is not a file: {}", path)
|
|
|
+ }
|
|
|
+ };
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+impl Command {
|
|
|
+ fn new(name: String, args: Vec<String>) -> Result<Command, CommandError> {
|
|
|
+ for BuildInCmd(cmd) in BuildInCmd::BUILD_IN_CMD {
|
|
|
+ if name == *cmd {
|
|
|
+ return Ok(Command {
|
|
|
+ args,
|
|
|
+ cmd_type: CommandType::InternalCommand(BuildInCmd(cmd)),
|
|
|
+ });
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ return Ok(Command {
|
|
|
+ args,
|
|
|
+ cmd_type: CommandType::ExternalCommand(name),
|
|
|
+ });
|
|
|
+ }
|
|
|
+
|
|
|
+ fn from_string(str: String) -> Result<Command, CommandError> {
|
|
|
+ let regex: Regex = Regex::new(r#"([^\s'"]|("[^"]*"|'[^']*'))+"#).unwrap();
|
|
|
+ let hay = str.clone();
|
|
|
+ let mut iter = regex
|
|
|
+ .captures_iter(hay.as_str())
|
|
|
+ .map(|c| String::from(c.get(0).unwrap().as_str()));
|
|
|
+ // let mut iter = str.split_ascii_whitespace();
|
|
|
+ let name = iter.next().unwrap();
|
|
|
+ let re: Regex = Regex::new(r"\$[\w_]+").unwrap();
|
|
|
+ let replacement = |caps: &Captures| -> String {
|
|
|
+ match Env::get(&String::from(&caps[0][1..])) {
|
|
|
+ Some(value) => value,
|
|
|
+ None => String::from(&caps[0]),
|
|
|
+ }
|
|
|
+ };
|
|
|
+ let mut args: Vec<String> = Vec::new();
|
|
|
+ for arg in iter.collect::<Vec<String>>().iter() {
|
|
|
+ let arg = re.replace_all(arg.as_str(), &replacement).to_string();
|
|
|
+ match re.captures(arg.as_str()) {
|
|
|
+ Some(caps) => {
|
|
|
+ return Err(CommandError::EnvironmentVariableNotFound(String::from(
|
|
|
+ caps.get(0).unwrap().as_str(),
|
|
|
+ )))
|
|
|
+ }
|
|
|
+ None => args.push(arg),
|
|
|
+ }
|
|
|
+ }
|
|
|
+ Command::new(name, args)
|
|
|
+ }
|
|
|
+
|
|
|
+ pub fn from_strings(str: String) -> Vec<Command> {
|
|
|
+ str.split(';')
|
|
|
+ .filter_map(|s| match Command::from_string(String::from(s)) {
|
|
|
+ Ok(s) => Some(s),
|
|
|
+ Err(e) => {
|
|
|
+ CommandError::handle(e);
|
|
|
+ None
|
|
|
+ }
|
|
|
+ })
|
|
|
+ .collect::<Vec<Command>>()
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+pub struct BuildInCmd(pub &'static str);
|
|
|
+
|
|
|
+impl BuildInCmd {
|
|
|
+ pub const BUILD_IN_CMD: &[BuildInCmd] = &[
|
|
|
+ BuildInCmd("cd"),
|
|
|
+ BuildInCmd("ls"),
|
|
|
+ BuildInCmd("cat"),
|
|
|
+ BuildInCmd("touch"),
|
|
|
+ BuildInCmd("mkdir"),
|
|
|
+ BuildInCmd("rm"),
|
|
|
+ BuildInCmd("rmdir"),
|
|
|
+ BuildInCmd("pwd"),
|
|
|
+ BuildInCmd("cp"),
|
|
|
+ BuildInCmd("exec"),
|
|
|
+ BuildInCmd("echo"),
|
|
|
+ BuildInCmd("reboot"),
|
|
|
+ BuildInCmd("free"),
|
|
|
+ BuildInCmd("kill"),
|
|
|
+ BuildInCmd("help"),
|
|
|
+ BuildInCmd("export"),
|
|
|
+ BuildInCmd("env"),
|
|
|
+ BuildInCmd("compgen"),
|
|
|
+ BuildInCmd("complete"),
|
|
|
+ ];
|
|
|
+}
|
|
|
+
|
|
|
+impl Shell {
|
|
|
+ pub fn exec_internal_command(
|
|
|
+ &mut self,
|
|
|
+ cmd: &str,
|
|
|
+ args: &Vec<String>,
|
|
|
+ ) -> Result<(), CommandError> {
|
|
|
+ match cmd {
|
|
|
+ "cd" => self.shell_cmd_cd(args),
|
|
|
+ "ls" => self.shell_cmd_ls(args),
|
|
|
+ "cat" => self.shell_cmd_cat(args),
|
|
|
+ "touch" => self.shell_cmd_touch(args),
|
|
|
+ "mkdir" => self.shell_cmd_mkdir(args),
|
|
|
+ "rm" => self.shell_cmd_rm(args),
|
|
|
+ "rmdir" => self.shell_cmd_rmdir(args),
|
|
|
+ "pwd" => self.shell_cmd_pwd(args),
|
|
|
+ "cp" => self.shell_cmd_cp(args),
|
|
|
+ "exec" => self.shell_cmd_exec(args),
|
|
|
+ "echo" => self.shell_cmd_echo(args),
|
|
|
+ "reboot" => self.shell_cmd_reboot(args),
|
|
|
+ "free" => self.shell_cmd_free(args),
|
|
|
+ "kill" => self.shell_cmd_kill(args),
|
|
|
+ "help" => self.shell_cmd_help(args),
|
|
|
+ "export" => self.shell_cmd_export(args),
|
|
|
+ "env" => self.shell_cmd_env(args),
|
|
|
+ "compgen" => self.shell_cmd_compgen(args),
|
|
|
+ "complete" => self.shell_cmd_complete(args),
|
|
|
+
|
|
|
+ _ => Err(CommandError::CommandNotFound(String::from(cmd))),
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ pub fn exec_external_command(&mut self, path: String, args: &Vec<String>) {
|
|
|
+ let mut full_args = args.clone();
|
|
|
+ full_args.insert(0, path.clone());
|
|
|
+ self.shell_cmd_exec(&full_args).unwrap_or_else(|e| {
|
|
|
+ let err = match e {
|
|
|
+ CommandError::FileNotFound(_) => CommandError::CommandNotFound(path.clone()),
|
|
|
+ _ => e,
|
|
|
+ };
|
|
|
+ CommandError::handle(err);
|
|
|
+ })
|
|
|
+ }
|
|
|
+
|
|
|
+ pub fn exec_command(&mut self, command: &Command) {
|
|
|
+ match &command.cmd_type {
|
|
|
+ CommandType::ExternalCommand(path) => {
|
|
|
+ self.exec_external_command(path.to_string(), &command.args);
|
|
|
+ }
|
|
|
+
|
|
|
+ CommandType::InternalCommand(BuildInCmd(cmd)) => {
|
|
|
+ match self.exec_internal_command(cmd, &command.args) {
|
|
|
+ Ok(_) => {}
|
|
|
+ Err(e) => CommandError::handle(e),
|
|
|
+ }
|
|
|
+ if command.args.contains(&String::from("--help")) {
|
|
|
+ Help::shell_help(cmd);
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ fn shell_cmd_cd(&mut self, args: &Vec<String>) -> Result<(), CommandError> {
|
|
|
+ if args.len() == 0 {
|
|
|
+ self.set_current_dir(String::from(ROOT_PATH));
|
|
|
+ return Ok(());
|
|
|
+ }
|
|
|
+ if args.len() == 1 {
|
|
|
+ let path = self.to_absolute_path(args.get(0).unwrap());
|
|
|
+ if Path::new(&path).is_dir() {
|
|
|
+ self.set_current_dir(String::from(path));
|
|
|
+ return Ok(());
|
|
|
+ } else {
|
|
|
+ return Err(CommandError::NotDirectory(path));
|
|
|
+ }
|
|
|
+ } else {
|
|
|
+ return Err(CommandError::WrongArgumentCount(args.len()));
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ fn shell_cmd_ls(&self, args: &Vec<String>) -> Result<(), CommandError> {
|
|
|
+ let mut path = String::new();
|
|
|
+ if args.len() == 0 {
|
|
|
+ path = self.current_dir();
|
|
|
+ }
|
|
|
+ if args.len() == 1 {
|
|
|
+ path = self.to_absolute_path(args.get(0).unwrap());
|
|
|
+ }
|
|
|
+
|
|
|
+ if path.is_empty() {
|
|
|
+ return Err(CommandError::WrongArgumentCount(args.len()));
|
|
|
+ }
|
|
|
+
|
|
|
+ let dir: fs::ReadDir;
|
|
|
+ match fs::read_dir(Path::new(&path)) {
|
|
|
+ Ok(readdir) => dir = readdir,
|
|
|
+ Err(_) => return Err(CommandError::InvalidArgument(path)),
|
|
|
+ }
|
|
|
+ for entry in dir {
|
|
|
+ let entry = entry.unwrap();
|
|
|
+ if entry.file_type().unwrap().is_dir() {
|
|
|
+ crate::shell::Printer::print_color(
|
|
|
+ entry.file_name().as_bytes(),
|
|
|
+ 0x000088ff,
|
|
|
+ 0x00000000,
|
|
|
+ );
|
|
|
+ print!(" ");
|
|
|
+ } else {
|
|
|
+ print!("{} ", entry.file_name().into_string().unwrap());
|
|
|
+ }
|
|
|
+ }
|
|
|
+ print!("{}", String::from_utf8(vec![CR, LF]).unwrap());
|
|
|
+ return Ok(());
|
|
|
+ }
|
|
|
+
|
|
|
+ fn shell_cmd_cat(&self, args: &Vec<String>) -> Result<(), CommandError> {
|
|
|
+ if args.len() > 0 {
|
|
|
+ let path = self.to_absolute_path(args.get(0).unwrap());
|
|
|
+ let mut buf: Vec<u8> = Vec::new();
|
|
|
+ if !Path::new(&path).exists() {
|
|
|
+ return Err(CommandError::PathNotFound(path));
|
|
|
+ } else if !Path::new(&path).is_file() {
|
|
|
+ return Err(CommandError::NotFile(path));
|
|
|
+ }
|
|
|
+
|
|
|
+ File::open(path).unwrap().read_to_end(&mut buf).unwrap();
|
|
|
+ if args.len() == 1 {
|
|
|
+ println!("{}", String::from_utf8(buf.clone()).unwrap());
|
|
|
+ }
|
|
|
+
|
|
|
+ if args.len() == 3 {
|
|
|
+ let target_path = self.to_absolute_path(args.get(2).unwrap());
|
|
|
+ if !Path::new(&target_path).is_file() {
|
|
|
+ return Err(CommandError::NotFile(target_path));
|
|
|
+ }
|
|
|
+
|
|
|
+ if args[1] == ">" {
|
|
|
+ match OpenOptions::new().write(true).open(target_path) {
|
|
|
+ Ok(mut file) => {
|
|
|
+ file.write_all(&buf).unwrap();
|
|
|
+ }
|
|
|
+ Err(e) => print!("{e}"),
|
|
|
+ }
|
|
|
+ } else if args[1] == ">>" {
|
|
|
+ match OpenOptions::new().append(true).open(target_path) {
|
|
|
+ Ok(mut file) => {
|
|
|
+ file.write_all(&buf).unwrap();
|
|
|
+ }
|
|
|
+ Err(e) => print!("{e}"),
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+ return Ok(());
|
|
|
+ }
|
|
|
+ return Err(CommandError::WrongArgumentCount(args.len()));
|
|
|
+ }
|
|
|
+
|
|
|
+ fn shell_cmd_touch(&self, args: &Vec<String>) -> Result<(), CommandError> {
|
|
|
+ if args.len() == 1 {
|
|
|
+ let path = self.to_absolute_path(args.get(0).unwrap());
|
|
|
+ match File::create(path) {
|
|
|
+ Ok(_) => {}
|
|
|
+ Err(e) => {
|
|
|
+ print!("{e}")
|
|
|
+ }
|
|
|
+ };
|
|
|
+ return Ok(());
|
|
|
+ }
|
|
|
+ return Err(CommandError::WrongArgumentCount(args.len()));
|
|
|
+ }
|
|
|
+
|
|
|
+ fn shell_cmd_mkdir(&self, args: &Vec<String>) -> Result<(), CommandError> {
|
|
|
+ if args.len() == 1 {
|
|
|
+ let path = self.to_absolute_path(args.get(0).unwrap());
|
|
|
+ let dir = &path[0..path.rfind('/').unwrap_or(0)];
|
|
|
+ if !Path::new(dir).is_dir() {
|
|
|
+ return Err(CommandError::DirectoryNotFound(dir.to_string()));
|
|
|
+ }
|
|
|
+ match fs::create_dir_all(path) {
|
|
|
+ Ok(_) => {}
|
|
|
+ Err(e) => {
|
|
|
+ print!("{e}")
|
|
|
+ }
|
|
|
+ }
|
|
|
+ return Ok(());
|
|
|
+ } else {
|
|
|
+ return Err(CommandError::WrongArgumentCount(args.len()));
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ fn shell_cmd_rm(&self, args: &Vec<String>) -> Result<(), CommandError> {
|
|
|
+ if args.len() == 1 {
|
|
|
+ let path = self.to_absolute_path(args.get(0).unwrap());
|
|
|
+ // match fs::remove_file(path) {
|
|
|
+ // Ok(_) => {}
|
|
|
+ // Err(e) => {
|
|
|
+ // print!("{e}")
|
|
|
+ // }
|
|
|
+ // }
|
|
|
+ if Path::new(&path).is_file() {
|
|
|
+ let path_cstr = std::ffi::CString::new(path).unwrap();
|
|
|
+ unsafe {
|
|
|
+ // libc::unlinkat(0, path_cstr.as_ptr(), 0);
|
|
|
+ libc::syscall(libc::SYS_unlinkat, 0, path_cstr.as_ptr(), 0, 0, 0, 0);
|
|
|
+ }
|
|
|
+ return Ok(());
|
|
|
+ } else {
|
|
|
+ return Err(CommandError::FileNotFound(path));
|
|
|
+ }
|
|
|
+ }
|
|
|
+ return Err(CommandError::WrongArgumentCount(args.len()));
|
|
|
+ }
|
|
|
+
|
|
|
+ fn shell_cmd_rmdir(&self, args: &Vec<String>) -> Result<(), CommandError> {
|
|
|
+ if args.len() == 1 {
|
|
|
+ let path = self.to_absolute_path(args.get(0).unwrap());
|
|
|
+ if Path::new(&path).is_dir() {
|
|
|
+ let path_cstr = std::ffi::CString::new(path).unwrap();
|
|
|
+ unsafe { libc::unlinkat(0, path_cstr.as_ptr(), libc::AT_REMOVEDIR) };
|
|
|
+ return Ok(());
|
|
|
+ } else {
|
|
|
+ return Err(CommandError::DirectoryNotFound(path));
|
|
|
+ }
|
|
|
+ } else {
|
|
|
+ return Err(CommandError::WrongArgumentCount(args.len()));
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ fn shell_cmd_pwd(&self, args: &Vec<String>) -> Result<(), CommandError> {
|
|
|
+ if args.len() == 0 {
|
|
|
+ println!("{}", self.current_dir());
|
|
|
+ return Ok(());
|
|
|
+ } else {
|
|
|
+ return Err(CommandError::WrongArgumentCount(args.len()));
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ fn shell_cmd_cp(&self, args: &Vec<String>) -> Result<(), CommandError> {
|
|
|
+ if args.len() == 2 {
|
|
|
+ let src_path = self.to_absolute_path(args.get(0).unwrap());
|
|
|
+ let mut target_path = self.to_absolute_path(args.get(1).unwrap());
|
|
|
+ if !Path::new(&src_path).is_file() {
|
|
|
+ return Err(CommandError::NotFile(src_path));
|
|
|
+ };
|
|
|
+ let mut src_file = File::open(&src_path).unwrap();
|
|
|
+ let name = &src_path[src_path.rfind('/').unwrap_or(0)..];
|
|
|
+
|
|
|
+ if !Path::new(&target_path).exists() {
|
|
|
+ return Err(CommandError::PathNotFound(target_path));
|
|
|
+ } else if Path::new(&target_path).is_dir() {
|
|
|
+ target_path = format!("{}/{}", target_path, name);
|
|
|
+ }
|
|
|
+ let mut target_file = File::create(target_path).unwrap();
|
|
|
+ let mut buf: Vec<u8> = Vec::new();
|
|
|
+ src_file.read_to_end(&mut buf).unwrap();
|
|
|
+ target_file.write_all(&buf).unwrap();
|
|
|
+ return Ok(());
|
|
|
+ } else {
|
|
|
+ return Err(CommandError::WrongArgumentCount(args.len()));
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ pub fn shell_cmd_exec(&self, args: &Vec<String>) -> Result<(), CommandError> {
|
|
|
+ let path = args.get(0).unwrap();
|
|
|
+ let mut real_path = String::new();
|
|
|
+ if !path.starts_with('/') {
|
|
|
+ let mut prefix_collection = Env::path();
|
|
|
+ prefix_collection.insert(0, self.current_dir());
|
|
|
+ for prefix in prefix_collection {
|
|
|
+ real_path = Self::path_format(&format!("{}/{}", prefix, path));
|
|
|
+ if Path::new(&real_path).is_file() {
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ if !Path::new(&real_path).is_file() {
|
|
|
+ return Err(CommandError::FileNotFound(path.clone()));
|
|
|
+ }
|
|
|
+
|
|
|
+ let pid: libc::pid_t = unsafe {
|
|
|
+ libc::syscall(libc::SYS_fork, 0, 0, 0, 0, 0, 0)
|
|
|
+ .try_into()
|
|
|
+ .unwrap()
|
|
|
+ };
|
|
|
+ let mut retval = 0;
|
|
|
+ if pid == 0 {
|
|
|
+ let path_cstr = std::ffi::CString::new(real_path).unwrap();
|
|
|
+ let mut argv: *const *const i8 = std::ptr::null();
|
|
|
+ if args.len() > 1 {
|
|
|
+ let args_cstr = args
|
|
|
+ .iter()
|
|
|
+ .skip(1)
|
|
|
+ .map(|str| std::ffi::CString::new(str.as_str()).unwrap())
|
|
|
+ .collect::<Vec<std::ffi::CString>>();
|
|
|
+ let mut args_ptr = args_cstr
|
|
|
+ .iter()
|
|
|
+ .map(|c_str| c_str.as_ptr())
|
|
|
+ .collect::<Vec<*const i8>>();
|
|
|
+ args_ptr.push(std::ptr::null());
|
|
|
+ argv = args_ptr.as_ptr();
|
|
|
+ }
|
|
|
+ unsafe {
|
|
|
+ libc::execv(path_cstr.as_ptr(), argv);
|
|
|
+ }
|
|
|
+ } else {
|
|
|
+ if args.last().unwrap() != &"&" {
|
|
|
+ unsafe { libc::waitpid(pid, &mut retval as *mut i32, 0) };
|
|
|
+ } else {
|
|
|
+ println!("[1] {}", pid);
|
|
|
+ }
|
|
|
+ }
|
|
|
+ return Ok(());
|
|
|
+ }
|
|
|
+
|
|
|
+ fn shell_cmd_echo(&self, args: &Vec<String>) -> Result<(), CommandError> {
|
|
|
+ if args.len() > 0 {
|
|
|
+ let str = args.get(0).unwrap();
|
|
|
+ if args.len() == 1 {
|
|
|
+ println!("{str}");
|
|
|
+ }
|
|
|
+
|
|
|
+ if args.len() == 3 {
|
|
|
+ let target_path = self.to_absolute_path(args.get(2).unwrap());
|
|
|
+ if args[1] == ">" {
|
|
|
+ match OpenOptions::new().write(true).open(target_path) {
|
|
|
+ Ok(mut file) => {
|
|
|
+ file.write_all(str.as_bytes()).unwrap();
|
|
|
+ }
|
|
|
+ Err(e) => print!("{e}"),
|
|
|
+ }
|
|
|
+ } else if args[1] == ">>" {
|
|
|
+ match OpenOptions::new().append(true).open(target_path) {
|
|
|
+ Ok(mut file) => {
|
|
|
+ file.write_all(str.as_bytes()).unwrap();
|
|
|
+ }
|
|
|
+ Err(e) => print!("{e}"),
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+ return Ok(());
|
|
|
+ } else {
|
|
|
+ return Err(CommandError::WrongArgumentCount(args.len()));
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ fn shell_cmd_reboot(&self, args: &Vec<String>) -> Result<(), CommandError> {
|
|
|
+ if args.len() == 0 {
|
|
|
+ unsafe { libc::syscall(libc::SYS_reboot, 0, 0, 0, 0, 0, 0) };
|
|
|
+ return Ok(());
|
|
|
+ } else {
|
|
|
+ return Err(CommandError::WrongArgumentCount(args.len()));
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ fn shell_cmd_free(&self, args: &Vec<String>) -> Result<(), CommandError> {
|
|
|
+ if args.len() == 1 && args.get(0).unwrap() != "-m" {
|
|
|
+ return Err(CommandError::InvalidArgument(
|
|
|
+ args.get(0).unwrap().to_string(),
|
|
|
+ ));
|
|
|
+ }
|
|
|
+
|
|
|
+ struct mstat_t {
|
|
|
+ total: u64, // 计算机的总内存数量大小
|
|
|
+ used: u64, // 已使用的内存大小
|
|
|
+ free: u64, // 空闲物理页所占的内存大小
|
|
|
+ shared: u64, // 共享的内存大小
|
|
|
+ cache_used: u64, // 位于slab缓冲区中的已使用的内存大小
|
|
|
+ cache_free: u64, // 位于slab缓冲区中的空闲的内存大小
|
|
|
+ available: u64, // 系统总空闲内存大小(包括kmalloc缓冲区)
|
|
|
+ };
|
|
|
+
|
|
|
+ let mut mst = mstat_t {
|
|
|
+ total: 0,
|
|
|
+ used: 0,
|
|
|
+ free: 0,
|
|
|
+ shared: 0,
|
|
|
+ cache_used: 0,
|
|
|
+ cache_free: 0,
|
|
|
+ available: 0,
|
|
|
+ };
|
|
|
+
|
|
|
+ let mut info_file = File::open("/proc/meminfo").unwrap();
|
|
|
+ let mut buf: Vec<u8> = Vec::new();
|
|
|
+ info_file.read_to_end(&mut buf).unwrap();
|
|
|
+ let str = String::from_utf8(buf).unwrap();
|
|
|
+ let info = str
|
|
|
+ .split(&['\n', '\t', ' '])
|
|
|
+ .filter_map(|str| str.parse::<u64>().ok())
|
|
|
+ .collect::<Vec<u64>>();
|
|
|
+ mst.total = *info.get(0).unwrap();
|
|
|
+ mst.free = *info.get(1).unwrap();
|
|
|
+ mst.used = mst.total - mst.free;
|
|
|
+
|
|
|
+ print!("\ttotal\tused\tfree\tshared\tcache\tavailable\n");
|
|
|
+ print!("Mem:\t");
|
|
|
+
|
|
|
+ if args.len() == 0 {
|
|
|
+ print!(
|
|
|
+ "{}\t{}\t{}\t{}\t{}\t{}\t\n",
|
|
|
+ mst.total, mst.used, mst.free, mst.shared, mst.cache_used, mst.available
|
|
|
+ );
|
|
|
+ } else {
|
|
|
+ print!(
|
|
|
+ "{}\t{}\t{}\t{}\t{}\t{}\t\n",
|
|
|
+ mst.total >> 10,
|
|
|
+ mst.used >> 10,
|
|
|
+ mst.free >> 10,
|
|
|
+ mst.shared >> 10,
|
|
|
+ mst.cache_used >> 10,
|
|
|
+ mst.available >> 10
|
|
|
+ );
|
|
|
+ }
|
|
|
+ Ok(())
|
|
|
+ }
|
|
|
+
|
|
|
+ fn shell_cmd_kill(&self, args: &Vec<String>) -> Result<(), CommandError> {
|
|
|
+ if args.len() == 1 {
|
|
|
+ let pid: i32;
|
|
|
+ match args.get(0).unwrap().parse::<i32>() {
|
|
|
+ Ok(x) => pid = x,
|
|
|
+ Err(_) => {
|
|
|
+ return Err(CommandError::InvalidArgument(
|
|
|
+ args.get(0).unwrap().to_string(),
|
|
|
+ ))
|
|
|
+ }
|
|
|
+ }
|
|
|
+ unsafe {
|
|
|
+ libc::kill(pid, libc::SIGKILL);
|
|
|
+ }
|
|
|
+ return Ok(());
|
|
|
+ } else {
|
|
|
+ return Err(CommandError::WrongArgumentCount(args.len()));
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ fn shell_cmd_help(&self, args: &Vec<String>) -> Result<(), CommandError> {
|
|
|
+ if args.len() == 0 {
|
|
|
+ for BuildInCmd(cmd) in BuildInCmd::BUILD_IN_CMD {
|
|
|
+ Help::shell_help(cmd)
|
|
|
+ }
|
|
|
+ return Ok(());
|
|
|
+ }
|
|
|
+ return Err(CommandError::WrongArgumentCount(args.len()));
|
|
|
+ }
|
|
|
+
|
|
|
+ fn shell_cmd_export(&self, args: &Vec<String>) -> Result<(), CommandError> {
|
|
|
+ if args.len() == 1 {
|
|
|
+ let pair = args.get(0).unwrap().split('=').collect::<Vec<&str>>();
|
|
|
+
|
|
|
+ if pair.len() == 2 && !pair.contains(&"") {
|
|
|
+ let name = pair.get(0).unwrap().to_string();
|
|
|
+ let value = pair.get(1).unwrap().to_string();
|
|
|
+ Env::insert(name, value);
|
|
|
+ return Ok(());
|
|
|
+ } else {
|
|
|
+ return Err(CommandError::InvalidArgument(args.get(0).unwrap().clone()));
|
|
|
+ }
|
|
|
+ }
|
|
|
+ return Err(CommandError::WrongArgumentCount(args.len()));
|
|
|
+ }
|
|
|
+
|
|
|
+ fn shell_cmd_env(&self, args: &Vec<String>) -> Result<(), CommandError> {
|
|
|
+ if args.len() == 0 {
|
|
|
+ let mut file = File::open("/etc/profile").unwrap();
|
|
|
+ let mut buf: Vec<u8> = Vec::new();
|
|
|
+ file.read_to_end(&mut buf).unwrap();
|
|
|
+ println!("{}", String::from_utf8(buf).unwrap());
|
|
|
+ return Ok(());
|
|
|
+ } else {
|
|
|
+ return Err(CommandError::InvalidArgument(args.get(0).unwrap().clone()));
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ fn shell_cmd_compgen(&self, args: &Vec<String>) -> Result<(), CommandError> {
|
|
|
+ Ok(())
|
|
|
+ }
|
|
|
+
|
|
|
+ fn shell_cmd_complete(&self, args: &Vec<String>) -> Result<(), CommandError> {
|
|
|
+ Ok(())
|
|
|
+ }
|
|
|
+
|
|
|
+ fn to_absolute_path(&self, path: &String) -> String {
|
|
|
+ let mut fmt_path = Self::path_format(path);
|
|
|
+ if !path.starts_with('/') {
|
|
|
+ if self.current_dir() == "/" {
|
|
|
+ fmt_path = format!("/{}", fmt_path);
|
|
|
+ } else {
|
|
|
+ fmt_path = format!("{}/{}", self.current_dir(), fmt_path);
|
|
|
+ }
|
|
|
+ }
|
|
|
+ if fmt_path.ends_with('/') {
|
|
|
+ fmt_path.pop().unwrap();
|
|
|
+ }
|
|
|
+
|
|
|
+ let re = regex::Regex::new(r"\/\.\/").unwrap();
|
|
|
+ let replacement = |_caps: ®ex::Captures| -> String { String::from("/") };
|
|
|
+ fmt_path = re.replace_all(fmt_path.as_str(), replacement).to_string();
|
|
|
+
|
|
|
+ let re = regex::Regex::new(r"\/[^\/]+\/\.\.").unwrap();
|
|
|
+ let replacement = |_caps: ®ex::Captures| -> String { String::from("/") };
|
|
|
+ fmt_path = re.replace_all(fmt_path.as_str(), replacement).to_string();
|
|
|
+ fmt_path = Self::path_format(&fmt_path);
|
|
|
+ return fmt_path;
|
|
|
+ }
|
|
|
+
|
|
|
+ fn path_format(path: &String) -> String {
|
|
|
+ let re = regex::Regex::new(r"\/{2,}").unwrap();
|
|
|
+ let replacement = |_caps: ®ex::Captures| -> String { String::from("/") };
|
|
|
+ let fmt_path = re.replace_all(path, replacement).to_string();
|
|
|
+ return fmt_path;
|
|
|
+ }
|
|
|
+}
|