/* * The contents of this file are subject to the Initial * Developer's Public License Version 1.0 (the "License"); * you may not use this file except in compliance with the * License. You may obtain a copy of the License at * http://www.ibphoenix.com/main.nfs?a=ibphoenix&page=ibp_idpl. * * Software distributed under the License is distributed AS IS, * WITHOUT WARRANTY OF ANY KIND, either express or implied. * See the License for the specific language governing rights * and limitations under the License. * * The Original Code was created by Adriano dos Santos Fernandes * for the Firebird Open Source RDBMS project. * * Copyright (c) 2008 Adriano dos Santos Fernandes * and all contributors signed below. * * All Rights Reserved. * Contributor(s): ______________________________________. */ #ifndef JRD_PREPARED_STATEMENT_H #define JRD_PREPARED_STATEMENT_H #include "firebird.h" #include "../jrd/common.h" #include "../jrd/dsc.h" #include "../jrd/intl.h" #include "../common/classes/alloc.h" #include "../common/classes/array.h" #include "../common/classes/auto.h" #include "../common/classes/fb_string.h" #include "../common/classes/MetaName.h" #include "../common/classes/Nullable.h" namespace Jrd { class thread_db; class jrd_tra; class Attachment; class dsql_req; class dsql_msg; class ResultSet; class PreparedStatement : public Firebird::PermanentStorage { friend class ResultSet; public: class Builder { private: enum Type { TYPE_SSHORT, TYPE_SLONG, TYPE_SINT64, TYPE_DOUBLE, TYPE_METANAME, TYPE_STRING, }; // This struct and the member outputParams are used to make the C++ undefined parameter // evaluation order do not interfere on the question marks / slots correspondence. struct OutputParam { OutputParam(const char* aChunk, size_t aNumber) : chunk(aChunk), number(aNumber) { } const char* chunk; size_t number; }; struct InputSlot { Type type; unsigned number; const void* address; const bool* specifiedAddress; }; struct OutputSlot { Type type; unsigned number; void* address; bool* specifiedAddress; }; public: Builder() : outputParams(0) { } public: // Output variables. template OutputParam operator ()(const char* chunk, Nullable& param) { OutputParam ret = (*this)(chunk, param.value); outputSlots[outputSlots.getCount() - 1].specifiedAddress = ¶m.specified; return ret; } template OutputParam operator ()(const char* chunk, T& param) { addOutput(getType(param), ¶m, outputSlots); return OutputParam(chunk, outputSlots.getCount() - 1); } // SQL text concatenation. Builder& operator <<(OutputParam outputParam) { text += " "; text += outputParam.chunk; outputSlots[outputParam.number].number = ++outputParams; return *this; } // Input variables. Builder& operator <<(const char* chunk) { text += " "; text += chunk; return *this; } template Builder& operator <<(const Nullable& param) { *this << param.value; inputSlots[inputSlots.getCount() - 1].specifiedAddress = ¶m.specified; return *this; } template Builder& operator <<(const T& param) { addInput(getType(param), ¶m, inputSlots); text += "?"; return *this; } public: const Firebird::string& getText() const { return text; } void moveFromResultSet(thread_db* tdbb, ResultSet* rs) const; void moveToStatement(thread_db* tdbb, PreparedStatement* stmt) const; private: // Make the C++ template engine return the constant for each type. static Type getType(SSHORT) { return TYPE_SSHORT; } static Type getType(SLONG) { return TYPE_SLONG; } static Type getType(SINT64) { return TYPE_SINT64; } static Type getType(double) { return TYPE_DOUBLE; } static Type getType(const Firebird::AbstractString&) { return TYPE_STRING; } static Type getType(const Firebird::MetaName&) { return TYPE_METANAME; } void addInput(Type type, const void* address, Firebird::Array& slots) { InputSlot slot; slot.type = type; slot.number = (unsigned) slots.getCount() + 1; slot.address = address; slot.specifiedAddress = NULL; slots.add(slot); } void addOutput(Type type, void* address, Firebird::Array& slots) { OutputSlot slot; slot.type = type; slot.number = (unsigned) slots.getCount() + 1; slot.address = address; slot.specifiedAddress = NULL; slots.add(slot); } private: Firebird::string text; Firebird::Array inputSlots; Firebird::Array outputSlots; unsigned outputParams; }; private: // Auxiliary class to use positional parameters with C++ variables. class PosBuilder { public: explicit PosBuilder(const Firebird::string& aText) : text(aText), params(0) { } PosBuilder& operator <<(const char* chunk) { text += chunk; return *this; } PosBuilder& operator <<(unsigned& param) { text += "?"; param = ++params; return *this; } operator const Firebird::string& () { return text; } private: Firebird::string text; unsigned params; }; public: // Create a PreparedStatement builder to use positional parameters with C++ variables. static PosBuilder build(const Firebird::string& text) { return PosBuilder(text); } // Escape a metadata name accordingly to SQL rules. static Firebird::string escapeName(const Firebird::MetaName& s) { Firebird::string ret; for (const char* p = s.begin(); p != s.end(); ++p) { ret += *p; if (*p == '\"') ret += '\"'; } return ret; } // Escape a string accordingly to SQL rules. template static Firebird::string escapeString(const T& s) { Firebird::string ret; for (const char* p = s.begin(); p != s.end(); ++p) { ret += *p; if (*p == '\'') ret += '\''; } return ret; } public: PreparedStatement(thread_db* tdbb, Firebird::MemoryPool& aPool, Attachment* attachment, jrd_tra* transaction, const Firebird::string& text, bool isInternalRequest); PreparedStatement(thread_db* tdbb, Firebird::MemoryPool& aPool, Attachment* attachment, jrd_tra* transaction, const Builder& aBuilder, bool isInternalRequest); ~PreparedStatement(); private: void init(thread_db* tdbb, Attachment* attachment, jrd_tra* transaction, const Firebird::string& text, bool isInternalRequest); public: void setDesc(thread_db* tdbb, unsigned param, const dsc& value); void setNull(unsigned param) { fb_assert(param > 0); dsc* desc = &inValues[(param - 1) * 2 + 1]; fb_assert(desc->dsc_dtype == dtype_short); *reinterpret_cast(desc->dsc_address) = -1; } void setSmallInt(thread_db* tdbb, unsigned param, SSHORT value, SCHAR scale = 0) { fb_assert(param > 0); dsc desc; desc.makeShort(scale, &value); setDesc(tdbb, param, desc); } void setInt(thread_db* tdbb, unsigned param, SLONG value, SCHAR scale = 0) { fb_assert(param > 0); dsc desc; desc.makeLong(scale, &value); setDesc(tdbb, param, desc); } void setBigInt(thread_db* tdbb, unsigned param, SINT64 value, SCHAR scale = 0) { fb_assert(param > 0); dsc desc; desc.makeInt64(scale, &value); setDesc(tdbb, param, desc); } void setDouble(thread_db* tdbb, unsigned param, double value) { fb_assert(param > 0); dsc desc; desc.makeDouble(&value); setDesc(tdbb, param, desc); } void setString(thread_db* tdbb, unsigned param, const Firebird::AbstractString& value) { fb_assert(param > 0); dsc desc; desc.makeText((USHORT) value.length(), inValues[(param - 1) * 2].getTextType(), (UCHAR*) value.c_str()); setDesc(tdbb, param, desc); } void setMetaName(thread_db* tdbb, unsigned param, const Firebird::MetaName& value) { fb_assert(param > 0); dsc desc; desc.makeText((USHORT) value.length(), CS_METADATA, (UCHAR*) value.c_str()); setDesc(tdbb, param, desc); } void execute(thread_db* tdbb, jrd_tra* transaction); ResultSet* executeQuery(thread_db* tdbb, jrd_tra* transaction); unsigned executeUpdate(thread_db* tdbb, jrd_tra* transaction); int getResultCount() const; dsql_req* getRequest() { return request; } static void parseDsqlMessage(const dsql_msg* dsqlMsg, Firebird::Array& values, Firebird::UCharBuffer& blr, Firebird::UCharBuffer& msg); static void generateBlr(const dsc* desc, Firebird::UCharBuffer& blr); private: const Builder* builder; dsql_req* request; Firebird::Array inValues, outValues; Firebird::UCharBuffer inBlr, outBlr; Firebird::UCharBuffer inMessage, outMessage; ResultSet* resultSet; }; typedef Firebird::AutoPtr AutoPreparedStatement; } // namespace #endif // JRD_PREPARED_STATEMENT_H