Substitutions

This commit is contained in:
Maxwell Jeffress 2025-03-24 15:36:08 +11:00
parent cbb9576778
commit 9f05a58157

View File

@ -1,25 +1,44 @@
#include <iostream> #include <iostream>
#include <string> #include <string>
#include <vector> #include <vector>
#include <ctime>
#include <filesystem> #include <filesystem>
#include <unistd.h> #include <unistd.h>
#include <sys/wait.h> #include <sys/wait.h>
#include <sys/types.h> #include <sys/types.h>
#include <pwd.h> #include <pwd.h>
#include <cstring> #include <cstring>
#include <limits.h>
#include <fstream>
using namespace std; using namespace std;
vector<string> builtInFunctions = {"exit", "cd"}; vector<string> builtInFunctions = {"exit", "cd", "sub"};
vector<pair<string, vector<string>>> substitutions = {{"ls", {"ls", "--color"}}};
void log(string input) { void log(string input) {
cout << input << endl; 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 prompt() {
string input; 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); getline(cin, input);
return input; return input;
} }
@ -58,8 +77,11 @@ string getHomeDir() {
vector<string> tokenise(string input) { vector<string> tokenise(string input) {
vector<string> output; vector<string> output;
string currentArg; string currentArg;
bool isString = false;
for (int i = 0; i < input.size(); i++) { 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); output.push_back(currentArg);
currentArg.clear(); currentArg.clear();
} else if (input[i] == '~') { } else if (input[i] == '~') {
@ -71,15 +93,48 @@ vector<string> tokenise(string input) {
if (!currentArg.empty()) { if (!currentArg.empty()) {
output.push_back(currentArg); 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]); string executableLocation = findExecutable(output[0]);
if (!executableLocation.empty()) { if (!executableLocation.empty()) {
output[0] = executableLocation; output[0] = executableLocation;
} else { } else {
log("Couldn't find an executable in the path"); log("Couldn't find an executable in the path");
output.clear();
} }
return output;
}
vector<string> tokeniseSubstitutions(string input) {
vector<string> 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; return output;
} }
@ -110,12 +165,34 @@ int runProcess(vector<string> args) {
return -1; 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<string> 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[]) { int main(int argc, char *argv[]) {
doStartup();
bool continueLoop = true; bool continueLoop = true;
while (continueLoop) { while (continueLoop) {
string input = prompt(); string input = prompt();
if (input == "") { if (input == "") {
break; continue;
} }
vector<string> tokens = tokenise(input); vector<string> tokens = tokenise(input);
auto path = filesystem::current_path(); auto path = filesystem::current_path();
@ -128,6 +205,23 @@ int main(int argc, char *argv[]) {
filesystem::current_path(tokens[1]); filesystem::current_path(tokens[1]);
} }
continue; continue;
} else if (tokens[0] == "sub") {
if (tokens.size() < 3) {
log("substitute requires 2 arguments");
continue;
}
vector<string> 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()) { if (tokens.empty()) {
continue; continue;