Swithced to 2 spaces indent

This commit is contained in:
Dmitriy Shishkov 2021-02-15 00:24:38 +05:00
parent bd61469247
commit 1a4792f417
No known key found for this signature in database
GPG Key ID: 7CAE12ED13853CAC
15 changed files with 1458 additions and 1458 deletions

View File

@ -7,17 +7,17 @@
#include <string.h> #include <string.h>
enum keys { enum keys {
DELETE_KEY = 1000, DELETE_KEY = 1000,
UP_KEY, UP_KEY,
DOWN_KEY, DOWN_KEY,
LEFT_KEY, LEFT_KEY,
RIGHT_KEY, RIGHT_KEY,
HOME_KEY, HOME_KEY,
END_KEY, END_KEY,
BACKSPACE_KEY, BACKSPACE_KEY,
ENTER_KEY, ENTER_KEY,
TAB_KEY, TAB_KEY,
ESCAPE_KEY ESCAPE_KEY
}; };
void change_mode(int on); void change_mode(int on);

View File

@ -25,60 +25,60 @@
// Types definitions // Types definitions
struct hist_sub struct hist_sub
{ {
char **content; char **content;
int length; int length;
int pos; int pos;
}; };
struct history struct history
{ {
char **content; char **content;
ssize_t length; ssize_t length;
int pos; int pos;
FILE *file; FILE *file;
struct hist_sub sub; struct hist_sub sub;
}; };
struct status struct status
{ {
int s; int s;
bool invert; bool invert;
}; };
typedef enum typedef enum
{ {
NO_SEP, NO_SEP,
SEMICOLON_SEP, SEMICOLON_SEP,
AND_SEP, AND_SEP,
OR_SEP OR_SEP
} cmd_sep; } cmd_sep;
typedef struct pipes typedef struct pipes
{ {
char **args; char **args;
ssize_t args_am; ssize_t args_am;
int pipefd[2]; int pipefd[2];
struct pipes *next; struct pipes *next;
} cmd_pipe; } cmd_pipe;
typedef struct commands typedef struct commands
{ {
cmd_pipe *pipe; cmd_pipe *pipe;
ssize_t pipes_am; ssize_t pipes_am;
char **envs; char **envs;
ssize_t envs_am; ssize_t envs_am;
struct status stat; struct status stat;
struct commands *next; struct commands *next;
cmd_sep sep_next; cmd_sep sep_next;
} cmds_p; } cmds_p;
typedef struct typedef struct
{ {
struct history hist; struct history hist;
int last_status; int last_status;
} t_; } t_;
//Globals defenition //Globals defenition

View File

@ -8,9 +8,9 @@
struct tree_node struct tree_node
{ {
struct tree_node *child[ALPHABET_SIZE]; struct tree_node *child[ALPHABET_SIZE];
int is_leaf; int is_leaf;
}; };
struct tree_node *get_new_node(); struct tree_node *get_new_node();

View File

@ -13,16 +13,16 @@
*/ */
bool check_if_executable(char *path, char *file_name) bool check_if_executable(char *path, char *file_name)
{ {
char *file_path = malloc(strlen(path) + strlen(file_name) + 2); char *file_path = malloc(strlen(path) + strlen(file_name) + 2);
file_path[0] = '\0'; file_path[0] = '\0';
file_path = strcat(file_path, path); file_path = strcat(file_path, path);
file_path = strcat(file_path, "/"); file_path = strcat(file_path, "/");
file_path = strcat(file_path, file_name); file_path = strcat(file_path, file_name);
bool ret = access(file_path, X_OK) == 0; bool ret = access(file_path, X_OK) == 0;
free(file_path); free(file_path);
return ret; return ret;
} }
/** /**
@ -35,38 +35,38 @@ bool check_if_executable(char *path, char *file_name)
*/ */
ssize_t get_dir_list(char ***dir_list, char *path, int ex) ssize_t get_dir_list(char ***dir_list, char *path, int ex)
{ {
DIR *dir; DIR *dir;
struct dirent *ent; struct dirent *ent;
if ((dir = opendir(path)) == NULL) if ((dir = opendir(path)) == NULL)
{
perror("\nOpendir");
return -1;
}
ssize_t n = 0;
while ((ent = readdir(dir)) != NULL)
{
if (ex != 0 && !check_if_executable(path, ent->d_name))
continue;
if (strcmp(ent->d_name, ".") == 0 || strcmp(ent->d_name, "..") == 0)
continue;
n++;
*dir_list = realloc(*dir_list, sizeof(char *) * n);
(*dir_list)[n - 1] = strdup(ent->d_name);
if (ent->d_type == DT_DIR)
{ {
perror("\nOpendir"); (*dir_list)[n - 1] = realloc((*dir_list)[n - 1], strlen((*dir_list)[n - 1]) + 2);
return -1; (*dir_list)[n - 1] = strcat((*dir_list)[n - 1], "/");
} }
}
ssize_t n = 0; closedir(dir);
while ((ent = readdir(dir)) != NULL)
{
if (ex != 0 && !check_if_executable(path, ent->d_name))
continue;
if (strcmp(ent->d_name, ".") == 0 || strcmp(ent->d_name, "..") == 0) return n;
continue;
n++;
*dir_list = realloc(*dir_list, sizeof(char *) * n);
(*dir_list)[n - 1] = strdup(ent->d_name);
if (ent->d_type == DT_DIR)
{
(*dir_list)[n - 1] = realloc((*dir_list)[n - 1], strlen((*dir_list)[n - 1]) + 2);
(*dir_list)[n - 1] = strcat((*dir_list)[n - 1], "/");
}
}
closedir(dir);
return n;
} }
/** /**
@ -78,14 +78,14 @@ ssize_t get_dir_list(char ***dir_list, char *path, int ex)
*/ */
ssize_t append_builtin_list(char ***commands_list, ssize_t *size) ssize_t append_builtin_list(char ***commands_list, ssize_t *size)
{ {
for (int i = 0; i < 3; i++) for (int i = 0; i < 3; i++)
{ {
(*size)++; (*size)++;
*commands_list = realloc(*commands_list, sizeof(char *) * *size); *commands_list = realloc(*commands_list, sizeof(char *) * *size);
(*commands_list)[*size - 1] = strdup(builtin[i]); (*commands_list)[*size - 1] = strdup(builtin[i]);
} }
return *size; return *size;
} }
/** /**
@ -98,70 +98,70 @@ 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 = calloc(0, sizeof(char *)); char **args = NULL, **folders = calloc(0, sizeof(char *));
ssize_t sz; ssize_t sz;
int am = sep_string(line, &args, " "), path_depth = 0; 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)
{
path_depth = sep_string(last_arg, &folders, "/");
*to_complete = strdup(folders[path_depth - 1]);
char *curr_pos = NULL;
if (last_arg[0] == '/')
{ {
path_depth = sep_string(last_arg, &folders, "/"); curr_pos = strdup("");
*to_complete = strdup(folders[path_depth - 1]); }
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] != '|')
char *curr_pos = NULL; {
curr_pos = strdup(".");
if (last_arg[0] == '/') }
{ else if (last_arg[0] == '.' && last_arg[1] == '/')
curr_pos = strdup(""); {
} curr_pos = strdup(folders[0]);
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(".");
}
else if (last_arg[0] == '.' && last_arg[1] == '/')
{
curr_pos = strdup(folders[0]);
}
else
goto ABSOLUTE;
for (int i = 0; i < (path_depth - 1); i++)
{
curr_pos = realloc(curr_pos, strlen(curr_pos) + strlen(folders[i]) + 2);
curr_pos = strcat(curr_pos, "/");
curr_pos = strcat(curr_pos, folders[i]);
}
sz = get_dir_list(opts, curr_pos, 0);
free(curr_pos);
} }
else else
goto ABSOLUTE;
for (int i = 0; i < (path_depth - 1); i++)
{ {
*to_complete = strdup(line); curr_pos = realloc(curr_pos, strlen(curr_pos) + strlen(folders[i]) + 2);
ABSOLUTE: curr_pos = strcat(curr_pos, "/");
if ((*to_complete)[0] != '\0' && am >= 2) curr_pos = strcat(curr_pos, folders[i]);
if (strcmp(args[am - 2], "||") == 0 || strcmp(args[am - 2], "&&") == 0 || strcmp(args[am - 2], ";") == 0)
*to_complete = strdup("");
sz = get_path_commands_list(opts);
append_builtin_list(opts, &sz);
} }
sz = get_dir_list(opts, curr_pos, 0);
free_str_arr(args, am); free(curr_pos);
free_str_arr(folders, path_depth); }
else
{
*to_complete = strdup(line);
ABSOLUTE:
if ((*to_complete)[0] != '\0' && am >= 2)
if (strcmp(args[am - 2], "||") == 0 || strcmp(args[am - 2], "&&") == 0 || strcmp(args[am - 2], ";") == 0)
*to_complete = strdup("");
if (sz < 0) sz = get_path_commands_list(opts);
return sz;
if ((*to_complete)[0] != '\0') append_builtin_list(opts, &sz);
{ }
sz = filter_options(opts, &sz, *to_complete);
}
free_str_arr(args, am);
free_str_arr(folders, path_depth);
if (sz < 0)
return sz; return sz;
if ((*to_complete)[0] != '\0')
{
sz = filter_options(opts, &sz, *to_complete);
}
return sz;
} }
/** /**
@ -174,59 +174,59 @@ ssize_t get_complete_options(char ***opts, char *line, char **to_complete)
*/ */
ssize_t filter_options(char ***comp_list, ssize_t *size, char *filter_string) ssize_t filter_options(char ***comp_list, ssize_t *size, char *filter_string)
{ {
if (*size < 0) if (*size < 0)
return *size;
struct tree_node *child_dirs_root = get_new_node();
for (ssize_t i = 0; i < *size; i++)
insert_tree(child_dirs_root, (*comp_list)[i]);
char **folders = NULL;
int path_depth = sep_string(filter_string, &folders, "/");
char *last_option = strdup(filter_string);
if (path_depth > 0)
{
free(last_option);
last_option = folders[path_depth - 1];
}
*size = list_strings_containing(child_dirs_root, last_option, comp_list);
free_tree(child_dirs_root);
free_str_arr(folders, path_depth);
return *size; return *size;
struct tree_node *child_dirs_root = get_new_node();
for (ssize_t i = 0; i < *size; i++)
insert_tree(child_dirs_root, (*comp_list)[i]);
char **folders = NULL;
int path_depth = sep_string(filter_string, &folders, "/");
char *last_option = strdup(filter_string);
if (path_depth > 0)
{
free(last_option);
last_option = folders[path_depth - 1];
}
*size = list_strings_containing(child_dirs_root, last_option, comp_list);
free_tree(child_dirs_root);
free_str_arr(folders, path_depth);
return *size;
} }
ssize_t get_path_commands_list(char ***opts) ssize_t get_path_commands_list(char ***opts)
{ {
char *paths_str = get_env_var("PATH"); char *paths_str = get_env_var("PATH");
if (paths_str == NULL) if (paths_str == NULL)
return -1; return -1;
char **paths = NULL; char **paths = NULL;
int path_am = sep_string(paths_str, &paths, ":"); int path_am = sep_string(paths_str, &paths, ":");
ssize_t sz = 0; ssize_t sz = 0;
for (int i = 0; i < path_am; i++) for (int i = 0; i < path_am; i++)
{
char **tmp_list = malloc(0);
ssize_t tmp_sz = get_dir_list(&tmp_list, paths[i], 1);
if (tmp_sz < 0)
continue;
for (long j = 0; j < tmp_sz; j++)
{ {
char **tmp_list = malloc(0); if (i != 0)
ssize_t tmp_sz = get_dir_list(&tmp_list, paths[i], 1); if (str_is_in_arr(*opts, sz, tmp_list[j]))
if (tmp_sz < 0) continue;
continue;
for (long j = 0; j < tmp_sz; j++) append_to_str_arr(opts, &sz, tmp_list[j]);
{
if (i != 0)
if (str_is_in_arr(*opts, sz, tmp_list[j]))
continue;
append_to_str_arr(opts, &sz, tmp_list[j]);
}
} }
}
free(paths_str); free(paths_str);
return sz; return sz;
} }

