libos: Make long value arguments use '=' and make value arguments' values always required
All checks were successful
continuous-integration/drone/push Build is passing

This commit is contained in:
apio 2023-06-17 20:58:54 +02:00
parent 266fa4a0d4
commit 2f08e0f5b0
Signed by: apio
GPG Key ID: B8A7D06E42258954
10 changed files with 64 additions and 72 deletions

View File

@ -11,8 +11,7 @@ Result<int> luna_main(int argc, char** argv)
os::ArgumentParser parser; os::ArgumentParser parser;
parser.add_description("Display the current (or another) date and time."_sv); parser.add_description("Display the current (or another) date and time."_sv);
parser.add_system_program_info("date"_sv); parser.add_system_program_info("date"_sv);
parser.add_value_argument(date, 'd', "date"_sv, true, parser.add_value_argument(date, 'd', "date"_sv, "the UNIX timestamp to display instead of the current time"_sv);
"the UNIX timestamp to display instead of the current time"_sv);
parser.parse(argc, argv); parser.parse(argc, argv);
time_t now; time_t now;

View File

@ -96,7 +96,7 @@ Result<int> luna_main(int argc, char** argv)
"follow symbolic links listed as arguments"_sv); "follow symbolic links listed as arguments"_sv);
parser.add_switch_argument(one_per_line, '1', ""_sv, "list one file per line"_sv); parser.add_switch_argument(one_per_line, '1', ""_sv, "list one file per line"_sv);
parser.add_switch_argument(list_directories, 'd', "directory"_sv, "list directories instead of their contents"_sv); parser.add_switch_argument(list_directories, 'd', "directory"_sv, "list directories instead of their contents"_sv);
parser.add_value_argument(sort_type, ' ', "sort"_sv, true, "sort by name, size or time"_sv); parser.add_value_argument(sort_type, ' ', "sort"_sv, "sort by name, size or time"_sv);
parser.add_switch_argument(reverse_sort, 'r', "reverse"_sv, "reverse order while sorting"_sv); parser.add_switch_argument(reverse_sort, 'r', "reverse"_sv, "reverse order while sorting"_sv);
parser.parse(argc, argv); parser.parse(argc, argv);

View File

@ -41,7 +41,7 @@ Result<int> luna_main(int argc, char** argv)
parser.add_description("Create directories."_sv); parser.add_description("Create directories."_sv);
parser.add_system_program_info("mkdir"_sv); parser.add_system_program_info("mkdir"_sv);
parser.add_positional_argument(path, "path"_sv, true); parser.add_positional_argument(path, "path"_sv, true);
parser.add_value_argument(mode_string, 'm', "mode"_sv, true, "set the mode for the newly created directory"); parser.add_value_argument(mode_string, 'm', "mode"_sv, "set the mode for the newly created directory");
parser.add_switch_argument(recursive, 'p', "parents"_sv, parser.add_switch_argument(recursive, 'p', "parents"_sv,
"if parent directories do not exist, create them as well"_sv); "if parent directories do not exist, create them as well"_sv);
parser.parse(argc, argv); parser.parse(argc, argv);

View File

