mshell/src/shell.c

397 lines
8.5 KiB
C

#include "../include/shell.h"
#include "../include/input.h"
#include "../include/output.h"
#include "../include/utils.h"
/* Global definitions */
char *builtin[] = {
"cd",
"exit",
"exec"};
int (*builtin_func[])(char **) = {
&sh_cd,
&sh_exit,
&sh_exec};
/**
* @brief Function for main loop. It prints prompt, reads user's input and executes it
*/
void process_command()
{
cmds_p *coms = NULL;
char *prompt = compose_prompt();
print_str(prompt, strlen(prompt));
char *line = read_line();
char *tmp = trim_string(line, false);
free(line);
line = tmp;
process_line(line, &coms);
cmds_p *curr = coms;
int status = 0;
while (curr != NULL)
{
status = execute(curr);
if (curr->stat.invert)
if (status == 0)
status = 1;
else
status = 0;
curr->stat.s = status;
term.last_status = curr->stat.s;
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);
}
/**
* @brief Extracts command and its arguments from line
*
* @param line
* @param args
*/
int process_line(char *line, cmds_p **coms)
{
char *tmp = strdup(line), *free_tmp = tmp;
int line_size = strlen(line), args_am = 0;
*coms = new_cmd();
cmds_p *curr_cmd = *coms;
for (int i = 0; i < line_size; i++)
{
if (line[i] == '"')
{
tmp++;
i++;
int j = i;
for (; j < line_size; j++)
if (line[j] == '"')
{
free_tmp[j] = '\0';
append_to_str_arr(&(curr_cmd->args), &args_am, trim_string(tmp, false));
tmp += strlen(curr_cmd->args[args_am - 1]) + 1;
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_cmd->args), &args_am, trim_string(tmp, false));
tmp += strlen(curr_cmd->args[args_am - 1]) + 1;
}
else
tmp++;
curr_cmd->args = realloc(curr_cmd->args, sizeof(char *) * (args_am + 1));
curr_cmd->args[args_am] = NULL;
switch (line[i])
{
case ';':
curr_cmd->sep_next = SEMICOLON_SEP;
break;
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++;
}
cmds_p *next = new_cmd();
curr_cmd->next = next;
curr_cmd = curr_cmd->next;
args_am = 0;
}
else if (line[i] == ' ')
{
free_tmp[i] = '\0';
append_to_str_arr(&(curr_cmd->args), &args_am, trim_string(tmp, false));
tmp += strlen(curr_cmd->args[args_am - 1]) + 1;
}
else if (line[i] == '#')
{
break;
}
}
if (tmp[0] != '\0' && tmp[0] != '#')
{
append_to_str_arr(&(curr_cmd->args), &args_am, trim_string(tmp, false));
}
if (curr_cmd->args[0] != NULL)
{
if (strcmp(curr_cmd->args[0], "!") == 0)
{
curr_cmd->stat.invert = true;
}
}
free(free_tmp);
curr_cmd->args = realloc(curr_cmd->args, sizeof(char *) * (args_am + 1));
curr_cmd->args[args_am] = NULL;
return args_am;
}
/**
* @brief Executes either builtin or system command
*
* @param args
* @return int
*/
int execute(cmds_p *command)
{
if (command->args[0] == NULL)
return 1;
char **args = command->args;
if (strcmp(command->args[0], "!") == 0)
args = slice_array(args, 1, -1, true);
for (int i = 0; i < BUILTIN_NUM; i++)
if (strcmp(args[0], builtin[i]) == 0)
return (*builtin_func[i])(args);
return launch(args);
}
/**
* @brief Launches separate process and executes specified command
*
* @param args
* @return int
*/
int launch(char **args)
{
pid_t pid, wpid;
int status;
pid = fork();
if (pid == 0)
{
sh_exec(args);
}
else if (pid < 0)
{
perror("myshell1");
}
else
do
wpid = waitpid(pid, &status, WUNTRACED);
while (!WIFEXITED(status) && !WIFSIGNALED(status));
change_mode(1);
return status;
}
/**
* @brief Shell builtin command. Changes current working directory
*
* @param args
* @return int
*/
int sh_cd(char **args)
{
if (args[1] == NULL)
chdir(getenv("HOME"));
else if (chdir(args[1]) < 0)
perror("myshell2");
return 0;
}
/**
* @brief Shell builtin command. Exits shell
*
* @param args
* @return int
*/
int sh_exit(char **args)
{
exit(0);
}
/**
* @brief Shell builtin command. Executes command replacing shell with it
*
* @param args
* @return int
*/
int sh_exec(char **args)
{
change_mode(0);
signal(SIGINT, SIG_DFL);
if (strcmp(args[0], "exec") == 0)
args = slice_array(args, 1, -1, 1);
if (execvp(args[0], args) < 0)
{
perror("myshell3");
}
exit(EXIT_FAILURE);
}
char *get_ip_addr()
{
struct ifaddrs *host, *tmp;
getifaddrs(&host);
tmp = host;
char *ip = NULL;
while (tmp)
{
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);
if (strncmp(ip, "127", 3) != 0)
break;
}
tmp = tmp->ifa_next;
}
freeifaddrs(host);
return ip;
}
/**
* @brief Creates prompt string
*
* @return char*
*/
char *compose_prompt()
{
// New line
char *prompt = strdup("");
// Username
char *username = getenv("USER");
if (username == NULL)
username = "none";
prompt = realloc(prompt, strlen(prompt) + strlen("\033[97;44m") + strlen(username) + 2);
prompt = strcat(prompt, "\033[97;44m");
prompt = strcat(prompt, username);
// Current host ip
char *ip = get_ip_addr();
if (ip == NULL)
ip = "none";
prompt = realloc(prompt, strlen(prompt) + 1 + strlen(ip) + strlen("\033[39m") + strlen("\033[48;2;28;32;35m") + 2);
prompt = strcat(prompt, "@");
prompt = strcat(prompt, ip);
prompt = strcat(prompt, "\033[39m");
prompt = strcat(prompt, "\033[48;2;28;32;35m");
prompt = strcat(prompt, ":");
// Current path
char *full_path = get_current_dir_name();
prompt = realloc(prompt, strlen(prompt) + strlen("\033[92;1m") + strlen("\033[39;0m") + strlen(full_path) + 2);
prompt = strcat(prompt, "\033[92;1m");
prompt = strcat(prompt, full_path);
prompt = strcat(prompt, "\033[39;0m");
free(full_path);
// Previous status
if (term.last_status != 0)
{
char *status = malloc(snprintf(NULL, 0, " \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 = strcat(prompt, status);
free(status);
}
// Permissions specific character before user input
prompt = realloc(prompt, strlen(prompt) + 4);
if (getuid() == 0)
{
prompt = strcat(prompt, "\n# ");
}
else
prompt = strcat(prompt, "\n% ");
return prompt;
}
cmds_p *new_cmd()
{
cmds_p *new = malloc(sizeof(cmds_p));
new->args = calloc(0, sizeof(char *));
new->stat.s = 0;
new->stat.invert = false;
new->next = NULL;
}