View File

@ -4,16 +4,16 @@
/* Global definitions */ /* Global definitions */
char *builtin[] = { char *builtin[] = {
"cd", "cd",
"exec", "exec",
"export", "export",
"exit"}; "exit"};
int (*builtin_func[])(char **) = { int (*builtin_func[])(char **) = {
&sh_cd, &sh_cd,
&sh_exec, &sh_exec,
&sh_export, &sh_export,
&sh_exit}; &sh_exit};
/** /**
* @brief Redirects command input/output and executes it * @brief Redirects command input/output and executes it
@ -23,28 +23,28 @@ int (*builtin_func[])(char **) = {
*/ */
int execute_with_pipes(cmds_p *command) int execute_with_pipes(cmds_p *command)
{ {
cmd_pipe *curr = command->pipe; cmd_pipe *curr = command->pipe;
int status = 0, tmp_fd[2] = {-1}; int status = 0, tmp_fd[2] = {-1};
for (int i = 0; i < command->pipes_am - 1 && curr != NULL && status == 0; i++) for (int i = 0; i < command->pipes_am - 1 && curr != NULL && status == 0; i++)
{
if (pipe(tmp_fd) < 0)
{ {
if (pipe(tmp_fd) < 0) perror("pipe");
{ return 1;
perror("pipe");
return 1;
}
curr->pipefd[1] = tmp_fd[1];
curr->next->pipefd[0] = tmp_fd[0];
status = execute(curr, command->envs);
curr = curr->next;
} }
curr->pipefd[1] = tmp_fd[1];
curr->next->pipefd[0] = tmp_fd[0];
status = execute(curr, command->envs); status = execute(curr, command->envs);
return status; curr = curr->next;
}
status = execute(curr, command->envs);
return status;
} }
/** /**
@ -55,22 +55,22 @@ int execute_with_pipes(cmds_p *command)
*/ */
int execute(cmd_pipe *command, char **envp) int execute(cmd_pipe *command, char **envp)
{ {
if (command->args[0] == NULL) if (command->args[0] == NULL)
return 1; return 1;
char **args = command->args; char **args = command->args;
if (strcmp(args[0], "!") == 0) if (strcmp(args[0], "!") == 0)
{ {
args = slice_array(args, 1, -1, true); args = slice_array(args, 1, -1, true);
command->args = args; command->args = args;
} }
for (int i = 0; i < BUILTIN_NUM; i++) for (int i = 0; i < BUILTIN_NUM; i++)
if (strcmp(args[0], builtin[i]) == 0) if (strcmp(args[0], builtin[i]) == 0)
return (*builtin_func[i])(args); return (*builtin_func[i])(args);
return launch(command, envp); return launch(command, envp);
} }
/** /**
@ -81,37 +81,37 @@ int execute(cmd_pipe *command, char **envp)
*/ */
int launch(cmd_pipe *command, char **envp) int launch(cmd_pipe *command, char **envp)
{ {
pid_t pid, wpid; pid_t pid, wpid;
int status; int status;
pid = fork(); pid = fork();
if (pid == 0) if (pid == 0)
{ {
redirect_fd(command->pipefd[0], STDIN_FILENO); redirect_fd(command->pipefd[0], STDIN_FILENO);
redirect_fd(command->pipefd[1], STDOUT_FILENO); redirect_fd(command->pipefd[1], STDOUT_FILENO);
sh_exec(command->args, envp); sh_exec(command->args, envp);
} }
else if (pid < 0) else if (pid < 0)
{ {
perror("mshell"); perror("mshell");
return -1; return -1;
} }
else else
{ {
for (int j = 0; j < 2; j++) for (int j = 0; j < 2; j++)
if (command->pipefd[j] > 2) if (command->pipefd[j] > 2)
close(command->pipefd[j]); close(command->pipefd[j]);
do do
wpid = waitpid(pid, &status, WUNTRACED); wpid = waitpid(pid, &status, WUNTRACED);
while (!WIFEXITED(status) && !WIFSIGNALED(status)); while (!WIFEXITED(status) && !WIFSIGNALED(status));
} }
change_mode(1); change_mode(1);
return status; return status;
} }
/** /**
@ -122,23 +122,23 @@ int launch(cmd_pipe *command, char **envp)
*/ */
int sh_cd(char **args) int sh_cd(char **args)
{ {
if (args[1] == NULL) if (args[1] == NULL)
{ {
char *home_env = get_env_var("HOME"); char *home_env = get_env_var("HOME");
if (home_env == NULL) if (home_env == NULL)
return 1; return 1;
chdir(home_env); chdir(home_env);
free(home_env); free(home_env);
} }
else if (chdir(args[1]) < 0) else if (chdir(args[1]) < 0)
{ {
perror("cd"); perror("cd");
return 1; return 1;
} }
return 0; return 0;
} }
/** /**
@ -149,44 +149,44 @@ int sh_cd(char **args)
*/ */
int sh_exec(char **args, char **envp) int sh_exec(char **args, char **envp)
{ {
change_mode(0); change_mode(0);
signal(SIGINT, SIG_DFL); signal(SIGINT, SIG_DFL);
if (strcmp(args[0], "exec") == 0) if (strcmp(args[0], "exec") == 0)
args = slice_array(args, 1, -1, 1); args = slice_array(args, 1, -1, 1);
if (mexecvpe(args[0], args, envp) < 0) if (mexecvpe(args[0], args, envp) < 0)
{ {
perror("mshell"); perror("mshell");
} }
exit(EXIT_FAILURE); exit(EXIT_FAILURE);
} }
int sh_export(char **args) int sh_export(char **args)
{ {
if (args[1] == NULL) if (args[1] == NULL)
{ {
for (int i = 0; environ[i] != NULL; i++) for (int i = 0; environ[i] != NULL; i++)
printf("declare -x %s\n", environ[i]); printf("declare -x %s\n", environ[i]);
return 0; return 0;
}
for (int i = 1; args[i] != NULL; i++)
{
char *name = strtok(args[i], "="), *val = strtok(NULL, "=");
if (name == NULL || val == NULL)
{
fprintf(stderr, "export: wrong arguments\n");
return 1;
} }
for (int i = 1; args[i] != NULL; i++) setenv(name, val, true);
{
char *name = strtok(args[i], "="), *val = strtok(NULL, "=");
if (name == NULL || val == NULL) printf("Set %s variable to %s\n", name, getenv(name));
{ }
fprintf(stderr, "export: wrong arguments\n");
return 1;
}
setenv(name, val, true);
printf("Set %s variable to %s\n", name, getenv(name));
}
} }
/** /**
@ -197,84 +197,84 @@ int sh_export(char **args)
*/ */
int sh_exit(char **args) int sh_exit(char **args)
{ {
exit(0); exit(0);
} }
void redirect_fd(int old, int new) void redirect_fd(int old, int new)
{ {
if (old > 0) if (old > 0)
if (old != new) if (old != new)
{ {
if (dup2(old, new) < 0) if (dup2(old, new) < 0)
perror("dup2"); perror("dup2");
else else
close(old); close(old);
} }
} }
int mexecvpe(char *file, char **argv, char **envp) int mexecvpe(char *file, char **argv, char **envp)
{ {
if (file == NULL || file[0] == '\0') if (file == NULL || file[0] == '\0')
return -1; return -1;
if (envp == NULL || envp[0] == NULL) if (envp == NULL || envp[0] == NULL)
execvp(file, argv); execvp(file, argv);
int res = complete_envs(&envp); int res = complete_envs(&envp);
if (res < 0) if (res < 0)
return -1; return -1;
if (strchr(file, '/') != NULL) if (strchr(file, '/') != NULL)
execve(file, argv, envp); execve(file, argv, envp);
char *path = get_env_var("PATH"); char *path = get_env_var("PATH");
if (path == NULL) if (path == NULL)
return -1; return -1;
size_t file_len = strlen(file) + 1, path_len = strlen(path) + 1; size_t file_len = strlen(file) + 1, path_len = strlen(path) + 1;
char buff[file_len + path_len + 1], *subp; char buff[file_len + path_len + 1], *subp;
for (char *p = path;; p = subp) for (char *p = path;; p = subp)
{ {
subp = strchr(p, ':'); subp = strchr(p, ':');
memcpy(buff, p, subp - p); memcpy(buff, p, subp - p);
buff[subp - p] = '/'; buff[subp - p] = '/';
memcpy(buff + (subp - p) + (p < subp), file, file_len); memcpy(buff + (subp - p) + (p < subp), file, file_len);
execve(buff, argv, envp); execve(buff, argv, envp);
if (*subp++ == '\0') if (*subp++ == '\0')
break; break;
} }
} }
int complete_envs(char ***envp) int complete_envs(char ***envp)
{ {
for (int i = 0; environ[i] != NULL; i++) for (int i = 0; environ[i] != NULL; i++)
{
char *name = strtok(strdup(environ[i]), "=");
bool modded = false;
for (int j = 0; (*envp)[j] != NULL; j++)
{ {
char *name = strtok(strdup(environ[i]), "="); if (strncmp((*envp)[j], name, strlen(name)) == 0)
bool modded = false; {
for (int j = 0; (*envp)[j] != NULL; j++) modded = true;
{ (*envp)[j] = strdup(environ[i]);
if (strncmp((*envp)[j], name, strlen(name)) == 0) }
{
modded = true;
(*envp)[j] = strdup(environ[i]);
}
}
if (!modded)
{
ssize_t size = get_null_term_arr_size(*envp);
int rs = append_to_str_arr(envp, &size, environ[i]);
if (rs < 0)
return rs;
}
} }
return 0; if (!modded)
{
ssize_t size = get_null_term_arr_size(*envp);
int rs = append_to_str_arr(envp, &size, environ[i]);
if (rs < 0)
return rs;
}
}
return 0;
} }

View File

@ -9,34 +9,34 @@
*/ */
void append_to_history(char *line) void append_to_history(char *line)
{ {
if (term.hist.length > 0) if (term.hist.length > 0)
if (strcmp(line, term.hist.content[term.hist.length - 1]) == 0) if (strcmp(line, term.hist.content[term.hist.length - 1]) == 0)
return; return;
if (strncmp(line, "exit", strlen("exit")) == 0) if (strncmp(line, "exit", strlen("exit")) == 0)
return; return;
if (line[0] == '\0') if (line[0] == '\0')
return; return;
term.hist.length++; term.hist.length++;
term.hist.content = (char **)realloc(term.hist.content, term.hist.length * sizeof(char *)); term.hist.content = (char **)realloc(term.hist.content, term.hist.length * sizeof(char *));
term.hist.content[term.hist.length - 1] = strdup(line); term.hist.content[term.hist.length - 1] = strdup(line);
fseek(term.hist.file, 0, SEEK_END); fseek(term.hist.file, 0, SEEK_END);
fprintf(term.hist.file, "%s\n", line); fprintf(term.hist.file, "%s\n", line);
fflush(term.hist.file); fflush(term.hist.file);
term.hist.pos = -1; term.hist.pos = -1;
} }
void clear_sub_history() void clear_sub_history()
{ {
free(term.hist.sub.content); free(term.hist.sub.content);
term.hist.sub.content = malloc(0); term.hist.sub.content = malloc(0);
term.hist.sub.length = -1; term.hist.sub.length = -1;
term.hist.sub.pos = -1; term.hist.sub.pos = -1;
} }
/** /**
@ -47,101 +47,101 @@ void clear_sub_history()
*/ */
char *previous_hist_entry(char *line) char *previous_hist_entry(char *line)
{ {
static int h_size; static int h_size;
int f_len = get_num_of_lines(term.hist.file); int f_len = get_num_of_lines(term.hist.file);
if (term.hist.length != f_len) if (term.hist.length != f_len)
{
clear_sub_history();
term.hist.pos = -1;
char **tmp_h = calloc(sizeof(char *), 0);
int tmp_h_pos = 0;
fseek(term.hist.file, 0, SEEK_END);
while (term.hist.length + tmp_h_pos < f_len && ftell(term.hist.file) > 1)
{ {
clear_sub_history(); char ch = '\0';
term.hist.pos = -1; size_t llen = 0;
while (ch != '\n')
{
fseek(term.hist.file, -2, SEEK_CUR);
if (ftell(term.hist.file) < 1)
break;
ch = fgetc(term.hist.file);
}
char **tmp_h = calloc(sizeof(char *), 0); long f_pos = ftell(term.hist.file);
int tmp_h_pos = 0;
fseek(term.hist.file, 0, SEEK_END); tmp_h = realloc(tmp_h, sizeof(char *) * (tmp_h_pos + 1));
getline(&tmp_h[tmp_h_pos], &llen, term.hist.file);
while (term.hist.length + tmp_h_pos < f_len && ftell(term.hist.file) > 1) fseek(term.hist.file, f_pos, SEEK_SET);
{
char ch = '\0';
size_t llen = 0;
while (ch != '\n')
{
fseek(term.hist.file, -2, SEEK_CUR);
if (ftell(term.hist.file) < 1)
break;
ch = fgetc(term.hist.file);
}
long f_pos = ftell(term.hist.file); if (tmp_h[tmp_h_pos][strlen(tmp_h[tmp_h_pos]) - 1] == '\n')
tmp_h[tmp_h_pos][strlen(tmp_h[tmp_h_pos]) - 1] = '\0';
tmp_h = realloc(tmp_h, sizeof(char *) * (tmp_h_pos + 1)); tmp_h_pos++;
getline(&tmp_h[tmp_h_pos], &llen, term.hist.file);
fseek(term.hist.file, f_pos, SEEK_SET); fseek(term.hist.file, -2, SEEK_CUR);
if (tmp_h[tmp_h_pos][strlen(tmp_h[tmp_h_pos]) - 1] == '\n')
tmp_h[tmp_h_pos][strlen(tmp_h[tmp_h_pos]) - 1] = '\0';
tmp_h_pos++;
fseek(term.hist.file, -2, SEEK_CUR);
}
for (int i = 0; i < tmp_h_pos; i++)
append_to_str_arr(&term.hist.content, &term.hist.length, tmp_h[i]);
free_str_arr(tmp_h, tmp_h_pos);
} }
if (line == NULL) for (int i = 0; i < tmp_h_pos; i++)
{ append_to_str_arr(&term.hist.content, &term.hist.length, tmp_h[i]);
if (term.hist.pos + 1 == term.hist.length)
return NULL;
term.hist.pos++; free_str_arr(tmp_h, tmp_h_pos);
return strdup(term.hist.content[term.hist.length - term.hist.pos - 1]); }
}
if (term.hist.sub.length < 0) if (line == NULL)
{ {
if (term.hist.pos + 1 == term.hist.length)
return NULL;
term.hist.pos++;
return strdup(term.hist.content[term.hist.length - term.hist.pos - 1]);
}
if (term.hist.sub.length < 0)
{
term.hist.sub.length++;
for (int i = 0; i < term.hist.length; i++)
if (strncmp(term.hist.content[i], line, strlen(line)) == 0)
{
term.hist.sub.length++; term.hist.sub.length++;
for (int i = 0; i < term.hist.length; i++)
if (strncmp(term.hist.content[i], line, strlen(line)) == 0)
{
term.hist.sub.length++;
term.hist.sub.content = realloc(term.hist.sub.content, term.hist.sub.length * sizeof(char *));
term.hist.sub.content[term.hist.sub.length - 1] = term.hist.content[i];
}
h_size = term.hist.sub.length;
}
else
{
int tmp_len = term.hist.sub.length;
term.hist.sub.length = 0;
for (int i = 0; i < tmp_len; i++)
if (strncmp(term.hist.sub.content[i], line, strlen(line)) == 0)
{
term.hist.sub.length++;
term.hist.sub.content[term.hist.sub.length - 1] = term.hist.sub.content[i];
}
term.hist.sub.content = realloc(term.hist.sub.content, term.hist.sub.length * sizeof(char *)); term.hist.sub.content = realloc(term.hist.sub.content, term.hist.sub.length * sizeof(char *));
term.hist.sub.content[term.hist.sub.length - 1] = term.hist.content[i];
}
h_size = term.hist.sub.length;
}
else
{
int tmp_len = term.hist.sub.length;
term.hist.sub.length = 0;
for (int i = 0; i < tmp_len; i++)
if (strncmp(term.hist.sub.content[i], line, strlen(line)) == 0)
{
term.hist.sub.length++;
term.hist.sub.content[term.hist.sub.length - 1] = term.hist.sub.content[i];
}
term.hist.sub.content = realloc(term.hist.sub.content, term.hist.sub.length * sizeof(char *));
if (term.hist.sub.length != h_size) if (term.hist.sub.length != h_size)
{
h_size = term.hist.sub.length;
term.hist.sub.pos = -1;
}
}
term.hist.sub.pos++;
if (term.hist.sub.pos >= term.hist.sub.length)
{ {
term.hist.sub.pos = term.hist.sub.length - 1; h_size = term.hist.sub.length;
return NULL; term.hist.sub.pos = -1;
} }
}
return strdup(term.hist.sub.content[term.hist.sub.length - term.hist.sub.pos - 1]); term.hist.sub.pos++;
if (term.hist.sub.pos >= term.hist.sub.length)
{
term.hist.sub.pos = term.hist.sub.length - 1;
return NULL;
}
return strdup(term.hist.sub.content[term.hist.sub.length - term.hist.sub.pos - 1]);
} }
/** /**
@ -152,90 +152,90 @@ char *previous_hist_entry(char *line)
*/ */
char *next_hist_entry(char *line) char *next_hist_entry(char *line)
{ {
if (line == NULL) if (line == NULL)
{ {
if (term.hist.pos == 0) if (term.hist.pos == 0)
return NULL; return NULL;
term.hist.pos--; term.hist.pos--;
return strdup(term.hist.content[term.hist.length - term.hist.pos - 1]); return strdup(term.hist.content[term.hist.length - term.hist.pos - 1]);
} }
if (term.hist.sub.pos <= 0) if (term.hist.sub.pos <= 0)
return NULL; return NULL;
term.hist.sub.pos--; term.hist.sub.pos--;
return strdup(term.hist.sub.content[term.hist.sub.length - term.hist.sub.pos - 1]); return strdup(term.hist.sub.content[term.hist.sub.length - term.hist.sub.pos - 1]);
} }
void init_history() void init_history()
{ {
term.hist.length = 0; term.hist.length = 0;
term.hist.pos = -1; term.hist.pos = -1;
term.hist.content = calloc(term.hist.length, sizeof(char *)); 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 = calloc(0, sizeof(char *)); term.hist.sub.content = calloc(0, sizeof(char *));
open_history_file(); open_history_file();
size_t sz = 0; size_t sz = 0;
int len = 0; int len = 0;
char *buf = NULL; char *buf = NULL;
while ((len = getline(&buf, &sz, term.hist.file)) != -1) while ((len = getline(&buf, &sz, term.hist.file)) != -1)
{ {
char *tmp = strdup(buf); char *tmp = strdup(buf);
if (tmp[strlen(tmp) - 1] == '\n') if (tmp[strlen(tmp) - 1] == '\n')
tmp[strlen(tmp) - 1] = '\0'; tmp[strlen(tmp) - 1] = '\0';
append_to_str_arr(&term.hist.content, &term.hist.length, tmp); append_to_str_arr(&term.hist.content, &term.hist.length, tmp);
free(tmp); free(tmp);
} }
free(buf); free(buf);
} }
void open_history_file() void open_history_file()
{ {
char *data_path; char *data_path;
if ((data_path = get_env_var("XDG_DATA_HOME")) == NULL) if ((data_path = get_env_var("XDG_DATA_HOME")) == NULL)
{ {
char *user_home; char *user_home;
if ((user_home = get_env_var("HOME")) == NULL) if ((user_home = get_env_var("HOME")) == NULL)
exit(EXIT_FAILURE); exit(EXIT_FAILURE);
data_path = user_home; data_path = user_home;
data_path = realloc(data_path, strlen(data_path) + strlen("/.local/share/") + 1); data_path = realloc(data_path, strlen(data_path) + strlen("/.local/share/") + 1);
data_path = strcat(data_path, "/.local/share/"); data_path = strcat(data_path, "/.local/share/");
} }
struct stat st = {0}; struct stat st = {0};
if (stat(data_path, &st) < 0) if (stat(data_path, &st) < 0)
{ {
printf("No %s dir, tying to create\n", data_path); printf("No %s dir, tying to create\n", data_path);
char *buff = malloc(strlen("mkdir -p ") + strlen(data_path) + 1); char *buff = malloc(strlen("mkdir -p ") + strlen(data_path) + 1);
buff = strdup("mkdir -p "); buff = strdup("mkdir -p ");
buff = strcat(buff, data_path); buff = strcat(buff, data_path);
system(buff); system(buff);
} }
data_path = realloc(data_path, strlen(data_path) + strlen(".mshistory") + 1); data_path = realloc(data_path, strlen(data_path) + strlen(".mshistory") + 1);
data_path = strcat(data_path, ".mshistory"); data_path = strcat(data_path, ".mshistory");
if ((term.hist.file = fopen(data_path, "a+")) == NULL) if ((term.hist.file = fopen(data_path, "a+")) == NULL)
{ {
perror("open"); perror("open");
exit(EXIT_FAILURE); exit(EXIT_FAILURE);
} }
fseek(term.hist.file, 0, SEEK_SET); fseek(term.hist.file, 0, SEEK_SET);
free(data_path); free(data_path);
} }
void close_history_file() void close_history_file()
{ {
fclose(term.hist.file); fclose(term.hist.file);
} }

