// // ODBCStatementImpl.cpp // // $Id: //poco/Main/Data/ODBC/src/ODBCStatementImpl.cpp#3 $ // // Library: ODBC // Package: ODBC // Module: ODBCStatementImpl // // Copyright (c) 2006, Applied Informatics Software Engineering GmbH. // and Contributors. // // Permission is hereby granted, free of charge, to any person or organization // obtaining a copy of the software and accompanying documentation covered by // this license (the "Software") to use, reproduce, display, distribute, // execute, and transmit the Software, and to prepare derivative works of the // Software, and to permit third-parties to whom the Software is furnished to // do so, all subject to the following: // // The copyright notices in the Software and this entire statement, including // the above license grant, this restriction and the following disclaimer, // must be included in all copies of the Software, in whole or in part, and // all derivative works of the Software, unless such copies or derivative // works are solely in the form of machine-executable object code generated by // a source language processor. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, // FITNESS FOR A PARTICULAR PURPOSE, TITLE AND NON-INFRINGEMENT. IN NO EVENT // SHALL THE COPYRIGHT HOLDERS OR ANYONE DISTRIBUTING THE SOFTWARE BE LIABLE // FOR ANY DAMAGES OR OTHER LIABILITY, WHETHER IN CONTRACT, TORT OR OTHERWISE, // ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER // DEALINGS IN THE SOFTWARE. // #include "Poco/Data/ODBC/ODBCStatementImpl.h" #include "Poco/Data/ODBC/ConnectionHandle.h" #include "Poco/Data/ODBC/Utility.h" #include "Poco/Data/ODBC/ODBCException.h" #include "Poco/Data/AbstractPrepare.h" #include namespace Poco { namespace Data { namespace ODBC { const std::string ODBCStatementImpl::INVALID_CURSOR_STATE = "24000"; ODBCStatementImpl::ODBCStatementImpl(SessionImpl& rSession): Poco::Data::StatementImpl(rSession), _rConnection(rSession.dbc()), _stmt(rSession.dbc()), _stepCalled(false), _nextResponse(0) { checkError(SQLSetStmtAttr(_stmt, SQL_CURSOR_TYPE, (SQLPOINTER) SQL_CURSOR_FORWARD_ONLY, 0), "SQLSetStmtAttr(SQL_CURSOR_TYPE, SQL_CURSOR_FORWARD_ONLY)"); if (session().getFeature("autoBind")) { SQLSetStmtAttr(_stmt, SQL_ATTR_PARAM_BIND_TYPE, (SQLPOINTER) SQL_PARAM_BIND_BY_COLUMN, 0); } else { SQLSetStmtAttr(_stmt, SQL_ATTR_ROW_ARRAY_SIZE, (SQLPOINTER) 1, 0); } } ODBCStatementImpl::~ODBCStatementImpl() { ColumnPtrVec::iterator it = _columnPtrs.begin(); ColumnPtrVec::iterator itEnd = _columnPtrs.end(); for(; it != itEnd; ++it) delete *it; } void ODBCStatementImpl::compileImpl() { _stepCalled = false; _nextResponse = 0; std::string statement(toString()); if (statement.empty()) throw ODBCException("Empty statements are illegal"); Preparation::DataExtraction ext = session().getFeature("autoExtract") ? Preparation::DE_BOUND : Preparation::DE_MANUAL; std::size_t maxFieldSize = AnyCast(session().getProperty("maxFieldSize")); _pPreparation = new Preparation(_stmt, statement, maxFieldSize, ext); Binder::ParameterBinding bind = session().getFeature("autoBind") ? Binder::PB_IMMEDIATE : Binder::PB_AT_EXEC; TypeInfo* pDT = 0; try { Poco::Any dti = session().getProperty("dataTypeInfo"); pDT = AnyCast(dti); }catch (NotSupportedException&) { } _pBinder = new Binder(_stmt, bind, pDT); _pExtractor = new Extractor(_stmt, *_pPreparation); // This is a hack to conform to some ODBC drivers behavior (e.g. MS SQLServer) with // stored procedure calls: driver refuses to report the number of columns, unless all // parameters for the stored procedure are bound. Since number of columns is essential // information for the internal extraction creation, in order to allow for querying it, // these calls must occur before. fixupBinding(); doBind(false, true); // Following code creates internal extraction storage in case when none is provided by user. // Under normal circumstances, this is the responsibility of the Data framework. Due to some // ODBC peculiarities, for ODBC it is implemented here. Data library detects this being already // done and does not try to do it again. bool dataAvailable = hasData(); if (dataAvailable && !extractions().size()) { fillColumns(); makeExtractors(columnsReturned()); } if (Preparation::DE_BOUND == ext && dataAvailable) { Extractions& extracts = extractions(); Extractions::iterator it = extracts.begin(); Extractions::iterator itEnd = extracts.end(); for (std::size_t pos = 0; it != itEnd; ++it) { AbstractPrepare* pAP = (*it)->createPrepareObject(_pPreparation, pos); pAP->prepare(); pos += (*it)->numOfColumnsHandled(); delete pAP; } } } bool ODBCStatementImpl::canBind() const { if (!bindings().empty()) return (*bindings().begin())->canBind(); return false; } void ODBCStatementImpl::doBind(bool clear, bool reset) { if (clear) this->clear(); Bindings& binds = bindings(); if (!binds.empty()) { Bindings::iterator it = binds.begin(); Bindings::iterator itEnd = binds.end(); for (std::size_t pos = 0; it != itEnd && (*it)->canBind(); ++it) { (*it)->bind(pos); pos += (*it)->numOfColumnsHandled(); } if (reset) { it = binds.begin(); for (; it != itEnd && (*it)->canBind(); ++it) (*it)->reset(); } } } void ODBCStatementImpl::bindImpl() { doBind(); SQLRETURN rc = SQLExecute(_stmt); if (SQL_NEED_DATA == rc) putData(); else checkError(rc, "SQLExecute()"); _pBinder->sync(Binder::PD_OUT); } void ODBCStatementImpl::putData() { SQLPOINTER pParam = 0; SQLRETURN rc = SQLParamData(_stmt, &pParam); do { poco_assert_dbg (pParam); SQLINTEGER dataSize = (SQLINTEGER) _pBinder->parameterSize(pParam); if (Utility::isError(SQLPutData(_stmt, pParam, dataSize))) throw StatementException(_stmt, "SQLPutData()"); }while (SQL_NEED_DATA == (rc = SQLParamData(_stmt, &pParam))); checkError(rc, "SQLParamData()"); /* TODO: this is how manual extraction of parameters should work in practice, for the time being, we only support automatic binding for output params Binder::ParameterMap outParamMap = _pBinder->outParameters(); Binder::ParameterMap::iterator it = outParamMap.begin(); Binder::ParameterMap::iterator end = outParamMap.end(); for (int i = 1; it != end; ++it, ++i) { SQLINTEGER retLen = 0; while (SQL_NO_DATA != (rc = SQLGetData(_stmt, i, SQL_C_TYPE_TIMESTAMP, (SQLPOINTER) (it->first + retLen), it->second - retLen, &retLen))) if (0 == retLen || SQL_NULL_DATA == retLen) break; StatementException se(_stmt); std::cout << se.toString(); } */ } void ODBCStatementImpl::clear() { SQLRETURN rc = SQLCloseCursor(_stmt); if (Utility::isError(rc)) { StatementError err(_stmt); bool ignoreError = false; const StatementDiagnostics& diagnostics = err.diagnostics(); //ignore "Invalid cursor state" error //(returned by 3.x drivers when cursor is not opened) for (int i = 0; i < diagnostics.count(); ++i) { if (ignoreError = (INVALID_CURSOR_STATE == std::string(diagnostics.sqlState(i)))) { break; } } if (!ignoreError) throw StatementException(_stmt, "SQLCloseCursor()"); } } bool ODBCStatementImpl::hasNext() { if (hasData()) { if (_stepCalled) return _stepCalled = nextRowReady(); _stepCalled = true; _nextResponse = SQLFetch(_stmt); if (!nextRowReady()) return false; else if (Utility::isError(_nextResponse)) checkError(_nextResponse, "SQLFetch()"); return true; } return false; } void ODBCStatementImpl::next() { if (nextRowReady()) { poco_assert (columnsExtracted() == _pPreparation->columns()); Extractions& extracts = extractions(); Extractions::iterator it = extracts.begin(); Extractions::iterator itEnd = extracts.end(); for (std::size_t pos = 0; it != itEnd; ++it) { (*it)->extract(pos); pos += (*it)->numOfColumnsHandled(); } _stepCalled = false; } else { throw StatementException(_stmt, std::string("Iterator Error: trying to access the next value")); } } std::string ODBCStatementImpl::nativeSQL() { std::string statement = toString(); //Hopefully, double the original statement length is enough. //If it is not, the total available length is indicated in the retlen parameter, //which is in turn used to resize the buffer and request the native SQL again. SQLINTEGER length = (SQLINTEGER) statement.size() * 2; char* pNative = 0; SQLINTEGER retlen = length; do { delete [] pNative; pNative = new char[retlen]; memset(pNative, 0, retlen); length = retlen; if (Utility::isError(SQLNativeSql(_rConnection, (POCO_SQLCHAR*) statement.c_str(), (SQLINTEGER) statement.size(), (POCO_SQLCHAR*) pNative, length, &retlen))) { delete [] pNative; throw ConnectionException(_rConnection, "SQLNativeSql()"); } ++retlen;//accomodate for terminating '\0' }while (retlen > length); std::string sql(pNative); delete [] pNative; return sql; } void ODBCStatementImpl::checkError(SQLRETURN rc, const std::string& msg) { if (Utility::isError(rc)) { std::ostringstream os; os << std::endl << "Requested SQL statement: " << toString() << std::endl; os << "Native SQL statement: " << nativeSQL() << std::endl; std::string str(msg); str += os.str(); throw StatementException(_stmt, str); } } void ODBCStatementImpl::fillColumns() { Poco::UInt32 colCount = columnsReturned(); for (int i = 0; i < colCount; ++i) _columnPtrs.push_back(new ODBCColumn(_stmt, i)); } } } } // namespace Poco::Data::ODBC