blob: 136f5c43a3d298ab80dc0314852c1825ef2c47d8 [file] [log] [blame]
Junxiao Shi64567bb2016-09-04 16:00:27 +00001/* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */
2/**
3 * Copyright (c) 2014-2016, Regents of the University of California,
4 * 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#ifndef NFD_TOOLS_NFDC_COMMAND_DEFINITION_HPP
27#define NFD_TOOLS_NFDC_COMMAND_DEFINITION_HPP
28
29#include "command-arguments.hpp"
30
31namespace nfd {
32namespace tools {
33namespace nfdc {
34
35/** \brief indicates argument value type
36 */
37enum class ArgValueType {
38 /** \brief boolean argument without value
39 *
40 * The argument appears in CommandArguments as bool value 'true'.
41 * It must not be declared as positional.
42 */
43 NONE,
44
45 /** \brief any arguments
46 *
47 * The argument appears in CommandArguments as std::vector<std::string>.
48 * It must be declared as positional, and will consume all subsequent tokens.
49 */
50 ANY,
51
52 /** \brief non-negative integer
53 *
54 * The argument appears in CommandArguments as uint64_t.
55 * Acceptable input range is [0, std::numeric_limits<int64_t>::max()].
56 */
57 UNSIGNED,
58
59 /** \brief arbitrary string
60 *
61 * The argument appears in CommandArguments as std::string.
62 */
63 STRING,
64
65 /** \brief report format 'xml' or 'text'
66 *
67 * The argument appears in CommandArguments as nfd::tools::nfdc::ReportFormat.
68 */
69 REPORT_FORMAT,
70
71 /** \brief Name prefix
72 *
73 * The argument appears in CommandArguments as ndn::Name.
74 */
75 NAME,
76
77 /** \brief FaceUri
78 *
79 * The argument appears in CommandArguments as ndn::util::FaceUri.
80 */
81 FACE_URI,
82
83 /** \brief FaceId or FaceUri
84 *
85 * The argument appears in CommandArguments as either uint64_t or ndn::util::FaceUri.
86 */
87 FACE_ID_OR_URI,
88
89 /** \brief face persistency 'persistent' or 'permanent'
90 *
91 * The argument appears in CommandArguments as ndn::nfd::FacePersistency.
92 */
93 FACE_PERSISTENCY
94};
95
96std::ostream&
97operator<<(std::ostream& os, ArgValueType vt);
98
99/** \brief indicates whether an argument is required
100 */
101enum class Required {
102 NO = false, ///< argument is optional
103 YES = true ///< argument is required
104};
105
106/** \brief indicates whether an argument can be specified as positional
107 */
108enum class Positional {
109 NO = false, ///< argument must be named
110 YES = true ///< argument can be specified as positional
111};
112
113/** \brief declares semantics of a command
114 */
115class CommandDefinition
116{
117public:
118 class Error : public std::invalid_argument
119 {
120 public:
121 explicit
122 Error(const std::string& what)
123 : std::invalid_argument(what)
124 {
125 }
126 };
127
128 CommandDefinition(const std::string& noun, const std::string& verb);
129
130 ~CommandDefinition();
131
132 const std::string
133 getNoun() const
134 {
135 return m_noun;
136 }
137
138 const std::string
139 getVerb() const
140 {
141 return m_verb;
142 }
143
144public: // help
145 /** \return one-line synopsis
146 */
147 const std::string&
148 getSynopsis() const
149 {
150 return m_synopsis;
151 }
152
153 /** \brief set one-line synopsis
154 */
155 CommandDefinition&
156 setSynopsis(const std::string& synopsis)
157 {
158 m_synopsis = synopsis;
159 return *this;
160 }
161
162public: // arguments
163 /** \brief declare an argument
164 * \param name argument name, must be unique
165 * \param valueType argument value type
166 * \param isRequired whether the argument is required
167 * \param allowPositional whether the argument value can be specified as positional
168 * \param metavar displayed argument value placeholder
169 */
170 CommandDefinition&
171 addArg(const std::string& name, ArgValueType valueType,
172 Required isRequired = Required::NO,
173 Positional allowPositional = Positional::NO,
174 const std::string& metavar = "");
175
176 /** \brief parse a command line
177 * \param tokens command line tokens
178 * \param start command line start position, after noun and verb
179 * \throw Error command line is invalid
180 */
181 CommandArguments
182 parse(const std::vector<std::string>& tokens, size_t start = 0) const;
183
184private:
185 boost::any
186 parseValue(ArgValueType valueType, const std::string& token) const;
187
188private:
189 std::string m_noun;
190 std::string m_verb;
191
192 std::string m_synopsis;
193
194 struct Arg
195 {
196 std::string name;
197 ArgValueType valueType;
198 bool isRequired;
199 std::string metavar;
200 };
201 typedef std::map<std::string, Arg> ArgMap;
202 ArgMap m_args;
203 std::set<std::string> m_requiredArgs;
204 std::vector<std::string> m_positionalArgs;
205};
206
207} // namespace nfdc
208} // namespace tools
209} // namespace nfd
210
211#endif // NFD_TOOLS_NFDC_COMMAND_DEFINITION_HPP