View File

@ -10,17 +10,17 @@
*/ */
void change_mode(int on) void change_mode(int on)
{ {
static struct termios oldt, newt; static struct termios oldt, newt;
if (on) if (on)
{ {
tcgetattr(STDIN_FILENO, &oldt); tcgetattr(STDIN_FILENO, &oldt);
newt = oldt; newt = oldt;
newt.c_lflag &= ~(ICANON | ECHO); newt.c_lflag &= ~(ICANON | ECHO);
tcsetattr(STDIN_FILENO, TCSANOW, &newt); tcsetattr(STDIN_FILENO, TCSANOW, &newt);
} }
else else
tcsetattr(STDIN_FILENO, TCSANOW, &oldt); tcsetattr(STDIN_FILENO, TCSANOW, &oldt);
} }
/** /**
@ -31,82 +31,82 @@ void change_mode(int on)
*/ */
char *read_line() char *read_line()
{ {
int c; int c;
int n = 0, pos = 0; int n = 0, pos = 0;
char *line = strdup(""); char *line = strdup("");
while (read(STDIN_FILENO, &c, 1)) while (read(STDIN_FILENO, &c, 1))
{
c = process_keypress(c);
if (c)
{ {
c = process_keypress(c); switch (c)
{
case UP_KEY:
up_key(&pos, &n, &line);
break;
if (c) case DOWN_KEY:
down_key(&pos, &n, &line);
break;
default:
term.hist.pos = -1;
switch (c)
{ {
switch (c)
{
case UP_KEY:
up_key(&pos, &n, &line);
break;
case DOWN_KEY: case DELETE_KEY:
down_key(&pos, &n, &line); delete_key(pos, &n, &line);
break; break;
default: case LEFT_KEY:
term.hist.pos = -1; move_left(&pos);
break;
switch (c) case RIGHT_KEY:
{ move_right(&pos, n);
break;
case DELETE_KEY: case HOME_KEY:
delete_key(pos, &n, &line); home_key(&pos);
break; break;
case LEFT_KEY: case END_KEY:
move_left(&pos); end_key(&pos, n);
break; break;
case RIGHT_KEY: case BACKSPACE_KEY:
move_right(&pos, n); backspace_key(&pos, &n, &line);
break; break;
case HOME_KEY: case ENTER_KEY:
home_key(&pos); new_line(line);
break;
case END_KEY: return line;
end_key(&pos, n); break;
break;
case BACKSPACE_KEY: case TAB_KEY:
backspace_key(&pos, &n, &line); {
break; tab_key(&pos, &n, &line);
case ENTER_KEY:
new_line(line);
return line;
break;
case TAB_KEY:
{
tab_key(&pos, &n, &line);
}
break;
case ESCAPE_KEY:
break;
default:
if ((c > 31 && c < 127) || (c > 127 && c < 255))
printable_key(&pos, &n, (char)c, &line);
break;
}
}
} }
break;
case ESCAPE_KEY:
break;
default:
if ((c > 31 && c < 127) || (c > 127 && c < 255))
printable_key(&pos, &n, (char)c, &line);
break;
}
}
} }
return line; }
return line;
} }
/** /**
@ -117,72 +117,72 @@ char *read_line()
*/ */
int process_keypress(char c) int process_keypress(char c)
{ {
char seq[3]; char seq[3];
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) < 0) if (read(STDIN_FILENO, &seq[i], 1) < 0)
return ESCAPE_KEY;
if (seq[0] == '[')
{
if (seq[1] >= '0' && seq[1] <= '9')
{
read(STDIN_FILENO, &seq[2], 1);
if (seq[2] == '~')
{
switch (seq[1])
{
case '3':
return DELETE_KEY;
break;
}
}
}
switch (seq[1])
{
case 'A':
return UP_KEY;
break;
case 'B':
return DOWN_KEY;
break;
case 'C':
return RIGHT_KEY;
break;
case 'D':
return LEFT_KEY;
break;
case 'F':
return END_KEY;
break;
case 'H':
return HOME_KEY;
break;
}
}
return ESCAPE_KEY; return ESCAPE_KEY;
} if (seq[0] == '[')
else if (c == 127)
{ {
return BACKSPACE_KEY; if (seq[1] >= '0' && seq[1] <= '9')
} {
else if (c == '\n') read(STDIN_FILENO, &seq[2], 1);
{ if (seq[2] == '~')
return ENTER_KEY; {
} switch (seq[1])
else if (c == '\t') {
{ case '3':
return TAB_KEY; return DELETE_KEY;
} break;
else }
{ }
return c; }
switch (seq[1])
{
case 'A':
return UP_KEY;
break;
case 'B':
return DOWN_KEY;
break;
case 'C':
return RIGHT_KEY;
break;
case 'D':
return LEFT_KEY;
break;
case 'F':
return END_KEY;
break;
case 'H':
return HOME_KEY;
break;
}
} }
return ESCAPE_KEY;
}
else if (c == 127)
{
return BACKSPACE_KEY;
}
else if (c == '\n')
{
return ENTER_KEY;
}
else if (c == '\t')
{
return TAB_KEY;
}
else
{
return c;
}
} }
/** /**
@ -194,20 +194,20 @@ int process_keypress(char c)
*/ */
void free_input(int *pos, int *n, char **line) void free_input(int *pos, int *n, char **line)
{ {
char *buff = strdup(""); char *buff = strdup("");
size_t buff_size = 1; size_t buff_size = 1;
free(*line); free(*line);
*line = strdup(""); *line = strdup("");
*n = 0; *n = 0;
for (int i = 0; i < *pos; i++) for (int i = 0; i < *pos; i++)
append_to_buff(&buff, &buff_size, "\033[D", 3); append_to_buff(&buff, &buff_size, "\033[D", 3);
append_to_buff(&buff, &buff_size, "\033[K", 3); append_to_buff(&buff, &buff_size, "\033[K", 3);
*pos = *n; *pos = *n;
print_str(buff, buff_size); print_str(buff, buff_size);
free(buff); free(buff);
} }

View File

@ -15,22 +15,22 @@
*/ */
void delete_key(int pos, int *n, char **line) void delete_key(int pos, int *n, char **line)
{ {
if (pos < *n) if (pos < *n)
{ {
remove_on_pos(line, pos + 1); remove_on_pos(line, pos + 1);
(*n)--; (*n)--;
char *buff = strdup(""); char *buff = strdup("");
size_t buff_size = 1; size_t buff_size = 1;
append_to_buff(&buff, &buff_size, "\0337", 2); append_to_buff(&buff, &buff_size, "\0337", 2);
append_to_buff(&buff, &buff_size, *line + pos, *n - pos); append_to_buff(&buff, &buff_size, *line + pos, *n - pos);
append_to_buff(&buff, &buff_size, " ", 1); append_to_buff(&buff, &buff_size, " ", 1);
append_to_buff(&buff, &buff_size, "\0338", 2); append_to_buff(&buff, &buff_size, "\0338", 2);
print_str(buff, buff_size); print_str(buff, buff_size);
free(buff); free(buff);
} }
} }
/** /**
@ -43,35 +43,35 @@ 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)
{ {
char *buff = strdup("\033[K"); char *buff = strdup("\033[K");
size_t buff_size = 4; size_t buff_size = 4;
char *entry = NULL; char *entry = NULL;
if (*pos == 0) if (*pos == 0)
entry = previous_hist_entry(NULL); entry = previous_hist_entry(NULL);
else else
{ {
(*line)[*pos] = '\0'; (*line)[*pos] = '\0';
entry = previous_hist_entry(*line); entry = previous_hist_entry(*line);
} }
if (entry == NULL) if (entry == NULL)
{ {
free(buff);
return;
}
free(*line);
*line = entry;
*n = strlen(*line);
append_to_buff(&buff, &buff_size, "\0337", 2);
append_to_buff(&buff, &buff_size, *line + *pos, *n - *pos);
append_to_buff(&buff, &buff_size, "\0338", 2);
print_str(buff, buff_size);
free(buff); free(buff);
return;
}
free(*line);
*line = entry;
*n = strlen(*line);
append_to_buff(&buff, &buff_size, "\0337", 2);
append_to_buff(&buff, &buff_size, *line + *pos, *n - *pos);
append_to_buff(&buff, &buff_size, "\0338", 2);
print_str(buff, buff_size);
free(buff);
} }
/** /**
@ -84,57 +84,57 @@ void up_key(int *pos, int *n, char **line)
void down_key(int *pos, int *n, char **line) void down_key(int *pos, int *n, char **line)
{ {
char *buff = strdup("\033[K"); char *buff = strdup("\033[K");
size_t buff_size = 4; size_t buff_size = 4;
char *entry = NULL; char *entry = NULL;
if (*pos == 0) if (*pos == 0)
{
if (term.hist.pos <= 0)
{ {
if (term.hist.pos <= 0) if (term.hist.pos == 0)
{ term.hist.pos--;
if (term.hist.pos == 0)
term.hist.pos--;
free(buff); free(buff);
free_input(pos, n, line); free_input(pos, n, line);
return; return;
}
entry = next_hist_entry(NULL);
}
else
{
(*line)[*pos] = '\0';
if (term.hist.sub.pos <= 0)
{
if (term.hist.sub.pos == 0)
term.hist.sub.pos--;
free(buff);
free_input(pos, n, line);
return;
}
entry = next_hist_entry(*line);
} }
if (entry == NULL) entry = next_hist_entry(NULL);
}
else
{
(*line)[*pos] = '\0';
if (term.hist.sub.pos <= 0)
{ {
free(buff); if (term.hist.sub.pos == 0)
return; term.hist.sub.pos--;
free(buff);
free_input(pos, n, line);
return;
} }
*line = entry; entry = next_hist_entry(*line);
*n = strlen(*line); }
append_to_buff(&buff, &buff_size, "\0337", 2); if (entry == NULL)
append_to_buff(&buff, &buff_size, *line + *pos, *n - *pos); {
append_to_buff(&buff, &buff_size, "\0338", 2);
print_str(buff, buff_size);
free(buff); free(buff);
return;
}
*line = entry;
*n = strlen(*line);
append_to_buff(&buff, &buff_size, "\0337", 2);
append_to_buff(&buff, &buff_size, *line + *pos, *n - *pos);
append_to_buff(&buff, &buff_size, "\0338", 2);
print_str(buff, buff_size);
free(buff);
} }
/** /**
@ -144,12 +144,12 @@ void down_key(int *pos, int *n, char **line)
*/ */
void move_left(int *pos) void move_left(int *pos)
{ {
clear_sub_history(); clear_sub_history();
if (*pos > 0) if (*pos > 0)
{ {
print_str("\033[D", 3); print_str("\033[D", 3);
(*pos)--; (*pos)--;
} }
} }
/** /**
@ -160,13 +160,13 @@ void move_left(int *pos)
*/ */
void move_right(int *pos, int n) void move_right(int *pos, int n)
{ {
if (*pos < n) if (*pos < n)
{ {
print_str("\033[C", 3); print_str("\033[C", 3);
(*pos)++; (*pos)++;
clear_sub_history(); clear_sub_history();
} }
} }
/** /**
@ -176,18 +176,18 @@ void move_right(int *pos, int n)
*/ */
void home_key(int *pos) void home_key(int *pos)
{ {
clear_sub_history(); clear_sub_history();
char *buff = strdup(""); char *buff = strdup("");
size_t buff_size = 1; size_t buff_size = 1;
for (int i = 0; i < *pos; i++) for (int i = 0; i < *pos; i++)
append_to_buff(&buff, &buff_size, "\033[D", 3); append_to_buff(&buff, &buff_size, "\033[D", 3);
print_str(buff, buff_size); print_str(buff, buff_size);
*pos = 0; *pos = 0;
free(buff); free(buff);
} }
/** /**
@ -198,17 +198,17 @@ void home_key(int *pos)
*/ */
void end_key(int *pos, int n) void end_key(int *pos, int n)
{ {
char *buff = strdup(""); char *buff = strdup("");
size_t buff_size = 1; size_t buff_size = 1;
for (int i = 0; i < n - *pos; i++) for (int i = 0; i < n - *pos; i++)
append_to_buff(&buff, &buff_size, "\033[C", 3); append_to_buff(&buff, &buff_size, "\033[C", 3);
print_str(buff, buff_size); print_str(buff, buff_size);
*pos = n; *pos = n;
free(buff); free(buff);
} }
/** /**
@ -220,27 +220,27 @@ void end_key(int *pos, int n)
*/ */
void backspace_key(int *pos, int *n, char **line) void backspace_key(int *pos, int *n, char **line)
{ {
if (*pos > 0) if (*pos > 0)
{ {
clear_sub_history(); clear_sub_history();
remove_on_pos(line, *pos); remove_on_pos(line, *pos);
(*n)--; (*n)--;
(*pos)--; (*pos)--;
char *buff = strdup(""); char *buff = strdup("");
size_t buff_size = 1; size_t buff_size = 1;
append_to_buff(&buff, &buff_size, "\033[D", 3); append_to_buff(&buff, &buff_size, "\033[D", 3);
append_to_buff(&buff, &buff_size, "\0337", 2); append_to_buff(&buff, &buff_size, "\0337", 2);
append_to_buff(&buff, &buff_size, *line + *pos, *n - *pos); append_to_buff(&buff, &buff_size, *line + *pos, *n - *pos);
append_to_buff(&buff, &buff_size, " ", 1); append_to_buff(&buff, &buff_size, " ", 1);
append_to_buff(&buff, &buff_size, "\0338", 2); append_to_buff(&buff, &buff_size, "\0338", 2);
print_str(buff, buff_size); print_str(buff, buff_size);
free(buff); free(buff);
} }
} }
/** /**
@ -249,9 +249,9 @@ void backspace_key(int *pos, int *n, char **line)
*/ */
void new_line(char *line) void new_line(char *line)
{ {
append_to_history(line); append_to_history(line);
clear_sub_history(); clear_sub_history();
print_str("\n", 1); print_str("\n", 1);
} }
/** /**
@ -263,63 +263,63 @@ void new_line(char *line)
*/ */
void tab_key(int *pos, int *n, char **line) void tab_key(int *pos, int *n, char **line)
{ {
clear_sub_history(); clear_sub_history();
(*line)[*pos] = '\0'; (*line)[*pos] = '\0';
*line = realloc(*line, strlen(*line) + 1); *line = realloc(*line, strlen(*line) + 1);
char *tmp = trim_string(*line, true); char *tmp = trim_string(*line, true);
free(*line); free(*line);
*line = tmp; *line = tmp;
*pos = strlen(*line); *pos = strlen(*line);
*n = *pos;
char *buff = strdup("");
size_t buff_size = 1;
char **complete_options = calloc(0, sizeof(char *)), *to_complete = NULL;
size_t opts_sz = get_complete_options(&complete_options, *line, &to_complete);
if (opts_sz == 1)
{
char *ending = complete_options[0] + strlen(to_complete);
*pos += strlen(ending);
*n = *pos; *n = *pos;
char *buff = strdup(""); *line = realloc(*line, strlen(*line) + strlen(ending) + 1);
size_t buff_size = 1; *line = strcat(*line, ending);
char **complete_options = calloc(0, sizeof(char *)), *to_complete = NULL; append_to_buff(&buff, &buff_size, ending, strlen(ending));
size_t opts_sz = get_complete_options(&complete_options, *line, &to_complete); }
else
{
append_to_buff(&buff, &buff_size, "\n", 1);
if (opts_sz == 1) if ((int)opts_sz < 1)
{ {
char *ending = complete_options[0] + strlen(to_complete); append_to_buff(&buff, &buff_size, "No suggestions", strlen("No suggestions"));
*pos += strlen(ending);
*n = *pos;
*line = realloc(*line, strlen(*line) + strlen(ending) + 1);
*line = strcat(*line, ending);
append_to_buff(&buff, &buff_size, ending, strlen(ending));
}
else
{
append_to_buff(&buff, &buff_size, "\n", 1);
if ((int)opts_sz < 1)
{
append_to_buff(&buff, &buff_size, "No suggestions", strlen("No suggestions"));
}
for (int i = 0; i < (int)opts_sz; i++)
{
append_to_buff(&buff, &buff_size, complete_options[i], strlen(complete_options[i]));
append_to_buff(&buff, &buff_size, " ", 1);
}
append_to_buff(&buff, &buff_size, "\n", 1);
char *prompt = compose_prompt();
append_to_buff(&buff, &buff_size, prompt, strlen(prompt));
free(prompt);
append_to_buff(&buff, &buff_size, *line, *pos);
} }
print_str(buff, buff_size); for (int i = 0; i < (int)opts_sz; i++)
{
append_to_buff(&buff, &buff_size, complete_options[i], strlen(complete_options[i]));
append_to_buff(&buff, &buff_size, " ", 1);
}
free(buff); append_to_buff(&buff, &buff_size, "\n", 1);
free(to_complete);
char *prompt = compose_prompt();
append_to_buff(&buff, &buff_size, prompt, strlen(prompt));
free(prompt);
append_to_buff(&buff, &buff_size, *line, *pos);
}
print_str(buff, buff_size);
free(buff);
free(to_complete);
} }
/** /**
@ -332,20 +332,20 @@ void tab_key(int *pos, int *n, char **line)
*/ */
void printable_key(int *pos, int *n, char c, char **line) void printable_key(int *pos, int *n, char c, char **line)
{ {
char *buff = strdup(""); char *buff = strdup("");
size_t buff_size = 1; size_t buff_size = 1;
(*n)++; (*n)++;
append_to_buff(&buff, &buff_size, &c, 1); append_to_buff(&buff, &buff_size, &c, 1);
append_to_pos(line, *pos, c); append_to_pos(line, *pos, c);
append_to_buff(&buff, &buff_size, "\0337", 2); append_to_buff(&buff, &buff_size, "\0337", 2);
(*pos)++; (*pos)++;
append_to_buff(&buff, &buff_size, *line + *pos, *n - *pos); append_to_buff(&buff, &buff_size, *line + *pos, *n - *pos);
append_to_buff(&buff, &buff_size, "\0338", 2); append_to_buff(&buff, &buff_size, "\0338", 2);
print_str(buff, buff_size); print_str(buff, buff_size);
free(buff); free(buff);
} }

View File

@ -10,307 +10,307 @@
*/ */
cmds_p *process_line(char *line) cmds_p *process_line(char *line)
{ {
int line_size = strlen(line); int line_size = strlen(line);
if (line_size < 1) if (line_size < 1)
return NULL; return NULL;
char *tmp = strdup(line), *free_tmp = tmp; char *tmp = strdup(line), *free_tmp = tmp;
cmds_p *coms = new_cmd(); cmds_p *coms = new_cmd();
cmds_p *curr_cmd = coms; cmds_p *curr_cmd = coms;
cmd_pipe *curr_pipe = curr_cmd->pipe; cmd_pipe *curr_pipe = curr_cmd->pipe;
curr_pipe->pipefd[0] = STDIN_FILENO; curr_pipe->pipefd[0] = STDIN_FILENO;
for (int i = 0; i < line_size; i++) for (int i = 0; i < line_size; i++)
{
if (line[i] == '"')
{ {
if (line[i] == '"') tmp++;
i++;
int j = i;
for (; j < line_size; j++)
if (line[j] == '"')
{ {
tmp++; free_tmp[j] = '\0';
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)); 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; tmp += strlen(curr_pipe->args[curr_pipe->args_am - 1]) + 1;
break; 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)); if (j >= line_size)
curr_pipe->args[curr_pipe->args_am] = NULL; {
char *ap_line = read_line();
switch (line[i]) line = realloc(line, strlen(line) + strlen(ap_line) + 1);
{ line = strcat(line, ap_line);
case ';':
curr_cmd->sep_next = SEMICOLON_SEP;
break;
case '&': line_size = strlen(line);
curr_cmd->sep_next = AND_SEP;
i++;
tmp++;
break;
case '|': free(ap_line);
curr_cmd->sep_next = OR_SEP; i--;
i++; continue;
tmp++; }
break;
}
if (tmp[0] == ' ') i = j;
{
tmp++;
i++;
}
if (curr_pipe->args[0] != NULL) if (tmp[0] == ' ')
{ {
if (strcmp(curr_pipe->args[0], "!") == 0) tmp++;
{ i++;
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] == '>' || line[i] == '<')
{
int flags = 0;
int f = 0;
if (line[i] == '>' && line[i + 1] == '>')
{
flags = O_CREAT | O_APPEND | O_WRONLY;
i++;
tmp++;
f = 0;
}
else if (line[i] == '>')
{
flags = O_CREAT | O_TRUNC | O_WRONLY;
f = 0;
}
else if (line[i] == '<' && line[i + 1] == '>')
{
flags = O_CREAT | O_RDWR;
i++;
tmp++;
f = 1;
}
else if (line[i] == '<')
{
flags = O_RDONLY;
f = 2;
}
i++;
tmp++;
if (line[i] == ' ')
{
i++;
tmp++;
}
int j = i;
for (; j < line_size; j++)
if (line[j] == ' ' || line[j] == '|' || line[j] == '&' || line[j] == ';')
break;
free_tmp[j] = '\0';
if (tmp[0] == '"')
tmp++;
if (free_tmp[j - 1] == '"')
free_tmp[j - 1] = '\0';
int ffd;
if ((ffd = open(tmp, flags)) < 0)
{
perror("file redirection: open");
return NULL;
}
if (f == 0)
curr_pipe->pipefd[1] = ffd;
else if (f == 1)
for (int k = 0; k < 2; k++)
curr_pipe->pipefd[k] = ffd;
else if (f == 2)
curr_pipe->pipefd[0] = ffd;
tmp += j - i + 1;
i = j;
}
else if (line[i] == '$')
{
tmp++;
i++;
int j = i;
for (; j < line_size; j++)
if (line[j] == ' ' || line[j] == ';' || line[j] == '&' || line[j] == '|' || line[j] == '>' || line[j] == '<')
break;
free_tmp[j] = '\0';
char *msg = NULL;
for (int k = 0; k < curr_cmd->envs_am; k++)
if (strncmp(curr_cmd->envs[k], tmp, strlen(tmp)) == 0)
msg = strchr(curr_cmd->envs[k], '=') + 1;
if (msg == NULL)
{
msg = get_env_var(tmp);
if (msg == NULL)
msg = strdup("");
}
append_to_str_arr(&curr_pipe->args, &curr_pipe->args_am, msg);
tmp += j - i + 1;
i = j;
}
else if (line[i] == ' ')
{
free_tmp[i] = '\0';
if (curr_cmd->pipe->args[0] == NULL && strchr(tmp, '='))
{
append_to_str_arr(&curr_cmd->envs, &curr_cmd->envs_am, tmp);
tmp += strlen(curr_cmd->envs[curr_cmd->envs_am - 1]) + 1;
continue;
}
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;
}
} }
else if (line[i] == ';' || (line[i] == '&' && line[i + 1] == '&') || (line[i] == '|' && line[i + 1] == '|'))
if (tmp[0] != '\0' && tmp[0] != '#')
{ {
if (tmp[strlen(tmp) - 1] != '"') free_tmp[i] = '\0';
{ if (line[i - 1] != ' ')
char *ap_string = trim_string(tmp, false), **exp = calloc(0, sizeof(char *)); {
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++;
int sz = expand_wildcatrd(&exp, ap_string); curr_pipe->args = realloc(curr_pipe->args, sizeof(char *) * (curr_pipe->args_am + 1));
curr_pipe->args[curr_pipe->args_am] = NULL;
for (int l = 0; l < sz; l++) switch (line[i])
append_to_str_arr(&curr_pipe->args, &curr_pipe->args_am, exp[l]); {
} case ';':
else curr_cmd->sep_next = SEMICOLON_SEP;
append_to_str_arr(&curr_pipe->args, &curr_pipe->args_am, trim_string(tmp, false)); break;
}
if (curr_pipe->args[0] != NULL) 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) if (strcmp(curr_pipe->args[0], "!") == 0)
{ {
curr_cmd->stat.invert = true; 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] == '|')
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)); if (line[i - 1] != ' ')
curr_pipe->args[curr_pipe->args_am] = NULL; {
} append_to_str_arr(&curr_pipe->args, &curr_pipe->args_am, trim_string(tmp, false));
if (curr_pipe->pipefd[1] < 0) tmp += strlen(curr_pipe->args[curr_pipe->args_am - 1]) + 1;
curr_pipe->pipefd[1] = STDOUT_FILENO; }
else
tmp++;
return coms; 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] == '>' || line[i] == '<')
{
int flags = 0;
int f = 0;
if (line[i] == '>' && line[i + 1] == '>')
{
flags = O_CREAT | O_APPEND | O_WRONLY;
i++;
tmp++;
f = 0;
}
else if (line[i] == '>')
{
flags = O_CREAT | O_TRUNC | O_WRONLY;
f = 0;
}
else if (line[i] == '<' && line[i + 1] == '>')
{
flags = O_CREAT | O_RDWR;
i++;
tmp++;
f = 1;
}
else if (line[i] == '<')
{
flags = O_RDONLY;
f = 2;
}
i++;
tmp++;
if (line[i] == ' ')
{
i++;
tmp++;
}
int j = i;
for (; j < line_size; j++)
if (line[j] == ' ' || line[j] == '|' || line[j] == '&' || line[j] == ';')
break;
free_tmp[j] = '\0';
if (tmp[0] == '"')
tmp++;
if (free_tmp[j - 1] == '"')
free_tmp[j - 1] = '\0';
int ffd;
if ((ffd = open(tmp, flags)) < 0)
{
perror("file redirection: open");
return NULL;
}
if (f == 0)
curr_pipe->pipefd[1] = ffd;
else if (f == 1)
for (int k = 0; k < 2; k++)
curr_pipe->pipefd[k] = ffd;
else if (f == 2)
curr_pipe->pipefd[0] = ffd;
tmp += j - i + 1;
i = j;
}
else if (line[i] == '$')
{
tmp++;
i++;
int j = i;
for (; j < line_size; j++)
if (line[j] == ' ' || line[j] == ';' || line[j] == '&' || line[j] == '|' || line[j] == '>' || line[j] == '<')
break;
free_tmp[j] = '\0';
char *msg = NULL;
for (int k = 0; k < curr_cmd->envs_am; k++)
if (strncmp(curr_cmd->envs[k], tmp, strlen(tmp)) == 0)
msg = strchr(curr_cmd->envs[k], '=') + 1;
if (msg == NULL)
{
msg = get_env_var(tmp);
if (msg == NULL)
msg = strdup("");
}
append_to_str_arr(&curr_pipe->args, &curr_pipe->args_am, msg);
tmp += j - i + 1;
i = j;
}
else if (line[i] == ' ')
{
free_tmp[i] = '\0';
if (curr_cmd->pipe->args[0] == NULL && strchr(tmp, '='))
{
append_to_str_arr(&curr_cmd->envs, &curr_cmd->envs_am, tmp);
tmp += strlen(curr_cmd->envs[curr_cmd->envs_am - 1]) + 1;
continue;
}
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;
}
if (curr_pipe->pipefd[1] < 0)
curr_pipe->pipefd[1] = STDOUT_FILENO;
return coms;
} }
/** /**
@ -322,20 +322,20 @@ cmds_p *process_line(char *line)
*/ */
int expand_wildcatrd(char ***arr, char *input) int expand_wildcatrd(char ***arr, char *input)
{ {
ssize_t size = 0; ssize_t size = 0;
if (strchr(input, '*') || strchr(input, '~') || strchr(input, '?')) if (strchr(input, '*') || strchr(input, '~') || strchr(input, '?'))
{ {
glob_t globbuf; glob_t globbuf;
int t = glob(input, GLOB_TILDE, NULL, &globbuf); int t = glob(input, GLOB_TILDE, NULL, &globbuf);
if (t != 0) if (t != 0)
perror("Glob"); perror("Glob");
for (int j = 0; j < globbuf.gl_pathc; j++) for (int j = 0; j < globbuf.gl_pathc; j++)
append_to_str_arr(arr, &size, globbuf.gl_pathv[j]); append_to_str_arr(arr, &size, globbuf.gl_pathv[j]);
} }
else else
append_to_str_arr(arr, &size, input); append_to_str_arr(arr, &size, input);
return size; return size;
} }

View File

@ -15,36 +15,36 @@ t_ init_term();
// Main // Main
int main() int main()
{ {
term = init_term(); term = init_term();
while (1) while (1)
{ {
process_command(); process_command();
} }
} }
// Init // Init
t_ init_term() t_ init_term()
{ {
// Entering raw mode // Entering raw mode
change_mode(1); change_mode(1);
// Init history // Init history
init_history(); init_history();
term.last_status = 0; term.last_status = 0;
// Disable Ctrl+C default behaviour for shell // Disable Ctrl+C default behaviour for shell
signal(SIGINT, SIG_IGN); signal(SIGINT, SIG_IGN);
// Set up function to run on shell exit // Set up function to run on shell exit
atexit(exit_shell); atexit(exit_shell);
return term; return term;
} }
// Exit // Exit
void exit_shell() void exit_shell()
{ {
change_mode(0); change_mode(0);
} }

View File

