X-Git-Url: https://git.dlugolecki.net.pl/?a=blobdiff_plain;f=include%2Foption.h;h=7e9917b589c7021a0c5852695e90eecb5fa36908;hb=71d10208f91a566b40ea1bdbb1e2f20f85f8182f;hp=516fce016eb80c6d6e350c6e050fc498e9d3c497;hpb=e7c2ad96ad56116b2bfca1000f425287b1a02da2;p=command.git diff --git a/include/option.h b/include/option.h index 516fce0..7e9917b 100644 --- a/include/option.h +++ b/include/option.h @@ -6,6 +6,9 @@ #include #include "parameter.h" +#include "exception/missingOptionValue.h" +#include "exception/optionFailedConversion.h" +#include "exception/optionValueNotSpecified.h" namespace command { /** @@ -55,6 +58,7 @@ namespace command { */ virtual void handle() { this->call(value); + used = true; } /** @@ -79,36 +83,49 @@ namespace command { * 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; + } }; /** @@ -150,6 +167,7 @@ namespace command { */ virtual void handle() { this->call(); + used = true; } /** @@ -168,13 +186,18 @@ namespace command { * 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"); + } }; }