sl@0
|
1 |
// Copyright (c) 2005-2010 Nokia Corporation and/or its subsidiary(-ies).
|
sl@0
|
2 |
// All rights reserved.
|
sl@0
|
3 |
// This component and the accompanying materials are made available
|
sl@0
|
4 |
// under the terms of "Eclipse Public License v1.0"
|
sl@0
|
5 |
// which accompanies this distribution, and is available
|
sl@0
|
6 |
// at the URL "http://www.eclipse.org/legal/epl-v10.html".
|
sl@0
|
7 |
//
|
sl@0
|
8 |
// Initial Contributors:
|
sl@0
|
9 |
// Nokia Corporation - initial contribution.
|
sl@0
|
10 |
//
|
sl@0
|
11 |
// Contributors:
|
sl@0
|
12 |
// NTT DOCOMO, INC - Fix for defect 1915 "SQL server panics when using long column type strings"
|
sl@0
|
13 |
//
|
sl@0
|
14 |
// Description:
|
sl@0
|
15 |
//
|
sl@0
|
16 |
|
sl@0
|
17 |
#ifndef __SQLSRVSTATEMENT_H__
|
sl@0
|
18 |
#define __SQLSRVSTATEMENT_H__
|
sl@0
|
19 |
|
sl@0
|
20 |
#include <e32base.h>
|
sl@0
|
21 |
#include <s32mem.h> //TBufBuf
|
sl@0
|
22 |
#include "SqlUtil.h" //Sql2OsErrCode()
|
sl@0
|
23 |
#include "SqlBufFlat.h" //RSqlBufFlat
|
sl@0
|
24 |
#include "SqlSrvStatementUtil.h" //Global SQL statement related functions
|
sl@0
|
25 |
|
sl@0
|
26 |
//Forward declarations
|
sl@0
|
27 |
struct sqlite3;
|
sl@0
|
28 |
struct sqlite3_stmt;
|
sl@0
|
29 |
class RWriteStream;
|
sl@0
|
30 |
class CSqlSrvStatement;
|
sl@0
|
31 |
|
sl@0
|
32 |
/////////////////////////////////////////////////////////////////////////////////////////////////////////
|
sl@0
|
33 |
////////////////////// HSqlSrvStmtParamBuf //////////////////////////////////
|
sl@0
|
34 |
/////////////////////////////////////////////////////////////////////////////////////////////////////////
|
sl@0
|
35 |
|
sl@0
|
36 |
/**
|
sl@0
|
37 |
Buffer for storing binary/text parameter values (an input from client side write stream objects or bound text/binary parameters).
|
sl@0
|
38 |
Every CSqlSrvStatement object maintains a collection of HSqlSrvStmtParamBuf objects.
|
sl@0
|
39 |
|
sl@0
|
40 |
Please, note that:
|
sl@0
|
41 |
|
sl@0
|
42 |
- The life-time of the HSqlSrvStmtParamBuf objects is controlled by a set of flags:
|
sl@0
|
43 |
= iStatementFinalized - False by default, set to true if the bound statement has been finalized.
|
sl@0
|
44 |
When DoRelease() is called, the method checks this poreperty and if the statement has been finalized,
|
sl@0
|
45 |
then obviously no more parameter binding operations can be performed and it is safe to destroy the
|
sl@0
|
46 |
current HSqlSrvStmtParamBuf object. Otherwise the HSqlSrvStmtParamBuf object won't be destroyed,
|
sl@0
|
47 |
it will be marked as "dead" - see the "iAlive" property;
|
sl@0
|
48 |
= iAlive - True by default, set to false by the DoRelease() call, if the object cannot be destroyed because
|
sl@0
|
49 |
the bound statement is not finalized yet;
|
sl@0
|
50 |
= iBufType - EBufIpcStream if the current HSqlSrvStmtParamBuf object is an IPC stream buffer, EBufSimpleBind
|
sl@0
|
51 |
if the object holds a parameter value, that has been bound not using an IPC stream. This property
|
sl@0
|
52 |
is checked when the bound statement gets finalized - inside the NotifyStatementFinalized() function.
|
sl@0
|
53 |
If the current object is not an IPC stream buffer, then it is safe to destroy the object because
|
sl@0
|
54 |
the bound statement is finalized;
|
sl@0
|
55 |
|
sl@0
|
56 |
All these checks allow keeping the buffer object alive when it is needed and code like the code bellow should work without problems:
|
sl@0
|
57 |
@code
|
sl@0
|
58 |
RSqlStatement stmt; //Prepare a statement with a binary/text parameter
|
sl@0
|
59 |
....
|
sl@0
|
60 |
RSqlParamWriteStream strm;
|
sl@0
|
61 |
strm.BindBinary(stmt,...); //Use a stream object to bind the parameter value
|
sl@0
|
62 |
strm.WriteL(...);
|
sl@0
|
63 |
strm.CommitL(); //The stream commit call will bind the streamed parameter value to the statement
|
sl@0
|
64 |
strm.Close(); //The stream close operation should not destroy the HSqlSrvStmtParamBuf object, because the parameter
|
sl@0
|
65 |
//value is kept in the buffer, and when that value was bound to the statement, the
|
sl@0
|
66 |
//SQLITE_STATIC constant was used, which means - SQLite won't make a copy of the parameter value,
|
sl@0
|
67 |
//SQLite will keep a pointer to the parameter value and use it for the Exec() operation.
|
sl@0
|
68 |
stmt.Exec(); //At this point the SQLite library should be able to get the parameter data and use it.
|
sl@0
|
69 |
@endcode
|
sl@0
|
70 |
|
sl@0
|
71 |
- Another reason for keeping the HSqlSrvStmtParamBuf object alive during the life-time of the CSqlSrvStatement object is that
|
sl@0
|
72 |
the RSqlStatement::Reset() call should not destroy the buffer. The client should be able to resuse
|
sl@0
|
73 |
the buffered parameter data for the next RSqlStatement::Exec() call. For example:
|
sl@0
|
74 |
@code
|
sl@0
|
75 |
RSqlStatement stmt; //Prepare a statement with two binary parameters
|
sl@0
|
76 |
....
|
sl@0
|
77 |
RSqlParamWriteStream strm1, strm2;
|
sl@0
|
78 |
|
sl@0
|
79 |
strm1.BindBinary(stmt,...); //Use a stream object to bind the 1st parameter value
|
sl@0
|
80 |
strm1.WriteL(...);
|
sl@0
|
81 |
strm1.CommitL();
|
sl@0
|
82 |
strm1.Close();
|
sl@0
|
83 |
|
sl@0
|
84 |
strm2.BindBinary(stmt,...); //Use a stream object to bind the 2nd parameter value
|
sl@0
|
85 |
strm2.WriteL(...);
|
sl@0
|
86 |
strm2.CommitL();
|
sl@0
|
87 |
strm2.Close();
|
sl@0
|
88 |
|
sl@0
|
89 |
stmt.Exec(); //Do the operation using the bound parameter values
|
sl@0
|
90 |
stmt.Reset(); //This call should not destroy the HSqlSrvStmtParamBuf objects that keep the parameter values
|
sl@0
|
91 |
|
sl@0
|
92 |
strm2.BindBinary(stmt,...); //Use a stream object to bind the 2nd parameter value
|
sl@0
|
93 |
strm2.WriteL(...);
|
sl@0
|
94 |
strm2.CommitL();
|
sl@0
|
95 |
strm2.Close();
|
sl@0
|
96 |
|
sl@0
|
97 |
stmt.Exec(); //Do the operation using the bound parameter values.
|
sl@0
|
98 |
//The operation should be able to use the old 1st parameter value without any problems,
|
sl@0
|
99 |
//and also the new 2nd parameter value.
|
sl@0
|
100 |
@endcode
|
sl@0
|
101 |
|
sl@0
|
102 |
- The HSqlSrvStmtParamBuf object can be synch-ed just once. The reason for that is to prevent the double commit-s,
|
sl@0
|
103 |
that will occur as a result of MStreamBuf::Close() calls on the client side, where the Close() method will try to sync the buffer.
|
sl@0
|
104 |
The single sync will prevent the SQL server from crashing, if the order of the RSqlStatement::Close() and
|
sl@0
|
105 |
RSqlParamWriteStream::Close() calls is reversed, e.g.:
|
sl@0
|
106 |
@code
|
sl@0
|
107 |
RSqlStatement stmt;
|
sl@0
|
108 |
....
|
sl@0
|
109 |
RSqlParamWriteStream strm;
|
sl@0
|
110 |
strm.BindBinary(stmt,...);
|
sl@0
|
111 |
strm.WriteL(...);
|
sl@0
|
112 |
strm.CommitL();
|
sl@0
|
113 |
stmt.Exec();
|
sl@0
|
114 |
stmt.Close();
|
sl@0
|
115 |
strm.Close();
|
sl@0
|
116 |
@endcode
|
sl@0
|
117 |
|
sl@0
|
118 |
- For BC reasons, even though the HSqlSrvStmtParamBuf object can be synch-ed just once (the operation is controlled by the iSynchDone
|
sl@0
|
119 |
flag), the client side still will be able to update the buffer content, because the related HIpcStream stream object keeps a copy
|
sl@0
|
120 |
of the buffer's base address. Since the parameter is bound using the SQLITE_STATIC constant (SQLite won't make a copy of the
|
sl@0
|
121 |
parameter value), at the moment of the Exec() call, SQLite will use the give pointer to the parameter value and will pick up
|
sl@0
|
122 |
the updated buffer even without a commit call;
|
sl@0
|
123 |
|
sl@0
|
124 |
@endcode
|
sl@0
|
125 |
|
sl@0
|
126 |
- The iSynchDone flag controls the number of buffer commit operations. It is false by default,
|
sl@0
|
127 |
set to true by the DoSynchL() call. The HSqlSrvStmtParamBuf object can be synch-ed just once, because DoSynchL()
|
sl@0
|
128 |
calls sqlite3_bind_text/binary() and the expectation is that the whole buffer with the parameter data is ready for binding;
|
sl@0
|
129 |
If the buffer is not commited by calling DoSynchL(), the buffer still will be commited by the DoSynchL() call performed from
|
sl@0
|
130 |
MStreamBuf::Close(). But in a case of a sync failure, no error will be reported to the client side;
|
sl@0
|
131 |
|
sl@0
|
132 |
@see HIpcStream
|
sl@0
|
133 |
@see CSqlSrvStatement
|
sl@0
|
134 |
|
sl@0
|
135 |
@internalComponent
|
sl@0
|
136 |
*/
|
sl@0
|
137 |
NONSHARABLE_CLASS(HSqlSrvStmtParamBuf) : public TBufBuf
|
sl@0
|
138 |
{
|
sl@0
|
139 |
public:
|
sl@0
|
140 |
enum TDataType {EBinary, EText16};
|
sl@0
|
141 |
enum TBufType {EBufSimpleBind, EBufIpcStream};
|
sl@0
|
142 |
|
sl@0
|
143 |
static inline HSqlSrvStmtParamBuf* NewL(CSqlSrvStatement& aStatement, TInt aParamIndex, HSqlSrvStmtParamBuf::TDataType aDataType,
|
sl@0
|
144 |
HSqlSrvStmtParamBuf::TBufType aBufType);
|
sl@0
|
145 |
inline void Reset(HSqlSrvStmtParamBuf::TDataType aDataType, HSqlSrvStmtParamBuf::TBufType aBufType);
|
sl@0
|
146 |
|
sl@0
|
147 |
inline const TPtrC8 SetDataL(const TDesC8& aData);
|
sl@0
|
148 |
inline const TPtrC8 Data() const;
|
sl@0
|
149 |
inline HSqlSrvStmtParamBuf::TDataType DataType() const;
|
sl@0
|
150 |
|
sl@0
|
151 |
void NotifyStatementFinalized();
|
sl@0
|
152 |
inline TInt ParamIndex() const;
|
sl@0
|
153 |
|
sl@0
|
154 |
private:
|
sl@0
|
155 |
inline HSqlSrvStmtParamBuf(CSqlSrvStatement& aStatement, TInt aParamIndex, HSqlSrvStmtParamBuf::TDataType aDataType,
|
sl@0
|
156 |
HSqlSrvStmtParamBuf::TBufType aBufType);
|
sl@0
|
157 |
virtual ~HSqlSrvStmtParamBuf();
|
sl@0
|
158 |
inline void ConstructL();
|
sl@0
|
159 |
virtual void DoSynchL();
|
sl@0
|
160 |
virtual void DoRelease();
|
sl@0
|
161 |
|
sl@0
|
162 |
private:
|
sl@0
|
163 |
enum {EExpandSize = 512};
|
sl@0
|
164 |
|
sl@0
|
165 |
CSqlSrvStatement& iStatement; //The bound CSqlSrvStatement object
|
sl@0
|
166 |
CBufFlat* iBuf; //Parameter buffer - not owned by HSqlSrvStmtParamBuf
|
sl@0
|
167 |
TInt iParamIndex; //The parameter index
|
sl@0
|
168 |
TBool iStatementFinalized;//True if the bound statement object has been finalized
|
sl@0
|
169 |
TBool iAlive; //True if DoRelease() has not been called yet
|
sl@0
|
170 |
HSqlSrvStmtParamBuf::TDataType iDataType;//The parameter type
|
sl@0
|
171 |
HSqlSrvStmtParamBuf::TBufType iBufType; //IPC stream buf or a simple "bind param" buf
|
sl@0
|
172 |
TBool iSynchDone; //True if the buffer data has been bound to the statement
|
sl@0
|
173 |
};
|
sl@0
|
174 |
|
sl@0
|
175 |
/////////////////////////////////////////////////////////////////////////////////////////////////////////
|
sl@0
|
176 |
////////////////// CSqlSrvStatement //////////////////////////////////
|
sl@0
|
177 |
/////////////////////////////////////////////////////////////////////////////////////////////////////////
|
sl@0
|
178 |
|
sl@0
|
179 |
/**
|
sl@0
|
180 |
SQL statement handle.
|
sl@0
|
181 |
|
sl@0
|
182 |
CSqlSrvStatement is a server side class which processes the client side requests for:
|
sl@0
|
183 |
- preparing 16-bit and 8-bit SQL statements;
|
sl@0
|
184 |
- executing prepared SQL statement;
|
sl@0
|
185 |
- retrieving parameter names and column names;
|
sl@0
|
186 |
- binding SQL parameters;
|
sl@0
|
187 |
- retrieving column values;
|
sl@0
|
188 |
- reseting the prepared SQL statement;
|
sl@0
|
189 |
- moving the cursor to the next record;
|
sl@0
|
190 |
|
sl@0
|
191 |
@see CSqlSrvStatement::NewLC()
|
sl@0
|
192 |
@see CSqlSrvStatement::NewLC()
|
sl@0
|
193 |
@see CSqlSrvStatement::BindL()
|
sl@0
|
194 |
@see CSqlSrvStatement::Next()
|
sl@0
|
195 |
@see CSqlSrvStatement::Reset()
|
sl@0
|
196 |
@see CSqlSrvStatement::Exec()
|
sl@0
|
197 |
@see CSqlSrvStatement::ColumnNamesL()
|
sl@0
|
198 |
@see CSqlSrvStatement::ParamNamesL()
|
sl@0
|
199 |
@see CSqlSrvStatement::ColumnValuesL()
|
sl@0
|
200 |
@see CSqlSrvStatement::ColumnSource()
|
sl@0
|
201 |
|
sl@0
|
202 |
@see HSqlSrvStmtParamBuf
|
sl@0
|
203 |
|
sl@0
|
204 |
@internalComponent
|
sl@0
|
205 |
*/
|
sl@0
|
206 |
NONSHARABLE_CLASS(CSqlSrvStatement) : public CBase
|
sl@0
|
207 |
{
|
sl@0
|
208 |
public:
|
sl@0
|
209 |
static CSqlSrvStatement* NewLC(sqlite3* aDbHandle, const TDesC16& aSqlStmt, TInt& aColumnCount, TInt& aParamCount);
|
sl@0
|
210 |
static CSqlSrvStatement* NewLC(sqlite3* aDbHandle, const TDesC8& aSqlStmt, TInt& aColumnCount, TInt& aParamCount);
|
sl@0
|
211 |
virtual ~CSqlSrvStatement();
|
sl@0
|
212 |
|
sl@0
|
213 |
void BindL(const RSqlBufFlat& aParamBuf);
|
sl@0
|
214 |
inline TInt Next();
|
sl@0
|
215 |
inline TInt Reset();
|
sl@0
|
216 |
inline TInt Exec();
|
sl@0
|
217 |
|
sl@0
|
218 |
inline const RSqlBufFlat& BufFlatL(TSqlBufFlatType aWhat) const;
|
sl@0
|
219 |
const RSqlBufFlat& ColumnNamesL();
|
sl@0
|
220 |
const RSqlBufFlat& ParamNamesL();
|
sl@0
|
221 |
const RSqlBufFlat& ColumnValuesL();
|
sl@0
|
222 |
TInt ColumnSource(TInt aColumnIndex, TPtrC8& aColumnSource) const;
|
sl@0
|
223 |
|
sl@0
|
224 |
TInt ColumnInt(TInt aColIdx) const;
|
sl@0
|
225 |
TInt64 ColumnInt64(TInt aColIdx) const;
|
sl@0
|
226 |
TReal ColumnReal(TInt aColIdx) const;
|
sl@0
|
227 |
TPtrC ColumnTextL(TInt aColIdx) const;
|
sl@0
|
228 |
TPtrC8 ColumnBinary(TInt aColIdx) const;
|
sl@0
|
229 |
|
sl@0
|
230 |
const RSqlBufFlat& GetDeclColumnTypesL();
|
sl@0
|
231 |
|
sl@0
|
232 |
HSqlSrvStmtParamBuf* GetParamBufL(TInt aParamIndex, HSqlSrvStmtParamBuf::TDataType aDataType, HSqlSrvStmtParamBuf::TBufType aBufType);
|
sl@0
|
233 |
void BindParamBufL(TInt aParamIndex);
|
sl@0
|
234 |
|
sl@0
|
235 |
private:
|
sl@0
|
236 |
inline CSqlSrvStatement();
|
sl@0
|
237 |
inline void ConstructL(sqlite3* aDbHandle, const TDesC16& aSqlStmt);
|
sl@0
|
238 |
inline void ConstructL(sqlite3* aDbHandle, const TDesC8& aSqlStmt);
|
sl@0
|
239 |
void DoCommonConstructL();
|
sl@0
|
240 |
void DestroyParamBufArray();
|
sl@0
|
241 |
void ExtendParamBufArrayL(TInt aParamIndex);
|
sl@0
|
242 |
TPtrC8 CopyAndStoreParamL(TInt aParamIndex, HSqlSrvStmtParamBuf::TDataType aDataType, const TDesC8& aParamValue);
|
sl@0
|
243 |
|
sl@0
|
244 |
private:
|
sl@0
|
245 |
sqlite3_stmt* iStmtHandle; //SQL statement handle
|
sl@0
|
246 |
TInt iColumnCount;
|
sl@0
|
247 |
TInt iParamCount;
|
sl@0
|
248 |
TSqlBufFlatType iBufFlatType; //What is in iFlatBuf data member
|
sl@0
|
249 |
RSqlBufFlat iBufFlat; //Flat buffer used for: column names, parameter names, column values
|
sl@0
|
250 |
RArray<HSqlSrvStmtParamBuf*> iParamBufArray;//An array with pointers to HSqlSrvStmtParamBuf buffers for the text/binary parameters
|
sl@0
|
251 |
|
sl@0
|
252 |
};
|
sl@0
|
253 |
|
sl@0
|
254 |
#include "SqlSrvStatement.inl"
|
sl@0
|
255 |
|
sl@0
|
256 |
#endif//__SQLSRVSTATEMENT_H__
|