blob: 1fe7a735ab3b19fb35195a78719fc7a1b5e991f0 [file] [log] [blame]
/** @file
Nest commands (for command line processing).
@section license License
Licensed to the Apache Software Foundation (ASF) under one
or more contributor license agreements. See the NOTICE file
distributed with this work for additional information
regarding copyright ownership. The ASF licenses this file
to you under the Apache License, Version 2.0 (the
"License"); you may not use this file except in compliance
with the License. You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
#include "Command.h"
#include <new>
#include <algorithm>
#include <cstring>
#include <sstream>
namespace ts
{
int CommandTable::_opt_idx = 0;
static const std::string LEADING(":- ");
// Error message functions.
ts::Errata
ERR_COMMAND_TAG_NOT_FOUND(char const *tag)
{
std::ostringstream s;
s << "Command tag " << tag << " not found";
return ts::Errata(s.str());
}
CommandTable::Command::Command() {}
CommandTable::Command::Command(std::string const &name, std::string const &help) : _name(name), _help(help) {}
CommandTable::Command::Command(std::string const &name, std::string const &help, LeafAction const &f) : _name(name), _help(help)
{
_action = f;
}
CommandTable::Command::Command(std::string const &name, std::string const &help, NullaryAction const &f) : _name(name), _help(help)
{
_action = f;
}
CommandTable::Command::~Command() {}
CommandTable::Command &
CommandTable::Command::subCommand(std::string const &name, std::string const &help, LeafAction const &f)
{
_group.push_back(Command(name, help, f));
return _group.back();
}
CommandTable::Command &
CommandTable::Command::subCommand(std::string const &name, std::string const &help, NullaryAction const &f)
{
_group.push_back(Command(name, help, f));
return _group.back();
}
auto
CommandTable::Command::subCommand(std::string const &name, std::string const &help) -> self &
{
_group.push_back(Command(name, help));
return _group.back();
}
ts::Errata
CommandTable::Command::invoke(int argc, char *argv[])
{
ts::Errata zret;
if (_action.is_leaf()) {
zret = _action.invoke(argc - CommandTable::_opt_idx, argv + CommandTable::_opt_idx);
} else if (CommandTable::_opt_idx >= argc || argv[CommandTable::_opt_idx][0] == '-') {
if (_action.is_nullary()) {
zret = _action.invoke();
} else {
std::ostringstream s;
s << "Incomplete command, additional keyword required";
s << std::endl;
this->helpMessage(0, nullptr, s, LEADING);
zret.push(s.str());
}
} else {
char const *tag = argv[CommandTable::_opt_idx];
auto spot = std::find_if(_group.begin(), _group.end(),
[tag](CommandGroup::value_type const &elt) { return 0 == strcasecmp(tag, elt._name.c_str()); });
if (spot != _group.end()) {
++CommandTable::_opt_idx;
zret = spot->invoke(argc, argv);
} else {
zret = ERR_COMMAND_TAG_NOT_FOUND(tag);
}
}
return zret;
}
void
CommandTable::Command::helpMessage(int argc, char *argv[], std::ostream &out, std::string const &prefix) const
{
if (CommandTable::_opt_idx >= argc || argv[CommandTable::_opt_idx][0] == '-') {
// Tail of command keywords, start listing
if (_name.empty()) { // root command group, don't print for that.
for (Command const &c : _group) {
c.helpMessage(argc, argv, out, prefix);
}
} else {
out << prefix << _name << ": " << _help << std::endl;
for (Command const &c : _group) {
c.helpMessage(argc, argv, out, " " + prefix);
}
}
} else {
char const *tag = argv[CommandTable::_opt_idx];
auto spot = std::find_if(_group.begin(), _group.end(),
[tag](CommandGroup::value_type const &elt) { return 0 == strcasecmp(tag, elt._name.c_str()); });
if (spot != _group.end()) {
++CommandTable::_opt_idx;
spot->helpMessage(argc, argv, out, prefix);
} else {
out << ERR_COMMAND_TAG_NOT_FOUND(tag) << std::endl;
}
}
}
CommandTable::CommandTable() {}
auto
CommandTable::add(std::string const &name, std::string const &help) -> Command &
{
return _top.subCommand(name, help);
}
auto
CommandTable::add(std::string const &name, std::string const &help, LeafAction const &f) -> Command &
{
return _top.subCommand(name, help, f);
}
auto
CommandTable::add(std::string const &name, std::string const &help, NullaryAction const &f) -> Command &
{
return _top.subCommand(name, help, f);
}
ts::Errata
CommandTable::invoke(int argc, char *argv[])
{
return _top.invoke(argc, argv);
}
// This is basically cloned from invoke(), need to find how to do some unification.
void
CommandTable::helpMessage(int argc, char *argv[]) const
{
_opt_idx = 0;
std::cerr << "Command tree" << std::endl;
_top.helpMessage(argc, argv, std::cerr, LEADING);
}
} // namespace ts