Refactored tab autocompletion, fixed some more memory leaks, added logging to separate file
This commit is contained in:
parent
a6ccd9344f
commit
dceafa7244
@ -9,8 +9,12 @@
|
|||||||
#include <unistd.h>
|
#include <unistd.h>
|
||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
#include <dirent.h>
|
#include <dirent.h>
|
||||||
|
#include <stdbool.h>
|
||||||
|
|
||||||
size_t get_dir_list(char ***dir_list, char *path);
|
bool check_if_executable(char *path, char *file_name);
|
||||||
void complete_line(int *pos, int *n, char **line, char **out);
|
size_t get_dir_list(char ***dir_list, char *path, int ex);
|
||||||
|
size_t get_complete_options(char ***opts, char *line, char **to_complete);
|
||||||
|
size_t complete_line(int *pos, int *n, char **line, char **out);
|
||||||
|
size_t filter_options(char ***comp_list, size_t *size, char *filter_string);
|
||||||
|
|
||||||
#endif
|
#endif
|
@ -13,7 +13,7 @@
|
|||||||
#define ARG_SIZE 32
|
#define ARG_SIZE 32
|
||||||
|
|
||||||
char *read_line();
|
char *read_line();
|
||||||
void process_line(char *line, char ***args);
|
int process_line(char *line, char ***args);
|
||||||
int launch(char **args);
|
int launch(char **args);
|
||||||
void process_command();
|
void process_command();
|
||||||
int execute(char **args);
|
int execute(char **args);
|
||||||
|
@ -7,7 +7,19 @@
|
|||||||
#include <unistd.h>
|
#include <unistd.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
|
||||||
|
extern FILE *log_file;
|
||||||
|
|
||||||
|
#define log(fmt, ...) \
|
||||||
|
{ \
|
||||||
|
log_file = fopen("/var/log/mshell.log", "a"); \
|
||||||
|
fprintf(log_file, fmt, __VA_ARGS__); \
|
||||||
|
fclose(log_file); \
|
||||||
|
}
|
||||||
|
|
||||||
void append_to_pos(char **str, int pos, char ch);
|
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);
|
||||||
|
char *trim_string(char **str);
|
||||||
|
void free_str_arr(char **arr);
|
||||||
|
|
||||||
#endif
|
#endif
|
135
src/complete.c
135
src/complete.c
@ -1,8 +1,23 @@
|
|||||||
#include "../include/complete.h"
|
#include "../include/complete.h"
|
||||||
#include "../include/shell.h"
|
#include "../include/shell.h"
|
||||||
#include "../include/tree.h"
|
#include "../include/tree.h"
|
||||||
|
#include "../include/utils.h"
|
||||||
|
|
||||||
size_t get_dir_list(char ***dir_list, char *path)
|
bool check_if_executable(char *path, char *file_name)
|
||||||
|
{
|
||||||
|
char *file_path = malloc(strlen(path) + strlen(file_name) + 2);
|
||||||
|
file_path[0] = '\0';
|
||||||
|
file_path = strcat(file_path, path);
|
||||||
|
file_path = strcat(file_path, "/");
|
||||||
|
file_path = strcat(file_path, file_name);
|
||||||
|
|
||||||
|
bool ret = access(file_path, X_OK) == 0;
|
||||||
|
free(file_path);
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
size_t get_dir_list(char ***dir_list, char *path, int ex)
|
||||||
{
|
{
|
||||||
size_t n = 0;
|
size_t n = 0;
|
||||||
*dir_list = malloc(sizeof(char *) * n);
|
*dir_list = malloc(sizeof(char *) * n);
|
||||||
@ -17,9 +32,18 @@ size_t get_dir_list(char ***dir_list, char *path)
|
|||||||
}
|
}
|
||||||
while ((ent = readdir(dir)) != NULL)
|
while ((ent = readdir(dir)) != NULL)
|
||||||
{
|
{
|
||||||
|
if (ex != 0 && !check_if_executable(path, ent->d_name))
|
||||||
|
continue;
|
||||||
|
|
||||||
n++;
|
n++;
|
||||||
*dir_list = realloc(*dir_list, sizeof(char *) * n);
|
*dir_list = realloc(*dir_list, sizeof(char *) * n);
|
||||||
(*dir_list)[n - 1] = strdup(ent->d_name);
|
(*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);
|
closedir(dir);
|
||||||
@ -27,74 +51,65 @@ size_t get_dir_list(char ***dir_list, char *path)
|
|||||||
return n;
|
return n;
|
||||||
}
|
}
|
||||||
|
|
||||||
void complete_line(int *pos, int *n, char **line, char **out)
|
size_t get_complete_options(char ***opts, char *line, char **to_complete)
|
||||||
{
|
{
|
||||||
(*line)[*pos] = '\0';
|
char **args, **folders = malloc(0);
|
||||||
*out = strdup("\x1b[2K");
|
|
||||||
|
|
||||||
char **comp_list;
|
|
||||||
|
|
||||||
char *curr_path, *tmp_line = strdup(*line);
|
|
||||||
size_t sz;
|
size_t sz;
|
||||||
|
|
||||||
if ((curr_path = strtok(tmp_line, " ")) != NULL && (curr_path = strtok(NULL, " ")) != NULL)
|
int am = sep_string(line, &args, " ");
|
||||||
|
|
||||||
|
char *last_arg = args[am - 1];
|
||||||
|
|
||||||
|
if (am > 0)
|
||||||
{
|
{
|
||||||
sz = get_dir_list(&comp_list, get_current_dir_name());
|
if (last_arg[0] == '/')
|
||||||
|
{
|
||||||
|
int path_depth = sep_string(last_arg, &folders, "/");
|
||||||
|
*to_complete = strdup(folders[path_depth - 1]);
|
||||||
|
sz = get_dir_list(opts, last_arg, 0); // pass / + all $folders instead of last_arg
|
||||||
|
}
|
||||||
|
else if (last_arg[0] == '.' && last_arg[1] == '/')
|
||||||
|
{
|
||||||
|
int path_depth = sep_string(last_arg + 1, &folders, "/");
|
||||||
|
*to_complete = strdup(folders[path_depth - 1]);
|
||||||
|
sz = get_dir_list(opts, get_current_dir_name(), 0); // pass get_current_dir_name() + all $folders instead of get_current_dir_name()
|
||||||
|
}
|
||||||
|
else if (strchr(line, ' '))
|
||||||
|
{
|
||||||
|
int path_depth = sep_string(last_arg, &folders, "/");
|
||||||
|
*to_complete = strdup(folders[path_depth - 1]);
|
||||||
|
sz = get_dir_list(opts, get_current_dir_name(), 0); // pass get_current_dir_name() + all $folders instead of get_current_dir_name()
|
||||||
|
}
|
||||||
|
else
|
||||||
|
goto ABSOLUTE;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
sz = get_dir_list(&comp_list, "/usr/bin");
|
ABSOLUTE:
|
||||||
curr_path = *line;
|
*to_complete = strdup(line);
|
||||||
|
sz = get_dir_list(opts, "/usr/bin", 1);
|
||||||
}
|
}
|
||||||
if (*pos > 0 && (*line)[*pos - 1] != ' ')
|
|
||||||
|
free_str_arr(args);
|
||||||
|
free_str_arr(folders);
|
||||||
|
|
||||||
|
if ((*to_complete)[0] != '\0')
|
||||||
{
|
{
|
||||||
curr_path = strdup(curr_path);
|
sz = filter_options(opts, &sz, *to_complete);
|
||||||
|
|
||||||
char *tmp;
|
|
||||||
while ((tmp = strtok(NULL, " ")) != NULL)
|
|
||||||
{
|
|
||||||
free(curr_path);
|
|
||||||
|
|
||||||
curr_path = strdup(tmp);
|
|
||||||
}
|
|
||||||
|
|
||||||
struct tree_node *child_dirs_root = get_new_node();
|
|
||||||
for (size_t i = 0; i < sz; i++)
|
|
||||||
{
|
|
||||||
insert_tree(child_dirs_root, comp_list[i]);
|
|
||||||
}
|
|
||||||
|
|
||||||
sz = list_strings_containing(child_dirs_root, curr_path, &comp_list);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (sz == 1)
|
return sz;
|
||||||
{
|
}
|
||||||
*out = strdup(comp_list[0] + strlen(curr_path));
|
|
||||||
*pos += strlen(*out);
|
size_t filter_options(char ***comp_list, size_t *size, char *filter_string)
|
||||||
*n = *pos;
|
{
|
||||||
|
struct tree_node *child_dirs_root = get_new_node();
|
||||||
*line = realloc(*line, strlen(*line) + strlen(*out));
|
for (size_t i = 0; i < *size; i++)
|
||||||
*line = strcat(*line, *out);
|
insert_tree(child_dirs_root, (*comp_list)[i]);
|
||||||
|
|
||||||
return;
|
*size = list_strings_containing(child_dirs_root, filter_string, comp_list);
|
||||||
}
|
|
||||||
|
free_tree(child_dirs_root);
|
||||||
for (int i = 0; i < sz; i++)
|
|
||||||
{
|
return *size;
|
||||||
*out = realloc(*out, strlen(*out) + strlen(comp_list[i]) + 2);
|
|
||||||
*out = strcat(*out, comp_list[i]);
|
|
||||||
*out = strcat(*out, " ");
|
|
||||||
}
|
|
||||||
|
|
||||||
*out = realloc(*out, strlen(*out) + 2);
|
|
||||||
*out = strcat(*out, "\n");
|
|
||||||
|
|
||||||
char *prompt = compose_prompt();
|
|
||||||
*out = realloc(*out, strlen(*out) + strlen(prompt) + 1);
|
|
||||||
*out = strcat(*out, prompt);
|
|
||||||
|
|
||||||
*out = realloc(*out, strlen(*out) + *pos);
|
|
||||||
*out = strncat(*out, *line, *pos + 1);
|
|
||||||
|
|
||||||
*n = *pos;
|
|
||||||
}
|
}
|
58
src/keys.c
58
src/keys.c
@ -2,6 +2,7 @@
|
|||||||
#include "../include/output.h"
|
#include "../include/output.h"
|
||||||
#include "../include/utils.h"
|
#include "../include/utils.h"
|
||||||
#include "../include/complete.h"
|
#include "../include/complete.h"
|
||||||
|
#include "../include/shell.h"
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @brief Delete key action
|
* @brief Delete key action
|
||||||
@ -27,6 +28,7 @@ void delete_key(int pos, int *n, char **line)
|
|||||||
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);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -75,6 +77,8 @@ void home_key(int *pos)
|
|||||||
|
|
||||||
print_str(buff, buff_size);
|
print_str(buff, buff_size);
|
||||||
*pos = 0;
|
*pos = 0;
|
||||||
|
|
||||||
|
free(buff);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -95,6 +99,8 @@ void end_key(int *pos, int n)
|
|||||||
print_str(buff, buff_size);
|
print_str(buff, buff_size);
|
||||||
|
|
||||||
*pos = n;
|
*pos = n;
|
||||||
|
|
||||||
|
free(buff);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -123,6 +129,8 @@ void backspace_key(int *pos, int *n, char **line)
|
|||||||
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);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -137,16 +145,56 @@ void new_line()
|
|||||||
|
|
||||||
void tab_key(int *pos, int *n, char **line)
|
void tab_key(int *pos, int *n, char **line)
|
||||||
{
|
{
|
||||||
char *buff = malloc(1);
|
(*line)[*pos] = '\0';
|
||||||
|
*line = realloc(*line, strlen(*line) + 1);
|
||||||
|
|
||||||
|
*line = trim_string(line);
|
||||||
|
*pos = strlen(*line);
|
||||||
|
*n = *pos;
|
||||||
|
|
||||||
|
char *buff = malloc(1), *output;
|
||||||
buff[0] = '\0';
|
buff[0] = '\0';
|
||||||
size_t buff_size = 1;
|
size_t buff_size = 1;
|
||||||
|
|
||||||
char *options;
|
char **complete_options, *to_complete;
|
||||||
|
size_t opts_sz = get_complete_options(&complete_options, *line, &to_complete);
|
||||||
|
|
||||||
complete_line(pos, n, line, &options);
|
if (opts_sz == 1)
|
||||||
append_to_buff(&buff, &buff_size, options, strlen(options));
|
{
|
||||||
|
char *ending = complete_options[0] + strlen(to_complete);
|
||||||
|
*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, "\x1b[2K", 4);
|
||||||
|
|
||||||
|
for (int i = 0; i < 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);
|
print_str(buff, buff_size);
|
||||||
|
|
||||||
|
free(buff);
|
||||||
|
free(to_complete);
|
||||||
|
free_str_arr(complete_options);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -174,4 +222,6 @@ void printable_key(int *pos, int *n, char c, char **line)
|
|||||||
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);
|
||||||
}
|
}
|
14
src/main.c
14
src/main.c
@ -1,9 +1,13 @@
|
|||||||
#include <signal.h>
|
#include <signal.h>
|
||||||
|
#include <sys/stat.h>
|
||||||
|
#include <fcntl.h>
|
||||||
|
|
||||||
#include "../include/input.h"
|
#include "../include/input.h"
|
||||||
#include "../include/utils.h"
|
#include "../include/utils.h"
|
||||||
#include "../include/shell.h"
|
#include "../include/shell.h"
|
||||||
|
|
||||||
|
FILE *log_file;
|
||||||
|
|
||||||
void exit_shell()
|
void exit_shell()
|
||||||
{
|
{
|
||||||
change_mode(0);
|
change_mode(0);
|
||||||
@ -11,6 +15,16 @@ void exit_shell()
|
|||||||
|
|
||||||
int main(int argc, char **argv)
|
int main(int argc, char **argv)
|
||||||
{
|
{
|
||||||
|
log_file = fopen("/var/log/mshell.log", "w");
|
||||||
|
if (log_file == NULL)
|
||||||
|
{
|
||||||
|
fprintf(stderr, "Couldn't open log file\n");
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
fprintf(log_file, "\n New session:\n");
|
||||||
|
fclose(log_file);
|
||||||
|
}
|
||||||
|
|
||||||
change_mode(1);
|
change_mode(1);
|
||||||
signal(SIGINT, SIG_IGN);
|
signal(SIGINT, SIG_IGN);
|
||||||
|
|
||||||
|
@ -44,13 +44,13 @@ void process_command()
|
|||||||
* @param line
|
* @param line
|
||||||
* @param args
|
* @param args
|
||||||
*/
|
*/
|
||||||
void process_line(char *line, char ***args)
|
int process_line(char *line, char ***args)
|
||||||
{
|
{
|
||||||
int buff_size = ARG_SIZE, pos = 0;
|
int buff_size = ARG_SIZE, pos = 0;
|
||||||
*args = malloc(buff_size * sizeof(char *));
|
*args = malloc(buff_size * sizeof(char *));
|
||||||
char *tok, *rest = strdup(line);
|
char *tok, *rest = strdup(line);
|
||||||
|
|
||||||
while ((tok = strtok_r(rest, " \t", &rest)) != NULL)
|
while ((tok = strtok_r(rest, " ", &rest)) != NULL)
|
||||||
{
|
{
|
||||||
(*args)[pos] = tok;
|
(*args)[pos] = tok;
|
||||||
pos++;
|
pos++;
|
||||||
@ -68,6 +68,7 @@ void process_line(char *line, char ***args)
|
|||||||
}
|
}
|
||||||
|
|
||||||
(*args)[pos] = NULL;
|
(*args)[pos] = NULL;
|
||||||
|
return pos;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -159,7 +160,7 @@ char *compose_prompt()
|
|||||||
char *prompt = malloc(1);
|
char *prompt = malloc(1);
|
||||||
prompt[0] = '\0';
|
prompt[0] = '\0';
|
||||||
|
|
||||||
prompt = realloc(prompt, strlen(prompt) + 3);
|
prompt = realloc(prompt, strlen(prompt) + 4);
|
||||||
if (getuid() == 0)
|
if (getuid() == 0)
|
||||||
{
|
{
|
||||||
prompt = strcat(prompt, "\n# ");
|
prompt = strcat(prompt, "\n# ");
|
||||||
|
@ -106,10 +106,11 @@ size_t list_strings_containing(struct tree_node *root, char *key, char ***string
|
|||||||
if (is_last == 0)
|
if (is_last == 0)
|
||||||
{
|
{
|
||||||
char *prefix = strdup(key);
|
char *prefix = strdup(key);
|
||||||
prefix = realloc(prefix, strlen(prefix) + 2);
|
|
||||||
|
|
||||||
get_all_substrings(current, &amount, &prefix, strings);
|
get_all_substrings(current, &amount, &prefix, strings);
|
||||||
|
|
||||||
|
free(prefix);
|
||||||
|
|
||||||
return amount;
|
return amount;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -132,6 +133,7 @@ void get_all_substrings(struct tree_node *root, size_t *amount, char **curr_pref
|
|||||||
{
|
{
|
||||||
if (root->child[index] != NULL)
|
if (root->child[index] != NULL)
|
||||||
{
|
{
|
||||||
|
*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);
|
||||||
|
44
src/utils.c
44
src/utils.c
@ -37,7 +37,7 @@ void remove_on_pos(char **str, int pos)
|
|||||||
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];
|
||||||
}
|
}
|
||||||
@ -47,3 +47,45 @@ void remove_on_pos(char **str, int pos)
|
|||||||
else
|
else
|
||||||
fprintf(stderr, "Can't remove symbol outside the string\n");
|
fprintf(stderr, "Can't remove symbol outside the string\n");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int sep_string(char *line, char ***toks, char *sep)
|
||||||
|
{
|
||||||
|
char *tmp_line = strdup(line);
|
||||||
|
int n = 0;
|
||||||
|
*toks = malloc(sizeof(char *) * n);
|
||||||
|
|
||||||
|
char *tmp;
|
||||||
|
while ((tmp = strsep(&tmp_line, sep)) != NULL)
|
||||||
|
{
|
||||||
|
n++;
|
||||||
|
*toks = realloc(*toks, sizeof(char *) * n);
|
||||||
|
(*toks)[n - 1] = strdup(tmp);
|
||||||
|
}
|
||||||
|
|
||||||
|
return n;
|
||||||
|
}
|
||||||
|
|
||||||
|
char *trim_string(char **str)
|
||||||
|
{
|
||||||
|
while ((*str)[0] == ' ')
|
||||||
|
memmove(*str, *str + 1, strlen(*str));
|
||||||
|
|
||||||
|
for (int i = 1; i < strlen(*str); i++)
|
||||||
|
if ((*str)[i] == ' ' && (*str)[i - 1] == ' ')
|
||||||
|
{
|
||||||
|
memmove(*str + i, *str + i + 1, strlen(*str + i));
|
||||||
|
i--;
|
||||||
|
}
|
||||||
|
|
||||||
|
return *str;
|
||||||
|
}
|
||||||
|
|
||||||
|
void free_str_arr(char **arr)
|
||||||
|
{
|
||||||
|
for (int i = 0; i < sizeof(arr) / sizeof(char *); i++)
|
||||||
|
{
|
||||||
|
free(arr[i]);
|
||||||
|
}
|
||||||
|
|
||||||
|
free(arr);
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user