#include <stdexcept>
#include "parameter.h"
+#include "exception/missingOptionValue.h"
+#include "exception/optionFailedConversion.h"
+#include "exception/optionValueNotSpecified.h"
namespace command {
/**
* @param description Description of current Option
* @param function Function used to handle current Option.
*/
- Option(const std::string & name, const std::string & description, void (*function)(ParameterType))
+ Option(const std::string & name, const std::string & description, std::function<void(ParameterType)> function)
: Parameter(description), Callable<ParameterType>(function), name(name) {
}
*/
virtual void handle() {
this->call(value);
+ used = true;
}
/**
* converted to ParameterType, returns true and Option is set as used one.
* Otherwise returns false and can be used to check against next value.
*
- * @throw std::invalid_argument when OptionName part has no equal sign
- * after itself
- * @throw std::invalid_argument when OptionValue part failed conversion
+ * @throw MissingOptionValue when OptionValue part is missing after
+ * equal sign
+ * @throw OptionFailedConversion when OptionValue part failed conversion
* to ParameterType
*/
- virtual bool understand(const std::string & argv)
- throw(std::invalid_argument) {
-
- if ((!isUsed()) && (argv.find(name) == 0)) {
- std::size_t pos = argv.find("=");
+ virtual bool understand(const std::string & argv) {
+ if (this->hasName(argv)) {
+ std::size_t pos = this->valuePosition(argv);
if (pos != name.size()) {
- throw std::invalid_argument("Option: " + name + " requires value but no one has been provided");
+ throw MissingOptionValue("Option: " + name + " requires value but no one has been provided");
}
std::stringstream ss;
- ss << argv.substr(pos + 1);
- ss >> value;// memory leak? when uncommented and exception is
- // thrown, valgrind shows e.g.:
- // possibly lost: 380 bytes in 7 blocks
+ ss << std::fixed << argv.substr(pos + 1);
+ ss >> value;
if (ss.fail()) {
- throw std::invalid_argument("Value for option: " + name + " failed conversion to the required type");
+ throw OptionFailedConversion("Option: " + name + " failed value conversion to the required type");
}
- used = true;
return true;
}
return false;
}
+
+ /**
+ * \inheritdoc
+ */
+ virtual unsigned int valuePosition(const std::string & value) {
+ std::size_t pos = value.find("=");
+
+ if ((this->hasName(value)) && (pos == std::string::npos)) {
+ throw OptionValueNotSpecified("Option: " + name + " requires value to be specified after equal sign, but no equal sign was found");
+ }
+
+ return pos;
+ }
+
+ protected:
+ bool hasName(const std::string & argv) {
+ return argv.find(name) == 0;
+ }
};
/**
* @param description Description of current Option
* @param function Function used to handle current Option.
*/
- Option(const std::string & name, const std::string & description, void (*function)(void))
+ Option(const std::string & name, const std::string & description, std::function<void(void)> function)
: Parameter(description), Callable<void>(function), name(name) {
}
*/
virtual void handle() {
this->call();
+ used = true;
}
/**
* used to check against next value.
*/
virtual bool understand(const std::string & argv) {
- if ((!isUsed()) &&
- (argv == name)) {
- used = true;
+ if (argv == name) {
return true;
}
return false;
}
+
+ /**
+ * \inheritdoc
+ */
+ virtual unsigned int valuePosition(const std::string & ) {
+ throw new std::invalid_argument(this->describe() + " is void Option, so it does not have value part");
+ }
};
}