mirror of
https://github.com/LadybirdBrowser/ladybird.git
synced 2025-01-24 10:12:25 -05:00
10e8b99038
A POSIX-compatibility fix was introduced in 64740a0214
to make the
compilation of the `diffutils` port work, which expected a
`char* const* argv` signature.
And indeed, the POSIX spec does not mention permutation of `argv`:
https://pubs.opengroup.org/onlinepubs/9699919799/functions/getopt.html
However, most implementations do modify `argv` as evidenced by
documentation such as:
https://refspecs.linuxbase.org/LSB_5.0.0/LSB-Core-generic
/LSB-Core-generic/libutil-getopt-3.html
"The function prototype was aligned with POSIX 1003.1-2008 (ISO/IEC
9945-2009) despite the fact that it modifies argv, and the library
maintainers are unwilling to change this."
Change the behavior back to permutate `argc` to allow for the following
command line argument order to work again:
unzip ./file.zip -o target-dir
Without this change, `./file.zip` in the example above would have been
ignored completely.
390 lines
14 KiB
C++
390 lines
14 KiB
C++
/*
|
|
* Copyright (c) 2021, the SerenityOS developers.
|
|
*
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
*/
|
|
|
|
#include <AK/Function.h>
|
|
#include <AK/Vector.h>
|
|
#include <LibCore/ArgsParser.h>
|
|
#include <LibTest/TestCase.h>
|
|
#include <string.h>
|
|
|
|
static char** build_argv(Vector<String> arguments)
|
|
{
|
|
auto argv = new char*[arguments.size() + 1];
|
|
size_t idx = 0;
|
|
for (auto& argument : arguments) {
|
|
auto char_argument = new char[argument.length() + 1];
|
|
memcpy(char_argument, argument.characters(), argument.length());
|
|
char_argument[argument.length()] = '\0';
|
|
argv[idx++] = char_argument;
|
|
}
|
|
argv[idx++] = nullptr;
|
|
return argv;
|
|
}
|
|
|
|
static void delete_argv(char** argv, size_t number_of_arguments)
|
|
{
|
|
for (size_t idx = 0; idx < number_of_arguments; ++idx)
|
|
delete[] argv[idx];
|
|
delete[] argv;
|
|
}
|
|
|
|
static bool run_parser(Vector<String> arguments, Function<void(Core::ArgsParser&)> parser_initialization = {})
|
|
{
|
|
Core::ArgsParser parser;
|
|
if (parser_initialization)
|
|
parser_initialization(parser);
|
|
|
|
auto argv = build_argv(arguments);
|
|
auto result = parser.parse(arguments.size(), argv, Core::ArgsParser::FailureBehavior::Ignore);
|
|
delete_argv(argv, arguments.size());
|
|
|
|
return result;
|
|
}
|
|
|
|
TEST_CASE(no_arguments)
|
|
{
|
|
auto parser_result = run_parser({ "app" });
|
|
EXPECT_EQ(parser_result, true);
|
|
}
|
|
|
|
TEST_CASE(bool_option)
|
|
{
|
|
// Short option
|
|
bool force = false;
|
|
auto parser_result = run_parser({ "app", "-f" }, [&](auto& parser) {
|
|
parser.add_option(force, "force", nullptr, 'f');
|
|
});
|
|
EXPECT_EQ(parser_result, true);
|
|
EXPECT_EQ(force, true);
|
|
|
|
// Short option, not given
|
|
force = false;
|
|
parser_result = run_parser({ "app" }, [&](auto& parser) {
|
|
parser.add_option(force, "force", nullptr, 'f');
|
|
});
|
|
EXPECT_EQ(parser_result, true);
|
|
EXPECT_EQ(force, false);
|
|
|
|
// Long option
|
|
force = false;
|
|
parser_result = run_parser({ "app", "--force" }, [&](auto& parser) {
|
|
parser.add_option(force, "force", "force", '\0');
|
|
});
|
|
EXPECT_EQ(parser_result, true);
|
|
EXPECT_EQ(force, true);
|
|
|
|
// Long option, not given
|
|
force = false;
|
|
parser_result = run_parser({ "app" }, [&](auto& parser) {
|
|
parser.add_option(force, "force", "force", '\0');
|
|
});
|
|
EXPECT_EQ(parser_result, true);
|
|
EXPECT_EQ(force, false);
|
|
|
|
// Allow both short and long option, provide short
|
|
force = false;
|
|
parser_result = run_parser({ "app", "-f" }, [&](auto& parser) {
|
|
parser.add_option(force, "force", "force", 'f');
|
|
});
|
|
EXPECT_EQ(parser_result, true);
|
|
EXPECT_EQ(force, true);
|
|
|
|
// Allow both short and long option, provide long
|
|
force = false;
|
|
parser_result = run_parser({ "app", "--force" }, [&](auto& parser) {
|
|
parser.add_option(force, "force", "force", 'f');
|
|
});
|
|
EXPECT_EQ(parser_result, true);
|
|
EXPECT_EQ(force, true);
|
|
|
|
// Allow both short and long option, provide both
|
|
force = false;
|
|
parser_result = run_parser({ "app", "--force", "-f" }, [&](auto& parser) {
|
|
parser.add_option(force, "force", "force", 'f');
|
|
});
|
|
EXPECT_EQ(parser_result, true);
|
|
EXPECT_EQ(force, true);
|
|
}
|
|
|
|
TEST_CASE(positional_string_argument)
|
|
{
|
|
// Single required string argument
|
|
String name = "";
|
|
auto parser_result = run_parser({ "app", "buggie" }, [&](auto& parser) {
|
|
parser.add_positional_argument(name, "name", "name", Core::ArgsParser::Required::Yes);
|
|
});
|
|
EXPECT_EQ(parser_result, true);
|
|
EXPECT_EQ(name, "buggie");
|
|
|
|
// Single required string argument, not given
|
|
name = "";
|
|
parser_result = run_parser({ "app" }, [&](auto& parser) {
|
|
parser.add_positional_argument(name, "name", "name", Core::ArgsParser::Required::Yes);
|
|
});
|
|
EXPECT_EQ(parser_result, false);
|
|
EXPECT_EQ(name, "");
|
|
|
|
// Single optional string argument
|
|
name = "";
|
|
parser_result = run_parser({ "app", "buggie" }, [&](auto& parser) {
|
|
parser.add_positional_argument(name, "name", "name", Core::ArgsParser::Required::No);
|
|
});
|
|
EXPECT_EQ(parser_result, true);
|
|
EXPECT_EQ(name, "buggie");
|
|
|
|
// Single optional string argument, not given
|
|
name = "";
|
|
parser_result = run_parser({ "app" }, [&](auto& parser) {
|
|
parser.add_positional_argument(name, "name", "name", Core::ArgsParser::Required::No);
|
|
});
|
|
EXPECT_EQ(parser_result, true);
|
|
EXPECT_EQ(name, "");
|
|
}
|
|
|
|
TEST_CASE(positional_vector_string_argument)
|
|
{
|
|
// Zero or more positional arguments, zero given
|
|
Vector<String> values = {};
|
|
auto parser_result = run_parser({ "app" }, [&](auto& parser) {
|
|
parser.add_positional_argument(values, "values", "values", Core::ArgsParser::Required::No);
|
|
});
|
|
EXPECT_EQ(parser_result, true);
|
|
EXPECT_EQ(values.size(), 0u);
|
|
|
|
// Zero or more positional arguments, one given
|
|
values = {};
|
|
parser_result = run_parser({ "app", "one" }, [&](auto& parser) {
|
|
parser.add_positional_argument(values, "values", "values", Core::ArgsParser::Required::No);
|
|
});
|
|
EXPECT_EQ(parser_result, true);
|
|
EXPECT_EQ(values.size(), 1u);
|
|
if (values.size() == 1u)
|
|
EXPECT_EQ(values[0], "one");
|
|
|
|
// Zero or more positional arguments, two given
|
|
values = {};
|
|
parser_result = run_parser({ "app", "one", "two" }, [&](auto& parser) {
|
|
parser.add_positional_argument(values, "values", "values", Core::ArgsParser::Required::No);
|
|
});
|
|
EXPECT_EQ(parser_result, true);
|
|
EXPECT_EQ(values.size(), 2u);
|
|
if (values.size() == 2u) {
|
|
EXPECT_EQ(values[0], "one");
|
|
EXPECT_EQ(values[1], "two");
|
|
}
|
|
|
|
// One or more positional arguments, zero given
|
|
values = {};
|
|
parser_result = run_parser({ "app" }, [&](auto& parser) {
|
|
parser.add_positional_argument(values, "values", "values", Core::ArgsParser::Required::Yes);
|
|
});
|
|
EXPECT_EQ(parser_result, false);
|
|
EXPECT_EQ(values.size(), 0u);
|
|
|
|
// One or more positional arguments, one given
|
|
values = {};
|
|
parser_result = run_parser({ "app", "one" }, [&](auto& parser) {
|
|
parser.add_positional_argument(values, "values", "values", Core::ArgsParser::Required::Yes);
|
|
});
|
|
EXPECT_EQ(parser_result, true);
|
|
EXPECT_EQ(values.size(), 1u);
|
|
if (values.size() == 1u)
|
|
EXPECT_EQ(values[0], "one");
|
|
|
|
// One or more positional arguments, two given
|
|
values = {};
|
|
parser_result = run_parser({ "app", "one", "two" }, [&](auto& parser) {
|
|
parser.add_positional_argument(values, "values", "values", Core::ArgsParser::Required::Yes);
|
|
});
|
|
EXPECT_EQ(parser_result, true);
|
|
EXPECT_EQ(values.size(), 2u);
|
|
if (values.size() == 2u) {
|
|
EXPECT_EQ(values[0], "one");
|
|
EXPECT_EQ(values[1], "two");
|
|
}
|
|
}
|
|
|
|
TEST_CASE(combination_of_bool_options_with_positional_vector_string)
|
|
{
|
|
// Bool options (given) and positional arguments (given)
|
|
// Expected: all arguments fill as given
|
|
bool bool_opt1 = false;
|
|
bool bool_opt2 = false;
|
|
Vector<String> positionals = {};
|
|
auto parser_result = run_parser({ "app", "-b", "-c", "one", "two" }, [&](auto& parser) {
|
|
parser.add_option(bool_opt1, "bool_opt1", nullptr, 'b');
|
|
parser.add_option(bool_opt2, "bool_opt2", nullptr, 'c');
|
|
parser.add_positional_argument(positionals, "pos", "pos", Core::ArgsParser::Required::No);
|
|
});
|
|
EXPECT_EQ(parser_result, true);
|
|
EXPECT_EQ(bool_opt1, true);
|
|
EXPECT_EQ(bool_opt2, true);
|
|
EXPECT_EQ(positionals.size(), 2u);
|
|
if (positionals.size() == 2u) {
|
|
EXPECT_EQ(positionals[0], "one");
|
|
EXPECT_EQ(positionals[1], "two");
|
|
}
|
|
|
|
// Bool options (missing) and positional arguments (given)
|
|
// Expected: only the positional arguments are filled
|
|
bool_opt1 = false;
|
|
bool_opt2 = false;
|
|
positionals = {};
|
|
parser_result = run_parser({ "app", "one", "two" }, [&](auto& parser) {
|
|
parser.add_option(bool_opt1, "bool_opt1", nullptr, 'b');
|
|
parser.add_option(bool_opt2, "bool_opt2", nullptr, 'c');
|
|
parser.add_positional_argument(positionals, "pos", "pos", Core::ArgsParser::Required::No);
|
|
});
|
|
EXPECT_EQ(parser_result, true);
|
|
EXPECT_EQ(bool_opt1, false);
|
|
EXPECT_EQ(bool_opt2, false);
|
|
EXPECT_EQ(positionals.size(), 2u);
|
|
if (positionals.size() == 2u) {
|
|
EXPECT_EQ(positionals[0], "one");
|
|
EXPECT_EQ(positionals[1], "two");
|
|
}
|
|
|
|
// Bool options (given) and positional arguments (missing)
|
|
// Expected: only the bool options are filled
|
|
bool_opt1 = false;
|
|
bool_opt2 = false;
|
|
positionals = {};
|
|
parser_result = run_parser({ "app", "-b", "-c" }, [&](auto& parser) {
|
|
parser.add_option(bool_opt1, "bool_opt1", nullptr, 'b');
|
|
parser.add_option(bool_opt2, "bool_opt2", nullptr, 'c');
|
|
parser.add_positional_argument(positionals, "pos", "pos", Core::ArgsParser::Required::No);
|
|
});
|
|
EXPECT_EQ(parser_result, true);
|
|
EXPECT_EQ(bool_opt1, true);
|
|
EXPECT_EQ(bool_opt2, true);
|
|
EXPECT_EQ(positionals.size(), 0u);
|
|
|
|
// Bool options (missing) and positional arguments (given) using double dash
|
|
// Expected: the bool options are interpreted as positional arguments
|
|
bool_opt1 = false;
|
|
bool_opt2 = false;
|
|
positionals = {};
|
|
parser_result = run_parser({ "app", "--", "-b", "-c" }, [&](auto& parser) {
|
|
parser.add_option(bool_opt1, "bool_opt1", nullptr, 'b');
|
|
parser.add_option(bool_opt2, "bool_opt2", nullptr, 'c');
|
|
parser.add_positional_argument(positionals, "pos", "pos", Core::ArgsParser::Required::No);
|
|
});
|
|
EXPECT_EQ(parser_result, true);
|
|
EXPECT_EQ(bool_opt1, false);
|
|
EXPECT_EQ(bool_opt2, false);
|
|
EXPECT_EQ(positionals.size(), 2u);
|
|
if (positionals.size() == 2u) {
|
|
EXPECT_EQ(positionals[0], "-b");
|
|
EXPECT_EQ(positionals[1], "-c");
|
|
}
|
|
|
|
// Bool options (one given) and positional arguments (one given) using double dash
|
|
// Expected: bool_opt1 is set, one positional is added
|
|
bool_opt1 = false;
|
|
bool_opt2 = false;
|
|
positionals = {};
|
|
parser_result = run_parser({ "app", "-b", "--", "-c" }, [&](auto& parser) {
|
|
parser.add_option(bool_opt1, "bool_opt1", nullptr, 'b');
|
|
parser.add_option(bool_opt2, "bool_opt2", nullptr, 'c');
|
|
parser.add_positional_argument(positionals, "pos", "pos", Core::ArgsParser::Required::No);
|
|
});
|
|
EXPECT_EQ(parser_result, true);
|
|
EXPECT_EQ(bool_opt1, true);
|
|
EXPECT_EQ(bool_opt2, false);
|
|
EXPECT_EQ(positionals.size(), 1u);
|
|
if (positionals.size() == 1u) {
|
|
EXPECT_EQ(positionals[0], "-c");
|
|
}
|
|
|
|
// Bool options (three given, one incorrect) and positional arguments (missing)
|
|
// Expected: parser fails
|
|
bool_opt1 = false;
|
|
bool_opt2 = false;
|
|
positionals = {};
|
|
parser_result = run_parser({ "app", "-b", "-d", "-c" }, [&](auto& parser) {
|
|
parser.add_option(bool_opt1, "bool_opt1", nullptr, 'b');
|
|
parser.add_option(bool_opt2, "bool_opt2", nullptr, 'c');
|
|
parser.add_positional_argument(positionals, "pos", "pos", Core::ArgsParser::Required::No);
|
|
});
|
|
EXPECT_EQ(parser_result, false);
|
|
};
|
|
|
|
TEST_CASE(stop_on_first_non_option)
|
|
{
|
|
// Do not stop on first non-option; arguments in correct order
|
|
// Expected: bool options are set and one positional argument is filled
|
|
bool bool_opt1 = false;
|
|
bool bool_opt2 = false;
|
|
Vector<String> positionals = {};
|
|
auto parser_result = run_parser({ "app", "-b", "-c", "one" }, [&](auto& parser) {
|
|
parser.set_stop_on_first_non_option(false);
|
|
parser.add_option(bool_opt1, "bool_opt1", nullptr, 'b');
|
|
parser.add_option(bool_opt2, "bool_opt2", nullptr, 'c');
|
|
parser.add_positional_argument(positionals, "pos", "pos", Core::ArgsParser::Required::Yes);
|
|
});
|
|
EXPECT_EQ(parser_result, true);
|
|
EXPECT_EQ(bool_opt1, true);
|
|
EXPECT_EQ(bool_opt2, true);
|
|
EXPECT_EQ(positionals.size(), 1u);
|
|
if (positionals.size() == 1u)
|
|
EXPECT_EQ(positionals[0], "one");
|
|
|
|
// Do not stop on first non-option; arguments in wrong order
|
|
// Expected: bool options are set and one positional argument is filled
|
|
bool_opt1 = false;
|
|
bool_opt2 = false;
|
|
positionals = {};
|
|
parser_result = run_parser({ "app", "-b", "one", "-c" }, [&](auto& parser) {
|
|
parser.set_stop_on_first_non_option(false);
|
|
parser.add_option(bool_opt1, "bool_opt1", nullptr, 'b');
|
|
parser.add_option(bool_opt2, "bool_opt2", nullptr, 'c');
|
|
parser.add_positional_argument(positionals, "pos", "pos", Core::ArgsParser::Required::Yes);
|
|
});
|
|
EXPECT_EQ(parser_result, true);
|
|
EXPECT_EQ(bool_opt1, true);
|
|
EXPECT_EQ(bool_opt2, true);
|
|
EXPECT_EQ(positionals.size(), 1u);
|
|
if (positionals.size() == 1u)
|
|
EXPECT_EQ(positionals[0], "one");
|
|
|
|
// Stop on first non-option; arguments in correct order
|
|
// Expected: bool options are set and one positional argument is filled
|
|
bool_opt1 = false;
|
|
bool_opt2 = false;
|
|
positionals = {};
|
|
parser_result = run_parser({ "app", "-b", "-c", "one" }, [&](auto& parser) {
|
|
parser.set_stop_on_first_non_option(true);
|
|
parser.add_option(bool_opt1, "bool_opt1", nullptr, 'b');
|
|
parser.add_option(bool_opt2, "bool_opt2", nullptr, 'c');
|
|
parser.add_positional_argument(positionals, "pos", "pos", Core::ArgsParser::Required::Yes);
|
|
});
|
|
EXPECT_EQ(parser_result, true);
|
|
EXPECT_EQ(bool_opt1, true);
|
|
EXPECT_EQ(bool_opt2, true);
|
|
EXPECT_EQ(positionals.size(), 1u);
|
|
if (positionals.size() == 1u)
|
|
EXPECT_EQ(positionals[0], "one");
|
|
|
|
// Stop on first non-option; arguments in wrong order
|
|
// Expected: bool_opt1 is set, other arguments are filled as positional arguments
|
|
bool_opt1 = false;
|
|
bool_opt2 = false;
|
|
positionals = {};
|
|
parser_result = run_parser({ "app", "-b", "one", "-c" }, [&](auto& parser) {
|
|
parser.set_stop_on_first_non_option(true);
|
|
parser.add_option(bool_opt1, "bool_opt1", nullptr, 'b');
|
|
parser.add_option(bool_opt2, "bool_opt2", nullptr, 'c');
|
|
parser.add_positional_argument(positionals, "pos", "pos", Core::ArgsParser::Required::Yes);
|
|
});
|
|
EXPECT_EQ(parser_result, true);
|
|
EXPECT_EQ(bool_opt1, true);
|
|
EXPECT_EQ(bool_opt2, false);
|
|
EXPECT_EQ(positionals.size(), 2u);
|
|
if (positionals.size() == 2u) {
|
|
EXPECT_EQ(positionals[0], "one");
|
|
EXPECT_EQ(positionals[1], "-c");
|
|
}
|
|
}
|