2023-04-19 17:16:45 +00:00
|
|
|
#include <luna/StringBuilder.h>
|
2023-03-29 16:27:02 +00:00
|
|
|
#include <os/ArgumentParser.h>
|
2023-05-01 17:32:00 +00:00
|
|
|
#include <os/File.h>
|
2023-03-29 16:27:02 +00:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
2023-04-28 14:33:05 +00:00
|
|
|
#include <sys/utsname.h>
|
2023-03-29 16:27:02 +00:00
|
|
|
|
2023-04-07 08:40:46 +00:00
|
|
|
namespace os
|
2023-03-29 16:27:02 +00:00
|
|
|
{
|
2023-04-28 14:33:05 +00:00
|
|
|
ArgumentParser::ArgumentParser() : m_add_short_help_flag(true), m_add_long_help_flag(true)
|
2023-04-19 17:16:45 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void ArgumentParser::add_description(StringView description)
|
|
|
|
{
|
|
|
|
m_description = description;
|
|
|
|
}
|
|
|
|
|
2023-04-07 08:40:46 +00:00
|
|
|
Result<void> ArgumentParser::add_positional_argument(StringView& out, StringView name, bool required)
|
|
|
|
{
|
|
|
|
PositionalArgument arg { &out, name, required, {} };
|
2023-03-29 16:27:02 +00:00
|
|
|
|
2023-04-07 08:40:46 +00:00
|
|
|
return m_positional_args.try_append(move(arg));
|
|
|
|
}
|
2023-03-29 16:27:02 +00:00
|
|
|
|
2023-04-07 08:40:46 +00:00
|
|
|
Result<void> ArgumentParser::add_positional_argument(StringView& out, StringView name, StringView fallback)
|
|
|
|
{
|
|
|
|
PositionalArgument arg { &out, name, false, fallback };
|
2023-03-29 16:27:02 +00:00
|
|
|
|
2023-04-07 08:40:46 +00:00
|
|
|
return m_positional_args.try_append(move(arg));
|
|
|
|
}
|
2023-03-29 17:25:11 +00:00
|
|
|
|
2023-04-19 17:16:45 +00:00
|
|
|
Result<void> ArgumentParser::add_switch_argument(bool& out, char short_flag, StringView long_flag, StringView help)
|
2023-04-07 08:40:46 +00:00
|
|
|
{
|
2023-04-19 17:16:45 +00:00
|
|
|
SwitchArgument 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;
|
2023-04-28 14:33:05 +00:00
|
|
|
if (short_flag == 'v') m_add_short_version_flag = false;
|
|
|
|
if (long_flag == "version"_sv) m_add_long_version_flag = false;
|
2023-03-29 17:25:11 +00:00
|
|
|
|
2023-04-07 08:40:46 +00:00
|
|
|
return m_switch_args.try_append(move(arg));
|
|
|
|
}
|
2023-03-29 19:46:07 +00:00
|
|
|
|
2023-04-07 08:40:46 +00:00
|
|
|
Result<void> ArgumentParser::add_value_argument(StringView& out, char short_flag, StringView long_flag,
|
2023-04-19 17:16:45 +00:00
|
|
|
bool value_required, StringView help)
|
2023-04-07 08:40:46 +00:00
|
|
|
{
|
2023-04-19 17:16:45 +00:00
|
|
|
ValueArgument arg { &out, short_flag, long_flag, value_required, {}, help };
|
|
|
|
|
|
|
|
if (short_flag == 'h') m_add_short_help_flag = false;
|
|
|
|
if (long_flag == "help"_sv) m_add_long_help_flag = false;
|
2023-04-28 14:33:05 +00:00
|
|
|
if (short_flag == 'v') m_add_short_version_flag = false;
|
|
|
|
if (long_flag == "version"_sv) m_add_long_version_flag = false;
|
2023-03-29 19:46:07 +00:00
|
|
|
|
2023-04-07 08:40:46 +00:00
|
|
|
return m_value_args.try_append(move(arg));
|
|
|
|
}
|
2023-03-29 19:46:07 +00:00
|
|
|
|
2023-04-07 08:40:46 +00:00
|
|
|
Result<void> ArgumentParser::add_value_argument(StringView& out, char short_flag, StringView long_flag,
|
2023-04-19 17:16:45 +00:00
|
|
|
StringView fallback, StringView help)
|
2023-04-07 08:40:46 +00:00
|
|
|
{
|
2023-04-19 17:16:45 +00:00
|
|
|
ValueArgument arg { &out, short_flag, long_flag, false, fallback, help };
|
|
|
|
|
|
|
|
if (short_flag == 'h') m_add_short_help_flag = false;
|
|
|
|
if (long_flag == "help"_sv) m_add_long_help_flag = false;
|
2023-04-28 14:33:05 +00:00
|
|
|
if (short_flag == 'v') m_add_short_version_flag = false;
|
|
|
|
if (long_flag == "version"_sv) m_add_long_version_flag = false;
|
2023-03-29 19:46:07 +00:00
|
|
|
|
2023-04-07 08:40:46 +00:00
|
|
|
return m_value_args.try_append(move(arg));
|
|
|
|
}
|
2023-03-29 17:25:11 +00:00
|
|
|
|
2023-04-28 14:33:05 +00:00
|
|
|
constexpr auto copyright_text = "Copyright (C) 2023, the Luna authors.";
|
|
|
|
constexpr auto license_text = "Licensed under the BSD-2 license <https://opensource.org/license/bsd-2-clause/>";
|
|
|
|
|
|
|
|
void ArgumentParser::add_program_info(ProgramInfo info)
|
|
|
|
{
|
|
|
|
m_program_info = info;
|
|
|
|
m_add_short_version_flag = m_add_long_version_flag = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ArgumentParser::add_system_program_info(StringView name)
|
|
|
|
{
|
|
|
|
static utsname info;
|
|
|
|
check(uname(&info) == 0);
|
|
|
|
|
|
|
|
ProgramInfo system_info {
|
|
|
|
.name = name,
|
|
|
|
.version = info.release,
|
|
|
|
.copyright = copyright_text,
|
|
|
|
.license = license_text,
|
|
|
|
.authors = {},
|
|
|
|
.package = "Luna system",
|
|
|
|
};
|
|
|
|
|
|
|
|
add_program_info(system_info);
|
|
|
|
}
|
|
|
|
|
2023-04-07 08:40:46 +00:00
|
|
|
static bool looks_like_short_flag(StringView arg)
|
|
|
|
{
|
|
|
|
return arg.length() > 1 && arg[0] == '-';
|
|
|
|
}
|
2023-03-29 17:25:11 +00:00
|
|
|
|
2023-04-07 08:40:46 +00:00
|
|
|
static bool looks_like_long_flag(StringView arg)
|
|
|
|
{
|
|
|
|
return arg.length() > 2 && arg[0] == '-' && arg[1] == '-';
|
|
|
|
}
|
2023-03-29 16:27:02 +00:00
|
|
|
|
2023-04-07 08:40:46 +00:00
|
|
|
Result<Vector<StringView>> ArgumentParser::parse(int argc, char* const* argv)
|
|
|
|
{
|
|
|
|
StringView program_name = argv[0];
|
2023-03-29 20:07:42 +00:00
|
|
|
|
2023-04-07 08:40:46 +00:00
|
|
|
Vector<StringView> leftovers;
|
2023-03-29 19:46:07 +00:00
|
|
|
|
2023-04-07 08:40:46 +00:00
|
|
|
Option<ValueArgument> current_value_argument = {};
|
|
|
|
bool is_parsing_value_argument = false;
|
2023-03-29 17:25:11 +00:00
|
|
|
|
2023-04-07 08:40:46 +00:00
|
|
|
bool is_still_parsing_flags = true;
|
2023-03-29 17:25:11 +00:00
|
|
|
|
2023-04-07 08:40:46 +00:00
|
|
|
for (int i = 1; i < argc; i++)
|
2023-03-29 19:46:07 +00:00
|
|
|
{
|
2023-04-07 08:40:46 +00:00
|
|
|
StringView arg = argv[i];
|
2023-03-29 19:46:07 +00:00
|
|
|
|
2023-04-07 08:40:46 +00:00
|
|
|
if (is_parsing_value_argument)
|
2023-03-29 17:25:11 +00:00
|
|
|
{
|
2023-04-07 08:40:46 +00:00
|
|
|
*current_value_argument->out = arg;
|
|
|
|
is_parsing_value_argument = false;
|
2023-03-29 17:25:11 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2023-04-07 08:40:46 +00:00
|
|
|
if (is_still_parsing_flags)
|
2023-03-29 17:25:11 +00:00
|
|
|
{
|
2023-04-07 08:40:46 +00:00
|
|
|
if (arg == "--")
|
2023-03-29 17:25:11 +00:00
|
|
|
{
|
2023-04-07 08:40:46 +00:00
|
|
|
is_still_parsing_flags = false;
|
|
|
|
continue;
|
2023-03-29 17:25:11 +00:00
|
|
|
}
|
|
|
|
|
2023-04-07 08:40:46 +00:00
|
|
|
if (looks_like_long_flag(arg))
|
2023-03-29 19:46:07 +00:00
|
|
|
{
|
2023-04-07 08:40:46 +00:00
|
|
|
StringView flag = &arg[2];
|
|
|
|
|
|
|
|
bool found = false;
|
|
|
|
|
2023-04-19 17:16:45 +00:00
|
|
|
if (m_add_long_help_flag && flag == "help"_sv) { TRY(usage(program_name)); }
|
2023-04-28 14:33:05 +00:00
|
|
|
if (m_add_long_version_flag && flag == "version"_sv) { version(); }
|
2023-04-19 17:16:45 +00:00
|
|
|
|
2023-04-07 08:40:46 +00:00
|
|
|
for (const auto& current : m_switch_args)
|
2023-03-29 19:46:07 +00:00
|
|
|
{
|
2023-04-07 08:40:46 +00:00
|
|
|
if (current.long_flag == flag)
|
|
|
|
{
|
|
|
|
*current.out = true;
|
|
|
|
found = true;
|
|
|
|
break;
|
|
|
|
}
|
2023-03-29 19:46:07 +00:00
|
|
|
}
|
|
|
|
|
2023-04-07 08:40:46 +00:00
|
|
|
for (const auto& current : m_value_args)
|
|
|
|
{
|
|
|
|
if (current.long_flag == flag)
|
|
|
|
{
|
|
|
|
current_value_argument = current;
|
|
|
|
is_parsing_value_argument = true;
|
|
|
|
found = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2023-03-29 17:25:11 +00:00
|
|
|
|
2023-04-07 08:40:46 +00:00
|
|
|
if (found) continue;
|
2023-03-29 17:25:11 +00:00
|
|
|
|
2023-05-01 17:32:00 +00:00
|
|
|
os::eprintln("%s: unrecognized option '%s'", program_name.chars(), arg.chars());
|
2023-04-19 17:16:45 +00:00
|
|
|
short_usage(program_name);
|
2023-04-07 08:40:46 +00:00
|
|
|
}
|
|
|
|
else if (looks_like_short_flag(arg))
|
2023-03-29 17:25:11 +00:00
|
|
|
{
|
2023-04-07 08:40:46 +00:00
|
|
|
StringView flags = &arg[1];
|
2023-03-29 17:25:11 +00:00
|
|
|
|
2023-04-07 08:40:46 +00:00
|
|
|
for (usize j = 0; j < flags.length(); j++)
|
2023-04-07 08:37:15 +00:00
|
|
|
{
|
2023-04-07 08:40:46 +00:00
|
|
|
char c = flags[j];
|
|
|
|
bool found = false;
|
|
|
|
|
2023-04-19 17:16:45 +00:00
|
|
|
if (m_add_short_help_flag && c == 'h') { TRY(usage(program_name)); }
|
2023-04-28 14:33:05 +00:00
|
|
|
if (m_add_short_version_flag && c == 'v') { version(); }
|
2023-04-19 17:16:45 +00:00
|
|
|
|
2023-04-07 08:40:46 +00:00
|
|
|
// Last flag, this could be a value flag
|
|
|
|
if (j + 1 == flags.length())
|
|
|
|
{
|
|
|
|
for (const auto& current : m_value_args)
|
|
|
|
{
|
|
|
|
if (current.short_flag == ' ') continue;
|
|
|
|
|
|
|
|
if (current.short_flag == c)
|
|
|
|
{
|
|
|
|
current_value_argument = current;
|
|
|
|
is_parsing_value_argument = true;
|
|
|
|
found = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (found) continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (const auto& current : m_switch_args)
|
2023-04-07 08:37:15 +00:00
|
|
|
{
|
|
|
|
if (current.short_flag == ' ') continue;
|
|
|
|
|
|
|
|
if (current.short_flag == c)
|
|
|
|
{
|
2023-04-07 08:40:46 +00:00
|
|
|
*current.out = true;
|
2023-04-07 08:37:15 +00:00
|
|
|
found = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (found) continue;
|
2023-03-29 17:25:11 +00:00
|
|
|
|
2023-05-01 17:32:00 +00:00
|
|
|
os::eprintln("%s: invalid option -- '%c'", program_name.chars(), c);
|
2023-04-19 17:16:45 +00:00
|
|
|
short_usage(program_name);
|
2023-03-29 17:25:11 +00:00
|
|
|
}
|
|
|
|
|
2023-04-07 08:40:46 +00:00
|
|
|
continue;
|
2023-03-29 17:25:11 +00:00
|
|
|
}
|
2023-04-07 08:40:46 +00:00
|
|
|
}
|
2023-03-29 17:25:11 +00:00
|
|
|
|
2023-04-07 08:40:46 +00:00
|
|
|
Option<PositionalArgument> current = m_positional_args.try_dequeue();
|
|
|
|
if (!current.has_value())
|
|
|
|
{
|
|
|
|
TRY(leftovers.try_append(arg));
|
2023-03-29 17:25:11 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2023-04-07 08:40:46 +00:00
|
|
|
*current->out = arg;
|
2023-03-29 16:27:02 +00:00
|
|
|
}
|
|
|
|
|
2023-04-07 08:40:46 +00:00
|
|
|
if (is_parsing_value_argument)
|
2023-03-29 19:46:07 +00:00
|
|
|
{
|
2023-04-07 08:40:46 +00:00
|
|
|
if (current_value_argument->required)
|
|
|
|
{
|
2023-05-01 17:32:00 +00:00
|
|
|
os::eprintln("%s: option '--%s' requires an argument", program_name.chars(),
|
|
|
|
current_value_argument->long_flag.chars());
|
2023-04-19 17:16:45 +00:00
|
|
|
short_usage(program_name);
|
2023-04-07 08:40:46 +00:00
|
|
|
}
|
|
|
|
else { *current_value_argument->out = current_value_argument->fallback; }
|
2023-03-29 19:46:07 +00:00
|
|
|
}
|
|
|
|
|
2023-04-07 08:40:46 +00:00
|
|
|
// Loop through all remaining positional arguments.
|
|
|
|
for (const auto& arg : m_positional_args)
|
2023-03-29 16:27:02 +00:00
|
|
|
{
|
2023-04-07 08:40:46 +00:00
|
|
|
if (arg.required)
|
|
|
|
{
|
2023-05-01 17:32:00 +00:00
|
|
|
os::eprintln("%s: required argument '%s' not provided", program_name.chars(), arg.name.chars());
|
2023-04-19 17:16:45 +00:00
|
|
|
short_usage(program_name);
|
2023-04-07 08:40:46 +00:00
|
|
|
}
|
|
|
|
else { *arg.out = arg.fallback; }
|
2023-03-29 16:27:02 +00:00
|
|
|
}
|
2023-03-29 20:07:42 +00:00
|
|
|
|
2023-04-07 08:40:46 +00:00
|
|
|
return leftovers;
|
|
|
|
}
|
2023-04-19 17:16:45 +00:00
|
|
|
|
|
|
|
Result<void> ArgumentParser::usage(StringView program_name)
|
|
|
|
{
|
|
|
|
StringBuilder sb;
|
|
|
|
TRY(sb.format("Usage: %s <options>"_sv, program_name.chars()));
|
|
|
|
|
|
|
|
for (const auto& arg : m_positional_args)
|
|
|
|
{
|
|
|
|
if (arg.required) { TRY(sb.format(" %s", arg.name.chars())); }
|
|
|
|
else { TRY(sb.format(" [%s]", arg.name.chars())); }
|
|
|
|
}
|
|
|
|
|
|
|
|
TRY(sb.add('\n'));
|
|
|
|
|
|
|
|
auto usage_line = TRY(sb.string());
|
|
|
|
fputs(usage_line.chars(), stdout);
|
|
|
|
|
|
|
|
if (!m_description.is_empty()) { puts(m_description.chars()); }
|
|
|
|
|
|
|
|
if (m_switch_args.size() || m_value_args.size())
|
|
|
|
{
|
|
|
|
putchar('\n');
|
|
|
|
puts("Options:");
|
|
|
|
}
|
|
|
|
|
|
|
|
for (const auto& arg : m_switch_args)
|
|
|
|
{
|
|
|
|
if (arg.long_flag.is_empty())
|
|
|
|
{
|
|
|
|
if (arg.short_flag == ' ') continue;
|
|
|
|
|
|
|
|
printf(" -%-25c %s\n", arg.short_flag, arg.help.chars());
|
|
|
|
}
|
|
|
|
else if (arg.short_flag == ' ') { printf(" --%-20s %s\n", arg.long_flag.chars(), arg.help.chars()); }
|
|
|
|
else { printf(" -%c, --%-20s %s\n", arg.short_flag, arg.long_flag.chars(), arg.help.chars()); }
|
|
|
|
}
|
|
|
|
|
|
|
|
for (const auto& arg : m_value_args)
|
|
|
|
{
|
|
|
|
StringView value_name;
|
|
|
|
if (arg.required) value_name = "VALUE"_sv;
|
|
|
|
else
|
|
|
|
value_name = "[VALUE]"_sv;
|
|
|
|
|
|
|
|
int field_size = 20 - (int)(arg.long_flag.length() + 1);
|
|
|
|
if (field_size < 0) field_size = 0;
|
|
|
|
|
|
|
|
if (arg.long_flag.is_empty())
|
|
|
|
{
|
|
|
|
if (arg.short_flag == ' ') continue;
|
|
|
|
|
|
|
|
printf(" -%c %-20s %s\n", arg.short_flag, value_name.chars(), arg.help.chars());
|
|
|
|
}
|
|
|
|
else if (arg.short_flag == ' ')
|
|
|
|
{
|
|
|
|
printf(" --%s %-*s %s\n", arg.long_flag.chars(), field_size, value_name.chars(), arg.help.chars());
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
printf(" -%c, --%s %-*s %s\n", arg.short_flag, arg.long_flag.chars(), field_size, value_name.chars(),
|
|
|
|
arg.help.chars());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-04-28 14:33:05 +00:00
|
|
|
if (m_add_short_help_flag && m_add_long_help_flag)
|
|
|
|
printf(" -h, --%-20s %s\n", "help", "show this help message and exit");
|
|
|
|
else if (m_add_long_help_flag)
|
|
|
|
printf(" --%-20s %s\n", "help", "show this help message and exit");
|
|
|
|
else
|
|
|
|
printf(" -%-25c %s\n", 'h', "show this help message and exit");
|
|
|
|
|
|
|
|
if (m_add_short_version_flag && m_add_long_version_flag)
|
|
|
|
printf(" -v, --%-20s %s\n", "version", "show version information and exit");
|
|
|
|
else if (m_add_long_version_flag)
|
|
|
|
printf(" --%-20s %s\n", "version", "show version information and exit");
|
|
|
|
else
|
|
|
|
printf(" -%-25c %s\n", 'v', "show version information and exit");
|
|
|
|
|
|
|
|
exit(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
void ArgumentParser::version()
|
|
|
|
{
|
|
|
|
if (m_program_info.package.is_empty())
|
|
|
|
printf("%s %s\n", m_program_info.name.chars(), m_program_info.version.chars());
|
|
|
|
else
|
|
|
|
printf("%s (%s) %s\n", m_program_info.name.chars(), m_program_info.package.chars(),
|
|
|
|
m_program_info.version.chars());
|
|
|
|
|
|
|
|
printf("%s\n%s\n", m_program_info.copyright.chars(), m_program_info.license.chars());
|
|
|
|
|
|
|
|
if (!m_program_info.authors.is_empty()) printf("\n%s\n", m_program_info.authors.chars());
|
|
|
|
|
2023-04-19 17:16:45 +00:00
|
|
|
exit(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
void ArgumentParser::short_usage(StringView program_name)
|
|
|
|
{
|
|
|
|
if (m_add_short_help_flag || m_add_long_help_flag)
|
2023-05-01 17:32:00 +00:00
|
|
|
os::eprintln("Try running '%s %s' for more information.", program_name.chars(),
|
|
|
|
m_add_long_help_flag ? "--help" : "-h");
|
2023-04-19 17:16:45 +00:00
|
|
|
|
|
|
|
exit(1);
|
|
|
|
}
|
2023-03-29 16:27:02 +00:00
|
|
|
}
|