mpw-shell/command.cpp

457 lines
8.9 KiB
C++
Raw Normal View History

2016-01-30 12:45:19 -05:00
#include "command.h"
#include "phase2-parser.h"
2016-02-01 20:38:29 -05:00
#include "environment.h"
#include "fdset.h"
#include "builtins.h"
#include "mpw-shell.h"
2016-02-05 13:19:20 -05:00
#include "error.h"
2016-02-01 20:38:29 -05:00
2016-01-30 12:45:19 -05:00
#include <stdexcept>
2016-02-01 20:38:29 -05:00
#include <unordered_map>
#include <cctype>
#include <cerrno>
#include <cstdlib>
2016-02-11 15:48:46 -05:00
#include "cxx/filesystem.h"
#include "cxx/string_splitter.h"
2016-02-01 20:38:29 -05:00
#include <unistd.h>
#include <sys/wait.h>
#include <sysexits.h>
2016-02-05 13:19:20 -05:00
extern volatile int control_c;
2016-02-11 15:48:46 -05:00
namespace fs = filesystem;
extern fs::path mpw_path();
namespace ToolBox {
std::string MacToUnix(const std::string path);
std::string UnixToMac(const std::string path);
}
fs::path which(const Environment &env, const std::string &name) {
std::error_code ec;
if (name.find_first_of("/:") != name.npos) {
// canonical?
fs::path p(name);
if (fs::exists(p, ec)) return name;
return "";
}
std::string s = env.get("commands");
for (string_splitter ss(s, ','); ss; ++ss) {
fs::path p(ToolBox::MacToUnix(*ss));
p /= name;
if (fs::exists(p, ec)) return p;
}
// error..
return "";
}
2016-02-05 23:00:42 -05:00
void launch_mpw(const Environment &env, const std::vector<std::string> &argv, const fdmask &fds) {
std::vector<char *> cargv;
cargv.reserve(argv.size() + 3);
cargv.push_back((char *)"mpw");
//cargv.push_back((char *)"--shell");
unsigned offset = cargv.size();
std::transform(argv.begin(), argv.end(), std::back_inserter(cargv),
[](const std::string &s) { return strdup(s.c_str()); }
);
cargv.push_back(nullptr);
// export environment...
for (const auto &kv : env) {
if (kv.second) { // exported
std::string name = "mpw$" + kv.first;
setenv(name.c_str(), kv.second.c_str(), 1);
}
}
// handle any indirection...
fds.dup();
2016-02-11 15:51:56 -05:00
execv(mpw_path().c_str(), cargv.data());
2016-02-05 23:00:42 -05:00
perror("execvp: ");
exit(EX_OSERR); // raise a signal?
}
2016-02-01 20:38:29 -05:00
namespace {
std::string &lowercase(std::string &s) {
std::transform(s.begin(), s.end(), s.begin(), [](char c){ return std::tolower(c); });
return s;
}
std::unordered_map<std::string, int (*)(Environment &, const std::vector<std::string> &, const fdmask &)> builtins = {
{"aboutbox", builtin_aboutbox},
2016-02-01 20:38:29 -05:00
{"directory", builtin_directory},
{"echo", builtin_echo},
2016-02-11 15:51:00 -05:00
{"export", builtin_export},
2016-02-01 20:38:29 -05:00
{"parameters", builtin_parameters},
{"quote", builtin_quote},
{"set", builtin_set},
{"unexport", builtin_unexport},
2016-02-11 15:51:00 -05:00
{"unset", builtin_unset},
{"which", builtin_which},
2016-02-01 20:38:29 -05:00
};
2016-02-05 23:00:42 -05:00
int execute_external(const Environment &env, const std::vector<std::string> &argv, const fdmask &fds) {
2016-02-01 20:38:29 -05:00
2016-02-05 23:00:42 -05:00
int status;
int pid;
2016-02-01 20:38:29 -05:00
pid = fork();
if (pid < 0) {
perror("fork: ");
exit(EX_OSERR);
}
if (pid == 0) {
2016-02-05 23:00:42 -05:00
launch_mpw(env, argv, fds);
2016-02-01 20:38:29 -05:00
}
for(;;) {
int status;
pid_t ok;
ok = waitpid(pid, &status, 0);
if (ok < 0) {
if (errno == EINTR) continue;
perror("waitpid:");
exit(EX_OSERR);
}
if (WIFEXITED(status)) return WEXITSTATUS(status);
if (WIFSIGNALED(status)) return -9; // command-. user abort exits via -9.
fprintf(stderr, "waitpid - unexpected result\n");
exit(EX_OSERR);
}
}
}
//std::string expand_vars(const std::string &s, const class Environment &);
2016-01-30 12:45:19 -05:00
command::~command()
{}
2016-02-01 20:38:29 -05:00
/*
* todo -- all errors should be handled the same, regardless of source. (throw, parse, etc).
* echo and error should respect the active fdmask.
*/
2016-01-30 12:45:19 -05:00
2016-02-02 16:19:13 -05:00
int simple_command::execute(Environment &env, const fdmask &fds, bool throwup) {
2016-02-05 13:19:20 -05:00
if (control_c) throw execution_of_input_terminated();
2016-02-01 20:38:29 -05:00
std::string s = expand_vars(text, env);
process p;
try {
auto tokens = tokenize(s, false);
2016-02-11 15:51:56 -05:00
if (tokens.empty()) return 0;
2016-02-01 20:38:29 -05:00
parse_tokens(std::move(tokens), p);
} catch(std::exception &e) {
2016-02-02 16:19:13 -05:00
fprintf(stderr, "%s\n", e.what());
return env.status(-4, throwup);
2016-02-01 20:38:29 -05:00
}
if (p.arguments.empty()) return 0;
2016-02-11 15:51:56 -05:00
env.echo("%s", s.c_str());
2016-02-01 20:38:29 -05:00
fdmask newfds = p.fds | fds;
std::string name = p.arguments.front();
lowercase(name);
2016-01-30 12:45:19 -05:00
2016-02-01 20:38:29 -05:00
auto iter = builtins.find(name);
if (iter != builtins.end()) {
2016-02-11 15:51:56 -05:00
env.set("command", name);
2016-02-01 20:38:29 -05:00
int status = iter->second(env, p.arguments, newfds);
2016-02-02 16:19:13 -05:00
return env.status(status, throwup);
2016-01-30 12:45:19 -05:00
}
2016-02-01 20:38:29 -05:00
if (env.test()) return env.status(0);
if (env.startup()) {
fprintf(stderr, "### MPW Shell - startup file may not contain external commands.\n");
return env.status(0);
}
2016-02-11 15:51:56 -05:00
name = p.arguments.front();
fs::path path = which(env, name);
if (path.empty()) {
fprintf(stderr, "### MPW Shell - Command \"%s\" was not found.\n", name.c_str());
return env.status(-1, throwup);
}
env.set("command", path);
p.arguments[0] = path;
2016-02-01 20:38:29 -05:00
int status = execute_external(env, p.arguments, newfds);
2016-02-02 16:19:13 -05:00
return env.status(status, throwup);
2016-02-01 20:38:29 -05:00
}
2016-02-02 16:19:13 -05:00
int evaluate_command::execute(Environment &env, const fdmask &fds, bool throwup) {
2016-02-01 20:38:29 -05:00
2016-02-05 13:19:20 -05:00
if (control_c) throw execution_of_input_terminated();
2016-02-01 20:38:29 -05:00
std::string s = expand_vars(text, env);
2016-02-11 15:51:56 -05:00
env.set("command", "evaluate");
2016-02-02 16:19:13 -05:00
env.echo("%s", s.c_str());
2016-02-01 20:38:29 -05:00
2016-02-02 16:19:13 -05:00
try {
auto tokens = tokenize(s, true);
if (tokens.empty()) return 0;
2016-02-01 20:38:29 -05:00
2016-02-02 16:19:13 -05:00
int status = builtin_evaluate(env, std::move(tokens), fds);
2016-02-01 20:38:29 -05:00
2016-02-02 16:19:13 -05:00
return env.status(status, throwup);
} catch (std::exception &e) {
fprintf(stderr, "%s\n", e.what());
return env.status(1, throwup);
}
2016-02-01 20:38:29 -05:00
}
2016-02-02 16:19:13 -05:00
int or_command::execute(Environment &e, const fdmask &fds, bool throwup) {
2016-01-30 12:45:19 -05:00
int rv = 0;
2016-02-01 20:38:29 -05:00
2016-01-30 12:45:19 -05:00
for (auto &c : children) {
2016-02-02 16:19:13 -05:00
if (!c) continue;
rv = c->execute(e, fds, false);
2016-01-30 12:45:19 -05:00
if (rv == 0) return rv;
}
2016-02-01 20:38:29 -05:00
return e.status(rv);
2016-01-30 12:45:19 -05:00
}
2016-02-02 16:19:13 -05:00
int and_command::execute(Environment &e, const fdmask &fds, bool throwup) {
2016-01-30 12:45:19 -05:00
int rv = 0;
for (auto &c : children) {
2016-02-02 16:19:13 -05:00
if (!c) continue;
2016-02-04 21:45:04 -05:00
rv = c->execute(e, fds, false);
2016-01-30 12:45:19 -05:00
if (rv != 0) return rv;
}
2016-02-01 20:38:29 -05:00
2016-01-30 12:45:19 -05:00
return rv;
}
2016-02-02 16:19:13 -05:00
int vector_command::execute(Environment &e, const fdmask &fds, bool throwup) {
2016-01-30 12:45:19 -05:00
int rv = 0;
for (auto &c : children) {
2016-02-04 21:45:04 -05:00
if (!c) continue;
2016-02-01 20:38:29 -05:00
rv = c->execute(e, fds);
}
return e.status(rv);
}
2016-02-02 16:19:13 -05:00
int error_command::execute(Environment &e, const fdmask &fds, bool throwup) {
2016-02-05 13:19:20 -05:00
if (control_c) throw execution_of_input_terminated();
2016-02-01 20:38:29 -05:00
std::string s = expand_vars(text, e);
2016-02-02 16:19:13 -05:00
e.echo("%s", s.c_str());
2016-02-01 20:38:29 -05:00
switch(type) {
case END:
fprintf(stderr, "### MPW Shell - Extra END command.\n");
break;
case RPAREN:
fprintf(stderr, "### MPW Shell - Extra ) command.\n");
break;
case ELSE:
case ELSE_IF:
fprintf(stderr, "### MPW Shell - ELSE must be within IF ... END.\n");
break;
}
return e.status(-3);
}
namespace {
int check_ends(const std::string &s, fdmask &fds) {
// MPW ignores any END tokens other than redirection.
process p;
try {
auto tokens = tokenize(s, false);
parse_tokens(std::move(tokens), p);
}
catch (std::exception &e) {
fprintf(stderr, "%s\n", e.what());
return -4;
}
fds = p.fds.to_mask();
return 0;
2016-01-30 12:45:19 -05:00
}
}
2016-02-02 16:19:13 -05:00
int begin_command::execute(Environment &env, const fdmask &fds, bool throwup) {
2016-01-30 12:45:19 -05:00
// todo -- parse end for redirection.
2016-02-01 20:38:29 -05:00
std::string b = expand_vars(begin, env);
std::string e = expand_vars(end, env);
2016-02-11 15:51:56 -05:00
env.set("command", type == BEGIN ? "end" : ")");
2016-01-31 00:41:16 -05:00
// echo!
2016-02-02 16:19:13 -05:00
env.echo("%s ... %s",
2016-02-01 20:38:29 -05:00
b.c_str(),
e.c_str()
2016-01-31 00:41:16 -05:00
);
2016-01-30 12:45:19 -05:00
2016-02-11 15:51:56 -05:00
2016-02-01 20:38:29 -05:00
// tokenize the begin and end commands.
// begin may not have extra arguments. end may have redirection.
auto bt = tokenize(b, true);
if (bt.size() != 1) {
fprintf(stderr, "### Begin - Too many parameters were specified.\n");
fprintf(stderr, "Usage - Begin\n");
return env.status(-3);
}
fdmask newfds;
int status = check_ends(e, newfds);
newfds |= fds;
if (status) return env.status(status);
2016-02-02 16:19:13 -05:00
int rv;
2016-02-10 21:58:00 -05:00
env.indent_and([&]{
rv = vector_command::execute(env, newfds);
});
2016-02-02 16:19:13 -05:00
env.echo("%s", type == BEGIN ? "end" : ")");
2016-02-01 20:38:29 -05:00
return env.status(rv);
2016-01-30 12:45:19 -05:00
}
2016-02-01 20:38:29 -05:00
namespace {
2016-02-02 21:57:42 -05:00
/*
* returns:
* <0 -> error
* 0 -> false
* >0 -> true
*/
2016-02-01 20:38:29 -05:00
2016-02-02 16:19:13 -05:00
int evaluate(int type, const std::string &s, Environment &env) {
2016-02-01 20:38:29 -05:00
auto tokens = tokenize(s, true);
std::reverse(tokens.begin(), tokens.end());
int32_t e;
switch(type) {
default: return 0;
case ELSE_IF:
tokens.pop_back();
case IF:
tokens.pop_back();
try {
e = evaluate_expression("If", std::move(tokens));
}
catch (std::exception &ex) {
fprintf(stderr, "%s\n", ex.what());
2016-02-02 16:19:13 -05:00
return -5;
2016-02-01 20:38:29 -05:00
}
break;
case ELSE:
e = 1;
if (tokens.size() > 1) {
fprintf(stderr, "### Else - Missing if keyword.\n");
fprintf(stderr, "# Usage - Else [if expression...]\n");
2016-02-02 16:19:13 -05:00
return -3;
2016-02-01 20:38:29 -05:00
}
}
2016-02-02 16:19:13 -05:00
return !!e;
2016-02-01 20:38:29 -05:00
}
}
2016-02-02 16:19:13 -05:00
/*
* the entire command prints even if there is an error with the expression.
*
*/
int if_command::execute(Environment &env, const fdmask &fds, bool throwup) {
2016-01-30 12:45:19 -05:00
int rv = 0;
bool ok = false;
2016-01-31 00:41:16 -05:00
2016-02-01 20:38:29 -05:00
std::string e = expand_vars(end, env);
2016-02-11 15:51:56 -05:00
env.set("command", "end");
2016-02-01 20:38:29 -05:00
// parse end for indirection.
fdmask newfds;
int status = check_ends(e, newfds);
newfds |= fds;
2016-02-02 16:19:13 -05:00
if (status) {
rv = status;
ok = true;
}
2016-01-31 00:41:16 -05:00
for (auto &c : clauses) {
2016-02-01 20:38:29 -05:00
std::string s = expand_vars(c->clause, env);
2016-02-02 16:19:13 -05:00
if (c->type == IF)
env.echo("%s ... %s", s.c_str(), e.c_str());
else
env.echo("%s", s.c_str());
2016-01-31 00:41:16 -05:00
2016-01-30 12:45:19 -05:00
if (ok) continue;
2016-02-10 21:58:00 -05:00
int tmp = evaluate(c->type, s, env);
if (tmp < 0) { ok = true; rv = tmp; continue; }
if (tmp == 0) continue;
env.indent_and([&](){
2016-02-02 21:57:42 -05:00
rv = c->execute(env, newfds);
2016-02-10 21:58:00 -05:00
});
2016-01-30 12:45:19 -05:00
}
2016-01-31 00:41:16 -05:00
2016-02-02 16:19:13 -05:00
env.echo("end");
2016-02-01 20:38:29 -05:00
return env.status(rv);
2016-01-30 12:45:19 -05:00
}