blob: a20127231499e2322edec2ade5df842afd9f0e40 [file] [log] [blame]
Junxiao Shi64567bb2016-09-04 16:00:27 +00001/* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */
Junxiao Shi15902ef2017-08-11 22:58:35 +00002/*
Davide Pesaventoe0bae0f2018-02-17 22:07:52 -05003 * Copyright (c) 2014-2018, Regents of the University of California,
Junxiao Shi64567bb2016-09-04 16:00:27 +00004 * Arizona Board of Regents,
5 * Colorado State University,
6 * University Pierre & Marie Curie, Sorbonne University,
7 * Washington University in St. Louis,
8 * Beijing Institute of Technology,
9 * The University of Memphis.
10 *
11 * This file is part of NFD (Named Data Networking Forwarding Daemon).
12 * See AUTHORS.md for complete list of NFD authors and contributors.
13 *
14 * NFD is free software: you can redistribute it and/or modify it under the terms
15 * of the GNU General Public License as published by the Free Software Foundation,
16 * either version 3 of the License, or (at your option) any later version.
17 *
18 * NFD is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;
19 * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
20 * PURPOSE. See the GNU General Public License for more details.
21 *
22 * You should have received a copy of the GNU General Public License along with
23 * NFD, e.g., in COPYING.md file. If not, see <http://www.gnu.org/licenses/>.
24 */
25
26#include "command-parser.hpp"
Junxiao Shic143afe2016-09-20 13:04:51 +000027#include "format-helpers.hpp"
Davide Pesaventoe0bae0f2018-02-17 22:07:52 -050028
Junxiao Shi64567bb2016-09-04 16:00:27 +000029#include <ndn-cxx/util/logger.hpp>
30
31namespace nfd {
32namespace tools {
33namespace nfdc {
34
Junxiao Shic143afe2016-09-20 13:04:51 +000035NDN_LOG_INIT(nfdc.CommandParser);
36
37static_assert(std::is_same<std::underlying_type<AvailableIn>::type,
38 std::underlying_type<ParseMode>::type>::value,
39 "AvailableIn and ParseMode must be declared with same underlying type");
Junxiao Shi64567bb2016-09-04 16:00:27 +000040
41std::ostream&
42operator<<(std::ostream& os, AvailableIn modes)
43{
Junxiao Shic143afe2016-09-20 13:04:51 +000044 text::Separator sep("|");
45 if ((modes & AVAILABLE_IN_ONE_SHOT) != 0) {
46 os << sep << "one-shot";
47 }
48 if ((modes & AVAILABLE_IN_BATCH) != 0) {
49 os << sep << "batch";
Junxiao Shi64567bb2016-09-04 16:00:27 +000050 }
Junxiao Shi6c135622016-11-21 14:30:33 +000051 if ((modes & AVAILABLE_IN_HELP) == 0) {
52 os << sep << "hidden";
53 }
Junxiao Shi64567bb2016-09-04 16:00:27 +000054
Junxiao Shic143afe2016-09-20 13:04:51 +000055 if (sep.getCount() == 0) {
Junxiao Shi64567bb2016-09-04 16:00:27 +000056 os << "none";
57 }
58 return os;
59}
60
61std::ostream&
62operator<<(std::ostream& os, ParseMode mode)
63{
64 switch (mode) {
65 case ParseMode::ONE_SHOT:
66 return os << "one-shot";
67 case ParseMode::BATCH:
68 return os << "batch";
69 }
70 return os << static_cast<int>(mode);
71}
72
73CommandParser&
Junxiao Shic143afe2016-09-20 13:04:51 +000074CommandParser::addCommand(const CommandDefinition& def, const ExecuteCommand& execute,
75 std::underlying_type<AvailableIn>::type modes)
Junxiao Shi64567bb2016-09-04 16:00:27 +000076{
77 BOOST_ASSERT(modes != AVAILABLE_IN_NONE);
Davide Pesaventoe0bae0f2018-02-17 22:07:52 -050078
Junxiao Shic143afe2016-09-20 13:04:51 +000079 m_commands[{def.getNoun(), def.getVerb()}].reset(
80 new Command{def, execute, static_cast<AvailableIn>(modes)});
Junxiao Shi6c135622016-11-21 14:30:33 +000081
82 if ((modes & AVAILABLE_IN_HELP) != 0) {
83 m_commandOrder.push_back(m_commands.find({def.getNoun(), def.getVerb()}));
84 }
85
Junxiao Shi64567bb2016-09-04 16:00:27 +000086 return *this;
87}
88
89CommandParser&
90CommandParser::addAlias(const std::string& noun, const std::string& verb, const std::string& verb2)
91{
92 m_commands[{noun, verb2}] = m_commands.at({noun, verb});
93 return *this;
94}
95
Junxiao Shi6c135622016-11-21 14:30:33 +000096std::vector<const CommandDefinition*>
97CommandParser::listCommands(const std::string& noun, ParseMode mode) const
98{
99 std::vector<const CommandDefinition*> results;
Davide Pesaventoe0bae0f2018-02-17 22:07:52 -0500100 for (auto i : m_commandOrder) {
Junxiao Shi6c135622016-11-21 14:30:33 +0000101 const Command& command = *i->second;
102 if ((command.modes & static_cast<AvailableIn>(mode)) != 0 &&
103 (noun.empty() || noun == command.def.getNoun())) {
104 results.push_back(&command.def);
105 }
106 }
107 return results;
108}
109
Junxiao Shi737c43c2016-09-14 02:51:44 +0000110std::tuple<std::string, std::string, CommandArguments, ExecuteCommand>
Davide Pesavento2a588152018-02-19 18:10:03 -0500111CommandParser::parse(const std::vector<std::string>& tokens, ParseMode mode) const
Junxiao Shi64567bb2016-09-04 16:00:27 +0000112{
113 BOOST_ASSERT(mode == ParseMode::ONE_SHOT);
114
115 const std::string& noun = tokens.size() > 0 ? tokens[0] : "";
116 const std::string& verb = tokens.size() > 1 ? tokens[1] : "";
Junxiao Shi64567bb2016-09-04 16:00:27 +0000117
Davide Pesaventoe0bae0f2018-02-17 22:07:52 -0500118 NDN_LOG_TRACE("parse mode=" << mode << " noun=" << noun << " verb=" << verb);
119
Junxiao Shi64567bb2016-09-04 16:00:27 +0000120 auto i = m_commands.find({noun, verb});
Junxiao Shi64567bb2016-09-04 16:00:27 +0000121 if (i == m_commands.end() || (i->second->modes & static_cast<AvailableIn>(mode)) == 0) {
Davide Pesaventoe0bae0f2018-02-17 22:07:52 -0500122 BOOST_THROW_EXCEPTION(NoSuchCommandError(noun, verb));
Junxiao Shi64567bb2016-09-04 16:00:27 +0000123 }
124
125 const CommandDefinition& def = i->second->def;
Davide Pesaventoe0bae0f2018-02-17 22:07:52 -0500126 NDN_LOG_TRACE("found command noun=" << def.getNoun() << " verb=" << def.getVerb());
Junxiao Shi64567bb2016-09-04 16:00:27 +0000127
Davide Pesaventod2147442018-02-19 23:58:17 -0500128 size_t nConsumed = std::min<size_t>(2, tokens.size());
129 return std::make_tuple(def.getNoun(), def.getVerb(), def.parse(tokens, nConsumed), i->second->execute);
Junxiao Shi64567bb2016-09-04 16:00:27 +0000130}
131
132} // namespace nfdc
133} // namespace tools
134} // namespace nfd