@ -11,13 +11,13 @@
*/ */
size_t append_to_buff(char **buff, size_t *buff_size, char *ap, size_t ap_size) size_t append_to_buff(char **buff, size_t *buff_size, char *ap, size_t ap_size)
{ {
*buff = realloc(*buff, *buff_size + ap_size); *buff = realloc(*buff, *buff_size + ap_size);
memcpy(*buff + *buff_size - 1, ap, ap_size); memcpy(*buff + *buff_size - 1, ap, ap_size);
*buff_size += ap_size; *buff_size += ap_size;
(*buff)[*buff_size - 1] = '\0'; (*buff)[*buff_size - 1] = '\0';
return *buff_size; return *buff_size;
} }
/** /**
@ -28,5 +28,5 @@ size_t append_to_buff(char **buff, size_t *buff_size, char *ap, size_t ap_size)
*/ */
void print_str(char *buff, size_t size) void print_str(char *buff, size_t size)
{ {
write(STDOUT_FILENO, buff, size); write(STDOUT_FILENO, buff, size);
} }

View File

@ -8,27 +8,27 @@
*/ */
char *get_ip_addr() char *get_ip_addr()
{ {
struct ifaddrs *host, *tmp; struct ifaddrs *host, *tmp;
getifaddrs(&host); getifaddrs(&host);
tmp = host; tmp = host;
char *ip = NULL; char *ip = NULL;
while (tmp) while (tmp)
{
if (tmp->ifa_addr && tmp->ifa_addr->sa_family == AF_INET)
{ {
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);
struct sockaddr_in *pAddr = (struct sockaddr_in *)tmp->ifa_addr; if (strncmp(ip, "127", 3) != 0)
ip = inet_ntoa(pAddr->sin_addr); break;
if (strncmp(ip, "127", 3) != 0)
break;
}
tmp = tmp->ifa_next;
} }
freeifaddrs(host); tmp = tmp->ifa_next;
}
return ip; freeifaddrs(host);
return ip;
} }
/** /**
@ -38,54 +38,54 @@ char *get_ip_addr()
*/ */
char *compose_prompt() char *compose_prompt()
{ {
// New line // New line
char *prompt = strdup(""); char *prompt = strdup("");
// Username // Username
char *username = strdup(getlogin()); char *username = strdup(getlogin());
if (username == NULL) if (username == NULL)
username = "none"; username = "none";
prompt = realloc(prompt, strlen(prompt) + strlen("\033[97;44m") + strlen(username) + 2); prompt = realloc(prompt, strlen(prompt) + strlen("\033[97;44m") + strlen(username) + 2);
prompt = strcat(prompt, "\033[97;44m"); prompt = strcat(prompt, "\033[97;44m");
prompt = strcat(prompt, username); prompt = strcat(prompt, username);
// Current host ip // Current host ip
char *ip = get_ip_addr(); char *ip = get_ip_addr();
if (ip == NULL) if (ip == NULL)
ip = "none"; ip = "none";
prompt = realloc(prompt, strlen(prompt) + 1 + strlen(ip) + strlen("\033[39m") + strlen("\033[0m") + 2); prompt = realloc(prompt, strlen(prompt) + 1 + strlen(ip) + strlen("\033[39m") + strlen("\033[0m") + 2);
prompt = strcat(prompt, "@"); prompt = strcat(prompt, "@");
prompt = strcat(prompt, ip); prompt = strcat(prompt, ip);
prompt = strcat(prompt, "\033[39m"); prompt = strcat(prompt, "\033[39m");
prompt = strcat(prompt, "\033[0m"); prompt = strcat(prompt, "\033[0m");
prompt = strcat(prompt, ":"); prompt = strcat(prompt, ":");
// Current path // Current path
char *full_path = get_curr_dir_name(); char *full_path = get_curr_dir_name();
prompt = realloc(prompt, strlen(prompt) + strlen("\033[92;1m") + strlen("\033[0m") + strlen(full_path) + 2); 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, "\033[92;1m");
prompt = strcat(prompt, full_path); prompt = strcat(prompt, full_path);
prompt = strcat(prompt, "\033[0m"); prompt = strcat(prompt, "\033[0m");
free(full_path); free(full_path);
// Previous status // Previous status
if (term.last_status != 0) if (term.last_status != 0)
{ {
char *status = malloc(snprintf(NULL, 0, " \033[91m%d\033[39m", term.last_status)); char *status = malloc(snprintf(NULL, 0, " \033[91m%d\033[39m", term.last_status));
sprintf(status, " \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 = realloc(prompt, strlen(prompt) + strlen(status) + 1);
prompt = strcat(prompt, status); prompt = strcat(prompt, status);
free(status); free(status);
} }
// Permissions specific character before user input // Permissions specific character before user input
prompt = realloc(prompt, strlen(prompt) + 4); prompt = realloc(prompt, strlen(prompt) + 4);
if (getuid() == 0) if (getuid() == 0)
{ {
prompt = strcat(prompt, "\n# "); prompt = strcat(prompt, "\n# ");
} }
else else
prompt = strcat(prompt, "\n% "); prompt = strcat(prompt, "\n% ");
return prompt; return prompt;
} }

View File

@ -8,50 +8,50 @@
*/ */
void process_command() void process_command()
{ {
char *prompt = compose_prompt(); char *prompt = compose_prompt();
print_str(prompt, strlen(prompt)); print_str(prompt, strlen(prompt));
char *line = read_line(); char *line = read_line();
char *tmp = trim_string(line, false); char *tmp = trim_string(line, false);
free(line); free(line);
line = tmp; line = tmp;
cmds_p *coms = process_line(line); 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_with_pipes(curr);
if (curr->stat.invert)
{ {
status = execute_with_pipes(curr); if (status == 0)
status = 1;
if (curr->stat.invert) else
{ status = 0;
if (status == 0)
status = 1;
else
status = 0;
}
curr->stat.s = term.last_status = status;
if (curr->sep_next == AND_SEP)
{
if (term.last_status != 0)
break;
}
else if (curr->sep_next == OR_SEP)
{
if (term.last_status == 0)
break;
}
curr = curr->next;
} }
free(line); curr->stat.s = term.last_status = status;
free(prompt);
free_cmds_p(coms); if (curr->sep_next == AND_SEP)
{
if (term.last_status != 0)
break;
}
else if (curr->sep_next == OR_SEP)
{
if (term.last_status == 0)
break;
}
curr = curr->next;
}
free(line);
free(prompt);
free_cmds_p(coms);
} }
/** /**
@ -61,20 +61,20 @@ void process_command()
*/ */
cmds_p *new_cmd() cmds_p *new_cmd()
{ {
cmds_p *new = malloc(sizeof(cmds_p)); cmds_p *new = malloc(sizeof(cmds_p));
new->pipe = new_cmd_pipe(); new->pipe = new_cmd_pipe();
new->pipes_am = 1; new->pipes_am = 1;
new->envs = calloc(0, sizeof(char *)); new->envs = calloc(0, sizeof(char *));
new->envs_am = 0; new->envs_am = 0;
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;
new->next = NULL; new->next = NULL;
return new; return new;
} }
/** /**
@ -84,38 +84,38 @@ cmds_p *new_cmd()
*/ */
cmd_pipe *new_cmd_pipe() cmd_pipe *new_cmd_pipe()
{ {
cmd_pipe *new = malloc(sizeof(cmd_pipe)); cmd_pipe *new = malloc(sizeof(cmd_pipe));
new->args = calloc(0, sizeof(char *)); new->args = calloc(0, sizeof(char *));
new->args_am = 0; new->args_am = 0;
new->next = NULL; new->next = NULL;
new->pipefd[0] = new->pipefd[1] = -1; new->pipefd[0] = new->pipefd[1] = -1;
return new; return new;
} }
void free_cmds_p(cmds_p *root) void free_cmds_p(cmds_p *root)
{ {
if (root != NULL) if (root != NULL)
{ {
if (root->pipe != NULL) if (root->pipe != NULL)
free_cmd_pipe(root->pipe); free_cmd_pipe(root->pipe);
if (root->next != NULL) if (root->next != NULL)
free_cmds_p(root->next); free_cmds_p(root->next);
free(root); free(root);
} }
return; return;
} }
void free_cmd_pipe(cmd_pipe *root) void free_cmd_pipe(cmd_pipe *root)
{ {
if (root != NULL) if (root != NULL)
{ {
if (root->next != NULL) if (root->next != NULL)
free_cmd_pipe(root->next); free_cmd_pipe(root->next);
free(root); free(root);
} }
return; return;
} }

View File

