source: src/UIElements/CommandLineUI/CommandLineParser.cpp@ d6203a

Action_Thermostats Add_AtomRandomPerturbation Add_FitFragmentPartialChargesAction Add_RotateAroundBondAction Add_SelectAtomByNameAction Added_ParseSaveFragmentResults AddingActions_SaveParseParticleParameters Adding_Graph_to_ChangeBondActions Adding_MD_integration_tests Adding_ParticleName_to_Atom Adding_StructOpt_integration_tests AtomFragments Automaking_mpqc_open AutomationFragmentation_failures Candidate_v1.5.4 Candidate_v1.6.0 Candidate_v1.6.1 Candidate_v1.7.0 ChangeBugEmailaddress ChangingTestPorts ChemicalSpaceEvaluator CombiningParticlePotentialParsing Combining_Subpackages Debian_Package_split Debian_package_split_molecuildergui_only Disabling_MemDebug Docu_Python_wait EmpiricalPotential_contain_HomologyGraph EmpiricalPotential_contain_HomologyGraph_documentation Enable_parallel_make_install Enhance_userguide Enhanced_StructuralOptimization Enhanced_StructuralOptimization_continued Example_ManyWaysToTranslateAtom Exclude_Hydrogens_annealWithBondGraph FitPartialCharges_GlobalError Fix_BoundInBox_CenterInBox_MoleculeActions Fix_ChargeSampling_PBC Fix_ChronosMutex Fix_FitPartialCharges Fix_FitPotential_needs_atomicnumbers Fix_ForceAnnealing Fix_IndependentFragmentGrids Fix_ParseParticles Fix_ParseParticles_split_forward_backward_Actions Fix_PopActions Fix_QtFragmentList_sorted_selection Fix_Restrictedkeyset_FragmentMolecule Fix_StatusMsg Fix_StepWorldTime_single_argument Fix_Verbose_Codepatterns Fix_fitting_potentials Fixes ForceAnnealing_goodresults ForceAnnealing_oldresults ForceAnnealing_tocheck ForceAnnealing_with_BondGraph ForceAnnealing_with_BondGraph_continued ForceAnnealing_with_BondGraph_continued_betteresults ForceAnnealing_with_BondGraph_contraction-expansion FragmentAction_writes_AtomFragments FragmentMolecule_checks_bonddegrees GeometryObjects Gui_Fixes Gui_displays_atomic_force_velocity ImplicitCharges IndependentFragmentGrids IndependentFragmentGrids_IndividualZeroInstances IndependentFragmentGrids_IntegrationTest IndependentFragmentGrids_Sole_NN_Calculation JobMarket_RobustOnKillsSegFaults JobMarket_StableWorkerPool JobMarket_unresolvable_hostname_fix MoreRobust_FragmentAutomation ODR_violation_mpqc_open PartialCharges_OrthogonalSummation PdbParser_setsAtomName PythonUI_with_named_parameters QtGui_reactivate_TimeChanged_changes Recreated_GuiChecks Rewrite_FitPartialCharges RotateToPrincipalAxisSystem_UndoRedo SaturateAtoms_findBestMatching SaturateAtoms_singleDegree StoppableMakroAction Subpackage_CodePatterns Subpackage_JobMarket Subpackage_LinearAlgebra Subpackage_levmar Subpackage_mpqc_open Subpackage_vmg Switchable_LogView ThirdParty_MPQC_rebuilt_buildsystem TrajectoryDependenant_MaxOrder TremoloParser_IncreasedPrecision TremoloParser_MultipleTimesteps TremoloParser_setsAtomName Ubuntu_1604_changes stable
Last change on this file since d6203a was 7d9416, checked in by Frederik Heber <heber@…>, 13 years ago

Turned query<Box> into query<RealSpaceMatrix>

  • Property mode set to 100644
File size: 20.2 KB
RevLine 
[e4afb4]1/*
2 * Project: MoleCuilder
3 * Description: creates and alters molecular systems
[0aa122]4 * Copyright (C) 2010-2012 University of Bonn. All rights reserved.
[e4afb4]5 * Please see the LICENSE file or "Copyright notice" in builder.cpp for details.
6 */
7
8/*
9 * CommandLineParser.cpp
10 *
11 * Created on: May 8, 2010
12 * Author: heber
13 */
14
15// include config.h
16#ifdef HAVE_CONFIG_H
17#include <config.h>
18#endif
19
[ad011c]20#include "CodePatterns/MemDebug.hpp"
[e4afb4]21
22#include <boost/filesystem.hpp>
23#include <boost/program_options.hpp>
24#include <fstream>
25#include <iostream>
[5a8f38]26#include <set>
[e4afb4]27#include <map>
28
29#include "Actions/Action.hpp"
30#include "Actions/ActionRegistry.hpp"
[3139b2]31#include "Actions/ActionTrait.hpp"
[e4afb4]32#include "Actions/OptionRegistry.hpp"
33#include "Actions/OptionTrait.hpp"
34#include "Actions/Values.hpp"
[ad011c]35#include "CodePatterns/Log.hpp"
36#include "CodePatterns/Verbose.hpp"
[e4afb4]37#include "CommandLineParser.hpp"
[3a21a7]38#include "CommandLineParser_validate.hpp"
[e4afb4]39
[ad011c]40#include "CodePatterns/Singleton_impl.hpp"
[e4afb4]41
[ce7fdc]42using namespace MoleCuilder;
43
[e4afb4]44class element;
45
46/** Constructor of class CommandLineParser.
47 *
48 */
49CommandLineParser::CommandLineParser() :
50 analysis("Analysis options"),
51 atom("Atom options"),
52 command("Command options"),
[a88452]53 fill("fill options"),
[e4afb4]54 fragmentation("Fragmentation options"),
[d09093]55 graph("Graph options"),
[e4afb4]56 molecule("Molecule options"),
[d09093]57 options("Secondary options"),
[e4afb4]58 parser("Parser options"),
59 selection("Selection options"),
60 tesselation("Tesselation options"),
[d09093]61 world("World options")
[e4afb4]62{
63 // put all options lists into a lookup
64 CmdParserLookup["analysis"] = &analysis;
65 CmdParserLookup["atom"] = &atom;
66 CmdParserLookup["command"] = &command;
67 CmdParserLookup["edit"] = &edit;
[a88452]68 CmdParserLookup["fill"] = &fill;
[e4afb4]69 CmdParserLookup["fragmentation"] = &fragmentation;
[d09093]70 CmdParserLookup["graph"] = &graph;
71 CmdParserLookup["options"] = &options;
[e4afb4]72 CmdParserLookup["molecule"] = &molecule;
73 CmdParserLookup["parser"] = &parser;
74 CmdParserLookup["selection"] = &selection;
75 CmdParserLookup["tesselation"] = &tesselation;
76 CmdParserLookup["world"] = &world;
77}
78
79/** Destructor of class CommandLineParser.
80 *
81 */
82CommandLineParser::~CommandLineParser()
83{}
84
85/** Initializes command arguments to accept.
86 * Goes through ActionRegistry and puts all actions therein into the map.
87 */
88void CommandLineParser::InitializeCommandArguments()
89{
[5a8f38]90 // we need a list of already added options, otherwise we get ambigious exceptions
91 std::set<std::string> AlreadyAddedOptionNames;
92
[e4afb4]93 ActionRegistry &AR = ActionRegistry::getInstance();
94 bool ActionAlreadyAdded_flag = false;
95 for (ActionRegistry::const_iterator actioniter = AR.getBeginIter(); actioniter != AR.getEndIter(); ++actioniter) {
96 ActionAlreadyAdded_flag = false;
97 Action* const currentAction = actioniter->second;
98 //std::cout << "Current Action to initialize is: " << actioniter->first << std::endl;
99
[3139b2]100 for (ActionTrait::options_const_iterator optioniter = currentAction->Traits.getBeginIter();
[e4afb4]101 optioniter != currentAction->Traits.getEndIter();
102 ++optioniter) {
103 if (optioniter->first == actioniter->first)
104 ActionAlreadyAdded_flag = true;
105 ASSERT( OptionRegistry::getInstance().isOptionPresentByName(optioniter->first),
[5a8f38]106 "CommandLineParser::Init() - Option "+optioniter->first+" not present in OptionRegistry." );
[e4afb4]107 const OptionTrait* const currentOption = OptionRegistry::getInstance().getOptionByName(optioniter->first);
108
[5a8f38]109 if (AlreadyAddedOptionNames.find(optioniter->first) == AlreadyAddedOptionNames.end()) {
110 // add the option
111// std::cout << "Registering Option "
112// << currentOption->getName()
113// << " with type '" << currentOption->getTypeName() << "' "
114// << " with description '" << currentOption->getDescription() << "' ";
115// if (currentOption->hasShortForm())
116// std::cout << ", with short form " << currentOption->getShortForm();
117// else
118// std::cout << ", with no short form ";
119// if (currentOption->hasDefaultValue())
120// std::cout << ", with default value " << currentOption->getDefaultValue();
121// else
122// std::cout << ", with no default value ";
123// std::cout << std::endl;
124
125 AddOptionToParser(currentOption, (CmdParserLookup["options"]));
126
127 AlreadyAddedOptionNames.insert(optioniter->first);
128 } else {
129// std::cout << "Option " << currentOption->getName() << " already registered." << std::endl;
130 }
[e4afb4]131 }
132
133 if (!ActionAlreadyAdded_flag) {
134 // add the action
[ad7270]135// std::cout << "Registering Action "
136// << currentAction->Traits.getName()
137// << " in menu " << currentAction->Traits.getMenuName()
138// << " with type '" << currentAction->Traits.getTypeName() << "' "
139// << " with description '" << currentAction->Traits.getDescription() << "' ";
140// if (currentAction->Traits.hasShortForm())
141// std::cout << ", with short form " << currentAction->Traits.getShortForm();
142// else
143// std::cout << ", with no short form ";
144// if (currentAction->Traits.hasDefaultValue())
145// std::cout << ", with default value " << currentAction->Traits.getDefaultValue();
146// else
147// std::cout << ", with no default value ";
148// std::cout << std::endl;
[e4afb4]149
150 ASSERT(CmdParserLookup.find(currentAction->Traits.getMenuName()) != CmdParserLookup.end(),
151 "CommandLineParser: boost::program_options::options_description for this Action not present.");
152 AddOptionToParser(dynamic_cast<const OptionTrait * const>(&(currentAction->Traits)),(CmdParserLookup[currentAction->Traits.getMenuName()]));
153 }
154 }
155 // note: positioning is not important on the command line
156}
157
158/** Adds an Action or Option to the CommandLineParser.
159 * Note that Action is derived from Option(Trait)
160 *
161 * This ugly switch function is necessary because of the compile-time problem:
162 * po::value<T> has to be instantiated at compile-time however we do know the type not until run-time.
163 * Not even a templated function like po::value<T> getProgramOptionValuefromType() does help, specialized
164 * to each available type, as the signatures of all the functions differ. Hence, they cannot not put into
165 * one type_info -> po::value<T> map ...
166 *
167 * \param *currentOption pointer to Action/Option to add
168 * \param *OptionList program_options list to add to
169 */
170void CommandLineParser::AddOptionToParser(const OptionTrait * const currentOption, po::options_description* OptionList)
171{
172 // check whether dynamic_cast in Init() suceeded
173 ASSERT(currentOption != NULL, "CommandLineParser::AddOptionToParser() - currentOption is NULL!");
174 // add other options
[ec098d]175// std::cout << "Adding Action " << currentOption->getName() << " with type "
176// << currentOption->getType()->name() << ", " << (currentOption->hasDefaultValue() ? "with" : "without")
177// << " default value, and KeyandShortform " << currentOption->getKeyAndShortForm()
178// << " to CommandLineParser." << std::endl;
[e4afb4]179 switch(TypeToEnums.getEnumforType(currentOption->getType())) {
180 default:
181 case TypeEnumContainer::NoneType:
182 OptionList->add_options()
183 (currentOption->getKeyAndShortForm().c_str(), currentOption->getDescription().c_str())
184 ;
185 break;
186 case TypeEnumContainer::BooleanType:
187 OptionList->add_options()
188 (currentOption->getKeyAndShortForm().c_str(),
189 currentOption->hasDefaultValue() ?
190 po::value < bool >()->default_value(boost::lexical_cast<int>(currentOption->getDefaultValue().c_str())) :
191 po::value < bool >(),
192 currentOption->getDescription().c_str())
193 ;
194 break;
195 case TypeEnumContainer::FileType:
196 OptionList->add_options()
197 (currentOption->getKeyAndShortForm().c_str(),
198// currentOption->hasDefaultValue() ?
199// po::value < boost::filesystem::path >()->default_value(boost::lexical_cast<boost::filesystem::path>(currentOption->getDefaultValue().c_str())) :
200 po::value < boost::filesystem::path >(),
201 currentOption->getDescription().c_str())
202 ;
203 break;
204 case TypeEnumContainer::ListOfFilesType:
205 OptionList->add_options()
206 (currentOption->getKeyAndShortForm().c_str(),
207// currentOption->hasDefaultValue() ?
208// po::value < std::vector<boost::filesystem::path> >()->default_value(boost::lexical_cast< std::vector<boost::filesystem::path> >(currentOption->getDefaultValue().c_str())) :
209 po::value < std::vector<boost::filesystem::path> >()->multitoken(),
210 currentOption->getDescription().c_str())
211 ;
212 break;
213 case TypeEnumContainer::IntegerType:
214 OptionList->add_options()
215 (currentOption->getKeyAndShortForm().c_str(),
216 currentOption->hasDefaultValue() ?
217 po::value < int >()->default_value(boost::lexical_cast<int>(currentOption->getDefaultValue().c_str())) :
218 po::value < int >(),
219 currentOption->getDescription().c_str())
220 ;
221 break;
222 case TypeEnumContainer::ListOfIntegersType:
223 OptionList->add_options()
224 (currentOption->getKeyAndShortForm().c_str(),
225// currentOption->hasDefaultValue() ?
226// po::value < std::vector<int> >()->default_value(boost::lexical_cast< std::vector<int> >(currentOption->getDefaultValue().c_str())) :
227 po::value < std::vector<int> >()->multitoken(),
228 currentOption->getDescription().c_str())
229 ;
230 break;
[838cd0]231 case TypeEnumContainer::UnsignedIntegerType:
232 OptionList->add_options()
233 (currentOption->getKeyAndShortForm().c_str(),
234 currentOption->hasDefaultValue() ?
235 po::value < unsigned int >()->default_value(boost::lexical_cast<unsigned int>(currentOption->getDefaultValue().c_str())) :
236 po::value < unsigned int >(),
237 currentOption->getDescription().c_str())
238 ;
239 break;
[12948c]240 case TypeEnumContainer::ListOfUnsignedIntegersType:
241 OptionList->add_options()
242 (currentOption->getKeyAndShortForm().c_str(),
243// currentOption->hasDefaultValue() ?
244// po::value < std::vector<unsigned int> >()->default_value(boost::lexical_cast< std::vector<unsigned int> >(currentOption->getDefaultValue().c_str())) :
245 po::value < std::vector<unsigned int> >()->multitoken(),
246 currentOption->getDescription().c_str())
247 ;
248 break;
[e4afb4]249 case TypeEnumContainer::DoubleType:
250 OptionList->add_options()
251 (currentOption->getKeyAndShortForm().c_str(),
252 currentOption->hasDefaultValue() ?
253 po::value < double >()->default_value(boost::lexical_cast<double>(currentOption->getDefaultValue().c_str())) :
254 po::value < double >(),
255 currentOption->getDescription().c_str())
256 ;
257 break;
258 case TypeEnumContainer::ListOfDoublesType:
259 OptionList->add_options()
260 (currentOption->getKeyAndShortForm().c_str(),
261// currentOption->hasDefaultValue() ?
262// po::value < std::vector<double> >()->default_value(boost::lexical_cast< std::vector<double> >(currentOption->getDefaultValue().c_str())) :
263 po::value < std::vector<double> >()->multitoken(),
264 currentOption->getDescription().c_str())
265 ;
266 break;
267 case TypeEnumContainer::StringType:
268 OptionList->add_options()
269 (currentOption->getKeyAndShortForm().c_str(),
270 currentOption->hasDefaultValue() ?
271 po::value < std::string >()->default_value(currentOption->getDefaultValue()) :
272 po::value < std::string >(),
273 currentOption->getDescription().c_str())
274 ;
275 break;
276 case TypeEnumContainer::ListOfStringsType:
277 OptionList->add_options()
278 (currentOption->getKeyAndShortForm().c_str(),
279// currentOption->hasDefaultValue() ?
280// po::value < std::vector<std::string> >()->default_value(boost::lexical_cast< std::vector<std::string> >(currentOption->getDefaultValue().c_str())) :
281 po::value < std::vector<std::string> >()->multitoken(),
282 currentOption->getDescription().c_str())
283 ;
284 break;
285 case TypeEnumContainer::VectorType:
286 OptionList->add_options()
287 (currentOption->getKeyAndShortForm().c_str(),
288// currentOption->hasDefaultValue() ?
289// po::value < VectorValue >()->default_value(boost::lexical_cast<VectorValue>(currentOption->getDefaultValue().c_str())) :
290 po::value < VectorValue >(),
291 currentOption->getDescription().c_str())
292 ;
293 break;
294 case TypeEnumContainer::ListOfVectorsType:
295 OptionList->add_options()
296 (currentOption->getKeyAndShortForm().c_str(),
297// currentOption->hasDefaultValue() ?
298// po::value < std::vector<VectorValue> >()->default_value(boost::lexical_cast< std::vector<VectorValue> >(currentOption->getDefaultValue().c_str())) :
299 po::value < std::vector<VectorValue> >()->multitoken(),
300 currentOption->getDescription().c_str())
301 ;
302 break;
303 case TypeEnumContainer::MoleculeType:
304 OptionList->add_options()
305 (currentOption->getKeyAndShortForm().c_str(),
306// currentOption->hasDefaultValue() ?
307// po::value < const molecule * >()->default_value(boost::lexical_cast<const molecule *>(currentOption->getDefaultValue().c_str())) :
308 po::value < int >(),
309 currentOption->getDescription().c_str())
310 ;
311 break;
312 case TypeEnumContainer::ListOfMoleculesType:
313 OptionList->add_options()
314 (currentOption->getKeyAndShortForm().c_str(),
315// currentOption->hasDefaultValue() ?
316// po::value < std::vector<const molecule *> >()->default_value(boost::lexical_cast< std::vector<const molecule *> >(currentOption->getDefaultValue().c_str())) :
317 po::value < std::vector<int> >()->multitoken(),
318 currentOption->getDescription().c_str())
319 ;
320 break;
321 case TypeEnumContainer::AtomType:
322 OptionList->add_options()
323 (currentOption->getKeyAndShortForm().c_str(),
324 currentOption->hasDefaultValue() ?
325 po::value < int >()->default_value(boost::lexical_cast<int>(currentOption->getDefaultValue().c_str())) :
326 po::value < int >(),
327 currentOption->getDescription().c_str())
328 ;
329 break;
330 case TypeEnumContainer::ListOfAtomsType:
331 OptionList->add_options()
332 (currentOption->getKeyAndShortForm().c_str(),
333// currentOption->hasDefaultValue() ?
334// po::value < std::vector<const atom *> >()->default_value(boost::lexical_cast< std::vector<const atom *> >(currentOption->getDefaultValue().c_str())) :
335 po::value < std::vector<int> >()->multitoken(),
336 currentOption->getDescription().c_str())
337 ;
338 break;
339 case TypeEnumContainer::ElementType:
340 OptionList->add_options()
341 (currentOption->getKeyAndShortForm().c_str(),
342// currentOption->hasDefaultValue() ?
343// po::value < const element * >()->default_value(boost::lexical_cast<const element *>(currentOption->getDefaultValue().c_str())) :
344 po::value < int >(),
345 currentOption->getDescription().c_str())
346 ;
347 break;
348 case TypeEnumContainer::ListOfElementsType:
349 OptionList->add_options()
350 (currentOption->getKeyAndShortForm().c_str(),
351// currentOption->hasDefaultValue() ?
352// po::value < std::vector<const element *> >()->default_value(boost::lexical_cast< std::vector<const element *> >(currentOption->getDefaultValue().c_str())) :
353 po::value < std::vector<int> >()->multitoken(),
354 currentOption->getDescription().c_str())
355 ;
356 break;
[0275ad]357 case TypeEnumContainer::RandomNumberDistribution_ParametersType:
358 OptionList->add_options()
359 (currentOption->getKeyAndShortForm().c_str(),
360 currentOption->hasDefaultValue() ?
361 po::value < std::string >()->default_value(boost::lexical_cast< std::string >(currentOption->getDefaultValue().c_str())) :
362 po::value < std::string >(),
363 currentOption->getDescription().c_str())
364 ;
365 break;
[7d9416]366 case TypeEnumContainer::RealSpaceMatrixType:
367 OptionList->add_options()
368 (currentOption->getKeyAndShortForm().c_str(),
369// currentOption->hasDefaultValue() ?
370// po::value < RealSpaceMatrixValue >()->default_value(boost::lexical_cast<BoxValue>(currentOption->getDefaultValue().c_str())) :
371 po::value < RealSpaceMatrixValue >(),
372 currentOption->getDescription().c_str())
373 ;
374 break;
[e4afb4]375 }
376}
377
378/** States whether there are command line arguments.
379 * \return true - there are none, false - there is at least one command line argument
380 */
381bool CommandLineParser::isEmpty()
382{
383 return vm.empty();
384}
385
386/** Sets the options.
387 * \param _argc arg count from main()
388 * \param **_argv argument array from main()
389 */
390void CommandLineParser::setOptions(int _argc, char **_argv)
391{
392 argc = _argc;
393 argv = _argv;
394 config_file_options.add(options);
[c6e5eb]395 // append all option_descriptions to both cmdline_options and visible
396 for (CmdParserLookupMap::iterator iter = CmdParserLookup.begin();
397 iter != CmdParserLookup.end();
398 ++iter) {
399 cmdline_options.add(*(iter->second));
400 visible.add(*(iter->second));
401 }
[e4afb4]402}
403
404/** Parses the command line arguments.
405 * Calls program_options::store() and program_options::notify()
406 */
407void CommandLineParser::Parse()
408{
409 po::store(po::command_line_parser(argc,argv).options(cmdline_options).run(), vm);
[3a21a7]410 std::ifstream input;
[e4afb4]411 input.open("example.cfg");
412 if (!input.fail())
413 po::store(po::parse_config_file(input, config_file_options), vm);
414 input.close();
415 po::notify(vm);
416}
417
418/** Scan the argument list for -a or --arguments and store their order for later use.
419 */
420void CommandLineParser::scanforSequenceOfArguments()
421{
[3a21a7]422 std::map <std::string, std::string> ShortFormToActionMap = getShortFormToActionMap();
[47d041]423 LOG(0, "Scanning command line arguments and recognizing Actions.");
[e4afb4]424 // go through all arguments
425 for (int i=1;i<argc;i++) {
[47d041]426 LOG(2, "Checking on " << argv[i]);
[e4afb4]427 // check whether they
428 if (argv[i][0] == '-') { // .. begin with -
[47d041]429 LOG(2, "Possible argument: " << argv[i]);
[e4afb4]430 if (argv[i][1] == '-') { // .. or --
[47d041]431 LOG(1, "Putting " << argv[i] << " into the sequence.");
[e4afb4]432 SequenceOfActions.push_back(&(argv[i][2]));
433 // .. and check that next letter is not numeric, if so insert
434 } else if (((argv[i][1] < '0') || (argv[i][1] > '9')) && ((argv[i][1] != '.'))) {
[3a21a7]435 std::map <std::string, std::string>::iterator iter = ShortFormToActionMap.find(&(argv[i][1]));
[e4afb4]436 if (iter != ShortFormToActionMap.end()) {
[47d041]437 LOG(1, "Putting " << iter->second << " for " << iter->first << " into the sequence.");
[e4afb4]438 SequenceOfActions.push_back(iter->second);
439 }
440 }
441 }
442 }
443}
444
445/** Makes the Parser parse the command line options with current known options.
446 * \param _argc arg count from main()
447 * \param **_argv argument array from main()
448 */
449void CommandLineParser::Run(int _argc, char **_argv)
450{
451 setOptions(_argc,_argv);
452 Parse();
453 scanforSequenceOfArguments();
454}
455
456/** Go through all Actions and create a map from short form to their token.
457 * \return map from Action's ShortForm to token.
458 */
[2a0a9e3]459std::map <std::string, std::string> CommandLineParser::getShortFormToActionMap() const
[e4afb4]460{
[3a21a7]461 std::map <std::string, std::string> result;
[e4afb4]462
463 ActionRegistry &AR = ActionRegistry::getInstance();
464 for (ActionRegistry::const_iterator iter = AR.getBeginIter(); iter != AR.getEndIter(); ++iter)
465 if ((iter->second)->Traits.hasShortForm()) {
466 ASSERT(result.find((iter->second)->Traits.getShortForm()) == result.end(),
[cfc53b]467 "Short form "+toString((iter->second)->Traits.getShortForm())+
468 " for action "+toString(iter->first)+" already present from "+
[e4afb4]469 std::string(result[(iter->second)->Traits.getShortForm()])+"!");
470 result[(iter->second)->Traits.getShortForm()] = (iter->second)->getName();
471 }
472
473 return result;
474}
475
476CONSTRUCT_SINGLETON(CommandLineParser)
Note: See TracBrowser for help on using the repository browser.