shell.c divided into multiple files, command pipes support
This commit is contained in:
parent
fcd73c9187
commit
56ae617430
@ -14,6 +14,7 @@ Work is still in porgress, buf when you will see a "finished" topic assigned to
|
|||||||
* Show previous command return status in prompt and invert it with `!`, separated with space, specified before it
|
* Show previous command return status in prompt and invert it with `!`, separated with space, specified before it
|
||||||
* Running multiple commands separated by `;`, `&&` or `||`
|
* Running multiple commands separated by `;`, `&&` or `||`
|
||||||
* Expand `*` wildcards
|
* Expand `*` wildcards
|
||||||
|
* Commands I/O redirection with `|` pipes
|
||||||
|
|
||||||
# Builtin commands
|
# Builtin commands
|
||||||
* `cd`: changes current working directory to the one specified by user. If no arguments provided, shows error.
|
* `cd`: changes current working directory to the one specified by user. If no arguments provided, shows error.
|
||||||
@ -21,7 +22,7 @@ Work is still in porgress, buf when you will see a "finished" topic assigned to
|
|||||||
* `exec`: executes entered command and exits
|
* `exec`: executes entered command and exits
|
||||||
|
|
||||||
# TODO
|
# TODO
|
||||||
* Pipes and files input/output
|
* Files input/output
|
||||||
* Replace linux `echo` command with builtin one with support of environmental variables
|
* Replace linux `echo` command with builtin one with support of environmental variables
|
||||||
* Environmental variables
|
* Environmental variables
|
||||||
* `Ctrl+Z` running programm with `fd`
|
* `Ctrl+Z` running programm with `fd`
|
||||||
|
@ -2,6 +2,7 @@
|
|||||||
#define _KEYS_H
|
#define _KEYS_H
|
||||||
|
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
|
#include <string.h>
|
||||||
|
|
||||||
void delete_key(int pos, int *n, char **line);
|
void delete_key(int pos, int *n, char **line);
|
||||||
void up_key(int *pos, int *n, char **line);
|
void up_key(int *pos, int *n, char **line);
|
||||||
|
@ -53,9 +53,19 @@ typedef enum
|
|||||||
OR_SEP
|
OR_SEP
|
||||||
} cmd_sep;
|
} cmd_sep;
|
||||||
|
|
||||||
typedef struct commands
|
typedef struct pipes
|
||||||
{
|
{
|
||||||
char **args;
|
char **args;
|
||||||
|
int args_am;
|
||||||
|
int pipefd[2];
|
||||||
|
struct pipes *next;
|
||||||
|
} cmd_pipe;
|
||||||
|
|
||||||
|
typedef struct commands
|
||||||
|
{
|
||||||
|
cmd_pipe *pipe;
|
||||||
|
int pipes_am;
|
||||||
|
|
||||||
struct status stat;
|
struct status stat;
|
||||||
struct commands *next;
|
struct commands *next;
|
||||||
cmd_sep sep_next;
|
cmd_sep sep_next;
|
||||||
@ -72,22 +82,31 @@ typedef struct
|
|||||||
extern t_ term;
|
extern t_ term;
|
||||||
extern char *builtin[];
|
extern char *builtin[];
|
||||||
|
|
||||||
// Functions prototypes
|
// shell.c file
|
||||||
int process_line(char *line, cmds_p **coms);
|
|
||||||
int launch(char **args);
|
|
||||||
void process_command();
|
void process_command();
|
||||||
int execute(cmds_p *command);
|
cmds_p *new_cmd();
|
||||||
|
cmd_pipe *new_cmd_pipe();
|
||||||
|
void free_cmds_p(cmds_p *root);
|
||||||
|
void free_cmd_pipe(cmd_pipe *root);
|
||||||
|
|
||||||
void sig_handler();
|
// line_parce.c file
|
||||||
|
cmds_p *process_line(char *line);
|
||||||
|
int expand_wildcatrd(char ***arr, char *input);
|
||||||
|
|
||||||
|
// execute.c file
|
||||||
|
int execute_with_pipes(cmds_p *command);
|
||||||
|
int execute(cmd_pipe *command);
|
||||||
|
int launch(cmd_pipe *command);
|
||||||
|
|
||||||
int sh_cd(char **args);
|
int sh_cd(char **args);
|
||||||
int sh_exit(char **args);
|
int sh_exit(char **args);
|
||||||
int sh_exec(char **args);
|
int sh_exec(char **args);
|
||||||
|
|
||||||
char *compose_prompt();
|
void redirect_fd(int old, int new);
|
||||||
|
|
||||||
cmds_p *new_cmd();
|
// prompt.c
|
||||||
int expand_wildcatrd(char ***arr, char *input);
|
char *get_ip_addr();
|
||||||
|
char *compose_prompt();
|
||||||
|
|
||||||
#define BUILTIN_NUM sizeof(builtin) / sizeof(char *)
|
#define BUILTIN_NUM sizeof(builtin) / sizeof(char *)
|
||||||
|
|
||||||
|
@ -19,7 +19,7 @@ void append_to_pos(char **str, int pos, char ch);
|
|||||||
void remove_on_pos(char **str, int pos);
|
void remove_on_pos(char **str, int pos);
|
||||||
int sep_string(char *line, char ***toks, char *sep);
|
int sep_string(char *line, char ***toks, char *sep);
|
||||||
char *trim_string(char *str, bool leave_trailing_space);
|
char *trim_string(char *str, bool leave_trailing_space);
|
||||||
void free_str_arr(char **arr);
|
void free_str_arr(char **arr, int sz);
|
||||||
char **slice_array(char **arr, int beg, int end, bool asc);
|
char **slice_array(char **arr, int beg, int end, bool asc);
|
||||||
int get_null_term_arr_size(char **arr);
|
int get_null_term_arr_size(char **arr);
|
||||||
int append_to_str_arr(char ***arr, int *sz, char *str);
|
int append_to_str_arr(char ***arr, int *sz, char *str);
|
||||||
|
@ -98,33 +98,31 @@ ssize_t append_builtin_list(char ***commands_list, ssize_t *size)
|
|||||||
*/
|
*/
|
||||||
ssize_t get_complete_options(char ***opts, char *line, char **to_complete)
|
ssize_t get_complete_options(char ***opts, char *line, char **to_complete)
|
||||||
{
|
{
|
||||||
char **args = NULL, **folders = malloc(0);
|
char **args = NULL, **folders = calloc(0, sizeof(char *));
|
||||||
ssize_t sz;
|
ssize_t sz;
|
||||||
|
|
||||||
int am = sep_string(line, &args, " ");
|
int am = sep_string(line, &args, " "), path_depth = 0;
|
||||||
|
|
||||||
char *last_arg = args[am - 1];
|
char *last_arg = args[am - 1];
|
||||||
|
|
||||||
if (am > 0)
|
if (am > 0)
|
||||||
{
|
{
|
||||||
int path_depth = sep_string(last_arg, &folders, "/");
|
path_depth = sep_string(last_arg, &folders, "/");
|
||||||
*to_complete = strdup(folders[path_depth - 1]);
|
*to_complete = strdup(folders[path_depth - 1]);
|
||||||
|
|
||||||
char *curr_pos = NULL;
|
char *curr_pos = NULL;
|
||||||
int i = 0;
|
|
||||||
|
|
||||||
if (last_arg[0] == '/')
|
if (last_arg[0] == '/')
|
||||||
{
|
{
|
||||||
curr_pos = strdup("");
|
curr_pos = strdup("");
|
||||||
}
|
}
|
||||||
else if (strchr(line, ' ') && strcmp(args[am - 2], "||") != 0 && strcmp(args[am - 2], "&&") != 0 && strcmp(args[am - 2], ";") != 0 && line[strlen(line) - 2] != ';' && line[strlen(line) - 2] != '&' && line[strlen(line) - 2] != '|')
|
else if (strchr(line, ' ') && strcmp(args[am - 2], "||") != 0 && strcmp(args[am - 2], "|") != 0 && strcmp(args[am - 2], "&&") != 0 && strcmp(args[am - 2], ";") != 0 && line[strlen(line) - 2] != ';' && line[strlen(line) - 2] != '&' && line[strlen(line) - 2] != '|')
|
||||||
{
|
{
|
||||||
curr_pos = strdup(".");
|
curr_pos = strdup(".");
|
||||||
}
|
}
|
||||||
else if (last_arg[0] == '.' && last_arg[1] == '/')
|
else if (last_arg[0] == '.' && last_arg[1] == '/')
|
||||||
{
|
{
|
||||||
curr_pos = strdup(folders[0]);
|
curr_pos = strdup(folders[0]);
|
||||||
i++;
|
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
goto ABSOLUTE;
|
goto ABSOLUTE;
|
||||||
@ -152,8 +150,8 @@ ssize_t get_complete_options(char ***opts, char *line, char **to_complete)
|
|||||||
append_builtin_list(opts, &sz);
|
append_builtin_list(opts, &sz);
|
||||||
}
|
}
|
||||||
|
|
||||||
free_str_arr(args);
|
free_str_arr(args, am);
|
||||||
free_str_arr(folders);
|
free_str_arr(folders, path_depth);
|
||||||
|
|
||||||
if (sz < 0)
|
if (sz < 0)
|
||||||
return sz;
|
return sz;
|
||||||
@ -197,7 +195,7 @@ ssize_t filter_options(char ***comp_list, ssize_t *size, char *filter_string)
|
|||||||
*size = list_strings_containing(child_dirs_root, last_option, comp_list);
|
*size = list_strings_containing(child_dirs_root, last_option, comp_list);
|
||||||
|
|
||||||
free_tree(child_dirs_root);
|
free_tree(child_dirs_root);
|
||||||
free_str_arr(folders);
|
free_str_arr(folders, path_depth);
|
||||||
|
|
||||||
return *size;
|
return *size;
|
||||||
}
|
}
|
178
src/execute.c
Normal file
178
src/execute.c
Normal file
@ -0,0 +1,178 @@
|
|||||||
|
#include "../include/shell.h"
|
||||||
|
#include "../include/utils.h"
|
||||||
|
#include "../include/input.h"
|
||||||
|
|
||||||
|
/* Global definitions */
|
||||||
|
char *builtin[] = {
|
||||||
|
"cd",
|
||||||
|
"exit",
|
||||||
|
"exec"};
|
||||||
|
|
||||||
|
int (*builtin_func[])(char **) = {
|
||||||
|
&sh_cd,
|
||||||
|
&sh_exit,
|
||||||
|
&sh_exec};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief Redirects command input/output and executes it
|
||||||
|
*
|
||||||
|
* @param command
|
||||||
|
* @return int
|
||||||
|
*/
|
||||||
|
int execute_with_pipes(cmds_p *command)
|
||||||
|
{
|
||||||
|
cmd_pipe *curr = command->pipe;
|
||||||
|
int status = 0, tmp_fd[2] = {-1};
|
||||||
|
|
||||||
|
for (int i = 0; i < command->pipes_am - 1 && curr != NULL && status == 0; i++)
|
||||||
|
{
|
||||||
|
|
||||||
|
if (pipe(tmp_fd) < 0)
|
||||||
|
{
|
||||||
|
perror("pipe");
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
curr->pipefd[1] = tmp_fd[1];
|
||||||
|
curr->next->pipefd[0] = tmp_fd[0];
|
||||||
|
|
||||||
|
status = execute(curr);
|
||||||
|
|
||||||
|
curr = curr->next;
|
||||||
|
}
|
||||||
|
|
||||||
|
status = execute(curr);
|
||||||
|
|
||||||
|
return status;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief Executes either builtin or system command
|
||||||
|
*
|
||||||
|
* @param args
|
||||||
|
* @return int
|
||||||
|
*/
|
||||||
|
int execute(cmd_pipe *command)
|
||||||
|
{
|
||||||
|
if (command->args[0] == NULL)
|
||||||
|
return 1;
|
||||||
|
|
||||||
|
char **args = command->args;
|
||||||
|
|
||||||
|
if (strcmp(args[0], "!") == 0)
|
||||||
|
{
|
||||||
|
args = slice_array(args, 1, -1, true);
|
||||||
|
command->args = args;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (int i = 0; i < BUILTIN_NUM; i++)
|
||||||
|
if (strcmp(args[0], builtin[i]) == 0)
|
||||||
|
return (*builtin_func[i])(args);
|
||||||
|
|
||||||
|
return launch(command);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief Launches separate process and executes specified command
|
||||||
|
*
|
||||||
|
* @param args
|
||||||
|
* @return int
|
||||||
|
*/
|
||||||
|
int launch(cmd_pipe *command)
|
||||||
|
{
|
||||||
|
pid_t pid, wpid;
|
||||||
|
int status;
|
||||||
|
|
||||||
|
pid = fork();
|
||||||
|
|
||||||
|
if (pid == 0)
|
||||||
|
{
|
||||||
|
redirect_fd(command->pipefd[0], STDIN_FILENO);
|
||||||
|
|
||||||
|
redirect_fd(command->pipefd[1], STDOUT_FILENO);
|
||||||
|
|
||||||
|
sh_exec(command->args);
|
||||||
|
}
|
||||||
|
else if (pid < 0)
|
||||||
|
{
|
||||||
|
perror("mshell");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
for (int j = 0; j < 2; j++)
|
||||||
|
if (command->pipefd[j] > 2)
|
||||||
|
close(command->pipefd[j]);
|
||||||
|
do
|
||||||
|
wpid = waitpid(pid, &status, WUNTRACED);
|
||||||
|
while (!WIFEXITED(status) && !WIFSIGNALED(status));
|
||||||
|
}
|
||||||
|
|
||||||
|
change_mode(1);
|
||||||
|
|
||||||
|
return status;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief Shell builtin command. Changes current working directory
|
||||||
|
*
|
||||||
|
* @param args
|
||||||
|
* @return int
|
||||||
|
*/
|
||||||
|
int sh_cd(char **args)
|
||||||
|
{
|
||||||
|
if (args[1] == NULL)
|
||||||
|
chdir(getenv("HOME"));
|
||||||
|
else if (chdir(args[1]) < 0)
|
||||||
|
{
|
||||||
|
perror("cd");
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief Shell builtin command. Exits shell
|
||||||
|
*
|
||||||
|
* @param args
|
||||||
|
* @return int
|
||||||
|
*/
|
||||||
|
int sh_exit(char **args)
|
||||||
|
{
|
||||||
|
exit(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief Shell builtin command. Executes command replacing shell with it
|
||||||
|
*
|
||||||
|
* @param args
|
||||||
|
* @return int
|
||||||
|
*/
|
||||||
|
int sh_exec(char **args)
|
||||||
|
{
|
||||||
|
change_mode(0);
|
||||||
|
signal(SIGINT, SIG_DFL);
|
||||||
|
|
||||||
|
if (strcmp(args[0], "exec") == 0)
|
||||||
|
args = slice_array(args, 1, -1, 1);
|
||||||
|
|
||||||
|
if (execvp(args[0], args) < 0)
|
||||||
|
{
|
||||||
|
perror("mshell");
|
||||||
|
}
|
||||||
|
|
||||||
|
exit(EXIT_FAILURE);
|
||||||
|
}
|
||||||
|
|
||||||
|
void redirect_fd(int old, int new)
|
||||||
|
{
|
||||||
|
if (old > 0)
|
||||||
|
if (old != new)
|
||||||
|
{
|
||||||
|
if (dup2(old, new) < 0)
|
||||||
|
perror("dup2");
|
||||||
|
else
|
||||||
|
close(old);
|
||||||
|
}
|
||||||
|
}
|
@ -121,7 +121,7 @@ int process_keypress(char c)
|
|||||||
if (c == '\x1b')
|
if (c == '\x1b')
|
||||||
{
|
{
|
||||||
for (int i = 0; i < 2; i++)
|
for (int i = 0; i < 2; i++)
|
||||||
if (read(STDIN_FILENO, &seq[i], 1) == -1)
|
if (read(STDIN_FILENO, &seq[i], 1) < 0)
|
||||||
return ESCAPE_KEY;
|
return ESCAPE_KEY;
|
||||||
if (seq[0] == '[')
|
if (seq[0] == '[')
|
||||||
{
|
{
|
||||||
|
@ -278,7 +278,7 @@ void tab_key(int *pos, int *n, char **line)
|
|||||||
char *buff = strdup("");
|
char *buff = strdup("");
|
||||||
size_t buff_size = 1;
|
size_t buff_size = 1;
|
||||||
|
|
||||||
char **complete_options = malloc(0), *to_complete = NULL;
|
char **complete_options = calloc(0, sizeof(char *)), *to_complete = NULL;
|
||||||
size_t opts_sz = get_complete_options(&complete_options, *line, &to_complete);
|
size_t opts_sz = get_complete_options(&complete_options, *line, &to_complete);
|
||||||
|
|
||||||
if (opts_sz == 1)
|
if (opts_sz == 1)
|
||||||
|
234
src/line_parce.c
Normal file
234
src/line_parce.c
Normal file
@ -0,0 +1,234 @@
|
|||||||
|
#include "../include/shell.h"
|
||||||
|
#include "../include/utils.h"
|
||||||
|
#include "../include/input.h"
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief Extracts command and its arguments from line
|
||||||
|
*
|
||||||
|
* @param line
|
||||||
|
* @param args
|
||||||
|
*/
|
||||||
|
cmds_p *process_line(char *line)
|
||||||
|
{
|
||||||
|
int line_size = strlen(line);
|
||||||
|
|
||||||
|
if (line_size < 1)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
char *tmp = strdup(line), *free_tmp = tmp;
|
||||||
|
cmds_p *coms = new_cmd();
|
||||||
|
|
||||||
|
cmds_p *curr_cmd = coms;
|
||||||
|
cmd_pipe *curr_pipe = curr_cmd->pipe;
|
||||||
|
curr_pipe->pipefd[0] = STDIN_FILENO;
|
||||||
|
|
||||||
|
for (int i = 0; i < line_size; i++)
|
||||||
|
{
|
||||||
|
if (line[i] == '"')
|
||||||
|
{
|
||||||
|
tmp++;
|
||||||
|
i++;
|
||||||
|
int j = i;
|
||||||
|
for (; j < line_size; j++)
|
||||||
|
if (line[j] == '"')
|
||||||
|
{
|
||||||
|
free_tmp[j] = '\0';
|
||||||
|
|
||||||
|
append_to_str_arr(&(curr_pipe->args), &curr_pipe->args_am, trim_string(tmp, false));
|
||||||
|
|
||||||
|
tmp += strlen(curr_pipe->args[curr_pipe->args_am - 1]) + 1;
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (j >= line_size)
|
||||||
|
{
|
||||||
|
char *ap_line = read_line();
|
||||||
|
|
||||||
|
line = realloc(line, strlen(line) + strlen(ap_line) + 1);
|
||||||
|
line = strcat(line, ap_line);
|
||||||
|
|
||||||
|
line_size = strlen(line);
|
||||||
|
|
||||||
|
free(ap_line);
|
||||||
|
i--;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
i = j;
|
||||||
|
|
||||||
|
if (tmp[0] == ' ')
|
||||||
|
{
|
||||||
|
tmp++;
|
||||||
|
i++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (line[i] == ';' || (line[i] == '&' && line[i + 1] == '&') || (line[i] == '|' && line[i + 1] == '|'))
|
||||||
|
{
|
||||||
|
free_tmp[i] = '\0';
|
||||||
|
if (line[i - 1] != ' ')
|
||||||
|
{
|
||||||
|
append_to_str_arr(&(curr_pipe->args), &curr_pipe->args_am, trim_string(tmp, false));
|
||||||
|
tmp += strlen(curr_pipe->args[curr_pipe->args_am - 1]) + 1;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
tmp++;
|
||||||
|
|
||||||
|
curr_pipe->args = realloc(curr_pipe->args, sizeof(char *) * (curr_pipe->args_am + 1));
|
||||||
|
curr_pipe->args[curr_pipe->args_am] = NULL;
|
||||||
|
|
||||||
|
switch (line[i])
|
||||||
|
{
|
||||||
|
case ';':
|
||||||
|
curr_cmd->sep_next = SEMICOLON_SEP;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case '&':
|
||||||
|
curr_cmd->sep_next = AND_SEP;
|
||||||
|
i++;
|
||||||
|
tmp++;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case '|':
|
||||||
|
curr_cmd->sep_next = OR_SEP;
|
||||||
|
i++;
|
||||||
|
tmp++;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (tmp[0] == ' ')
|
||||||
|
{
|
||||||
|
tmp++;
|
||||||
|
i++;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (curr_pipe->args[0] != NULL)
|
||||||
|
{
|
||||||
|
if (strcmp(curr_pipe->args[0], "!") == 0)
|
||||||
|
{
|
||||||
|
curr_cmd->stat.invert = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
cmds_p *next = new_cmd();
|
||||||
|
|
||||||
|
curr_cmd->next = next;
|
||||||
|
|
||||||
|
curr_cmd = curr_cmd->next;
|
||||||
|
|
||||||
|
curr_pipe = curr_cmd->pipe;
|
||||||
|
}
|
||||||
|
else if (line[i] == '|')
|
||||||
|
{
|
||||||
|
if (line[i - 1] != ' ')
|
||||||
|
{
|
||||||
|
append_to_str_arr(&(curr_pipe->args), &curr_pipe->args_am, trim_string(tmp, false));
|
||||||
|
tmp += strlen(curr_pipe->args[curr_pipe->args_am - 1]) + 1;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
tmp++;
|
||||||
|
|
||||||
|
curr_pipe->args = realloc(curr_pipe->args, sizeof(char *) * (curr_pipe->args_am + 1));
|
||||||
|
curr_pipe->args[curr_pipe->args_am] = NULL;
|
||||||
|
|
||||||
|
if (tmp[0] == ' ')
|
||||||
|
{
|
||||||
|
tmp++;
|
||||||
|
i++;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (curr_pipe->args[0] != NULL)
|
||||||
|
{
|
||||||
|
if (strcmp(curr_pipe->args[0], "!") == 0)
|
||||||
|
{
|
||||||
|
curr_cmd->stat.invert = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
cmd_pipe *next = new_cmd_pipe();
|
||||||
|
|
||||||
|
curr_pipe->next = next;
|
||||||
|
curr_cmd->pipes_am++;
|
||||||
|
curr_pipe = curr_pipe->next;
|
||||||
|
}
|
||||||
|
else if (line[i] == ' ')
|
||||||
|
{
|
||||||
|
free_tmp[i] = '\0';
|
||||||
|
|
||||||
|
char *ap_string = trim_string(tmp, false), **exp = calloc(0, sizeof(char *));
|
||||||
|
|
||||||
|
int sz = expand_wildcatrd(&exp, ap_string);
|
||||||
|
|
||||||
|
for (int l = 0; l < sz; l++)
|
||||||
|
append_to_str_arr(&(curr_pipe->args), &curr_pipe->args_am, exp[l]);
|
||||||
|
|
||||||
|
tmp += strlen(ap_string) + 1;
|
||||||
|
}
|
||||||
|
else if (line[i] == '#')
|
||||||
|
{
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (tmp[0] != '\0' && tmp[0] != '#')
|
||||||
|
{
|
||||||
|
if (tmp[strlen(tmp) - 1] != '"')
|
||||||
|
{
|
||||||
|
char *ap_string = trim_string(tmp, false), **exp = calloc(0, sizeof(char *));
|
||||||
|
|
||||||
|
int sz = expand_wildcatrd(&exp, ap_string);
|
||||||
|
|
||||||
|
for (int l = 0; l < sz; l++)
|
||||||
|
append_to_str_arr(&(curr_pipe->args), &curr_pipe->args_am, exp[l]);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
append_to_str_arr(&curr_pipe->args, &curr_pipe->args_am, trim_string(tmp, false));
|
||||||
|
}
|
||||||
|
|
||||||
|
if (curr_pipe->args[0] != NULL)
|
||||||
|
{
|
||||||
|
if (strcmp(curr_pipe->args[0], "!") == 0)
|
||||||
|
{
|
||||||
|
curr_cmd->stat.invert = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
free(free_tmp);
|
||||||
|
|
||||||
|
if (curr_pipe->args[curr_pipe->args_am] != NULL)
|
||||||
|
{
|
||||||
|
curr_pipe->args = realloc(curr_pipe->args, sizeof(char *) * (curr_pipe->args_am + 1));
|
||||||
|
curr_pipe->args[curr_pipe->args_am] = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
curr_pipe->pipefd[1] = STDOUT_FILENO;
|
||||||
|
|
||||||
|
return coms;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief Expands wildcard in argument
|
||||||
|
*
|
||||||
|
* @param arr
|
||||||
|
* @param input
|
||||||
|
* @return int
|
||||||
|
*/
|
||||||
|
int expand_wildcatrd(char ***arr, char *input)
|
||||||
|
{
|
||||||
|
int size = 0;
|
||||||
|
|
||||||
|
if (strchr(input, '*') || strchr(input, '~') || strchr(input, '?'))
|
||||||
|
{
|
||||||
|
glob_t globbuf;
|
||||||
|
int t = glob(input, GLOB_TILDE, NULL, &globbuf);
|
||||||
|
if (t != 0)
|
||||||
|
perror("Glob");
|
||||||
|
|
||||||
|
for (int j = 0; j < globbuf.gl_pathc; j++)
|
||||||
|
append_to_str_arr(arr, &size, globbuf.gl_pathv[j]);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
append_to_str_arr(arr, &size, input);
|
||||||
|
|
||||||
|
return size;
|
||||||
|
}
|
@ -12,7 +12,7 @@ void exit_shell();
|
|||||||
t_ init_term();
|
t_ init_term();
|
||||||
|
|
||||||
// Main
|
// Main
|
||||||
int main(int argc, char **argv)
|
int main()
|
||||||
{
|
{
|
||||||
term = init_term();
|
term = init_term();
|
||||||
|
|
||||||
@ -25,7 +25,6 @@ int main(int argc, char **argv)
|
|||||||
// Init
|
// Init
|
||||||
t_ init_term()
|
t_ init_term()
|
||||||
{
|
{
|
||||||
t_ term;
|
|
||||||
|
|
||||||
FILE *log_file = fopen("/var/log/mshell.log", "w");
|
FILE *log_file = fopen("/var/log/mshell.log", "w");
|
||||||
if (log_file == NULL)
|
if (log_file == NULL)
|
||||||
@ -42,11 +41,11 @@ t_ init_term()
|
|||||||
|
|
||||||
term.hist.length = 0;
|
term.hist.length = 0;
|
||||||
term.hist.pos = -1;
|
term.hist.pos = -1;
|
||||||
term.hist.content = (char **)malloc(sizeof(char *) * term.hist.length);
|
term.hist.content = calloc(term.hist.length, sizeof(char *));
|
||||||
|
|
||||||
term.hist.sub.length = -1;
|
term.hist.sub.length = -1;
|
||||||
term.hist.sub.pos = -1;
|
term.hist.sub.pos = -1;
|
||||||
term.hist.sub.content = (char **)malloc(sizeof(char *) * 0);
|
term.hist.sub.content = calloc(0, sizeof(char *));
|
||||||
|
|
||||||
term.last_status = 0;
|
term.last_status = 0;
|
||||||
|
|
||||||
|
91
src/prompt.c
Normal file
91
src/prompt.c
Normal file
@ -0,0 +1,91 @@
|
|||||||
|
#include "../include/shell.h"
|
||||||
|
#include "../include/utils.h"
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief Get IP adress of system
|
||||||
|
*
|
||||||
|
* @return char*
|
||||||
|
*/
|
||||||
|
char *get_ip_addr()
|
||||||
|
{
|
||||||
|
struct ifaddrs *host, *tmp;
|
||||||
|
getifaddrs(&host);
|
||||||
|
tmp = host;
|
||||||
|
char *ip = NULL;
|
||||||
|
|
||||||
|
while (tmp)
|
||||||
|
{
|
||||||
|
if (tmp->ifa_addr && tmp->ifa_addr->sa_family == AF_INET)
|
||||||
|
{
|
||||||
|
struct sockaddr_in *pAddr = (struct sockaddr_in *)tmp->ifa_addr;
|
||||||
|
ip = inet_ntoa(pAddr->sin_addr);
|
||||||
|
if (strncmp(ip, "127", 3) != 0)
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
tmp = tmp->ifa_next;
|
||||||
|
}
|
||||||
|
|
||||||
|
freeifaddrs(host);
|
||||||
|
|
||||||
|
return ip;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief Creates prompt string
|
||||||
|
*
|
||||||
|
* @return char*
|
||||||
|
*/
|
||||||
|
char *compose_prompt()
|
||||||
|
{
|
||||||
|
// New line
|
||||||
|
char *prompt = strdup("");
|
||||||
|
|
||||||
|
// Username
|
||||||
|
char *username = getenv("USER");
|
||||||
|
if (username == NULL)
|
||||||
|
username = "none";
|
||||||
|
prompt = realloc(prompt, strlen(prompt) + strlen("\033[97;44m") + strlen(username) + 2);
|
||||||
|
prompt = strcat(prompt, "\033[97;44m");
|
||||||
|
prompt = strcat(prompt, username);
|
||||||
|
|
||||||
|
// Current host ip
|
||||||
|
char *ip = get_ip_addr();
|
||||||
|
if (ip == NULL)
|
||||||
|
ip = "none";
|
||||||
|
prompt = realloc(prompt, strlen(prompt) + 1 + strlen(ip) + strlen("\033[39m") + strlen("\033[0m") + 2);
|
||||||
|
prompt = strcat(prompt, "@");
|
||||||
|
prompt = strcat(prompt, ip);
|
||||||
|
prompt = strcat(prompt, "\033[39m");
|
||||||
|
prompt = strcat(prompt, "\033[0m");
|
||||||
|
prompt = strcat(prompt, ":");
|
||||||
|
|
||||||
|
// Current path
|
||||||
|
char *full_path = get_curr_dir_name();
|
||||||
|
prompt = realloc(prompt, strlen(prompt) + strlen("\033[92;1m") + strlen("\033[0m") + strlen(full_path) + 2);
|
||||||
|
prompt = strcat(prompt, "\033[92;1m");
|
||||||
|
prompt = strcat(prompt, full_path);
|
||||||
|
prompt = strcat(prompt, "\033[0m");
|
||||||
|
free(full_path);
|
||||||
|
|
||||||
|
// Previous status
|
||||||
|
if (term.last_status != 0)
|
||||||
|
{
|
||||||
|
char *status = malloc(snprintf(NULL, 0, " \033[91m%d\033[39m", term.last_status));
|
||||||
|
sprintf(status, " \033[91m%d\033[39m", term.last_status);
|
||||||
|
prompt = realloc(prompt, strlen(prompt) + strlen(status) + 1);
|
||||||
|
prompt = strcat(prompt, status);
|
||||||
|
free(status);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Permissions specific character before user input
|
||||||
|
prompt = realloc(prompt, strlen(prompt) + 4);
|
||||||
|
if (getuid() == 0)
|
||||||
|
{
|
||||||
|
prompt = strcat(prompt, "\n# ");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
prompt = strcat(prompt, "\n% ");
|
||||||
|
|
||||||
|
return prompt;
|
||||||
|
}
|
423
src/shell.c
423
src/shell.c
@ -1,26 +1,13 @@
|
|||||||
#include "../include/shell.h"
|
#include "../include/shell.h"
|
||||||
|
#include "../include/utils.h"
|
||||||
#include "../include/input.h"
|
#include "../include/input.h"
|
||||||
#include "../include/output.h"
|
#include "../include/output.h"
|
||||||
#include "../include/utils.h"
|
|
||||||
|
|
||||||
/* Global definitions */
|
|
||||||
char *builtin[] = {
|
|
||||||
"cd",
|
|
||||||
"exit",
|
|
||||||
"exec"};
|
|
||||||
|
|
||||||
int (*builtin_func[])(char **) = {
|
|
||||||
&sh_cd,
|
|
||||||
&sh_exit,
|
|
||||||
&sh_exec};
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @brief Function for main loop. It prints prompt, reads user's input and executes it
|
* @brief Function for main loop. It prints prompt, reads user's input and executes it
|
||||||
*/
|
*/
|
||||||
void process_command()
|
void process_command()
|
||||||
{
|
{
|
||||||
cmds_p *coms = NULL;
|
|
||||||
|
|
||||||
char *prompt = compose_prompt();
|
char *prompt = compose_prompt();
|
||||||
print_str(prompt, strlen(prompt));
|
print_str(prompt, strlen(prompt));
|
||||||
|
|
||||||
@ -30,13 +17,13 @@ void process_command()
|
|||||||
free(line);
|
free(line);
|
||||||
line = tmp;
|
line = tmp;
|
||||||
|
|
||||||
process_line(line, &coms);
|
cmds_p *coms = process_line(line);
|
||||||
|
|
||||||
cmds_p *curr = coms;
|
cmds_p *curr = coms;
|
||||||
int status = 0;
|
int status = 0;
|
||||||
while (curr != NULL)
|
while (curr != NULL)
|
||||||
{
|
{
|
||||||
status = execute(curr);
|
status = execute_with_pipes(curr);
|
||||||
|
|
||||||
if (curr->stat.invert)
|
if (curr->stat.invert)
|
||||||
{
|
{
|
||||||
@ -46,8 +33,7 @@ void process_command()
|
|||||||
status = 0;
|
status = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
curr->stat.s = status;
|
curr->stat.s = term.last_status = status;
|
||||||
term.last_status = curr->stat.s;
|
|
||||||
|
|
||||||
if (curr->sep_next == AND_SEP)
|
if (curr->sep_next == AND_SEP)
|
||||||
{
|
{
|
||||||
@ -65,348 +51,7 @@ void process_command()
|
|||||||
|
|
||||||
free(line);
|
free(line);
|
||||||
free(prompt);
|
free(prompt);
|
||||||
}
|
free_cmds_p(coms);
|
||||||
|
|
||||||
/**
|
|
||||||
* @brief Extracts command and its arguments from line
|
|
||||||
*
|
|
||||||
* @param line
|
|
||||||
* @param args
|
|
||||||
*/
|
|
||||||
int process_line(char *line, cmds_p **coms)
|
|
||||||
{
|
|
||||||
char *tmp = strdup(line), *free_tmp = tmp;
|
|
||||||
int line_size = strlen(line), args_am = 0;
|
|
||||||
|
|
||||||
*coms = new_cmd();
|
|
||||||
|
|
||||||
cmds_p *curr_cmd = *coms;
|
|
||||||
|
|
||||||
for (int i = 0; i < line_size; i++)
|
|
||||||
{
|
|
||||||
if (line[i] == '"')
|
|
||||||
{
|
|
||||||
tmp++;
|
|
||||||
i++;
|
|
||||||
int j = i;
|
|
||||||
for (; j < line_size; j++)
|
|
||||||
if (line[j] == '"')
|
|
||||||
{
|
|
||||||
free_tmp[j] = '\0';
|
|
||||||
|
|
||||||
append_to_str_arr(&(curr_cmd->args), &args_am, trim_string(tmp, false));
|
|
||||||
|
|
||||||
tmp += strlen(curr_cmd->args[args_am - 1]) + 1;
|
|
||||||
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (j >= line_size)
|
|
||||||
{
|
|
||||||
char *ap_line = read_line();
|
|
||||||
|
|
||||||
line = realloc(line, strlen(line) + strlen(ap_line) + 1);
|
|
||||||
line = strcat(line, ap_line);
|
|
||||||
|
|
||||||
line_size = strlen(line);
|
|
||||||
|
|
||||||
free(ap_line);
|
|
||||||
i--;
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
i = j;
|
|
||||||
|
|
||||||
if (tmp[0] == ' ')
|
|
||||||
{
|
|
||||||
tmp++;
|
|
||||||
i++;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else if (line[i] == ';' || (line[i] == '&' && line[i + 1] == '&') || (line[i] == '|' && line[i + 1] == '|'))
|
|
||||||
{
|
|
||||||
free_tmp[i] = '\0';
|
|
||||||
if (line[i - 1] != ' ')
|
|
||||||
{
|
|
||||||
append_to_str_arr(&(curr_cmd->args), &args_am, trim_string(tmp, false));
|
|
||||||
tmp += strlen(curr_cmd->args[args_am - 1]) + 1;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
tmp++;
|
|
||||||
|
|
||||||
curr_cmd->args = realloc(curr_cmd->args, sizeof(char *) * (args_am + 1));
|
|
||||||
curr_cmd->args[args_am] = NULL;
|
|
||||||
|
|
||||||
switch (line[i])
|
|
||||||
{
|
|
||||||
case ';':
|
|
||||||
curr_cmd->sep_next = SEMICOLON_SEP;
|
|
||||||
break;
|
|
||||||
|
|
||||||
case '&':
|
|
||||||
curr_cmd->sep_next = AND_SEP;
|
|
||||||
i++;
|
|
||||||
tmp++;
|
|
||||||
break;
|
|
||||||
|
|
||||||
case '|':
|
|
||||||
curr_cmd->sep_next = OR_SEP;
|
|
||||||
i++;
|
|
||||||
tmp++;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (tmp[0] == ' ')
|
|
||||||
{
|
|
||||||
tmp++;
|
|
||||||
i++;
|
|
||||||
}
|
|
||||||
|
|
||||||
cmds_p *next = new_cmd();
|
|
||||||
|
|
||||||
curr_cmd->next = next;
|
|
||||||
curr_cmd = curr_cmd->next;
|
|
||||||
args_am = 0;
|
|
||||||
}
|
|
||||||
else if (line[i] == ' ')
|
|
||||||
{
|
|
||||||
free_tmp[i] = '\0';
|
|
||||||
|
|
||||||
char *ap_string = trim_string(tmp, false), **exp = malloc(0);
|
|
||||||
|
|
||||||
int sz = expand_wildcatrd(&exp, ap_string);
|
|
||||||
|
|
||||||
for (int l = 0; l < sz; l++)
|
|
||||||
append_to_str_arr(&(curr_cmd->args), &args_am, exp[l]);
|
|
||||||
|
|
||||||
tmp += strlen(ap_string) + 1;
|
|
||||||
}
|
|
||||||
else if (line[i] == '#')
|
|
||||||
{
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (tmp[0] != '\0' && tmp[0] != '#')
|
|
||||||
{
|
|
||||||
if (tmp[strlen(tmp) - 1] != '"')
|
|
||||||
{
|
|
||||||
char *ap_string = trim_string(tmp, false), **exp = malloc(0);
|
|
||||||
|
|
||||||
int sz = expand_wildcatrd(&exp, ap_string);
|
|
||||||
|
|
||||||
for (int l = 0; l < sz; l++)
|
|
||||||
append_to_str_arr(&(curr_cmd->args), &args_am, exp[l]);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
append_to_str_arr(&(curr_cmd->args), &args_am, trim_string(tmp, false));
|
|
||||||
}
|
|
||||||
|
|
||||||
if (curr_cmd->args[0] != NULL)
|
|
||||||
{
|
|
||||||
if (strcmp(curr_cmd->args[0], "!") == 0)
|
|
||||||
{
|
|
||||||
curr_cmd->stat.invert = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
free(free_tmp);
|
|
||||||
|
|
||||||
curr_cmd->args = realloc(curr_cmd->args, sizeof(char *) * (args_am + 1));
|
|
||||||
curr_cmd->args[args_am] = NULL;
|
|
||||||
return args_am;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @brief Executes either builtin or system command
|
|
||||||
*
|
|
||||||
* @param args
|
|
||||||
* @return int
|
|
||||||
*/
|
|
||||||
int execute(cmds_p *command)
|
|
||||||
{
|
|
||||||
if (command->args[0] == NULL)
|
|
||||||
return 1;
|
|
||||||
|
|
||||||
char **args = command->args;
|
|
||||||
|
|
||||||
if (strcmp(args[0], "!") == 0)
|
|
||||||
args = slice_array(args, 1, -1, true);
|
|
||||||
|
|
||||||
for (int i = 0; i < BUILTIN_NUM; i++)
|
|
||||||
if (strcmp(args[0], builtin[i]) == 0)
|
|
||||||
return (*builtin_func[i])(args);
|
|
||||||
|
|
||||||
return launch(args);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @brief Launches separate process and executes specified command
|
|
||||||
*
|
|
||||||
* @param args
|
|
||||||
* @return int
|
|
||||||
*/
|
|
||||||
int launch(char **args)
|
|
||||||
{
|
|
||||||
pid_t pid, wpid;
|
|
||||||
int status;
|
|
||||||
|
|
||||||
pid = fork();
|
|
||||||
|
|
||||||
if (pid == 0)
|
|
||||||
{
|
|
||||||
sh_exec(args);
|
|
||||||
}
|
|
||||||
else if (pid < 0)
|
|
||||||
{
|
|
||||||
perror("myshell1");
|
|
||||||
}
|
|
||||||
else
|
|
||||||
do
|
|
||||||
wpid = waitpid(pid, &status, WUNTRACED);
|
|
||||||
while (!WIFEXITED(status) && !WIFSIGNALED(status));
|
|
||||||
|
|
||||||
change_mode(1);
|
|
||||||
|
|
||||||
return status;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @brief Shell builtin command. Changes current working directory
|
|
||||||
*
|
|
||||||
* @param args
|
|
||||||
* @return int
|
|
||||||
*/
|
|
||||||
int sh_cd(char **args)
|
|
||||||
{
|
|
||||||
if (args[1] == NULL)
|
|
||||||
chdir(getenv("HOME"));
|
|
||||||
else if (chdir(args[1]) < 0)
|
|
||||||
perror("myshell2");
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @brief Shell builtin command. Exits shell
|
|
||||||
*
|
|
||||||
* @param args
|
|
||||||
* @return int
|
|
||||||
*/
|
|
||||||
int sh_exit(char **args)
|
|
||||||
{
|
|
||||||
exit(0);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @brief Shell builtin command. Executes command replacing shell with it
|
|
||||||
*
|
|
||||||
* @param args
|
|
||||||
* @return int
|
|
||||||
*/
|
|
||||||
int sh_exec(char **args)
|
|
||||||
{
|
|
||||||
change_mode(0);
|
|
||||||
signal(SIGINT, SIG_DFL);
|
|
||||||
|
|
||||||
if (strcmp(args[0], "exec") == 0)
|
|
||||||
args = slice_array(args, 1, -1, 1);
|
|
||||||
|
|
||||||
if (execvp(args[0], args) < 0)
|
|
||||||
{
|
|
||||||
perror("myshell3");
|
|
||||||
}
|
|
||||||
|
|
||||||
exit(EXIT_FAILURE);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @brief Get IP adress of system
|
|
||||||
*
|
|
||||||
* @return char*
|
|
||||||
*/
|
|
||||||
char *get_ip_addr()
|
|
||||||
{
|
|
||||||
struct ifaddrs *host, *tmp;
|
|
||||||
getifaddrs(&host);
|
|
||||||
tmp = host;
|
|
||||||
char *ip = NULL;
|
|
||||||
|
|
||||||
while (tmp)
|
|
||||||
{
|
|
||||||
if (tmp->ifa_addr && tmp->ifa_addr->sa_family == AF_INET)
|
|
||||||
{
|
|
||||||
struct sockaddr_in *pAddr = (struct sockaddr_in *)tmp->ifa_addr;
|
|
||||||
ip = inet_ntoa(pAddr->sin_addr);
|
|
||||||
if (strncmp(ip, "127", 3) != 0)
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
tmp = tmp->ifa_next;
|
|
||||||
}
|
|
||||||
|
|
||||||
freeifaddrs(host);
|
|
||||||
|
|
||||||
return ip;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @brief Creates prompt string
|
|
||||||
*
|
|
||||||
* @return char*
|
|
||||||
*/
|
|
||||||
char *compose_prompt()
|
|
||||||
{
|
|
||||||
// New line
|
|
||||||
char *prompt = strdup("");
|
|
||||||
|
|
||||||
// Username
|
|
||||||
char *username = getenv("USER");
|
|
||||||
if (username == NULL)
|
|
||||||
username = "none";
|
|
||||||
prompt = realloc(prompt, strlen(prompt) + strlen("\033[97;44m") + strlen(username) + 2);
|
|
||||||
prompt = strcat(prompt, "\033[97;44m");
|
|
||||||
prompt = strcat(prompt, username);
|
|
||||||
|
|
||||||
// Current host ip
|
|
||||||
char *ip = get_ip_addr();
|
|
||||||
if (ip == NULL)
|
|
||||||
ip = "none";
|
|
||||||
prompt = realloc(prompt, strlen(prompt) + 1 + strlen(ip) + strlen("\033[39m") + strlen("\033[0m") + 2);
|
|
||||||
prompt = strcat(prompt, "@");
|
|
||||||
prompt = strcat(prompt, ip);
|
|
||||||
prompt = strcat(prompt, "\033[39m");
|
|
||||||
prompt = strcat(prompt, "\033[0m");
|
|
||||||
prompt = strcat(prompt, ":");
|
|
||||||
|
|
||||||
// Current path
|
|
||||||
char *full_path = get_curr_dir_name();
|
|
||||||
prompt = realloc(prompt, strlen(prompt) + strlen("\033[92;1m") + strlen("\033[0m") + strlen(full_path) + 2);
|
|
||||||
prompt = strcat(prompt, "\033[92;1m");
|
|
||||||
prompt = strcat(prompt, full_path);
|
|
||||||
prompt = strcat(prompt, "\033[0m");
|
|
||||||
free(full_path);
|
|
||||||
|
|
||||||
// Previous status
|
|
||||||
if (term.last_status != 0)
|
|
||||||
{
|
|
||||||
char *status = malloc(snprintf(NULL, 0, " \033[91m%d\033[39m", term.last_status));
|
|
||||||
sprintf(status, " \033[91m%d\033[39m", term.last_status);
|
|
||||||
prompt = realloc(prompt, strlen(prompt) + strlen(status) + 1);
|
|
||||||
prompt = strcat(prompt, status);
|
|
||||||
free(status);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Permissions specific character before user input
|
|
||||||
prompt = realloc(prompt, strlen(prompt) + 4);
|
|
||||||
if (getuid() == 0)
|
|
||||||
{
|
|
||||||
prompt = strcat(prompt, "\n# ");
|
|
||||||
}
|
|
||||||
else
|
|
||||||
prompt = strcat(prompt, "\n% ");
|
|
||||||
|
|
||||||
return prompt;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -417,7 +62,10 @@ char *compose_prompt()
|
|||||||
cmds_p *new_cmd()
|
cmds_p *new_cmd()
|
||||||
{
|
{
|
||||||
cmds_p *new = malloc(sizeof(cmds_p));
|
cmds_p *new = malloc(sizeof(cmds_p));
|
||||||
new->args = calloc(0, sizeof(char *));
|
|
||||||
|
new->pipe = new_cmd_pipe();
|
||||||
|
new->pipes_am = 1;
|
||||||
|
|
||||||
new->stat.s = 0;
|
new->stat.s = 0;
|
||||||
new->sep_next = NO_SEP;
|
new->sep_next = NO_SEP;
|
||||||
new->stat.invert = false;
|
new->stat.invert = false;
|
||||||
@ -426,20 +74,45 @@ cmds_p *new_cmd()
|
|||||||
return new;
|
return new;
|
||||||
}
|
}
|
||||||
|
|
||||||
int expand_wildcatrd(char ***arr, char *input)
|
/**
|
||||||
|
* @brief Initialize command pipe structure
|
||||||
|
*
|
||||||
|
* @return cmd_pipe*
|
||||||
|
*/
|
||||||
|
cmd_pipe *new_cmd_pipe()
|
||||||
{
|
{
|
||||||
int size = 0;
|
cmd_pipe *new = malloc(sizeof(cmd_pipe));
|
||||||
|
new->args = calloc(0, sizeof(char *));
|
||||||
|
new->args_am = 0;
|
||||||
|
new->next = NULL;
|
||||||
|
new->pipefd[0] = new->pipefd[1] = -1;
|
||||||
|
|
||||||
if (strchr(input, '*') || strchr(input, '~') || strchr(input, '?'))
|
return new;
|
||||||
{
|
}
|
||||||
glob_t globbuf;
|
|
||||||
int t = glob(input, GLOB_TILDE, NULL, &globbuf);
|
void free_cmds_p(cmds_p *root)
|
||||||
|
{
|
||||||
for (int j = 0; j < globbuf.gl_pathc; j++)
|
if (root != NULL)
|
||||||
append_to_str_arr(arr, &size, globbuf.gl_pathv[j]);
|
{
|
||||||
}
|
if (root->pipe != NULL)
|
||||||
else
|
free_cmd_pipe(root->pipe);
|
||||||
append_to_str_arr(arr, &size, input);
|
|
||||||
|
if (root->next != NULL)
|
||||||
return size;
|
free_cmds_p(root->next);
|
||||||
|
|
||||||
|
free(root);
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
void free_cmd_pipe(cmd_pipe *root)
|
||||||
|
{
|
||||||
|
if (root != NULL)
|
||||||
|
{
|
||||||
|
if (root->next != NULL)
|
||||||
|
free_cmd_pipe(root->next);
|
||||||
|
|
||||||
|
free(root);
|
||||||
|
}
|
||||||
|
return;
|
||||||
}
|
}
|
@ -7,7 +7,7 @@
|
|||||||
*/
|
*/
|
||||||
struct tree_node *get_new_node()
|
struct tree_node *get_new_node()
|
||||||
{
|
{
|
||||||
struct tree_node *node = (struct tree_node *)malloc(sizeof(struct tree_node));
|
struct tree_node *node = malloc(sizeof(struct tree_node));
|
||||||
node->is_leaf = 0;
|
node->is_leaf = 0;
|
||||||
|
|
||||||
for (int i = 0; i < ALPHABET_SIZE; i++)
|
for (int i = 0; i < ALPHABET_SIZE; i++)
|
||||||
@ -119,7 +119,7 @@ ssize_t list_strings_containing(struct tree_node *root, char *key, char ***strin
|
|||||||
|
|
||||||
free(*strings);
|
free(*strings);
|
||||||
|
|
||||||
*strings = malloc(amount * sizeof(char *));
|
*strings = calloc(amount, sizeof(char *));
|
||||||
|
|
||||||
struct tree_node *current = root;
|
struct tree_node *current = root;
|
||||||
|
|
||||||
|
@ -56,7 +56,7 @@ int sep_string(char *line, char ***toks, char *sep)
|
|||||||
char *tmp_line = strdup(line);
|
char *tmp_line = strdup(line);
|
||||||
char *free_tmp_line = tmp_line;
|
char *free_tmp_line = tmp_line;
|
||||||
int n = 0;
|
int n = 0;
|
||||||
*toks = malloc(sizeof(char *) * n);
|
*toks = calloc(n, sizeof(char *));
|
||||||
|
|
||||||
char *tmp = NULL;
|
char *tmp = NULL;
|
||||||
while ((tmp = strsep(&tmp_line, sep)) != NULL)
|
while ((tmp = strsep(&tmp_line, sep)) != NULL)
|
||||||
@ -101,11 +101,12 @@ char *trim_string(char *str, bool leave_trailing_space)
|
|||||||
* @brief Frees array of strings
|
* @brief Frees array of strings
|
||||||
*
|
*
|
||||||
* @param arr
|
* @param arr
|
||||||
|
* @param sz
|
||||||
*/
|
*/
|
||||||
void free_str_arr(char **arr)
|
void free_str_arr(char **arr, int sz)
|
||||||
{
|
{
|
||||||
if (arr[0] != NULL)
|
if (arr[0] != NULL)
|
||||||
for (int i = 0; i < sizeof(arr) / sizeof(char *); i++)
|
for (int i = 0; i < sz; i++)
|
||||||
free(arr[i]);
|
free(arr[i]);
|
||||||
free(arr);
|
free(arr);
|
||||||
}
|
}
|
||||||
@ -126,7 +127,7 @@ char **slice_array(char **arr, int beg, int end, bool asc)
|
|||||||
if (end == -1)
|
if (end == -1)
|
||||||
end = get_null_term_arr_size(arr);
|
end = get_null_term_arr_size(arr);
|
||||||
|
|
||||||
char **new_arr = malloc(abs(end - beg) * sizeof(char *));
|
char **new_arr = calloc(abs(end - beg), sizeof(char *));
|
||||||
|
|
||||||
if (asc)
|
if (asc)
|
||||||
for (int i = beg, j = 0; i < end; i++, j++)
|
for (int i = beg, j = 0; i < end; i++, j++)
|
||||||
|
Loading…
x
Reference in New Issue
Block a user