@ -7,15 +7,15 @@
*/ */
struct tree_node *get_new_node() struct tree_node *get_new_node()
{ {
struct tree_node *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++)
{ {
node->child[i] = NULL; node->child[i] = NULL;
} }
return node; return node;
} }
/** /**
@ -26,20 +26,20 @@ struct tree_node *get_new_node()
*/ */
void insert_tree(struct tree_node *root, char *key) void insert_tree(struct tree_node *root, char *key)
{ {
struct tree_node *current = root; struct tree_node *current = root;
for (int i = 0; i < strlen(key); i++) for (int i = 0; i < strlen(key); i++)
{ {
char index = key[i]; char index = key[i];
if (current->child[index] == NULL) if (current->child[index] == NULL)
{ {
current->child[index] = get_new_node(); current->child[index] = get_new_node();
} }
current = current->child[index]; current = current->child[index];
} }
current->is_leaf = 1; current->is_leaf = 1;
} }
/** /**
@ -49,19 +49,19 @@ void insert_tree(struct tree_node *root, char *key)
*/ */
void free_tree(struct tree_node *root) void free_tree(struct tree_node *root)
{ {
if (root != NULL) if (root != NULL)
{
for (int i = 0; i < ALPHABET_SIZE; i++)
{
if (root->child[i] != NULL)
{ {
for (int i = 0; i < ALPHABET_SIZE; i++) free_tree(root->child[i]);
{
if (root->child[i] != NULL)
{
free_tree(root->child[i]);
}
}
free(root);
} }
return; }
free(root);
}
return;
} }
/** /**
@ -73,21 +73,21 @@ void free_tree(struct tree_node *root)
*/ */
int search_tree(struct tree_node *root, char *key) int search_tree(struct tree_node *root, char *key)
{ {
struct tree_node *current = root; struct tree_node *current = root;
for (int i = 0; i < strlen(key); i++) for (int i = 0; i < strlen(key); i++)
{ {
char index = key[i]; char index = key[i];
if (current->child[index] == NULL) if (current->child[index] == NULL)
{ {
return 0; return 0;
} }
current = current->child[index]; current = current->child[index];
} }
return (current != NULL && current->is_leaf); return (current != NULL && current->is_leaf);
} }
/** /**
@ -98,11 +98,11 @@ int search_tree(struct tree_node *root, char *key)
*/ */
int is_last_node(struct tree_node *root) int is_last_node(struct tree_node *root)
{ {
for (int i = 0; i < ALPHABET_SIZE; i++) for (int i = 0; i < ALPHABET_SIZE; i++)
if (root->child[i] != NULL) if (root->child[i] != NULL)
return 0; return 0;
return 1; return 1;
} }
/** /**
@ -115,48 +115,48 @@ int is_last_node(struct tree_node *root)
*/ */
ssize_t list_strings_containing(struct tree_node *root, char *key, char ***strings) ssize_t list_strings_containing(struct tree_node *root, char *key, char ***strings)
{ {
ssize_t amount = 0; ssize_t amount = 0;
free(*strings); free(*strings);
*strings = calloc(amount, sizeof(char *)); *strings = calloc(amount, sizeof(char *));
struct tree_node *current = root; struct tree_node *current = root;
for (int level = 0; level < strlen(key); level++) for (int level = 0; level < strlen(key); level++)
{ {
int index = key[level]; int index = key[level];
if (current->child[index] == NULL) if (current->child[index] == NULL)
return amount; return amount;
current = current->child[index]; current = current->child[index];
} }
int key_presents = current->is_leaf; int key_presents = current->is_leaf;
int is_last = is_last_node(current); int is_last = is_last_node(current);
if (key_presents && is_last) if (key_presents && is_last)
{ {
amount++; amount++;
*strings = (char **)realloc(*strings, amount * sizeof(char *)); *strings = (char **)realloc(*strings, amount * sizeof(char *));
(*strings)[0] = strdup(key); (*strings)[0] = strdup(key);
return amount; return amount;
} }
if (is_last == 0) if (is_last == 0)
{ {
char *prefix = strdup(key); char *prefix = strdup(key);
get_all_substrings(current, &amount, &prefix, strings); get_all_substrings(current, &amount, &prefix, strings);
free(prefix); free(prefix);
return amount; return amount;
} }
return -1; return -1;
} }
/** /**
@ -169,26 +169,26 @@ ssize_t list_strings_containing(struct tree_node *root, char *key, char ***strin
*/ */
void get_all_substrings(struct tree_node *root, ssize_t *amount, char **curr_prefix, char ***strings) void get_all_substrings(struct tree_node *root, ssize_t *amount, char **curr_prefix, char ***strings)
{ {
if (root->is_leaf) if (root->is_leaf)
{ {
(*amount)++; (*amount)++;
*strings = (char **)realloc(*strings, *amount * sizeof(char *)); *strings = (char **)realloc(*strings, *amount * sizeof(char *));
(*strings)[*amount - 1] = strdup(*curr_prefix); (*strings)[*amount - 1] = strdup(*curr_prefix);
} }
if (is_last_node(root)) if (is_last_node(root))
return; return;
for (int index = 0; index < ALPHABET_SIZE; index++) for (int index = 0; index < ALPHABET_SIZE; index++)
{ {
if (root->child[index] != NULL) if (root->child[index] != NULL)
{ {
*curr_prefix = realloc(*curr_prefix, strlen(*curr_prefix) + 2); *curr_prefix = realloc(*curr_prefix, strlen(*curr_prefix) + 2);
*curr_prefix = strncat(*curr_prefix, (char *)(&index), 1); *curr_prefix = strncat(*curr_prefix, (char *)(&index), 1);
get_all_substrings(root->child[index], amount, curr_prefix, strings); get_all_substrings(root->child[index], amount, curr_prefix, strings);
(*curr_prefix)[strlen(*curr_prefix) - 1] = '\0'; (*curr_prefix)[strlen(*curr_prefix) - 1] = '\0';
} }
} }
} }

View File

@ -9,18 +9,18 @@
*/ */
void append_to_pos(char **str, int pos, char ch) void append_to_pos(char **str, int pos, char ch)
{ {
size_t len = strlen(*str); size_t len = strlen(*str);
if (pos <= len + 1) if (pos <= len + 1)
{ {
*str = realloc(*str, len + 2); *str = realloc(*str, len + 2);
for (int i = len; i >= pos; i--) for (int i = len; i >= pos; i--)
{ {
(*str)[i + 1] = (*str)[i]; (*str)[i + 1] = (*str)[i];
} }
(*str)[pos] = ch; (*str)[pos] = ch;
(*str)[len + 1] = '\0'; (*str)[len + 1] = '\0';
} }
} }
/** /**
@ -31,15 +31,15 @@ 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)
{ {
size_t len = strlen(*str); size_t len = strlen(*str);
if (pos <= len) if (pos <= len)
{ {
for (int i = pos - 1; i < len; i++) for (int i = pos - 1; i < len; i++)
(*str)[i] = (*str)[i + 1]; (*str)[i] = (*str)[i + 1];
(*str)[len] = '\0'; (*str)[len] = '\0';
*str = realloc(*str, len); *str = realloc(*str, len);
} }
} }
/** /**
@ -52,23 +52,23 @@ 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)
{ {
free(*toks); free(*toks);
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 = calloc(n, sizeof(char *)); *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)
{ {
n++; n++;
*toks = realloc(*toks, sizeof(char *) * n); *toks = realloc(*toks, sizeof(char *) * n);
(*toks)[n - 1] = strdup(tmp); (*toks)[n - 1] = strdup(tmp);
} }
free(free_tmp_line); free(free_tmp_line);
return n; return n;
} }
/** /**
@ -79,22 +79,22 @@ 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)
{ {
char *tmp = strdup(str); char *tmp = strdup(str);
while (tmp[0] == ' ') while (tmp[0] == ' ')
remove_on_pos(&tmp, 1); remove_on_pos(&tmp, 1);
for (int i = 1; i < strlen(tmp); i++) for (int i = 1; i < strlen(tmp); i++)
if (tmp[i] == ' ' && tmp[i - 1] == ' ') if (tmp[i] == ' ' && tmp[i - 1] == ' ')
{ {
remove_on_pos(&tmp, i); remove_on_pos(&tmp, i);
i--; i--;
} }
if (!leave_trailing_space) if (!leave_trailing_space)
if (tmp[strlen(tmp) - 1] == ' ' && tmp[strlen(tmp) - 2] == ' ') if (tmp[strlen(tmp) - 1] == ' ' && tmp[strlen(tmp) - 2] == ' ')
remove_on_pos(&tmp, strlen(tmp) - 1); remove_on_pos(&tmp, strlen(tmp) - 1);
return tmp; return tmp;
} }
/** /**
@ -105,10 +105,10 @@ char *trim_string(char *str, bool leave_trailing_space)
*/ */
void free_str_arr(char **arr, int sz) void free_str_arr(char **arr, int sz)
{ {
if (arr[0] != NULL) if (arr[0] != NULL)
for (int i = 0; i < sz; i++) for (int i = 0; i < sz; i++)
free(arr[i]); free(arr[i]);
free(arr); free(arr);
} }
/** /**
@ -122,21 +122,21 @@ 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)
{ {
if (beg == -1) if (beg == -1)
beg = 0; beg = 0;
if (end == -1) if (end == -1)
end = get_null_term_arr_size(arr); end = get_null_term_arr_size(arr);
char **new_arr = calloc(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++)
new_arr[j] = arr[i]; new_arr[j] = arr[i];
else else
for (int i = end, j = 0; i > beg; i--, j++) for (int i = end, j = 0; i > beg; i--, j++)
new_arr[j] = arr[i]; new_arr[j] = arr[i];
return new_arr; return new_arr;
} }
/** /**
@ -147,11 +147,11 @@ 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 k = 0; int k = 0;
for (int i = 0; arr[i] != NULL; i++) for (int i = 0; arr[i] != NULL; i++)
k++; k++;
return ++k; return ++k;
} }
/** /**
@ -164,19 +164,19 @@ int get_null_term_arr_size(char **arr)
*/ */
int append_to_str_arr(char ***arr, ssize_t *sz, char *str) int append_to_str_arr(char ***arr, ssize_t *sz, char *str)
{ {
(*sz)++; (*sz)++;
char **tmp = realloc(*arr, sizeof(char *) * *sz); char **tmp = realloc(*arr, sizeof(char *) * *sz);
if (tmp == NULL) if (tmp == NULL)
{ {
fprintf(stderr, "Couldn't reallocate memory\n"); fprintf(stderr, "Couldn't reallocate memory\n");
return -1; return -1;
} }
*arr = tmp; *arr = tmp;
(*arr)[*sz - 1] = strdup(str); (*arr)[*sz - 1] = strdup(str);
return 0; return 0;
} }
/** /**
@ -186,10 +186,10 @@ int append_to_str_arr(char ***arr, ssize_t *sz, char *str)
*/ */
char *get_curr_dir_name() char *get_curr_dir_name()
{ {
char *pwd = malloc(FILENAME_MAX); char *pwd = malloc(FILENAME_MAX);
getcwd(pwd, FILENAME_MAX); getcwd(pwd, FILENAME_MAX);
return pwd; return pwd;
} }
/** /**
@ -200,60 +200,60 @@ char *get_curr_dir_name()
*/ */
int get_num_of_lines(FILE *file) int get_num_of_lines(FILE *file)
{ {
int n = 0; int n = 0;
int ch, pch; int ch, pch;
long curr_pos = ftell(file); long curr_pos = ftell(file);
fseek(file, 0, SEEK_SET); fseek(file, 0, SEEK_SET);
while ((ch = fgetc(file)) != EOF) while ((ch = fgetc(file)) != EOF)
{ {
if (ch == '\n') if (ch == '\n')
n++; n++;
pch = ch; pch = ch;
} }
if (pch != '\n') if (pch != '\n')
n++; n++;
fseek(file, curr_pos, SEEK_SET); fseek(file, curr_pos, SEEK_SET);
return n; return n;
} }
bool str_is_in_arr(char **arr, size_t sz, char *str) bool str_is_in_arr(char **arr, size_t sz, char *str)
{ {
int first = 0; int first = 0;
int last = sz - 1; int last = sz - 1;
int middle; int middle;
while (first <= last) while (first <= last)
{ {
middle = (first + last) / 2; middle = (first + last) / 2;
int diff = strcmp(arr[middle], str); int diff = strcmp(arr[middle], str);
if (diff > 0) if (diff > 0)
{ {
first = middle + 1; first = middle + 1;
} }
else if (diff == 0) else if (diff == 0)
return true; return true;
else else
{ {
last = middle - 1; last = middle - 1;
} }
} }
return false; return false;
} }
char *get_env_var(char *name) char *get_env_var(char *name)
{ {
char *env_var = getenv(name); char *env_var = getenv(name);
if (env_var == NULL) if (env_var == NULL)
{ {
fprintf(stderr, "No $%s environment variable\n", name); fprintf(stderr, "No $%s environment variable\n", name);
return NULL; return NULL;
} }
return strdup(env_var); return strdup(env_var);
} }