Added permanent history saving into file and fixed hidded files suggestion
This commit is contained in:
parent
89c637eaa3
commit
d7037ec237
@ -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
|
|
||||||
|
@ -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
|
@ -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;
|
||||||
};
|
};
|
||||||
|
@ -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
|
@ -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++;
|
||||||
|
113
src/history.c
113
src/history.c
@ -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);
|
||||||
}
|
}
|
15
src/main.c
15
src/main.c
@ -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;
|
||||||
|
29
src/utils.c
29
src/utils.c
@ -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;
|
||||||
}
|
}
|
@ -1 +0,0 @@
|
|||||||
Test test.txt
|
|
Loading…
x
Reference in New Issue
Block a user