diff options
author | Ben Noordhuis <info@bnoordhuis.nl> | 2013-09-02 16:42:01 +0200 |
---|---|---|
committer | Ben Noordhuis <info@bnoordhuis.nl> | 2013-09-02 21:41:12 +0200 |
commit | 185c515c9febf2229ed2ac76bfdd0c767ea7fd43 (patch) | |
tree | 798f110afacb802ee60e577344e7673543f1650e /src/node.cc | |
parent | d6fb764cc2b77e439e842ae26811b34ff3f50235 (diff) | |
download | node-new-185c515c9febf2229ed2ac76bfdd0c767ea7fd43.tar.gz |
src: clean up CLI argument parser
* Exit with an error message when the option is not a node or V8 option.
* Remove the option_end_index global. Needs to happen anyway for
the multi-context work, might as well land it in master now.
* Add a smidgen of const-correctness.
* Pay off a few years of accrued technical debt.
Diffstat (limited to 'src/node.cc')
-rw-r--r-- | src/node.cc | 245 |
1 files changed, 133 insertions, 112 deletions
diff --git a/src/node.cc b/src/node.cc index 18f23a5874..c26c1d2384 100644 --- a/src/node.cc +++ b/src/node.cc @@ -159,8 +159,7 @@ static bool print_eval = false; static bool force_repl = false; static bool trace_deprecation = false; static bool throw_deprecation = false; -static char *eval_string = NULL; -static int option_end_index = 0; +static const char* eval_string = NULL; static bool use_debug_agent = false; static bool debug_wait_connect = false; static int debug_port = 5858; @@ -2285,7 +2284,10 @@ static void NeedImmediateCallbackSetter(Local<String> property, } while (0) -Handle<Object> SetupProcessObject(int argc, char *argv[]) { +Handle<Object> SetupProcessObject(int argc, + const char* const* argv, + int exec_argc, + const char* const* exec_argv) { HandleScope scope(node_isolate); int i, j; @@ -2374,19 +2376,18 @@ Handle<Object> SetupProcessObject(int argc, char *argv[]) { OneByteString(node_isolate, PLATFORM)); // process.argv - Local<Array> arguments = Array::New(argc - option_end_index + 1); - arguments->Set(0, String::NewFromUtf8(node_isolate, argv[0])); - for (j = 1, i = option_end_index; i < argc; j++, i++) { - arguments->Set(j, String::NewFromUtf8(node_isolate, argv[i])); + Local<Array> arguments = Array::New(argc); + for (int i = 0; i < argc; ++i) { + arguments->Set(i, String::NewFromUtf8(node_isolate, argv[i])); } process->Set(FIXED_ONE_BYTE_STRING(node_isolate, "argv"), arguments); // process.execArgv - Local<Array> exec_argv = Array::New(option_end_index - 1); - for (j = 1, i = 0; j < option_end_index; j++, i++) { - exec_argv->Set(i, String::NewFromUtf8(node_isolate, argv[j])); + Local<Array> exec_arguments = Array::New(exec_argc); + for (int i = 0; i < exec_argc; ++i) { + exec_arguments->Set(i, String::NewFromUtf8(node_isolate, exec_argv[i])); } - process->Set(FIXED_ONE_BYTE_STRING(node_isolate, "execArgv"), exec_argv); + process->Set(FIXED_ONE_BYTE_STRING(node_isolate, "execArgv"), exec_arguments); // create process.env Local<ObjectTemplate> envTemplate = ObjectTemplate::New(); @@ -2648,72 +2649,117 @@ static void PrintHelp() { } -// Parse node command line arguments. -static void ParseArgs(int argc, char **argv) { - int i; +// Parse command line arguments. +// +// argv is modified in place. exec_argv and v8_argv are out arguments that +// ParseArgs() allocates memory for and stores a pointer to the output +// vector in. The caller should free them with delete[]. +// +// On exit: +// +// * argv contains the arguments with node and V8 options filtered out. +// * exec_argv contains both node and V8 options and nothing else. +// * v8_argv contains argv[0] plus any V8 options +static void ParseArgs(int* argc, + const char** argv, + int* exec_argc, + const char*** exec_argv, + int* v8_argc, + const char*** v8_argv) { + const unsigned int nargs = static_cast<unsigned int>(*argc); + const char** new_exec_argv = new const char*[nargs]; + const char** new_v8_argv = new const char*[nargs]; + const char** new_argv = new const char*[nargs]; + + for (unsigned int i = 0; i < nargs; ++i) { + new_exec_argv[i] = NULL; + new_v8_argv[i] = NULL; + new_argv[i] = NULL; + } + + // exec_argv starts with the first option, the other two start with argv[0]. + unsigned int new_exec_argc = 0; + unsigned int new_v8_argc = 1; + unsigned int new_argc = 1; + new_v8_argv[0] = argv[0]; + new_argv[0] = argv[0]; + + unsigned int index = 1; + while (index < nargs && argv[index][0] == '-') { + const char* const arg = argv[index]; + unsigned int args_consumed = 1; - // TODO(bnoordhuis) use parse opts - for (i = 1; i < argc; i++) { - const char *arg = argv[i]; if (strstr(arg, "--debug") == arg) { ParseDebugOpt(arg); - argv[i] = const_cast<char*>(""); } else if (strcmp(arg, "--version") == 0 || strcmp(arg, "-v") == 0) { printf("%s\n", NODE_VERSION); exit(0); } else if (strcmp(arg, "--help") == 0 || strcmp(arg, "-h") == 0) { PrintHelp(); exit(0); - } else if (strcmp(arg, "--eval") == 0 || - strcmp(arg, "-e") == 0 || - strcmp(arg, "--print") == 0 || - strcmp(arg, "-pe") == 0 || + } else if (strcmp(arg, "--eval") == 0 || + strcmp(arg, "-e") == 0 || + strcmp(arg, "--print") == 0 || + strcmp(arg, "-pe") == 0 || strcmp(arg, "-p") == 0) { bool is_eval = strchr(arg, 'e') != NULL; bool is_print = strchr(arg, 'p') != NULL; - - // argument to -p and --print is optional - if (is_eval == true && i + 1 >= argc) { - fprintf(stderr, "Error: %s requires an argument\n", arg); - exit(13); - } - print_eval = print_eval || is_print; - argv[i] = const_cast<char*>(""); - - // --eval, -e and -pe always require an argument + // --eval, -e and -pe always require an argument. if (is_eval == true) { - eval_string = argv[++i]; - continue; + args_consumed += 1; + eval_string = argv[index + 1]; + if (eval_string == NULL) { + fprintf(stderr, "%s: %s requires an argument\n", argv[0], arg); + exit(1); + } + } else if (argv[index + 1] != NULL && argv[index + 1][0] != '-') { + args_consumed += 1; + eval_string = argv[index + 1]; + if (strncmp(eval_string, "\\-", 2) == 0) { + // Starts with "\\-": escaped expression, drop the backslash. + eval_string += 1; + } } - - // next arg is the expression to evaluate unless it starts with: - // - a dash, then it's another switch - // - "\\-", then it's an escaped expression, drop the backslash - if (argv[i + 1] == NULL) continue; - if (argv[i + 1][0] == '-') continue; - eval_string = argv[++i]; - if (strncmp(eval_string, "\\-", 2) == 0) ++eval_string; } else if (strcmp(arg, "--interactive") == 0 || strcmp(arg, "-i") == 0) { force_repl = true; - argv[i] = const_cast<char*>(""); - } else if (strcmp(arg, "--v8-options") == 0) { - argv[i] = const_cast<char*>("--help"); } else if (strcmp(arg, "--no-deprecation") == 0) { - argv[i] = const_cast<char*>(""); no_deprecation = true; } else if (strcmp(arg, "--trace-deprecation") == 0) { - argv[i] = const_cast<char*>(""); trace_deprecation = true; } else if (strcmp(arg, "--throw-deprecation") == 0) { - argv[i] = const_cast<char*>(""); throw_deprecation = true; - } else if (argv[i][0] != '-') { - break; + } else if (strcmp(arg, "--v8-options") == 0) { + new_v8_argv[new_v8_argc] = "--help"; + new_v8_argc += 1; + } else { + // V8 option. Pass through as-is. + new_v8_argv[new_v8_argc] = arg; + new_v8_argc += 1; } + + memcpy(new_exec_argv + new_exec_argc, + argv + index, + args_consumed * sizeof(*argv)); + + new_exec_argc += args_consumed; + index += args_consumed; } - option_end_index = i; + // Copy remaining arguments. + const unsigned int args_left = nargs - index; + memcpy(new_argv + new_argc, argv + index, args_left * sizeof(*argv)); + new_argc += args_left; + + *exec_argc = new_exec_argc; + *exec_argv = new_exec_argv; + *v8_argc = new_v8_argc; + *v8_argv = new_v8_argv; + + // Copy new_argv over argv and update argc. + memcpy(argv, new_argv, new_argc * sizeof(*argv)); + delete[] new_argv; + *argc = static_cast<int>(new_argc); } @@ -2981,7 +3027,10 @@ static void DebugEnd(const FunctionCallbackInfo<Value>& args) { } -char** Init(int argc, char *argv[]) { +void Init(int* argc, + const char** argv, + int* exec_argc, + const char*** exec_argv) { // Initialize prog_start_time to get relative uptime. uv_uptime(&prog_start_time); @@ -3001,24 +3050,29 @@ char** Init(int argc, char *argv[]) { uv_unref(reinterpret_cast<uv_handle_t*>(&emit_debug_enabled_async)); // Parse a few arguments which are specific to Node. - node::ParseArgs(argc, argv); - // Parse the rest of the args (up to the 'option_end_index' (where '--' was - // in the command line)) - int v8argc = option_end_index; - char **v8argv = argv; + int v8_argc; + const char** v8_argv; + ParseArgs(argc, argv, exec_argc, exec_argv, &v8_argc, &v8_argv); - if (debug_wait_connect) { - // v8argv is a copy of argv up to the script file argument +2 if --debug-brk - // to expose the v8 debugger js object so that node.js can set - // a breakpoint on the first line of the startup script - v8argc += 2; - v8argv = new char*[v8argc]; - memcpy(v8argv, argv, sizeof(*argv) * option_end_index); - v8argv[option_end_index] = const_cast<char*>("--expose_debug_as"); - v8argv[option_end_index + 1] = const_cast<char*>("v8debug"); + // The const_cast doesn't violate conceptual const-ness. V8 doesn't modify + // the argv array or the elements it points to. + V8::SetFlagsFromCommandLine(&v8_argc, const_cast<char**>(v8_argv), true); + + // Anything that's still in v8_argv is not a V8 or a node option. + for (int i = 1; i < v8_argc; i++) { + fprintf(stderr, "%s: bad option: %s\n", argv[0], v8_argv[i]); } + delete[] v8_argv; + v8_argv = NULL; - V8::SetFlagsFromCommandLine(&v8argc, v8argv, false); + if (v8_argc > 1) { + exit(1); + } + + if (debug_wait_connect) { + const char expose_debug_as[] = "--expose_debug_as=v8debug"; + V8::SetFlagsFromString(expose_debug_as, sizeof(expose_debug_as) - 1); + } const char typed_arrays_flag[] = "--harmony_typed_arrays"; V8::SetFlagsFromString(typed_arrays_flag, sizeof(typed_arrays_flag) - 1); @@ -3055,8 +3109,6 @@ char** Init(int argc, char *argv[]) { uv_unref(reinterpret_cast<uv_handle_t*>(&signal_watcher)); #endif // __POSIX__ } - - return argv; } @@ -3102,50 +3154,18 @@ void EmitExit(v8::Handle<v8::Object> process_l) { MakeCallback(process_l, "emit", ARRAY_SIZE(args), args); } -static char **copy_argv(int argc, char **argv) { - size_t strlen_sum; - char **argv_copy; - char *argv_data; - size_t len; - int i; - strlen_sum = 0; - for (i = 0; i < argc; i++) { - strlen_sum += strlen(argv[i]) + 1; - } +int Start(int argc, char** argv) { + assert(argc > 0); - argv_copy = static_cast<char**>( - malloc(sizeof(*argv_copy) * (argc + 1) + strlen_sum)); - if (!argv_copy) { - return NULL; - } - - argv_data = reinterpret_cast<char*>(argv_copy) + - sizeof(*argv_copy) * (argc + 1); - - for (i = 0; i < argc; i++) { - argv_copy[i] = argv_data; - len = strlen(argv[i]) + 1; - memcpy(argv_data, argv[i], len); - argv_data += len; - } - - argv_copy[argc] = NULL; - - return argv_copy; -} - -int Start(int argc, char *argv[]) { - // Hack aroung with the argv pointer. Used for process.title = "blah". + // Hack around with the argv pointer. Used for process.title = "blah". argv = uv_setup_args(argc, argv); - // Logic to duplicate argv as Init() modifies arguments - // that are passed into it. - char **argv_copy = copy_argv(argc, argv); - - // This needs to run *before* V8::Initialize() - // Use copy here as to not modify the original argv: - Init(argc, argv_copy); + // This needs to run *before* V8::Initialize(). The const_cast is not + // optional, in case you're wondering. + int exec_argc; + const char** exec_argv; + Init(&argc, const_cast<const char**>(argv), &exec_argc, &exec_argv); V8::Initialize(); { @@ -3159,7 +3179,8 @@ int Start(int argc, char *argv[]) { binding_cache.Reset(node_isolate, Object::New()); // Use original argv, as we're just copying values out of it. - Local<Object> process_l = SetupProcessObject(argc, argv); + Local<Object> process_l = + SetupProcessObject(argc, argv, exec_argc, exec_argv); // Create all the objects, load modules, do everything. // so your next reading stop should be node::Load()! @@ -3181,8 +3202,8 @@ int Start(int argc, char *argv[]) { V8::Dispose(); #endif // NDEBUG - // Clean up the copy: - free(argv_copy); + delete[] exec_argv; + exec_argv = NULL; return 0; } |