@ -5,13 +5,13 @@
Result<int> luna_main(int argc, char** argv) Result<int> luna_main(int argc, char** argv)
{ {
StringView target; StringView target;
StringView fstype; StringView fstype { "auto" };
os::ArgumentParser parser; os::ArgumentParser parser;
parser.add_description("Mount a file system."); parser.add_description("Mount a file system.");
parser.add_system_program_info("mount"_sv); parser.add_system_program_info("mount"_sv);
parser.add_positional_argument(target, "mountpoint"_sv, true); parser.add_positional_argument(target, "mountpoint"_sv, true);
parser.add_value_argument(fstype, 't', "type"_sv, "auto"_sv, "the file system type to use"); parser.add_value_argument(fstype, 't', "type"_sv, "the file system type to use");
parser.parse(argc, argv); parser.parse(argc, argv);
if (mount(target.chars(), fstype.chars()) < 0) if (mount(target.chars(), fstype.chars()) < 0)

View File

@ -48,7 +48,7 @@ Result<int> luna_main(int argc, char** argv)
parser.add_description("The Luna system's command shell."_sv); parser.add_description("The Luna system's command shell."_sv);
parser.add_system_program_info("sh"_sv); parser.add_system_program_info("sh"_sv);
parser.add_positional_argument(path, "path"_sv, "-"_sv); parser.add_positional_argument(path, "path"_sv, "-"_sv);
parser.add_value_argument(command, 'c', "command"_sv, true, "execute a single command and then exit"_sv); parser.add_value_argument(command, 'c', "command"_sv, "execute a single command and then exit"_sv);
parser.parse(argc, argv); parser.parse(argc, argv);
if (!command.is_empty()) TRY(execute_command(command)); if (!command.is_empty()) TRY(execute_command(command));

View File

@ -40,8 +40,8 @@ int main(int argc, char** argv)
os::ArgumentParser parser; os::ArgumentParser parser;
parser.add_description("System call fuzzer (invokes system calls with random arguments to test system stability)"); parser.add_description("System call fuzzer (invokes system calls with random arguments to test system stability)");
parser.add_system_program_info("sysfuzz"_sv); parser.add_system_program_info("sysfuzz"_sv);
parser.add_value_argument(times_sv, 't', "times"_sv, true, "the number of syscalls to invoke"_sv); parser.add_value_argument(times_sv, 't', "times"_sv, "the number of syscalls to invoke"_sv);
parser.add_value_argument(interval_sv, 'i', "interval"_sv, true, parser.add_value_argument(interval_sv, 'i', "interval"_sv,
"the interval between system calls (in milliseconds)"_sv); "the interval between system calls (in milliseconds)"_sv);
parser.parse(argc, argv); parser.parse(argc, argv);

View File

@ -42,6 +42,9 @@ class StringView
Result<Vector<String>> split(StringView delim) const; Result<Vector<String>> split(StringView delim) const;
Result<Vector<String>> split_once(char delim) const; Result<Vector<String>> split_once(char delim) const;
Result<Vector<StringView>> split_view(char delim) const;
bool contains(char v) const;
static StringView from_fixed_size_cstring(const char* string, usize max); static StringView from_fixed_size_cstring(const char* string, usize max);

View File

@ -50,12 +50,12 @@ const char& StringView::operator[](usize index) const
bool StringView::operator==(const char* other) const bool StringView::operator==(const char* other) const
{ {
return !strcmp(m_string, other); return !strncmp(m_string, other, m_length);
} }
bool StringView::operator==(StringView other) const bool StringView::operator==(StringView other) const
{ {
return !strcmp(m_string, other.m_string); return !strncmp(m_string, other.m_string, m_length);
} }
Result<Vector<String>> StringView::split(StringView delim) const Result<Vector<String>> StringView::split(StringView delim) const
@ -110,6 +110,32 @@ Result<Vector<String>> StringView::split_once(char delim) const
return result; return result;
} }
Result<Vector<StringView>> StringView::split_view(char delim) const
{
Vector<StringView> result;
char* middle = strchr(m_string, delim);
if (!middle)
{
TRY(result.try_append(*this));
return result;
}
// begin will not contain a null terminator.
auto begin = StringView::from_fixed_size_cstring(m_string, middle - m_string);
auto end = StringView { middle + 1 };
TRY(result.try_append(begin));
TRY(result.try_append(end));
return result;
}
bool StringView::contains(char v) const
{
return strchr(m_string, v);
}
Result<usize> StringView::to_uint() const Result<usize> StringView::to_uint() const
{ {
const char* endptr = nullptr; const char* endptr = nullptr;

View File

@ -78,32 +78,10 @@ namespace os
* @param long_flag The long flag to use for this argument, excluding the '--' prefix: an option '--example' * @param long_flag The long flag to use for this argument, excluding the '--' prefix: an option '--example'
* would be passed as 'example'. Can be a string of any length. If you do not wish this argument to have a long * would be passed as 'example'. Can be a string of any length. If you do not wish this argument to have a long
* flag, use an empty string: "". * flag, use an empty string: "".
* @param fallback The value to use if the user uses the argument but does not provide a value. If, however, the
* user does not specify this argument at all, out WILL NOT BE MODIFIED!
* @param help The help text to show for this argument (optional). * @param help The help text to show for this argument (optional).
* @return Result<void> Whether the operation succeeded. * @return Result<void> Whether the operation succeeded.
*/ */
Result<void> add_value_argument(StringView& out, char short_flag, StringView long_flag, StringView fallback, Result<void> add_value_argument(StringView& out, char short_flag, StringView long_flag, StringView help = {});
StringView help = {});
/**
* @brief Register a new value argument.
*
* @param out The variable where the argument's value will be stored.
* @param short_flag The short flag to use for this argument, excluding the '-' prefix: an option '-c' would be
* passed as 'c'. Can only be a single ASCII character. If you do not wish this argument to have a short flag,
* use the space character: ' '.
* @param long_flag The long flag to use for this argument, excluding the '--' prefix: an option '--example'
* would be passed as 'example'. Can be a string of any length. If you do not wish this argument to have a long
* flag, use an empty string: "".
* @param value_required Whether the user is required to pass a value when using this argument. If this is
* false and the user does not enter a value along with the argument, out will be set to an empty string. If,
* however, the user does not specify this argument at all, out WILL NOT BE MODIFIED!
* @param help The help text to show for this argument (optional).
* @return Result<void> Whether the operation succeeded.
*/
Result<void> add_value_argument(StringView& out, char short_flag, StringView long_flag, bool value_required,
StringView help = {});
/** /**
* @brief Set the vector to append extra unregistered positional arguments to, after all * @brief Set the vector to append extra unregistered positional arguments to, after all
@ -192,8 +170,6 @@ namespace os
StringView* out; StringView* out;
char short_flag; char short_flag;
StringView long_flag; StringView long_flag;
bool required;
StringView fallback;
StringView help; StringView help;
}; };

View File

@ -8,6 +8,7 @@
*/ */
#include <luna/StringBuilder.h> #include <luna/StringBuilder.h>
#include <luna/TypeTraits.h>
#include <os/ArgumentParser.h> #include <os/ArgumentParser.h>
#include <os/File.h> #include <os/File.h>
#include <stdio.h> #include <stdio.h>
@ -52,22 +53,9 @@ namespace os
} }
Result<void> ArgumentParser::add_value_argument(StringView& out, char short_flag, StringView long_flag, Result<void> ArgumentParser::add_value_argument(StringView& out, char short_flag, StringView long_flag,
bool value_required, StringView help) StringView help)
{ {
ValueArgument arg { &out, short_flag, long_flag, value_required, {}, help }; ValueArgument arg { &out, short_flag, long_flag, help };
if (short_flag == 'h') m_add_short_help_flag = false;
if (long_flag == "help"_sv) m_add_long_help_flag = false;
if (short_flag == 'v') m_add_short_version_flag = false;
if (long_flag == "version"_sv) m_add_long_version_flag = false;
return m_value_args.try_append(move(arg));
}
Result<void> ArgumentParser::add_value_argument(StringView& out, char short_flag, StringView long_flag,
StringView fallback, StringView help)
{
ValueArgument arg { &out, short_flag, long_flag, false, fallback, help };
if (short_flag == 'h') m_add_short_help_flag = false; if (short_flag == 'h') m_add_short_help_flag = false;
if (long_flag == "help"_sv) m_add_long_help_flag = false; if (long_flag == "help"_sv) m_add_long_help_flag = false;
@ -130,6 +118,8 @@ namespace os
bool is_still_parsing_flags = true; bool is_still_parsing_flags = true;
Vector<PositionalArgument> positional_args = TRY(m_positional_args.deep_copy());
for (int i = 1; i < argc; i++) for (int i = 1; i < argc; i++)
{ {
StringView arg = argv[i]; StringView arg = argv[i];
@ -168,14 +158,19 @@ namespace os
} }
} }
for (const auto& current : m_value_args) if (flag.contains('='))
{ {
if (current.long_flag == flag) auto v = TRY(flag.split_view('='));
StringView actual_flag = v[0];
StringView data = v[1];
for (const auto& current : m_value_args)
{ {
current_value_argument = current; if (current.long_flag == actual_flag)
is_parsing_value_argument = true; {
found = true; *current.out = data;
break; found = true;
break;
}
} }
} }
@ -237,7 +232,7 @@ namespace os
} }
} }
Option<PositionalArgument> current = m_positional_args.try_dequeue(); Option<PositionalArgument> current = positional_args.try_dequeue();
if (!current.has_value()) if (!current.has_value())
{ {
if (m_vector_argument) if (m_vector_argument)
@ -253,17 +248,13 @@ namespace os
if (is_parsing_value_argument) if (is_parsing_value_argument)
{ {
if (current_value_argument->required) os::eprintln("%s: option '-%c' requires an argument", program_name.chars(),
{ current_value_argument->short_flag);
os::eprintln("%s: option '--%s' requires an argument", program_name.chars(), short_usage(program_name);
current_value_argument->long_flag.chars());
short_usage(program_name);
}
else { *current_value_argument->out = current_value_argument->fallback; }
} }
// Loop through all remaining positional arguments. // Loop through all remaining positional arguments.
for (const auto& arg : m_positional_args) for (const auto& arg : positional_args)
{ {
if (arg.required) if (arg.required)
{ {
@ -317,10 +308,7 @@ namespace os
for (const auto& arg : m_value_args) for (const auto& arg : m_value_args)
{ {
StringView value_name; StringView value_name = "VALUE"_sv;
if (arg.required) value_name = "VALUE"_sv;
else
value_name = "[VALUE]"_sv;
int field_size = 20 - (int)(arg.long_flag.length() + 1); int field_size = 20 - (int)(arg.long_flag.length() + 1);
if (field_size < 0) field_size = 0; if (field_size < 0) field_size = 0;
@ -333,11 +321,11 @@ namespace os
} }
else if (arg.short_flag == ' ') else if (arg.short_flag == ' ')
{ {
printf(" --%s %-*s %s\n", arg.long_flag.chars(), field_size, value_name.chars(), arg.help.chars()); printf(" --%s=%-*s %s\n", arg.long_flag.chars(), field_size, value_name.chars(), arg.help.chars());
} }
else else
{ {
printf(" -%c, --%s %-*s %s\n", arg.short_flag, arg.long_flag.chars(), field_size, value_name.chars(), printf(" -%c, --%s=%-*s %s\n", arg.short_flag, arg.long_flag.chars(), field_size, value_name.chars(),
arg.help.chars()); arg.help.chars());
} }
} }