#include "vw/slim/opts.h"

#include <cctype>
#include <cstdlib>
#include <utility>

namespace vw_slim
{
template <class T>
void find_opt(std::string const& command_line_args, std::string arg_name, std::vector<T>& out_values)
{
  // append space to search for '--quadratic '
  arg_name += ' ';

  // support -q ab -v -q cd
  for (size_t start = 0; start < command_line_args.size();)
  {
    auto idx = command_line_args.find(arg_name, start);
    if (idx == std::string::npos)
    {
      return;  // no more occurences found, exit
    }

    auto idx_after_arg = idx + arg_name.size();

    // skip all white space
    for (; idx_after_arg < command_line_args.size() && std::isspace(command_line_args[idx_after_arg]); ++idx_after_arg)
    { ; }

    if (idx_after_arg == command_line_args.size()) { return; }

    if (command_line_args[idx_after_arg] == '-' &&
        // make sure we allow -5.2 (negative numbers)
        !(idx_after_arg + 1 < command_line_args.size() &&
            (command_line_args[idx_after_arg + 1] >= '0' && command_line_args[idx_after_arg + 1] <= '9')))
    {
      start = idx_after_arg;
      continue;  // next option found
    }

    // find next non-white space character
    auto idx_after_value = idx_after_arg;
    while (idx_after_value < command_line_args.size() && !std::isspace(command_line_args[idx_after_value]))
    { ++idx_after_value; }

    auto value_size = idx_after_value - idx_after_arg;
    if (value_size > 0)
    {
      std::string args = command_line_args.substr(idx_after_arg, value_size);
      out_values.emplace_back(args.begin(), args.end());
    }
    start = idx_after_arg + 1;
  }
}

template void find_opt<std::string>(
    std::string const& command_line_args, std::string arg_name, std::vector<std::string>& out_values);

template void find_opt<std::vector<VW::namespace_index>>(std::string const& command_line_args, std::string arg_name,
    std::vector<std::vector<VW::namespace_index>>& out_values);

std::vector<std::string> find_opt(std::string const& command_line_args, std::string arg_name)
{
  std::vector<std::string> values;
  find_opt(command_line_args, std::move(arg_name), values);
  return values;
}

template <typename T, typename S, S (*F)(const char*)>
bool find_opt_parse(std::string const& command_line_args, std::string arg_name, T& value)
{
  std::vector<std::string> opts = find_opt(command_line_args, std::move(arg_name));

  if (opts.size() != 1) { return false; }

  value = (T)F(opts[0].c_str());

  return true;
}

bool find_opt_float(std::string const& command_line_args, std::string arg_name, float& value)
{
  return find_opt_parse<float, double, atof>(command_line_args, std::move(arg_name), value);
}

bool find_opt_int(std::string const& command_line_args, std::string arg_name, int& value)
{
  return find_opt_parse<int, int, atoi>(command_line_args, std::move(arg_name), value);
}
}  // namespace vw_slim