/** * WinPR: Windows Portable Runtime * Command-Line Utils * * Copyright 2012 Marc-Andre Moreau * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #ifdef HAVE_CONFIG_H #include "config.h" #endif #include #include /** * Command-line syntax: some basic concepts: * https://pythonconquerstheuniverse.wordpress.com/2010/07/25/command-line-syntax-some-basic-concepts/ */ /** * Command-Line Syntax: * * * * : '/' or '-' or ('+' | '-') * * : option, named argument, flag * * : ':' or '=' * * : argument value * */ int CommandLineParseArgumentsA(int argc, LPCSTR* argv, COMMAND_LINE_ARGUMENT_A* options, DWORD flags, void* context, COMMAND_LINE_PRE_FILTER_FN_A preFilter, COMMAND_LINE_POST_FILTER_FN_A postFilter) { int i, j; int status; int count; size_t length; BOOL notescaped; const char* sigil; size_t sigil_length; const char* keyword; SSIZE_T keyword_length; SSIZE_T keyword_index; char* separator; const char* value; int toggle; status = 0; notescaped = FALSE; if (!argv) return status; if (argc == 1) { if (flags & COMMAND_LINE_IGN_UNKNOWN_KEYWORD) status = 0; else status = COMMAND_LINE_STATUS_PRINT_HELP; return status; } for (i = 1; i < argc; i++) { BOOL found = FALSE; BOOL escaped = TRUE; if (preFilter) { count = preFilter(context, i, argc, argv); if (count < 0) { status = COMMAND_LINE_ERROR; return status; } if (count > 0) { i += (count - 1); continue; } } sigil = argv[i]; length = strlen(argv[i]); if ((sigil[0] == '/') && (flags & COMMAND_LINE_SIGIL_SLASH)) { sigil_length = 1; } else if ((sigil[0] == '-') && (flags & COMMAND_LINE_SIGIL_DASH)) { sigil_length = 1; if (length > 2) { if ((sigil[1] == '-') && (flags & COMMAND_LINE_SIGIL_DOUBLE_DASH)) sigil_length = 2; } } else if ((sigil[0] == '+') && (flags & COMMAND_LINE_SIGIL_PLUS_MINUS)) { sigil_length = 1; } else if ((sigil[0] == '-') && (flags & COMMAND_LINE_SIGIL_PLUS_MINUS)) { sigil_length = 1; } else if (flags & COMMAND_LINE_SIGIL_NONE) { sigil_length = 0; } else if (flags & COMMAND_LINE_SIGIL_NOT_ESCAPED) { if (notescaped) return COMMAND_LINE_ERROR; sigil_length = 0; escaped = FALSE; notescaped = TRUE; } else { return COMMAND_LINE_ERROR; } if ((sigil_length > 0) || (flags & COMMAND_LINE_SIGIL_NONE) || (flags & COMMAND_LINE_SIGIL_NOT_ESCAPED)) { if (length < (sigil_length + 1)) { if ((flags & COMMAND_LINE_IGN_UNKNOWN_KEYWORD)) continue; return COMMAND_LINE_ERROR_NO_KEYWORD; } keyword_index = sigil_length; keyword = &argv[i][keyword_index]; toggle = -1; if (flags & COMMAND_LINE_SIGIL_ENABLE_DISABLE) { if (strncmp(keyword, "enable-", 7) == 0) { toggle = TRUE; keyword_index += 7; keyword = &argv[i][keyword_index]; } else if (strncmp(keyword, "disable-", 8) == 0) { toggle = FALSE; keyword_index += 8; keyword = &argv[i][keyword_index]; } } separator = NULL; if ((flags & COMMAND_LINE_SEPARATOR_COLON) && (!separator)) separator = strchr(keyword, ':'); if ((flags & COMMAND_LINE_SEPARATOR_EQUAL) && (!separator)) separator = strchr(keyword, '='); if (separator) { SSIZE_T separator_index = (separator - argv[i]); SSIZE_T value_index = separator_index + 1; keyword_length = (separator - keyword); value = &argv[i][value_index]; } else { keyword_length = (length - keyword_index); value = NULL; } if (!escaped) continue; for (j = 0; options[j].Name != NULL; j++) { BOOL match = FALSE; if (strncmp(options[j].Name, keyword, keyword_length) == 0) { if (strlen(options[j].Name) == keyword_length) match = TRUE; } if ((!match) && (options[j].Alias != NULL)) { if (strncmp(options[j].Alias, keyword, keyword_length) == 0) { if (strlen(options[j].Alias) == keyword_length) match = TRUE; } } if (!match) continue; found = match; options[j].Index = i; if ((flags & COMMAND_LINE_SEPARATOR_SPACE) && ((i + 1) < argc)) { BOOL argument; int value_present = 1; if (flags & COMMAND_LINE_SIGIL_DASH) { if (strncmp(argv[i + 1], "-", 1) == 0) value_present = 0; } if (flags & COMMAND_LINE_SIGIL_DOUBLE_DASH) { if (strncmp(argv[i + 1], "--", 2) == 0) value_present = 0; } if (flags & COMMAND_LINE_SIGIL_SLASH) { if (strncmp(argv[i + 1], "/", 1) == 0) value_present = 0; } if ((options[j].Flags & COMMAND_LINE_VALUE_REQUIRED) || (options[j].Flags & COMMAND_LINE_VALUE_OPTIONAL)) argument = TRUE; else argument = FALSE; if (value_present && argument) { i++; value = argv[i]; } else if (!value_present && (options[j].Flags & COMMAND_LINE_VALUE_OPTIONAL)) { value = NULL; } else if (!value_present && argument) return COMMAND_LINE_ERROR; } if (!(flags & COMMAND_LINE_SEPARATOR_SPACE)) { if (value && (options[j].Flags & COMMAND_LINE_VALUE_FLAG)) return COMMAND_LINE_ERROR_UNEXPECTED_VALUE; } else { if (value && (options[j].Flags & COMMAND_LINE_VALUE_FLAG)) { i--; value = NULL; } } if (!value && (options[j].Flags & COMMAND_LINE_VALUE_REQUIRED)) { status = COMMAND_LINE_ERROR_MISSING_VALUE; return status; } options[j].Flags |= COMMAND_LINE_ARGUMENT_PRESENT; if (value) { options[j].Value = value; options[j].Flags |= COMMAND_LINE_VALUE_PRESENT; } else { if (options[j].Flags & COMMAND_LINE_VALUE_FLAG) { options[j].Value = (LPSTR) 1; options[j].Flags |= COMMAND_LINE_VALUE_PRESENT; } else if (options[j].Flags & COMMAND_LINE_VALUE_BOOL) { if (flags & COMMAND_LINE_SIGIL_ENABLE_DISABLE) { if (toggle == -1) options[j].Value = BoolValueTrue; else if (!toggle) options[j].Value = BoolValueFalse; else options[j].Value = BoolValueTrue; } else { if (sigil[0] == '+') options[j].Value = BoolValueTrue; else if (sigil[0] == '-') options[j].Value = BoolValueFalse; else options[j].Value = BoolValueTrue; } options[j].Flags |= COMMAND_LINE_VALUE_PRESENT; } } if (postFilter) { count = postFilter(context, &options[j]); if (count < 0) { status = COMMAND_LINE_ERROR; return status; } } if (options[j].Flags & COMMAND_LINE_PRINT) return COMMAND_LINE_STATUS_PRINT; else if (options[j].Flags & COMMAND_LINE_PRINT_HELP) return COMMAND_LINE_STATUS_PRINT_HELP; else if (options[j].Flags & COMMAND_LINE_PRINT_VERSION) return COMMAND_LINE_STATUS_PRINT_VERSION; else if (options[j].Flags & COMMAND_LINE_PRINT_BUILDCONFIG) return COMMAND_LINE_STATUS_PRINT_BUILDCONFIG; } if (!found && (flags & COMMAND_LINE_IGN_UNKNOWN_KEYWORD) == 0) return COMMAND_LINE_ERROR_NO_KEYWORD; } } return status; } int CommandLineParseArgumentsW(int argc, LPCWSTR* argv, COMMAND_LINE_ARGUMENT_W* options, DWORD flags, void* context, COMMAND_LINE_PRE_FILTER_FN_W preFilter, COMMAND_LINE_POST_FILTER_FN_W postFilter) { return 0; } int CommandLineClearArgumentsA(COMMAND_LINE_ARGUMENT_A* options) { size_t i; for (i = 0; options[i].Name != NULL; i++) { options[i].Flags &= COMMAND_LINE_INPUT_FLAG_MASK; options[i].Value = NULL; } return 0; } int CommandLineClearArgumentsW(COMMAND_LINE_ARGUMENT_W* options) { int i; for (i = 0; options[i].Name != NULL; i++) { options[i].Flags &= COMMAND_LINE_INPUT_FLAG_MASK; options[i].Value = NULL; } return 0; } COMMAND_LINE_ARGUMENT_A* CommandLineFindArgumentA(COMMAND_LINE_ARGUMENT_A* options, LPCSTR Name) { int i; for (i = 0; options[i].Name != NULL; i++) { if (strcmp(options[i].Name, Name) == 0) return &options[i]; if (options[i].Alias != NULL) { if (strcmp(options[i].Alias, Name) == 0) return &options[i]; } } return NULL; } COMMAND_LINE_ARGUMENT_W* CommandLineFindArgumentW(COMMAND_LINE_ARGUMENT_W* options, LPCWSTR Name) { int i; for (i = 0; options[i].Name != NULL; i++) { if (_wcscmp(options[i].Name, Name) == 0) return &options[i]; if (options[i].Alias != NULL) { if (_wcscmp(options[i].Alias, Name) == 0) return &options[i]; } } return NULL; } COMMAND_LINE_ARGUMENT_A* CommandLineFindNextArgumentA(COMMAND_LINE_ARGUMENT_A* argument) { COMMAND_LINE_ARGUMENT_A* nextArgument; nextArgument = &argument[1]; if (nextArgument->Name == NULL) return NULL; return nextArgument; }