Improved error checking in Value + more to array conversion control.
This commit is contained in:
parent
1fe7d3c56d
commit
d9854d81fa
3 changed files with 153 additions and 18 deletions
|
|
@ -45,24 +45,44 @@ Value::operator std::string() const
|
||||||
return m_val;
|
return m_val;
|
||||||
}
|
}
|
||||||
|
|
||||||
Value::operator short() const
|
Value::operator int16_t() const
|
||||||
{
|
{
|
||||||
return (short)std::atoi(m_val);
|
int32_t r = operator int32_t();
|
||||||
|
if (r <= std::numeric_limits<int16_t>::max()
|
||||||
|
&& r >= std::numeric_limits<int16_t>::min())
|
||||||
|
return int16_t(r);
|
||||||
|
|
||||||
|
throw std::runtime_error("Value conversion to int16_t failed (out of range)");
|
||||||
}
|
}
|
||||||
|
|
||||||
Value::operator int() const
|
Value::operator int32_t() const
|
||||||
{
|
{
|
||||||
return std::atoi(m_val);
|
const int len = std::strlen(m_val);
|
||||||
|
if (len > 0) {
|
||||||
|
char *endptr = nullptr;
|
||||||
|
long result = std::strtol(m_val, &endptr, 10);
|
||||||
|
if (endptr == m_val + len)
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
throw std::runtime_error("Value conversion to int32_t failed");
|
||||||
}
|
}
|
||||||
|
|
||||||
Value::operator Oid() const
|
Value::operator Oid() const
|
||||||
{
|
{
|
||||||
return operator int();
|
return operator int32_t();
|
||||||
}
|
}
|
||||||
|
|
||||||
Value::operator long long() const
|
Value::operator int64_t() const
|
||||||
{
|
{
|
||||||
return std::strtoull(m_val, nullptr, 10);
|
const int len = std::strlen(m_val);
|
||||||
|
if (len > 0) {
|
||||||
|
char *endptr = nullptr;
|
||||||
|
int64_t result = std::strtoll(m_val, &endptr, 10);
|
||||||
|
if (endptr == m_val + len)
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
throw std::runtime_error("Value conversion to int64_t failed");
|
||||||
|
// return std::strtoll(m_val, nullptr, 10);
|
||||||
}
|
}
|
||||||
|
|
||||||
Value::operator bool() const
|
Value::operator bool() const
|
||||||
|
|
|
||||||
|
|
@ -23,36 +23,92 @@ namespace Pgsql {
|
||||||
operator QDate() const;
|
operator QDate() const;
|
||||||
operator QTime() const;
|
operator QTime() const;
|
||||||
operator std::string() const;
|
operator std::string() const;
|
||||||
operator short() const;
|
operator int16_t() const;
|
||||||
operator int() const;
|
operator int32_t() const;
|
||||||
operator Oid() const;
|
operator Oid() const;
|
||||||
operator long long() const;
|
operator int64_t() const;
|
||||||
operator bool() const;
|
operator bool() const;
|
||||||
operator float() const;
|
operator float() const;
|
||||||
operator double() const;
|
operator double() const;
|
||||||
|
|
||||||
bool isString() const;
|
bool isString() const;
|
||||||
|
|
||||||
|
enum class NullHandling {
|
||||||
|
Ignore,
|
||||||
|
Throw
|
||||||
|
};
|
||||||
/**
|
/**
|
||||||
*
|
*
|
||||||
* \param insert_iter An insert_iterator which is used to add the elements of the array to a container.
|
* \param insert_iter An insert_iterator which is used to add the elements of the array to a container.
|
||||||
*/
|
*/
|
||||||
template <typename E, typename I>
|
template <typename E, typename I>
|
||||||
void getAsArray(I insert_iter) const
|
void getAsArray(I insert_iter, NullHandling nullhandling = NullHandling::Throw) const
|
||||||
{
|
{
|
||||||
using value_type = E;
|
using value_type = E;
|
||||||
ArrayParser parser(m_val);
|
ArrayParser parser(m_val);
|
||||||
for (;;) {
|
for (;;) {
|
||||||
auto res = parser.GetNextElem();
|
auto res = parser.GetNextElem();
|
||||||
if (res.ok) {
|
if (res.ok) {
|
||||||
|
if (res.value) {
|
||||||
std::string str(res.value->data(), res.value->length());
|
std::string str(res.value->data(), res.value->length());
|
||||||
// std::istringstream iss(str);
|
|
||||||
Value val(str.c_str(), ANYOID);
|
Value val(str.c_str(), ANYOID);
|
||||||
value_type v;
|
value_type v;
|
||||||
// iss >> v;
|
|
||||||
v << val;
|
v << val;
|
||||||
insert_iter = v;
|
insert_iter = v;
|
||||||
}
|
}
|
||||||
|
else {
|
||||||
|
if (nullhandling == NullHandling::Throw)
|
||||||
|
throw std::runtime_error("Unexpected NULL value in array");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
template <typename E, typename I>
|
||||||
|
void getAsArray(I insert_iter, const E &value_for_nulls) const
|
||||||
|
{
|
||||||
|
using value_type = E;
|
||||||
|
ArrayParser parser(m_val);
|
||||||
|
for (;;) {
|
||||||
|
auto res = parser.GetNextElem();
|
||||||
|
if (res.ok) {
|
||||||
|
if (res.value) {
|
||||||
|
std::string str(res.value->data(), res.value->length());
|
||||||
|
Value val(str.c_str(), ANYOID);
|
||||||
|
value_type v;
|
||||||
|
v << val;
|
||||||
|
insert_iter = v;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
insert_iter = value_for_nulls;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
template <typename E, typename I>
|
||||||
|
void getAsArrayOfOptional(I insert_iter) const
|
||||||
|
{
|
||||||
|
using value_type = E;
|
||||||
|
ArrayParser parser(m_val);
|
||||||
|
for (;;) {
|
||||||
|
auto res = parser.GetNextElem();
|
||||||
|
if (res.ok) {
|
||||||
|
if (res.value) {
|
||||||
|
std::string str(res.value->data(), res.value->length());
|
||||||
|
Value val(str.c_str(), ANYOID);
|
||||||
|
value_type v;
|
||||||
|
v << val;
|
||||||
|
insert_iter = v;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
insert_iter = std::nullopt;
|
||||||
|
}
|
||||||
|
}
|
||||||
else
|
else
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -87,3 +87,62 @@ TEST(Pgsql_Value, getAsArray_QDateTime)
|
||||||
|
|
||||||
ASSERT_EQ(r.size(), 2);
|
ASSERT_EQ(r.size(), 2);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
TEST(Pgsql_Value, getAsArray_throws_on_NULL)
|
||||||
|
{
|
||||||
|
Pgsql::Value v("{1,NULL,2}", TEXTARRAYOID);
|
||||||
|
std::vector<int> r;
|
||||||
|
try {
|
||||||
|
v.getAsArray<int>(std::back_inserter(r));
|
||||||
|
FAIL();
|
||||||
|
} catch (std::runtime_error &) {
|
||||||
|
SUCCEED();
|
||||||
|
} catch (...) {
|
||||||
|
FAIL();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(Pgsql_Value, getAsArray_default_on_NULL)
|
||||||
|
{
|
||||||
|
Pgsql::Value v("{1,NULL,2}", TEXTARRAYOID);
|
||||||
|
std::vector<int> r;
|
||||||
|
try {
|
||||||
|
v.getAsArray<int>(std::back_inserter(r), -1);
|
||||||
|
ASSERT_EQ(r.size(), 3);
|
||||||
|
ASSERT_EQ(r[0], 1);
|
||||||
|
ASSERT_EQ(r[1], -1);
|
||||||
|
ASSERT_EQ(r[2], 2);
|
||||||
|
} catch (...) {
|
||||||
|
FAIL();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(Pgsql_Value, getAsArray_ignore_NULL)
|
||||||
|
{
|
||||||
|
Pgsql::Value v("{1,NULL,2}", TEXTARRAYOID);
|
||||||
|
std::vector<int> r;
|
||||||
|
try {
|
||||||
|
v.getAsArray<int>(std::back_inserter(r), Value::NullHandling::Ignore);
|
||||||
|
ASSERT_EQ(r.size(), 2);
|
||||||
|
ASSERT_EQ(r[0], 1);
|
||||||
|
ASSERT_EQ(r[1], 2);
|
||||||
|
} catch (...) {
|
||||||
|
FAIL();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(Pgsql_Value, getAsArrayOptional)
|
||||||
|
{
|
||||||
|
Pgsql::Value v("{1,NULL,2}", TEXTARRAYOID);
|
||||||
|
std::vector<std::optional<int>> r;
|
||||||
|
try {
|
||||||
|
v.getAsArrayOfOptional<int>(std::back_inserter(r));
|
||||||
|
ASSERT_EQ(r.size(), 3);
|
||||||
|
ASSERT_EQ(r[0], 1);
|
||||||
|
ASSERT_EQ(r[1], std::nullopt);
|
||||||
|
ASSERT_EQ(r[2], 2);
|
||||||
|
} catch (...) {
|
||||||
|
FAIL();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue