fish/fish_shell/fish_core.c

226 lines
4.1 KiB
C
Raw Normal View History

2017-05-11 15:16:06 +00:00
//
// Created by Antoine Bartuccio on 11/05/2017.
2017-05-15 12:32:41 +00:00
// Dont forget that Antoine Bartuccio is a faggot since 1784 (tm)
2017-05-11 15:16:06 +00:00
//
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <sys/wait.h>
#include "fish_core.h"
2017-05-15 12:32:41 +00:00
#include "fish_globbing.h"
#include "fish_types.h"
2017-05-11 15:16:06 +00:00
void fishLoop(Settings * settings){
2017-05-15 09:22:53 +00:00
char * line = NULL;
2017-05-15 14:53:56 +00:00
WordList* splited = NULL;
2017-05-15 09:22:53 +00:00
int status = 1;
2017-05-15 12:32:41 +00:00
do {
printf("%s", settings->PS1);
line = fishReadLine();
splited = split(line, (char*) FISH_TOKENS);
2017-05-15 12:32:41 +00:00
splited = fishExpand(splited);
status = fishExecute(splited);
2017-05-15 09:22:53 +00:00
free(line);
} while(status != EXIT_SIGNAL);
2017-05-11 15:16:06 +00:00
}
int countSeparators(char *string, char *separators) {
2017-05-15 09:22:53 +00:00
int nb = 0;
int i = 0;
int k = 0;
while (string[i] != '\0'){
while (separators[k] != '\0'){
if (string[i] == separators[k]){
nb++;
}
k++;
}
i++;
k = 0;
}
return nb;
2017-05-11 15:16:06 +00:00
}
2017-05-15 14:53:56 +00:00
WordList * split(char *string, char *separator){
WordList *list = createWordList();
2017-05-15 09:22:53 +00:00
char *to_delete = strdup(string);
2017-05-15 10:32:00 +00:00
char *to_delete_bak = to_delete;
2017-05-15 09:22:53 +00:00
char *token = NULL;
if (to_delete == NULL){
2017-05-15 12:08:10 +00:00
crash();
2017-05-15 09:22:53 +00:00
}
while((token = strsep(&to_delete, separator)) != NULL)
addEndWordList(list, token);
2017-05-15 09:22:53 +00:00
2017-05-15 10:32:00 +00:00
free(to_delete_bak);
2017-05-15 09:22:53 +00:00
2017-05-15 14:53:56 +00:00
return list;
2017-05-11 15:16:06 +00:00
}
char *fishReadLine() {
2017-05-15 09:22:53 +00:00
size_t bufferSize = FISH_BUFFER_SIZE;
int position = 0;
2017-05-15 22:08:07 +00:00
char *line = (char*) malloc(sizeof(char*) * bufferSize);
2017-05-15 09:22:53 +00:00
int c;
if (line == NULL){
2017-05-15 12:08:10 +00:00
crash();
2017-05-15 09:22:53 +00:00
}
while (1){
c = getchar();
switch (c){
case '\n':
line[position] = '\0';
return line;
case EOF:
exit(EXIT_SUCCESS);
default:
line[position] = (char) c;
}
position++;
if ((size_t) position > bufferSize){
bufferSize+=bufferSize;
2017-05-15 22:08:07 +00:00
line = (char*) realloc(line, bufferSize);
2017-05-15 09:22:53 +00:00
if (line == NULL){
2017-05-15 12:08:10 +00:00
crash();
2017-05-15 09:22:53 +00:00
}
}
}
return NULL;
2017-05-11 15:16:06 +00:00
}
Settings *getSettings() {
2017-05-15 09:22:53 +00:00
Settings *s = (Settings*) malloc(sizeof(Settings));
if (s == NULL){
2017-05-15 12:08:10 +00:00
crash();
2017-05-15 14:53:56 +00:00
} else {
s->PS1 = strdup("\n~>");
2017-05-15 09:22:53 +00:00
}
return s;
}
2017-05-15 10:32:00 +00:00
void freeSettings(Settings *settings){
if (settings != NULL){
free(settings->PS1);
free(settings);
}
}
int fishLoad(WordArray *array) {
2017-05-15 09:22:53 +00:00
pid_t pid;
int status = 1;
2017-05-15 09:22:53 +00:00
pid = fork();
if (pid == 0){
/* Executes only in the child process */
if (execvp(array->words[0], array->words) == -1){
/* Error during system call */
exit(EXIT_FAILURE);
2017-05-15 09:22:53 +00:00
}
exit(EXIT_SUCCESS);
2017-05-15 09:22:53 +00:00
} else if (pid < 0){
/* Fork failed */
perror("fish");
} else {
/* Handle parent process */
/* Wait for the child process to finish */
do {
waitpid(pid, &status, WUNTRACED);
} while (!WIFEXITED(status) && !WIFSIGNALED(status));
if (status) fprintf(stderr, "%s\n", getInsult());
}
2017-05-17 23:30:46 +00:00
freeWordArray(array);
return status;
}
int fishExecute(WordList *list) {
WordList *splited = NULL;
shell_operator op = NONE;
WordArray *array = NULL;
int signal = 1;
splited = parseWordList(list, &op);
array = wordListToWordArray(list);
signal = loadRightCommand(array);
if (signal == EXIT_SIGNAL){
if (splited != NULL) freeWordList(splited);
if (array != NULL) freeWordArray(array);
return signal;
}
switch (op) {
case AND:
if (!signal) signal = fishExecute(splited);
else {
if (splited != NULL) freeWordList(splited);
}
break;
case REVERSE_AND:
if (signal) signal = fishExecute(splited);
else {
if (splited != NULL) freeWordList(splited);
}
break;
case OR:
signal = fishExecute(splited);
break;
default:
break;
2017-05-15 09:22:53 +00:00
}
return signal;
2017-05-15 09:22:53 +00:00
}
int loadRightCommand(WordArray *array){
2017-05-15 09:22:53 +00:00
int i;
if (array->size <= 0) return 1;
for (i = 0; i < getNbBuiltins(); i++) {
if (!strcmp(array->words[0], getBuiltinCommandsStr()[i])) {
2017-05-15 09:22:53 +00:00
return getBuiltinCommands()[i](array);
}
}
return fishLoad(array);
}
WordList * parseWordList(WordList *list, shell_operator *an_operator) {
char *op_str[] = {
(char*) ";",
(char*) "\\|\\|",
(char*) "&&"
};
shell_operator op[] = {
OR,
REVERSE_AND,
AND
};
WordList *newList = NULL;
int max = sizeof(op_str) / sizeof(char*);
int i = 0;
while (i < max && newList == NULL){
newList = splitWordList(list, op_str[i]);
i++;
}
if (newList != NULL) *an_operator = op[i-1];
else *an_operator = NONE;
return newList;
}