diff --git a/libsolidity/interface/StandardCompiler.cpp b/libsolidity/interface/StandardCompiler.cpp index c46be3b22..03adddc60 100644 --- a/libsolidity/interface/StandardCompiler.cpp +++ b/libsolidity/interface/StandardCompiler.cpp @@ -881,29 +881,31 @@ std::variant StandardCompiler: if (!jsonSourceName[library].isString()) return formatFatalError(Error::Type::JSONError, "Library address must be a string."); std::string address = jsonSourceName[library].asString(); + util::h160 addr; - if (!boost::starts_with(address, "0x")) + util::ValidationError error = util::validateAddress(address, addr); + + switch (error) + { + case util::ValidationError::emptyAddress: + case util::ValidationError::wrongPrefix: return formatFatalError( Error::Type::JSONError, "Library address is not prefixed with \"0x\"." ); - - if (address.length() != 42) + case util::ValidationError::wrongAddressLength: return formatFatalError( Error::Type::JSONError, "Library address is of invalid length." ); - - try - { - ret.libraries[sourceName + ":" + library] = util::h160(address); - } - catch (util::BadHexCharacter const&) - { + case util::ValidationError::checksumNotPassed: + case util::ValidationError::invalidAddress: return formatFatalError( Error::Type::JSONError, - "Invalid library address (\"" + address + "\") supplied." + "Invalid library address (\"0x" + address + "\") supplied." ); + case util::ValidationError::noError: + ret.libraries[sourceName + ":" + library] = addr; } } } diff --git a/libsolutil/StringUtils.cpp b/libsolutil/StringUtils.cpp index 1dc86714d..e9b016612 100644 --- a/libsolutil/StringUtils.cpp +++ b/libsolutil/StringUtils.cpp @@ -22,6 +22,7 @@ * String routines */ +#include "FixedHash.h" #include #include #include @@ -190,3 +191,26 @@ std::string solidity::util::formatNumberReadable(bigint const& _value, bool _use return sign + str; } +ValidationError solidity::util::validateAddress(std::string& addrString, util::h160& address) +{ + if (addrString.empty()) + return ValidationError::emptyAddress; + + if (addrString.substr(0, 2) != "0x") + return ValidationError::wrongPrefix; + addrString = addrString.substr(2); + + if (addrString.length() != 40) + return ValidationError::wrongAddressLength; + + if (!util::passesAddressChecksum(addrString, false)) + return ValidationError::checksumNotPassed; + + bytes binAddr = util::fromHex(addrString); + address = util::h160(binAddr, util::h160::AlignRight); + + if (binAddr.size() > 20 || address == util::h160()) + return ValidationError::invalidAddress; + + return ValidationError::noError; +} diff --git a/libsolutil/StringUtils.h b/libsolutil/StringUtils.h index 0d5bc7839..d45932d7b 100644 --- a/libsolutil/StringUtils.h +++ b/libsolutil/StringUtils.h @@ -26,6 +26,7 @@ #include #include +#include #include @@ -115,6 +116,17 @@ std::string joinHumanReadablePrefixed /// @example formatNumberReadable(-57896044618658097711785492504343953926634992332820282019728792003956564819968) = -2**255 std::string formatNumberReadable(bigint const& _value, bool _useTruncation = false); +enum ValidationError { + noError = 0, + emptyAddress, + wrongPrefix, + wrongAddressLength, + checksumNotPassed, + invalidAddress +}; + +ValidationError validateAddress(std::string& addrString, util::h160& address); + /// Safely converts an unsigned integer as string into an unsigned int type. /// /// @return the converted number or nullopt in case of an failure (including if it would not fit). diff --git a/solc/CommandLineParser.cpp b/solc/CommandLineParser.cpp index 53f481ae8..8cb6ffac3 100644 --- a/solc/CommandLineParser.cpp +++ b/solc/CommandLineParser.cpp @@ -23,6 +23,7 @@ #include #include +#include #include @@ -411,43 +412,51 @@ void CommandLineParser::parseLibraryOption(std::string const& _input) std::string addrString(lib.begin() + static_cast(separator) + 1, lib.end()); boost::trim(addrString); - if (addrString.empty()) + + util::h160 address; + util::ValidationError error = util::validateAddress(addrString, address); + + switch (error) + { + case util::ValidationError::emptyAddress: solThrow( CommandLineValidationError, "Empty address provided for library \"" + libName + "\".\n" - "Note that there should not be any whitespace after the " + + "Note that there should not be any whitespace after the " + (isSeparatorEqualSign ? "equal sign" : "colon") + "." ); - - if (addrString.substr(0, 2) == "0x") - addrString = addrString.substr(2); - else + break; + case util::ValidationError::wrongPrefix: solThrow( CommandLineValidationError, "The address " + addrString + " is not prefixed with \"0x\".\n" - "Note that the address must be prefixed with \"0x\"." + "Note that the address must be prefixed with \"0x\"." ); - - if (addrString.length() != 40) + break; + case util::ValidationError::wrongAddressLength: solThrow( CommandLineValidationError, "Invalid length for address for library \"" + libName + "\": " + std::to_string(addrString.length()) + " instead of 40 characters." ); - if (!util::passesAddressChecksum(addrString, false)) + break; + case util::ValidationError::checksumNotPassed: solThrow( CommandLineValidationError, "Invalid checksum on address for library \"" + libName + "\": " + addrString + "\n" - "The correct checksum is " + util::getChecksummedAddress(addrString) + "The correct checksum is " + util::getChecksummedAddress(addrString) ); - bytes binAddr = util::fromHex(addrString); - util::h160 address(binAddr, util::h160::AlignRight); - if (binAddr.size() > 20 || address == util::h160()) + break; + case util::ValidationError::invalidAddress: solThrow( CommandLineValidationError, "Invalid address for library \"" + libName + "\": " + addrString ); - m_options.linker.libraries[libName] = address; + break; + case util::ValidationError::noError: + m_options.linker.libraries[libName] = address; + break; + } } }