X-Git-Url: https://git.armaanb.net/?p=gen-shell.git;a=blobdiff_plain;f=src%2Fmain.cpp;h=0d7a03412413b4ec790c1f64b06d8442df1d2c8d;hp=afba7277ec511044acd58b0b19a9f0e3209dbd68;hb=cc55b39b55f9906bf5364c8f69d9c87ae928400c;hpb=a5d9cd320e9ed83e36de2b5326f1201ea16b697f diff --git a/src/main.cpp b/src/main.cpp index afba727..0d7a034 100644 --- a/src/main.cpp +++ b/src/main.cpp @@ -1,6 +1,6 @@ //////////////////////////////////////////////////////////////////////////////// // -// Copyright 2006 - 2017, Paul Beckingham, Federico Hernandez. +// Copyright 2006 - 2017, Paul Beckingham, Federico Hernandez, 2020 Armaan Bhojwani // // Permission is hereby granted, free of charge, to any person obtaining a copy // of this software and associated documentation files (the "Software"), to deal @@ -26,167 +26,125 @@ #include #include -#include -#include -#include -#include +#include <../Sarge/src/sarge.h> #include -#include -#include #ifdef HAVE_READLINE -#include -#include + #include + #include #endif -// TODO These conflict with tw commands. This needs to be resolved. -// Perhaps an escape, such as '-- help' could invoke local help, or using -// a 'task' prefix could disambiguate. - -// tasksh commands. -int cmdHelp (); -int cmdDiagnostics (); -int cmdReview (const std::vector &, bool); -int cmdShell (const std::vector &); -std::string promptCompose (); -std::string findTaskwarrior (); - //////////////////////////////////////////////////////////////////////////////// -static void welcome () -{ - std::cout << PACKAGE_STRING << "\n"; - cmdHelp (); -} -//////////////////////////////////////////////////////////////////////////////// -const std::string getResponse (const std::string& prompt) -{ - std::string response {""}; +using namespace std; - // Display prompt, get input. -#ifdef HAVE_READLINE - char *line_read = readline (prompt.c_str ()); - if (! line_read) - { +const std::string getResponse(const std::string & prompt) { + std::string response { + "" + }; + + // Display prompt, get input + #ifdef HAVE_READLINE + char * line_read = readline(prompt.c_str()); + if (!line_read) { std::cout << "\n"; response = ""; + } else { + // Save history + if ( * line_read) + add_history(line_read); + + response = std::string(line_read); + free(line_read); } - else - { - // Save history. - if (*line_read) - add_history (line_read); - - response = std::string (line_read); - free (line_read); - } -#else + #else std::cout << prompt; - std::getline (std::cin, response); - if (std::cin.eof () == 1) - { + std::getline(std::cin, response); + if (std::cin.eof() == 1) { std::cout << "\n"; response = ""; } -#endif + #endif return response; } //////////////////////////////////////////////////////////////////////////////// -static int commandLoop (bool autoClear) -{ - // Compose the prompt. - auto prompt = promptCompose (); - // Display prompt, get input. - auto command = getResponse (prompt); - - // Obey Taskwarrior's rc.tasksh.autoclear. - if (autoClear) - std::cout << "\033[2J\033[0;0H"; - - int status = 0; - if (! isatty (fileno (stdin)) && command == "") - { - status = -1; - } - else if (command != "") - { - auto args = split (command, ' '); - - // Dispatch command. - if (args[0] == "") status = -1; - else if (closeEnough ("exit", args[0], 3)) status = -1; - else if (closeEnough ("quit", args[0], 3)) status = -1; - else if (closeEnough ("help", args[0], 3)) status = cmdHelp (); - else if (closeEnough ("diagnostics", args[0], 3)) status = cmdDiagnostics (); - else if (closeEnough ("review", args[0], 3)) status = cmdReview (args, autoClear); - else if (closeEnough ("exec", args[0], 3) || - args[0][0] == '!') status = cmdShell (args); - else if (command != "") - { - command = "task " + command; - std::cout << "[" << command << "]\n"; - system (command.c_str ()); - - // Deliberately ignoreѕ taskwarrior exit status, otherwise empty filters - // cause the shell to terminate. - } - } - - return status; -} - -//////////////////////////////////////////////////////////////////////////////// -int main (int argc, const char** argv) +int main(int argc, char** argv) { - int status = 0; - // Lightweight version checking that doesn't require initialization or any I/O. - if (argc == 2 && !strcmp (argv[1], "--version")) - { - std::cout << VERSION << "\n"; - } - else - { - try + // Command line arguments + Sarge sarge; + + sarge.setArgument("a", "after", "Command to execute before leaving the shell", true); + sarge.setArgument("b", "before", "Command to execute before entering the shell", true); + sarge.setArgument("c", "command", "Command to convert to shell", true); + sarge.setArgument("h", "help", "Get help.", false); + sarge.setArgument("p", "prompt", "Define a custom prompt", true); + sarge.setArgument("", "no-space", "Dont automatically add spaces after custom prompt and command", false); + sarge.setDescription("Make a shell from any executable"); + sarge.setUsage("gen-shell "); + + if (!sarge.parseArguments(argc, argv)) { + std::cerr << "Couldn't parse arguments..." << std::endl; + return 1; + } + + if (sarge.exists("help")) { + sarge.printHelp(); + return 0; + } + + bool space; + if (sarge.exists("no-space")) { + space = false; + } + + // Define input command + string arg_cmd; + sarge.getFlag("command", arg_cmd); + if ( space ) + arg_cmd += " "; + + // Define prompt + string prompt = ""; + sarge.getFlag("prompt", prompt); + + if ( prompt == "" ) + prompt = "% "; + if ( space ) + prompt += " "; + + // Execute before-command + string before_command; + sarge.getFlag("before", before_command); + system (before_command.c_str ()); + + // Execute after-command + string after_command; + sarge.getFlag("after", after_command); + + // Main program + while (true) { + // Display prompt, get input + auto command = getResponse(prompt); + + if (command != "") { - // Get the Taskwarrior rc.tasksh.autoclear Boolean setting. - bool autoClear = false; - std::string input; - std::string output; - execute ("task", {"_get", "rc.tasksh.autoclear"}, input, output); - output = lowerCase (output); - autoClear = (output == "true\n" || - output == "1\n" || - output == "y\n" || - output == "yes\n" || - output == "on\n"); - - if (isatty (fileno (stdin))) - welcome (); - - while ((status = commandLoop (autoClear)) == 0) - ; - } - - catch (const std::string& error) - { - std::cerr << error << "\n"; - status = -1; - } - - catch (...) - { - std::cerr << "Unknown error." << "\n"; - status = -2; + // Dispatch command + if (command == "") + { + system (after_command.c_str ()); + return 0; + } + else if (command != "") + { + string whole_command = arg_cmd + command; + system (whole_command.c_str ()); + } } } - - // Returning -1 drops out of the command loop, but gets translated to 0 here, - // so that there is a clean way to exit. - return status == -1 ? 0 : status; } ////////////////////////////////////////////////////////////////////////////////