diff --git a/src/main.cpp b/src/main.cpp index a5e0130..f411ac0 100644 --- a/src/main.cpp +++ b/src/main.cpp @@ -1,25 +1,44 @@ #include #include #include - +#include #include #include #include #include #include #include +#include +#include using namespace std; -vector builtInFunctions = {"exit", "cd"}; +vector builtInFunctions = {"exit", "cd", "sub"}; + +vector>> substitutions = {{"ls", {"ls", "--color"}}}; void log(string input) { cout << input << endl; } +string getTime() { + time_t rawtime; + struct tm * timeinfo; + char buffer[40]; + + time(&rawtime); + timeinfo = localtime(&rawtime); + + strftime(buffer, sizeof(buffer),"%H:%M:%S", timeinfo); + string str(buffer); + return str; +} + string prompt() { string input; - cout << filesystem::current_path() << " > "; + char hostname[HOST_NAME_MAX]; + gethostname(hostname, HOST_NAME_MAX); + cout << "\x1B[0m{" << getTime() << "}\x1B[32m " << getenv("USER") << "@" << hostname << "\x1B[96m " << string(filesystem::current_path()) << endl << "\x1B[0m> "; getline(cin, input); return input; } @@ -58,8 +77,11 @@ string getHomeDir() { vector tokenise(string input) { vector output; string currentArg; + bool isString = false; for (int i = 0; i < input.size(); i++) { - if (input[i] == ' ' && !currentArg.empty()) { + if (input[i] == '"') { + isString = !isString; + } else if (input[i] == ' ' && !currentArg.empty() && !isString) { output.push_back(currentArg); currentArg.clear(); } else if (input[i] == '~') { @@ -71,15 +93,48 @@ vector tokenise(string input) { if (!currentArg.empty()) { output.push_back(currentArg); } - + for (int i = 0; i < substitutions.size(); i++) { + for (int j = 0; j < output.size(); j++) { + if (output[j] == substitutions[i].first) { + output.erase(output.begin() + j); + int substitutionSize = substitutions[i].second.size(); + for (int k = 0; k < substitutionSize; k++) { + output.insert(output.begin() + j, substitutions[i].second[k]); + j++; + } + j -= substitutionSize; + break; + } + } + } string executableLocation = findExecutable(output[0]); if (!executableLocation.empty()) { output[0] = executableLocation; } else { log("Couldn't find an executable in the path"); - output.clear(); } + return output; +} +vector tokeniseSubstitutions(string input) { + vector output; + string currentArg; + bool isString = false; + for (int i = 0; i < input.size(); i++) { + if (input[i] == '"') { + isString = !isString; + } else if (input[i] == ' ' && !isString) { + if (!currentArg.empty()) { + output.push_back(currentArg); + currentArg.clear(); + } + } else { + currentArg += input[i]; + } + } + if (!currentArg.empty()) { + output.push_back(currentArg); + } return output; } @@ -110,12 +165,34 @@ int runProcess(vector args) { return -1; } +void doStartup() { + struct passwd* pw = getpwuid(getuid()); + const char* homedir = pw->pw_dir; + string configLocation = string(homedir) + "/.config/space/Spacefile"; + if (filesystem::exists(configLocation)) { + string config; + ifstream configFile(configLocation); + while (getline(configFile, config)) { + vector tokens = tokenise(config); + if (tokens.empty()) continue; + int returnCode = runProcess(tokens); + if (returnCode != 0) { + log("Config file error:\nCommand " + tokens[0] + " failed with exit code " + to_string(returnCode)); + } + } + configFile.close(); + } else { + log("Spacefile does not exist"); + } +} + int main(int argc, char *argv[]) { + doStartup(); bool continueLoop = true; while (continueLoop) { string input = prompt(); if (input == "") { - break; + continue; } vector tokens = tokenise(input); auto path = filesystem::current_path(); @@ -128,6 +205,23 @@ int main(int argc, char *argv[]) { filesystem::current_path(tokens[1]); } continue; + } else if (tokens[0] == "sub") { + if (tokens.size() < 3) { + log("substitute requires 2 arguments"); + continue; + } + vector args = tokeniseSubstitutions(tokens[2]); + substitutions.push_back({tokens[1], args}); + continue; + } else if (tokens[0] == "listsubs") { + log("Substitutions:"); + for (int i = 0; i < substitutions.size(); i++) { + string substring; + for (int j = 0; j < substitutions[i].second.size(); j++) { + substring += substitutions[i].second[j] + " "; + } + cout << substitutions[i].first << " -> " << substring << endl; + } } if (tokens.empty()) { continue;