// // Extractor.cpp // // Library: Data/PostgreSQL // Package: PostgreSQL // Module: Extractor // // Copyright (c) 2015, Applied Informatics Software Engineering GmbH. // and Contributors. // // SPDX-License-Identifier: BSL-1.0 // #include "Poco/Data/PostgreSQL/Extractor.h" #include "Poco/Data/Date.h" #include "Poco/Data/Time.h" #include "Poco/NumberParser.h" #include "Poco/DateTimeParser.h" #include "Poco/MemoryStream.h" #include "Poco/HexBinaryDecoder.h" #include namespace Poco { namespace Data { namespace PostgreSQL { Extractor::Extractor(StatementExecutor& st /*, ResultMetadata& md */): _statementExecutor(st) { } Extractor::~Extractor() { } bool Extractor::extract(std::size_t pos, Poco::Int8& val) { const OutputParameter& outputParameter = extractPreamble(pos); int tempVal = 0; if (isColumnNull(outputParameter) || !Poco::NumberParser::tryParse(outputParameter.pData(), tempVal)) { return false; } val = static_cast(tempVal); return true; } bool Extractor::extract(std::size_t pos, Poco::UInt8& val) { const OutputParameter& outputParameter = extractPreamble(pos); unsigned int tempVal = 0; if (isColumnNull(outputParameter)|| !Poco::NumberParser::tryParseUnsigned(outputParameter.pData(), tempVal)) { return false; } val = static_cast(tempVal); return true; } bool Extractor::extract(std::size_t pos, Poco::Int16& val) { const OutputParameter& outputParameter = extractPreamble(pos); int tempVal = 0; if (isColumnNull(outputParameter) || !Poco::NumberParser::tryParse(outputParameter.pData(), tempVal)) { return false; } val = static_cast(tempVal); return true; } bool Extractor::extract(std::size_t pos, Poco::UInt16& val) { const OutputParameter& outputParameter = extractPreamble(pos); unsigned int tempVal = 0; if (isColumnNull(outputParameter) || !Poco::NumberParser::tryParseUnsigned(outputParameter.pData(), tempVal)) { return false; } val = static_cast(tempVal); return true; } bool Extractor::extract(std::size_t pos, Poco::Int32& val) { const OutputParameter& outputParameter = extractPreamble(pos); if (isColumnNull(outputParameter) || !Poco::NumberParser::tryParse(outputParameter.pData(), val)) { return false; } return true; } bool Extractor::extract(std::size_t pos, Poco::UInt32& val) { const OutputParameter& outputParameter = extractPreamble(pos); if (isColumnNull(outputParameter) || !Poco::NumberParser::tryParseUnsigned(outputParameter.pData(), val)) { return false; } return true; } bool Extractor::extract(std::size_t pos, Poco::Int64& val) { const OutputParameter& outputParameter = extractPreamble(pos); if (isColumnNull(outputParameter) || !Poco::NumberParser::tryParse64(outputParameter.pData(), val)) { return false; } return true; } bool Extractor::extract(std::size_t pos, Poco::UInt64& val) { const OutputParameter& outputParameter = extractPreamble(pos); if (isColumnNull(outputParameter) || !Poco::NumberParser::tryParseUnsigned64(outputParameter.pData(), val)) { return false; } return true; } #ifndef POCO_INT64_IS_LONG bool Extractor::extract(std::size_t pos, long& val) { const OutputParameter& outputParameter = extractPreamble(pos); Poco::Int64 tempVal = 0; if (isColumnNull(outputParameter) || !Poco::NumberParser::tryParse64(outputParameter.pData(), tempVal)) { return false; } val = (long)tempVal; return true; } bool Extractor::extract(std::size_t pos, unsigned long& val) { const OutputParameter& outputParameter = extractPreamble(pos); Poco::UInt64 tempVal = 0; if (isColumnNull(outputParameter) || !Poco::NumberParser::tryParseUnsigned64(outputParameter.pData(), tempVal)) { return false; } val = (unsigned long)tempVal; return true; } #endif bool Extractor::extract(std::size_t pos, bool& val) { const OutputParameter& outputParameter = extractPreamble(pos); if (isColumnNull(outputParameter)) { return false; } val = 't' == *outputParameter.pData(); return true; } bool Extractor::extract(std::size_t pos, float& val) { const OutputParameter& outputParameter = extractPreamble(pos); double tempVal = 0.0; if (isColumnNull(outputParameter) || !Poco::NumberParser::tryParseFloat(outputParameter.pData(), tempVal)) { return false; } val = static_cast(tempVal); return true; } bool Extractor::extract(std::size_t pos, double& val) { const OutputParameter& outputParameter = extractPreamble(pos); if (isColumnNull(outputParameter) || !Poco::NumberParser::tryParseFloat(outputParameter.pData(), val)) { return false; } return true; } bool Extractor::extract(std::size_t pos, char& val) { const OutputParameter& outputParameter = extractPreamble(pos); if (isColumnNull(outputParameter)) { return false; } val = *outputParameter.pData(); return true; } bool Extractor::extract(std::size_t pos, std::string& val) { const OutputParameter& outputParameter = extractPreamble(pos); if (isColumnNull(outputParameter)) { return false; } val.assign(outputParameter.pData(), outputParameter.size()); return true; } bool Extractor::extract(std::size_t pos, Poco::Data::BLOB& val) { const OutputParameter& outputParameter = extractPreamble(pos); if (isColumnNull(outputParameter)) { return false; } // convert the PostgreSQL text format to binary and append to the BLOB // Format: \x10843029479abcf ... two characters for every byte const char * pBLOB = reinterpret_cast(outputParameter.pData()); std::size_t blobSize = outputParameter.size(); val = Poco::Data::BLOB(); // don't share contents with _default if (blobSize > 2 && '\\' == pBLOB[0] && 'x' == pBLOB[1]) // preamble to BYTEA data format in text form is \x { blobSize -= 2; // lose the preamble Poco::MemoryInputStream mistr(pBLOB + 2, blobSize); Poco::HexBinaryDecoder decoder(mistr); auto* pDecoderBuf = decoder.rdbuf(); blobSize /= 2; val.resize(blobSize); char* pData = reinterpret_cast(val.rawContent()); while (blobSize-- > 0) { *pData++ = pDecoderBuf->sbumpc(); } } return true; } bool Extractor::extract(std::size_t pos, Poco::Data::CLOB& val) { const OutputParameter& outputParameter = extractPreamble(pos); if (isColumnNull(outputParameter)) { return false; } val.assignRaw(outputParameter.pData(), outputParameter.size()); return true; } bool Extractor::extract(std::size_t pos, DateTime& val) { const OutputParameter& outputParameter = extractPreamble(pos); if (isColumnNull(outputParameter)) { return false; } int tzd = -1; DateTime dateTime; if (!DateTimeParser::tryParse("%Y-%m-%d %H:%M:%s", outputParameter.pData(), dateTime, tzd)) { return false; } dateTime.makeUTC(tzd); val = dateTime; return true; } bool Extractor::extract(std::size_t pos, Date& val) { const OutputParameter& outputParameter = extractPreamble(pos); if (isColumnNull(outputParameter)) { return false; } int tzd = -1; DateTime dateTime; if (!DateTimeParser::tryParse(outputParameter.pData(), dateTime, tzd)) { return false; } dateTime.makeUTC(tzd); val.assign(dateTime.year(), dateTime.month(), dateTime.day()); return true; } bool Extractor::extract(std::size_t pos, Time& val) { const OutputParameter& outputParameter = extractPreamble(pos); if (isColumnNull(outputParameter)) { return false; } int tzd = -1; DateTime dateTime; if (! DateTimeParser::tryParse("%H:%M:%s%z", outputParameter.pData(), dateTime, tzd)) { return false; } // dateTime.makeUTC(tzd); // TODO // Note: Poco::Data::Time should be extended to support the fractional components of Poco::DateTime val.assign(dateTime.hour(), dateTime.minute(), dateTime.second()); return true; } bool Extractor::extract(std::size_t pos, UUID& val) { const OutputParameter& outputParameter = extractPreamble(pos); if (isColumnNull(outputParameter)) { return false; } return val.tryParse(outputParameter.pData()); } bool Extractor::extract(std::size_t pos, Any& val) { return extractStringImpl(pos, val); } bool Extractor::extract(std::size_t pos, Dynamic::Var& val) { return extractToDynamic(pos, val); } bool Extractor::isNull(std::size_t col, std::size_t /*row*/) { const OutputParameter& outputParameter = extractPreamble(col); if (isColumnNull(outputParameter)) { return true; } return false; } void Extractor::reset() { AbstractExtractor::reset(); } const OutputParameter& Extractor::extractPreamble(std::size_t aPosition) const { if (_statementExecutor.columnsReturned() <= aPosition) { throw PostgreSQLException("Extractor: attempt to extract more parameters than query result contains"); } return _statementExecutor.resultColumn(aPosition); } bool Extractor::extractToDynamic(std::size_t pos, Dynamic::Var& val) { const OutputParameter& outputParameter = _statementExecutor.resultColumn(pos); if (isColumnNull(outputParameter)) { return false; } const std::string tempString{outputParameter.pData(), outputParameter.size()}; const Oid oid = outputParameter.internalFieldType(); bool success = false; switch (oid) { case BOOLOID: { success = true; if (tempString[0] == 't') val = true; else val = false; break; } case INT2OID: case INT4OID: case INT8OID: { Poco::Int64 tempValue = 0; success = Poco::NumberParser::tryParse64(tempString, tempValue); if (success) val = tempValue; break; } // floating point case FLOAT8OID: case FLOAT4OID: case NUMERICOID: { double tempValue = 0; success = Poco::NumberParser::tryParseFloat(tempString, tempValue); if (success) val = tempValue; break; } // character strings case CHAROID: case BPCHAROID: case VARCHAROID: default: { success = true; val = tempString; break; } // BLOB, CLOB case BYTEAOID: { Poco::Data::BLOB blob; success = extract(pos, blob); if (success) val = blob; break; } case TEXTOID: { Poco::Data::CLOB clob; success = extract(pos, clob); if (success) val = clob; break; } // date case DATEOID: { Date d; success = extract(pos, d); if (success) val = d; break; } // time case TIMEOID: case TIMETZOID: { Time t; success = extract(pos, t); if (success) val = t; break; } //timestamp case TIMESTAMPOID: case TIMESTAMPTZOID: { DateTime dt; success = extract(pos, dt); if (success) val = dt; break; } case UUIDOID: { UUID uuid; success = extract(pos, uuid); if (success) val = uuid; break; } } return success; } ////////////// // Not implemented ////////////// bool Extractor::extract(std::size_t, std::vector&) { throw NotImplementedException("std::vector extractor must be implemented."); } bool Extractor::extract(std::size_t, std::deque&) { throw NotImplementedException("std::deque extractor must be implemented."); } bool Extractor::extract(std::size_t, std::list&) { throw NotImplementedException("std::list extractor must be implemented."); } bool Extractor::extract(std::size_t, std::vector&) { throw NotImplementedException("std::vector extractor must be implemented."); } bool Extractor::extract(std::size_t, std::deque&) { throw NotImplementedException("std::deque extractor must be implemented."); } bool Extractor::extract(std::size_t, std::list&) { throw NotImplementedException("std::list extractor must be implemented."); } bool Extractor::extract(std::size_t, std::vector&) { throw NotImplementedException("std::vector extractor must be implemented."); } bool Extractor::extract(std::size_t, std::deque&) { throw NotImplementedException("std::deque extractor must be implemented."); } bool Extractor::extract(std::size_t, std::list&) { throw NotImplementedException("std::list extractor must be implemented."); } bool Extractor::extract(std::size_t, std::vector&) { throw NotImplementedException("std::vector extractor must be implemented."); } bool Extractor::extract(std::size_t, std::deque&) { throw NotImplementedException("std::deque extractor must be implemented."); } bool Extractor::extract(std::size_t, std::list&) { throw NotImplementedException("std::list extractor must be implemented."); } bool Extractor::extract(std::size_t, std::vector&) { throw NotImplementedException("std::vector extractor must be implemented."); } bool Extractor::extract(std::size_t, std::deque&) { throw NotImplementedException("std::deque extractor must be implemented."); } bool Extractor::extract(std::size_t, std::list&) { throw NotImplementedException("std::list extractor must be implemented."); } bool Extractor::extract(std::size_t, std::vector&) { throw NotImplementedException("std::vector extractor must be implemented."); } bool Extractor::extract(std::size_t, std::deque&) { throw NotImplementedException("std::deque extractor must be implemented."); } bool Extractor::extract(std::size_t, std::list&) { throw NotImplementedException("std::list extractor must be implemented."); } bool Extractor::extract(std::size_t, std::vector&) { throw NotImplementedException("std::vector extractor must be implemented."); } bool Extractor::extract(std::size_t, std::deque&) { throw NotImplementedException("std::deque extractor must be implemented."); } bool Extractor::extract(std::size_t, std::list&) { throw NotImplementedException("std::list extractor must be implemented."); } bool Extractor::extract(std::size_t, std::vector&) { throw NotImplementedException("std::vector extractor must be implemented."); } bool Extractor::extract(std::size_t, std::deque&) { throw NotImplementedException("std::deque extractor must be implemented."); } bool Extractor::extract(std::size_t, std::list&) { throw NotImplementedException("std::list extractor must be implemented."); } #ifndef POCO_INT64_IS_LONG bool Extractor::extract(std::size_t, std::vector&) { throw NotImplementedException("std::vector extractor must be implemented."); } bool Extractor::extract(std::size_t, std::deque&) { throw NotImplementedException("std::deque extractor must be implemented."); } bool Extractor::extract(std::size_t, std::list&) { throw NotImplementedException("std::list extractor must be implemented."); } #endif bool Extractor::extract(std::size_t, std::vector&) { throw NotImplementedException("std::vector extractor must be implemented."); } bool Extractor::extract(std::size_t, std::deque&) { throw NotImplementedException("std::deque extractor must be implemented."); } bool Extractor::extract(std::size_t, std::list&) { throw NotImplementedException("std::list extractor must be implemented."); } bool Extractor::extract(std::size_t, std::vector&) { throw NotImplementedException("std::vector extractor must be implemented."); } bool Extractor::extract(std::size_t, std::deque&) { throw NotImplementedException("std::deque extractor must be implemented."); } bool Extractor::extract(std::size_t, std::list&) { throw NotImplementedException("std::list extractor must be implemented."); } bool Extractor::extract(std::size_t, std::vector&) { throw NotImplementedException("std::vector extractor must be implemented."); } bool Extractor::extract(std::size_t, std::deque&) { throw NotImplementedException("std::deque extractor must be implemented."); } bool Extractor::extract(std::size_t, std::list&) { throw NotImplementedException("std::list extractor must be implemented."); } bool Extractor::extract(std::size_t, std::vector&) { throw NotImplementedException("std::vector extractor must be implemented."); } bool Extractor::extract(std::size_t, std::deque&) { throw NotImplementedException("std::deque extractor must be implemented."); } bool Extractor::extract(std::size_t, std::list&) { throw NotImplementedException("std::list extractor must be implemented."); } bool Extractor::extract(std::size_t, std::vector&) { throw NotImplementedException("std::vector extractor must be implemented."); } bool Extractor::extract(std::size_t, std::deque&) { throw NotImplementedException("std::deque extractor must be implemented."); } bool Extractor::extract(std::size_t, std::list&) { throw NotImplementedException("std::list extractor must be implemented."); } bool Extractor::extract(std::size_t, std::vector&) { throw NotImplementedException("std::vector extractor must be implemented."); } bool Extractor::extract(std::size_t, std::deque&) { throw NotImplementedException("std::deque extractor must be implemented."); } bool Extractor::extract(std::size_t, std::list&) { throw NotImplementedException("std::list extractor must be implemented."); } bool Extractor::extract(std::size_t, std::vector&) { throw NotImplementedException("std::vector extractor must be implemented."); } bool Extractor::extract(std::size_t, std::deque&) { throw NotImplementedException("std::deque extractor must be implemented."); } bool Extractor::extract(std::size_t, std::list&) { throw NotImplementedException("std::list extractor must be implemented."); } bool Extractor::extract(std::size_t, std::vector&) { throw NotImplementedException("std::vector extractor must be implemented."); } bool Extractor::extract(std::size_t, std::deque&) { throw NotImplementedException("std::deque extractor must be implemented."); } bool Extractor::extract(std::size_t, std::list&) { throw NotImplementedException("std::list extractor must be implemented."); } bool Extractor::extract(std::size_t, std::vector&) { throw NotImplementedException("std::vector extractor must be implemented."); } bool Extractor::extract(std::size_t, std::deque&) { throw NotImplementedException("std::deque extractor must be implemented."); } bool Extractor::extract(std::size_t, std::list&) { throw NotImplementedException("std::list extractor must be implemented."); } bool Extractor::extract(std::size_t, std::vector