#include "mpw-shell.h" #include "fdset.h" #include "value.h" #include "environment.h" #include #include #include #include #include #include #include #include "cxx/string_splitter.h" #include "cxx/filesystem.h" //MAXPATHLEN #include #include #include "version.h" namespace ToolBox { std::string MacToUnix(const std::string path); std::string UnixToMac(const std::string path); } namespace fs = filesystem; namespace { std::string &lowercase(std::string &s) { std::transform(s.begin(), s.end(), s.begin(), [](char c){ return std::tolower(c); }); return s; } // doesn't handle flag arguments but builtins don't have arguments. template std::vector getopt(const std::vector &argv, FX fx) { std::vector out; out.reserve(argv.size()); std::copy_if(argv.begin()+1, argv.end(), std::back_inserter(out), [&fx](const std::string &s){ if (s.empty()) return false; // ? if (s.front() == '-') { std::for_each(s.begin() + 1, s.end(), fx); return false; } return true; }); return out; } template class offset_range { public: typedef typename T::iterator iterator; typedef typename T::const_iterator const_iterator; offset_range(T &t, size_t offset) : _t(t), _offset(offset) {} offset_range(const offset_range &) = default; offset_range(offset_range &&) = default; offset_range() = delete; auto begin() { return std::next(_t.begin(), _offset); } auto begin() const { return std::next(_t.begin(), _offset); } auto cbegin() { return std::next(_t.begin(), _offset); } auto end() { return _t.end(); } auto end() const { return _t.end(); } auto cend() const { return _t.cend(); } private: T &_t; size_t _offset; }; template offset_range make_offset_range(T &t, size_t offset) { return offset_range(t, offset); } } #undef stdin #undef stdout #undef stderr #define stdin fds[0] #define stdout fds[1] #define stderr fds[2] #undef fprintf #undef fputs #undef fputc #define fprintf(...) dprintf(__VA_ARGS__) inline int fputs(const char *data, int fd) { auto rv = write(fd, data, strlen(data)); return rv < 0 ? EOF : rv; } inline int fputc(int c, int fd) { unsigned char tmp = c; auto rv = write(fd, &tmp, 1); return rv < 0 ? EOF : c; } std::vector load_argv(Environment &env) { std::vector rv; int n = env.pound(); if ( n <= 0 ) return rv; n = std::min(n, (int)255); rv.reserve(n); for (int i = 1; i <= n; ++i) { rv.push_back(env.get(std::to_string(i))); } return rv; } int builtin_shift(Environment &env, const std::vector &tokens, const fdmask &fds) { int n = 1; if (tokens.size() > 3) { fputs("### Shift - Too many parameters were specified.\n", stderr); fputs("# Usage - Shift [number]\n", stderr); return 1; } if (tokens.size() == 2) { value v(tokens[1]); if (v.is_number() && v.number >= 0) n = v.number; else { fputs("### Shift - The parameter must be a positive number.\n", stderr); fputs("# Usage - Shift [number]\n", stderr); return 1; } } if (n == 0) return 0; auto argv = load_argv(env); if (argv.empty()) return 0; std::move(argv.begin() + n , argv.end(), argv.begin()); do { env.unset(std::to_string(argv.size())); argv.pop_back(); } while (--n); env.set_argv(argv); return 0; } int builtin_unset(Environment &env, const std::vector &tokens, const fdmask &) { for (const auto &s : make_offset_range(tokens, 1)) { env.unset(s); } // unset [no arg] removes ALL variables if (tokens.size() == 1) { env.unset(); } return 0; } int builtin_set(Environment &env, const std::vector &tokens, const fdmask &fds) { // set var name -- set // set var -- just print the value // 3.5 supports -e to also export it. //io_helper io(fds); if (tokens.size() == 1) { for (const auto &kv : env) { std::string name = quote(kv.first); std::string value = quote(kv.second); fprintf(stdout, "Set %s%s %s\n", bool(kv.second) ? "-e " : "", name.c_str(), value.c_str()); } return 0; } if (tokens.size() == 2) { std::string name = tokens[1]; auto iter = env.find(name); if (iter == env.end()) { fprintf(stderr, "### Set - No variable definition exists for %s.\n", name.c_str()); return 2; } name = quote(name); std::string value = quote(iter->second); fprintf(stdout, "Set %s%s %s\n", bool(iter->second) ? "-e " : "", name.c_str(), value.c_str()); return 0; } bool exported = false; if (tokens.size() == 4 && tokens[1] == "-e") { exported = true; } if (tokens.size() > 3 && !exported) { fputs("### Set - Too many parameters were specified.\n", stderr); fputs("# Usage - set [name [value]]\n", stderr); return 1; } std::string name = tokens[1+exported]; std::string value = tokens[2+exported]; env.set(name, value, exported); return 0; } static int export_common(Environment &env, bool export_or_unexport, const std::vector &tokens, const fdmask &fds) { const char *name = export_or_unexport ? "Export" : "Unexport"; struct { int _r = 0; int _s = 0; } flags; bool error = false; std::vector argv = getopt(tokens, [&](char c){ switch(c) { case 'r': case 'R': flags._r = true; break; case 's': case 'S': flags._s = true; break; default: fprintf(stderr, "### %s - \"-%c\" is not an option.\n", name, c); error = true; break; } }); if (error) { fprintf(stderr, "# Usage - %s [-r | -s | name...]\n", name); return 1; } if (argv.empty()) { if (flags._r && flags._s) goto conflict; // list of exported vars. // -r will generate unexport commands for exported variables. // -s will only print the names. name = export_or_unexport ? "Export " : "Unexport "; for (const auto &kv : env) { const std::string& vname = kv.first; if (kv.second == export_or_unexport) fprintf(stdout, "%s%s\n", flags._s ? "" : name, quote(vname).c_str()); } return 0; } else { // mark as exported. if (flags._r || flags._s) goto conflict; for (std::string s : argv) { auto iter = env.find(s); if (iter != env.end()) iter->second = export_or_unexport; } return 0; } conflict: fprintf(stderr, "### %s - Conflicting options or parameters were specified.\n", name); fprintf(stderr, "# Usage - %s [-r | -s | name...]\n", name); return 1; } int builtin_export(Environment &env, const std::vector &tokens, const fdmask &fds) { return export_common(env, true, tokens, fds); } int builtin_unexport(Environment &env, const std::vector &tokens, const fdmask &fds) { return export_common(env, false, tokens, fds); } int builtin_alias(Environment &env, const std::vector &tokens, const fdmask &fds) { // alias -> lists all aliases // alias name -> list single alias // alias name parms... -> add a new alias. if (tokens.size() == 1) { for (const auto &p : env.aliases()) { fprintf(stdout, "Alias %s %s\n", quote(p.first).c_str(), quote(p.second).c_str()); } return 0; } std::string name = tokens[1]; if (tokens.size() == 2) { const auto as = env.find_alias(name); if (as.empty()) { fprintf(stderr, "### Alias - No alias exists for %s\n", quote(name).c_str()); return 1; } fprintf(stdout, "Alias %s %s\n", quote(name).c_str(), quote(as).c_str()); return 0; } std::string as; for (const auto &s : make_offset_range(tokens, 2)) { as += s; as.push_back(' '); } as.pop_back(); // add/remove it to the alias table... if (as.empty()) { env.remove_alias(name); } else { env.add_alias(std::move(name), std::move(as)); } return 0; } int builtin_unalias(Environment &env, const std::vector &tokens, const fdmask &) { // unalias -> remove all aliases. // unalias name -> remove single alias. if (tokens.size() == 1) { env.remove_alias(); return 0; } for (const auto &x : make_offset_range(tokens, 1)) { env.remove_alias(x); } return 0; } int builtin_echo(Environment &env, const std::vector &tokens, const fdmask &fds) { //io_helper io(fds); bool space = false; bool n = false; for (const auto &s : make_offset_range(tokens, 1)) { if (s == "-n" || s == "-N") { n = true; continue; } if (space) { fputs(" ", stdout); } fputs(s.c_str(), stdout); space = true; } if (!n) fputs("\n", stdout); return 0; } int builtin_quote(Environment &env, const std::vector &tokens, const fdmask &fds) { // todo... //io_helper io(fds); bool space = false; bool n = false; for (const auto &s : make_offset_range(tokens, 1)) { if (s == "-n" || s == "-N") { n = true; continue; } if (space) { fputs(" ", stdout); } fputs(quote(s).c_str(), stdout); space = true; } if (!n) fputs("\n", stdout); return 0; } int builtin_parameters(Environment &env, const std::vector &argv, const fdmask &fds) { //io_helper io(fds); int i = 0; for (const auto &s : argv) { fprintf(stdout, "{%d} %s\n", i++, s.c_str()); } return 0; } int builtin_directory(Environment &env, const std::vector &tokens, const fdmask &fds) { // directory [-q] // directory path // for relative names, uses {DirectoryPath} (if set) rather than . // set DirectoryPath ":,{MPW},{MPW}Projects:" /* * Parameters: * ---------- * directory * Sets the default directory to directory. If you specify directory * as a leafname (that is, the final portion of a full pathname), the * MPW Shell searches for the directory in the current directory path * (for example, searching "{MPW}Examples:" for CExamples). However, if * the MPW Shell fails to find the directory in the current directory * path, it searches the directories listed in the {DirectoryPath} MPW * Shell variable, which contains a list of directories to be searched * in order of precedence. The last example illustrates how to do this. * * Options: * ------- * -q * Inhibits quoting the directory pathname written to standard * output. This option applies only if you omit the directory * parameter Normally the MPW Shell quotes the current default * directory name if it contains spaces or other special characters * * Status: * ------ * Directory can return the following status codes: * * 0 no errors * 1 directory not found; command aborted; or parameter error * */ //io_helper io(fds); bool q = false; bool error = false; std::vector argv = getopt(tokens, [&](char c){ switch(c) { case 'q': case 'Q': q = true; break; default: fprintf(stderr, "### Directory - \"-%c\" is not an option.\n", c); error = true; break; } }); if (error) { fputs("# Usage - Directory [-q | directory]\n", stderr); return 1; } if (argv.size() > 1) { fputs("### Directory - Too many parameters were specified.\n", stderr); fputs("# Usage - Directory [-q | directory]\n", stderr); return 1; } if (argv.size() == 1) { //cd if (q) { fputs("### Directory - Conflicting options or parameters were specified.\n", stderr); return 1; } // todo -- if relative path does not exist, check {DirectoryPath} fs::path path = ToolBox::MacToUnix(argv.front()); std::error_code ec; current_path(path, ec); if (ec) { fputs("### Directory - Unable to set current directory.\n", stderr); fprintf(stderr, "# %s\n", ec.message().c_str()); return 1; } } else { // pwd std::error_code ec; fs::path path = fs::current_path(ec); if (ec) { fputs("### Directory - Unable to get current directory.\n", stderr); fprintf(stderr, "# %s\n", ec.message().c_str()); return 1; } // todo -- pathname translation? std::string s = path; if (!q) s = quote(std::move(s)); fprintf(stdout, "%s\n", s.c_str()); } return 0; } int builtin_exists(Environment &env, const std::vector &tokens, const fdmask &fds) { bool _a = false; // print if alias/symlink bool _d = false; // print if directory bool _f = false; // print if normal file bool _n = false; // don't follow alias bool _w = false; // print if normal file + writable bool _q = false; // don't quote names bool error = false; std::vector argv = getopt(tokens, [&](char c){ switch(tolower(c)) { case 'a': _a = true; break; case 'd': _d = true; break; case 'f': _f = true; break; case 'n': _n = true; break; case 'q': _q = true; break; case 'w': _w = true; break; default: fprintf(stderr, "### Exists - \"-%c\" is not an option.\n", c); error = true; break; } }); if (_w) _f = false; if (_a + _d + _f + _w > 1) { fputs("### Exists - Conflicting options were specified.\n", stderr); error = true; } if (argv.size() < 1) { fputs("### Exists - Not enough parameters were specified.\n", stderr); error = true; } if (error) { fputs("# Usage - Exists [-a | -d | -f | -w] [-n] [-q] name...\n", stderr); return 1; } for (auto &s : argv) { std::string path = ToolBox::MacToUnix(s); std::error_code ec; fs::file_status st = _n ? fs::symlink_status(path, ec) : fs::status(path, ec); if (ec) continue; if (_d && !fs::is_directory(st)) continue; if (_a && !fs::is_symlink(st)) continue; if (_f && !fs::is_regular_file(st)) continue; if (_w && !fs::is_regular_file(st)) continue; if (_w && (access(path.c_str(), W_OK | F_OK) < 0)) continue; if (!_q) s = quote(std::move(s)); fprintf(stdout, "%s\n", s.c_str()); } return 0; } static bool is_assignment(int type) { switch(type) { case '=': case '+=': case '-=': return true; default: return false; } } int builtin_evaluate(Environment &env, std::vector &&tokens, const fdmask &fds) { // evaluate expression // evaluate variable = expression // evaluate variable += expression // evaluate variable -= expression // flags -- -h -o -b -- print in hex, octal, or binary // convert the arguments to a stack. int output = 'd'; //io_helper io(fds); std::reverse(tokens.begin(), tokens.end()); // remove 'Evaluate' tokens.pop_back(); // check for -h -x -o if (tokens.size() >= 2 && tokens.back().type == '-') { const token &t = tokens[tokens.size() - 2]; if (t.type == token::text && t.string.length() == 1) { int flag = tolower(t.string[0]); switch(flag) { case 'o': case 'h': case 'b': output = flag; tokens.pop_back(); tokens.pop_back(); } } } if (tokens.size() >= 2 && tokens.back().type == token::text) { int type = tokens[tokens.size() -2].type; if (is_assignment(type)) { std::string name = tokens.back().string; tokens.pop_back(); tokens.pop_back(); int32_t i = evaluate_expression("Evaluate", std::move(tokens)); switch(type) { case '=': env.set(name, i); break; case '+=': case '-=': { value old; auto iter = env.find(name); if (iter != env.end()) old = (const std::string &)iter->second; switch(type) { case '+=': i = old.to_number() + i; break; case '-=': i = old.to_number() - i; break; } env.set(name, i); } break; } return 0; } } int32_t i = evaluate_expression("Evaluate", std::move(tokens)); if (output == 'h') { fprintf(stdout, "0x%08x\n", i); return 0; } if (output == 'b') { std::string tmp("0b"); for (int j = 0; j < 32; ++j) { tmp.push_back(i & 0x80000000 ? '1' : '0'); i <<= 1; } tmp.push_back('\n'); fputs(tmp.c_str(), stdout); return 0; } if (output == 'o') { // octal. fprintf(stdout, "0%o\n", i); return 0; } fprintf(stdout, "%d\n", i); return 0; } int builtin_which(Environment &env, const std::vector &tokens, const fdmask &fds) { // which [-a] [-p] [command] bool a = false; bool p = false; bool error = false; std::vector argv = getopt(tokens, [&](char c){ switch(c) { case 'a': case 'A': a = true; break; case 'p': case 'P': p = true; break; default: fprintf(stderr, "### Which - \"-%c\" is not an option.\n", c); error = true; break; } }); if (argv.size() > 1) { fprintf(stderr, "### Which - Too many parameters were specified.\n"); error = true; } if (error) { fprintf(stderr, "# Usage - Which [-a] [-p] [name]\n"); return 1; } std::string s = env.get("commands"); string_splitter ss(s, ','); if (argv.empty()) { // just print the paths. for (; ss; ++ss) { fprintf(stdout, "%s\n", ss->c_str()); } return 0; } std::string target = argv[0]; bool found = false; // if absolute or relative path, check that. if (target.find_first_of("/:") != target.npos) { std::error_code ec; fs::path p(ToolBox::MacToUnix(target)); if (fs::exists(p, ec)) { fprintf(stdout, "%s\n", quote(p).c_str()); return 0; } else { fprintf(stderr, "### Which - File \"%s\" not found.\n", target.c_str()); return 2; } } for(; ss; ++ss) { if (p) fprintf(stderr, "checking %s\n", ss->c_str()); std::error_code ec; fs::path p(ToolBox::MacToUnix(ss->c_str())); p /= target; if (fs::exists(p, ec)) { found = true; fprintf(stdout, "%s\n", quote(p).c_str()); if (!a) break; } } // check builtins... if (!found || a) { static const char *builtins[] = { "aboutbox", "alias", "catenate", "directory", "echo", "exists", "export", "parameters", "quote", "set", "shift", "unalias", "unexport", "unset", "version", "which", }; lowercase(target); auto iter = std::find(std::begin(builtins), std::end(builtins), target); if (iter != std::end(builtins)) { fprintf(stdout, "%s\n", *iter); found = true; } } if (found) return 0; // also check built-ins? fprintf(stderr, "### Which - Command \"%s\" was not found.\n", target.c_str()); return 2; // not found. } int cat_helper(int in, int out) { static uint8_t buffer[4096]; for(;;) { ssize_t rcount = read(in, buffer, sizeof(buffer)); if (rcount < 0) { if (errno == EINTR) continue; return 2; } if (rcount == 0) break; for (;;) { ssize_t wcount = write(out, buffer, rcount); if (wcount < 0) { if (errno == EINTR) continue; return 2; } if (wcount != rcount) return 2; break; } } return 0; } int builtin_catenate(Environment &env, const std::vector &tokens, const fdmask &fds) { if (tokens.size() == 1) { int rv = cat_helper(stdin, stdout); if (rv) fputs("### Catenate - I/O Error\n", stderr); return rv; } for (const auto &s : make_offset_range(tokens, 1)) { std::string path = ToolBox::MacToUnix(s); int fd = open(path.c_str(), O_RDONLY); if (fd < 0) { fprintf(stderr, "### Catenate - Unable to open \"%s\".\n", path.c_str()); return 1; } int rv = cat_helper(fd, stdout); close(fd); if (rv) { fputs("### Catenate - I/O Error\n", stderr); return rv; } } return 0; } int builtin_version(Environment &env, const std::vector &tokens, const fdmask &fds) { bool _v = false; bool error = false; auto argv = getopt(tokens, [&](char c){ switch(tolower(c)) { case 'v': _v = true; break; default: fprintf(stderr, "### Version - \"-%c\" is not an option.\n", c); error = true; break; } }); if (argv.size() != 0) { fprintf(stderr, "### Version - Too many parameters were specified.\n"); error = true; } if (error) { fprintf(stderr, "# Usage - Version [-v]\n"); return 1; } //fputs("MPW Shell 3.5, Copyright Apple Computer, Inc. 1985-99. All rights reserved.\n", stdout); fputs("MPW Shell " VERSION ", Copyright Kelvin W Sherlock 2016. All rights reserved.\n", stdout); fputs("based on MPW Shell 3.5, Copyright Apple Computer, Inc. 1985-99. All rights reserved.\n", stdout); if (_v) { fputs("This version built on " __DATE__ " at " __TIME__ ".\n", stdout); } return 0; } int builtin_aboutbox(Environment &env, const std::vector &tokens, const fdmask &fds) { // the most important command of all! if (tokens.size() == 2 && tokens[1] == "--moof") { fputs( "\n" " ## \n" " ## ## #### \n" " ## #### ## \n" " ## ## \n" " ## ## ## ## \n" " ## ## #### \n" " ## ## ## ## \n" " ######## #### ## ## \n" " ## #################### ## \n" " ## ############## ## \n" " #### ############ ## \n" " ###### ###### ## \n" " ###### ## \n" " #### ## \n" " ## ## \n" " ## ################ ## \n" " ## ## ## ## \n" " ## ## ## ## \n" " ## ## ## ## \n" " ## ## ## ## \n" " ## ## ## ## \n" " ###### ###### \n" "\n" ,stdout); return 0; } fprintf(stdout, "+--------------------------------------+\n" "| MPW Shell %-4s |\n" "| |\n" "| |\n" "| (c) 2016 Kelvin W Sherlock |\n" "+--------------------------------------+\n" , VERSION); return 0; }