vcpkg/toolsrc/src/vcpkg_cmd_arguments.cpp

250 lines
8.6 KiB
C++
Raw Normal View History

2016-09-19 11:50:08 +08:00
#define WIN32_LEAN_AND_MEAN
#include <Windows.h>
#include "vcpkg_cmd_arguments.h"
#include "vcpkg_Commands.h"
#include "vcpkg_Graphs.h"
#include <unordered_set>
#include "metrics.h"
#include "vcpkg.h"
#include "vcpkg_System.h"
namespace vcpkg
{
static void parse_value(
const std::string* arg_begin,
const std::string* arg_end,
const std::string& option_name,
std::unique_ptr<std::string>& option_field)
{
if (arg_begin == arg_end)
{
System::println(System::color::error, "Error: expected value after %s", option_name);
TrackProperty("error", "error option name");
print_usage();
exit(EXIT_FAILURE);
}
if (option_field != nullptr)
{
System::println(System::color::error, "Error: %s specified multiple times", option_name);
TrackProperty("error", "error option specified multiple times");
print_usage();
exit(EXIT_FAILURE);
}
option_field = std::make_unique<std::string>(*arg_begin);
}
static void parse_switch(
opt_bool new_setting,
const std::string& option_name,
opt_bool& option_field)
{
if (option_field != opt_bool::unspecified && option_field != new_setting)
{
System::println(System::color::error, "Error: conflicting values specified for --%s", option_name);
TrackProperty("error", "error conflicting switches");
print_usage();
exit(EXIT_FAILURE);
}
option_field = new_setting;
}
vcpkg_cmd_arguments vcpkg_cmd_arguments::create_from_command_line(const int argc, const wchar_t* const* const argv)
{
std::vector<std::string> v;
for (int i = 1; i < argc; ++i)
{
v.push_back(Strings::utf16_to_utf8(argv[i]));
}
return vcpkg_cmd_arguments::create_from_arg_sequence(v.data(), v.data() + v.size());
}
vcpkg_cmd_arguments vcpkg_cmd_arguments::create_from_arg_sequence(const std::string* arg_begin, const std::string* arg_end)
{
vcpkg_cmd_arguments args;
for (; arg_begin != arg_end; ++arg_begin)
{
std::string arg = *arg_begin;
if (arg.empty())
{
continue;
}
if (arg[0] == '-' && arg[1] != '-')
{
System::println(System::color::error, "Error: short options are not supported: %s", arg);
TrackProperty("error", "error short options are not supported");
exit(EXIT_FAILURE);
}
if (arg[0] == '-' && arg[1] == '-')
{
// command switch
if (arg == "--vcpkg-root")
{
++arg_begin;
parse_value(arg_begin, arg_end, "--vcpkg-root", args.vcpkg_root_dir);
continue;
}
if (arg == "--triplet")
{
++arg_begin;
parse_value(arg_begin, arg_end, "--triplet", args.target_triplet);
continue;
}
if (arg == "--debug")
{
parse_switch(opt_bool::enabled, "debug", args.debug);
continue;
}
if (arg == "--sendmetrics")
{
parse_switch(opt_bool::enabled, "sendmetrics", args.sendmetrics);
continue;
}
if (arg == "--printmetrics")
{
parse_switch(opt_bool::enabled, "printmetrics", args.printmetrics);
continue;
}
if (arg == "--no-sendmetrics")
{
parse_switch(opt_bool::disabled, "sendmetrics", args.sendmetrics);
continue;
}
if (arg == "--no-printmetrics")
{
parse_switch(opt_bool::disabled, "printmetrics", args.printmetrics);
continue;
}
args.optional_command_arguments.insert(arg);
continue;
}
if (args.command.empty())
{
args.command = arg;
}
else
{
args.command_arguments.push_back(arg);
}
}
return args;
}
std::unordered_set<std::string> vcpkg_cmd_arguments::check_and_get_optional_command_arguments(const std::vector<std::string>& valid_options) const
{
std::unordered_set<std::string> output;
auto options_copy = this->optional_command_arguments;
for (const std::string& option : valid_options)
{
auto it = options_copy.find(option);
if (it != options_copy.end())
{
output.insert(option);
options_copy.erase(it);
}
}
if (!options_copy.empty())
{
System::println(System::color::error, "Unknown option(s) for command '%s':", this->command);
for (const std::string& option : options_copy)
{
System::println(option.c_str());
}
exit(EXIT_FAILURE);
}
return output;
}
void vcpkg_cmd_arguments::check_max_arg_count(const size_t expected_arg_count) const
2016-09-19 11:50:08 +08:00
{
return check_max_arg_count(expected_arg_count, "");
}
void vcpkg_cmd_arguments::check_min_arg_count(const size_t expected_arg_count) const
{
return check_min_arg_count(expected_arg_count, "");
}
void vcpkg_cmd_arguments::check_exact_arg_count(const size_t expected_arg_count) const
{
return check_exact_arg_count(expected_arg_count, "");
}
void vcpkg_cmd_arguments::check_max_arg_count(const size_t expected_arg_count, const char* example_text) const
{
const size_t actual_arg_count = command_arguments.size();
if (actual_arg_count > expected_arg_count)
2016-09-19 11:50:08 +08:00
{
System::println(System::color::error, "Error: %s requires at most %u arguments, but %u were provided", this->command, expected_arg_count, actual_arg_count);
System::print(example_text);
exit(EXIT_FAILURE);
}
}
void vcpkg_cmd_arguments::check_min_arg_count(const size_t expected_arg_count, const char* example_text) const
{
const size_t actual_arg_count = command_arguments.size();
if (actual_arg_count < expected_arg_count)
{
System::println(System::color::error, "Error: %s requires at least %u arguments, but %u were provided", this->command, expected_arg_count, actual_arg_count);
System::print(example_text);
exit(EXIT_FAILURE);
}
}
void vcpkg_cmd_arguments::check_exact_arg_count(const size_t expected_arg_count, const char* example_text) const
{
const size_t actual_arg_count = command_arguments.size();
if (actual_arg_count != expected_arg_count)
{
System::println(System::color::error, "Error: %s requires %u arguments, but %u were provided", this->command, expected_arg_count, actual_arg_count);
System::print(example_text);
2016-09-19 11:50:08 +08:00
exit(EXIT_FAILURE);
}
}
std::vector<package_spec> vcpkg_cmd_arguments::parse_all_arguments_as_package_specs(const triplet& default_target_triplet, const char* example_text) const
2016-09-19 11:50:08 +08:00
{
size_t arg_count = command_arguments.size();
if (arg_count < 1)
{
System::println(System::color::error, "Error: %s requires one or more package specifiers", this->command);
if (example_text == nullptr)
print_example(Strings::format("%s zlib zlib:x64-windows curl boost", this->command).c_str());
else
print_example(example_text);
exit(EXIT_FAILURE);
}
std::vector<package_spec> specs;
specs.reserve(arg_count);
for (const std::string& command_argument : command_arguments)
{
expected<package_spec> current_spec = package_spec::from_string(command_argument, default_target_triplet);
2016-09-19 11:50:08 +08:00
if (auto spec = current_spec.get())
{
specs.push_back(std::move(*spec));
}
else
{
System::println(System::color::error, "Error: %s: %s", current_spec.error_code().message(), command_argument);
print_example(Strings::format("%s zlib:x64-windows", this->command).c_str());
exit(EXIT_FAILURE);
}
}
return specs;
}
}