mxlib
c++ tools for analyzing astronomical data and other tasks by Jared R. Males. [git repo]
Loading...
Searching...
No Matches
clOptions.cpp
Go to the documentation of this file.
1/** \file clOptions.cpp
2 * \author Jared R. Males
3 * \brief Implementatino of a command line parser
4 *
5 * \ingroup mxApp_files
6 */
7
8//***********************************************************************//
9// Copyright 2021 Jared R. Males (jaredmales@gmail.com)
10//
11// This file is part of mxlib.
12//
13// mxlib is free software: you can redistribute it and/or modify
14// it under the terms of the GNU General Public License as published by
15// the Free Software Foundation, either version 3 of the License, or
16// (at your option) any later version.
17//
18// mxlib is distributed in the hope that it will be useful,
19// but WITHOUT ANY WARRANTY; without even the implied warranty of
20// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
21// GNU General Public License for more details.
22//
23// You should have received a copy of the GNU General Public License
24// along with mxlib. If not, see <http://www.gnu.org/licenses/>.
25//***********************************************************************//
26
27#include "app/clOptions.hpp"
28
29namespace mx
30{
31namespace app
32{
33
34static const char *falseStr = "false";
35static const char *trueStr = "true";
36static const char *blankStr = "";
37
38static option::ArgStatus Arg_Required( const option::Option &option, bool msg )
39{
40 static_cast<void>( msg );
41
42 if( option.arg != 0 )
43 return option::ARG_OK;
44
45 return option::ARG_ILLEGAL;
46}
47
49{
50 if( options )
51 delete[] options;
52 if( buffer )
53 delete[] buffer;
54}
55
57{
58 if( options )
59 delete[] options;
60 options = 0;
61
62 if( buffer )
63 delete[] buffer;
64 buffer = 0;
65
66 map.clear();
67 typeMap.clear();
68 descriptions.clear();
69}
70
71void clOptions::add( const std::string &optName, const char *const shortOpt, const char *const longOpt, int argT )
72{
73 mapIterator it;
74 it = map.find( optName );
75
76 if( it == map.end() )
77 {
78 if( argT == argType::Optional )
79 {
80 descriptions.push_back( { nOpts, argT, shortOpt, longOpt, option::Arg::Optional, "" } );
81 }
82 else if( argT == argType::Required )
83 {
84 descriptions.push_back( { nOpts, argT, shortOpt, longOpt, Arg_Required, "" } );
85 }
86 else
87 {
88 descriptions.push_back( { nOpts, argT, shortOpt, longOpt, option::Arg::None, "" } );
89 }
90
91 map.insert( { optName, nOpts } );
92 typeMap.insert( { optName, argT } );
93
94 ++nOpts;
95 return;
96 }
97}
98
99void clOptions::parse( int argc, char **argv, std::vector<std::string> *nonOptions )
100{
101 argc -= ( argc > 0 );
102 argv += ( argc > 0 ); // skip program name argv[0] if present
103
104 // If not already done, we push the unknown catcher and the termination descriptor.
105 if( descriptions.back().index != 0 )
106 {
107 descriptions.push_back(
108 { nOpts, 0, "", "", option::Arg::None, "" } ); // This is inserted to catch unknown options
109 descriptions.push_back( { 0, 0, 0, 0, 0, 0 } );
110 }
111
112 // Now allocate.
113 option::Stats stats( descriptions.data(), argc, argv );
114 options = new option::Option[stats.options_max];
115 buffer = new option::Option[stats.buffer_max];
116
117 option::Parser parse( false, descriptions.data(), argc, argv, options, buffer );
118
119 if( nonOptions )
120 {
121 nonOptions->resize( parse.nonOptionsCount() );
122
123 for( int i = 0; i < parse.nonOptionsCount(); ++i )
124 {
125 ( *nonOptions )[i] = parse.nonOption( i );
126 }
127 }
128}
129
130const char *clOptions::operator[]( const std::string &key )
131{
132 mapIterator it = map.find( key );
133
134 if( it == map.end() )
135 {
136 std::cerr << "oh no\n";
137 return 0;
138 }
139
140 int typeDesc = typeMap[key];
141
142 // If this option is not pressent, either return the opposite T/F condition or blank
143 if( options[it->second].type() == argType::None )
144 {
145 if( typeDesc == argType::False )
146 return trueStr;
147 if( typeDesc == argType::True )
148 return falseStr;
149 return blankStr;
150 }
151
152 if( typeDesc == argType::False || typeDesc == argType::True )
153 {
154 if( options[it->second].last()->type() == argType::False )
155 return falseStr;
156 else
157 return trueStr;
158 }
159
160 if( options[it->second].arg == 0 )
161 return blankStr;
162
163 return options[it->second].last()->arg;
164}
165
166int clOptions::count( const std::string &key )
167{
168 mapIterator it = map.find( key );
169
170 if( it == map.end() )
171 return -1;
172
173 return options[it->second].count();
174}
175
176void clOptions::getAll( std::vector<std::string> &args, const std::string &key )
177{
178 mapIterator it = map.find( key );
179
180 if( it == map.end() )
181 {
182 std::cerr << "oh no\n";
183 return;
184 }
185
186 int typeDesc = typeMap[key];
187
188 // If this option is not present, either return the opposite T/F condition or blank
189 if( options[it->second].type() == argType::None )
190 {
191
192 if( typeDesc == argType::False )
193 {
194 args.resize( 1 );
195 args[0] = trueStr;
196 return;
197 }
198
199 if( typeDesc == argType::True )
200 {
201 args.resize( 1 );
202 args[0] = falseStr;
203 return;
204 }
205
206 args.clear();
207
208 return;
209 }
210
211 if( typeDesc == argType::False || typeDesc == argType::True )
212 {
213 args.resize( 1 );
214
215 if( options[it->second].last()->type() == argType::False )
216 args[0] = falseStr;
217 else
218 args[0] = trueStr;
219 return;
220 }
221
222 if( options[it->second].arg == 0 )
223 {
224 args.clear();
225 return;
226 }
227
228 int N = options[it->second].count();
229
230 args.resize( N );
231
232 int i = 0;
233
234 for( option::Option *opt = options[it->second]; opt != NULL && i < N; opt = opt->next() )
235 {
236 args[i] = opt->arg;
237 ++i;
238 }
239}
240
241bool clOptions::optSet( const std::string &key )
242{
243 mapIterator it = map.find( key );
244
245 if( it == map.end() )
246 return false; // Not found --> e.g. if neither command line short nor long option set.
247
248 if( options[it->second].type() != argType::None )
249 return true;
250 return false;
251};
252
254{
255 // The dummy description to catch unknown options is inserted at position nOpts.
256 return options[nOpts].count();
257}
258
259int clOptions::unknown( std::vector<std::string> &unk )
260{
261 unk.clear();
262
263 if( numUnknown() == 0 )
264 return 0;
265
266 for( option::Option *opt = options[nOpts]; opt != NULL; opt = opt->next() )
267 {
268 unk.push_back( opt->name );
269 }
270
271 return 0;
272}
273
274} // namespace app
275} // namespace mx
A command line parser.
The mxlib c++ namespace.
Definition mxError.hpp:106
void parse(int argc, char **argv, std::vector< std::string > *nonOptions=0)
Parse the command line.
Definition clOptions.cpp:99
const char * operator[](const std::string &key)
Get the value of the option, if present.
void clear()
Clear the memory held by this object.
Definition clOptions.cpp:56
int numUnknown()
Get the number of unknown options found by the parser.
void add(const std::string &optName, const char *const shortOpt, const char *const longOpt, int argT)
Add a command line option target.
Definition clOptions.cpp:71
int count(const std::string &key)
Get the number of times the option was set on the command line.
unsigned int nOpts
The number of options added.
Definition clOptions.hpp:70
void getAll(std::vector< std::string > &args, const std::string &key)
Fill a vector of strings with the arguments supplied for key, last to first.
~clOptions()
D'tor. Deletes the options and buffer pointers.
Definition clOptions.cpp:48
int unknown(std::vector< std::string > &unk)
Get a vector of the unknown options found by the parser.
bool optSet(const std::string &key)
Test whether this option was set.