Added permanent history saving into file and fixed hidded files suggestion

This commit is contained in:
Dmitriy Shishkov 2020-07-17 21:06:56 +05:00
parent 89c637eaa3
commit d7037ec237
11 changed files with 156 additions and 14 deletions

View File

@ -15,6 +15,7 @@ Work is still in porgress, buf when you will see a "finished" topic assigned to
* Running multiple commands separated by `;`, `&&` or `||` * Running multiple commands separated by `;`, `&&` or `||`
* Expand `*` wildcards * Expand `*` wildcards
* Commands I/O redirection with `|` pipes and `>`, `<`, `>>`, `<>` file input/output * Commands I/O redirection with `|` pipes and `>`, `<`, `>>`, `<>` file input/output
* Save history into file and access recent command in another instance of shell
# Builtin commands # Builtin commands
* `cd`: changes current working directory to the one specified by user. If no arguments provided, shows error. * `cd`: changes current working directory to the one specified by user. If no arguments provided, shows error.
@ -27,4 +28,3 @@ Work is still in porgress, buf when you will see a "finished" topic assigned to
* `Ctrl+Z` running programm with `fd` * `Ctrl+Z` running programm with `fd`
* Gettingcommands path from system `PATH` environment variable for autocomplete * Gettingcommands path from system `PATH` environment variable for autocomplete
* `$()` subcommands * `$()` subcommands
* Save history into file

1
foo
View File

@ -1 +0,0 @@
zim

View File

@ -5,5 +5,8 @@ void append_to_history(char *line);
void clear_sub_history(); void clear_sub_history();
char *previous_hist_entry(char *line); char *previous_hist_entry(char *line);
char *next_hist_entry(char *line); char *next_hist_entry(char *line);
void init_history();
void open_history_file();
void close_history_file();
#endif #endif

View File

@ -35,7 +35,7 @@ struct history
char **content; char **content;
int length; int length;
int pos; int pos;
char *curr_command; FILE *file;
struct hist_sub sub; struct hist_sub sub;
}; };

View File

@ -24,5 +24,6 @@ char **slice_array(char **arr, int beg, int end, bool asc);
int get_null_term_arr_size(char **arr); int get_null_term_arr_size(char **arr);
int append_to_str_arr(char ***arr, int *sz, char *str); int append_to_str_arr(char ***arr, int *sz, char *str);
char *get_curr_dir_name(); char *get_curr_dir_name();
int get_num_of_lines(FILE *file);
#endif #endif

View File

@ -50,7 +50,7 @@ ssize_t get_dir_list(char ***dir_list, char *path, int ex)
if (ex != 0 && !check_if_executable(path, ent->d_name)) if (ex != 0 && !check_if_executable(path, ent->d_name))
continue; continue;
if (ent->d_name[0] == '.') if (strcmp(ent->d_name, ".") == 0 || strcmp(ent->d_name, "..") == 0)
continue; continue;
n++; n++;

View File

@ -13,6 +13,9 @@ void append_to_history(char *line)
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)
return;
if (line[0] == '\0') if (line[0] == '\0')
return; return;
@ -20,6 +23,10 @@ void append_to_history(char *line)
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);
fprintf(term.hist.file, "%s\n", line);
fflush(term.hist.file);
term.hist.pos = -1; term.hist.pos = -1;
} }
@ -41,6 +48,51 @@ 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);
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)
{
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);
tmp_h = realloc(tmp_h, sizeof(char *) * (tmp_h_pos + 1));
getline(&tmp_h[tmp_h_pos], &llen, term.hist.file);
fseek(term.hist.file, f_pos, SEEK_SET);
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) if (line == NULL)
{ {
if (term.hist.pos + 1 == term.hist.length) if (term.hist.pos + 1 == term.hist.length)
@ -114,4 +166,65 @@ char *next_hist_entry(char *line)
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()
{
term.hist.length = 0;
term.hist.pos = -1;
term.hist.content = calloc(term.hist.length, sizeof(char *));
term.hist.sub.length = -1;
term.hist.sub.pos = -1;
term.hist.sub.content = calloc(0, sizeof(char *));
open_history_file();
size_t sz = 0;
int len = 0;
char *buf = NULL;
while ((len = getline(&buf, &sz, term.hist.file)) != -1)
{
char *tmp = strdup(buf);
if (tmp[strlen(tmp) - 1] == '\n')
tmp[strlen(tmp) - 1] = '\0';
append_to_str_arr(&term.hist.content, &term.hist.length, tmp);
free(tmp);
}
free(buf);
}
void open_history_file()
{
char *data_path;
if ((data_path = getenv("XDG_DATA_HOME")) == NULL)
{
char *user_home;
if ((user_home = getenv("HOME")) == NULL)
{
perror("getenv");
exit(EXIT_FAILURE);
}
data_path = strdup(user_home);
data_path = realloc(data_path, strlen(data_path) + strlen("/.local/share/") + 1);
data_path = strcat(data_path, "/.local/share/");
}
data_path = realloc(data_path, strlen(data_path) + strlen(".mshistory") + 1);
data_path = strcat(data_path, ".mshistory");
if ((term.hist.file = fopen(data_path, "a+")) == NULL)
{
perror("open");
exit(EXIT_FAILURE);
}
fseek(term.hist.file, 0, SEEK_SET);
}
void close_history_file()
{
fclose(term.hist.file);
} }

View File

@ -5,6 +5,7 @@
#include "../include/input.h" #include "../include/input.h"
#include "../include/utils.h" #include "../include/utils.h"
#include "../include/shell.h" #include "../include/shell.h"
#include "../include/history.h"
// Definitions // Definitions
t_ term; t_ term;
@ -25,7 +26,7 @@ int main()
// Init // Init
t_ init_term() t_ init_term()
{ {
// Log file
FILE *log_file = fopen("/var/log/mshell.log", "w"); FILE *log_file = fopen("/var/log/mshell.log", "w");
if (log_file == NULL) if (log_file == NULL)
{ {
@ -37,20 +38,18 @@ t_ init_term()
fclose(log_file); fclose(log_file);
} }
// Entering raw mode
change_mode(1); change_mode(1);
term.hist.length = 0; // Init history
term.hist.pos = -1; init_history();
term.hist.content = calloc(term.hist.length, sizeof(char *));
term.hist.sub.length = -1;
term.hist.sub.pos = -1;
term.hist.sub.content = calloc(0, sizeof(char *));
term.last_status = 0; term.last_status = 0;
// Disable Ctrl+C default behaviour for shell
signal(SIGINT, SIG_IGN); signal(SIGINT, SIG_IGN);
// Set up function to run on shell exit
atexit(exit_shell); atexit(exit_shell);
return term; return term;

View File

@ -190,4 +190,33 @@ char *get_curr_dir_name()
getcwd(pwd, FILENAME_MAX); getcwd(pwd, FILENAME_MAX);
return pwd; return pwd;
}
/**
* @brief Get the number of lines in file
*
* @param file
* @return int
*/
int get_num_of_lines(FILE *file)
{
int n = 0;
char ch, pch;
long curr_pos = ftell(file);
fseek(file, 0, SEEK_SET);
while ((ch = fgetc(file)) != EOF)
{
if (ch == '\n')
n++;
pch = ch;
}
if (pch != '\n')
n++;
fseek(file, curr_pos, SEEK_SET);
return n;
} }

View File

@ -1 +0,0 @@
Test

View File

@ -1 +0,0 @@
Test test.txt