Refactored tab autocompletion, fixed some more memory leaks, added logging to separate file

This commit is contained in:
Dmitriy Shishkov 2020-07-10 18:03:13 +05:00
parent a6ccd9344f
commit dceafa7244
9 changed files with 208 additions and 68 deletions

View File

@ -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

View File

@ -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);

View File

@ -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

View File

@ -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(); return sz;
for (size_t i = 0; i < sz; i++) }
{
insert_tree(child_dirs_root, comp_list[i]); size_t filter_options(char ***comp_list, size_t *size, char *filter_string)
} {
struct tree_node *child_dirs_root = get_new_node();
sz = list_strings_containing(child_dirs_root, curr_path, &comp_list); for (size_t i = 0; i < *size; i++)
} insert_tree(child_dirs_root, (*comp_list)[i]);
if (sz == 1) *size = list_strings_containing(child_dirs_root, filter_string, comp_list);
{
*out = strdup(comp_list[0] + strlen(curr_path)); free_tree(child_dirs_root);
*pos += strlen(*out);
*n = *pos; return *size;
*line = realloc(*line, strlen(*line) + strlen(*out));
*line = strcat(*line, *out);
return;
}
for (int i = 0; i < sz; i++)
{
*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;
} }

View File

@ -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);
} }

View File

@ -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);

View File

@ -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# ");

View File

@ -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);

View File

@ -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);
}