a44caf65fe Merge bitcoin-core/univalue-subtree#28: Import fixes for sanitizer reported issues 135254331e Import fixes for sanitizer reported issues d5fb86940e refactor: use c++11 range based for loop in checkObject ff9c379304 refactor: Use nullptr (c++11) instead of NULL 08a99754d5 build: use ax_cxx_compile_stdcxx.m4 to check for C++11 support 66d3713ce7 Merge bitcoin-core/univalue#29: ci: travis -> cirrus 808d487292 ci: travis -> cirrus c390ac375f Merge bitcoin-core/univalue#19: Split sources for easier buildsystem integration 4a5b0a1c65 build: Move source entries out to sources.mk 6c7d94b33c build: cleanup wonky gen usage a222637c6d Merge #23: Merge changes from jgarzik/univalue@1ae6a23 f77d0f718d Merge commit '1ae6a231a0169938eb3972c1d48dd17cba5947e1' into HEAD 1ae6a231a0 Merge pull request #57 from MarcoFalke/test_fix 92bdd11f0b univalue_write: remove unneeded sstream.h include ffb621c130 Merge pull request #56 from drodil/remove_sstream_header f33acf9fe8 Merge commit '7890db9~' into HEAD 66e0adec4d Remove unnecessary sstream header from univalue.h 88967f6586 Version 1.0.4 1dc113dbef Merge pull request #50 from luke-jr/pushKV_bool 72392fb227 [tests] test pushKV for boolean values c23132bcf4 Pushing boolean value to univalue correctly 81faab26a1 Merge pull request #48 from fwolfst/47-UPDATE_MIT_LINK_TO_HTTPS b17634ef24 Update URLs to MIT license. 88ab64f6b5 Merge pull request #46 from jasonbcox/master 35ed96da31 Merge pull request #44 from MarcoFalke/Mf1709-univalue-cherrypick-explicit 420c226290 Merge pull request #45 from MarcoFalke/Mf1710-univalue-revert-test git-subtree-dir: src/univalue git-subtree-split: a44caf65fe55b9dd8ddb08f04c0f70409efd53b3
149 lines
4.3 KiB
C++
149 lines
4.3 KiB
C++
// Copyright 2014 BitPay Inc.
|
|
// Copyright 2015 Bitcoin Core Developers
|
|
// Distributed under the MIT software license, see the accompanying
|
|
// file COPYING or https://opensource.org/licenses/mit-license.php.
|
|
|
|
#include <stdint.h>
|
|
#include <errno.h>
|
|
#include <string.h>
|
|
#include <stdlib.h>
|
|
#include <stdexcept>
|
|
#include <vector>
|
|
#include <limits>
|
|
#include <string>
|
|
#include <sstream>
|
|
|
|
#include "univalue.h"
|
|
|
|
namespace
|
|
{
|
|
static bool ParsePrechecks(const std::string& str)
|
|
{
|
|
if (str.empty()) // No empty string allowed
|
|
return false;
|
|
if (str.size() >= 1 && (json_isspace(str[0]) || json_isspace(str[str.size()-1]))) // No padding allowed
|
|
return false;
|
|
if (str.size() != strlen(str.c_str())) // No embedded NUL characters allowed
|
|
return false;
|
|
return true;
|
|
}
|
|
|
|
bool ParseInt32(const std::string& str, int32_t *out)
|
|
{
|
|
if (!ParsePrechecks(str))
|
|
return false;
|
|
char *endp = nullptr;
|
|
errno = 0; // strtol will not set errno if valid
|
|
long int n = strtol(str.c_str(), &endp, 10);
|
|
if(out) *out = (int32_t)n;
|
|
// Note that strtol returns a *long int*, so even if strtol doesn't report an over/underflow
|
|
// we still have to check that the returned value is within the range of an *int32_t*. On 64-bit
|
|
// platforms the size of these types may be different.
|
|
return endp && *endp == 0 && !errno &&
|
|
n >= std::numeric_limits<int32_t>::min() &&
|
|
n <= std::numeric_limits<int32_t>::max();
|
|
}
|
|
|
|
bool ParseInt64(const std::string& str, int64_t *out)
|
|
{
|
|
if (!ParsePrechecks(str))
|
|
return false;
|
|
char *endp = nullptr;
|
|
errno = 0; // strtoll will not set errno if valid
|
|
long long int n = strtoll(str.c_str(), &endp, 10);
|
|
if(out) *out = (int64_t)n;
|
|
// Note that strtoll returns a *long long int*, so even if strtol doesn't report a over/underflow
|
|
// we still have to check that the returned value is within the range of an *int64_t*.
|
|
return endp && *endp == 0 && !errno &&
|
|
n >= std::numeric_limits<int64_t>::min() &&
|
|
n <= std::numeric_limits<int64_t>::max();
|
|
}
|
|
|
|
bool ParseDouble(const std::string& str, double *out)
|
|
{
|
|
if (!ParsePrechecks(str))
|
|
return false;
|
|
if (str.size() >= 2 && str[0] == '0' && str[1] == 'x') // No hexadecimal floats allowed
|
|
return false;
|
|
std::istringstream text(str);
|
|
text.imbue(std::locale::classic());
|
|
double result;
|
|
text >> result;
|
|
if(out) *out = result;
|
|
return text.eof() && !text.fail();
|
|
}
|
|
}
|
|
|
|
const std::vector<std::string>& UniValue::getKeys() const
|
|
{
|
|
if (typ != VOBJ)
|
|
throw std::runtime_error("JSON value is not an object as expected");
|
|
return keys;
|
|
}
|
|
|
|
const std::vector<UniValue>& UniValue::getValues() const
|
|
{
|
|
if (typ != VOBJ && typ != VARR)
|
|
throw std::runtime_error("JSON value is not an object or array as expected");
|
|
return values;
|
|
}
|
|
|
|
bool UniValue::get_bool() const
|
|
{
|
|
if (typ != VBOOL)
|
|
throw std::runtime_error("JSON value is not a boolean as expected");
|
|
return getBool();
|
|
}
|
|
|
|
const std::string& UniValue::get_str() const
|
|
{
|
|
if (typ != VSTR)
|
|
throw std::runtime_error("JSON value is not a string as expected");
|
|
return getValStr();
|
|
}
|
|
|
|
int UniValue::get_int() const
|
|
{
|
|
if (typ != VNUM)
|
|
throw std::runtime_error("JSON value is not an integer as expected");
|
|
int32_t retval;
|
|
if (!ParseInt32(getValStr(), &retval))
|
|
throw std::runtime_error("JSON integer out of range");
|
|
return retval;
|
|
}
|
|
|
|
int64_t UniValue::get_int64() const
|
|
{
|
|
if (typ != VNUM)
|
|
throw std::runtime_error("JSON value is not an integer as expected");
|
|
int64_t retval;
|
|
if (!ParseInt64(getValStr(), &retval))
|
|
throw std::runtime_error("JSON integer out of range");
|
|
return retval;
|
|
}
|
|
|
|
double UniValue::get_real() const
|
|
{
|
|
if (typ != VNUM)
|
|
throw std::runtime_error("JSON value is not a number as expected");
|
|
double retval;
|
|
if (!ParseDouble(getValStr(), &retval))
|
|
throw std::runtime_error("JSON double out of range");
|
|
return retval;
|
|
}
|
|
|
|
const UniValue& UniValue::get_obj() const
|
|
{
|
|
if (typ != VOBJ)
|
|
throw std::runtime_error("JSON value is not an object as expected");
|
|
return *this;
|
|
}
|
|
|
|
const UniValue& UniValue::get_array() const
|
|
{
|
|
if (typ != VARR)
|
|
throw std::runtime_error("JSON value is not an array as expected");
|
|
return *this;
|
|
}
|
|
|