mirror of
https://github.com/pocoproject/poco.git
synced 2025-01-23 10:36:37 +01:00
648 lines
17 KiB
C++
648 lines
17 KiB
C++
//
|
|
// StatementImpl.h
|
|
//
|
|
// Library: Data
|
|
// Package: DataCore
|
|
// Module: StatementImpl
|
|
//
|
|
// Definition of the StatementImpl class.
|
|
//
|
|
// Copyright (c) 2006, Applied Informatics Software Engineering GmbH.
|
|
// and Contributors.
|
|
//
|
|
// SPDX-License-Identifier: BSL-1.0
|
|
//
|
|
|
|
|
|
#ifndef Data_StatementImpl_INCLUDED
|
|
#define Data_StatementImpl_INCLUDED
|
|
|
|
|
|
#include "Poco/Data/Data.h"
|
|
#include "Poco/Data/AbstractBinding.h"
|
|
#include "Poco/Data/AbstractExtraction.h"
|
|
#include "Poco/Data/Range.h"
|
|
#include "Poco/Data/Bulk.h"
|
|
#include "Poco/Data/Column.h"
|
|
#include "Poco/Data/Extraction.h"
|
|
#include "Poco/Data/BulkExtraction.h"
|
|
#include "Poco/Data/SessionImpl.h"
|
|
#include "Poco/RefCountedObject.h"
|
|
#include "Poco/String.h"
|
|
#include "Poco/Format.h"
|
|
#include "Poco/Exception.h"
|
|
#include <vector>
|
|
#include <list>
|
|
#include <deque>
|
|
#include <string>
|
|
#include <sstream>
|
|
|
|
|
|
namespace Poco {
|
|
namespace Data {
|
|
|
|
|
|
class Data_API StatementImpl
|
|
/// StatementImpl interface that subclasses must implement to define database dependent query execution.
|
|
///
|
|
/// StatementImpl's are noncopyable.
|
|
{
|
|
public:
|
|
using Ptr = Poco::SharedPtr<StatementImpl>;
|
|
|
|
enum State
|
|
{
|
|
ST_INITIALIZED,
|
|
ST_COMPILED,
|
|
ST_BOUND,
|
|
ST_PAUSED,
|
|
ST_DONE,
|
|
ST_RESET
|
|
};
|
|
|
|
enum Storage
|
|
{
|
|
STORAGE_DEQUE_IMPL,
|
|
STORAGE_VECTOR_IMPL,
|
|
STORAGE_LIST_IMPL,
|
|
STORAGE_UNKNOWN_IMPL
|
|
};
|
|
|
|
enum BulkType
|
|
{
|
|
BULK_UNDEFINED,
|
|
/// Bulk mode not defined yet.
|
|
BULK_BINDING,
|
|
/// Binding in bulk mode.
|
|
/// If extraction is present in the same statement,
|
|
/// it must also be bulk.
|
|
BULK_EXTRACTION,
|
|
/// Extraction in bulk mode.
|
|
/// If binding is present in the same statement,
|
|
/// it must also be bulk.
|
|
BULK_FORBIDDEN
|
|
/// Bulk forbidden.
|
|
/// Happens when the statement has already been
|
|
/// configured as non-bulk.
|
|
};
|
|
|
|
static const std::string DEQUE;
|
|
static const std::string VECTOR;
|
|
static const std::string LIST;
|
|
static const std::string UNKNOWN;
|
|
|
|
static const int USE_CURRENT_DATA_SET = -1;
|
|
|
|
StatementImpl(SessionImpl& rSession);
|
|
/// Creates the StatementImpl.
|
|
|
|
virtual ~StatementImpl();
|
|
/// Destroys the StatementImpl.
|
|
|
|
template <typename T>
|
|
void add(const T& t)
|
|
/// Appends SQL statement (fragments).
|
|
{
|
|
_ostr << t;
|
|
}
|
|
|
|
void addBind(AbstractBinding::Ptr pBinding);
|
|
/// Registers the Binding with the StatementImpl.
|
|
|
|
void removeBind(const std::string& name);
|
|
/// Unregisters all the bindings having specified name with the StatementImpl.
|
|
/// Bindings are released and, if this class was the sole owner, deleted.
|
|
|
|
void addExtract(AbstractExtraction::Ptr pExtraction);
|
|
/// Registers objects used for extracting data with the StatementImpl.
|
|
|
|
void setExtractionLimit(const Limit& extrLimit);
|
|
/// Changes the extractionLimit to extrLimit.
|
|
/// Per default no limit (EXTRACT_UNLIMITED) is set.
|
|
|
|
std::string toString() const;
|
|
/// Create a string version of the SQL statement.
|
|
|
|
std::size_t execute(const bool& reset = true);
|
|
/// Executes a statement. Returns the number of rows
|
|
/// extracted for statements returning data or number of rows
|
|
/// affected for all other statements (insert, update, delete).
|
|
/// If reset is true (default), the underlying bound storage is
|
|
/// reset and reused. In case of containers, this means they are
|
|
/// cleared and resized to accomodate the number of rows returned by
|
|
/// this execution step. When reset is false, data is appended to the
|
|
/// bound containers during multiple execute calls.
|
|
|
|
void executeDirect(const std::string& query);
|
|
/// Execute query directly.
|
|
|
|
void reset();
|
|
/// Resets the statement, so that we can reuse all bindings and re-execute again.
|
|
|
|
State getState() const;
|
|
/// Returns the state of the Statement.
|
|
|
|
void setStorage(Storage storage);
|
|
/// Sets the storage type for this statement;
|
|
|
|
void setStorage(const std::string& storage);
|
|
/// Sets the storage type for this statement;
|
|
|
|
Storage getStorage() const;
|
|
/// Returns the storage type for this statement.
|
|
|
|
std::size_t extractionCount() const;
|
|
/// Returns the number of extraction storage buffers associated
|
|
/// with the statement.
|
|
|
|
std::size_t dataSetCount() const;
|
|
/// Returns the number of data sets associated with the statement.
|
|
|
|
protected:
|
|
virtual std::size_t columnsReturned() const = 0;
|
|
/// Returns number of columns returned by query.
|
|
|
|
virtual int affectedRowCount() const = 0;
|
|
/// Returns the number of affected rows.
|
|
/// Used to find out the number of rows affected by insert, delete or update.
|
|
///
|
|
/// Some back-ends may return a negative number in certain circumstances (e.g.
|
|
/// some ODBC drivers when this function is called after a select statement
|
|
/// execution).
|
|
|
|
virtual const MetaColumn& metaColumn(std::size_t pos) const = 0;
|
|
/// Returns column meta data.
|
|
|
|
const MetaColumn& metaColumn(const std::string& name) const;
|
|
/// Returns column meta data.
|
|
|
|
virtual bool hasNext() = 0;
|
|
/// Returns true if a call to next() will return data.
|
|
///
|
|
/// Note that the implementation must support
|
|
/// several consecutive calls to hasNext without data getting lost,
|
|
/// ie. hasNext(); hasNext(); next() must be equal to hasNext(); next();
|
|
|
|
virtual std::size_t next() = 0;
|
|
/// Retrieves the next row or set of rows from the resultset and
|
|
/// returns the number of rows retreved.
|
|
///
|
|
/// Will throw, if the resultset is empty.
|
|
/// Expects the statement to be compiled and bound.
|
|
|
|
virtual bool canBind() const = 0;
|
|
/// Returns true if another bind is possible.
|
|
|
|
virtual bool canCompile() const = 0;
|
|
/// Returns true if another compile is possible.
|
|
|
|
virtual void compileImpl() = 0;
|
|
/// Compiles the statement, doesn't bind yet.
|
|
|
|
virtual void bindImpl() = 0;
|
|
/// Binds parameters.
|
|
|
|
virtual void execDirectImpl(const std::string& query);
|
|
/// Execute query directly.
|
|
|
|
virtual AbstractExtraction::ExtractorPtr extractor() = 0;
|
|
/// Returns the concrete extractor used by the statement.
|
|
|
|
const AbstractExtractionVec& extractions() const;
|
|
/// Returns the const reference to extractions vector.
|
|
|
|
AbstractExtractionVec& extractions();
|
|
/// Returns the reference to extractions vector.
|
|
|
|
void fixupExtraction();
|
|
/// Sets the AbstractExtractor at the extractors.
|
|
|
|
Limit::SizeT getExtractionLimit();
|
|
/// Returns the extraction limit value.
|
|
|
|
const Limit& extractionLimit() const;
|
|
/// Returns the extraction limit.
|
|
|
|
std::size_t columnsExtracted(int dataSet = USE_CURRENT_DATA_SET) const;
|
|
/// Returns the number of columns that the extractors handle.
|
|
|
|
std::size_t rowsExtracted(int dataSet = USE_CURRENT_DATA_SET) const;
|
|
/// Returns the number of rows extracted for current data set.
|
|
/// Default value (USE_CURRENT_DATA_SET) indicates current data set (if any).
|
|
|
|
std::size_t subTotalRowCount(int dataSet = USE_CURRENT_DATA_SET) const;
|
|
/// Returns the number of rows extracted so far for the data set.
|
|
/// Default value indicates current data set (if any).
|
|
|
|
void makeExtractors(std::size_t count);
|
|
/// Determines the type of the internal extraction container and
|
|
/// calls the extraction creation function (addInternalExtract)
|
|
/// with appropriate data type and container type arguments.
|
|
///
|
|
/// This function is only called in cases when there is data
|
|
/// returned by query, but no data storage supplied by user.
|
|
///
|
|
/// The type of the internal container is determined in the
|
|
/// following order:
|
|
/// 1. If statement has the container type set, the type is used.
|
|
/// 2. If statement does not have the container type set,
|
|
/// session is queried for container type setting. If the
|
|
/// session container type setting is found, it is used.
|
|
/// 3. If neither session nor statement have the internal
|
|
/// container type set, std::deque is used.
|
|
///
|
|
/// Supported internal extraction container types are:
|
|
/// - std::deque (default)
|
|
/// - std::vector
|
|
/// - std::list
|
|
|
|
SessionImpl& session();
|
|
/// Rteurns session associated with this statement.
|
|
|
|
const SessionImpl& session() const;
|
|
/// Rteurns session associated with this statement.
|
|
|
|
virtual AbstractBinding::BinderPtr binder() = 0;
|
|
/// Returns the concrete binder used by the statement.
|
|
|
|
const AbstractBindingVec& bindings() const;
|
|
/// Returns the const reference to bindings vector.
|
|
|
|
AbstractBindingVec& bindings();
|
|
/// Returns the reference to bindings.
|
|
|
|
void fixupBinding();
|
|
/// Sets the AbstractBinder at the bindings.
|
|
|
|
void resetBinding();
|
|
/// Resets binding so it can be reused again.
|
|
|
|
virtual bool isStoredProcedure() const;
|
|
/// Returns true if the statement is stored procedure.
|
|
/// Used as a help to determine whether to automatically create the
|
|
/// internal extractions when no outside extraction is supplied.
|
|
/// The reason for this function is to prevent unnecessary internal
|
|
/// extraction creation in cases (behavior exhibited by some ODBC drivers)
|
|
/// when there is data available from the stored procedure call
|
|
/// statement execution but no external extraction is supplied (as is
|
|
/// usually the case when stored procedures are called). In such cases
|
|
/// no storage is needed because output parameters serve as storage.
|
|
/// At the Data framework level, this function always returns false.
|
|
/// When connector-specific behavior is desired, it should be overridden
|
|
/// by the statement implementation.
|
|
|
|
std::size_t currentDataSet() const;
|
|
/// Returns the current data set.
|
|
|
|
std::size_t activateNextDataSet();
|
|
/// Returns the next data set index, or throws NoDataException if the last
|
|
/// data set was reached.
|
|
|
|
std::size_t activatePreviousDataSet();
|
|
/// Returns the previous data set index, or throws NoDataException if the last
|
|
/// data set was reached.
|
|
|
|
bool hasMoreDataSets() const;
|
|
/// Returns true if there are data sets not activated yet.
|
|
|
|
private:
|
|
void compile();
|
|
/// Compiles the statement.
|
|
|
|
void bind();
|
|
/// Binds the statement, if not yet bound.
|
|
|
|
std::size_t executeWithLimit();
|
|
/// Executes with an upper limit set. Returns the number of rows
|
|
/// extracted for statements returning data or number of rows
|
|
/// affected for all other statements (insert, update, delete).
|
|
|
|
std::size_t executeWithoutLimit();
|
|
/// Executes without an upper limit set. Returns the number of rows
|
|
/// extracted for statements returning data or number of rows
|
|
/// affected for all other statements (insert, update, delete).
|
|
|
|
void resetExtraction();
|
|
/// Resets extraction so it can be reused again.
|
|
|
|
template <class C>
|
|
SharedPtr<InternalExtraction<C>> createExtract(const MetaColumn& mc)
|
|
{
|
|
C* pData = new C;
|
|
Column<C>* pCol = new Column<C>(mc, pData);
|
|
return new InternalExtraction<C>(*pData, pCol, Poco::UInt32(currentDataSet()));
|
|
}
|
|
|
|
template <class C>
|
|
SharedPtr<InternalBulkExtraction<C>> createBulkExtract(const MetaColumn& mc)
|
|
{
|
|
C* pData = new C;
|
|
Column<C>* pCol = new Column<C>(mc, pData);
|
|
return new InternalBulkExtraction<C>(*pData,
|
|
pCol,
|
|
static_cast<Poco::UInt32>(getExtractionLimit()),
|
|
Position(static_cast<Poco::UInt32>(currentDataSet())));
|
|
}
|
|
|
|
template <class T>
|
|
void addInternalExtract(const MetaColumn& mc)
|
|
/// Creates and adds the internal extraction.
|
|
///
|
|
/// The decision about internal extraction container is done
|
|
/// in a following way:
|
|
///
|
|
/// If this statement has _storage member set, that setting
|
|
/// overrides the session setting for storage, otherwise the
|
|
/// session setting is used.
|
|
/// If neither this statement nor the session have the storage
|
|
/// type set, std::vector is the default container type used.
|
|
{
|
|
std::string storage;
|
|
|
|
switch (_storage)
|
|
{
|
|
case STORAGE_DEQUE_IMPL:
|
|
storage = DEQUE; break;
|
|
case STORAGE_VECTOR_IMPL:
|
|
storage = VECTOR; break;
|
|
case STORAGE_LIST_IMPL:
|
|
storage = LIST; break;
|
|
case STORAGE_UNKNOWN_IMPL:
|
|
storage = AnyCast<std::string>(session().getProperty("storage"));
|
|
break;
|
|
}
|
|
|
|
if (storage.empty()) storage = DEQUE;
|
|
|
|
if (0 == icompare(DEQUE, storage))
|
|
{
|
|
if (!isBulkExtraction())
|
|
addExtract(createExtract<std::deque<T>>(mc));
|
|
else
|
|
addExtract(createBulkExtract<std::deque<T>>(mc));
|
|
}
|
|
else if (0 == icompare(VECTOR, storage))
|
|
{
|
|
if (!isBulkExtraction())
|
|
addExtract(createExtract<std::vector<T>>(mc));
|
|
else
|
|
addExtract(createBulkExtract<std::vector<T>>(mc));
|
|
}
|
|
else if (0 == icompare(LIST, storage))
|
|
{
|
|
if (!isBulkExtraction())
|
|
addExtract(createExtract<std::list<T>>(mc));
|
|
else
|
|
addExtract(createBulkExtract<std::list<T>>(mc));
|
|
}
|
|
}
|
|
|
|
bool isNull(std::size_t col, std::size_t row) const;
|
|
/// Returns true if the value in [col, row] is null.
|
|
|
|
void forbidBulk();
|
|
/// Forbids bulk operations.
|
|
|
|
void setBulkBinding();
|
|
/// Sets the bulk binding flag.
|
|
|
|
void setBulkExtraction(const Bulk& l);
|
|
/// Sets the bulk extraction flag and extraction limit.
|
|
|
|
void resetBulk();
|
|
/// Resets the bulk extraction and binding flag.
|
|
|
|
bool bulkBindingAllowed() const;
|
|
/// Returns true if statement can be set to bind data in bulk.
|
|
/// Once bulk binding is set for a statement, it can be
|
|
/// neither altered nor mixed with non-bulk mode binding.
|
|
|
|
bool bulkExtractionAllowed() const;
|
|
/// Returns true if statement can be set to extract data in bulk.
|
|
/// Once bulk extraction is set for a statement, it can be
|
|
/// neither altered nor mixed with non-bulk mode extraction.
|
|
|
|
bool isBulkBinding() const;
|
|
/// Returns true if statement is set to bind data in bulk.
|
|
|
|
bool isBulkExtraction() const;
|
|
/// Returns true if statement is set to extract data in bulk.
|
|
|
|
bool isBulkSupported() const;
|
|
/// Returns true if connector and session support bulk operation.
|
|
|
|
void formatSQL(std::vector<Any>& arguments);
|
|
/// Formats the SQL string by filling in placeholders with values from supplied vector.
|
|
|
|
void assignSubTotal(bool reset);
|
|
|
|
StatementImpl(const StatementImpl& stmt);
|
|
StatementImpl& operator = (const StatementImpl& stmt);
|
|
|
|
using CountVec = std::vector<std::size_t>;
|
|
|
|
std::atomic<State> _state;
|
|
Limit _extrLimit;
|
|
std::size_t _lowerLimit;
|
|
std::vector<int> _columnsExtracted;
|
|
SessionImpl& _rSession;
|
|
Storage _storage;
|
|
std::ostringstream _ostr;
|
|
AbstractBindingVec _bindings;
|
|
AbstractExtractionVecVec _extractors;
|
|
std::size_t _curDataSet;
|
|
BulkType _bulkBinding;
|
|
BulkType _bulkExtraction;
|
|
CountVec _subTotalRowCount;
|
|
|
|
friend class Statement;
|
|
};
|
|
|
|
|
|
//
|
|
// inlines
|
|
//
|
|
|
|
|
|
inline void StatementImpl::addBind(AbstractBinding::Ptr pBinding)
|
|
{
|
|
poco_check_ptr (pBinding);
|
|
_bindings.push_back(pBinding);
|
|
}
|
|
|
|
|
|
inline std::string StatementImpl::toString() const
|
|
{
|
|
return _ostr.str();
|
|
}
|
|
|
|
|
|
inline const AbstractBindingVec& StatementImpl::bindings() const
|
|
{
|
|
return _bindings;
|
|
}
|
|
|
|
|
|
inline AbstractBindingVec& StatementImpl::bindings()
|
|
{
|
|
return _bindings;
|
|
}
|
|
|
|
|
|
inline const AbstractExtractionVec& StatementImpl::extractions() const
|
|
{
|
|
poco_assert (_curDataSet < _extractors.size());
|
|
return _extractors[_curDataSet];
|
|
}
|
|
|
|
|
|
inline AbstractExtractionVec& StatementImpl::extractions()
|
|
{
|
|
poco_assert (_curDataSet < _extractors.size());
|
|
return _extractors[_curDataSet];
|
|
}
|
|
|
|
|
|
inline StatementImpl::State StatementImpl::getState() const
|
|
{
|
|
return _state;
|
|
}
|
|
|
|
|
|
inline SessionImpl& StatementImpl::session()
|
|
{
|
|
return _rSession;
|
|
}
|
|
|
|
|
|
inline const SessionImpl& StatementImpl::session() const
|
|
{
|
|
return _rSession;
|
|
}
|
|
|
|
|
|
inline void StatementImpl::setStorage(Storage storage)
|
|
{
|
|
_storage = storage;
|
|
}
|
|
|
|
|
|
inline StatementImpl::Storage StatementImpl::getStorage() const
|
|
{
|
|
return _storage;
|
|
}
|
|
|
|
|
|
inline std::size_t StatementImpl::extractionCount() const
|
|
{
|
|
return static_cast<std::size_t>(extractions().size());
|
|
}
|
|
|
|
|
|
inline std::size_t StatementImpl::dataSetCount() const
|
|
{
|
|
return static_cast<std::size_t>(_extractors.size());
|
|
}
|
|
|
|
|
|
inline bool StatementImpl::isStoredProcedure() const
|
|
{
|
|
return false;
|
|
}
|
|
|
|
|
|
inline bool StatementImpl::isNull(std::size_t col, std::size_t row) const
|
|
{
|
|
try
|
|
{
|
|
return extractions().at(col)->isNull(row);
|
|
}
|
|
catch (std::out_of_range& ex)
|
|
{
|
|
throw RangeException(ex.what());
|
|
}
|
|
}
|
|
|
|
|
|
inline std::size_t StatementImpl::currentDataSet() const
|
|
{
|
|
return _curDataSet;
|
|
}
|
|
|
|
|
|
inline Limit::SizeT StatementImpl::getExtractionLimit()
|
|
{
|
|
return _extrLimit.value();
|
|
}
|
|
|
|
|
|
inline const Limit& StatementImpl::extractionLimit() const
|
|
{
|
|
return _extrLimit;
|
|
}
|
|
|
|
|
|
inline void StatementImpl::forbidBulk()
|
|
{
|
|
_bulkBinding = BULK_FORBIDDEN;
|
|
_bulkExtraction = BULK_FORBIDDEN;
|
|
}
|
|
|
|
|
|
inline void StatementImpl::setBulkBinding()
|
|
{
|
|
_bulkBinding = BULK_BINDING;
|
|
}
|
|
|
|
|
|
inline bool StatementImpl::bulkBindingAllowed() const
|
|
{
|
|
return BULK_UNDEFINED == _bulkBinding ||
|
|
BULK_BINDING == _bulkBinding;
|
|
}
|
|
|
|
|
|
inline bool StatementImpl::bulkExtractionAllowed() const
|
|
{
|
|
return BULK_UNDEFINED == _bulkExtraction ||
|
|
BULK_EXTRACTION == _bulkExtraction;
|
|
}
|
|
|
|
|
|
inline bool StatementImpl::isBulkBinding() const
|
|
{
|
|
return BULK_BINDING == _bulkBinding;
|
|
}
|
|
|
|
|
|
inline bool StatementImpl::isBulkExtraction() const
|
|
{
|
|
return BULK_EXTRACTION == _bulkExtraction;
|
|
}
|
|
|
|
|
|
inline void StatementImpl::resetBulk()
|
|
{
|
|
_bulkExtraction = BULK_UNDEFINED;
|
|
_bulkBinding = BULK_UNDEFINED;\
|
|
setExtractionLimit(Limit(Limit::LIMIT_UNLIMITED, false, false));
|
|
}
|
|
|
|
|
|
inline bool StatementImpl::isBulkSupported() const
|
|
{
|
|
return _rSession.getFeature("bulk");
|
|
}
|
|
|
|
|
|
inline bool StatementImpl::hasMoreDataSets() const
|
|
{
|
|
return currentDataSet() + 1 < dataSetCount();
|
|
}
|
|
|
|
|
|
} } // namespace Poco::Data
|
|
|
|
|
|
#endif // Data_StatementImpl_INCLUDED
|