67 myCopyrightNotices.push_back(
TL(
"Copyright (C) 2001-2026 German Aerospace Center (DLR) and others; https://sumo.dlr.de"));
84 throw ProcessError(name +
" is an already used option name.");
87 bool isSynonym =
false;
89 if (addresse.second == o) {
113 throw ProcessError(
"Neither the option '" + name1 +
"' nor the option '" + name2 +
"' is known yet");
116 if ((*i1).second == (*i2).second) {
119 throw ProcessError(
"Both options '" + name1 +
"' and '" + name2 +
"' do exist and differ.");
152 if (failOnNonExistant) {
153 throw ProcessError(
TLF(
"Internal request for unknown option '%'!", name));
158 return (*i).second->isSet();
168 return (*i).second->isDefault();
174 const auto& valuesFinder =
myValues.find(name);
175 if (valuesFinder ==
myValues.end()) {
180 std::string defaultName;
182 for (
const auto& value : subtopicEntry.second) {
183 const auto l =
myValues.find(value);
184 if ((l !=
myValues.end()) && (l->second == valuesFinder->second)) {
189 if (defaultName !=
"") {
193 WRITE_WARNINGF(
TL(
"Please note that '%' is deprecated.\n Use '%' instead."), name, defaultName);
194 synonymFinder->second =
true;
196 return valuesFinder->second;
261 WRITE_ERROR(
"While processing option '" + name +
"':\n " + e.what());
291std::vector<std::string>
294 std::vector<std::string> synonymes;
295 for (
const auto& value :
myValues) {
296 if ((value.second == o) && (name != value.first)) {
297 synonymes.push_back(value.first);
318 std::vector<std::string> done;
319 os <<
"Options set:" << std::endl;
320 for (
const auto& value : oc.
myValues) {
321 const auto& finder = std::find(done.begin(), done.end(), value.first);
322 if (finder == done.end()) {
323 std::vector<std::string> synonymes = oc.
getSynonymes(value.first);
324 if (synonymes.size() != 0) {
325 os << value.first <<
" (";
326 for (
auto synonym = synonymes.begin(); synonym != synonymes.end(); synonym++) {
327 if (synonym != synonymes.begin()) {
336 if (value.second->isSet()) {
337 os <<
": " << value.second->getValueString() << std::endl;
339 os <<
": <INVALID>" << std::endl;
341 done.push_back(value.first);
342 copy(synonymes.begin(), synonymes.end(), back_inserter(done));
352 if (addresse.second->isFileName() && addresse.second->isSet()) {
354 for (
auto& file : fileList) {
355 if (addresse.first !=
"configuration-file") {
365 for (
auto& file : rawList) {
369 if (conv !=
joinToString(addresse.second->getStringVector(),
',')) {
370 const bool hadDefault = addresse.second->isDefault();
371 addresse.second->set(conv,
joinToString(rawList,
','),
false);
373 addresse.second->resetDefault();
390 if (files.size() == 0) {
394 for (
const auto& file : files) {
397 WRITE_ERRORF(
TL(
"File '%' is not accessible (%)."), file, std::strerror(errno));
415 std::vector<std::string> seenSynonymes;
416 for (
const auto& value :
myValues) {
417 if (std::find(seenSynonymes.begin(), seenSynonymes.end(), value.first) != seenSynonymes.end()) {
420 if (value.second->isSet() && !value.second->isDefault() && value.first.find(prefix) == 0) {
421 WRITE_ERRORF(
TL(
"Option '%' needs option '%'."), value.first, name);
422 std::vector<std::string> synonymes =
getSynonymes(value.first);
423 std::copy(synonymes.begin(), synonymes.end(), std::back_inserter(seenSynonymes));
434 std::ostringstream s;
435 s <<
TLF(
"A value for the option '%' was already set.\n Possible synonymes: ", arg);
436 auto synonym = synonymes.begin();
437 while (synonym != synonymes.end()) {
440 if (synonym != synonymes.end()) {
468 addresse.second->resetWritable();
482 addresse.second->resetDefault();
510 delete addresse.second;
521 const std::string& description) {
527 throw ProcessError(
"SubTopic '" + subtopic +
"' doesn't exist");
542 throw ProcessError(
"SubTopic '" + subtopic +
"' doesn't exist");
606 int offset,
int nextOffset) {
607 while (what.length() > 0) {
608 if ((
int)what.length() > 79 - offset) {
609 std::string::size_type splitPos = what.rfind(
';', 79 - offset);
610 if (splitPos == std::string::npos) {
611 splitPos = what.rfind(
' ', 79 - offset);
615 if (splitPos != std::string::npos) {
616 os << what.substr(0, splitPos) << std::endl;
617 what = what.substr(splitPos + 1);
618 for (
int r = 0; r < (nextOffset + 1); ++r) {
639 if (missingOptions) {
642 std::cout <<
TL(
" Build features: ") << HAVE_ENABLED << std::endl;
644 std::cout <<
" " << copyrightNotice.data() << std::endl;
646 std::cout <<
TL(
" License EPL-2.0: Eclipse Public License Version 2 <https://eclipse.org/legal/epl-v20.html>") << std::endl;
647 std::cout <<
TL(
" Use --help to get the list of options.") << std::endl;
655 std::cout <<
" " << copyrightNotice.data() << std::endl;
663 std::cout <<
TL(
" Build features: ") << HAVE_ENABLED << std::endl;
665 std::cout <<
" " << copyrightNotice.data() << std::endl;
667 std::cout <<
"\n" <<
myFullName <<
" is part of SUMO.\n";
668 std::cout <<
"This program and the accompanying materials\n";
669 std::cout <<
"are made available under the terms of the Eclipse Public License v2.0\n";
670 std::cout <<
"which accompanies this distribution, and is available at\n";
671 std::cout <<
"http://www.eclipse.org/legal/epl-v20.html\n";
672 std::cout <<
"This program may also be made available under the following Secondary\n";
673 std::cout <<
"Licenses when the conditions for such availability set forth in the Eclipse\n";
674 std::cout <<
"Public License 2.0 are satisfied: GNU General Public License, version 2\n";
675 std::cout <<
"or later which is available at\n";
676 std::cout <<
"https://www.gnu.org/licenses/old-licenses/gpl-2.0-standalone.html\n";
677 std::cout <<
"SPDX-License-Identifier: EPL-2.0 OR GPL-2.0-or-later" << std::endl;
681 if (
getBool(
"print-options")) {
682 std::cout << (*this);
686 if (
isSet(
"save-configuration")) {
687 const std::string& configPath =
getString(
"save-configuration");
688 if (configPath ==
"-" || configPath ==
"stdout") {
694 throw ProcessError(
TLF(
"Could not save configuration to '%'", configPath));
704 if (
isSet(
"save-template")) {
720 if (
isSet(
"save-schema")) {
745 option.second->setDescription(
TL(option.second->getDescription().c_str()));
749 example.second =
TL(example.second.c_str());
758const std::vector<std::string>&
764std::vector<std::string>
769 return std::vector<std::string>();
792std::vector<std::pair<std::string, Option*> >::const_iterator
798std::vector<std::pair<std::string, Option*> >::const_iterator
818 int csize = (int)entry.length() + 2 + 4;
821 for (
const auto& synonym : synonymes) {
833 if (csize < tooLarge && maxSize < csize) {
840 if (helpTopic !=
"") {
841 bool foundTopic =
false;
850 os <<
TL(
"Help Topics:") << std::endl;
852 os <<
" " << t << std::endl;
858 os <<
TL(
"Usage: ") <<
myAppName <<
TL(
" [OPTION]*") << std::endl;
870 os <<
TL(
"Examples:") << std::endl;
872 os <<
" " <<
myAppName <<
' ' << callExample.first << std::endl;
873 os <<
" " << callExample.second << std::endl;
877 os <<
TLF(
"Report bugs at %.",
"<https://github.com/eclipse-sumo/sumo/issues>") << std::endl;
878 os <<
TLF(
"Get in contact via %.",
"<sumo@dlr.de>") << std::endl;
884 os <<
TLF(
"% Options:", topic) << std::endl;
887 int csize = (int)entry.length() + 2;
892 for (
const auto& synonym : synonymes) {
894 os <<
'-' << synonym <<
", ";
912 for (
int r = maxSize; r > csize; --r) {
915 int offset = csize > tooLarge ? csize : maxSize;
924 const bool complete,
const bool addComments,
const std::string& relativeTo,
925 const bool forceRelative,
const bool inComment,
const std::string& indent)
const {
930 os << indent <<
"<" << app <<
"Configuration xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\" "
931 <<
"xsi:noNamespaceSchemaLocation=\"http://sumo.dlr.de/xsd/" << app <<
"Configuration.xsd\">\n\n";
933 if (subtopic ==
"Configuration" && !complete) {
936 const std::vector<std::string>& entries =
mySubTopicEntries.find(subtopic)->second;
937 std::replace(subtopic.begin(), subtopic.end(),
' ',
'_');
940 for (
const std::string& name : entries) {
942 bool write = complete || (filled && !o->
isDefault());
946 if (name ==
"registry-viewport" && !complete) {
950 os << indent <<
" <" << subtopic <<
">\n";
957 os << indent <<
" <" << name <<
" value=\"";
961 for (
auto& file : fileList) {
970 forceRelative ||
getBool(
"save-configuration.relative"));
979 const std::vector<std::string> synonymes =
getSynonymes(name);
980 if (!synonymes.empty()) {
981 os <<
"\" synonymes=\"" <<
toString(synonymes);
983 std::string deprecated;
984 for (
const auto& synonym : synonymes) {
986 deprecated +=
" " + synonym;
989 if (deprecated !=
"") {
990 os <<
"\" deprecated=\"" << deprecated.substr(1);
1005 os << indent <<
" </" << subtopic <<
">\n\n";
1008 os << indent <<
"</" << app <<
"Configuration>" << std::endl;
1016 os <<
"<xsd:schema elementFormDefault=\"qualified\" xmlns:xsd=\"http://www.w3.org/2001/XMLSchema\">\n";
1017 os <<
" <xsd:complexType name=\"" << app <<
"ConfigurationType\">\n";
1018 os <<
" <xsd:all>\n";
1020 if (subtopic ==
"Configuration") {
1023 std::replace(subtopic.begin(), subtopic.end(),
' ',
'_');
1025 os <<
" <xsd:element name=\"" << subtopic <<
"\" type=\"" << app << subtopic <<
"TopicType\" minOccurs=\"0\"/>\n";
1027 os <<
" </xsd:all>\n";
1028 os <<
" </xsd:complexType>\n\n";
1030 if (subtopic ==
"Configuration") {
1033 const std::vector<std::string>& entries =
mySubTopicEntries.find(subtopic)->second;
1034 std::replace(subtopic.begin(), subtopic.end(),
' ',
'_');
1036 os <<
" <xsd:complexType name=\"" << app << subtopic <<
"TopicType\">\n";
1037 os <<
" <xsd:all>\n";
1038 for (
const auto& entry : entries) {
1042 if (type ==
"int[]") {
1045 if (type ==
"str[]") {
1048 os <<
" <xsd:element name=\"" << entry <<
"\" type=\"" << type <<
"OptionType\" minOccurs=\"0\"/>\n";
1050 os <<
" </xsd:all>\n";
1051 os <<
" </xsd:complexType>\n\n";
1053 os <<
"</xsd:schema>\n";
1061 if (!
getBool(
"write-metadata")) {
1064 if (
getBool(
"write-license")) {
1065 os <<
"This data file and the accompanying materials\n"
1066 "are made available under the terms of the Eclipse Public License v2.0\n"
1067 "which accompanies this distribution, and is available at\n"
1068 "http://www.eclipse.org/legal/epl-v20.html\n"
1069 "This file may also be made available under the following Secondary\n"
1070 "Licenses when the conditions for such availability set forth in the Eclipse\n"
1071 "Public License 2.0 are satisfied: GNU General Public License, version 2\n"
1072 "or later which is available at\n"
1073 "https://www.gnu.org/licenses/old-licenses/gpl-2.0-standalone.html\n"
1074 "SPDX-License-Identifier: EPL-2.0 OR GPL-2.0-or-later\n";
1076 if (includeConfig && !
getBool(
"write-metadata")) {
1085 const std::string& itemName)
const {
1086 if (
isSet(optionName)) {
1088 return std::find(values.begin(), values.end(), itemName) != values.end();
1101 addr.second = addr.second->clone();
#define WRITE_WARNINGF(...)
#define WRITE_MESSAGEF(...)
#define WRITE_ERRORF(...)
#define WRITE_WARNING(msg)
std::vector< std::string > StringVector
Definition of a vector of strings.
std::vector< int > IntVector
Definition of a vector of ints.
std::ostream & operator<<(std::ostream &os, const OptionsCont &oc)
std::string joinToString(const std::vector< T > &v, const T_BETWEEN &between, std::streamsize accuracy=gPrecision)
std::string toString(const T &t, std::streamsize accuracy=gPrecision)
static std::string fixRelative(const std::string &filename, const std::string &basePath, const bool force, std::string curDir="")
Fixes the relative path for the given filename in relation to the basePath (usually a config file).
static std::string checkForRelativity(const std::string &filename, const std::string &basePath)
Returns the path from a configuration so that it is accessible from the current working directory.
static bool isReadable(std::string path)
Checks whether the given file is readable.
static void setupI18n(const std::string &locale="")
set up gettext stuff
A class representing a single program option.
bool isWriteable() const
Returns the information whether the option may be set a further time.
bool isSet() const
returns the information whether this options holds a valid value
virtual bool isDefault() const
Returns the information whether the option holds the default value.
void setRequired()
mark option as required
virtual std::string getString() const
Returns the stored string value.
virtual const IntVector & getIntVector() const
Returns the stored integer vector.
void resetWritable()
Resets the option to be writeable.
const std::string & getDescription() const
Returns the description of what this option does.
void setListSeparator(const std::string &listSep)
set list separator
virtual bool isFileName() const
Returns the information whether this option is a file name.
virtual const StringVector & getStringVector() const
Returns the stored string vector.
void setDescription(const std::string &desc)
Sets the description of what this option does.
virtual const std::string & getTypeName() const
Returns the mml-type name of this option.
virtual int getInt() const
Returns the stored integer value.
virtual double getFloat() const
Returns the stored double value.
virtual bool getBool() const
Returns the stored boolean value.
void setPositional()
mark option as positional
void resetDefault()
Resets the option to be on its default value.
const std::string & getSubTopic() const
Returns the subtopic to which this option belongs.
virtual bool set(const std::string &v, const std::string &orig, const bool append)=0
Stores the given value.
bool isEditable() const
check if this option is editable
virtual bool isBool() const
Returns the information whether the option is a bool option.
void setEditable(const bool value)
set editable
const std::string & getValueString() const
Returns the string-representation of the value.
void setSubtopic(const std::string &subtopic)
Sets the subtopic to which this option belongs.
A storage for options typed value containers)
void setAdditionalHelpMessage(const std::string &add)
Sets an additional message to be printed at the begin of the help screen.
~OptionsCont()
Destructor.
void addDescription(const std::string &name, const std::string &subtopic, const std::string &description)
Adds a description for an option.
void writeConfiguration(std::ostream &os, const bool filled, const bool complete, const bool addComments, const std::string &relativeTo="", const bool forceRelative=false, const bool inComment=false, const std::string &indent="") const
Writes the configuration.
void resetDefault()
Resets all options to default.
void setFurtherAttributes(const std::string &name, const std::string &subtopic, bool required, bool positional, const std::string &listSep)
mark option as required
bool isSet(const std::string &name, bool failOnNonExistant=true) const
Returns the information whether the named option is set.
std::vector< std::pair< std::string, std::string > > myCallExamples
list of call examples
bool isWriteable(const std::string &name)
Returns the information whether the named option may be set.
std::map< std::string, std::vector< std::string > > mySubTopicEntries
A map from subtopic to option.
void writeXMLHeader(std::ostream &os, const bool includeConfig=true) const
Writes a standard XML header, including the configuration.
double getFloat(const std::string &name) const
Returns the double-value of the named option (only for Option_Float)
int getInt(const std::string &name) const
Returns the int-value of the named option (only for Option_Integer)
void splitLines(std::ostream &os, std::string what, int offset, int nextOffset)
Writes the given string 'formatted'.
void setApplicationName(const std::string &appName, const std::string &fullName)
Sets the application name.
OptionsCont * clone() const
make a copy of this OptionsCont instance
void printHelpOnTopic(const std::string &topic, int tooLarge, int maxSize, std::ostream &os)
Prints help on the given topic.
const std::string & getSubTopic(const std::string &name) const
Returns the option category.
std::map< std::string, Option * > myValues
option maps sorted by name (for addresses AND their synonyms)
bool isEmpty() const
check if options container is empty
std::string myAdditionalMessage
std::vector< std::string > myCopyrightNotices
const std::vector< std::string > & getSubTopics() const
return the list of subtopics
void setOptionEditable(const std::string &name, const bool value)
set option editable
bool myAmLocalized
Whether the descriptino has already been translated to the locale language.
std::vector< std::pair< std::string, Option * > > myAddresses
option-addresses
const IntVector & getIntVector(const std::string &name) const
Returns the list of integer-value of the named option (only for Option_IntVector)
std::vector< std::string > getSynonymes(const std::string &name) const
Returns the synonymes of an option name.
void reportDoubleSetting(const std::string &arg) const
Reports an error that the option has already been set.
std::vector< std::string > mySubTopics
lists of option subtopics and copyright notices
std::string getString(const std::string &name) const
Returns the string-value of the named option (only for Option_String)
static OptionsCont EMPTY_OPTIONS
empty option container
void localizeDescriptions()
void addSynonyme(const std::string &name1, const std::string &name2, bool isDeprecated=false)
Adds a synonyme for an options name (any order)
bool isDefault(const std::string &name) const
Returns the information whether the named option has still the default value.
bool setDefault(const std::string &name, const std::string &value)
Sets the given value for the named option as new default value.
void doRegister(const std::string &name, Option *o)
Adds an option under the given name.
bool exists(const std::string &name) const
Returns the information whether the named option is known.
const std::string & getFullName() const
get options full name
bool isBool(const std::string &name) const
Returns the information whether the option is a boolean option.
void addCopyrightNotice(const std::string ©rightLine)
Adds a copyright notice to the help output.
std::string getTypeName(const std::string name)
return the type name for the given option
void writeSchema(std::ostream &os)
Writes the xml schema for the configuration.
void clear()
Removes all information from the container.
void setApplicationDescription(const std::string &appDesc)
Sets the application description.
std::vector< std::pair< std::string, Option * > >::const_iterator begin() const
get begin addresses iterator
bool set(const std::string &name, const std::string &value, const bool append=false)
Sets the given value for the named option.
bool isEditable(const std::string &name)
Returns the information whether the named option is editable.
void clearCopyrightNotices()
Removes all copyright information.
std::string convertChar(char abbr) const
Converts an abbreviation into a name.
void addOptionSubTopic(const std::string &topic)
Adds an option subtopic.
bool getBool(const std::string &name) const
Returns the boolean-value of the named option (only for Option_Bool)
const StringVector & getStringVector(const std::string &name) const
Returns the list of string-value of the named option (only for Option_StringVector)
OptionsCont()
Constructor.
void printHelp(std::ostream &os)
Prints the help.
std::vector< std::pair< std::string, Option * > >::const_iterator end() const
get begin addresses iterator
std::string getValueString(const std::string &name) const
Returns the string-value of the named option (all options)
std::string myAppDescription
const std::string & getDescription(const std::string &name) const
Returns the option description.
bool setByRootElement(const std::string &name, const std::string &value)
Sets the given value for the option which can handle the given XML root.
std::map< std::string, bool > myDeprecatedSynonymes
A map from deprecated options to a bool indicating whether we warned about deprecation.
static OptionsCont myOptions
The static options container used.
bool checkDependingSuboptions(const std::string &name, const std::string &prefix) const
Checks whether an option is set, which has options with a prefix depending on it.
std::map< std::string, std::string > myXMLDefaults
A map from XML root element to option.
std::string myAppName
some information on the application
void resetWritable()
Resets all options to be writeable.
void addXMLDefault(const std::string &name, const std::string &xmlRoot="")
Adds an XML root element to handle by default. The special root "" denotes the default handler.
static OptionsCont & getOptions()
Retrieves the options.
std::vector< std::string > getSubTopicsEntries(const std::string &subtopic) const
return the list of entries for the given subtopic
Option * getSecure(const std::string &name) const
Returns the named option.
void relocateFiles(const std::string &configuration) const
Modifies file name options according to the configuration path.
bool isInStringVector(const std::string &optionName, const std::string &itemName) const
Returns the named option is a list of string values containing the specified item.
bool processMetaOptions(bool missingOptions)
Checks for help and configuration output, returns whether we should exit.
bool isUsableFileList(const std::string &name) const
Checks whether the named option is usable as a file list (with at least a single file)
void addCallExample(const std::string &example, const std::string &desc)
Add a call example.
static const std::chrono::time_point< std::chrono::system_clock > & getLoadTime()
Return the time stamp of the last init.
static const std::string ENCODING
The encoding of parsed strings.
std::vector< std::string > getVector()
return vector of strings
static std::string urlEncode(const std::string &url, const std::string encodeWhich="")
encode url (stem from http://bogomip.net/blog/cpp-url-encoding-and-decoding/)
static std::string urlDecode(const std::string &encoded)
decode url (stem from http://bogomip.net/blog/cpp-url-encoding-and-decoding/)
static std::string to_lower_case(const std::string &str)
Transfers the content to lower case.
static std::string escapeXML(const std::string &orig, const bool maskDoubleHyphen=false)
Replaces the standard escapes by their XML entities.
static bool startsWith(const std::string &str, const std::string prefix)
Checks whether a given string starts with the prefix.
static std::string substituteEnvironment(const std::string &str, const std::chrono::time_point< std::chrono::system_clock > *const timeRef=nullptr)
Replaces an environment variable with its value (similar to bash); syntax for a variable is ${NAME}.
static std::string transcodeToLocal(const std::string &utf8String)
convert a string from UTF-8 to the local codepage
static std::string isoTimeString(const std::chrono::time_point< std::chrono::system_clock > *const timeRef=nullptr)
Returns an ISO8601 formatted time string with microsecond precision.