14fccf43aSdan 
29b1c62d4Sdrh #if defined(SQLITE_ENABLE_SESSION) && defined(SQLITE_ENABLE_PREUPDATE_HOOK)
34fccf43aSdan #include "sqlite3session.h"
44fccf43aSdan #include <assert.h>
54fccf43aSdan #include <string.h>
64fccf43aSdan 
75d8a2984Sdan #ifndef SQLITE_AMALGAMATION
84fccf43aSdan # include "sqliteInt.h"
94fccf43aSdan # include "vdbeInt.h"
105d8a2984Sdan #endif
114fccf43aSdan 
124fccf43aSdan typedef struct SessionTable SessionTable;
134fccf43aSdan typedef struct SessionChange SessionChange;
14296c7658Sdan typedef struct SessionBuffer SessionBuffer;
15ef7a6304Sdan typedef struct SessionInput SessionInput;
16ef7a6304Sdan 
17ef7a6304Sdan /*
18ef7a6304Sdan ** Minimum chunk size used by streaming versions of functions.
19ef7a6304Sdan */
20f1a08ad8Sdrh #ifndef SESSIONS_STRM_CHUNK_SIZE
214757c658Sdan # ifdef SQLITE_TEST
22f1a08ad8Sdrh #   define SESSIONS_STRM_CHUNK_SIZE 64
234757c658Sdan # else
24f1a08ad8Sdrh #   define SESSIONS_STRM_CHUNK_SIZE 1024
25f1a08ad8Sdrh # endif
264757c658Sdan #endif
274fccf43aSdan 
281f48e67dSdan static int sessions_strm_chunk_size = SESSIONS_STRM_CHUNK_SIZE;
291f48e67dSdan 
30cf8e9144Sdan typedef struct SessionHook SessionHook;
31cf8e9144Sdan struct SessionHook {
32cf8e9144Sdan   void *pCtx;
33cf8e9144Sdan   int (*xOld)(void*,int,sqlite3_value**);
34cf8e9144Sdan   int (*xNew)(void*,int,sqlite3_value**);
35cf8e9144Sdan   int (*xCount)(void*);
36cf8e9144Sdan   int (*xDepth)(void*);
37cf8e9144Sdan };
38cf8e9144Sdan 
39296c7658Sdan /*
40296c7658Sdan ** Session handle structure.
41296c7658Sdan */
424fccf43aSdan struct sqlite3_session {
434fccf43aSdan   sqlite3 *db;                    /* Database handle session is attached to */
444fccf43aSdan   char *zDb;                      /* Name of database session is attached to */
456d29a4feSdan   int bEnableSize;                /* True if changeset_size() enabled */
46296c7658Sdan   int bEnable;                    /* True if currently recording */
47b4480e94Sdan   int bIndirect;                  /* True if all changes are indirect */
48ff4d0f41Sdan   int bAutoAttach;                /* True to auto-attach tables */
494fccf43aSdan   int rc;                         /* Non-zero if an error has occurred */
507531a5a3Sdan   void *pFilterCtx;               /* First argument to pass to xTableFilter */
517531a5a3Sdan   int (*xTableFilter)(void *pCtx, const char *zTab);
520cb735b9Sdan   i64 nMalloc;                    /* Number of bytes of data allocated */
53a23a873fSdan   i64 nMaxChangesetSize;
541611e5a3Sdan   sqlite3_value *pZeroBlob;       /* Value containing X'' */
554fccf43aSdan   sqlite3_session *pNext;         /* Next session object on same db. */
564fccf43aSdan   SessionTable *pTable;           /* List of attached tables */
57cf8e9144Sdan   SessionHook hook;               /* APIs to grab new and old data with */
584fccf43aSdan };
594fccf43aSdan 
604fccf43aSdan /*
61ef7a6304Sdan ** Instances of this structure are used to build strings or binary records.
62ef7a6304Sdan */
63ef7a6304Sdan struct SessionBuffer {
64ef7a6304Sdan   u8 *aBuf;                       /* Pointer to changeset buffer */
65ef7a6304Sdan   int nBuf;                       /* Size of buffer aBuf */
66ef7a6304Sdan   int nAlloc;                     /* Size of allocation containing aBuf */
67ef7a6304Sdan };
68ef7a6304Sdan 
69ef7a6304Sdan /*
7016228167Sdan ** An object of this type is used internally as an abstraction for
7116228167Sdan ** input data. Input data may be supplied either as a single large buffer
7216228167Sdan ** (e.g. sqlite3changeset_start()) or using a stream function (e.g.
73f1a08ad8Sdrh **  sqlite3changeset_start_strm()).
74ef7a6304Sdan */
75ef7a6304Sdan struct SessionInput {
763e259bcdSdan   int bNoDiscard;                 /* If true, do not discard in InputBuffer() */
77d9151526Sdan   int iCurrent;                   /* Offset in aData[] of current change */
784757c658Sdan   int iNext;                      /* Offset in aData[] of next change */
794757c658Sdan   u8 *aData;                      /* Pointer to buffer containing changeset */
804757c658Sdan   int nData;                      /* Number of bytes in aData */
814757c658Sdan 
82ef7a6304Sdan   SessionBuffer buf;              /* Current read buffer */
83ef7a6304Sdan   int (*xInput)(void*, void*, int*);        /* Input stream call (or NULL) */
84ef7a6304Sdan   void *pIn;                                /* First argument to xInput */
85ef7a6304Sdan   int bEof;                       /* Set to true after xInput finished */
86ef7a6304Sdan };
87ef7a6304Sdan 
88ef7a6304Sdan /*
89296c7658Sdan ** Structure for changeset iterators.
90296c7658Sdan */
91296c7658Sdan struct sqlite3_changeset_iter {
92ef7a6304Sdan   SessionInput in;                /* Input buffer or stream */
93ef7a6304Sdan   SessionBuffer tblhdr;           /* Buffer to hold apValue/zTab/abPK/ */
9473b3c055Sdan   int bPatchset;                  /* True if this is a patchset */
9544748f27Sdan   int bInvert;                    /* True to invert changeset */
961e25d20cSdan   int bSkipEmpty;                 /* Skip noop UPDATE changes */
97296c7658Sdan   int rc;                         /* Iterator error code */
98296c7658Sdan   sqlite3_stmt *pConflict;        /* Points to conflicting row, if any */
99296c7658Sdan   char *zTab;                     /* Current table */
100296c7658Sdan   int nCol;                       /* Number of columns in zTab */
101296c7658Sdan   int op;                         /* Current operation */
102b4480e94Sdan   int bIndirect;                  /* True if current change was indirect */
103244593c8Sdan   u8 *abPK;                       /* Primary key array */
104296c7658Sdan   sqlite3_value **apValue;        /* old.* and new.* values */
105296c7658Sdan };
106296c7658Sdan 
107296c7658Sdan /*
1084fccf43aSdan ** Each session object maintains a set of the following structures, one
1094fccf43aSdan ** for each table the session object is monitoring. The structures are
1104fccf43aSdan ** stored in a linked list starting at sqlite3_session.pTable.
1114fccf43aSdan **
1124fccf43aSdan ** The keys of the SessionTable.aChange[] hash table are all rows that have
1134fccf43aSdan ** been modified in any way since the session object was attached to the
1144fccf43aSdan ** table.
1154fccf43aSdan **
1164fccf43aSdan ** The data associated with each hash-table entry is a structure containing
1174fccf43aSdan ** a subset of the initial values that the modified row contained at the
1184fccf43aSdan ** start of the session. Or no initial values if the row was inserted.
1194fccf43aSdan */
1204fccf43aSdan struct SessionTable {
1214fccf43aSdan   SessionTable *pNext;
1224fccf43aSdan   char *zName;                    /* Local name of table */
1234fccf43aSdan   int nCol;                       /* Number of columns in table zName */
1243739f298Sdan   int bStat1;                     /* True if this is sqlite_stat1 */
125e8d5648eSdan   const char **azCol;             /* Column names */
126e8d5648eSdan   u8 *abPK;                       /* Array of primary key flags */
127296c7658Sdan   int nEntry;                     /* Total number of entries in hash table */
1284fccf43aSdan   int nChange;                    /* Size of apChange[] array */
1294fccf43aSdan   SessionChange **apChange;       /* Hash table buckets */
1304fccf43aSdan };
1314fccf43aSdan 
1324fccf43aSdan /*
1334fccf43aSdan ** RECORD FORMAT:
1344fccf43aSdan **
1354fccf43aSdan ** The following record format is similar to (but not compatible with) that
1364fccf43aSdan ** used in SQLite database files. This format is used as part of the
1374fccf43aSdan ** change-set binary format, and so must be architecture independent.
1384fccf43aSdan **
1394fccf43aSdan ** Unlike the SQLite database record format, each field is self-contained -
1404fccf43aSdan ** there is no separation of header and data. Each field begins with a
1414fccf43aSdan ** single byte describing its type, as follows:
1424fccf43aSdan **
1434fccf43aSdan **       0x00: Undefined value.
1444fccf43aSdan **       0x01: Integer value.
1454fccf43aSdan **       0x02: Real value.
1464fccf43aSdan **       0x03: Text value.
1474fccf43aSdan **       0x04: Blob value.
1484fccf43aSdan **       0x05: SQL NULL value.
1494fccf43aSdan **
1504fccf43aSdan ** Note that the above match the definitions of SQLITE_INTEGER, SQLITE_TEXT
1514fccf43aSdan ** and so on in sqlite3.h. For undefined and NULL values, the field consists
1524fccf43aSdan ** only of the single type byte. For other types of values, the type byte
1534fccf43aSdan ** is followed by:
1544fccf43aSdan **
1554fccf43aSdan **   Text values:
1564fccf43aSdan **     A varint containing the number of bytes in the value (encoded using
1574fccf43aSdan **     UTF-8). Followed by a buffer containing the UTF-8 representation
1584fccf43aSdan **     of the text value. There is no nul terminator.
1594fccf43aSdan **
1604fccf43aSdan **   Blob values:
1614fccf43aSdan **     A varint containing the number of bytes in the value, followed by
1624fccf43aSdan **     a buffer containing the value itself.
1634fccf43aSdan **
1644fccf43aSdan **   Integer values:
1654fccf43aSdan **     An 8-byte big-endian integer value.
1664fccf43aSdan **
1674fccf43aSdan **   Real values:
1684fccf43aSdan **     An 8-byte big-endian IEEE 754-2008 real value.
1694fccf43aSdan **
1704fccf43aSdan ** Varint values are encoded in the same way as varints in the SQLite
1714fccf43aSdan ** record format.
1724fccf43aSdan **
1734fccf43aSdan ** CHANGESET FORMAT:
1744fccf43aSdan **
1754fccf43aSdan ** A changeset is a collection of DELETE, UPDATE and INSERT operations on
1764fccf43aSdan ** one or more tables. Operations on a single table are grouped together,
1774fccf43aSdan ** but may occur in any order (i.e. deletes, updates and inserts are all
1784fccf43aSdan ** mixed together).
1794fccf43aSdan **
1804fccf43aSdan ** Each group of changes begins with a table header:
1814fccf43aSdan **
1824fccf43aSdan **   1 byte: Constant 0x54 (capital 'T')
183730bb805Sdan **   Varint: Number of columns in the table.
18473b3c055Sdan **   nCol bytes: 0x01 for PK columns, 0x00 otherwise.
1854fccf43aSdan **   N bytes: Unqualified table name (encoded using UTF-8). Nul-terminated.
1864fccf43aSdan **
1874fccf43aSdan ** Followed by one or more changes to the table.
1884fccf43aSdan **
189c8be6437Sdrh **   1 byte: Either SQLITE_INSERT (0x12), UPDATE (0x17) or DELETE (0x09).
1905d607a6eSdan **   1 byte: The "indirect-change" flag.
1914fccf43aSdan **   old.* record: (delete and update only)
1924fccf43aSdan **   new.* record: (insert and update only)
19373b3c055Sdan **
194730bb805Sdan ** The "old.*" and "new.*" records, if present, are N field records in the
195730bb805Sdan ** format described above under "RECORD FORMAT", where N is the number of
196730bb805Sdan ** columns in the table. The i'th field of each record is associated with
197730bb805Sdan ** the i'th column of the table, counting from left to right in the order
198730bb805Sdan ** in which columns were declared in the CREATE TABLE statement.
199730bb805Sdan **
200730bb805Sdan ** The new.* record that is part of each INSERT change contains the values
201730bb805Sdan ** that make up the new row. Similarly, the old.* record that is part of each
202730bb805Sdan ** DELETE change contains the values that made up the row that was deleted
203730bb805Sdan ** from the database. In the changeset format, the records that are part
204730bb805Sdan ** of INSERT or DELETE changes never contain any undefined (type byte 0x00)
205730bb805Sdan ** fields.
206730bb805Sdan **
207730bb805Sdan ** Within the old.* record associated with an UPDATE change, all fields
208730bb805Sdan ** associated with table columns that are not PRIMARY KEY columns and are
209730bb805Sdan ** not modified by the UPDATE change are set to "undefined". Other fields
210730bb805Sdan ** are set to the values that made up the row before the UPDATE that the
211730bb805Sdan ** change records took place. Within the new.* record, fields associated
212730bb805Sdan ** with table columns modified by the UPDATE change contain the new
213730bb805Sdan ** values. Fields associated with table columns that are not modified
214730bb805Sdan ** are set to "undefined".
215730bb805Sdan **
21673b3c055Sdan ** PATCHSET FORMAT:
21773b3c055Sdan **
21873b3c055Sdan ** A patchset is also a collection of changes. It is similar to a changeset,
219730bb805Sdan ** but leaves undefined those fields that are not useful if no conflict
220730bb805Sdan ** resolution is required when applying the changeset.
22173b3c055Sdan **
22273b3c055Sdan ** Each group of changes begins with a table header:
22373b3c055Sdan **
22473b3c055Sdan **   1 byte: Constant 0x50 (capital 'P')
225730bb805Sdan **   Varint: Number of columns in the table.
22673b3c055Sdan **   nCol bytes: 0x01 for PK columns, 0x00 otherwise.
22773b3c055Sdan **   N bytes: Unqualified table name (encoded using UTF-8). Nul-terminated.
22873b3c055Sdan **
22973b3c055Sdan ** Followed by one or more changes to the table.
23073b3c055Sdan **
231c8be6437Sdrh **   1 byte: Either SQLITE_INSERT (0x12), UPDATE (0x17) or DELETE (0x09).
23273b3c055Sdan **   1 byte: The "indirect-change" flag.
233730bb805Sdan **   single record: (PK fields for DELETE, PK and modified fields for UPDATE,
234730bb805Sdan **                   full record for INSERT).
235730bb805Sdan **
236730bb805Sdan ** As in the changeset format, each field of the single record that is part
237730bb805Sdan ** of a patchset change is associated with the correspondingly positioned
238730bb805Sdan ** table column, counting from left to right within the CREATE TABLE
239730bb805Sdan ** statement.
240730bb805Sdan **
241730bb805Sdan ** For a DELETE change, all fields within the record except those associated
242f01d3a7eSdan ** with PRIMARY KEY columns are omitted. The PRIMARY KEY fields contain the
243f01d3a7eSdan ** values identifying the row to delete.
244730bb805Sdan **
245730bb805Sdan ** For an UPDATE change, all fields except those associated with PRIMARY KEY
246730bb805Sdan ** columns and columns that are modified by the UPDATE are set to "undefined".
247730bb805Sdan ** PRIMARY KEY fields contain the values identifying the table row to update,
248730bb805Sdan ** and fields associated with modified columns contain the new column values.
249730bb805Sdan **
250730bb805Sdan ** The records associated with INSERT changes are in the same format as for
251730bb805Sdan ** changesets. It is not possible for a record associated with an INSERT
252730bb805Sdan ** change to contain a field set to "undefined".
253b74cf4b6Sdan **
254b74cf4b6Sdan ** REBASE BLOB FORMAT:
255b74cf4b6Sdan **
256b74cf4b6Sdan ** A rebase blob may be output by sqlite3changeset_apply_v2() and its
257b74cf4b6Sdan ** streaming equivalent for use with the sqlite3_rebaser APIs to rebase
258b74cf4b6Sdan ** existing changesets. A rebase blob contains one entry for each conflict
259b74cf4b6Sdan ** resolved using either the OMIT or REPLACE strategies within the apply_v2()
260b74cf4b6Sdan ** call.
261b74cf4b6Sdan **
262b74cf4b6Sdan ** The format used for a rebase blob is very similar to that used for
263b74cf4b6Sdan ** changesets. All entries related to a single table are grouped together.
264b74cf4b6Sdan **
265b74cf4b6Sdan ** Each group of entries begins with a table header in changeset format:
266b74cf4b6Sdan **
267b74cf4b6Sdan **   1 byte: Constant 0x54 (capital 'T')
268b74cf4b6Sdan **   Varint: Number of columns in the table.
269b74cf4b6Sdan **   nCol bytes: 0x01 for PK columns, 0x00 otherwise.
270b74cf4b6Sdan **   N bytes: Unqualified table name (encoded using UTF-8). Nul-terminated.
271b74cf4b6Sdan **
272b74cf4b6Sdan ** Followed by one or more entries associated with the table.
273b74cf4b6Sdan **
274b74cf4b6Sdan **   1 byte: Either SQLITE_INSERT (0x12), DELETE (0x09).
275b74cf4b6Sdan **   1 byte: Flag. 0x01 for REPLACE, 0x00 for OMIT.
276b74cf4b6Sdan **   record: (in the record format defined above).
277b74cf4b6Sdan **
278b74cf4b6Sdan ** In a rebase blob, the first field is set to SQLITE_INSERT if the change
279b74cf4b6Sdan ** that caused the conflict was an INSERT or UPDATE, or to SQLITE_DELETE if
280b74cf4b6Sdan ** it was a DELETE. The second field is set to 0x01 if the conflict
281b74cf4b6Sdan ** resolution strategy was REPLACE, or 0x00 if it was OMIT.
282b74cf4b6Sdan **
283b74cf4b6Sdan ** If the change that caused the conflict was a DELETE, then the single
284b74cf4b6Sdan ** record is a copy of the old.* record from the original changeset. If it
285b74cf4b6Sdan ** was an INSERT, then the single record is a copy of the new.* record. If
286b74cf4b6Sdan ** the conflicting change was an UPDATE, then the single record is a copy
287b74cf4b6Sdan ** of the new.* record with the PK fields filled in based on the original
288b74cf4b6Sdan ** old.* record.
2894fccf43aSdan */
2904fccf43aSdan 
2914fccf43aSdan /*
2924fccf43aSdan ** For each row modified during a session, there exists a single instance of
2934fccf43aSdan ** this structure stored in a SessionTable.aChange[] hash table.
2944fccf43aSdan */
2954fccf43aSdan struct SessionChange {
296a23a873fSdan   u8 op;                          /* One of UPDATE, DELETE, INSERT */
297a23a873fSdan   u8 bIndirect;                   /* True if this change is "indirect" */
298a23a873fSdan   int nMaxSize;                   /* Max size of eventual changeset record */
2994fccf43aSdan   int nRecord;                    /* Number of bytes in buffer aRecord[] */
3004fccf43aSdan   u8 *aRecord;                    /* Buffer containing old.* record */
3014fccf43aSdan   SessionChange *pNext;           /* For hash-table collisions */
3024fccf43aSdan };
3034fccf43aSdan 
304296c7658Sdan /*
305296c7658Sdan ** Write a varint with value iVal into the buffer at aBuf. Return the
306296c7658Sdan ** number of bytes written.
307296c7658Sdan */
sessionVarintPut(u8 * aBuf,int iVal)308296c7658Sdan static int sessionVarintPut(u8 *aBuf, int iVal){
309296c7658Sdan   return putVarint32(aBuf, iVal);
3104fccf43aSdan }
3114fccf43aSdan 
312296c7658Sdan /*
313296c7658Sdan ** Return the number of bytes required to store value iVal as a varint.
314296c7658Sdan */
sessionVarintLen(int iVal)315296c7658Sdan static int sessionVarintLen(int iVal){
316296c7658Sdan   return sqlite3VarintLen(iVal);
317296c7658Sdan }
318296c7658Sdan 
319296c7658Sdan /*
320296c7658Sdan ** Read a varint value from aBuf[] into *piVal. Return the number of
321296c7658Sdan ** bytes read.
322296c7658Sdan */
sessionVarintGet(u8 * aBuf,int * piVal)3234fccf43aSdan static int sessionVarintGet(u8 *aBuf, int *piVal){
324296c7658Sdan   return getVarint32(aBuf, *piVal);
3254fccf43aSdan }
3264fccf43aSdan 
32748cd59a5Sdrh /* Load an unaligned and unsigned 32-bit integer */
32848cd59a5Sdrh #define SESSION_UINT32(x) (((u32)(x)[0]<<24)|((x)[1]<<16)|((x)[2]<<8)|(x)[3])
32948cd59a5Sdrh 
330296c7658Sdan /*
331296c7658Sdan ** Read a 64-bit big-endian integer value from buffer aRec[]. Return
332296c7658Sdan ** the value read.
333296c7658Sdan */
sessionGetI64(u8 * aRec)3344fccf43aSdan static sqlite3_int64 sessionGetI64(u8 *aRec){
33548cd59a5Sdrh   u64 x = SESSION_UINT32(aRec);
33648cd59a5Sdrh   u32 y = SESSION_UINT32(aRec+4);
33748cd59a5Sdrh   x = (x<<32) + y;
33848cd59a5Sdrh   return (sqlite3_int64)x;
3394fccf43aSdan }
3404fccf43aSdan 
3414fccf43aSdan /*
342296c7658Sdan ** Write a 64-bit big-endian integer value to the buffer aBuf[].
343296c7658Sdan */
sessionPutI64(u8 * aBuf,sqlite3_int64 i)344296c7658Sdan static void sessionPutI64(u8 *aBuf, sqlite3_int64 i){
345296c7658Sdan   aBuf[0] = (i>>56) & 0xFF;
346296c7658Sdan   aBuf[1] = (i>>48) & 0xFF;
347296c7658Sdan   aBuf[2] = (i>>40) & 0xFF;
348296c7658Sdan   aBuf[3] = (i>>32) & 0xFF;
349296c7658Sdan   aBuf[4] = (i>>24) & 0xFF;
350296c7658Sdan   aBuf[5] = (i>>16) & 0xFF;
351296c7658Sdan   aBuf[6] = (i>> 8) & 0xFF;
352296c7658Sdan   aBuf[7] = (i>> 0) & 0xFF;
353296c7658Sdan }
354296c7658Sdan 
355296c7658Sdan /*
3564fccf43aSdan ** This function is used to serialize the contents of value pValue (see
3574fccf43aSdan ** comment titled "RECORD FORMAT" above).
3584fccf43aSdan **
3594fccf43aSdan ** If it is non-NULL, the serialized form of the value is written to
3604fccf43aSdan ** buffer aBuf. *pnWrite is set to the number of bytes written before
3614fccf43aSdan ** returning. Or, if aBuf is NULL, the only thing this function does is
3624fccf43aSdan ** set *pnWrite.
3634fccf43aSdan **
3644fccf43aSdan ** If no error occurs, SQLITE_OK is returned. Or, if an OOM error occurs
3654fccf43aSdan ** within a call to sqlite3_value_text() (may fail if the db is utf-16))
3664fccf43aSdan ** SQLITE_NOMEM is returned.
3674fccf43aSdan */
sessionSerializeValue(u8 * aBuf,sqlite3_value * pValue,sqlite3_int64 * pnWrite)3684fccf43aSdan static int sessionSerializeValue(
3694fccf43aSdan   u8 *aBuf,                       /* If non-NULL, write serialized value here */
3704fccf43aSdan   sqlite3_value *pValue,          /* Value to serialize */
3712d77d80aSdrh   sqlite3_int64 *pnWrite          /* IN/OUT: Increment by bytes written */
3724fccf43aSdan ){
373296c7658Sdan   int nByte;                      /* Size of serialized value in bytes */
3744fccf43aSdan 
37580fe2d93Sdan   if( pValue ){
37680fe2d93Sdan     int eType;                    /* Value type (SQLITE_NULL, TEXT etc.) */
37780fe2d93Sdan 
3784fccf43aSdan     eType = sqlite3_value_type(pValue);
3794fccf43aSdan     if( aBuf ) aBuf[0] = eType;
3804fccf43aSdan 
3814fccf43aSdan     switch( eType ){
3824fccf43aSdan       case SQLITE_NULL:
3834fccf43aSdan         nByte = 1;
3844fccf43aSdan         break;
3854fccf43aSdan 
3864fccf43aSdan       case SQLITE_INTEGER:
3874fccf43aSdan       case SQLITE_FLOAT:
3884fccf43aSdan         if( aBuf ){
3894fccf43aSdan           /* TODO: SQLite does something special to deal with mixed-endian
3904fccf43aSdan           ** floating point values (e.g. ARM7). This code probably should
3914fccf43aSdan           ** too.  */
3924fccf43aSdan           u64 i;
3934fccf43aSdan           if( eType==SQLITE_INTEGER ){
3944fccf43aSdan             i = (u64)sqlite3_value_int64(pValue);
3954fccf43aSdan           }else{
3964fccf43aSdan             double r;
3974fccf43aSdan             assert( sizeof(double)==8 && sizeof(u64)==8 );
3984fccf43aSdan             r = sqlite3_value_double(pValue);
3994fccf43aSdan             memcpy(&i, &r, 8);
4004fccf43aSdan           }
401296c7658Sdan           sessionPutI64(&aBuf[1], i);
4024fccf43aSdan         }
4034fccf43aSdan         nByte = 9;
4044fccf43aSdan         break;
4054fccf43aSdan 
4064e895da1Sdan       default: {
40780fe2d93Sdan         u8 *z;
40880fe2d93Sdan         int n;
40980fe2d93Sdan         int nVarint;
41080fe2d93Sdan 
4114e895da1Sdan         assert( eType==SQLITE_TEXT || eType==SQLITE_BLOB );
41280fe2d93Sdan         if( eType==SQLITE_TEXT ){
41380fe2d93Sdan           z = (u8 *)sqlite3_value_text(pValue);
41480fe2d93Sdan         }else{
41580fe2d93Sdan           z = (u8 *)sqlite3_value_blob(pValue);
41680fe2d93Sdan         }
41780fe2d93Sdan         n = sqlite3_value_bytes(pValue);
4183cc89d95Sdan         if( z==0 && (eType!=SQLITE_BLOB || n>0) ) return SQLITE_NOMEM;
41980fe2d93Sdan         nVarint = sessionVarintLen(n);
42080fe2d93Sdan 
4214fccf43aSdan         if( aBuf ){
4224fccf43aSdan           sessionVarintPut(&aBuf[1], n);
42311a9ad56Sdrh           if( n>0 ) memcpy(&aBuf[nVarint + 1], z, n);
4244fccf43aSdan         }
4254fccf43aSdan 
4264fccf43aSdan         nByte = 1 + nVarint + n;
4274fccf43aSdan         break;
4284fccf43aSdan       }
4294fccf43aSdan     }
43080fe2d93Sdan   }else{
43180fe2d93Sdan     nByte = 1;
43280fe2d93Sdan     if( aBuf ) aBuf[0] = '\0';
43380fe2d93Sdan   }
4344fccf43aSdan 
435fa122adaSdan   if( pnWrite ) *pnWrite += nByte;
4364fccf43aSdan   return SQLITE_OK;
4374fccf43aSdan }
4384fccf43aSdan 
4390cb735b9Sdan /*
4400cb735b9Sdan ** Allocate and return a pointer to a buffer nByte bytes in size. If
4410cb735b9Sdan ** pSession is not NULL, increase the sqlite3_session.nMalloc variable
4420cb735b9Sdan ** by the number of bytes allocated.
4430cb735b9Sdan */
sessionMalloc64(sqlite3_session * pSession,i64 nByte)4440cb735b9Sdan static void *sessionMalloc64(sqlite3_session *pSession, i64 nByte){
4450cb735b9Sdan   void *pRet = sqlite3_malloc64(nByte);
4460cb735b9Sdan   if( pSession ) pSession->nMalloc += sqlite3_msize(pRet);
4470cb735b9Sdan   return pRet;
4480cb735b9Sdan }
4490cb735b9Sdan 
4500cb735b9Sdan /*
4510cb735b9Sdan ** Free buffer pFree, which must have been allocated by an earlier
4520cb735b9Sdan ** call to sessionMalloc64(). If pSession is not NULL, decrease the
4530cb735b9Sdan ** sqlite3_session.nMalloc counter by the number of bytes freed.
4540cb735b9Sdan */
sessionFree(sqlite3_session * pSession,void * pFree)4550cb735b9Sdan static void sessionFree(sqlite3_session *pSession, void *pFree){
4560cb735b9Sdan   if( pSession ) pSession->nMalloc -= sqlite3_msize(pFree);
4570cb735b9Sdan   sqlite3_free(pFree);
4580cb735b9Sdan }
459fa122adaSdan 
460798693b2Sdan /*
461798693b2Sdan ** This macro is used to calculate hash key values for data structures. In
462798693b2Sdan ** order to use this macro, the entire data structure must be represented
463798693b2Sdan ** as a series of unsigned integers. In order to calculate a hash-key value
464798693b2Sdan ** for a data structure represented as three such integers, the macro may
465798693b2Sdan ** then be used as follows:
466798693b2Sdan **
467798693b2Sdan **    int hash_key_value;
468798693b2Sdan **    hash_key_value = HASH_APPEND(0, <value 1>);
469798693b2Sdan **    hash_key_value = HASH_APPEND(hash_key_value, <value 2>);
470798693b2Sdan **    hash_key_value = HASH_APPEND(hash_key_value, <value 3>);
471798693b2Sdan **
472798693b2Sdan ** In practice, the data structures this macro is used for are the primary
473798693b2Sdan ** key values of modified rows.
474798693b2Sdan */
4754131639cSdan #define HASH_APPEND(hash, add) ((hash) << 3) ^ (hash) ^ (unsigned int)(add)
476798693b2Sdan 
477798693b2Sdan /*
478798693b2Sdan ** Append the hash of the 64-bit integer passed as the second argument to the
479798693b2Sdan ** hash-key value passed as the first. Return the new hash-key value.
480798693b2Sdan */
sessionHashAppendI64(unsigned int h,i64 i)4814131639cSdan static unsigned int sessionHashAppendI64(unsigned int h, i64 i){
482e8d5648eSdan   h = HASH_APPEND(h, i & 0xFFFFFFFF);
483e8d5648eSdan   return HASH_APPEND(h, (i>>32)&0xFFFFFFFF);
484e8d5648eSdan }
485798693b2Sdan 
486798693b2Sdan /*
487798693b2Sdan ** Append the hash of the blob passed via the second and third arguments to
488798693b2Sdan ** the hash-key value passed as the first. Return the new hash-key value.
489798693b2Sdan */
sessionHashAppendBlob(unsigned int h,int n,const u8 * z)4904131639cSdan static unsigned int sessionHashAppendBlob(unsigned int h, int n, const u8 *z){
491e8d5648eSdan   int i;
492e8d5648eSdan   for(i=0; i<n; i++) h = HASH_APPEND(h, z[i]);
493e8d5648eSdan   return h;
494e8d5648eSdan }
495e8d5648eSdan 
4964fccf43aSdan /*
497798693b2Sdan ** Append the hash of the data type passed as the second argument to the
498798693b2Sdan ** hash-key value passed as the first. Return the new hash-key value.
499798693b2Sdan */
sessionHashAppendType(unsigned int h,int eType)500798693b2Sdan static unsigned int sessionHashAppendType(unsigned int h, int eType){
501798693b2Sdan   return HASH_APPEND(h, eType);
502798693b2Sdan }
503798693b2Sdan 
504798693b2Sdan /*
5054131639cSdan ** This function may only be called from within a pre-update callback.
5064131639cSdan ** It calculates a hash based on the primary key values of the old.* or
507798693b2Sdan ** new.* row currently available and, assuming no error occurs, writes it to
508798693b2Sdan ** *piHash before returning. If the primary key contains one or more NULL
509798693b2Sdan ** values, *pbNullPK is set to true before returning.
510798693b2Sdan **
511798693b2Sdan ** If an error occurs, an SQLite error code is returned and the final values
512798693b2Sdan ** of *piHash asn *pbNullPK are undefined. Otherwise, SQLITE_OK is returned
513798693b2Sdan ** and the output variables are set as described above.
5144fccf43aSdan */
sessionPreupdateHash(sqlite3_session * pSession,SessionTable * pTab,int bNew,int * piHash,int * pbNullPK)515798693b2Sdan static int sessionPreupdateHash(
516cf8e9144Sdan   sqlite3_session *pSession,      /* Session object that owns pTab */
517e8d5648eSdan   SessionTable *pTab,             /* Session table handle */
518e8d5648eSdan   int bNew,                       /* True to hash the new.* PK */
51927453faeSdan   int *piHash,                    /* OUT: Hash value */
520798693b2Sdan   int *pbNullPK                   /* OUT: True if there are NULL values in PK */
521e8d5648eSdan ){
5224131639cSdan   unsigned int h = 0;             /* Hash value to return */
5234131639cSdan   int i;                          /* Used to iterate through columns */
524e8d5648eSdan 
52527453faeSdan   assert( *pbNullPK==0 );
526cf8e9144Sdan   assert( pTab->nCol==pSession->hook.xCount(pSession->hook.pCtx) );
527e8d5648eSdan   for(i=0; i<pTab->nCol; i++){
528e8d5648eSdan     if( pTab->abPK[i] ){
529e8d5648eSdan       int rc;
530e8d5648eSdan       int eType;
531e8d5648eSdan       sqlite3_value *pVal;
532e8d5648eSdan 
533e8d5648eSdan       if( bNew ){
534cf8e9144Sdan         rc = pSession->hook.xNew(pSession->hook.pCtx, i, &pVal);
535e8d5648eSdan       }else{
536cf8e9144Sdan         rc = pSession->hook.xOld(pSession->hook.pCtx, i, &pVal);
537e8d5648eSdan       }
53812ca0b56Sdan       if( rc!=SQLITE_OK ) return rc;
539e8d5648eSdan 
540e8d5648eSdan       eType = sqlite3_value_type(pVal);
541798693b2Sdan       h = sessionHashAppendType(h, eType);
5426734007dSdan       if( eType==SQLITE_INTEGER || eType==SQLITE_FLOAT ){
543e8d5648eSdan         i64 iVal;
544e8d5648eSdan         if( eType==SQLITE_INTEGER ){
545e8d5648eSdan           iVal = sqlite3_value_int64(pVal);
546e8d5648eSdan         }else{
547e8d5648eSdan           double rVal = sqlite3_value_double(pVal);
548e8d5648eSdan           assert( sizeof(iVal)==8 && sizeof(rVal)==8 );
549e8d5648eSdan           memcpy(&iVal, &rVal, 8);
550e8d5648eSdan         }
551e8d5648eSdan         h = sessionHashAppendI64(h, iVal);
5526734007dSdan       }else if( eType==SQLITE_TEXT || eType==SQLITE_BLOB ){
5536734007dSdan         const u8 *z;
5543cc89d95Sdan         int n;
5556734007dSdan         if( eType==SQLITE_TEXT ){
5566734007dSdan           z = (const u8 *)sqlite3_value_text(pVal);
5576734007dSdan         }else{
5586734007dSdan           z = (const u8 *)sqlite3_value_blob(pVal);
559e8d5648eSdan         }
5603cc89d95Sdan         n = sqlite3_value_bytes(pVal);
5613cc89d95Sdan         if( !z && (eType!=SQLITE_BLOB || n>0) ) return SQLITE_NOMEM;
5623cc89d95Sdan         h = sessionHashAppendBlob(h, n, z);
5636734007dSdan       }else{
56427453faeSdan         assert( eType==SQLITE_NULL );
5651611e5a3Sdan         assert( pTab->bStat1==0 || i!=1 );
56627453faeSdan         *pbNullPK = 1;
567e8d5648eSdan       }
568e8d5648eSdan     }
569e8d5648eSdan   }
570e8d5648eSdan 
571e8d5648eSdan   *piHash = (h % pTab->nChange);
572e8d5648eSdan   return SQLITE_OK;
573e8d5648eSdan }
574e8d5648eSdan 
5754131639cSdan /*
5766cda207fSdan ** The buffer that the argument points to contains a serialized SQL value.
5776cda207fSdan ** Return the number of bytes of space occupied by the value (including
5786cda207fSdan ** the type byte).
5796cda207fSdan */
sessionSerialLen(u8 * a)5806cda207fSdan static int sessionSerialLen(u8 *a){
5816cda207fSdan   int e = *a;
5826cda207fSdan   int n;
58324a0c453Sdan   if( e==0 || e==0xFF ) return 1;
5846cda207fSdan   if( e==SQLITE_NULL ) return 1;
5856cda207fSdan   if( e==SQLITE_INTEGER || e==SQLITE_FLOAT ) return 9;
5866cda207fSdan   return sessionVarintGet(&a[1], &n) + 1 + n;
5876cda207fSdan }
5886cda207fSdan 
5896cda207fSdan /*
5905d607a6eSdan ** Based on the primary key values stored in change aRecord, calculate a
591798693b2Sdan ** hash key. Assume the has table has nBucket buckets. The hash keys
5924131639cSdan ** calculated by this function are compatible with those calculated by
5934131639cSdan ** sessionPreupdateHash().
59464277f4aSdan **
59564277f4aSdan ** The bPkOnly argument is non-zero if the record at aRecord[] is from
59664277f4aSdan ** a patchset DELETE. In this case the non-PK fields are omitted entirely.
5974131639cSdan */
sessionChangeHash(SessionTable * pTab,int bPkOnly,u8 * aRecord,int nBucket)5984131639cSdan static unsigned int sessionChangeHash(
5994131639cSdan   SessionTable *pTab,             /* Table handle */
60064277f4aSdan   int bPkOnly,                    /* Record consists of PK fields only */
6015d607a6eSdan   u8 *aRecord,                    /* Change record */
6024131639cSdan   int nBucket                     /* Assume this many buckets in hash table */
603e8d5648eSdan ){
6044131639cSdan   unsigned int h = 0;             /* Value to return */
6054131639cSdan   int i;                          /* Used to iterate through columns */
6065d607a6eSdan   u8 *a = aRecord;                /* Used to iterate through change record */
607e8d5648eSdan 
608e8d5648eSdan   for(i=0; i<pTab->nCol; i++){
6096cda207fSdan     int eType = *a;
610e8d5648eSdan     int isPK = pTab->abPK[i];
61164277f4aSdan     if( bPkOnly && isPK==0 ) continue;
612e8d5648eSdan 
61327453faeSdan     /* It is not possible for eType to be SQLITE_NULL here. The session
61427453faeSdan     ** module does not record changes for rows with NULL values stored in
61527453faeSdan     ** primary key columns. */
61627453faeSdan     assert( eType==SQLITE_INTEGER || eType==SQLITE_FLOAT
61727453faeSdan          || eType==SQLITE_TEXT || eType==SQLITE_BLOB
6186cda207fSdan          || eType==SQLITE_NULL || eType==0
61927453faeSdan     );
6201611e5a3Sdan     assert( !isPK || (eType!=0 && eType!=SQLITE_NULL) );
62127453faeSdan 
6226cda207fSdan     if( isPK ){
6236cda207fSdan       a++;
624798693b2Sdan       h = sessionHashAppendType(h, eType);
62527453faeSdan       if( eType==SQLITE_INTEGER || eType==SQLITE_FLOAT ){
6266734007dSdan         h = sessionHashAppendI64(h, sessionGetI64(a));
627e8d5648eSdan         a += 8;
6281611e5a3Sdan       }else{
629e8d5648eSdan         int n;
630e8d5648eSdan         a += sessionVarintGet(a, &n);
6316734007dSdan         h = sessionHashAppendBlob(h, n, a);
632e8d5648eSdan         a += n;
633e8d5648eSdan       }
6346cda207fSdan     }else{
6356cda207fSdan       a += sessionSerialLen(a);
6366cda207fSdan     }
637e8d5648eSdan   }
638e8d5648eSdan   return (h % nBucket);
639e8d5648eSdan }
640e8d5648eSdan 
641798693b2Sdan /*
642798693b2Sdan ** Arguments aLeft and aRight are pointers to change records for table pTab.
643798693b2Sdan ** This function returns true if the two records apply to the same row (i.e.
644798693b2Sdan ** have the same values stored in the primary key columns), or false
645798693b2Sdan ** otherwise.
646798693b2Sdan */
sessionChangeEqual(SessionTable * pTab,int bLeftPkOnly,u8 * aLeft,int bRightPkOnly,u8 * aRight)6475d607a6eSdan static int sessionChangeEqual(
648798693b2Sdan   SessionTable *pTab,             /* Table used for PK definition */
649a71d2371Sdan   int bLeftPkOnly,                /* True if aLeft[] contains PK fields only */
6505d607a6eSdan   u8 *aLeft,                      /* Change record */
651a71d2371Sdan   int bRightPkOnly,               /* True if aRight[] contains PK fields only */
6525d607a6eSdan   u8 *aRight                      /* Change record */
6535d607a6eSdan ){
654798693b2Sdan   u8 *a1 = aLeft;                 /* Cursor to iterate through aLeft */
655798693b2Sdan   u8 *a2 = aRight;                /* Cursor to iterate through aRight */
656798693b2Sdan   int iCol;                       /* Used to iterate through table columns */
6575d607a6eSdan 
658798693b2Sdan   for(iCol=0; iCol<pTab->nCol; iCol++){
65914faa061Sdan     if( pTab->abPK[iCol] ){
6605d607a6eSdan       int n1 = sessionSerialLen(a1);
6615d607a6eSdan       int n2 = sessionSerialLen(a2);
6625d607a6eSdan 
663bd45374cSdan       if( n1!=n2 || memcmp(a1, a2, n1) ){
6645d607a6eSdan         return 0;
6655d607a6eSdan       }
66614faa061Sdan       a1 += n1;
66714faa061Sdan       a2 += n2;
66814faa061Sdan     }else{
66914faa061Sdan       if( bLeftPkOnly==0 ) a1 += sessionSerialLen(a1);
67014faa061Sdan       if( bRightPkOnly==0 ) a2 += sessionSerialLen(a2);
67114faa061Sdan     }
6725d607a6eSdan   }
6735d607a6eSdan 
6745d607a6eSdan   return 1;
6755d607a6eSdan }
6765d607a6eSdan 
677798693b2Sdan /*
678798693b2Sdan ** Arguments aLeft and aRight both point to buffers containing change
679798693b2Sdan ** records with nCol columns. This function "merges" the two records into
680798693b2Sdan ** a single records which is written to the buffer at *paOut. *paOut is
681798693b2Sdan ** then set to point to one byte after the last byte written before
682798693b2Sdan ** returning.
683798693b2Sdan **
684798693b2Sdan ** The merging of records is done as follows: For each column, if the
685798693b2Sdan ** aRight record contains a value for the column, copy the value from
686798693b2Sdan ** their. Otherwise, if aLeft contains a value, copy it. If neither
687798693b2Sdan ** record contains a value for a given column, then neither does the
688798693b2Sdan ** output record.
689798693b2Sdan */
sessionMergeRecord(u8 ** paOut,int nCol,u8 * aLeft,u8 * aRight)6905d607a6eSdan static void sessionMergeRecord(
6915d607a6eSdan   u8 **paOut,
692798693b2Sdan   int nCol,
6935d607a6eSdan   u8 *aLeft,
6945d607a6eSdan   u8 *aRight
6955d607a6eSdan ){
696798693b2Sdan   u8 *a1 = aLeft;                 /* Cursor used to iterate through aLeft */
697798693b2Sdan   u8 *a2 = aRight;                /* Cursor used to iterate through aRight */
698798693b2Sdan   u8 *aOut = *paOut;              /* Output cursor */
699798693b2Sdan   int iCol;                       /* Used to iterate from 0 to nCol */
7005d607a6eSdan 
701798693b2Sdan   for(iCol=0; iCol<nCol; iCol++){
7025d607a6eSdan     int n1 = sessionSerialLen(a1);
7035d607a6eSdan     int n2 = sessionSerialLen(a2);
7045d607a6eSdan     if( *a2 ){
7055d607a6eSdan       memcpy(aOut, a2, n2);
7065d607a6eSdan       aOut += n2;
7075d607a6eSdan     }else{
7085d607a6eSdan       memcpy(aOut, a1, n1);
7095d607a6eSdan       aOut += n1;
7105d607a6eSdan     }
7115d607a6eSdan     a1 += n1;
7125d607a6eSdan     a2 += n2;
7135d607a6eSdan   }
7145d607a6eSdan 
7155d607a6eSdan   *paOut = aOut;
7165d607a6eSdan }
7175d607a6eSdan 
718798693b2Sdan /*
719798693b2Sdan ** This is a helper function used by sessionMergeUpdate().
720798693b2Sdan **
721798693b2Sdan ** When this function is called, both *paOne and *paTwo point to a value
722798693b2Sdan ** within a change record. Before it returns, both have been advanced so
723798693b2Sdan ** as to point to the next value in the record.
724798693b2Sdan **
725798693b2Sdan ** If, when this function is called, *paTwo points to a valid value (i.e.
726fa29ecc4Sdan ** *paTwo[0] is not 0x00 - the "no value" placeholder), a copy of the *paTwo
727798693b2Sdan ** pointer is returned and *pnVal is set to the number of bytes in the
728798693b2Sdan ** serialized value. Otherwise, a copy of *paOne is returned and *pnVal
729798693b2Sdan ** set to the number of bytes in the value at *paOne. If *paOne points
730fa29ecc4Sdan ** to the "no value" placeholder, *pnVal is set to 1. In other words:
731fa29ecc4Sdan **
732fa29ecc4Sdan **   if( *paTwo is valid ) return *paTwo;
733fa29ecc4Sdan **   return *paOne;
734fa29ecc4Sdan **
735798693b2Sdan */
sessionMergeValue(u8 ** paOne,u8 ** paTwo,int * pnVal)7365d607a6eSdan static u8 *sessionMergeValue(
737798693b2Sdan   u8 **paOne,                     /* IN/OUT: Left-hand buffer pointer */
738798693b2Sdan   u8 **paTwo,                     /* IN/OUT: Right-hand buffer pointer */
739798693b2Sdan   int *pnVal                      /* OUT: Bytes in returned value */
7405d607a6eSdan ){
7415d607a6eSdan   u8 *a1 = *paOne;
7425d607a6eSdan   u8 *a2 = *paTwo;
7435d607a6eSdan   u8 *pRet = 0;
7445d607a6eSdan   int n1;
7455d607a6eSdan 
7465d607a6eSdan   assert( a1 );
7475d607a6eSdan   if( a2 ){
7485d607a6eSdan     int n2 = sessionSerialLen(a2);
7495d607a6eSdan     if( *a2 ){
7505d607a6eSdan       *pnVal = n2;
7515d607a6eSdan       pRet = a2;
7525d607a6eSdan     }
7535d607a6eSdan     *paTwo = &a2[n2];
7545d607a6eSdan   }
7555d607a6eSdan 
7565d607a6eSdan   n1 = sessionSerialLen(a1);
7575d607a6eSdan   if( pRet==0 ){
7585d607a6eSdan     *pnVal = n1;
7595d607a6eSdan     pRet = a1;
7605d607a6eSdan   }
7615d607a6eSdan   *paOne = &a1[n1];
7625d607a6eSdan 
7635d607a6eSdan   return pRet;
7645d607a6eSdan }
7655d607a6eSdan 
766798693b2Sdan /*
767798693b2Sdan ** This function is used by changeset_concat() to merge two UPDATE changes
768798693b2Sdan ** on the same row.
769798693b2Sdan */
sessionMergeUpdate(u8 ** paOut,SessionTable * pTab,int bPatchset,u8 * aOldRecord1,u8 * aOldRecord2,u8 * aNewRecord1,u8 * aNewRecord2)7705d607a6eSdan static int sessionMergeUpdate(
771798693b2Sdan   u8 **paOut,                     /* IN/OUT: Pointer to output buffer */
772798693b2Sdan   SessionTable *pTab,             /* Table change pertains to */
773a71d2371Sdan   int bPatchset,                  /* True if records are patchset records */
774798693b2Sdan   u8 *aOldRecord1,                /* old.* record for first change */
775798693b2Sdan   u8 *aOldRecord2,                /* old.* record for second change */
776798693b2Sdan   u8 *aNewRecord1,                /* new.* record for first change */
777798693b2Sdan   u8 *aNewRecord2                 /* new.* record for second change */
7785d607a6eSdan ){
7795d607a6eSdan   u8 *aOld1 = aOldRecord1;
7805d607a6eSdan   u8 *aOld2 = aOldRecord2;
7815d607a6eSdan   u8 *aNew1 = aNewRecord1;
7825d607a6eSdan   u8 *aNew2 = aNewRecord2;
7835d607a6eSdan 
7845d607a6eSdan   u8 *aOut = *paOut;
7855d607a6eSdan   int i;
78664277f4aSdan 
78764277f4aSdan   if( bPatchset==0 ){
7885d607a6eSdan     int bRequired = 0;
7895d607a6eSdan 
7905d607a6eSdan     assert( aOldRecord1 && aNewRecord1 );
7915d607a6eSdan 
7925d607a6eSdan     /* Write the old.* vector first. */
7935d607a6eSdan     for(i=0; i<pTab->nCol; i++){
7945d607a6eSdan       int nOld;
7955d607a6eSdan       u8 *aOld;
7965d607a6eSdan       int nNew;
7975d607a6eSdan       u8 *aNew;
7985d607a6eSdan 
7995d607a6eSdan       aOld = sessionMergeValue(&aOld1, &aOld2, &nOld);
8005d607a6eSdan       aNew = sessionMergeValue(&aNew1, &aNew2, &nNew);
8015d607a6eSdan       if( pTab->abPK[i] || nOld!=nNew || memcmp(aOld, aNew, nNew) ){
8025d607a6eSdan         if( pTab->abPK[i]==0 ) bRequired = 1;
8035d607a6eSdan         memcpy(aOut, aOld, nOld);
8045d607a6eSdan         aOut += nOld;
8055d607a6eSdan       }else{
8065d607a6eSdan         *(aOut++) = '\0';
8075d607a6eSdan       }
8085d607a6eSdan     }
8095d607a6eSdan 
8105d607a6eSdan     if( !bRequired ) return 0;
81164277f4aSdan   }
8125d607a6eSdan 
8135d607a6eSdan   /* Write the new.* vector */
8145d607a6eSdan   aOld1 = aOldRecord1;
8155d607a6eSdan   aOld2 = aOldRecord2;
8165d607a6eSdan   aNew1 = aNewRecord1;
8175d607a6eSdan   aNew2 = aNewRecord2;
8185d607a6eSdan   for(i=0; i<pTab->nCol; i++){
8195d607a6eSdan     int nOld;
8205d607a6eSdan     u8 *aOld;
8215d607a6eSdan     int nNew;
8225d607a6eSdan     u8 *aNew;
8235d607a6eSdan 
8245d607a6eSdan     aOld = sessionMergeValue(&aOld1, &aOld2, &nOld);
8255d607a6eSdan     aNew = sessionMergeValue(&aNew1, &aNew2, &nNew);
82664277f4aSdan     if( bPatchset==0
82764277f4aSdan      && (pTab->abPK[i] || (nOld==nNew && 0==memcmp(aOld, aNew, nNew)))
82864277f4aSdan     ){
8295d607a6eSdan       *(aOut++) = '\0';
8305d607a6eSdan     }else{
8315d607a6eSdan       memcpy(aOut, aNew, nNew);
8325d607a6eSdan       aOut += nNew;
8335d607a6eSdan     }
8345d607a6eSdan   }
8355d607a6eSdan 
8365d607a6eSdan   *paOut = aOut;
8375d607a6eSdan   return 1;
8385d607a6eSdan }
8395d607a6eSdan 
84077fc1d5bSdan /*
84177fc1d5bSdan ** This function is only called from within a pre-update-hook callback.
84277fc1d5bSdan ** It determines if the current pre-update-hook change affects the same row
84377fc1d5bSdan ** as the change stored in argument pChange. If so, it returns true. Otherwise
84477fc1d5bSdan ** if the pre-update-hook does not affect the same row as pChange, it returns
84577fc1d5bSdan ** false.
84677fc1d5bSdan */
sessionPreupdateEqual(sqlite3_session * pSession,SessionTable * pTab,SessionChange * pChange,int op)84777fc1d5bSdan static int sessionPreupdateEqual(
848cf8e9144Sdan   sqlite3_session *pSession,      /* Session object that owns SessionTable */
84977fc1d5bSdan   SessionTable *pTab,             /* Table associated with change */
85077fc1d5bSdan   SessionChange *pChange,         /* Change to compare to */
85177fc1d5bSdan   int op                          /* Current pre-update operation */
852e8d5648eSdan ){
85377fc1d5bSdan   int iCol;                       /* Used to iterate through columns */
85477fc1d5bSdan   u8 *a = pChange->aRecord;       /* Cursor used to scan change record */
855e8d5648eSdan 
85677fc1d5bSdan   assert( op==SQLITE_INSERT || op==SQLITE_UPDATE || op==SQLITE_DELETE );
85777fc1d5bSdan   for(iCol=0; iCol<pTab->nCol; iCol++){
85877fc1d5bSdan     if( !pTab->abPK[iCol] ){
859798693b2Sdan       a += sessionSerialLen(a);
860e8d5648eSdan     }else{
8616734007dSdan       sqlite3_value *pVal;        /* Value returned by preupdate_new/old */
8626734007dSdan       int rc;                     /* Error code from preupdate_new/old */
863798693b2Sdan       int eType = *a++;           /* Type of value from change record */
8646734007dSdan 
8656734007dSdan       /* The following calls to preupdate_new() and preupdate_old() can not
8666734007dSdan       ** fail. This is because they cache their return values, and by the
8676734007dSdan       ** time control flows to here they have already been called once from
8686734007dSdan       ** within sessionPreupdateHash(). The first two asserts below verify
8696734007dSdan       ** this (that the method has already been called). */
87077fc1d5bSdan       if( op==SQLITE_INSERT ){
871cf8e9144Sdan         /* assert( db->pPreUpdate->pNewUnpacked || db->pPreUpdate->aNew ); */
872cf8e9144Sdan         rc = pSession->hook.xNew(pSession->hook.pCtx, iCol, &pVal);
873e8d5648eSdan       }else{
874cf8e9144Sdan         /* assert( db->pPreUpdate->pUnpacked ); */
875cf8e9144Sdan         rc = pSession->hook.xOld(pSession->hook.pCtx, iCol, &pVal);
876e8d5648eSdan       }
8776734007dSdan       assert( rc==SQLITE_OK );
8781611e5a3Sdan       if( sqlite3_value_type(pVal)!=eType ) return 0;
879e8d5648eSdan 
88012ca0b56Sdan       /* A SessionChange object never has a NULL value in a PK column */
88112ca0b56Sdan       assert( eType==SQLITE_INTEGER || eType==SQLITE_FLOAT
88212ca0b56Sdan            || eType==SQLITE_BLOB    || eType==SQLITE_TEXT
88312ca0b56Sdan       );
88412ca0b56Sdan 
88512ca0b56Sdan       if( eType==SQLITE_INTEGER || eType==SQLITE_FLOAT ){
886e8d5648eSdan         i64 iVal = sessionGetI64(a);
887e8d5648eSdan         a += 8;
888e8d5648eSdan         if( eType==SQLITE_INTEGER ){
88977fc1d5bSdan           if( sqlite3_value_int64(pVal)!=iVal ) return 0;
890e8d5648eSdan         }else{
891e8d5648eSdan           double rVal;
892e8d5648eSdan           assert( sizeof(iVal)==8 && sizeof(rVal)==8 );
893e8d5648eSdan           memcpy(&rVal, &iVal, 8);
89477fc1d5bSdan           if( sqlite3_value_double(pVal)!=rVal ) return 0;
895e8d5648eSdan         }
89612ca0b56Sdan       }else{
897e8d5648eSdan         int n;
898e8d5648eSdan         const u8 *z;
899e8d5648eSdan         a += sessionVarintGet(a, &n);
90077fc1d5bSdan         if( sqlite3_value_bytes(pVal)!=n ) return 0;
90112ca0b56Sdan         if( eType==SQLITE_TEXT ){
90212ca0b56Sdan           z = sqlite3_value_text(pVal);
90312ca0b56Sdan         }else{
90412ca0b56Sdan           z = sqlite3_value_blob(pVal);
90512ca0b56Sdan         }
9063f2cebb6Sdan         if( n>0 && memcmp(a, z, n) ) return 0;
907e8d5648eSdan         a += n;
908e8d5648eSdan       }
909e8d5648eSdan     }
910e8d5648eSdan   }
911e8d5648eSdan 
91277fc1d5bSdan   return 1;
9134fccf43aSdan }
9144fccf43aSdan 
9154fccf43aSdan /*
9164fccf43aSdan ** If required, grow the hash table used to store changes on table pTab
9174fccf43aSdan ** (part of the session pSession). If a fatal OOM error occurs, set the
9184fccf43aSdan ** session object to failed and return SQLITE_ERROR. Otherwise, return
9194fccf43aSdan ** SQLITE_OK.
9204fccf43aSdan **
9214fccf43aSdan ** It is possible that a non-fatal OOM error occurs in this function. In
9224fccf43aSdan ** that case the hash-table does not grow, but SQLITE_OK is returned anyway.
9234fccf43aSdan ** Growing the hash table in this case is a performance optimization only,
9244fccf43aSdan ** it is not required for correct operation.
9254fccf43aSdan */
sessionGrowHash(sqlite3_session * pSession,int bPatchset,SessionTable * pTab)9260cb735b9Sdan static int sessionGrowHash(
9270cb735b9Sdan   sqlite3_session *pSession,      /* For memory accounting. May be NULL */
9280cb735b9Sdan   int bPatchset,
9290cb735b9Sdan   SessionTable *pTab
9300cb735b9Sdan ){
9314fccf43aSdan   if( pTab->nChange==0 || pTab->nEntry>=(pTab->nChange/2) ){
9324fccf43aSdan     int i;
9334fccf43aSdan     SessionChange **apNew;
934f6ad201aSdrh     sqlite3_int64 nNew = 2*(sqlite3_int64)(pTab->nChange ? pTab->nChange : 128);
9354fccf43aSdan 
9360cb735b9Sdan     apNew = (SessionChange**)sessionMalloc64(
9370cb735b9Sdan         pSession, sizeof(SessionChange*) * nNew
9380cb735b9Sdan     );
9394fccf43aSdan     if( apNew==0 ){
9404fccf43aSdan       if( pTab->nChange==0 ){
9414fccf43aSdan         return SQLITE_ERROR;
9424fccf43aSdan       }
9434fccf43aSdan       return SQLITE_OK;
9444fccf43aSdan     }
9454fccf43aSdan     memset(apNew, 0, sizeof(SessionChange *) * nNew);
9464fccf43aSdan 
9474fccf43aSdan     for(i=0; i<pTab->nChange; i++){
9484fccf43aSdan       SessionChange *p;
9494fccf43aSdan       SessionChange *pNext;
9504fccf43aSdan       for(p=pTab->apChange[i]; p; p=pNext){
95164277f4aSdan         int bPkOnly = (p->op==SQLITE_DELETE && bPatchset);
95264277f4aSdan         int iHash = sessionChangeHash(pTab, bPkOnly, p->aRecord, nNew);
9534fccf43aSdan         pNext = p->pNext;
9544fccf43aSdan         p->pNext = apNew[iHash];
9554fccf43aSdan         apNew[iHash] = p;
9564fccf43aSdan       }
9574fccf43aSdan     }
9584fccf43aSdan 
9590cb735b9Sdan     sessionFree(pSession, pTab->apChange);
9604fccf43aSdan     pTab->nChange = nNew;
9614fccf43aSdan     pTab->apChange = apNew;
9624fccf43aSdan   }
9634fccf43aSdan 
9644fccf43aSdan   return SQLITE_OK;
9654fccf43aSdan }
9664fccf43aSdan 
967296c7658Sdan /*
968e8d5648eSdan ** This function queries the database for the names of the columns of table
9693f975373Sdrh ** zThis, in schema zDb.
970e8d5648eSdan **
97177fc1d5bSdan ** Otherwise, if they are not NULL, variable *pnCol is set to the number
97277fc1d5bSdan ** of columns in the database table and variable *pzTab is set to point to a
973e8d5648eSdan ** nul-terminated copy of the table name. *pazCol (if not NULL) is set to
974e8d5648eSdan ** point to an array of pointers to column names. And *pabPK (again, if not
975e8d5648eSdan ** NULL) is set to point to an array of booleans - true if the corresponding
976e8d5648eSdan ** column is part of the primary key.
977e8d5648eSdan **
978e8d5648eSdan ** For example, if the table is declared as:
979e8d5648eSdan **
980e8d5648eSdan **     CREATE TABLE tbl1(w, x, y, z, PRIMARY KEY(w, z));
981e8d5648eSdan **
98277fc1d5bSdan ** Then the four output variables are populated as follows:
983e8d5648eSdan **
98477fc1d5bSdan **     *pnCol  = 4
985e8d5648eSdan **     *pzTab  = "tbl1"
986e8d5648eSdan **     *pazCol = {"w", "x", "y", "z"}
987e8d5648eSdan **     *pabPK  = {1, 0, 0, 1}
988e8d5648eSdan **
989e8d5648eSdan ** All returned buffers are part of the same single allocation, which must
9903f975373Sdrh ** be freed using sqlite3_free() by the caller
991e8d5648eSdan */
sessionTableInfo(sqlite3_session * pSession,sqlite3 * db,const char * zDb,const char * zThis,int * pnCol,const char ** pzTab,const char *** pazCol,u8 ** pabPK)992e8d5648eSdan static int sessionTableInfo(
9930cb735b9Sdan   sqlite3_session *pSession,      /* For memory accounting. May be NULL */
994e8d5648eSdan   sqlite3 *db,                    /* Database connection */
995e8d5648eSdan   const char *zDb,                /* Name of attached database (e.g. "main") */
996e8d5648eSdan   const char *zThis,              /* Table name */
997ca62ad57Sdan   int *pnCol,                     /* OUT: number of columns */
998e8d5648eSdan   const char **pzTab,             /* OUT: Copy of zThis */
999e8d5648eSdan   const char ***pazCol,           /* OUT: Array of column names for table */
1000e8d5648eSdan   u8 **pabPK                      /* OUT: Array of booleans - true for PK col */
1001e8d5648eSdan ){
1002e8d5648eSdan   char *zPragma;
1003e8d5648eSdan   sqlite3_stmt *pStmt;
1004e8d5648eSdan   int rc;
10052d77d80aSdrh   sqlite3_int64 nByte;
1006e8d5648eSdan   int nDbCol = 0;
1007e8d5648eSdan   int nThis;
1008e8d5648eSdan   int i;
100974f598b6Smistachkin   u8 *pAlloc = 0;
1010db04571cSdan   char **azCol = 0;
101174f598b6Smistachkin   u8 *abPK = 0;
1012e8d5648eSdan 
1013db04571cSdan   assert( pazCol && pabPK );
1014e8d5648eSdan 
1015cfdbde21Sdrh   nThis = sqlite3Strlen30(zThis);
1016614efe2bSdan   if( nThis==12 && 0==sqlite3_stricmp("sqlite_stat1", zThis) ){
101758713184Sdan     rc = sqlite3_table_column_metadata(db, zDb, zThis, 0, 0, 0, 0, 0, 0);
101858713184Sdan     if( rc==SQLITE_OK ){
1019614efe2bSdan       /* For sqlite_stat1, pretend that (tbl,idx) is the PRIMARY KEY. */
1020614efe2bSdan       zPragma = sqlite3_mprintf(
1021614efe2bSdan           "SELECT 0, 'tbl',  '', 0, '', 1     UNION ALL "
1022614efe2bSdan           "SELECT 1, 'idx',  '', 0, '', 2     UNION ALL "
1023614efe2bSdan           "SELECT 2, 'stat', '', 0, '', 0"
1024614efe2bSdan       );
102558713184Sdan     }else if( rc==SQLITE_ERROR ){
102658713184Sdan       zPragma = sqlite3_mprintf("");
102758713184Sdan     }else{
1028*7a3b4451Sdrh       *pazCol = 0;
1029*7a3b4451Sdrh       *pabPK = 0;
1030*7a3b4451Sdrh       *pnCol = 0;
1031*7a3b4451Sdrh       if( pzTab ) *pzTab = 0;
103258713184Sdan       return rc;
103358713184Sdan     }
1034614efe2bSdan   }else{
1035e8d5648eSdan     zPragma = sqlite3_mprintf("PRAGMA '%q'.table_info('%q')", zDb, zThis);
1036614efe2bSdan   }
1037*7a3b4451Sdrh   if( !zPragma ){
1038*7a3b4451Sdrh     *pazCol = 0;
1039*7a3b4451Sdrh     *pabPK = 0;
1040*7a3b4451Sdrh     *pnCol = 0;
1041*7a3b4451Sdrh     if( pzTab ) *pzTab = 0;
1042*7a3b4451Sdrh     return SQLITE_NOMEM;
1043*7a3b4451Sdrh   }
1044e8d5648eSdan 
1045e8d5648eSdan   rc = sqlite3_prepare_v2(db, zPragma, -1, &pStmt, 0);
1046e8d5648eSdan   sqlite3_free(zPragma);
1047*7a3b4451Sdrh   if( rc!=SQLITE_OK ){
1048*7a3b4451Sdrh     *pazCol = 0;
1049*7a3b4451Sdrh     *pabPK = 0;
1050*7a3b4451Sdrh     *pnCol = 0;
1051*7a3b4451Sdrh     if( pzTab ) *pzTab = 0;
1052*7a3b4451Sdrh     return rc;
1053*7a3b4451Sdrh   }
1054e8d5648eSdan 
1055e8d5648eSdan   nByte = nThis + 1;
1056e8d5648eSdan   while( SQLITE_ROW==sqlite3_step(pStmt) ){
1057e8d5648eSdan     nByte += sqlite3_column_bytes(pStmt, 1);
1058e8d5648eSdan     nDbCol++;
1059e8d5648eSdan   }
1060e8d5648eSdan   rc = sqlite3_reset(pStmt);
1061e8d5648eSdan 
1062e8d5648eSdan   if( rc==SQLITE_OK ){
1063e8d5648eSdan     nByte += nDbCol * (sizeof(const char *) + sizeof(u8) + 1);
10640cb735b9Sdan     pAlloc = sessionMalloc64(pSession, nByte);
1065e8d5648eSdan     if( pAlloc==0 ){
1066e8d5648eSdan       rc = SQLITE_NOMEM;
1067e8d5648eSdan     }
1068e8d5648eSdan   }
1069e8d5648eSdan   if( rc==SQLITE_OK ){
1070e8d5648eSdan     azCol = (char **)pAlloc;
1071ca62ad57Sdan     pAlloc = (u8 *)&azCol[nDbCol];
1072e8d5648eSdan     abPK = (u8 *)pAlloc;
1073ca62ad57Sdan     pAlloc = &abPK[nDbCol];
1074e8d5648eSdan     if( pzTab ){
1075e8d5648eSdan       memcpy(pAlloc, zThis, nThis+1);
1076e8d5648eSdan       *pzTab = (char *)pAlloc;
1077e8d5648eSdan       pAlloc += nThis+1;
1078e8d5648eSdan     }
1079e8d5648eSdan 
1080e8d5648eSdan     i = 0;
1081e8d5648eSdan     while( SQLITE_ROW==sqlite3_step(pStmt) ){
1082e8d5648eSdan       int nName = sqlite3_column_bytes(pStmt, 1);
1083e8d5648eSdan       const unsigned char *zName = sqlite3_column_text(pStmt, 1);
1084e8d5648eSdan       if( zName==0 ) break;
1085e8d5648eSdan       memcpy(pAlloc, zName, nName+1);
1086e8d5648eSdan       azCol[i] = (char *)pAlloc;
1087e8d5648eSdan       pAlloc += nName+1;
1088db04571cSdan       abPK[i] = sqlite3_column_int(pStmt, 5);
1089e8d5648eSdan       i++;
1090e8d5648eSdan     }
1091e8d5648eSdan     rc = sqlite3_reset(pStmt);
1092e8d5648eSdan 
1093e8d5648eSdan   }
1094e8d5648eSdan 
1095e8d5648eSdan   /* If successful, populate the output variables. Otherwise, zero them and
1096e8d5648eSdan   ** free any allocation made. An error code will be returned in this case.
1097e8d5648eSdan   */
1098e8d5648eSdan   if( rc==SQLITE_OK ){
1099db04571cSdan     *pazCol = (const char **)azCol;
1100db04571cSdan     *pabPK = abPK;
1101ca62ad57Sdan     *pnCol = nDbCol;
1102e8d5648eSdan   }else{
1103db04571cSdan     *pazCol = 0;
1104db04571cSdan     *pabPK = 0;
1105ca62ad57Sdan     *pnCol = 0;
1106e8d5648eSdan     if( pzTab ) *pzTab = 0;
11070cb735b9Sdan     sessionFree(pSession, azCol);
1108e8d5648eSdan   }
1109e8d5648eSdan   sqlite3_finalize(pStmt);
1110e8d5648eSdan   return rc;
1111e8d5648eSdan }
1112e8d5648eSdan 
1113e8d5648eSdan /*
1114296c7658Sdan ** This function is only called from within a pre-update handler for a
1115296c7658Sdan ** write to table pTab, part of session pSession. If this is the first
11166dc29e60Sdan ** write to this table, initalize the SessionTable.nCol, azCol[] and
11176dc29e60Sdan ** abPK[] arrays accordingly.
1118296c7658Sdan **
11196dc29e60Sdan ** If an error occurs, an error code is stored in sqlite3_session.rc and
11206dc29e60Sdan ** non-zero returned. Or, if no error occurs but the table has no primary
11216dc29e60Sdan ** key, sqlite3_session.rc is left set to SQLITE_OK and non-zero returned to
11226dc29e60Sdan ** indicate that updates on this table should be ignored. SessionTable.abPK
11236dc29e60Sdan ** is set to NULL in this case.
1124296c7658Sdan */
sessionInitTable(sqlite3_session * pSession,SessionTable * pTab)11254fccf43aSdan static int sessionInitTable(sqlite3_session *pSession, SessionTable *pTab){
11264fccf43aSdan   if( pTab->nCol==0 ){
11276dc29e60Sdan     u8 *abPK;
1128e8d5648eSdan     assert( pTab->azCol==0 || pTab->abPK==0 );
11290cb735b9Sdan     pSession->rc = sessionTableInfo(pSession, pSession->db, pSession->zDb,
11306dc29e60Sdan         pTab->zName, &pTab->nCol, 0, &pTab->azCol, &abPK
1131e8d5648eSdan     );
11326dc29e60Sdan     if( pSession->rc==SQLITE_OK ){
11336dc29e60Sdan       int i;
11346dc29e60Sdan       for(i=0; i<pTab->nCol; i++){
11356dc29e60Sdan         if( abPK[i] ){
11366dc29e60Sdan           pTab->abPK = abPK;
11376dc29e60Sdan           break;
1138ca62ad57Sdan         }
11394fccf43aSdan       }
11403739f298Sdan       if( 0==sqlite3_stricmp("sqlite_stat1", pTab->zName) ){
11413739f298Sdan         pTab->bStat1 = 1;
11423739f298Sdan       }
1143a23a873fSdan 
11446d29a4feSdan       if( pSession->bEnableSize ){
1145a23a873fSdan         pSession->nMaxChangesetSize += (
1146a23a873fSdan           1 + sessionVarintLen(pTab->nCol) + pTab->nCol + strlen(pTab->zName)+1
1147a23a873fSdan         );
11486dc29e60Sdan       }
11496dc29e60Sdan     }
11506d29a4feSdan   }
11516dc29e60Sdan   return (pSession->rc || pTab->abPK==0);
1152e8d5648eSdan }
1153e8d5648eSdan 
115477fc1d5bSdan /*
11551611e5a3Sdan ** Versions of the four methods in object SessionHook for use with the
11561611e5a3Sdan ** sqlite_stat1 table. The purpose of this is to substitute a zero-length
11571611e5a3Sdan ** blob each time a NULL value is read from the "idx" column of the
11581611e5a3Sdan ** sqlite_stat1 table.
11591611e5a3Sdan */
11601611e5a3Sdan typedef struct SessionStat1Ctx SessionStat1Ctx;
11611611e5a3Sdan struct SessionStat1Ctx {
11621611e5a3Sdan   SessionHook hook;
11631611e5a3Sdan   sqlite3_session *pSession;
11641611e5a3Sdan };
sessionStat1Old(void * pCtx,int iCol,sqlite3_value ** ppVal)11651611e5a3Sdan static int sessionStat1Old(void *pCtx, int iCol, sqlite3_value **ppVal){
11661611e5a3Sdan   SessionStat1Ctx *p = (SessionStat1Ctx*)pCtx;
11671611e5a3Sdan   sqlite3_value *pVal = 0;
11681611e5a3Sdan   int rc = p->hook.xOld(p->hook.pCtx, iCol, &pVal);
11691611e5a3Sdan   if( rc==SQLITE_OK && iCol==1 && sqlite3_value_type(pVal)==SQLITE_NULL ){
11701611e5a3Sdan     pVal = p->pSession->pZeroBlob;
11711611e5a3Sdan   }
11721611e5a3Sdan   *ppVal = pVal;
11731611e5a3Sdan   return rc;
11741611e5a3Sdan }
sessionStat1New(void * pCtx,int iCol,sqlite3_value ** ppVal)11751611e5a3Sdan static int sessionStat1New(void *pCtx, int iCol, sqlite3_value **ppVal){
11761611e5a3Sdan   SessionStat1Ctx *p = (SessionStat1Ctx*)pCtx;
11771611e5a3Sdan   sqlite3_value *pVal = 0;
11781611e5a3Sdan   int rc = p->hook.xNew(p->hook.pCtx, iCol, &pVal);
11791611e5a3Sdan   if( rc==SQLITE_OK && iCol==1 && sqlite3_value_type(pVal)==SQLITE_NULL ){
11801611e5a3Sdan     pVal = p->pSession->pZeroBlob;
11811611e5a3Sdan   }
11821611e5a3Sdan   *ppVal = pVal;
11831611e5a3Sdan   return rc;
11841611e5a3Sdan }
sessionStat1Count(void * pCtx)11851611e5a3Sdan static int sessionStat1Count(void *pCtx){
11861611e5a3Sdan   SessionStat1Ctx *p = (SessionStat1Ctx*)pCtx;
11871611e5a3Sdan   return p->hook.xCount(p->hook.pCtx);
11881611e5a3Sdan }
sessionStat1Depth(void * pCtx)11891611e5a3Sdan static int sessionStat1Depth(void *pCtx){
11901611e5a3Sdan   SessionStat1Ctx *p = (SessionStat1Ctx*)pCtx;
11911611e5a3Sdan   return p->hook.xDepth(p->hook.pCtx);
11921611e5a3Sdan }
11931611e5a3Sdan 
sessionUpdateMaxSize(int op,sqlite3_session * pSession,SessionTable * pTab,SessionChange * pC)1194a23a873fSdan static int sessionUpdateMaxSize(
1195a23a873fSdan   int op,
1196a23a873fSdan   sqlite3_session *pSession,      /* Session object pTab is attached to */
1197a23a873fSdan   SessionTable *pTab,             /* Table that change applies to */
1198a23a873fSdan   SessionChange *pC               /* Update pC->nMaxSize */
1199a23a873fSdan ){
1200a23a873fSdan   i64 nNew = 2;
1201a23a873fSdan   if( pC->op==SQLITE_INSERT ){
1202a23a873fSdan     if( op!=SQLITE_DELETE ){
1203a23a873fSdan       int ii;
1204a23a873fSdan       for(ii=0; ii<pTab->nCol; ii++){
1205a23a873fSdan         sqlite3_value *p = 0;
1206a23a873fSdan         pSession->hook.xNew(pSession->hook.pCtx, ii, &p);
1207a23a873fSdan         sessionSerializeValue(0, p, &nNew);
1208a23a873fSdan       }
1209a23a873fSdan     }
1210a23a873fSdan   }else if( op==SQLITE_DELETE ){
1211a23a873fSdan     nNew += pC->nRecord;
1212a23a873fSdan     if( sqlite3_preupdate_blobwrite(pSession->db)>=0 ){
1213a23a873fSdan       nNew += pC->nRecord;
1214a23a873fSdan     }
1215a23a873fSdan   }else{
1216a23a873fSdan     int ii;
1217a23a873fSdan     u8 *pCsr = pC->aRecord;
1218a23a873fSdan     for(ii=0; ii<pTab->nCol; ii++){
1219a23a873fSdan       int bChanged = 1;
1220a23a873fSdan       int nOld = 0;
1221a23a873fSdan       int eType;
1222a23a873fSdan       sqlite3_value *p = 0;
1223a23a873fSdan       pSession->hook.xNew(pSession->hook.pCtx, ii, &p);
1224a23a873fSdan       if( p==0 ){
1225a23a873fSdan         return SQLITE_NOMEM;
1226a23a873fSdan       }
1227a23a873fSdan 
1228a23a873fSdan       eType = *pCsr++;
1229a23a873fSdan       switch( eType ){
1230a23a873fSdan         case SQLITE_NULL:
1231a23a873fSdan           bChanged = sqlite3_value_type(p)!=SQLITE_NULL;
1232a23a873fSdan           break;
1233a23a873fSdan 
1234a23a873fSdan         case SQLITE_FLOAT:
1235a23a873fSdan         case SQLITE_INTEGER: {
1236a23a873fSdan           if( eType==sqlite3_value_type(p) ){
1237a23a873fSdan             sqlite3_int64 iVal = sessionGetI64(pCsr);
1238a23a873fSdan             if( eType==SQLITE_INTEGER ){
1239a23a873fSdan               bChanged = (iVal!=sqlite3_value_int64(p));
1240a23a873fSdan             }else{
1241a23a873fSdan               double dVal;
1242a23a873fSdan               memcpy(&dVal, &iVal, 8);
1243a23a873fSdan               bChanged = (dVal!=sqlite3_value_double(p));
1244a23a873fSdan             }
1245a23a873fSdan           }
1246a23a873fSdan           nOld = 8;
1247a23a873fSdan           pCsr += 8;
1248a23a873fSdan           break;
1249a23a873fSdan         }
1250a23a873fSdan 
1251a23a873fSdan         default: {
1252a23a873fSdan           int nByte;
1253a23a873fSdan           nOld = sessionVarintGet(pCsr, &nByte);
1254a23a873fSdan           pCsr += nOld;
1255a23a873fSdan           nOld += nByte;
1256a23a873fSdan           assert( eType==SQLITE_TEXT || eType==SQLITE_BLOB );
1257a23a873fSdan           if( eType==sqlite3_value_type(p)
1258a23a873fSdan            && nByte==sqlite3_value_bytes(p)
1259a23a873fSdan            && (nByte==0 || 0==memcmp(pCsr, sqlite3_value_blob(p), nByte))
1260a23a873fSdan           ){
1261a23a873fSdan             bChanged = 0;
1262a23a873fSdan           }
1263a23a873fSdan           pCsr += nByte;
1264a23a873fSdan           break;
1265a23a873fSdan         }
1266a23a873fSdan       }
1267a23a873fSdan 
1268a23a873fSdan       if( bChanged && pTab->abPK[ii] ){
1269a23a873fSdan         nNew = pC->nRecord + 2;
1270a23a873fSdan         break;
1271a23a873fSdan       }
1272a23a873fSdan 
1273a23a873fSdan       if( bChanged ){
1274a23a873fSdan         nNew += 1 + nOld;
1275a23a873fSdan         sessionSerializeValue(0, p, &nNew);
1276a23a873fSdan       }else if( pTab->abPK[ii] ){
1277a23a873fSdan         nNew += 2 + nOld;
1278a23a873fSdan       }else{
1279a23a873fSdan         nNew += 2;
1280a23a873fSdan       }
1281a23a873fSdan     }
1282a23a873fSdan   }
1283a23a873fSdan 
1284a23a873fSdan   if( nNew>pC->nMaxSize ){
1285a23a873fSdan     int nIncr = nNew - pC->nMaxSize;
1286a23a873fSdan     pC->nMaxSize = nNew;
1287a23a873fSdan     pSession->nMaxChangesetSize += nIncr;
1288a23a873fSdan   }
1289a23a873fSdan   return SQLITE_OK;
1290a23a873fSdan }
12911611e5a3Sdan 
12921611e5a3Sdan /*
129377fc1d5bSdan ** This function is only called from with a pre-update-hook reporting a
129477fc1d5bSdan ** change on table pTab (attached to session pSession). The type of change
129577fc1d5bSdan ** (UPDATE, INSERT, DELETE) is specified by the first argument.
129677fc1d5bSdan **
129777fc1d5bSdan ** Unless one is already present or an error occurs, an entry is added
129877fc1d5bSdan ** to the changed-rows hash table associated with table pTab.
129977fc1d5bSdan */
sessionPreupdateOneChange(int op,sqlite3_session * pSession,SessionTable * pTab)1300e8d5648eSdan static void sessionPreupdateOneChange(
130177fc1d5bSdan   int op,                         /* One of SQLITE_UPDATE, INSERT, DELETE */
130277fc1d5bSdan   sqlite3_session *pSession,      /* Session object pTab is attached to */
130377fc1d5bSdan   SessionTable *pTab              /* Table that change applies to */
1304e8d5648eSdan ){
1305e8d5648eSdan   int iHash;
1306cf8e9144Sdan   int bNull = 0;
1307e8d5648eSdan   int rc = SQLITE_OK;
1308fa94d492Sdrh   SessionStat1Ctx stat1 = {{0,0,0,0,0},0};
1309e8d5648eSdan 
1310e8d5648eSdan   if( pSession->rc ) return;
1311e8d5648eSdan 
1312e8d5648eSdan   /* Load table details if required */
1313e8d5648eSdan   if( sessionInitTable(pSession, pTab) ) return;
1314e8d5648eSdan 
13156dc29e60Sdan   /* Check the number of columns in this xPreUpdate call matches the
13166dc29e60Sdan   ** number of columns in the table.  */
13176dc29e60Sdan   if( pTab->nCol!=pSession->hook.xCount(pSession->hook.pCtx) ){
13186dc29e60Sdan     pSession->rc = SQLITE_SCHEMA;
13196dc29e60Sdan     return;
13206dc29e60Sdan   }
13216dc29e60Sdan 
1322e8d5648eSdan   /* Grow the hash table if required */
13230cb735b9Sdan   if( sessionGrowHash(pSession, 0, pTab) ){
13245d607a6eSdan     pSession->rc = SQLITE_NOMEM;
13255d607a6eSdan     return;
13265d607a6eSdan   }
1327e8d5648eSdan 
13281611e5a3Sdan   if( pTab->bStat1 ){
13291611e5a3Sdan     stat1.hook = pSession->hook;
13301611e5a3Sdan     stat1.pSession = pSession;
13311611e5a3Sdan     pSession->hook.pCtx = (void*)&stat1;
13321611e5a3Sdan     pSession->hook.xNew = sessionStat1New;
13331611e5a3Sdan     pSession->hook.xOld = sessionStat1Old;
13341611e5a3Sdan     pSession->hook.xCount = sessionStat1Count;
13351611e5a3Sdan     pSession->hook.xDepth = sessionStat1Depth;
13361611e5a3Sdan     if( pSession->pZeroBlob==0 ){
13371611e5a3Sdan       sqlite3_value *p = sqlite3ValueNew(0);
13381611e5a3Sdan       if( p==0 ){
13391611e5a3Sdan         rc = SQLITE_NOMEM;
13401611e5a3Sdan         goto error_out;
13411611e5a3Sdan       }
13421611e5a3Sdan       sqlite3ValueSetStr(p, 0, "", 0, SQLITE_STATIC);
13431611e5a3Sdan       pSession->pZeroBlob = p;
13441611e5a3Sdan     }
13451611e5a3Sdan   }
13461611e5a3Sdan 
134780fe2d93Sdan   /* Calculate the hash-key for this change. If the primary key of the row
134880fe2d93Sdan   ** includes a NULL value, exit early. Such changes are ignored by the
134980fe2d93Sdan   ** session module. */
1350cf8e9144Sdan   rc = sessionPreupdateHash(pSession, pTab, op==SQLITE_INSERT, &iHash, &bNull);
135180fe2d93Sdan   if( rc!=SQLITE_OK ) goto error_out;
135280fe2d93Sdan 
13531611e5a3Sdan   if( bNull==0 ){
135480fe2d93Sdan     /* Search the hash table for an existing record for this row. */
1355b4480e94Sdan     SessionChange *pC;
13566734007dSdan     for(pC=pTab->apChange[iHash]; pC; pC=pC->pNext){
1357cf8e9144Sdan       if( sessionPreupdateEqual(pSession, pTab, pC, op) ) break;
1358e8d5648eSdan     }
135980fe2d93Sdan 
1360e8d5648eSdan     if( pC==0 ){
1361e8d5648eSdan       /* Create a new change object containing all the old values (if
1362e8d5648eSdan       ** this is an SQLITE_UPDATE or SQLITE_DELETE), or just the PK
1363e8d5648eSdan       ** values (if this is an INSERT). */
13642d77d80aSdrh       sqlite3_int64 nByte;    /* Number of bytes to allocate */
1365e8d5648eSdan       int i;                  /* Used to iterate through columns */
1366e8d5648eSdan 
1367b4480e94Sdan       assert( rc==SQLITE_OK );
1368e8d5648eSdan       pTab->nEntry++;
1369e8d5648eSdan 
1370e8d5648eSdan       /* Figure out how large an allocation is required */
1371e8d5648eSdan       nByte = sizeof(SessionChange);
137280fe2d93Sdan       for(i=0; i<pTab->nCol; i++){
1373e8d5648eSdan         sqlite3_value *p = 0;
1374e8d5648eSdan         if( op!=SQLITE_INSERT ){
1375cf8e9144Sdan           TESTONLY(int trc = ) pSession->hook.xOld(pSession->hook.pCtx, i, &p);
137680fe2d93Sdan           assert( trc==SQLITE_OK );
137780fe2d93Sdan         }else if( pTab->abPK[i] ){
1378cf8e9144Sdan           TESTONLY(int trc = ) pSession->hook.xNew(pSession->hook.pCtx, i, &p);
137980fe2d93Sdan           assert( trc==SQLITE_OK );
1380e8d5648eSdan         }
138180fe2d93Sdan 
138280fe2d93Sdan         /* This may fail if SQLite value p contains a utf-16 string that must
138380fe2d93Sdan         ** be converted to utf-8 and an OOM error occurs while doing so. */
1384e8d5648eSdan         rc = sessionSerializeValue(0, p, &nByte);
138580fe2d93Sdan         if( rc!=SQLITE_OK ) goto error_out;
1386e8d5648eSdan       }
1387e8d5648eSdan 
1388e8d5648eSdan       /* Allocate the change object */
1389a23a873fSdan       pC = (SessionChange *)sessionMalloc64(pSession, nByte);
1390a23a873fSdan       if( !pC ){
1391e8d5648eSdan         rc = SQLITE_NOMEM;
139280fe2d93Sdan         goto error_out;
1393e8d5648eSdan       }else{
1394a23a873fSdan         memset(pC, 0, sizeof(SessionChange));
1395a23a873fSdan         pC->aRecord = (u8 *)&pC[1];
1396e8d5648eSdan       }
1397e8d5648eSdan 
139880fe2d93Sdan       /* Populate the change object. None of the preupdate_old(),
139980fe2d93Sdan       ** preupdate_new() or SerializeValue() calls below may fail as all
140080fe2d93Sdan       ** required values and encodings have already been cached in memory.
140180fe2d93Sdan       ** It is not possible for an OOM to occur in this block. */
1402e8d5648eSdan       nByte = 0;
140380fe2d93Sdan       for(i=0; i<pTab->nCol; i++){
1404e8d5648eSdan         sqlite3_value *p = 0;
1405e8d5648eSdan         if( op!=SQLITE_INSERT ){
1406cf8e9144Sdan           pSession->hook.xOld(pSession->hook.pCtx, i, &p);
140780fe2d93Sdan         }else if( pTab->abPK[i] ){
1408cf8e9144Sdan           pSession->hook.xNew(pSession->hook.pCtx, i, &p);
1409e8d5648eSdan         }
1410a23a873fSdan         sessionSerializeValue(&pC->aRecord[nByte], p, &nByte);
1411e8d5648eSdan       }
141280fe2d93Sdan 
141380fe2d93Sdan       /* Add the change to the hash-table */
1414cf8e9144Sdan       if( pSession->bIndirect || pSession->hook.xDepth(pSession->hook.pCtx) ){
1415a23a873fSdan         pC->bIndirect = 1;
1416b4480e94Sdan       }
1417a23a873fSdan       pC->nRecord = nByte;
1418a23a873fSdan       pC->op = op;
1419a23a873fSdan       pC->pNext = pTab->apChange[iHash];
1420a23a873fSdan       pTab->apChange[iHash] = pC;
142180fe2d93Sdan 
142280fe2d93Sdan     }else if( pC->bIndirect ){
1423b4480e94Sdan       /* If the existing change is considered "indirect", but this current
1424b4480e94Sdan       ** change is "direct", mark the change object as direct. */
1425cf8e9144Sdan       if( pSession->hook.xDepth(pSession->hook.pCtx)==0
1426cf8e9144Sdan        && pSession->bIndirect==0
1427cf8e9144Sdan       ){
1428b4480e94Sdan         pC->bIndirect = 0;
1429b4480e94Sdan       }
1430e8d5648eSdan     }
1431a23a873fSdan 
1432a23a873fSdan     assert( rc==SQLITE_OK );
14336d29a4feSdan     if( pSession->bEnableSize ){
1434a23a873fSdan       rc = sessionUpdateMaxSize(op, pSession, pTab, pC);
14354fccf43aSdan     }
14366d29a4feSdan   }
143712ca0b56Sdan 
1438a23a873fSdan 
143912ca0b56Sdan   /* If an error has occurred, mark the session object as failed. */
144080fe2d93Sdan  error_out:
14411611e5a3Sdan   if( pTab->bStat1 ){
14421611e5a3Sdan     pSession->hook = stat1.hook;
14431611e5a3Sdan   }
144412ca0b56Sdan   if( rc!=SQLITE_OK ){
144512ca0b56Sdan     pSession->rc = rc;
144612ca0b56Sdan   }
144727453faeSdan }
14484fccf43aSdan 
sessionFindTable(sqlite3_session * pSession,const char * zName,SessionTable ** ppTab)1449cf8e9144Sdan static int sessionFindTable(
1450cf8e9144Sdan   sqlite3_session *pSession,
1451cf8e9144Sdan   const char *zName,
1452cf8e9144Sdan   SessionTable **ppTab
1453cf8e9144Sdan ){
1454cf8e9144Sdan   int rc = SQLITE_OK;
1455cf8e9144Sdan   int nName = sqlite3Strlen30(zName);
1456cf8e9144Sdan   SessionTable *pRet;
1457cf8e9144Sdan 
1458cf8e9144Sdan   /* Search for an existing table */
1459cf8e9144Sdan   for(pRet=pSession->pTable; pRet; pRet=pRet->pNext){
1460cf8e9144Sdan     if( 0==sqlite3_strnicmp(pRet->zName, zName, nName+1) ) break;
1461cf8e9144Sdan   }
1462cf8e9144Sdan 
1463cf8e9144Sdan   if( pRet==0 && pSession->bAutoAttach ){
1464cf8e9144Sdan     /* If there is a table-filter configured, invoke it. If it returns 0,
1465cf8e9144Sdan     ** do not automatically add the new table. */
1466cf8e9144Sdan     if( pSession->xTableFilter==0
1467cf8e9144Sdan      || pSession->xTableFilter(pSession->pFilterCtx, zName)
1468cf8e9144Sdan     ){
1469cf8e9144Sdan       rc = sqlite3session_attach(pSession, zName);
1470cf8e9144Sdan       if( rc==SQLITE_OK ){
1471*7a3b4451Sdrh         pRet = pSession->pTable;
1472*7a3b4451Sdrh         while( ALWAYS(pRet) && pRet->pNext ){
1473*7a3b4451Sdrh           pRet = pRet->pNext;
1474*7a3b4451Sdrh         }
1475*7a3b4451Sdrh         assert( pRet!=0 );
1476cf8e9144Sdan         assert( 0==sqlite3_strnicmp(pRet->zName, zName, nName+1) );
1477cf8e9144Sdan       }
1478cf8e9144Sdan     }
1479cf8e9144Sdan   }
1480cf8e9144Sdan 
1481cf8e9144Sdan   assert( rc==SQLITE_OK || pRet==0 );
1482cf8e9144Sdan   *ppTab = pRet;
1483cf8e9144Sdan   return rc;
1484cf8e9144Sdan }
1485cf8e9144Sdan 
14864fccf43aSdan /*
14874fccf43aSdan ** The 'pre-update' hook registered by this module with SQLite databases.
14884fccf43aSdan */
xPreUpdate(void * pCtx,sqlite3 * db,int op,char const * zDb,char const * zName,sqlite3_int64 iKey1,sqlite3_int64 iKey2)14894fccf43aSdan static void xPreUpdate(
14904fccf43aSdan   void *pCtx,                     /* Copy of third arg to preupdate_hook() */
14914fccf43aSdan   sqlite3 *db,                    /* Database handle */
14924fccf43aSdan   int op,                         /* SQLITE_UPDATE, DELETE or INSERT */
14934fccf43aSdan   char const *zDb,                /* Database name */
14944fccf43aSdan   char const *zName,              /* Table name */
14954fccf43aSdan   sqlite3_int64 iKey1,            /* Rowid of row about to be deleted/updated */
14964fccf43aSdan   sqlite3_int64 iKey2             /* New rowid value (for a rowid UPDATE) */
14974fccf43aSdan ){
14984fccf43aSdan   sqlite3_session *pSession;
1499cfdbde21Sdrh   int nDb = sqlite3Strlen30(zDb);
15004fccf43aSdan 
15014c220252Sdan   assert( sqlite3_mutex_held(db->mutex) );
15024c220252Sdan 
15034fccf43aSdan   for(pSession=(sqlite3_session *)pCtx; pSession; pSession=pSession->pNext){
15044fccf43aSdan     SessionTable *pTab;
1505296c7658Sdan 
1506e8d5648eSdan     /* If this session is attached to a different database ("main", "temp"
1507e8d5648eSdan     ** etc.), or if it is not currently enabled, there is nothing to do. Skip
1508e8d5648eSdan     ** to the next session object attached to this database. */
1509296c7658Sdan     if( pSession->bEnable==0 ) continue;
15104fccf43aSdan     if( pSession->rc ) continue;
15114fccf43aSdan     if( sqlite3_strnicmp(zDb, pSession->zDb, nDb+1) ) continue;
1512296c7658Sdan 
1513cf8e9144Sdan     pSession->rc = sessionFindTable(pSession, zName, &pTab);
1514cf8e9144Sdan     if( pTab ){
1515cf8e9144Sdan       assert( pSession->rc==SQLITE_OK );
1516e8d5648eSdan       sessionPreupdateOneChange(op, pSession, pTab);
1517e8d5648eSdan       if( op==SQLITE_UPDATE ){
1518e8d5648eSdan         sessionPreupdateOneChange(SQLITE_INSERT, pSession, pTab);
15194fccf43aSdan       }
15204fccf43aSdan     }
15214fccf43aSdan   }
15224fccf43aSdan }
1523cf8e9144Sdan 
1524cf8e9144Sdan /*
1525cf8e9144Sdan ** The pre-update hook implementations.
1526cf8e9144Sdan */
sessionPreupdateOld(void * pCtx,int iVal,sqlite3_value ** ppVal)1527cf8e9144Sdan static int sessionPreupdateOld(void *pCtx, int iVal, sqlite3_value **ppVal){
1528cf8e9144Sdan   return sqlite3_preupdate_old((sqlite3*)pCtx, iVal, ppVal);
1529cf8e9144Sdan }
sessionPreupdateNew(void * pCtx,int iVal,sqlite3_value ** ppVal)1530cf8e9144Sdan static int sessionPreupdateNew(void *pCtx, int iVal, sqlite3_value **ppVal){
1531cf8e9144Sdan   return sqlite3_preupdate_new((sqlite3*)pCtx, iVal, ppVal);
1532cf8e9144Sdan }
sessionPreupdateCount(void * pCtx)1533cf8e9144Sdan static int sessionPreupdateCount(void *pCtx){
1534cf8e9144Sdan   return sqlite3_preupdate_count((sqlite3*)pCtx);
1535cf8e9144Sdan }
sessionPreupdateDepth(void * pCtx)1536cf8e9144Sdan static int sessionPreupdateDepth(void *pCtx){
1537cf8e9144Sdan   return sqlite3_preupdate_depth((sqlite3*)pCtx);
1538cf8e9144Sdan }
1539cf8e9144Sdan 
1540cf8e9144Sdan /*
1541cf8e9144Sdan ** Install the pre-update hooks on the session object passed as the only
1542cf8e9144Sdan ** argument.
1543cf8e9144Sdan */
sessionPreupdateHooks(sqlite3_session * pSession)1544cf8e9144Sdan static void sessionPreupdateHooks(
1545cf8e9144Sdan   sqlite3_session *pSession
1546cf8e9144Sdan ){
1547cf8e9144Sdan   pSession->hook.pCtx = (void*)pSession->db;
1548cf8e9144Sdan   pSession->hook.xOld = sessionPreupdateOld;
1549cf8e9144Sdan   pSession->hook.xNew = sessionPreupdateNew;
1550cf8e9144Sdan   pSession->hook.xCount = sessionPreupdateCount;
1551cf8e9144Sdan   pSession->hook.xDepth = sessionPreupdateDepth;
1552cf8e9144Sdan }
1553cf8e9144Sdan 
1554cf8e9144Sdan typedef struct SessionDiffCtx SessionDiffCtx;
1555cf8e9144Sdan struct SessionDiffCtx {
1556cf8e9144Sdan   sqlite3_stmt *pStmt;
1557cf8e9144Sdan   int nOldOff;
1558cf8e9144Sdan };
1559cf8e9144Sdan 
1560cf8e9144Sdan /*
1561cf8e9144Sdan ** The diff hook implementations.
1562cf8e9144Sdan */
sessionDiffOld(void * pCtx,int iVal,sqlite3_value ** ppVal)1563cf8e9144Sdan static int sessionDiffOld(void *pCtx, int iVal, sqlite3_value **ppVal){
1564cf8e9144Sdan   SessionDiffCtx *p = (SessionDiffCtx*)pCtx;
1565cf8e9144Sdan   *ppVal = sqlite3_column_value(p->pStmt, iVal+p->nOldOff);
1566cf8e9144Sdan   return SQLITE_OK;
1567cf8e9144Sdan }
sessionDiffNew(void * pCtx,int iVal,sqlite3_value ** ppVal)1568cf8e9144Sdan static int sessionDiffNew(void *pCtx, int iVal, sqlite3_value **ppVal){
1569cf8e9144Sdan   SessionDiffCtx *p = (SessionDiffCtx*)pCtx;
1570cf8e9144Sdan   *ppVal = sqlite3_column_value(p->pStmt, iVal);
1571cf8e9144Sdan    return SQLITE_OK;
1572cf8e9144Sdan }
sessionDiffCount(void * pCtx)1573cf8e9144Sdan static int sessionDiffCount(void *pCtx){
1574cf8e9144Sdan   SessionDiffCtx *p = (SessionDiffCtx*)pCtx;
1575cf8e9144Sdan   return p->nOldOff ? p->nOldOff : sqlite3_column_count(p->pStmt);
1576cf8e9144Sdan }
sessionDiffDepth(void * pCtx)1577cf8e9144Sdan static int sessionDiffDepth(void *pCtx){
1578cf8e9144Sdan   return 0;
1579cf8e9144Sdan }
1580cf8e9144Sdan 
1581cf8e9144Sdan /*
1582cf8e9144Sdan ** Install the diff hooks on the session object passed as the only
1583cf8e9144Sdan ** argument.
1584cf8e9144Sdan */
sessionDiffHooks(sqlite3_session * pSession,SessionDiffCtx * pDiffCtx)1585cf8e9144Sdan static void sessionDiffHooks(
1586cf8e9144Sdan   sqlite3_session *pSession,
1587cf8e9144Sdan   SessionDiffCtx *pDiffCtx
1588cf8e9144Sdan ){
1589cf8e9144Sdan   pSession->hook.pCtx = (void*)pDiffCtx;
1590cf8e9144Sdan   pSession->hook.xOld = sessionDiffOld;
1591cf8e9144Sdan   pSession->hook.xNew = sessionDiffNew;
1592cf8e9144Sdan   pSession->hook.xCount = sessionDiffCount;
1593cf8e9144Sdan   pSession->hook.xDepth = sessionDiffDepth;
1594cf8e9144Sdan }
1595cf8e9144Sdan 
sessionExprComparePK(int nCol,const char * zDb1,const char * zDb2,const char * zTab,const char ** azCol,u8 * abPK)1596cf8e9144Sdan static char *sessionExprComparePK(
1597cf8e9144Sdan   int nCol,
1598cf8e9144Sdan   const char *zDb1, const char *zDb2,
1599cf8e9144Sdan   const char *zTab,
1600cf8e9144Sdan   const char **azCol, u8 *abPK
1601cf8e9144Sdan ){
1602cf8e9144Sdan   int i;
1603cf8e9144Sdan   const char *zSep = "";
1604cf8e9144Sdan   char *zRet = 0;
1605cf8e9144Sdan 
1606cf8e9144Sdan   for(i=0; i<nCol; i++){
1607cf8e9144Sdan     if( abPK[i] ){
1608cf8e9144Sdan       zRet = sqlite3_mprintf("%z%s\"%w\".\"%w\".\"%w\"=\"%w\".\"%w\".\"%w\"",
1609cf8e9144Sdan           zRet, zSep, zDb1, zTab, azCol[i], zDb2, zTab, azCol[i]
1610cf8e9144Sdan       );
1611cf8e9144Sdan       zSep = " AND ";
1612cf8e9144Sdan       if( zRet==0 ) break;
1613cf8e9144Sdan     }
1614cf8e9144Sdan   }
1615cf8e9144Sdan 
1616cf8e9144Sdan   return zRet;
1617cf8e9144Sdan }
1618cf8e9144Sdan 
sessionExprCompareOther(int nCol,const char * zDb1,const char * zDb2,const char * zTab,const char ** azCol,u8 * abPK)1619cf8e9144Sdan static char *sessionExprCompareOther(
1620cf8e9144Sdan   int nCol,
1621cf8e9144Sdan   const char *zDb1, const char *zDb2,
1622cf8e9144Sdan   const char *zTab,
1623cf8e9144Sdan   const char **azCol, u8 *abPK
1624cf8e9144Sdan ){
1625cf8e9144Sdan   int i;
1626cf8e9144Sdan   const char *zSep = "";
1627cf8e9144Sdan   char *zRet = 0;
1628cf8e9144Sdan   int bHave = 0;
1629cf8e9144Sdan 
1630cf8e9144Sdan   for(i=0; i<nCol; i++){
1631cf8e9144Sdan     if( abPK[i]==0 ){
1632cf8e9144Sdan       bHave = 1;
1633cf8e9144Sdan       zRet = sqlite3_mprintf(
1634cf8e9144Sdan           "%z%s\"%w\".\"%w\".\"%w\" IS NOT \"%w\".\"%w\".\"%w\"",
1635cf8e9144Sdan           zRet, zSep, zDb1, zTab, azCol[i], zDb2, zTab, azCol[i]
1636cf8e9144Sdan       );
1637cf8e9144Sdan       zSep = " OR ";
1638cf8e9144Sdan       if( zRet==0 ) break;
1639cf8e9144Sdan     }
1640cf8e9144Sdan   }
1641cf8e9144Sdan 
1642cf8e9144Sdan   if( bHave==0 ){
1643cf8e9144Sdan     assert( zRet==0 );
1644cf8e9144Sdan     zRet = sqlite3_mprintf("0");
1645cf8e9144Sdan   }
1646cf8e9144Sdan 
1647cf8e9144Sdan   return zRet;
1648cf8e9144Sdan }
1649cf8e9144Sdan 
sessionSelectFindNew(int nCol,const char * zDb1,const char * zDb2,const char * zTbl,const char * zExpr)1650cf8e9144Sdan static char *sessionSelectFindNew(
1651cf8e9144Sdan   int nCol,
1652cf8e9144Sdan   const char *zDb1,      /* Pick rows in this db only */
1653cf8e9144Sdan   const char *zDb2,      /* But not in this one */
1654cf8e9144Sdan   const char *zTbl,      /* Table name */
1655cf8e9144Sdan   const char *zExpr
1656cf8e9144Sdan ){
1657cf8e9144Sdan   char *zRet = sqlite3_mprintf(
1658cf8e9144Sdan       "SELECT * FROM \"%w\".\"%w\" WHERE NOT EXISTS ("
1659cf8e9144Sdan       "  SELECT 1 FROM \"%w\".\"%w\" WHERE %s"
1660cf8e9144Sdan       ")",
1661cf8e9144Sdan       zDb1, zTbl, zDb2, zTbl, zExpr
1662cf8e9144Sdan   );
1663cf8e9144Sdan   return zRet;
1664cf8e9144Sdan }
1665cf8e9144Sdan 
sessionDiffFindNew(int op,sqlite3_session * pSession,SessionTable * pTab,const char * zDb1,const char * zDb2,char * zExpr)1666cf8e9144Sdan static int sessionDiffFindNew(
1667cf8e9144Sdan   int op,
1668cf8e9144Sdan   sqlite3_session *pSession,
1669cf8e9144Sdan   SessionTable *pTab,
1670cf8e9144Sdan   const char *zDb1,
1671cf8e9144Sdan   const char *zDb2,
1672cf8e9144Sdan   char *zExpr
1673cf8e9144Sdan ){
1674cf8e9144Sdan   int rc = SQLITE_OK;
1675cf8e9144Sdan   char *zStmt = sessionSelectFindNew(pTab->nCol, zDb1, zDb2, pTab->zName,zExpr);
1676cf8e9144Sdan 
1677cf8e9144Sdan   if( zStmt==0 ){
1678cf8e9144Sdan     rc = SQLITE_NOMEM;
1679cf8e9144Sdan   }else{
1680cf8e9144Sdan     sqlite3_stmt *pStmt;
1681cf8e9144Sdan     rc = sqlite3_prepare(pSession->db, zStmt, -1, &pStmt, 0);
1682cf8e9144Sdan     if( rc==SQLITE_OK ){
1683cf8e9144Sdan       SessionDiffCtx *pDiffCtx = (SessionDiffCtx*)pSession->hook.pCtx;
1684cf8e9144Sdan       pDiffCtx->pStmt = pStmt;
1685cf8e9144Sdan       pDiffCtx->nOldOff = 0;
1686cf8e9144Sdan       while( SQLITE_ROW==sqlite3_step(pStmt) ){
1687cf8e9144Sdan         sessionPreupdateOneChange(op, pSession, pTab);
1688cf8e9144Sdan       }
1689cf8e9144Sdan       rc = sqlite3_finalize(pStmt);
1690cf8e9144Sdan     }
1691cf8e9144Sdan     sqlite3_free(zStmt);
1692cf8e9144Sdan   }
1693cf8e9144Sdan 
1694cf8e9144Sdan   return rc;
1695cf8e9144Sdan }
1696cf8e9144Sdan 
sessionDiffFindModified(sqlite3_session * pSession,SessionTable * pTab,const char * zFrom,const char * zExpr)1697cf8e9144Sdan static int sessionDiffFindModified(
1698cf8e9144Sdan   sqlite3_session *pSession,
1699cf8e9144Sdan   SessionTable *pTab,
1700cf8e9144Sdan   const char *zFrom,
1701cf8e9144Sdan   const char *zExpr
1702cf8e9144Sdan ){
1703cf8e9144Sdan   int rc = SQLITE_OK;
1704cf8e9144Sdan 
1705cf8e9144Sdan   char *zExpr2 = sessionExprCompareOther(pTab->nCol,
1706cf8e9144Sdan       pSession->zDb, zFrom, pTab->zName, pTab->azCol, pTab->abPK
1707cf8e9144Sdan   );
1708cf8e9144Sdan   if( zExpr2==0 ){
1709cf8e9144Sdan     rc = SQLITE_NOMEM;
1710cf8e9144Sdan   }else{
1711cf8e9144Sdan     char *zStmt = sqlite3_mprintf(
1712dd009f83Sdan         "SELECT * FROM \"%w\".\"%w\", \"%w\".\"%w\" WHERE %s AND (%z)",
1713cf8e9144Sdan         pSession->zDb, pTab->zName, zFrom, pTab->zName, zExpr, zExpr2
1714cf8e9144Sdan     );
1715cf8e9144Sdan     if( zStmt==0 ){
1716cf8e9144Sdan       rc = SQLITE_NOMEM;
1717cf8e9144Sdan     }else{
1718cf8e9144Sdan       sqlite3_stmt *pStmt;
1719cf8e9144Sdan       rc = sqlite3_prepare(pSession->db, zStmt, -1, &pStmt, 0);
1720cf8e9144Sdan 
1721cf8e9144Sdan       if( rc==SQLITE_OK ){
1722cf8e9144Sdan         SessionDiffCtx *pDiffCtx = (SessionDiffCtx*)pSession->hook.pCtx;
1723cf8e9144Sdan         pDiffCtx->pStmt = pStmt;
1724cf8e9144Sdan         pDiffCtx->nOldOff = pTab->nCol;
1725cf8e9144Sdan         while( SQLITE_ROW==sqlite3_step(pStmt) ){
1726cf8e9144Sdan           sessionPreupdateOneChange(SQLITE_UPDATE, pSession, pTab);
1727cf8e9144Sdan         }
1728cf8e9144Sdan         rc = sqlite3_finalize(pStmt);
1729cf8e9144Sdan       }
1730cf8e9144Sdan       sqlite3_free(zStmt);
1731cf8e9144Sdan     }
1732cf8e9144Sdan   }
1733cf8e9144Sdan 
1734cf8e9144Sdan   return rc;
1735cf8e9144Sdan }
1736cf8e9144Sdan 
sqlite3session_diff(sqlite3_session * pSession,const char * zFrom,const char * zTbl,char ** pzErrMsg)1737cf8e9144Sdan int sqlite3session_diff(
1738cf8e9144Sdan   sqlite3_session *pSession,
1739cf8e9144Sdan   const char *zFrom,
1740cf8e9144Sdan   const char *zTbl,
1741cf8e9144Sdan   char **pzErrMsg
1742cf8e9144Sdan ){
1743cf8e9144Sdan   const char *zDb = pSession->zDb;
1744cf8e9144Sdan   int rc = pSession->rc;
1745cf8e9144Sdan   SessionDiffCtx d;
1746cf8e9144Sdan 
1747cf8e9144Sdan   memset(&d, 0, sizeof(d));
1748cf8e9144Sdan   sessionDiffHooks(pSession, &d);
1749cf8e9144Sdan 
175010dc553cSdan   sqlite3_mutex_enter(sqlite3_db_mutex(pSession->db));
1751cf8e9144Sdan   if( pzErrMsg ) *pzErrMsg = 0;
1752cf8e9144Sdan   if( rc==SQLITE_OK ){
1753cf8e9144Sdan     char *zExpr = 0;
1754cf8e9144Sdan     sqlite3 *db = pSession->db;
1755cf8e9144Sdan     SessionTable *pTo;            /* Table zTbl */
1756cf8e9144Sdan 
1757cf8e9144Sdan     /* Locate and if necessary initialize the target table object */
1758cf8e9144Sdan     rc = sessionFindTable(pSession, zTbl, &pTo);
1759cf8e9144Sdan     if( pTo==0 ) goto diff_out;
17606dc29e60Sdan     if( sessionInitTable(pSession, pTo) ){
17616dc29e60Sdan       rc = pSession->rc;
17626dc29e60Sdan       goto diff_out;
1763cf8e9144Sdan     }
1764cf8e9144Sdan 
1765cf8e9144Sdan     /* Check the table schemas match */
1766cf8e9144Sdan     if( rc==SQLITE_OK ){
17674cc923e3Sdan       int bHasPk = 0;
1768b9db9099Sdan       int bMismatch = 0;
1769cf8e9144Sdan       int nCol;                   /* Columns in zFrom.zTbl */
1770cf8e9144Sdan       u8 *abPK;
1771cf8e9144Sdan       const char **azCol = 0;
17720cb735b9Sdan       rc = sessionTableInfo(0, db, zFrom, zTbl, &nCol, 0, &azCol, &abPK);
1773cf8e9144Sdan       if( rc==SQLITE_OK ){
17744cc923e3Sdan         if( pTo->nCol!=nCol ){
1775cf8e9144Sdan           bMismatch = 1;
1776cf8e9144Sdan         }else{
1777cf8e9144Sdan           int i;
1778cf8e9144Sdan           for(i=0; i<nCol; i++){
17794cc923e3Sdan             if( pTo->abPK[i]!=abPK[i] ) bMismatch = 1;
1780cf8e9144Sdan             if( sqlite3_stricmp(azCol[i], pTo->azCol[i]) ) bMismatch = 1;
17814cc923e3Sdan             if( abPK[i] ) bHasPk = 1;
1782cf8e9144Sdan           }
1783cf8e9144Sdan         }
1784cf8e9144Sdan       }
1785dbbd8160Sdrh       sqlite3_free((char*)azCol);
1786b9db9099Sdan       if( bMismatch ){
1787595d9f5fSdan         if( pzErrMsg ){
1788b9db9099Sdan           *pzErrMsg = sqlite3_mprintf("table schemas do not match");
1789595d9f5fSdan         }
1790b9db9099Sdan         rc = SQLITE_SCHEMA;
1791b9db9099Sdan       }
1792b9db9099Sdan       if( bHasPk==0 ){
1793b9db9099Sdan         /* Ignore tables with no primary keys */
1794b9db9099Sdan         goto diff_out;
1795b9db9099Sdan       }
1796cf8e9144Sdan     }
1797cf8e9144Sdan 
1798cf8e9144Sdan     if( rc==SQLITE_OK ){
1799cf8e9144Sdan       zExpr = sessionExprComparePK(pTo->nCol,
1800cf8e9144Sdan           zDb, zFrom, pTo->zName, pTo->azCol, pTo->abPK
1801cf8e9144Sdan       );
1802cf8e9144Sdan     }
1803cf8e9144Sdan 
1804cf8e9144Sdan     /* Find new rows */
1805cf8e9144Sdan     if( rc==SQLITE_OK ){
1806cf8e9144Sdan       rc = sessionDiffFindNew(SQLITE_INSERT, pSession, pTo, zDb, zFrom, zExpr);
1807cf8e9144Sdan     }
1808cf8e9144Sdan 
1809cf8e9144Sdan     /* Find old rows */
1810cf8e9144Sdan     if( rc==SQLITE_OK ){
1811cf8e9144Sdan       rc = sessionDiffFindNew(SQLITE_DELETE, pSession, pTo, zFrom, zDb, zExpr);
1812cf8e9144Sdan     }
1813cf8e9144Sdan 
1814cf8e9144Sdan     /* Find modified rows */
1815cf8e9144Sdan     if( rc==SQLITE_OK ){
1816cf8e9144Sdan       rc = sessionDiffFindModified(pSession, pTo, zFrom, zExpr);
1817cf8e9144Sdan     }
1818cf8e9144Sdan 
1819cf8e9144Sdan     sqlite3_free(zExpr);
1820cf8e9144Sdan   }
1821cf8e9144Sdan 
1822cf8e9144Sdan  diff_out:
1823cf8e9144Sdan   sessionPreupdateHooks(pSession);
182410dc553cSdan   sqlite3_mutex_leave(sqlite3_db_mutex(pSession->db));
1825cf8e9144Sdan   return rc;
1826296c7658Sdan }
18274fccf43aSdan 
18284fccf43aSdan /*
18294fccf43aSdan ** Create a session object. This session object will record changes to
18304fccf43aSdan ** database zDb attached to connection db.
18314fccf43aSdan */
sqlite3session_create(sqlite3 * db,const char * zDb,sqlite3_session ** ppSession)18324fccf43aSdan int sqlite3session_create(
18334fccf43aSdan   sqlite3 *db,                    /* Database handle */
18344fccf43aSdan   const char *zDb,                /* Name of db (e.g. "main") */
18354fccf43aSdan   sqlite3_session **ppSession     /* OUT: New session object */
18364fccf43aSdan ){
1837296c7658Sdan   sqlite3_session *pNew;          /* Newly allocated session object */
1838296c7658Sdan   sqlite3_session *pOld;          /* Session object already attached to db */
1839cfdbde21Sdrh   int nDb = sqlite3Strlen30(zDb); /* Length of zDb in bytes */
18404fccf43aSdan 
1841296c7658Sdan   /* Zero the output value in case an error occurs. */
18424fccf43aSdan   *ppSession = 0;
18434fccf43aSdan 
18444fccf43aSdan   /* Allocate and populate the new session object. */
18452d77d80aSdrh   pNew = (sqlite3_session *)sqlite3_malloc64(sizeof(sqlite3_session) + nDb + 1);
18464fccf43aSdan   if( !pNew ) return SQLITE_NOMEM;
18474fccf43aSdan   memset(pNew, 0, sizeof(sqlite3_session));
18484fccf43aSdan   pNew->db = db;
18494fccf43aSdan   pNew->zDb = (char *)&pNew[1];
1850296c7658Sdan   pNew->bEnable = 1;
18514fccf43aSdan   memcpy(pNew->zDb, zDb, nDb+1);
1852cf8e9144Sdan   sessionPreupdateHooks(pNew);
18534fccf43aSdan 
18544fccf43aSdan   /* Add the new session object to the linked list of session objects
18554fccf43aSdan   ** attached to database handle $db. Do this under the cover of the db
18564fccf43aSdan   ** handle mutex.  */
18574fccf43aSdan   sqlite3_mutex_enter(sqlite3_db_mutex(db));
18584fccf43aSdan   pOld = (sqlite3_session*)sqlite3_preupdate_hook(db, xPreUpdate, (void*)pNew);
18594fccf43aSdan   pNew->pNext = pOld;
18604fccf43aSdan   sqlite3_mutex_leave(sqlite3_db_mutex(db));
18614fccf43aSdan 
18624fccf43aSdan   *ppSession = pNew;
18634fccf43aSdan   return SQLITE_OK;
18644fccf43aSdan }
18654fccf43aSdan 
186677fc1d5bSdan /*
186777fc1d5bSdan ** Free the list of table objects passed as the first argument. The contents
186877fc1d5bSdan ** of the changed-rows hash tables are also deleted.
186977fc1d5bSdan */
sessionDeleteTable(sqlite3_session * pSession,SessionTable * pList)18700cb735b9Sdan static void sessionDeleteTable(sqlite3_session *pSession, SessionTable *pList){
18715d607a6eSdan   SessionTable *pNext;
18725d607a6eSdan   SessionTable *pTab;
18735d607a6eSdan 
18745d607a6eSdan   for(pTab=pList; pTab; pTab=pNext){
18755d607a6eSdan     int i;
18765d607a6eSdan     pNext = pTab->pNext;
18775d607a6eSdan     for(i=0; i<pTab->nChange; i++){
18785d607a6eSdan       SessionChange *p;
187902d436b1Smistachkin       SessionChange *pNextChange;
188002d436b1Smistachkin       for(p=pTab->apChange[i]; p; p=pNextChange){
188102d436b1Smistachkin         pNextChange = p->pNext;
18820cb735b9Sdan         sessionFree(pSession, p);
18835d607a6eSdan       }
18845d607a6eSdan     }
18850cb735b9Sdan     sessionFree(pSession, (char*)pTab->azCol);  /* cast works around VC++ bug */
18860cb735b9Sdan     sessionFree(pSession, pTab->apChange);
18870cb735b9Sdan     sessionFree(pSession, pTab);
18885d607a6eSdan   }
18895d607a6eSdan }
18905d607a6eSdan 
18914fccf43aSdan /*
18924fccf43aSdan ** Delete a session object previously allocated using sqlite3session_create().
18934fccf43aSdan */
sqlite3session_delete(sqlite3_session * pSession)18944fccf43aSdan void sqlite3session_delete(sqlite3_session *pSession){
18954fccf43aSdan   sqlite3 *db = pSession->db;
18964fccf43aSdan   sqlite3_session *pHead;
18974fccf43aSdan   sqlite3_session **pp;
18984fccf43aSdan 
1899296c7658Sdan   /* Unlink the session from the linked list of sessions attached to the
1900296c7658Sdan   ** database handle. Hold the db mutex while doing so.  */
19014fccf43aSdan   sqlite3_mutex_enter(sqlite3_db_mutex(db));
19024fccf43aSdan   pHead = (sqlite3_session*)sqlite3_preupdate_hook(db, 0, 0);
190350d348b1Sdrh   for(pp=&pHead; ALWAYS((*pp)!=0); pp=&((*pp)->pNext)){
190450d348b1Sdrh     if( (*pp)==pSession ){
19054fccf43aSdan       *pp = (*pp)->pNext;
19064fccf43aSdan       if( pHead ) sqlite3_preupdate_hook(db, xPreUpdate, (void*)pHead);
190750d348b1Sdrh       break;
190850d348b1Sdrh     }
190950d348b1Sdrh   }
19104fccf43aSdan   sqlite3_mutex_leave(sqlite3_db_mutex(db));
19111611e5a3Sdan   sqlite3ValueFree(pSession->pZeroBlob);
19124fccf43aSdan 
1913296c7658Sdan   /* Delete all attached table objects. And the contents of their
1914296c7658Sdan   ** associated hash-tables. */
19150cb735b9Sdan   sessionDeleteTable(pSession, pSession->pTable);
19164fccf43aSdan 
19170cb735b9Sdan   /* Assert that all allocations have been freed and then free the
19180cb735b9Sdan   ** session object itself. */
19190cb735b9Sdan   assert( pSession->nMalloc==0 );
19204fccf43aSdan   sqlite3_free(pSession);
19214fccf43aSdan }
19224fccf43aSdan 
19234fccf43aSdan /*
19247531a5a3Sdan ** Set a table filter on a Session Object.
19257531a5a3Sdan */
sqlite3session_table_filter(sqlite3_session * pSession,int (* xFilter)(void *,const char *),void * pCtx)19267531a5a3Sdan void sqlite3session_table_filter(
19277531a5a3Sdan   sqlite3_session *pSession,
19287531a5a3Sdan   int(*xFilter)(void*, const char*),
19297531a5a3Sdan   void *pCtx                      /* First argument passed to xFilter */
19307531a5a3Sdan ){
19317531a5a3Sdan   pSession->bAutoAttach = 1;
19327531a5a3Sdan   pSession->pFilterCtx = pCtx;
19337531a5a3Sdan   pSession->xTableFilter = xFilter;
19347531a5a3Sdan }
19357531a5a3Sdan 
19367531a5a3Sdan /*
19374fccf43aSdan ** Attach a table to a session. All subsequent changes made to the table
19384fccf43aSdan ** while the session object is enabled will be recorded.
19394fccf43aSdan **
19404fccf43aSdan ** Only tables that have a PRIMARY KEY defined may be attached. It does
19414fccf43aSdan ** not matter if the PRIMARY KEY is an "INTEGER PRIMARY KEY" (rowid alias)
19424fccf43aSdan ** or not.
19434fccf43aSdan */
sqlite3session_attach(sqlite3_session * pSession,const char * zName)19444fccf43aSdan int sqlite3session_attach(
19454fccf43aSdan   sqlite3_session *pSession,      /* Session object */
19464fccf43aSdan   const char *zName               /* Table name */
19474fccf43aSdan ){
1948ff4d0f41Sdan   int rc = SQLITE_OK;
1949ff4d0f41Sdan   sqlite3_mutex_enter(sqlite3_db_mutex(pSession->db));
1950ff4d0f41Sdan 
1951ff4d0f41Sdan   if( !zName ){
1952ff4d0f41Sdan     pSession->bAutoAttach = 1;
1953ff4d0f41Sdan   }else{
1954296c7658Sdan     SessionTable *pTab;           /* New table object (if required) */
1955296c7658Sdan     int nName;                    /* Number of bytes in string zName */
19564fccf43aSdan 
19574fccf43aSdan     /* First search for an existing entry. If one is found, this call is
19584fccf43aSdan     ** a no-op. Return early. */
1959cfdbde21Sdrh     nName = sqlite3Strlen30(zName);
19604fccf43aSdan     for(pTab=pSession->pTable; pTab; pTab=pTab->pNext){
19614c220252Sdan       if( 0==sqlite3_strnicmp(pTab->zName, zName, nName+1) ) break;
19624fccf43aSdan     }
19634fccf43aSdan 
19644c220252Sdan     if( !pTab ){
19654fccf43aSdan       /* Allocate new SessionTable object. */
19660cb735b9Sdan       int nByte = sizeof(SessionTable) + nName + 1;
19670cb735b9Sdan       pTab = (SessionTable*)sessionMalloc64(pSession, nByte);
19684c220252Sdan       if( !pTab ){
19694c220252Sdan         rc = SQLITE_NOMEM;
19704c220252Sdan       }else{
19716c39e6a8Sdan         /* Populate the new SessionTable object and link it into the list.
19726c39e6a8Sdan         ** The new object must be linked onto the end of the list, not
19736c39e6a8Sdan         ** simply added to the start of it in order to ensure that tables
19746c39e6a8Sdan         ** appear in the correct order when a changeset or patchset is
19756c39e6a8Sdan         ** eventually generated. */
19766c39e6a8Sdan         SessionTable **ppTab;
19774fccf43aSdan         memset(pTab, 0, sizeof(SessionTable));
19784fccf43aSdan         pTab->zName = (char *)&pTab[1];
19794fccf43aSdan         memcpy(pTab->zName, zName, nName+1);
19806c39e6a8Sdan         for(ppTab=&pSession->pTable; *ppTab; ppTab=&(*ppTab)->pNext);
19816c39e6a8Sdan         *ppTab = pTab;
19824c220252Sdan       }
19834c220252Sdan     }
1984ff4d0f41Sdan   }
19854fccf43aSdan 
19864c220252Sdan   sqlite3_mutex_leave(sqlite3_db_mutex(pSession->db));
19874c220252Sdan   return rc;
19884fccf43aSdan }
19894fccf43aSdan 
1990296c7658Sdan /*
1991296c7658Sdan ** Ensure that there is room in the buffer to append nByte bytes of data.
1992296c7658Sdan ** If not, use sqlite3_realloc() to grow the buffer so that there is.
1993296c7658Sdan **
1994296c7658Sdan ** If successful, return zero. Otherwise, if an OOM condition is encountered,
1995296c7658Sdan ** set *pRc to SQLITE_NOMEM and return non-zero.
1996296c7658Sdan */
sessionBufferGrow(SessionBuffer * p,i64 nByte,int * pRc)19974df68e0aSdan static int sessionBufferGrow(SessionBuffer *p, i64 nByte, int *pRc){
19984df68e0aSdan #define SESSION_MAX_BUFFER_SZ (0x7FFFFF00 - 1)
19994df68e0aSdan   i64 nReq = p->nBuf + nByte;
20004df68e0aSdan   if( *pRc==SQLITE_OK && nReq>p->nAlloc ){
20014fccf43aSdan     u8 *aNew;
20029c18ef09Sdan     i64 nNew = p->nAlloc ? p->nAlloc : 128;
20034df68e0aSdan 
20044fccf43aSdan     do {
20054fccf43aSdan       nNew = nNew*2;
20064df68e0aSdan     }while( nNew<nReq );
20074df68e0aSdan 
20084df68e0aSdan     /* The value of SESSION_MAX_BUFFER_SZ is copied from the implementation
20094df68e0aSdan     ** of sqlite3_realloc64(). Allocations greater than this size in bytes
20104df68e0aSdan     ** always fail. It is used here to ensure that this routine can always
20114df68e0aSdan     ** allocate up to this limit - instead of up to the largest power of
20124df68e0aSdan     ** two smaller than the limit.  */
20134df68e0aSdan     if( nNew>SESSION_MAX_BUFFER_SZ ){
20144df68e0aSdan       nNew = SESSION_MAX_BUFFER_SZ;
20154df68e0aSdan       if( nNew<nReq ){
20164df68e0aSdan         *pRc = SQLITE_NOMEM;
20174df68e0aSdan         return 1;
20184df68e0aSdan       }
20194df68e0aSdan     }
20204fccf43aSdan 
20219c18ef09Sdan     aNew = (u8 *)sqlite3_realloc64(p->aBuf, nNew);
20224fccf43aSdan     if( 0==aNew ){
20234fccf43aSdan       *pRc = SQLITE_NOMEM;
202480fe2d93Sdan     }else{
20254fccf43aSdan       p->aBuf = aNew;
20264fccf43aSdan       p->nAlloc = nNew;
20274fccf43aSdan     }
202880fe2d93Sdan   }
202980fe2d93Sdan   return (*pRc!=SQLITE_OK);
20304fccf43aSdan }
20314fccf43aSdan 
2032296c7658Sdan /*
2033fa122adaSdan ** Append the value passed as the second argument to the buffer passed
2034fa122adaSdan ** as the first.
2035fa122adaSdan **
2036fa122adaSdan ** This function is a no-op if *pRc is non-zero when it is called.
2037fa122adaSdan ** Otherwise, if an error occurs, *pRc is set to an SQLite error code
2038fa122adaSdan ** before returning.
2039fa122adaSdan */
sessionAppendValue(SessionBuffer * p,sqlite3_value * pVal,int * pRc)2040fa122adaSdan static void sessionAppendValue(SessionBuffer *p, sqlite3_value *pVal, int *pRc){
2041fa122adaSdan   int rc = *pRc;
2042fa122adaSdan   if( rc==SQLITE_OK ){
20432d77d80aSdrh     sqlite3_int64 nByte = 0;
2044e8fa8c96Sdan     rc = sessionSerializeValue(0, pVal, &nByte);
2045fa122adaSdan     sessionBufferGrow(p, nByte, &rc);
2046fa122adaSdan     if( rc==SQLITE_OK ){
2047fa122adaSdan       rc = sessionSerializeValue(&p->aBuf[p->nBuf], pVal, 0);
2048fa122adaSdan       p->nBuf += nByte;
2049fa122adaSdan     }else{
2050fa122adaSdan       *pRc = rc;
2051fa122adaSdan     }
2052fa122adaSdan   }
2053fa122adaSdan }
2054fa122adaSdan 
2055fa122adaSdan /*
2056296c7658Sdan ** This function is a no-op if *pRc is other than SQLITE_OK when it is
2057296c7658Sdan ** called. Otherwise, append a single byte to the buffer.
2058296c7658Sdan **
2059296c7658Sdan ** If an OOM condition is encountered, set *pRc to SQLITE_NOMEM before
2060296c7658Sdan ** returning.
2061296c7658Sdan */
sessionAppendByte(SessionBuffer * p,u8 v,int * pRc)20624fccf43aSdan static void sessionAppendByte(SessionBuffer *p, u8 v, int *pRc){
206380fe2d93Sdan   if( 0==sessionBufferGrow(p, 1, pRc) ){
20644fccf43aSdan     p->aBuf[p->nBuf++] = v;
20654fccf43aSdan   }
20664fccf43aSdan }
20674fccf43aSdan 
2068296c7658Sdan /*
2069296c7658Sdan ** This function is a no-op if *pRc is other than SQLITE_OK when it is
2070296c7658Sdan ** called. Otherwise, append a single varint to the buffer.
2071296c7658Sdan **
2072296c7658Sdan ** If an OOM condition is encountered, set *pRc to SQLITE_NOMEM before
2073296c7658Sdan ** returning.
2074296c7658Sdan */
sessionAppendVarint(SessionBuffer * p,int v,int * pRc)2075cfdbde21Sdrh static void sessionAppendVarint(SessionBuffer *p, int v, int *pRc){
207680fe2d93Sdan   if( 0==sessionBufferGrow(p, 9, pRc) ){
20774fccf43aSdan     p->nBuf += sessionVarintPut(&p->aBuf[p->nBuf], v);
20784fccf43aSdan   }
20794fccf43aSdan }
20804fccf43aSdan 
2081296c7658Sdan /*
2082296c7658Sdan ** This function is a no-op if *pRc is other than SQLITE_OK when it is
2083296c7658Sdan ** called. Otherwise, append a blob of data to the buffer.
2084296c7658Sdan **
2085296c7658Sdan ** If an OOM condition is encountered, set *pRc to SQLITE_NOMEM before
2086296c7658Sdan ** returning.
2087296c7658Sdan */
sessionAppendBlob(SessionBuffer * p,const u8 * aBlob,int nBlob,int * pRc)20884fccf43aSdan static void sessionAppendBlob(
20894fccf43aSdan   SessionBuffer *p,
20904fccf43aSdan   const u8 *aBlob,
20914fccf43aSdan   int nBlob,
20924fccf43aSdan   int *pRc
20934fccf43aSdan ){
2094895decf6Sdan   if( nBlob>0 && 0==sessionBufferGrow(p, nBlob, pRc) ){
20954fccf43aSdan     memcpy(&p->aBuf[p->nBuf], aBlob, nBlob);
20964fccf43aSdan     p->nBuf += nBlob;
20974fccf43aSdan   }
20984fccf43aSdan }
20994fccf43aSdan 
2100296c7658Sdan /*
2101296c7658Sdan ** This function is a no-op if *pRc is other than SQLITE_OK when it is
2102296c7658Sdan ** called. Otherwise, append a string to the buffer. All bytes in the string
2103296c7658Sdan ** up to (but not including) the nul-terminator are written to the buffer.
2104296c7658Sdan **
2105296c7658Sdan ** If an OOM condition is encountered, set *pRc to SQLITE_NOMEM before
2106296c7658Sdan ** returning.
2107296c7658Sdan */
sessionAppendStr(SessionBuffer * p,const char * zStr,int * pRc)2108d5f0767cSdan static void sessionAppendStr(
2109d5f0767cSdan   SessionBuffer *p,
2110d5f0767cSdan   const char *zStr,
2111d5f0767cSdan   int *pRc
2112d5f0767cSdan ){
2113cfdbde21Sdrh   int nStr = sqlite3Strlen30(zStr);
211480fe2d93Sdan   if( 0==sessionBufferGrow(p, nStr, pRc) ){
2115d5f0767cSdan     memcpy(&p->aBuf[p->nBuf], zStr, nStr);
2116d5f0767cSdan     p->nBuf += nStr;
2117d5f0767cSdan   }
2118d5f0767cSdan }
2119d5f0767cSdan 
2120296c7658Sdan /*
2121296c7658Sdan ** This function is a no-op if *pRc is other than SQLITE_OK when it is
2122296c7658Sdan ** called. Otherwise, append the string representation of integer iVal
2123296c7658Sdan ** to the buffer. No nul-terminator is written.
2124296c7658Sdan **
2125296c7658Sdan ** If an OOM condition is encountered, set *pRc to SQLITE_NOMEM before
2126296c7658Sdan ** returning.
2127296c7658Sdan */
sessionAppendInteger(SessionBuffer * p,int iVal,int * pRc)2128d5f0767cSdan static void sessionAppendInteger(
2129296c7658Sdan   SessionBuffer *p,               /* Buffer to append to */
2130296c7658Sdan   int iVal,                       /* Value to write the string rep. of */
2131296c7658Sdan   int *pRc                        /* IN/OUT: Error code */
2132d5f0767cSdan ){
2133d5f0767cSdan   char aBuf[24];
2134d5f0767cSdan   sqlite3_snprintf(sizeof(aBuf)-1, aBuf, "%d", iVal);
2135d5f0767cSdan   sessionAppendStr(p, aBuf, pRc);
2136d5f0767cSdan }
2137d5f0767cSdan 
2138296c7658Sdan /*
2139296c7658Sdan ** This function is a no-op if *pRc is other than SQLITE_OK when it is
2140296c7658Sdan ** called. Otherwise, append the string zStr enclosed in quotes (") and
2141296c7658Sdan ** with any embedded quote characters escaped to the buffer. No
2142296c7658Sdan ** nul-terminator byte is written.
2143296c7658Sdan **
2144296c7658Sdan ** If an OOM condition is encountered, set *pRc to SQLITE_NOMEM before
2145296c7658Sdan ** returning.
2146296c7658Sdan */
sessionAppendIdent(SessionBuffer * p,const char * zStr,int * pRc)2147d5f0767cSdan static void sessionAppendIdent(
2148296c7658Sdan   SessionBuffer *p,               /* Buffer to a append to */
2149296c7658Sdan   const char *zStr,               /* String to quote, escape and append */
2150296c7658Sdan   int *pRc                        /* IN/OUT: Error code */
2151d5f0767cSdan ){
2152cfdbde21Sdrh   int nStr = sqlite3Strlen30(zStr)*2 + 2 + 1;
215380fe2d93Sdan   if( 0==sessionBufferGrow(p, nStr, pRc) ){
2154d5f0767cSdan     char *zOut = (char *)&p->aBuf[p->nBuf];
2155d5f0767cSdan     const char *zIn = zStr;
2156d5f0767cSdan     *zOut++ = '"';
2157d5f0767cSdan     while( *zIn ){
2158d5f0767cSdan       if( *zIn=='"' ) *zOut++ = '"';
2159d5f0767cSdan       *zOut++ = *(zIn++);
2160d5f0767cSdan     }
2161d5f0767cSdan     *zOut++ = '"';
2162cfdbde21Sdrh     p->nBuf = (int)((u8 *)zOut - p->aBuf);
2163d5f0767cSdan   }
2164d5f0767cSdan }
2165d5f0767cSdan 
2166296c7658Sdan /*
2167296c7658Sdan ** This function is a no-op if *pRc is other than SQLITE_OK when it is
2168296c7658Sdan ** called. Otherwse, it appends the serialized version of the value stored
2169296c7658Sdan ** in column iCol of the row that SQL statement pStmt currently points
2170296c7658Sdan ** to to the buffer.
2171296c7658Sdan */
sessionAppendCol(SessionBuffer * p,sqlite3_stmt * pStmt,int iCol,int * pRc)21724fccf43aSdan static void sessionAppendCol(
2173296c7658Sdan   SessionBuffer *p,               /* Buffer to append to */
2174296c7658Sdan   sqlite3_stmt *pStmt,            /* Handle pointing to row containing value */
2175296c7658Sdan   int iCol,                       /* Column to read value from */
2176296c7658Sdan   int *pRc                        /* IN/OUT: Error code */
21774fccf43aSdan ){
21784fccf43aSdan   if( *pRc==SQLITE_OK ){
21794fccf43aSdan     int eType = sqlite3_column_type(pStmt, iCol);
21804fccf43aSdan     sessionAppendByte(p, (u8)eType, pRc);
21814fccf43aSdan     if( eType==SQLITE_INTEGER || eType==SQLITE_FLOAT ){
21824fccf43aSdan       sqlite3_int64 i;
21834fccf43aSdan       u8 aBuf[8];
21844fccf43aSdan       if( eType==SQLITE_INTEGER ){
21854fccf43aSdan         i = sqlite3_column_int64(pStmt, iCol);
21864fccf43aSdan       }else{
21874fccf43aSdan         double r = sqlite3_column_double(pStmt, iCol);
21884fccf43aSdan         memcpy(&i, &r, 8);
21894fccf43aSdan       }
2190296c7658Sdan       sessionPutI64(aBuf, i);
21914fccf43aSdan       sessionAppendBlob(p, aBuf, 8, pRc);
21924fccf43aSdan     }
21934fccf43aSdan     if( eType==SQLITE_BLOB || eType==SQLITE_TEXT ){
21946734007dSdan       u8 *z;
21953cc89d95Sdan       int nByte;
21966734007dSdan       if( eType==SQLITE_BLOB ){
21976734007dSdan         z = (u8 *)sqlite3_column_blob(pStmt, iCol);
21986734007dSdan       }else{
21996734007dSdan         z = (u8 *)sqlite3_column_text(pStmt, iCol);
22006734007dSdan       }
22013cc89d95Sdan       nByte = sqlite3_column_bytes(pStmt, iCol);
22023cc89d95Sdan       if( z || (eType==SQLITE_BLOB && nByte==0) ){
22034fccf43aSdan         sessionAppendVarint(p, nByte, pRc);
22046734007dSdan         sessionAppendBlob(p, z, nByte, pRc);
22056734007dSdan       }else{
22066734007dSdan         *pRc = SQLITE_NOMEM;
22076734007dSdan       }
22084fccf43aSdan     }
22094fccf43aSdan   }
22104fccf43aSdan }
22114fccf43aSdan 
2212296c7658Sdan /*
2213296c7658Sdan **
221480fe2d93Sdan ** This function appends an update change to the buffer (see the comments
221580fe2d93Sdan ** under "CHANGESET FORMAT" at the top of the file). An update change
221680fe2d93Sdan ** consists of:
2217296c7658Sdan **
2218296c7658Sdan **   1 byte:  SQLITE_UPDATE (0x17)
2219296c7658Sdan **   n bytes: old.* record (see RECORD FORMAT)
2220296c7658Sdan **   m bytes: new.* record (see RECORD FORMAT)
2221296c7658Sdan **
2222296c7658Sdan ** The SessionChange object passed as the third argument contains the
2223296c7658Sdan ** values that were stored in the row when the session began (the old.*
2224296c7658Sdan ** values). The statement handle passed as the second argument points
2225296c7658Sdan ** at the current version of the row (the new.* values).
2226296c7658Sdan **
2227296c7658Sdan ** If all of the old.* values are equal to their corresponding new.* value
2228296c7658Sdan ** (i.e. nothing has changed), then no data at all is appended to the buffer.
2229296c7658Sdan **
2230296c7658Sdan ** Otherwise, the old.* record contains all primary key values and the
2231296c7658Sdan ** original values of any fields that have been modified. The new.* record
2232296c7658Sdan ** contains the new values of only those fields that have been modified.
2233296c7658Sdan */
sessionAppendUpdate(SessionBuffer * pBuf,int bPatchset,sqlite3_stmt * pStmt,SessionChange * p,u8 * abPK)223480fe2d93Sdan static int sessionAppendUpdate(
2235296c7658Sdan   SessionBuffer *pBuf,            /* Buffer to append to */
223673b3c055Sdan   int bPatchset,                  /* True for "patchset", 0 for "changeset" */
2237296c7658Sdan   sqlite3_stmt *pStmt,            /* Statement handle pointing at new row */
2238296c7658Sdan   SessionChange *p,               /* Object containing old values */
223980fe2d93Sdan   u8 *abPK                        /* Boolean array - true for PK columns */
22404fccf43aSdan ){
224180fe2d93Sdan   int rc = SQLITE_OK;
2242296c7658Sdan   SessionBuffer buf2 = {0,0,0}; /* Buffer to accumulate new.* record in */
2243296c7658Sdan   int bNoop = 1;                /* Set to zero if any values are modified */
22441f34f8ccSdan   int nRewind = pBuf->nBuf;     /* Set to zero if any values are modified */
2245296c7658Sdan   int i;                        /* Used to iterate through columns */
2246296c7658Sdan   u8 *pCsr = p->aRecord;        /* Used to iterate through old.* values */
2247296c7658Sdan 
2248*7a3b4451Sdrh   assert( abPK!=0 );
224980fe2d93Sdan   sessionAppendByte(pBuf, SQLITE_UPDATE, &rc);
225080fe2d93Sdan   sessionAppendByte(pBuf, p->bIndirect, &rc);
22514fccf43aSdan   for(i=0; i<sqlite3_column_count(pStmt); i++){
225237f133ecSdan     int bChanged = 0;
22534fccf43aSdan     int nAdvance;
22544fccf43aSdan     int eType = *pCsr;
22554fccf43aSdan     switch( eType ){
22564fccf43aSdan       case SQLITE_NULL:
22574fccf43aSdan         nAdvance = 1;
22584fccf43aSdan         if( sqlite3_column_type(pStmt, i)!=SQLITE_NULL ){
225937f133ecSdan           bChanged = 1;
22604fccf43aSdan         }
22614fccf43aSdan         break;
22624fccf43aSdan 
22634fccf43aSdan       case SQLITE_FLOAT:
22644fccf43aSdan       case SQLITE_INTEGER: {
22654fccf43aSdan         nAdvance = 9;
22664fccf43aSdan         if( eType==sqlite3_column_type(pStmt, i) ){
22674fccf43aSdan           sqlite3_int64 iVal = sessionGetI64(&pCsr[1]);
22684fccf43aSdan           if( eType==SQLITE_INTEGER ){
22694fccf43aSdan             if( iVal==sqlite3_column_int64(pStmt, i) ) break;
22704fccf43aSdan           }else{
22714fccf43aSdan             double dVal;
22724fccf43aSdan             memcpy(&dVal, &iVal, 8);
22734fccf43aSdan             if( dVal==sqlite3_column_double(pStmt, i) ) break;
22744fccf43aSdan           }
22754fccf43aSdan         }
227637f133ecSdan         bChanged = 1;
22774fccf43aSdan         break;
22784fccf43aSdan       }
22794fccf43aSdan 
2280e5754eecSdan       default: {
2281895decf6Sdan         int n;
2282895decf6Sdan         int nHdr = 1 + sessionVarintGet(&pCsr[1], &n);
2283e5754eecSdan         assert( eType==SQLITE_TEXT || eType==SQLITE_BLOB );
2284895decf6Sdan         nAdvance = nHdr + n;
22854fccf43aSdan         if( eType==sqlite3_column_type(pStmt, i)
2286895decf6Sdan          && n==sqlite3_column_bytes(pStmt, i)
2287895decf6Sdan          && (n==0 || 0==memcmp(&pCsr[nHdr], sqlite3_column_blob(pStmt, i), n))
22884fccf43aSdan         ){
22894fccf43aSdan           break;
22904fccf43aSdan         }
229137f133ecSdan         bChanged = 1;
22924fccf43aSdan       }
22934fccf43aSdan     }
22944fccf43aSdan 
229573b3c055Sdan     /* If at least one field has been modified, this is not a no-op. */
229673b3c055Sdan     if( bChanged ) bNoop = 0;
229773b3c055Sdan 
229873b3c055Sdan     /* Add a field to the old.* record. This is omitted if this modules is
229973b3c055Sdan     ** currently generating a patchset. */
230073b3c055Sdan     if( bPatchset==0 ){
230137f133ecSdan       if( bChanged || abPK[i] ){
230280fe2d93Sdan         sessionAppendBlob(pBuf, pCsr, nAdvance, &rc);
23034fccf43aSdan       }else{
230480fe2d93Sdan         sessionAppendByte(pBuf, 0, &rc);
230537f133ecSdan       }
230673b3c055Sdan     }
230737f133ecSdan 
230873b3c055Sdan     /* Add a field to the new.* record. Or the only record if currently
230973b3c055Sdan     ** generating a patchset.  */
231073b3c055Sdan     if( bChanged || (bPatchset && abPK[i]) ){
231180fe2d93Sdan       sessionAppendCol(&buf2, pStmt, i, &rc);
231237f133ecSdan     }else{
231380fe2d93Sdan       sessionAppendByte(&buf2, 0, &rc);
23144fccf43aSdan     }
231537f133ecSdan 
23164fccf43aSdan     pCsr += nAdvance;
23174fccf43aSdan   }
23184fccf43aSdan 
23194fccf43aSdan   if( bNoop ){
23201f34f8ccSdan     pBuf->nBuf = nRewind;
23214fccf43aSdan   }else{
232280fe2d93Sdan     sessionAppendBlob(pBuf, buf2.aBuf, buf2.nBuf, &rc);
23234fccf43aSdan   }
23241f34f8ccSdan   sqlite3_free(buf2.aBuf);
232580fe2d93Sdan 
232680fe2d93Sdan   return rc;
2327d5f0767cSdan }
23284fccf43aSdan 
2329a71d2371Sdan /*
2330a71d2371Sdan ** Append a DELETE change to the buffer passed as the first argument. Use
2331a71d2371Sdan ** the changeset format if argument bPatchset is zero, or the patchset
2332a71d2371Sdan ** format otherwise.
2333a71d2371Sdan */
sessionAppendDelete(SessionBuffer * pBuf,int bPatchset,SessionChange * p,int nCol,u8 * abPK)233473b3c055Sdan static int sessionAppendDelete(
233573b3c055Sdan   SessionBuffer *pBuf,            /* Buffer to append to */
233673b3c055Sdan   int bPatchset,                  /* True for "patchset", 0 for "changeset" */
233773b3c055Sdan   SessionChange *p,               /* Object containing old values */
2338a71d2371Sdan   int nCol,                       /* Number of columns in table */
233973b3c055Sdan   u8 *abPK                        /* Boolean array - true for PK columns */
234073b3c055Sdan ){
234173b3c055Sdan   int rc = SQLITE_OK;
234273b3c055Sdan 
234373b3c055Sdan   sessionAppendByte(pBuf, SQLITE_DELETE, &rc);
234473b3c055Sdan   sessionAppendByte(pBuf, p->bIndirect, &rc);
234573b3c055Sdan 
234673b3c055Sdan   if( bPatchset==0 ){
234773b3c055Sdan     sessionAppendBlob(pBuf, p->aRecord, p->nRecord, &rc);
234873b3c055Sdan   }else{
234973b3c055Sdan     int i;
235073b3c055Sdan     u8 *a = p->aRecord;
235173b3c055Sdan     for(i=0; i<nCol; i++){
235273b3c055Sdan       u8 *pStart = a;
235373b3c055Sdan       int eType = *a++;
235473b3c055Sdan 
235573b3c055Sdan       switch( eType ){
235673b3c055Sdan         case 0:
235773b3c055Sdan         case SQLITE_NULL:
235873b3c055Sdan           assert( abPK[i]==0 );
235973b3c055Sdan           break;
236073b3c055Sdan 
236173b3c055Sdan         case SQLITE_FLOAT:
236273b3c055Sdan         case SQLITE_INTEGER:
236373b3c055Sdan           a += 8;
236473b3c055Sdan           break;
236573b3c055Sdan 
236673b3c055Sdan         default: {
236773b3c055Sdan           int n;
236873b3c055Sdan           a += sessionVarintGet(a, &n);
236973b3c055Sdan           a += n;
237073b3c055Sdan           break;
237173b3c055Sdan         }
237273b3c055Sdan       }
237373b3c055Sdan       if( abPK[i] ){
2374f5ab08c7Sdrh         sessionAppendBlob(pBuf, pStart, (int)(a-pStart), &rc);
237573b3c055Sdan       }
237673b3c055Sdan     }
237773b3c055Sdan     assert( (a - p->aRecord)==p->nRecord );
237873b3c055Sdan   }
237973b3c055Sdan 
238073b3c055Sdan   return rc;
238173b3c055Sdan }
238273b3c055Sdan 
238377fc1d5bSdan /*
238477fc1d5bSdan ** Formulate and prepare a SELECT statement to retrieve a row from table
238577fc1d5bSdan ** zTab in database zDb based on its primary key. i.e.
238677fc1d5bSdan **
238777fc1d5bSdan **   SELECT * FROM zDb.zTab WHERE pk1 = ? AND pk2 = ? AND ...
238877fc1d5bSdan */
sessionSelectStmt(sqlite3 * db,const char * zDb,const char * zTab,int nCol,const char ** azCol,u8 * abPK,sqlite3_stmt ** ppStmt)2389e8d5648eSdan static int sessionSelectStmt(
2390e8d5648eSdan   sqlite3 *db,                    /* Database handle */
2391d7fb7d24Sdan   const char *zDb,                /* Database name */
2392e8d5648eSdan   const char *zTab,               /* Table name */
239377fc1d5bSdan   int nCol,                       /* Number of columns in table */
239477fc1d5bSdan   const char **azCol,             /* Names of table columns */
239577fc1d5bSdan   u8 *abPK,                       /* PRIMARY KEY  array */
239677fc1d5bSdan   sqlite3_stmt **ppStmt           /* OUT: Prepared SELECT statement */
2397d5f0767cSdan ){
2398e8d5648eSdan   int rc = SQLITE_OK;
23993739f298Sdan   char *zSql = 0;
24003739f298Sdan   int nSql = -1;
24013739f298Sdan 
24023739f298Sdan   if( 0==sqlite3_stricmp("sqlite_stat1", zTab) ){
24033739f298Sdan     zSql = sqlite3_mprintf(
24043739f298Sdan         "SELECT tbl, ?2, stat FROM %Q.sqlite_stat1 WHERE tbl IS ?1 AND "
24053739f298Sdan         "idx IS (CASE WHEN ?2=X'' THEN NULL ELSE ?2 END)", zDb
24063739f298Sdan     );
2407bd45374cSdan     if( zSql==0 ) rc = SQLITE_NOMEM;
24083739f298Sdan   }else{
2409d5f0767cSdan     int i;
2410e8d5648eSdan     const char *zSep = "";
2411e8d5648eSdan     SessionBuffer buf = {0, 0, 0};
2412d5f0767cSdan 
2413e8d5648eSdan     sessionAppendStr(&buf, "SELECT * FROM ", &rc);
2414d7fb7d24Sdan     sessionAppendIdent(&buf, zDb, &rc);
2415d7fb7d24Sdan     sessionAppendStr(&buf, ".", &rc);
2416e8d5648eSdan     sessionAppendIdent(&buf, zTab, &rc);
2417e8d5648eSdan     sessionAppendStr(&buf, " WHERE ", &rc);
2418e8d5648eSdan     for(i=0; i<nCol; i++){
2419e8d5648eSdan       if( abPK[i] ){
2420e8d5648eSdan         sessionAppendStr(&buf, zSep, &rc);
2421e8d5648eSdan         sessionAppendIdent(&buf, azCol[i], &rc);
24223739f298Sdan         sessionAppendStr(&buf, " IS ?", &rc);
2423e8d5648eSdan         sessionAppendInteger(&buf, i+1, &rc);
2424e8d5648eSdan         zSep = " AND ";
2425d5f0767cSdan       }
2426d5f0767cSdan     }
24273739f298Sdan     zSql = (char*)buf.aBuf;
24283739f298Sdan     nSql = buf.nBuf;
2429d5f0767cSdan   }
24303739f298Sdan 
24313739f298Sdan   if( rc==SQLITE_OK ){
24323739f298Sdan     rc = sqlite3_prepare_v2(db, zSql, nSql, ppStmt, 0);
24333739f298Sdan   }
24343739f298Sdan   sqlite3_free(zSql);
2435e8d5648eSdan   return rc;
2436d5f0767cSdan }
2437d5f0767cSdan 
243877fc1d5bSdan /*
243977fc1d5bSdan ** Bind the PRIMARY KEY values from the change passed in argument pChange
244077fc1d5bSdan ** to the SELECT statement passed as the first argument. The SELECT statement
244177fc1d5bSdan ** is as prepared by function sessionSelectStmt().
244277fc1d5bSdan **
244377fc1d5bSdan ** Return SQLITE_OK if all PK values are successfully bound, or an SQLite
244477fc1d5bSdan ** error code (e.g. SQLITE_NOMEM) otherwise.
244577fc1d5bSdan */
sessionSelectBind(sqlite3_stmt * pSelect,int nCol,u8 * abPK,SessionChange * pChange)2446e8d5648eSdan static int sessionSelectBind(
244777fc1d5bSdan   sqlite3_stmt *pSelect,          /* SELECT from sessionSelectStmt() */
244877fc1d5bSdan   int nCol,                       /* Number of columns in table */
244977fc1d5bSdan   u8 *abPK,                       /* PRIMARY KEY array */
245077fc1d5bSdan   SessionChange *pChange          /* Change structure */
2451e8d5648eSdan ){
2452e8d5648eSdan   int i;
2453e8d5648eSdan   int rc = SQLITE_OK;
2454e5754eecSdan   u8 *a = pChange->aRecord;
2455d5f0767cSdan 
2456e8d5648eSdan   for(i=0; i<nCol && rc==SQLITE_OK; i++){
2457e8d5648eSdan     int eType = *a++;
2458e8d5648eSdan 
2459e8d5648eSdan     switch( eType ){
246080fe2d93Sdan       case 0:
2461e8d5648eSdan       case SQLITE_NULL:
24621611e5a3Sdan         assert( abPK[i]==0 );
2463e8d5648eSdan         break;
2464e8d5648eSdan 
2465e8d5648eSdan       case SQLITE_INTEGER: {
2466e8d5648eSdan         if( abPK[i] ){
2467e8d5648eSdan           i64 iVal = sessionGetI64(a);
2468e8d5648eSdan           rc = sqlite3_bind_int64(pSelect, i+1, iVal);
2469e8d5648eSdan         }
2470e8d5648eSdan         a += 8;
2471e8d5648eSdan         break;
2472d5f0767cSdan       }
2473296c7658Sdan 
2474e8d5648eSdan       case SQLITE_FLOAT: {
2475e8d5648eSdan         if( abPK[i] ){
2476e8d5648eSdan           double rVal;
2477e8d5648eSdan           i64 iVal = sessionGetI64(a);
2478e8d5648eSdan           memcpy(&rVal, &iVal, 8);
24794e895da1Sdan           rc = sqlite3_bind_double(pSelect, i+1, rVal);
2480d5f0767cSdan         }
2481e8d5648eSdan         a += 8;
2482e8d5648eSdan         break;
2483e8d5648eSdan       }
2484e8d5648eSdan 
2485e8d5648eSdan       case SQLITE_TEXT: {
2486e8d5648eSdan         int n;
2487e8d5648eSdan         a += sessionVarintGet(a, &n);
2488e8d5648eSdan         if( abPK[i] ){
2489e8d5648eSdan           rc = sqlite3_bind_text(pSelect, i+1, (char *)a, n, SQLITE_TRANSIENT);
2490e8d5648eSdan         }
2491e8d5648eSdan         a += n;
2492e8d5648eSdan         break;
2493e8d5648eSdan       }
2494e8d5648eSdan 
2495e5754eecSdan       default: {
2496e8d5648eSdan         int n;
2497e5754eecSdan         assert( eType==SQLITE_BLOB );
2498e8d5648eSdan         a += sessionVarintGet(a, &n);
2499e8d5648eSdan         if( abPK[i] ){
2500e8d5648eSdan           rc = sqlite3_bind_blob(pSelect, i+1, a, n, SQLITE_TRANSIENT);
2501e8d5648eSdan         }
2502e8d5648eSdan         a += n;
2503e8d5648eSdan         break;
2504e8d5648eSdan       }
2505e8d5648eSdan     }
2506e8d5648eSdan   }
2507e8d5648eSdan 
2508d5f0767cSdan   return rc;
25094fccf43aSdan }
25104fccf43aSdan 
251177fc1d5bSdan /*
251277fc1d5bSdan ** This function is a no-op if *pRc is set to other than SQLITE_OK when it
251377fc1d5bSdan ** is called. Otherwise, append a serialized table header (part of the binary
251477fc1d5bSdan ** changeset format) to buffer *pBuf. If an error occurs, set *pRc to an
251577fc1d5bSdan ** SQLite error code before returning.
251677fc1d5bSdan */
sessionAppendTableHdr(SessionBuffer * pBuf,int bPatchset,SessionTable * pTab,int * pRc)25175d607a6eSdan static void sessionAppendTableHdr(
2518a71d2371Sdan   SessionBuffer *pBuf,            /* Append header to this buffer */
2519a71d2371Sdan   int bPatchset,                  /* Use the patchset format if true */
2520a71d2371Sdan   SessionTable *pTab,             /* Table object to append header for */
2521a71d2371Sdan   int *pRc                        /* IN/OUT: Error code */
25225d607a6eSdan ){
25235d607a6eSdan   /* Write a table header */
252473b3c055Sdan   sessionAppendByte(pBuf, (bPatchset ? 'P' : 'T'), pRc);
25255d607a6eSdan   sessionAppendVarint(pBuf, pTab->nCol, pRc);
25265d607a6eSdan   sessionAppendBlob(pBuf, pTab->abPK, pTab->nCol, pRc);
25274f528042Sdan   sessionAppendBlob(pBuf, (u8 *)pTab->zName, (int)strlen(pTab->zName)+1, pRc);
25285d607a6eSdan }
25295d607a6eSdan 
2530a71d2371Sdan /*
2531a71d2371Sdan ** Generate either a changeset (if argument bPatchset is zero) or a patchset
2532a71d2371Sdan ** (if it is non-zero) based on the current contents of the session object
2533a71d2371Sdan ** passed as the first argument.
2534a71d2371Sdan **
2535a71d2371Sdan ** If no error occurs, SQLITE_OK is returned and the new changeset/patchset
2536a71d2371Sdan ** stored in output variables *pnChangeset and *ppChangeset. Or, if an error
2537a71d2371Sdan ** occurs, an SQLite error code is returned and both output variables set
2538a71d2371Sdan ** to 0.
2539a71d2371Sdan */
sessionGenerateChangeset(sqlite3_session * pSession,int bPatchset,int (* xOutput)(void * pOut,const void * pData,int nData),void * pOut,int * pnChangeset,void ** ppChangeset)2540adf3bf58Sdrh static int sessionGenerateChangeset(
25414fccf43aSdan   sqlite3_session *pSession,      /* Session object */
254273b3c055Sdan   int bPatchset,                  /* True for patchset, false for changeset */
2543ef7a6304Sdan   int (*xOutput)(void *pOut, const void *pData, int nData),
2544ef7a6304Sdan   void *pOut,                     /* First argument for xOutput */
25454fccf43aSdan   int *pnChangeset,               /* OUT: Size of buffer at *ppChangeset */
25464fccf43aSdan   void **ppChangeset              /* OUT: Buffer containing changeset */
25474fccf43aSdan ){
2548296c7658Sdan   sqlite3 *db = pSession->db;     /* Source database handle */
2549296c7658Sdan   SessionTable *pTab;             /* Used to iterate through attached tables */
2550296c7658Sdan   SessionBuffer buf = {0,0,0};    /* Buffer in which to accumlate changeset */
2551296c7658Sdan   int rc;                         /* Return code */
25524fccf43aSdan 
2553ef7a6304Sdan   assert( xOutput==0 || (pnChangeset==0 && ppChangeset==0) );
2554*7a3b4451Sdrh   assert( xOutput!=0 || (pnChangeset!=0 && ppChangeset!=0) );
2555ef7a6304Sdan 
2556296c7658Sdan   /* Zero the output variables in case an error occurs. If this session
2557296c7658Sdan   ** object is already in the error state (sqlite3_session.rc != SQLITE_OK),
2558296c7658Sdan   ** this call will be a no-op.  */
2559ef7a6304Sdan   if( xOutput==0 ){
2560*7a3b4451Sdrh     assert( pnChangeset!=0  && ppChangeset!=0 );
25614fccf43aSdan     *pnChangeset = 0;
25624fccf43aSdan     *ppChangeset = 0;
2563ef7a6304Sdan   }
2564e5754eecSdan 
2565e5754eecSdan   if( pSession->rc ) return pSession->rc;
2566e5754eecSdan   rc = sqlite3_exec(pSession->db, "SAVEPOINT changeset", 0, 0, 0);
2567e5754eecSdan   if( rc!=SQLITE_OK ) return rc;
2568e5754eecSdan 
2569e5754eecSdan   sqlite3_mutex_enter(sqlite3_db_mutex(db));
25704fccf43aSdan 
25714fccf43aSdan   for(pTab=pSession->pTable; rc==SQLITE_OK && pTab; pTab=pTab->pNext){
25724fccf43aSdan     if( pTab->nEntry ){
2573d7fb7d24Sdan       const char *zName = pTab->zName;
2574*7a3b4451Sdrh       int nCol = 0;               /* Number of columns in table */
2575*7a3b4451Sdrh       u8 *abPK = 0;               /* Primary key array */
2576a9605b91Sdan       const char **azCol = 0;     /* Table columns */
25771f34f8ccSdan       int i;                      /* Used to iterate through hash buckets */
25781f34f8ccSdan       sqlite3_stmt *pSel = 0;     /* SELECT statement to query table pTab */
25791f34f8ccSdan       int nRewind = buf.nBuf;     /* Initial size of write buffer */
25801f34f8ccSdan       int nNoop;                  /* Size of buffer after writing tbl header */
25814fccf43aSdan 
2582a9605b91Sdan       /* Check the table schema is still Ok. */
25830cb735b9Sdan       rc = sessionTableInfo(0, db, pSession->zDb, zName, &nCol, 0,&azCol,&abPK);
2584a9605b91Sdan       if( !rc && (pTab->nCol!=nCol || memcmp(abPK, pTab->abPK, nCol)) ){
2585a9605b91Sdan         rc = SQLITE_SCHEMA;
2586a9605b91Sdan       }
2587a9605b91Sdan 
25884fccf43aSdan       /* Write a table header */
258973b3c055Sdan       sessionAppendTableHdr(&buf, bPatchset, pTab, &rc);
25904fccf43aSdan 
25914fccf43aSdan       /* Build and compile a statement to execute: */
25924fccf43aSdan       if( rc==SQLITE_OK ){
2593d7fb7d24Sdan         rc = sessionSelectStmt(
2594a9605b91Sdan             db, pSession->zDb, zName, nCol, azCol, abPK, &pSel);
25954fccf43aSdan       }
25964fccf43aSdan 
25971f34f8ccSdan       nNoop = buf.nBuf;
259812ca0b56Sdan       for(i=0; i<pTab->nChange && rc==SQLITE_OK; i++){
2599e8d5648eSdan         SessionChange *p;         /* Used to iterate through changes */
2600e8d5648eSdan 
26014fccf43aSdan         for(p=pTab->apChange[i]; rc==SQLITE_OK && p; p=p->pNext){
2602e5754eecSdan           rc = sessionSelectBind(pSel, nCol, abPK, p);
260380fe2d93Sdan           if( rc!=SQLITE_OK ) continue;
26041f34f8ccSdan           if( sqlite3_step(pSel)==SQLITE_ROW ){
2605798693b2Sdan             if( p->op==SQLITE_INSERT ){
26064fccf43aSdan               int iCol;
26074fccf43aSdan               sessionAppendByte(&buf, SQLITE_INSERT, &rc);
2608b4480e94Sdan               sessionAppendByte(&buf, p->bIndirect, &rc);
2609e8d5648eSdan               for(iCol=0; iCol<nCol; iCol++){
26101f34f8ccSdan                 sessionAppendCol(&buf, pSel, iCol, &rc);
26114fccf43aSdan               }
2612e8d5648eSdan             }else{
2613*7a3b4451Sdrh               assert( abPK!=0 );  /* Because sessionSelectStmt() returned ok */
261473b3c055Sdan               rc = sessionAppendUpdate(&buf, bPatchset, pSel, p, abPK);
26154fccf43aSdan             }
2616798693b2Sdan           }else if( p->op!=SQLITE_INSERT ){
2617a71d2371Sdan             rc = sessionAppendDelete(&buf, bPatchset, p, nCol, abPK);
26184fccf43aSdan           }
261912ca0b56Sdan           if( rc==SQLITE_OK ){
26201f34f8ccSdan             rc = sqlite3_reset(pSel);
26214fccf43aSdan           }
2622ef7a6304Sdan 
26231f48e67dSdan           /* If the buffer is now larger than sessions_strm_chunk_size, pass
2624ef7a6304Sdan           ** its contents to the xOutput() callback. */
2625ef7a6304Sdan           if( xOutput
2626ef7a6304Sdan            && rc==SQLITE_OK
2627ef7a6304Sdan            && buf.nBuf>nNoop
26281f48e67dSdan            && buf.nBuf>sessions_strm_chunk_size
2629ef7a6304Sdan           ){
2630ef7a6304Sdan             rc = xOutput(pOut, (void*)buf.aBuf, buf.nBuf);
2631ef7a6304Sdan             nNoop = -1;
2632ef7a6304Sdan             buf.nBuf = 0;
2633ef7a6304Sdan           }
2634ef7a6304Sdan 
26354fccf43aSdan         }
2636e8d5648eSdan       }
26374fccf43aSdan 
26381f34f8ccSdan       sqlite3_finalize(pSel);
26391f34f8ccSdan       if( buf.nBuf==nNoop ){
26404fccf43aSdan         buf.nBuf = nRewind;
26414fccf43aSdan       }
2642cfdbde21Sdrh       sqlite3_free((char*)azCol);  /* cast works around VC++ bug */
26434fccf43aSdan     }
26444fccf43aSdan   }
26454fccf43aSdan 
26464fccf43aSdan   if( rc==SQLITE_OK ){
2647ef7a6304Sdan     if( xOutput==0 ){
26484fccf43aSdan       *pnChangeset = buf.nBuf;
26494fccf43aSdan       *ppChangeset = buf.aBuf;
2650ef7a6304Sdan       buf.aBuf = 0;
2651ef7a6304Sdan     }else if( buf.nBuf>0 ){
2652ef7a6304Sdan       rc = xOutput(pOut, (void*)buf.aBuf, buf.nBuf);
2653ef7a6304Sdan     }
26544fccf43aSdan   }
26554c220252Sdan 
2656ef7a6304Sdan   sqlite3_free(buf.aBuf);
2657e5754eecSdan   sqlite3_exec(db, "RELEASE changeset", 0, 0, 0);
26584c220252Sdan   sqlite3_mutex_leave(sqlite3_db_mutex(db));
26594fccf43aSdan   return rc;
26604fccf43aSdan }
26614fccf43aSdan 
2662296c7658Sdan /*
266373b3c055Sdan ** Obtain a changeset object containing all changes recorded by the
266473b3c055Sdan ** session object passed as the first argument.
266573b3c055Sdan **
266673b3c055Sdan ** It is the responsibility of the caller to eventually free the buffer
266773b3c055Sdan ** using sqlite3_free().
266873b3c055Sdan */
sqlite3session_changeset(sqlite3_session * pSession,int * pnChangeset,void ** ppChangeset)266973b3c055Sdan int sqlite3session_changeset(
267073b3c055Sdan   sqlite3_session *pSession,      /* Session object */
267173b3c055Sdan   int *pnChangeset,               /* OUT: Size of buffer at *ppChangeset */
267273b3c055Sdan   void **ppChangeset              /* OUT: Buffer containing changeset */
267373b3c055Sdan ){
2674*7a3b4451Sdrh   int rc;
2675*7a3b4451Sdrh 
2676*7a3b4451Sdrh   if( pnChangeset==0 || ppChangeset==0 ) return SQLITE_MISUSE;
2677*7a3b4451Sdrh   rc = sessionGenerateChangeset(pSession, 0, 0, 0, pnChangeset,ppChangeset);
26786d29a4feSdan   assert( rc || pnChangeset==0
26796d29a4feSdan        || pSession->bEnableSize==0 || *pnChangeset<=pSession->nMaxChangesetSize
26806d29a4feSdan   );
2681a23a873fSdan   return rc;
2682ef7a6304Sdan }
2683ef7a6304Sdan 
2684ef7a6304Sdan /*
2685ef7a6304Sdan ** Streaming version of sqlite3session_changeset().
2686ef7a6304Sdan */
sqlite3session_changeset_strm(sqlite3_session * pSession,int (* xOutput)(void * pOut,const void * pData,int nData),void * pOut)2687f1a08ad8Sdrh int sqlite3session_changeset_strm(
2688ef7a6304Sdan   sqlite3_session *pSession,
2689ef7a6304Sdan   int (*xOutput)(void *pOut, const void *pData, int nData),
2690ef7a6304Sdan   void *pOut
2691ef7a6304Sdan ){
2692*7a3b4451Sdrh   if( xOutput==0 ) return SQLITE_MISUSE;
2693ef7a6304Sdan   return sessionGenerateChangeset(pSession, 0, xOutput, pOut, 0, 0);
2694ef7a6304Sdan }
2695ef7a6304Sdan 
2696ef7a6304Sdan /*
2697ef7a6304Sdan ** Streaming version of sqlite3session_patchset().
2698ef7a6304Sdan */
sqlite3session_patchset_strm(sqlite3_session * pSession,int (* xOutput)(void * pOut,const void * pData,int nData),void * pOut)2699f1a08ad8Sdrh int sqlite3session_patchset_strm(
2700ef7a6304Sdan   sqlite3_session *pSession,
2701ef7a6304Sdan   int (*xOutput)(void *pOut, const void *pData, int nData),
2702ef7a6304Sdan   void *pOut
2703ef7a6304Sdan ){
2704*7a3b4451Sdrh   if( xOutput==0 ) return SQLITE_MISUSE;
2705ef7a6304Sdan   return sessionGenerateChangeset(pSession, 1, xOutput, pOut, 0, 0);
270673b3c055Sdan }
270773b3c055Sdan 
270873b3c055Sdan /*
270973b3c055Sdan ** Obtain a patchset object containing all changes recorded by the
271073b3c055Sdan ** session object passed as the first argument.
271173b3c055Sdan **
271273b3c055Sdan ** It is the responsibility of the caller to eventually free the buffer
271373b3c055Sdan ** using sqlite3_free().
271473b3c055Sdan */
sqlite3session_patchset(sqlite3_session * pSession,int * pnPatchset,void ** ppPatchset)271573b3c055Sdan int sqlite3session_patchset(
271673b3c055Sdan   sqlite3_session *pSession,      /* Session object */
271773b3c055Sdan   int *pnPatchset,                /* OUT: Size of buffer at *ppChangeset */
271873b3c055Sdan   void **ppPatchset               /* OUT: Buffer containing changeset */
271973b3c055Sdan ){
2720*7a3b4451Sdrh   if( pnPatchset==0 || ppPatchset==0 ) return SQLITE_MISUSE;
2721ef7a6304Sdan   return sessionGenerateChangeset(pSession, 1, 0, 0, pnPatchset, ppPatchset);
272273b3c055Sdan }
272373b3c055Sdan 
272473b3c055Sdan /*
2725296c7658Sdan ** Enable or disable the session object passed as the first argument.
2726296c7658Sdan */
sqlite3session_enable(sqlite3_session * pSession,int bEnable)27274fccf43aSdan int sqlite3session_enable(sqlite3_session *pSession, int bEnable){
27284c220252Sdan   int ret;
27294c220252Sdan   sqlite3_mutex_enter(sqlite3_db_mutex(pSession->db));
2730296c7658Sdan   if( bEnable>=0 ){
2731296c7658Sdan     pSession->bEnable = bEnable;
27324fccf43aSdan   }
27334c220252Sdan   ret = pSession->bEnable;
27344c220252Sdan   sqlite3_mutex_leave(sqlite3_db_mutex(pSession->db));
27354c220252Sdan   return ret;
2736296c7658Sdan }
27374fccf43aSdan 
27384fccf43aSdan /*
2739b4480e94Sdan ** Enable or disable the session object passed as the first argument.
2740b4480e94Sdan */
sqlite3session_indirect(sqlite3_session * pSession,int bIndirect)2741b4480e94Sdan int sqlite3session_indirect(sqlite3_session *pSession, int bIndirect){
2742b4480e94Sdan   int ret;
2743b4480e94Sdan   sqlite3_mutex_enter(sqlite3_db_mutex(pSession->db));
2744b4480e94Sdan   if( bIndirect>=0 ){
2745b4480e94Sdan     pSession->bIndirect = bIndirect;
2746b4480e94Sdan   }
2747b4480e94Sdan   ret = pSession->bIndirect;
2748b4480e94Sdan   sqlite3_mutex_leave(sqlite3_db_mutex(pSession->db));
2749b4480e94Sdan   return ret;
2750b4480e94Sdan }
2751b4480e94Sdan 
2752b4480e94Sdan /*
2753b69ec348Sdan ** Return true if there have been no changes to monitored tables recorded
2754b69ec348Sdan ** by the session object passed as the only argument.
2755b69ec348Sdan */
sqlite3session_isempty(sqlite3_session * pSession)2756b69ec348Sdan int sqlite3session_isempty(sqlite3_session *pSession){
2757b69ec348Sdan   int ret = 0;
2758b69ec348Sdan   SessionTable *pTab;
2759b69ec348Sdan 
2760b69ec348Sdan   sqlite3_mutex_enter(sqlite3_db_mutex(pSession->db));
2761b69ec348Sdan   for(pTab=pSession->pTable; pTab && ret==0; pTab=pTab->pNext){
2762b69ec348Sdan     ret = (pTab->nEntry>0);
2763b69ec348Sdan   }
2764b69ec348Sdan   sqlite3_mutex_leave(sqlite3_db_mutex(pSession->db));
2765b69ec348Sdan 
2766ff530326Sdan   return (ret==0);
2767b69ec348Sdan }
2768b69ec348Sdan 
2769b69ec348Sdan /*
27700cb735b9Sdan ** Return the amount of heap memory in use.
27710cb735b9Sdan */
sqlite3session_memory_used(sqlite3_session * pSession)27720cb735b9Sdan sqlite3_int64 sqlite3session_memory_used(sqlite3_session *pSession){
27730cb735b9Sdan   return pSession->nMalloc;
27740cb735b9Sdan }
27750cb735b9Sdan 
27760cb735b9Sdan /*
27776d29a4feSdan ** Configure the session object passed as the first argument.
27786d29a4feSdan */
sqlite3session_object_config(sqlite3_session * pSession,int op,void * pArg)27796d29a4feSdan int sqlite3session_object_config(sqlite3_session *pSession, int op, void *pArg){
27806d29a4feSdan   int rc = SQLITE_OK;
27816d29a4feSdan   switch( op ){
27826d29a4feSdan     case SQLITE_SESSION_OBJCONFIG_SIZE: {
27836d29a4feSdan       int iArg = *(int*)pArg;
27846d29a4feSdan       if( iArg>=0 ){
27856d29a4feSdan         if( pSession->pTable ){
27866d29a4feSdan           rc = SQLITE_MISUSE;
27876d29a4feSdan         }else{
27886d29a4feSdan           pSession->bEnableSize = (iArg!=0);
27896d29a4feSdan         }
27906d29a4feSdan       }
27916d29a4feSdan       *(int*)pArg = pSession->bEnableSize;
27926d29a4feSdan       break;
27936d29a4feSdan     }
27946d29a4feSdan 
27956d29a4feSdan     default:
27966d29a4feSdan       rc = SQLITE_MISUSE;
27976d29a4feSdan   }
27986d29a4feSdan 
27996d29a4feSdan   return rc;
28006d29a4feSdan }
28016d29a4feSdan 
28026d29a4feSdan /*
2803a23a873fSdan ** Return the maximum size of sqlite3session_changeset() output.
2804a23a873fSdan */
sqlite3session_changeset_size(sqlite3_session * pSession)2805a23a873fSdan sqlite3_int64 sqlite3session_changeset_size(sqlite3_session *pSession){
2806a23a873fSdan   return pSession->nMaxChangesetSize;
2807a23a873fSdan }
2808a23a873fSdan 
2809a23a873fSdan /*
2810f1a08ad8Sdrh ** Do the work for either sqlite3changeset_start() or start_strm().
28114fccf43aSdan */
sessionChangesetStart(sqlite3_changeset_iter ** pp,int (* xInput)(void * pIn,void * pData,int * pnData),void * pIn,int nChangeset,void * pChangeset,int bInvert,int bSkipEmpty)2812adf3bf58Sdrh static int sessionChangesetStart(
2813296c7658Sdan   sqlite3_changeset_iter **pp,    /* OUT: Changeset iterator handle */
2814ef7a6304Sdan   int (*xInput)(void *pIn, void *pData, int *pnData),
2815ef7a6304Sdan   void *pIn,
2816296c7658Sdan   int nChangeset,                 /* Size of buffer pChangeset in bytes */
281744748f27Sdan   void *pChangeset,               /* Pointer to buffer containing changeset */
28181e25d20cSdan   int bInvert,                    /* True to invert changeset */
28191e25d20cSdan   int bSkipEmpty                  /* True to skip empty UPDATE changes */
28204fccf43aSdan ){
28214fccf43aSdan   sqlite3_changeset_iter *pRet;   /* Iterator to return */
28224fccf43aSdan   int nByte;                      /* Number of bytes to allocate for iterator */
28234fccf43aSdan 
2824ef7a6304Sdan   assert( xInput==0 || (pChangeset==0 && nChangeset==0) );
2825ef7a6304Sdan 
2826296c7658Sdan   /* Zero the output variable in case an error occurs. */
2827296c7658Sdan   *pp = 0;
28284fccf43aSdan 
2829296c7658Sdan   /* Allocate and initialize the iterator structure. */
28304fccf43aSdan   nByte = sizeof(sqlite3_changeset_iter);
28314fccf43aSdan   pRet = (sqlite3_changeset_iter *)sqlite3_malloc(nByte);
28324fccf43aSdan   if( !pRet ) return SQLITE_NOMEM;
28334fccf43aSdan   memset(pRet, 0, sizeof(sqlite3_changeset_iter));
28344757c658Sdan   pRet->in.aData = (u8 *)pChangeset;
28354757c658Sdan   pRet->in.nData = nChangeset;
2836ef7a6304Sdan   pRet->in.xInput = xInput;
2837ef7a6304Sdan   pRet->in.pIn = pIn;
2838ef7a6304Sdan   pRet->in.bEof = (xInput ? 0 : 1);
283944748f27Sdan   pRet->bInvert = bInvert;
28401e25d20cSdan   pRet->bSkipEmpty = bSkipEmpty;
28414fccf43aSdan 
2842296c7658Sdan   /* Populate the output variable and return success. */
2843296c7658Sdan   *pp = pRet;
28444fccf43aSdan   return SQLITE_OK;
28454fccf43aSdan }
28464fccf43aSdan 
2847296c7658Sdan /*
2848ef7a6304Sdan ** Create an iterator used to iterate through the contents of a changeset.
2849ef7a6304Sdan */
sqlite3changeset_start(sqlite3_changeset_iter ** pp,int nChangeset,void * pChangeset)2850ef7a6304Sdan int sqlite3changeset_start(
2851ef7a6304Sdan   sqlite3_changeset_iter **pp,    /* OUT: Changeset iterator handle */
2852ef7a6304Sdan   int nChangeset,                 /* Size of buffer pChangeset in bytes */
2853ef7a6304Sdan   void *pChangeset                /* Pointer to buffer containing changeset */
2854ef7a6304Sdan ){
28551e25d20cSdan   return sessionChangesetStart(pp, 0, 0, nChangeset, pChangeset, 0, 0);
2856ef7a6304Sdan }
sqlite3changeset_start_v2(sqlite3_changeset_iter ** pp,int nChangeset,void * pChangeset,int flags)285746de0728Sdan int sqlite3changeset_start_v2(
285846de0728Sdan   sqlite3_changeset_iter **pp,    /* OUT: Changeset iterator handle */
285946de0728Sdan   int nChangeset,                 /* Size of buffer pChangeset in bytes */
286046de0728Sdan   void *pChangeset,               /* Pointer to buffer containing changeset */
286146de0728Sdan   int flags
286246de0728Sdan ){
286346de0728Sdan   int bInvert = !!(flags & SQLITE_CHANGESETSTART_INVERT);
28641e25d20cSdan   return sessionChangesetStart(pp, 0, 0, nChangeset, pChangeset, bInvert, 0);
286546de0728Sdan }
2866ef7a6304Sdan 
2867ef7a6304Sdan /*
2868ef7a6304Sdan ** Streaming version of sqlite3changeset_start().
2869ef7a6304Sdan */
sqlite3changeset_start_strm(sqlite3_changeset_iter ** pp,int (* xInput)(void * pIn,void * pData,int * pnData),void * pIn)2870f1a08ad8Sdrh int sqlite3changeset_start_strm(
2871ef7a6304Sdan   sqlite3_changeset_iter **pp,    /* OUT: Changeset iterator handle */
2872ef7a6304Sdan   int (*xInput)(void *pIn, void *pData, int *pnData),
2873ef7a6304Sdan   void *pIn
2874ef7a6304Sdan ){
28751e25d20cSdan   return sessionChangesetStart(pp, xInput, pIn, 0, 0, 0, 0);
2876ef7a6304Sdan }
sqlite3changeset_start_v2_strm(sqlite3_changeset_iter ** pp,int (* xInput)(void * pIn,void * pData,int * pnData),void * pIn,int flags)287746de0728Sdan int sqlite3changeset_start_v2_strm(
287846de0728Sdan   sqlite3_changeset_iter **pp,    /* OUT: Changeset iterator handle */
287946de0728Sdan   int (*xInput)(void *pIn, void *pData, int *pnData),
288046de0728Sdan   void *pIn,
288146de0728Sdan   int flags
288246de0728Sdan ){
288346de0728Sdan   int bInvert = !!(flags & SQLITE_CHANGESETSTART_INVERT);
28841e25d20cSdan   return sessionChangesetStart(pp, xInput, pIn, 0, 0, bInvert, 0);
288546de0728Sdan }
2886ef7a6304Sdan 
2887ef7a6304Sdan /*
2888d9151526Sdan ** If the SessionInput object passed as the only argument is a streaming
2889d9151526Sdan ** object and the buffer is full, discard some data to free up space.
2890d9151526Sdan */
sessionDiscardData(SessionInput * pIn)2891d9151526Sdan static void sessionDiscardData(SessionInput *pIn){
28921f48e67dSdan   if( pIn->xInput && pIn->iNext>=sessions_strm_chunk_size ){
2893d9151526Sdan     int nMove = pIn->buf.nBuf - pIn->iNext;
2894d9151526Sdan     assert( nMove>=0 );
2895d9151526Sdan     if( nMove>0 ){
2896d9151526Sdan       memmove(pIn->buf.aBuf, &pIn->buf.aBuf[pIn->iNext], nMove);
2897d9151526Sdan     }
2898d9151526Sdan     pIn->buf.nBuf -= pIn->iNext;
2899d9151526Sdan     pIn->iNext = 0;
2900d9151526Sdan     pIn->nData = pIn->buf.nBuf;
2901d9151526Sdan   }
2902d9151526Sdan }
2903d9151526Sdan 
2904d9151526Sdan /*
2905ef7a6304Sdan ** Ensure that there are at least nByte bytes available in the buffer. Or,
2906ef7a6304Sdan ** if there are not nByte bytes remaining in the input, that all available
2907ef7a6304Sdan ** data is in the buffer.
2908ef7a6304Sdan **
2909ef7a6304Sdan ** Return an SQLite error code if an error occurs, or SQLITE_OK otherwise.
2910ef7a6304Sdan */
sessionInputBuffer(SessionInput * pIn,int nByte)29114757c658Sdan static int sessionInputBuffer(SessionInput *pIn, int nByte){
2912ef7a6304Sdan   int rc = SQLITE_OK;
29134757c658Sdan   if( pIn->xInput ){
29144757c658Sdan     while( !pIn->bEof && (pIn->iNext+nByte)>=pIn->nData && rc==SQLITE_OK ){
29151f48e67dSdan       int nNew = sessions_strm_chunk_size;
29164757c658Sdan 
2917d9151526Sdan       if( pIn->bNoDiscard==0 ) sessionDiscardData(pIn);
29184757c658Sdan       if( SQLITE_OK==sessionBufferGrow(&pIn->buf, nNew, &rc) ){
29194757c658Sdan         rc = pIn->xInput(pIn->pIn, &pIn->buf.aBuf[pIn->buf.nBuf], &nNew);
29204757c658Sdan         if( nNew==0 ){
29214757c658Sdan           pIn->bEof = 1;
29224757c658Sdan         }else{
29234757c658Sdan           pIn->buf.nBuf += nNew;
29244757c658Sdan         }
29254757c658Sdan       }
29264757c658Sdan 
29274757c658Sdan       pIn->aData = pIn->buf.aBuf;
29284757c658Sdan       pIn->nData = pIn->buf.nBuf;
29294757c658Sdan     }
2930ef7a6304Sdan   }
2931ef7a6304Sdan   return rc;
2932ef7a6304Sdan }
2933ef7a6304Sdan 
2934ef7a6304Sdan /*
2935ef7a6304Sdan ** When this function is called, *ppRec points to the start of a record
2936ef7a6304Sdan ** that contains nCol values. This function advances the pointer *ppRec
2937ef7a6304Sdan ** until it points to the byte immediately following that record.
2938ef7a6304Sdan */
sessionSkipRecord(u8 ** ppRec,int nCol)2939ef7a6304Sdan static void sessionSkipRecord(
2940ef7a6304Sdan   u8 **ppRec,                     /* IN/OUT: Record pointer */
2941ef7a6304Sdan   int nCol                        /* Number of values in record */
2942ef7a6304Sdan ){
2943ef7a6304Sdan   u8 *aRec = *ppRec;
2944ef7a6304Sdan   int i;
2945ef7a6304Sdan   for(i=0; i<nCol; i++){
2946ef7a6304Sdan     int eType = *aRec++;
2947ef7a6304Sdan     if( eType==SQLITE_TEXT || eType==SQLITE_BLOB ){
2948ef7a6304Sdan       int nByte;
2949ef7a6304Sdan       aRec += sessionVarintGet((u8*)aRec, &nByte);
2950ef7a6304Sdan       aRec += nByte;
2951ef7a6304Sdan     }else if( eType==SQLITE_INTEGER || eType==SQLITE_FLOAT ){
2952ef7a6304Sdan       aRec += 8;
2953ef7a6304Sdan     }
2954ef7a6304Sdan   }
2955ef7a6304Sdan 
2956ef7a6304Sdan   *ppRec = aRec;
2957ef7a6304Sdan }
2958ef7a6304Sdan 
2959ef7a6304Sdan /*
29604757c658Sdan ** This function sets the value of the sqlite3_value object passed as the
29614757c658Sdan ** first argument to a copy of the string or blob held in the aData[]
29624757c658Sdan ** buffer. SQLITE_OK is returned if successful, or SQLITE_NOMEM if an OOM
29634757c658Sdan ** error occurs.
29644757c658Sdan */
sessionValueSetStr(sqlite3_value * pVal,u8 * aData,int nData,u8 enc)29654757c658Sdan static int sessionValueSetStr(
29664757c658Sdan   sqlite3_value *pVal,            /* Set the value of this object */
29674757c658Sdan   u8 *aData,                      /* Buffer containing string or blob data */
29684757c658Sdan   int nData,                      /* Size of buffer aData[] in bytes */
29694757c658Sdan   u8 enc                          /* String encoding (0 for blobs) */
29704757c658Sdan ){
297116228167Sdan   /* In theory this code could just pass SQLITE_TRANSIENT as the final
297216228167Sdan   ** argument to sqlite3ValueSetStr() and have the copy created
297316228167Sdan   ** automatically. But doing so makes it difficult to detect any OOM
297416228167Sdan   ** error. Hence the code to create the copy externally. */
29752d77d80aSdrh   u8 *aCopy = sqlite3_malloc64((sqlite3_int64)nData+1);
29764757c658Sdan   if( aCopy==0 ) return SQLITE_NOMEM;
29774757c658Sdan   memcpy(aCopy, aData, nData);
29784757c658Sdan   sqlite3ValueSetStr(pVal, nData, (char*)aCopy, enc, sqlite3_free);
29794757c658Sdan   return SQLITE_OK;
29804757c658Sdan }
29814757c658Sdan 
29824757c658Sdan /*
2983296c7658Sdan ** Deserialize a single record from a buffer in memory. See "RECORD FORMAT"
2984296c7658Sdan ** for details.
2985296c7658Sdan **
2986296c7658Sdan ** When this function is called, *paChange points to the start of the record
2987296c7658Sdan ** to deserialize. Assuming no error occurs, *paChange is set to point to
2988296c7658Sdan ** one byte after the end of the same record before this function returns.
2989a71d2371Sdan ** If the argument abPK is NULL, then the record contains nCol values. Or,
2990a71d2371Sdan ** if abPK is other than NULL, then the record contains only the PK fields
2991a71d2371Sdan ** (in other words, it is a patchset DELETE record).
2992296c7658Sdan **
2993296c7658Sdan ** If successful, each element of the apOut[] array (allocated by the caller)
2994296c7658Sdan ** is set to point to an sqlite3_value object containing the value read
2995296c7658Sdan ** from the corresponding position in the record. If that value is not
2996296c7658Sdan ** included in the record (i.e. because the record is part of an UPDATE change
2997296c7658Sdan ** and the field was not modified), the corresponding element of apOut[] is
2998296c7658Sdan ** set to NULL.
2999296c7658Sdan **
3000296c7658Sdan ** It is the responsibility of the caller to free all sqlite_value structures
3001296c7658Sdan ** using sqlite3_free().
3002296c7658Sdan **
3003296c7658Sdan ** If an error occurs, an SQLite error code (e.g. SQLITE_NOMEM) is returned.
3004296c7658Sdan ** The apOut[] array may have been partially populated in this case.
3005296c7658Sdan */
sessionReadRecord(SessionInput * pIn,int nCol,u8 * abPK,sqlite3_value ** apOut,int * pbEmpty)30064fccf43aSdan static int sessionReadRecord(
3007ef7a6304Sdan   SessionInput *pIn,              /* Input data */
30084fccf43aSdan   int nCol,                       /* Number of values in record */
300973b3c055Sdan   u8 *abPK,                       /* Array of primary key flags, or NULL */
30101e25d20cSdan   sqlite3_value **apOut,          /* Write values to this array */
30111e25d20cSdan   int *pbEmpty
30124fccf43aSdan ){
3013296c7658Sdan   int i;                          /* Used to iterate through columns */
3014ef7a6304Sdan   int rc = SQLITE_OK;
30154fccf43aSdan 
30161e25d20cSdan   assert( pbEmpty==0 || *pbEmpty==0 );
30171e25d20cSdan   if( pbEmpty ) *pbEmpty = 1;
3018ef7a6304Sdan   for(i=0; i<nCol && rc==SQLITE_OK; i++){
3019ef7a6304Sdan     int eType = 0;                /* Type of value (SQLITE_NULL, TEXT etc.) */
302073b3c055Sdan     if( abPK && abPK[i]==0 ) continue;
3021ef7a6304Sdan     rc = sessionInputBuffer(pIn, 9);
3022ef7a6304Sdan     if( rc==SQLITE_OK ){
3023e341ec69Sdan       if( pIn->iNext>=pIn->nData ){
3024e341ec69Sdan         rc = SQLITE_CORRUPT_BKPT;
3025e341ec69Sdan       }else{
30264757c658Sdan         eType = pIn->aData[pIn->iNext++];
3027e8fa8c96Sdan         assert( apOut[i]==0 );
30284fccf43aSdan         if( eType ){
30291e25d20cSdan           if( pbEmpty ) *pbEmpty = 0;
30304fccf43aSdan           apOut[i] = sqlite3ValueNew(0);
3031ef7a6304Sdan           if( !apOut[i] ) rc = SQLITE_NOMEM;
3032ef7a6304Sdan         }
3033dd8a4af8Sdan       }
3034e341ec69Sdan     }
30354fccf43aSdan 
3036ef7a6304Sdan     if( rc==SQLITE_OK ){
30374757c658Sdan       u8 *aVal = &pIn->aData[pIn->iNext];
30384fccf43aSdan       if( eType==SQLITE_TEXT || eType==SQLITE_BLOB ){
30394fccf43aSdan         int nByte;
3040ef7a6304Sdan         pIn->iNext += sessionVarintGet(aVal, &nByte);
3041ef7a6304Sdan         rc = sessionInputBuffer(pIn, nByte);
3042e8fa8c96Sdan         if( rc==SQLITE_OK ){
3043e341ec69Sdan           if( nByte<0 || nByte>pIn->nData-pIn->iNext ){
3044e341ec69Sdan             rc = SQLITE_CORRUPT_BKPT;
3045e341ec69Sdan           }else{
30466734007dSdan             u8 enc = (eType==SQLITE_TEXT ? SQLITE_UTF8 : 0);
30474757c658Sdan             rc = sessionValueSetStr(apOut[i],&pIn->aData[pIn->iNext],nByte,enc);
3048ef7a6304Sdan             pIn->iNext += nByte;
30494fccf43aSdan           }
3050e341ec69Sdan         }
3051e341ec69Sdan       }
30524fccf43aSdan       if( eType==SQLITE_INTEGER || eType==SQLITE_FLOAT ){
3053ef7a6304Sdan         sqlite3_int64 v = sessionGetI64(aVal);
30544fccf43aSdan         if( eType==SQLITE_INTEGER ){
30554fccf43aSdan           sqlite3VdbeMemSetInt64(apOut[i], v);
30564fccf43aSdan         }else{
30574fccf43aSdan           double d;
30584e895da1Sdan           memcpy(&d, &v, 8);
30594fccf43aSdan           sqlite3VdbeMemSetDouble(apOut[i], d);
30604fccf43aSdan         }
3061ef7a6304Sdan         pIn->iNext += 8;
306291ddd559Sdan       }
30634fccf43aSdan     }
30644fccf43aSdan   }
30654fccf43aSdan 
3066ef7a6304Sdan   return rc;
3067ef7a6304Sdan }
3068ef7a6304Sdan 
3069ef7a6304Sdan /*
3070ef7a6304Sdan ** The input pointer currently points to the second byte of a table-header.
3071ef7a6304Sdan ** Specifically, to the following:
3072ef7a6304Sdan **
3073ef7a6304Sdan **   + number of columns in table (varint)
3074ef7a6304Sdan **   + array of PK flags (1 byte per column),
3075ef7a6304Sdan **   + table name (nul terminated).
3076ef7a6304Sdan **
3077ef7a6304Sdan ** This function ensures that all of the above is present in the input
3078ef7a6304Sdan ** buffer (i.e. that it can be accessed without any calls to xInput()).
3079ef7a6304Sdan ** If successful, SQLITE_OK is returned. Otherwise, an SQLite error code.
3080ef7a6304Sdan ** The input pointer is not moved.
3081ef7a6304Sdan */
sessionChangesetBufferTblhdr(SessionInput * pIn,int * pnByte)3082ef7a6304Sdan static int sessionChangesetBufferTblhdr(SessionInput *pIn, int *pnByte){
3083ef7a6304Sdan   int rc = SQLITE_OK;
3084ef7a6304Sdan   int nCol = 0;
30854757c658Sdan   int nRead = 0;
3086ef7a6304Sdan 
3087ef7a6304Sdan   rc = sessionInputBuffer(pIn, 9);
3088ef7a6304Sdan   if( rc==SQLITE_OK ){
30894757c658Sdan     nRead += sessionVarintGet(&pIn->aData[pIn->iNext + nRead], &nCol);
30906344eddaSdan     /* The hard upper limit for the number of columns in an SQLite
30916344eddaSdan     ** database table is, according to sqliteLimit.h, 32676. So
30926344eddaSdan     ** consider any table-header that purports to have more than 65536
30936344eddaSdan     ** columns to be corrupt. This is convenient because otherwise,
30946344eddaSdan     ** if the (nCol>65536) condition below were omitted, a sufficiently
30956344eddaSdan     ** large value for nCol may cause nRead to wrap around and become
30966344eddaSdan     ** negative. Leading to a crash. */
30976344eddaSdan     if( nCol<0 || nCol>65536 ){
3098e341ec69Sdan       rc = SQLITE_CORRUPT_BKPT;
3099e341ec69Sdan     }else{
31004757c658Sdan       rc = sessionInputBuffer(pIn, nRead+nCol+100);
31014757c658Sdan       nRead += nCol;
3102ef7a6304Sdan     }
3103e341ec69Sdan   }
31044757c658Sdan 
3105ef7a6304Sdan   while( rc==SQLITE_OK ){
31064757c658Sdan     while( (pIn->iNext + nRead)<pIn->nData && pIn->aData[pIn->iNext + nRead] ){
31074757c658Sdan       nRead++;
3108ef7a6304Sdan     }
3109e8fa8c96Sdan     if( (pIn->iNext + nRead)<pIn->nData ) break;
31104757c658Sdan     rc = sessionInputBuffer(pIn, nRead + 100);
31114757c658Sdan   }
3112e8fa8c96Sdan   *pnByte = nRead+1;
3113ef7a6304Sdan   return rc;
3114ef7a6304Sdan }
3115ef7a6304Sdan 
3116ef7a6304Sdan /*
3117fa122adaSdan ** The input pointer currently points to the first byte of the first field
3118fa122adaSdan ** of a record consisting of nCol columns. This function ensures the entire
311916228167Sdan ** record is buffered. It does not move the input pointer.
312016228167Sdan **
312116228167Sdan ** If successful, SQLITE_OK is returned and *pnByte is set to the size of
312216228167Sdan ** the record in bytes. Otherwise, an SQLite error code is returned. The
312316228167Sdan ** final value of *pnByte is undefined in this case.
3124fa122adaSdan */
sessionChangesetBufferRecord(SessionInput * pIn,int nCol,int * pnByte)3125fa122adaSdan static int sessionChangesetBufferRecord(
312616228167Sdan   SessionInput *pIn,              /* Input data */
312716228167Sdan   int nCol,                       /* Number of columns in record */
312816228167Sdan   int *pnByte                     /* OUT: Size of record in bytes */
3129fa122adaSdan ){
3130fa122adaSdan   int rc = SQLITE_OK;
3131fa122adaSdan   int nByte = 0;
3132fa122adaSdan   int i;
3133fa122adaSdan   for(i=0; rc==SQLITE_OK && i<nCol; i++){
3134fa122adaSdan     int eType;
3135fa122adaSdan     rc = sessionInputBuffer(pIn, nByte + 10);
3136fa122adaSdan     if( rc==SQLITE_OK ){
3137fa122adaSdan       eType = pIn->aData[pIn->iNext + nByte++];
3138fa122adaSdan       if( eType==SQLITE_TEXT || eType==SQLITE_BLOB ){
3139fa122adaSdan         int n;
3140fa122adaSdan         nByte += sessionVarintGet(&pIn->aData[pIn->iNext+nByte], &n);
3141fa122adaSdan         nByte += n;
3142fa122adaSdan         rc = sessionInputBuffer(pIn, nByte);
3143fa122adaSdan       }else if( eType==SQLITE_INTEGER || eType==SQLITE_FLOAT ){
3144fa122adaSdan         nByte += 8;
3145fa122adaSdan       }
3146fa122adaSdan     }
3147fa122adaSdan   }
3148fa122adaSdan   *pnByte = nByte;
3149fa122adaSdan   return rc;
3150fa122adaSdan }
3151fa122adaSdan 
3152fa122adaSdan /*
3153ef7a6304Sdan ** The input pointer currently points to the second byte of a table-header.
3154ef7a6304Sdan ** Specifically, to the following:
3155ef7a6304Sdan **
3156ef7a6304Sdan **   + number of columns in table (varint)
3157ef7a6304Sdan **   + array of PK flags (1 byte per column),
3158ef7a6304Sdan **   + table name (nul terminated).
315916228167Sdan **
316016228167Sdan ** This function decodes the table-header and populates the p->nCol,
316116228167Sdan ** p->zTab and p->abPK[] variables accordingly. The p->apValue[] array is
316216228167Sdan ** also allocated or resized according to the new value of p->nCol. The
316316228167Sdan ** input pointer is left pointing to the byte following the table header.
316416228167Sdan **
316516228167Sdan ** If successful, SQLITE_OK is returned. Otherwise, an SQLite error code
316616228167Sdan ** is returned and the final values of the various fields enumerated above
316716228167Sdan ** are undefined.
3168ef7a6304Sdan */
sessionChangesetReadTblhdr(sqlite3_changeset_iter * p)3169ef7a6304Sdan static int sessionChangesetReadTblhdr(sqlite3_changeset_iter *p){
3170ef7a6304Sdan   int rc;
3171ef7a6304Sdan   int nCopy;
3172ef7a6304Sdan   assert( p->rc==SQLITE_OK );
3173ef7a6304Sdan 
3174ef7a6304Sdan   rc = sessionChangesetBufferTblhdr(&p->in, &nCopy);
3175ef7a6304Sdan   if( rc==SQLITE_OK ){
3176ef7a6304Sdan     int nByte;
3177ef7a6304Sdan     int nVarint;
31784757c658Sdan     nVarint = sessionVarintGet(&p->in.aData[p->in.iNext], &p->nCol);
3179dd8a4af8Sdan     if( p->nCol>0 ){
3180ef7a6304Sdan       nCopy -= nVarint;
3181ef7a6304Sdan       p->in.iNext += nVarint;
3182ef7a6304Sdan       nByte = p->nCol * sizeof(sqlite3_value*) * 2 + nCopy;
3183ef7a6304Sdan       p->tblhdr.nBuf = 0;
3184ef7a6304Sdan       sessionBufferGrow(&p->tblhdr, nByte, &rc);
3185dd8a4af8Sdan     }else{
3186e341ec69Sdan       rc = SQLITE_CORRUPT_BKPT;
3187dd8a4af8Sdan     }
3188ef7a6304Sdan   }
3189ef7a6304Sdan 
3190ef7a6304Sdan   if( rc==SQLITE_OK ){
3191f6ad201aSdrh     size_t iPK = sizeof(sqlite3_value*)*p->nCol*2;
3192ef7a6304Sdan     memset(p->tblhdr.aBuf, 0, iPK);
31934757c658Sdan     memcpy(&p->tblhdr.aBuf[iPK], &p->in.aData[p->in.iNext], nCopy);
3194ef7a6304Sdan     p->in.iNext += nCopy;
3195ef7a6304Sdan   }
3196ef7a6304Sdan 
3197ef7a6304Sdan   p->apValue = (sqlite3_value**)p->tblhdr.aBuf;
3198c08716a3Sdrh   if( p->apValue==0 ){
3199c08716a3Sdrh     p->abPK = 0;
3200c08716a3Sdrh     p->zTab = 0;
3201c08716a3Sdrh   }else{
3202ef7a6304Sdan     p->abPK = (u8*)&p->apValue[p->nCol*2];
3203c08716a3Sdrh     p->zTab = p->abPK ? (char*)&p->abPK[p->nCol] : 0;
3204c08716a3Sdrh   }
3205ef7a6304Sdan   return (p->rc = rc);
32064fccf43aSdan }
32074fccf43aSdan 
320877fc1d5bSdan /*
32091e25d20cSdan ** Advance the changeset iterator to the next change. The differences between
32101e25d20cSdan ** this function and sessionChangesetNext() are that
321177fc1d5bSdan **
32121e25d20cSdan **   * If pbEmpty is not NULL and the change is a no-op UPDATE (an UPDATE
32131e25d20cSdan **     that modifies no columns), this function sets (*pbEmpty) to 1.
321477fc1d5bSdan **
32151e25d20cSdan **   * If the iterator is configured to skip no-op UPDATEs,
32161e25d20cSdan **     sessionChangesetNext() does that. This function does not.
321777fc1d5bSdan */
sessionChangesetNextOne(sqlite3_changeset_iter * p,u8 ** paRec,int * pnRec,int * pbNew,int * pbEmpty)32181e25d20cSdan static int sessionChangesetNextOne(
321977fc1d5bSdan   sqlite3_changeset_iter *p,      /* Changeset iterator */
322077fc1d5bSdan   u8 **paRec,                     /* If non-NULL, store record pointer here */
3221c0a499eaSdan   int *pnRec,                     /* If non-NULL, store size of record here */
32221e25d20cSdan   int *pbNew,                     /* If non-NULL, true if new table */
32231e25d20cSdan   int *pbEmpty
32245d607a6eSdan ){
32254fccf43aSdan   int i;
3226ef7a6304Sdan   u8 op;
32274fccf43aSdan 
32285d607a6eSdan   assert( (paRec==0 && pnRec==0) || (paRec && pnRec) );
32291e25d20cSdan   assert( pbEmpty==0 || *pbEmpty==0 );
32305d607a6eSdan 
3231296c7658Sdan   /* If the iterator is in the error-state, return immediately. */
32324fccf43aSdan   if( p->rc!=SQLITE_OK ) return p->rc;
32334fccf43aSdan 
32345d607a6eSdan   /* Free the current contents of p->apValue[], if any. */
32354fccf43aSdan   if( p->apValue ){
32364fccf43aSdan     for(i=0; i<p->nCol*2; i++){
32374fccf43aSdan       sqlite3ValueFree(p->apValue[i]);
32384fccf43aSdan     }
32394fccf43aSdan     memset(p->apValue, 0, sizeof(sqlite3_value*)*p->nCol*2);
32404fccf43aSdan   }
32414fccf43aSdan 
3242ef7a6304Sdan   /* Make sure the buffer contains at least 10 bytes of input data, or all
3243ef7a6304Sdan   ** remaining data if there are less than 10 bytes available. This is
3244ef7a6304Sdan   ** sufficient either for the 'T' or 'P' byte and the varint that follows
3245ef7a6304Sdan   ** it, or for the two single byte values otherwise. */
3246ef7a6304Sdan   p->rc = sessionInputBuffer(&p->in, 2);
3247ef7a6304Sdan   if( p->rc!=SQLITE_OK ) return p->rc;
3248ef7a6304Sdan 
32494fccf43aSdan   /* If the iterator is already at the end of the changeset, return DONE. */
32504757c658Sdan   if( p->in.iNext>=p->in.nData ){
32514fccf43aSdan     return SQLITE_DONE;
32524fccf43aSdan   }
32534fccf43aSdan 
3254d9151526Sdan   sessionDiscardData(&p->in);
3255d9151526Sdan   p->in.iCurrent = p->in.iNext;
3256d9151526Sdan 
32574757c658Sdan   op = p->in.aData[p->in.iNext++];
325807d0f15eSdan   while( op=='T' || op=='P' ){
3259c0a499eaSdan     if( pbNew ) *pbNew = 1;
3260ef7a6304Sdan     p->bPatchset = (op=='P');
3261ef7a6304Sdan     if( sessionChangesetReadTblhdr(p) ) return p->rc;
3262ef7a6304Sdan     if( (p->rc = sessionInputBuffer(&p->in, 2)) ) return p->rc;
3263d9151526Sdan     p->in.iCurrent = p->in.iNext;
326407d0f15eSdan     if( p->in.iNext>=p->in.nData ) return SQLITE_DONE;
32654757c658Sdan     op = p->in.aData[p->in.iNext++];
32665d607a6eSdan   }
32675d607a6eSdan 
326844748f27Sdan   if( p->zTab==0 || (p->bPatchset && p->bInvert) ){
3269dd8a4af8Sdan     /* The first record in the changeset is not a table header. Must be a
3270dd8a4af8Sdan     ** corrupt changeset. */
327144748f27Sdan     assert( p->in.iNext==1 || p->zTab );
3272dd8a4af8Sdan     return (p->rc = SQLITE_CORRUPT_BKPT);
3273dd8a4af8Sdan   }
3274dd8a4af8Sdan 
3275ef7a6304Sdan   p->op = op;
32764757c658Sdan   p->bIndirect = p->in.aData[p->in.iNext++];
32774fccf43aSdan   if( p->op!=SQLITE_UPDATE && p->op!=SQLITE_DELETE && p->op!=SQLITE_INSERT ){
32784757c658Sdan     return (p->rc = SQLITE_CORRUPT_BKPT);
32794fccf43aSdan   }
32804fccf43aSdan 
3281cbf6d2d2Sdan   if( paRec ){
3282cbf6d2d2Sdan     int nVal;                     /* Number of values to buffer */
3283cbf6d2d2Sdan     if( p->bPatchset==0 && op==SQLITE_UPDATE ){
3284cbf6d2d2Sdan       nVal = p->nCol * 2;
3285cbf6d2d2Sdan     }else if( p->bPatchset && op==SQLITE_DELETE ){
3286cbf6d2d2Sdan       nVal = 0;
3287cbf6d2d2Sdan       for(i=0; i<p->nCol; i++) if( p->abPK[i] ) nVal++;
3288cbf6d2d2Sdan     }else{
3289cbf6d2d2Sdan       nVal = p->nCol;
3290cbf6d2d2Sdan     }
3291cbf6d2d2Sdan     p->rc = sessionChangesetBufferRecord(&p->in, nVal, pnRec);
3292cbf6d2d2Sdan     if( p->rc!=SQLITE_OK ) return p->rc;
3293cbf6d2d2Sdan     *paRec = &p->in.aData[p->in.iNext];
3294cbf6d2d2Sdan     p->in.iNext += *pnRec;
3295cbf6d2d2Sdan   }else{
329644748f27Sdan     sqlite3_value **apOld = (p->bInvert ? &p->apValue[p->nCol] : p->apValue);
329744748f27Sdan     sqlite3_value **apNew = (p->bInvert ? p->apValue : &p->apValue[p->nCol]);
32985d607a6eSdan 
32994fccf43aSdan     /* If this is an UPDATE or DELETE, read the old.* record. */
330073b3c055Sdan     if( p->op!=SQLITE_INSERT && (p->bPatchset==0 || p->op==SQLITE_DELETE) ){
330173b3c055Sdan       u8 *abPK = p->bPatchset ? p->abPK : 0;
33021e25d20cSdan       p->rc = sessionReadRecord(&p->in, p->nCol, abPK, apOld, 0);
33034fccf43aSdan       if( p->rc!=SQLITE_OK ) return p->rc;
33044fccf43aSdan     }
33054fccf43aSdan 
33064fccf43aSdan     /* If this is an INSERT or UPDATE, read the new.* record. */
33074fccf43aSdan     if( p->op!=SQLITE_DELETE ){
33081e25d20cSdan       p->rc = sessionReadRecord(&p->in, p->nCol, 0, apNew, pbEmpty);
33094fccf43aSdan       if( p->rc!=SQLITE_OK ) return p->rc;
33104fccf43aSdan     }
33114fccf43aSdan 
331244748f27Sdan     if( (p->bPatchset || p->bInvert) && p->op==SQLITE_UPDATE ){
331373b3c055Sdan       /* If this is an UPDATE that is part of a patchset, then all PK and
331473b3c055Sdan       ** modified fields are present in the new.* record. The old.* record
331573b3c055Sdan       ** is currently completely empty. This block shifts the PK fields from
331673b3c055Sdan       ** new.* to old.*, to accommodate the code that reads these arrays.  */
331773b3c055Sdan       for(i=0; i<p->nCol; i++){
331844748f27Sdan         assert( p->bPatchset==0 || p->apValue[i]==0 );
331973b3c055Sdan         if( p->abPK[i] ){
332044748f27Sdan           assert( p->apValue[i]==0 );
332173b3c055Sdan           p->apValue[i] = p->apValue[i+p->nCol];
3322e341ec69Sdan           if( p->apValue[i]==0 ) return (p->rc = SQLITE_CORRUPT_BKPT);
332373b3c055Sdan           p->apValue[i+p->nCol] = 0;
332473b3c055Sdan         }
332573b3c055Sdan       }
332644748f27Sdan     }else if( p->bInvert ){
332744748f27Sdan       if( p->op==SQLITE_INSERT ) p->op = SQLITE_DELETE;
332844748f27Sdan       else if( p->op==SQLITE_DELETE ) p->op = SQLITE_INSERT;
332973b3c055Sdan     }
3330cbf6d2d2Sdan   }
3331ef7a6304Sdan 
33324fccf43aSdan   return SQLITE_ROW;
33334fccf43aSdan }
33344fccf43aSdan 
33354fccf43aSdan /*
33361e25d20cSdan ** Advance the changeset iterator to the next change.
33371e25d20cSdan **
33381e25d20cSdan ** If both paRec and pnRec are NULL, then this function works like the public
33391e25d20cSdan ** API sqlite3changeset_next(). If SQLITE_ROW is returned, then the
33401e25d20cSdan ** sqlite3changeset_new() and old() APIs may be used to query for values.
33411e25d20cSdan **
33421e25d20cSdan ** Otherwise, if paRec and pnRec are not NULL, then a pointer to the change
33431e25d20cSdan ** record is written to *paRec before returning and the number of bytes in
33441e25d20cSdan ** the record to *pnRec.
33451e25d20cSdan **
33461e25d20cSdan ** Either way, this function returns SQLITE_ROW if the iterator is
33471e25d20cSdan ** successfully advanced to the next change in the changeset, an SQLite
33481e25d20cSdan ** error code if an error occurs, or SQLITE_DONE if there are no further
33491e25d20cSdan ** changes in the changeset.
33501e25d20cSdan */
sessionChangesetNext(sqlite3_changeset_iter * p,u8 ** paRec,int * pnRec,int * pbNew)33511e25d20cSdan static int sessionChangesetNext(
33521e25d20cSdan   sqlite3_changeset_iter *p,      /* Changeset iterator */
33531e25d20cSdan   u8 **paRec,                     /* If non-NULL, store record pointer here */
33541e25d20cSdan   int *pnRec,                     /* If non-NULL, store size of record here */
33551e25d20cSdan   int *pbNew                      /* If non-NULL, true if new table */
33561e25d20cSdan ){
33571e25d20cSdan   int bEmpty;
33581e25d20cSdan   int rc;
33591e25d20cSdan   do {
33601e25d20cSdan     bEmpty = 0;
33611e25d20cSdan     rc = sessionChangesetNextOne(p, paRec, pnRec, pbNew, &bEmpty);
33621e25d20cSdan   }while( rc==SQLITE_ROW && p->bSkipEmpty && bEmpty);
33631e25d20cSdan   return rc;
33641e25d20cSdan }
33651e25d20cSdan 
33661e25d20cSdan /*
33675d607a6eSdan ** Advance an iterator created by sqlite3changeset_start() to the next
33685d607a6eSdan ** change in the changeset. This function may return SQLITE_ROW, SQLITE_DONE
33695d607a6eSdan ** or SQLITE_CORRUPT.
33705d607a6eSdan **
33715d607a6eSdan ** This function may not be called on iterators passed to a conflict handler
33725d607a6eSdan ** callback by changeset_apply().
33735d607a6eSdan */
sqlite3changeset_next(sqlite3_changeset_iter * p)33745d607a6eSdan int sqlite3changeset_next(sqlite3_changeset_iter *p){
3375c0a499eaSdan   return sessionChangesetNext(p, 0, 0, 0);
33765d607a6eSdan }
33775d607a6eSdan 
33785d607a6eSdan /*
3379244593c8Sdan ** The following function extracts information on the current change
338077fc1d5bSdan ** from a changeset iterator. It may only be called after changeset_next()
33814fccf43aSdan ** has returned SQLITE_ROW.
33824fccf43aSdan */
sqlite3changeset_op(sqlite3_changeset_iter * pIter,const char ** pzTab,int * pnCol,int * pOp,int * pbIndirect)33834fccf43aSdan int sqlite3changeset_op(
3384296c7658Sdan   sqlite3_changeset_iter *pIter,  /* Iterator handle */
33854fccf43aSdan   const char **pzTab,             /* OUT: Pointer to table name */
33864fccf43aSdan   int *pnCol,                     /* OUT: Number of columns in table */
3387b4480e94Sdan   int *pOp,                       /* OUT: SQLITE_INSERT, DELETE or UPDATE */
3388b4480e94Sdan   int *pbIndirect                 /* OUT: True if change is indirect */
33894fccf43aSdan ){
33904fccf43aSdan   *pOp = pIter->op;
33914fccf43aSdan   *pnCol = pIter->nCol;
33924fccf43aSdan   *pzTab = pIter->zTab;
3393b4480e94Sdan   if( pbIndirect ) *pbIndirect = pIter->bIndirect;
33944fccf43aSdan   return SQLITE_OK;
33954fccf43aSdan }
33964fccf43aSdan 
339777fc1d5bSdan /*
339877fc1d5bSdan ** Return information regarding the PRIMARY KEY and number of columns in
339977fc1d5bSdan ** the database table affected by the change that pIter currently points
340077fc1d5bSdan ** to. This function may only be called after changeset_next() returns
340177fc1d5bSdan ** SQLITE_ROW.
340277fc1d5bSdan */
sqlite3changeset_pk(sqlite3_changeset_iter * pIter,unsigned char ** pabPK,int * pnCol)3403244593c8Sdan int sqlite3changeset_pk(
3404244593c8Sdan   sqlite3_changeset_iter *pIter,  /* Iterator object */
3405244593c8Sdan   unsigned char **pabPK,          /* OUT: Array of boolean - true for PK cols */
3406244593c8Sdan   int *pnCol                      /* OUT: Number of entries in output array */
3407244593c8Sdan ){
3408244593c8Sdan   *pabPK = pIter->abPK;
3409244593c8Sdan   if( pnCol ) *pnCol = pIter->nCol;
3410244593c8Sdan   return SQLITE_OK;
3411244593c8Sdan }
3412244593c8Sdan 
3413296c7658Sdan /*
3414296c7658Sdan ** This function may only be called while the iterator is pointing to an
3415296c7658Sdan ** SQLITE_UPDATE or SQLITE_DELETE change (see sqlite3changeset_op()).
3416296c7658Sdan ** Otherwise, SQLITE_MISUSE is returned.
3417296c7658Sdan **
3418296c7658Sdan ** It sets *ppValue to point to an sqlite3_value structure containing the
3419296c7658Sdan ** iVal'th value in the old.* record. Or, if that particular value is not
3420296c7658Sdan ** included in the record (because the change is an UPDATE and the field
3421296c7658Sdan ** was not modified and is not a PK column), set *ppValue to NULL.
3422296c7658Sdan **
3423296c7658Sdan ** If value iVal is out-of-range, SQLITE_RANGE is returned and *ppValue is
3424296c7658Sdan ** not modified. Otherwise, SQLITE_OK.
3425296c7658Sdan */
sqlite3changeset_old(sqlite3_changeset_iter * pIter,int iVal,sqlite3_value ** ppValue)34264fccf43aSdan int sqlite3changeset_old(
3427296c7658Sdan   sqlite3_changeset_iter *pIter,  /* Changeset iterator */
3428296c7658Sdan   int iVal,                       /* Index of old.* value to retrieve */
34294fccf43aSdan   sqlite3_value **ppValue         /* OUT: Old value (or NULL pointer) */
34304fccf43aSdan ){
3431d5f0767cSdan   if( pIter->op!=SQLITE_UPDATE && pIter->op!=SQLITE_DELETE ){
3432d5f0767cSdan     return SQLITE_MISUSE;
3433d5f0767cSdan   }
34344fccf43aSdan   if( iVal<0 || iVal>=pIter->nCol ){
34354fccf43aSdan     return SQLITE_RANGE;
34364fccf43aSdan   }
34374fccf43aSdan   *ppValue = pIter->apValue[iVal];
34384fccf43aSdan   return SQLITE_OK;
34394fccf43aSdan }
34404fccf43aSdan 
3441296c7658Sdan /*
3442296c7658Sdan ** This function may only be called while the iterator is pointing to an
3443296c7658Sdan ** SQLITE_UPDATE or SQLITE_INSERT change (see sqlite3changeset_op()).
3444296c7658Sdan ** Otherwise, SQLITE_MISUSE is returned.
3445296c7658Sdan **
3446296c7658Sdan ** It sets *ppValue to point to an sqlite3_value structure containing the
3447296c7658Sdan ** iVal'th value in the new.* record. Or, if that particular value is not
3448296c7658Sdan ** included in the record (because the change is an UPDATE and the field
3449296c7658Sdan ** was not modified), set *ppValue to NULL.
3450296c7658Sdan **
3451296c7658Sdan ** If value iVal is out-of-range, SQLITE_RANGE is returned and *ppValue is
3452296c7658Sdan ** not modified. Otherwise, SQLITE_OK.
3453296c7658Sdan */
sqlite3changeset_new(sqlite3_changeset_iter * pIter,int iVal,sqlite3_value ** ppValue)34544fccf43aSdan int sqlite3changeset_new(
3455296c7658Sdan   sqlite3_changeset_iter *pIter,  /* Changeset iterator */
3456296c7658Sdan   int iVal,                       /* Index of new.* value to retrieve */
34574fccf43aSdan   sqlite3_value **ppValue         /* OUT: New value (or NULL pointer) */
34584fccf43aSdan ){
3459d5f0767cSdan   if( pIter->op!=SQLITE_UPDATE && pIter->op!=SQLITE_INSERT ){
3460d5f0767cSdan     return SQLITE_MISUSE;
3461d5f0767cSdan   }
34624fccf43aSdan   if( iVal<0 || iVal>=pIter->nCol ){
34634fccf43aSdan     return SQLITE_RANGE;
34644fccf43aSdan   }
34654fccf43aSdan   *ppValue = pIter->apValue[pIter->nCol+iVal];
34664fccf43aSdan   return SQLITE_OK;
34674fccf43aSdan }
34684fccf43aSdan 
3469296c7658Sdan /*
34707aa469cdSdan ** The following two macros are used internally. They are similar to the
34717aa469cdSdan ** sqlite3changeset_new() and sqlite3changeset_old() functions, except that
34727aa469cdSdan ** they omit all error checking and return a pointer to the requested value.
34737aa469cdSdan */
34747aa469cdSdan #define sessionChangesetNew(pIter, iVal) (pIter)->apValue[(pIter)->nCol+(iVal)]
34757aa469cdSdan #define sessionChangesetOld(pIter, iVal) (pIter)->apValue[(iVal)]
34767aa469cdSdan 
34777aa469cdSdan /*
3478296c7658Sdan ** This function may only be called with a changeset iterator that has been
3479296c7658Sdan ** passed to an SQLITE_CHANGESET_DATA or SQLITE_CHANGESET_CONFLICT
3480296c7658Sdan ** conflict-handler function. Otherwise, SQLITE_MISUSE is returned.
3481296c7658Sdan **
3482296c7658Sdan ** If successful, *ppValue is set to point to an sqlite3_value structure
3483296c7658Sdan ** containing the iVal'th value of the conflicting record.
3484296c7658Sdan **
3485296c7658Sdan ** If value iVal is out-of-range or some other error occurs, an SQLite error
3486296c7658Sdan ** code is returned. Otherwise, SQLITE_OK.
3487296c7658Sdan */
sqlite3changeset_conflict(sqlite3_changeset_iter * pIter,int iVal,sqlite3_value ** ppValue)3488d5f0767cSdan int sqlite3changeset_conflict(
3489296c7658Sdan   sqlite3_changeset_iter *pIter,  /* Changeset iterator */
3490296c7658Sdan   int iVal,                       /* Index of conflict record value to fetch */
3491d5f0767cSdan   sqlite3_value **ppValue         /* OUT: Value from conflicting row */
3492d5f0767cSdan ){
3493d5f0767cSdan   if( !pIter->pConflict ){
3494d5f0767cSdan     return SQLITE_MISUSE;
3495d5f0767cSdan   }
3496ff677b20Sdan   if( iVal<0 || iVal>=pIter->nCol ){
3497d5f0767cSdan     return SQLITE_RANGE;
3498d5f0767cSdan   }
3499d5f0767cSdan   *ppValue = sqlite3_column_value(pIter->pConflict, iVal);
3500d5f0767cSdan   return SQLITE_OK;
3501d5f0767cSdan }
3502d5f0767cSdan 
35034fccf43aSdan /*
3504cb3e4b79Sdan ** This function may only be called with an iterator passed to an
3505cb3e4b79Sdan ** SQLITE_CHANGESET_FOREIGN_KEY conflict handler callback. In this case
3506cb3e4b79Sdan ** it sets the output variable to the total number of known foreign key
3507cb3e4b79Sdan ** violations in the destination database and returns SQLITE_OK.
3508cb3e4b79Sdan **
3509cb3e4b79Sdan ** In all other cases this function returns SQLITE_MISUSE.
3510cb3e4b79Sdan */
sqlite3changeset_fk_conflicts(sqlite3_changeset_iter * pIter,int * pnOut)3511cb3e4b79Sdan int sqlite3changeset_fk_conflicts(
3512cb3e4b79Sdan   sqlite3_changeset_iter *pIter,  /* Changeset iterator */
3513cb3e4b79Sdan   int *pnOut                      /* OUT: Number of FK violations */
3514cb3e4b79Sdan ){
3515cb3e4b79Sdan   if( pIter->pConflict || pIter->apValue ){
3516cb3e4b79Sdan     return SQLITE_MISUSE;
3517cb3e4b79Sdan   }
3518cb3e4b79Sdan   *pnOut = pIter->nCol;
3519cb3e4b79Sdan   return SQLITE_OK;
3520cb3e4b79Sdan }
3521cb3e4b79Sdan 
3522cb3e4b79Sdan 
3523cb3e4b79Sdan /*
35244fccf43aSdan ** Finalize an iterator allocated with sqlite3changeset_start().
35254fccf43aSdan **
35264fccf43aSdan ** This function may not be called on iterators passed to a conflict handler
35274fccf43aSdan ** callback by changeset_apply().
35284fccf43aSdan */
sqlite3changeset_finalize(sqlite3_changeset_iter * p)35294fccf43aSdan int sqlite3changeset_finalize(sqlite3_changeset_iter *p){
3530cbf6d2d2Sdan   int rc = SQLITE_OK;
3531cbf6d2d2Sdan   if( p ){
3532296c7658Sdan     int i;                        /* Used to iterate through p->apValue[] */
3533cbf6d2d2Sdan     rc = p->rc;
353412ca0b56Sdan     if( p->apValue ){
35354fccf43aSdan       for(i=0; i<p->nCol*2; i++) sqlite3ValueFree(p->apValue[i]);
353612ca0b56Sdan     }
3537ef7a6304Sdan     sqlite3_free(p->tblhdr.aBuf);
35384757c658Sdan     sqlite3_free(p->in.buf.aBuf);
35394fccf43aSdan     sqlite3_free(p);
3540cbf6d2d2Sdan   }
35414fccf43aSdan   return rc;
35424fccf43aSdan }
35434fccf43aSdan 
sessionChangesetInvert(SessionInput * pInput,int (* xOutput)(void * pOut,const void * pData,int nData),void * pOut,int * pnInverted,void ** ppInverted)3544fa122adaSdan static int sessionChangesetInvert(
3545fa122adaSdan   SessionInput *pInput,           /* Input changeset */
3546fa122adaSdan   int (*xOutput)(void *pOut, const void *pData, int nData),
3547fa122adaSdan   void *pOut,
354891ddd559Sdan   int *pnInverted,                /* OUT: Number of bytes in output changeset */
354991ddd559Sdan   void **ppInverted               /* OUT: Inverse of pChangeset */
355091ddd559Sdan ){
3551cfec7eeeSdan   int rc = SQLITE_OK;             /* Return value */
3552fa122adaSdan   SessionBuffer sOut;             /* Output buffer */
3553cfec7eeeSdan   int nCol = 0;                   /* Number of cols in current table */
3554cfec7eeeSdan   u8 *abPK = 0;                   /* PK array for current table */
3555cfec7eeeSdan   sqlite3_value **apVal = 0;      /* Space for values for UPDATE inversion */
3556ef7a6304Sdan   SessionBuffer sPK = {0, 0, 0};  /* PK array for current table */
355791ddd559Sdan 
3558fa122adaSdan   /* Initialize the output buffer */
3559fa122adaSdan   memset(&sOut, 0, sizeof(SessionBuffer));
3560fa122adaSdan 
356191ddd559Sdan   /* Zero the output variables in case an error occurs. */
3562fa122adaSdan   if( ppInverted ){
356391ddd559Sdan     *ppInverted = 0;
356491ddd559Sdan     *pnInverted = 0;
3565fa122adaSdan   }
356691ddd559Sdan 
3567fa122adaSdan   while( 1 ){
3568ef7a6304Sdan     u8 eType;
3569fa122adaSdan 
3570fa122adaSdan     /* Test for EOF. */
3571fa122adaSdan     if( (rc = sessionInputBuffer(pInput, 2)) ) goto finished_invert;
3572fa122adaSdan     if( pInput->iNext>=pInput->nData ) break;
3573fa122adaSdan     eType = pInput->aData[pInput->iNext];
3574fa122adaSdan 
357591ddd559Sdan     switch( eType ){
357691ddd559Sdan       case 'T': {
3577244593c8Sdan         /* A 'table' record consists of:
3578244593c8Sdan         **
3579244593c8Sdan         **   * A constant 'T' character,
3580244593c8Sdan         **   * Number of columns in said table (a varint),
3581ef7a6304Sdan         **   * An array of nCol bytes (sPK),
3582244593c8Sdan         **   * A nul-terminated table name.
3583244593c8Sdan         */
3584ef7a6304Sdan         int nByte;
3585fa122adaSdan         int nVar;
3586fa122adaSdan         pInput->iNext++;
3587fa122adaSdan         if( (rc = sessionChangesetBufferTblhdr(pInput, &nByte)) ){
3588ef7a6304Sdan           goto finished_invert;
3589ef7a6304Sdan         }
3590fa122adaSdan         nVar = sessionVarintGet(&pInput->aData[pInput->iNext], &nCol);
3591ef7a6304Sdan         sPK.nBuf = 0;
3592fa122adaSdan         sessionAppendBlob(&sPK, &pInput->aData[pInput->iNext+nVar], nCol, &rc);
3593fa122adaSdan         sessionAppendByte(&sOut, eType, &rc);
3594fa122adaSdan         sessionAppendBlob(&sOut, &pInput->aData[pInput->iNext], nByte, &rc);
3595ef7a6304Sdan         if( rc ) goto finished_invert;
3596fa122adaSdan 
3597fa122adaSdan         pInput->iNext += nByte;
3598cfec7eeeSdan         sqlite3_free(apVal);
3599cfec7eeeSdan         apVal = 0;
3600ef7a6304Sdan         abPK = sPK.aBuf;
360191ddd559Sdan         break;
360291ddd559Sdan       }
360391ddd559Sdan 
360491ddd559Sdan       case SQLITE_INSERT:
360591ddd559Sdan       case SQLITE_DELETE: {
360691ddd559Sdan         int nByte;
3607fa122adaSdan         int bIndirect = pInput->aData[pInput->iNext+1];
3608fa122adaSdan         int eType2 = (eType==SQLITE_DELETE ? SQLITE_INSERT : SQLITE_DELETE);
3609fa122adaSdan         pInput->iNext += 2;
3610fa122adaSdan         assert( rc==SQLITE_OK );
3611fa122adaSdan         rc = sessionChangesetBufferRecord(pInput, nCol, &nByte);
3612fa122adaSdan         sessionAppendByte(&sOut, eType2, &rc);
3613fa122adaSdan         sessionAppendByte(&sOut, bIndirect, &rc);
3614fa122adaSdan         sessionAppendBlob(&sOut, &pInput->aData[pInput->iNext], nByte, &rc);
3615fa122adaSdan         pInput->iNext += nByte;
3616fa122adaSdan         if( rc ) goto finished_invert;
361791ddd559Sdan         break;
361891ddd559Sdan       }
361991ddd559Sdan 
362091ddd559Sdan       case SQLITE_UPDATE: {
3621cfec7eeeSdan         int iCol;
362291ddd559Sdan 
3623cfec7eeeSdan         if( 0==apVal ){
36242d77d80aSdrh           apVal = (sqlite3_value **)sqlite3_malloc64(sizeof(apVal[0])*nCol*2);
3625cfec7eeeSdan           if( 0==apVal ){
3626cfec7eeeSdan             rc = SQLITE_NOMEM;
3627cfec7eeeSdan             goto finished_invert;
3628cfec7eeeSdan           }
3629cfec7eeeSdan           memset(apVal, 0, sizeof(apVal[0])*nCol*2);
3630cfec7eeeSdan         }
363191ddd559Sdan 
3632cfec7eeeSdan         /* Write the header for the new UPDATE change. Same as the original. */
3633fa122adaSdan         sessionAppendByte(&sOut, eType, &rc);
3634fa122adaSdan         sessionAppendByte(&sOut, pInput->aData[pInput->iNext+1], &rc);
363591ddd559Sdan 
3636ef7a6304Sdan         /* Read the old.* and new.* records for the update change. */
3637fa122adaSdan         pInput->iNext += 2;
36381e25d20cSdan         rc = sessionReadRecord(pInput, nCol, 0, &apVal[0], 0);
3639ef7a6304Sdan         if( rc==SQLITE_OK ){
36401e25d20cSdan           rc = sessionReadRecord(pInput, nCol, 0, &apVal[nCol], 0);
3641ef7a6304Sdan         }
3642ef7a6304Sdan 
3643cfec7eeeSdan         /* Write the new old.* record. Consists of the PK columns from the
3644cfec7eeeSdan         ** original old.* record, and the other values from the original
3645cfec7eeeSdan         ** new.* record. */
3646e8fa8c96Sdan         for(iCol=0; iCol<nCol; iCol++){
3647cfec7eeeSdan           sqlite3_value *pVal = apVal[iCol + (abPK[iCol] ? 0 : nCol)];
3648fa122adaSdan           sessionAppendValue(&sOut, pVal, &rc);
3649cfec7eeeSdan         }
3650cfec7eeeSdan 
3651cfec7eeeSdan         /* Write the new new.* record. Consists of a copy of all values
3652cfec7eeeSdan         ** from the original old.* record, except for the PK columns, which
3653cfec7eeeSdan         ** are set to "undefined". */
3654e8fa8c96Sdan         for(iCol=0; iCol<nCol; iCol++){
3655cfec7eeeSdan           sqlite3_value *pVal = (abPK[iCol] ? 0 : apVal[iCol]);
3656fa122adaSdan           sessionAppendValue(&sOut, pVal, &rc);
3657cfec7eeeSdan         }
3658cfec7eeeSdan 
3659cfec7eeeSdan         for(iCol=0; iCol<nCol*2; iCol++){
3660cfec7eeeSdan           sqlite3ValueFree(apVal[iCol]);
3661cfec7eeeSdan         }
3662cfec7eeeSdan         memset(apVal, 0, sizeof(apVal[0])*nCol*2);
3663cfec7eeeSdan         if( rc!=SQLITE_OK ){
3664cfec7eeeSdan           goto finished_invert;
3665cfec7eeeSdan         }
3666cfec7eeeSdan 
366791ddd559Sdan         break;
366891ddd559Sdan       }
366991ddd559Sdan 
367091ddd559Sdan       default:
36714757c658Sdan         rc = SQLITE_CORRUPT_BKPT;
3672cfec7eeeSdan         goto finished_invert;
367391ddd559Sdan     }
3674fa122adaSdan 
3675fa122adaSdan     assert( rc==SQLITE_OK );
36761f48e67dSdan     if( xOutput && sOut.nBuf>=sessions_strm_chunk_size ){
3677fa122adaSdan       rc = xOutput(pOut, sOut.aBuf, sOut.nBuf);
3678fa122adaSdan       sOut.nBuf = 0;
3679fa122adaSdan       if( rc!=SQLITE_OK ) goto finished_invert;
3680fa122adaSdan     }
368191ddd559Sdan   }
368291ddd559Sdan 
3683cfec7eeeSdan   assert( rc==SQLITE_OK );
368411a9ad56Sdrh   if( pnInverted && ALWAYS(ppInverted) ){
3685fa122adaSdan     *pnInverted = sOut.nBuf;
3686fa122adaSdan     *ppInverted = sOut.aBuf;
3687fa122adaSdan     sOut.aBuf = 0;
368811a9ad56Sdrh   }else if( sOut.nBuf>0 && ALWAYS(xOutput!=0) ){
3689fa122adaSdan     rc = xOutput(pOut, sOut.aBuf, sOut.nBuf);
3690fa122adaSdan   }
3691cfec7eeeSdan 
3692cfec7eeeSdan  finished_invert:
3693fa122adaSdan   sqlite3_free(sOut.aBuf);
3694cfec7eeeSdan   sqlite3_free(apVal);
3695ef7a6304Sdan   sqlite3_free(sPK.aBuf);
3696cfec7eeeSdan   return rc;
369791ddd559Sdan }
369891ddd559Sdan 
3699fa122adaSdan 
3700fa122adaSdan /*
3701fa122adaSdan ** Invert a changeset object.
3702fa122adaSdan */
sqlite3changeset_invert(int nChangeset,const void * pChangeset,int * pnInverted,void ** ppInverted)3703fa122adaSdan int sqlite3changeset_invert(
3704fa122adaSdan   int nChangeset,                 /* Number of bytes in input */
3705fa122adaSdan   const void *pChangeset,         /* Input changeset */
3706fa122adaSdan   int *pnInverted,                /* OUT: Number of bytes in output changeset */
3707fa122adaSdan   void **ppInverted               /* OUT: Inverse of pChangeset */
3708fa122adaSdan ){
3709fa122adaSdan   SessionInput sInput;
3710fa122adaSdan 
3711fa122adaSdan   /* Set up the input stream */
3712fa122adaSdan   memset(&sInput, 0, sizeof(SessionInput));
3713fa122adaSdan   sInput.nData = nChangeset;
3714fa122adaSdan   sInput.aData = (u8*)pChangeset;
3715fa122adaSdan 
3716fa122adaSdan   return sessionChangesetInvert(&sInput, 0, 0, pnInverted, ppInverted);
3717fa122adaSdan }
3718fa122adaSdan 
3719fa122adaSdan /*
3720fa122adaSdan ** Streaming version of sqlite3changeset_invert().
3721fa122adaSdan */
sqlite3changeset_invert_strm(int (* xInput)(void * pIn,void * pData,int * pnData),void * pIn,int (* xOutput)(void * pOut,const void * pData,int nData),void * pOut)3722f1a08ad8Sdrh int sqlite3changeset_invert_strm(
3723fa122adaSdan   int (*xInput)(void *pIn, void *pData, int *pnData),
3724fa122adaSdan   void *pIn,
3725fa122adaSdan   int (*xOutput)(void *pOut, const void *pData, int nData),
3726fa122adaSdan   void *pOut
3727fa122adaSdan ){
3728fa122adaSdan   SessionInput sInput;
3729fa122adaSdan   int rc;
3730fa122adaSdan 
3731fa122adaSdan   /* Set up the input stream */
3732fa122adaSdan   memset(&sInput, 0, sizeof(SessionInput));
3733fa122adaSdan   sInput.xInput = xInput;
3734fa122adaSdan   sInput.pIn = pIn;
3735fa122adaSdan 
3736fa122adaSdan   rc = sessionChangesetInvert(&sInput, xOutput, pOut, 0, 0);
3737fa122adaSdan   sqlite3_free(sInput.buf.aBuf);
3738fa122adaSdan   return rc;
3739fa122adaSdan }
3740fa122adaSdan 
3741e0d2096aSdan 
3742e0d2096aSdan typedef struct SessionUpdate SessionUpdate;
3743e0d2096aSdan struct SessionUpdate {
3744e0d2096aSdan   sqlite3_stmt *pStmt;
3745e0d2096aSdan   u32 *aMask;
3746e0d2096aSdan   SessionUpdate *pNext;
3747e0d2096aSdan };
3748e0d2096aSdan 
37490c698471Sdan typedef struct SessionApplyCtx SessionApplyCtx;
37500c698471Sdan struct SessionApplyCtx {
37510c698471Sdan   sqlite3 *db;
37520c698471Sdan   sqlite3_stmt *pDelete;          /* DELETE statement */
37530c698471Sdan   sqlite3_stmt *pInsert;          /* INSERT statement */
37540c698471Sdan   sqlite3_stmt *pSelect;          /* SELECT statement */
37550c698471Sdan   int nCol;                       /* Size of azCol[] and abPK[] arrays */
37560c698471Sdan   const char **azCol;             /* Array of column names */
37570c698471Sdan   u8 *abPK;                       /* Boolean array - true if column is in PK */
3758e0d2096aSdan   u32 *aUpdateMask;               /* Used by sessionUpdateFind */
3759e0d2096aSdan   SessionUpdate *pUp;
3760d1cccf19Sdan   int bStat1;                     /* True if table is sqlite_stat1 */
3761d9151526Sdan   int bDeferConstraints;          /* True to defer constraints */
37625d237bfaSdan   int bInvertConstraints;         /* Invert when iterating constraints buffer */
3763d9151526Sdan   SessionBuffer constraints;      /* Deferred constraints are stored here */
3764a38e6c57Sdan   SessionBuffer rebase;           /* Rebase information (if any) here */
3765dbe7d37aSdan   u8 bRebaseStarted;              /* If table header is already in rebase */
3766dbe7d37aSdan   u8 bRebase;                     /* True to collect rebase information */
37670c698471Sdan };
37680c698471Sdan 
3769e0d2096aSdan /* Number of prepared UPDATE statements to cache. */
3770e0d2096aSdan #define SESSION_UPDATE_CACHE_SZ 12
3771e0d2096aSdan 
3772e0d2096aSdan /*
3773e0d2096aSdan ** Find a prepared UPDATE statement suitable for the UPDATE step currently
3774e0d2096aSdan ** being visited by the iterator. The UPDATE is of the form:
3775e0d2096aSdan **
3776e0d2096aSdan **   UPDATE tbl SET col = ?, col2 = ? WHERE pk1 IS ? AND pk2 IS ?
3777e0d2096aSdan */
sessionUpdateFind(sqlite3_changeset_iter * pIter,SessionApplyCtx * p,int bPatchset,sqlite3_stmt ** ppStmt)3778e0d2096aSdan static int sessionUpdateFind(
3779e0d2096aSdan   sqlite3_changeset_iter *pIter,
3780e0d2096aSdan   SessionApplyCtx *p,
3781e0d2096aSdan   int bPatchset,
3782e0d2096aSdan   sqlite3_stmt **ppStmt
3783e0d2096aSdan ){
3784e0d2096aSdan   int rc = SQLITE_OK;
3785e0d2096aSdan   SessionUpdate *pUp = 0;
3786e0d2096aSdan   int nCol = pIter->nCol;
3787e0d2096aSdan   int nU32 = (pIter->nCol+33)/32;
3788e0d2096aSdan   int ii;
3789e0d2096aSdan 
3790e0d2096aSdan   if( p->aUpdateMask==0 ){
3791e0d2096aSdan     p->aUpdateMask = sqlite3_malloc(nU32*sizeof(u32));
3792e0d2096aSdan     if( p->aUpdateMask==0 ){
3793e0d2096aSdan       rc = SQLITE_NOMEM;
3794e0d2096aSdan     }
3795e0d2096aSdan   }
3796e0d2096aSdan 
3797e0d2096aSdan   if( rc==SQLITE_OK ){
3798e0d2096aSdan     memset(p->aUpdateMask, 0, nU32*sizeof(u32));
3799e0d2096aSdan     rc = SQLITE_CORRUPT;
3800e0d2096aSdan     for(ii=0; ii<pIter->nCol; ii++){
3801e0d2096aSdan       if( sessionChangesetNew(pIter, ii) ){
3802e0d2096aSdan         p->aUpdateMask[ii/32] |= (1<<(ii%32));
3803e0d2096aSdan         rc = SQLITE_OK;
3804e0d2096aSdan       }
3805e0d2096aSdan     }
3806e0d2096aSdan   }
3807e0d2096aSdan 
3808e0d2096aSdan   if( rc==SQLITE_OK ){
3809e0d2096aSdan     if( bPatchset ) p->aUpdateMask[nCol/32] |= (1<<(nCol%32));
3810e0d2096aSdan 
3811e0d2096aSdan     if( p->pUp ){
3812e0d2096aSdan       int nUp = 0;
3813e0d2096aSdan       SessionUpdate **pp = &p->pUp;
3814e0d2096aSdan       while( 1 ){
3815e0d2096aSdan         nUp++;
3816e0d2096aSdan         if( 0==memcmp(p->aUpdateMask, (*pp)->aMask, nU32*sizeof(u32)) ){
3817e0d2096aSdan           pUp = *pp;
3818e0d2096aSdan           *pp = pUp->pNext;
3819e0d2096aSdan           pUp->pNext = p->pUp;
3820e0d2096aSdan           p->pUp = pUp;
3821e0d2096aSdan           break;
3822e0d2096aSdan         }
3823e0d2096aSdan 
3824e0d2096aSdan         if( (*pp)->pNext ){
3825e0d2096aSdan           pp = &(*pp)->pNext;
3826e0d2096aSdan         }else{
3827e0d2096aSdan           if( nUp>=SESSION_UPDATE_CACHE_SZ ){
3828e0d2096aSdan             sqlite3_finalize((*pp)->pStmt);
3829e0d2096aSdan             sqlite3_free(*pp);
3830e0d2096aSdan             *pp = 0;
3831e0d2096aSdan           }
3832e0d2096aSdan           break;
3833e0d2096aSdan         }
3834e0d2096aSdan       }
3835e0d2096aSdan     }
3836e0d2096aSdan 
3837e0d2096aSdan     if( pUp==0 ){
3838e0d2096aSdan       int nByte = sizeof(SessionUpdate) * nU32*sizeof(u32);
3839e0d2096aSdan       int bStat1 = (sqlite3_stricmp(pIter->zTab, "sqlite_stat1")==0);
3840e0d2096aSdan       pUp = (SessionUpdate*)sqlite3_malloc(nByte);
3841e0d2096aSdan       if( pUp==0 ){
3842e0d2096aSdan         rc = SQLITE_NOMEM;
3843e0d2096aSdan       }else{
3844e0d2096aSdan         const char *zSep = "";
3845e0d2096aSdan         SessionBuffer buf;
3846e0d2096aSdan 
3847e0d2096aSdan         memset(&buf, 0, sizeof(buf));
3848e0d2096aSdan         pUp->aMask = (u32*)&pUp[1];
3849e0d2096aSdan         memcpy(pUp->aMask, p->aUpdateMask, nU32*sizeof(u32));
3850e0d2096aSdan 
3851e0d2096aSdan         sessionAppendStr(&buf, "UPDATE main.", &rc);
3852e0d2096aSdan         sessionAppendIdent(&buf, pIter->zTab, &rc);
3853e0d2096aSdan         sessionAppendStr(&buf, " SET ", &rc);
3854e0d2096aSdan 
3855e0d2096aSdan         /* Create the assignments part of the UPDATE */
3856e0d2096aSdan         for(ii=0; ii<pIter->nCol; ii++){
3857e0d2096aSdan           if( p->abPK[ii]==0 && sessionChangesetNew(pIter, ii) ){
3858e0d2096aSdan             sessionAppendStr(&buf, zSep, &rc);
3859e0d2096aSdan             sessionAppendIdent(&buf, p->azCol[ii], &rc);
3860e0d2096aSdan             sessionAppendStr(&buf, " = ?", &rc);
3861e0d2096aSdan             sessionAppendInteger(&buf, ii*2+1, &rc);
3862e0d2096aSdan             zSep = ", ";
3863e0d2096aSdan           }
3864e0d2096aSdan         }
3865e0d2096aSdan 
3866e0d2096aSdan         /* Create the WHERE clause part of the UPDATE */
3867e0d2096aSdan         zSep = "";
3868e0d2096aSdan         sessionAppendStr(&buf, " WHERE ", &rc);
3869e0d2096aSdan         for(ii=0; ii<pIter->nCol; ii++){
3870e0d2096aSdan           if( p->abPK[ii] || (bPatchset==0 && sessionChangesetOld(pIter, ii)) ){
3871e0d2096aSdan             sessionAppendStr(&buf, zSep, &rc);
3872e0d2096aSdan             if( bStat1 && ii==1 ){
3873e0d2096aSdan               assert( sqlite3_stricmp(p->azCol[ii], "idx")==0 );
3874e0d2096aSdan               sessionAppendStr(&buf,
3875e0d2096aSdan                   "idx IS CASE "
3876e0d2096aSdan                   "WHEN length(?4)=0 AND typeof(?4)='blob' THEN NULL "
3877e0d2096aSdan                   "ELSE ?4 END ", &rc
3878e0d2096aSdan               );
3879e0d2096aSdan             }else{
3880e0d2096aSdan               sessionAppendIdent(&buf, p->azCol[ii], &rc);
3881e0d2096aSdan               sessionAppendStr(&buf, " IS ?", &rc);
3882e0d2096aSdan               sessionAppendInteger(&buf, ii*2+2, &rc);
3883e0d2096aSdan             }
3884e0d2096aSdan             zSep = " AND ";
3885e0d2096aSdan           }
3886e0d2096aSdan         }
3887e0d2096aSdan 
3888e0d2096aSdan         if( rc==SQLITE_OK ){
3889e0d2096aSdan           char *zSql = (char*)buf.aBuf;
3890e0d2096aSdan           rc = sqlite3_prepare_v2(p->db, zSql, buf.nBuf, &pUp->pStmt, 0);
3891e0d2096aSdan         }
3892e0d2096aSdan 
3893e0d2096aSdan         if( rc!=SQLITE_OK ){
3894e0d2096aSdan           sqlite3_free(pUp);
3895e0d2096aSdan           pUp = 0;
3896e0d2096aSdan         }else{
3897e0d2096aSdan           pUp->pNext = p->pUp;
3898e0d2096aSdan           p->pUp = pUp;
3899e0d2096aSdan         }
3900e0d2096aSdan         sqlite3_free(buf.aBuf);
3901e0d2096aSdan       }
3902e0d2096aSdan     }
3903e0d2096aSdan   }
3904e0d2096aSdan 
3905e0d2096aSdan   assert( (rc==SQLITE_OK)==(pUp!=0) );
3906e0d2096aSdan   if( pUp ){
3907e0d2096aSdan     *ppStmt = pUp->pStmt;
3908e0d2096aSdan   }else{
3909e0d2096aSdan     *ppStmt = 0;
3910e0d2096aSdan   }
3911e0d2096aSdan   return rc;
3912e0d2096aSdan }
3913e0d2096aSdan 
3914e0d2096aSdan /*
3915e0d2096aSdan ** Free all cached UPDATE statements.
3916e0d2096aSdan */
sessionUpdateFree(SessionApplyCtx * p)3917e0d2096aSdan static void sessionUpdateFree(SessionApplyCtx *p){
3918e0d2096aSdan   SessionUpdate *pUp;
3919e0d2096aSdan   SessionUpdate *pNext;
3920e0d2096aSdan   for(pUp=p->pUp; pUp; pUp=pNext){
3921e0d2096aSdan     pNext = pUp->pNext;
3922e0d2096aSdan     sqlite3_finalize(pUp->pStmt);
3923e0d2096aSdan     sqlite3_free(pUp);
3924e0d2096aSdan   }
3925e0d2096aSdan   p->pUp = 0;
3926e0d2096aSdan   sqlite3_free(p->aUpdateMask);
3927e0d2096aSdan   p->aUpdateMask = 0;
3928e0d2096aSdan }
3929e0d2096aSdan 
3930d5f0767cSdan /*
3931d5f0767cSdan ** Formulate a statement to DELETE a row from database db. Assuming a table
3932d5f0767cSdan ** structure like this:
3933d5f0767cSdan **
3934d5f0767cSdan **     CREATE TABLE x(a, b, c, d, PRIMARY KEY(a, c));
3935d5f0767cSdan **
3936d5f0767cSdan ** The DELETE statement looks like this:
3937d5f0767cSdan **
3938db04571cSdan **     DELETE FROM x WHERE a = :1 AND c = :3 AND (:5 OR b IS :2 AND d IS :4)
3939d5f0767cSdan **
3940d5f0767cSdan ** Variable :5 (nCol+1) is a boolean. It should be set to 0 if we require
3941d5f0767cSdan ** matching b and d values, or 1 otherwise. The second case comes up if the
3942d5f0767cSdan ** conflict handler is invoked with NOTFOUND and returns CHANGESET_REPLACE.
3943296c7658Sdan **
3944296c7658Sdan ** If successful, SQLITE_OK is returned and SessionApplyCtx.pDelete is left
3945296c7658Sdan ** pointing to the prepared version of the SQL statement.
3946d5f0767cSdan */
sessionDeleteRow(sqlite3 * db,const char * zTab,SessionApplyCtx * p)3947d5f0767cSdan static int sessionDeleteRow(
3948d5f0767cSdan   sqlite3 *db,                    /* Database handle */
3949d5f0767cSdan   const char *zTab,               /* Table name */
39500c698471Sdan   SessionApplyCtx *p              /* Session changeset-apply context */
3951d5f0767cSdan ){
3952296c7658Sdan   int i;
3953296c7658Sdan   const char *zSep = "";
3954d5f0767cSdan   int rc = SQLITE_OK;
3955d5f0767cSdan   SessionBuffer buf = {0, 0, 0};
39567cf7df7dSdan   int nPk = 0;
3957d5f0767cSdan 
39589e5ecdc1Sdan   sessionAppendStr(&buf, "DELETE FROM main.", &rc);
3959d5f0767cSdan   sessionAppendIdent(&buf, zTab, &rc);
3960296c7658Sdan   sessionAppendStr(&buf, " WHERE ", &rc);
3961296c7658Sdan 
3962296c7658Sdan   for(i=0; i<p->nCol; i++){
3963296c7658Sdan     if( p->abPK[i] ){
39647cf7df7dSdan       nPk++;
3965296c7658Sdan       sessionAppendStr(&buf, zSep, &rc);
3966296c7658Sdan       sessionAppendIdent(&buf, p->azCol[i], &rc);
3967296c7658Sdan       sessionAppendStr(&buf, " = ?", &rc);
3968296c7658Sdan       sessionAppendInteger(&buf, i+1, &rc);
3969296c7658Sdan       zSep = " AND ";
3970296c7658Sdan     }
3971296c7658Sdan   }
3972296c7658Sdan 
39737cf7df7dSdan   if( nPk<p->nCol ){
3974296c7658Sdan     sessionAppendStr(&buf, " AND (?", &rc);
3975296c7658Sdan     sessionAppendInteger(&buf, p->nCol+1, &rc);
3976296c7658Sdan     sessionAppendStr(&buf, " OR ", &rc);
3977296c7658Sdan 
3978296c7658Sdan     zSep = "";
3979296c7658Sdan     for(i=0; i<p->nCol; i++){
3980296c7658Sdan       if( !p->abPK[i] ){
3981296c7658Sdan         sessionAppendStr(&buf, zSep, &rc);
3982296c7658Sdan         sessionAppendIdent(&buf, p->azCol[i], &rc);
3983296c7658Sdan         sessionAppendStr(&buf, " IS ?", &rc);
3984296c7658Sdan         sessionAppendInteger(&buf, i+1, &rc);
3985296c7658Sdan         zSep = "AND ";
3986296c7658Sdan       }
3987296c7658Sdan     }
3988296c7658Sdan     sessionAppendStr(&buf, ")", &rc);
39897cf7df7dSdan   }
3990d5f0767cSdan 
3991d5f0767cSdan   if( rc==SQLITE_OK ){
39920c698471Sdan     rc = sqlite3_prepare_v2(db, (char *)buf.aBuf, buf.nBuf, &p->pDelete, 0);
3993d5f0767cSdan   }
3994d5f0767cSdan   sqlite3_free(buf.aBuf);
3995d5f0767cSdan 
3996d5f0767cSdan   return rc;
3997d5f0767cSdan }
3998d5f0767cSdan 
3999d5f0767cSdan /*
4000296c7658Sdan ** Formulate and prepare an SQL statement to query table zTab by primary
4001296c7658Sdan ** key. Assuming the following table structure:
4002296c7658Sdan **
4003296c7658Sdan **     CREATE TABLE x(a, b, c, d, PRIMARY KEY(a, c));
4004296c7658Sdan **
4005296c7658Sdan ** The SELECT statement looks like this:
4006296c7658Sdan **
4007296c7658Sdan **     SELECT * FROM x WHERE a = ?1 AND c = ?3
4008296c7658Sdan **
4009296c7658Sdan ** If successful, SQLITE_OK is returned and SessionApplyCtx.pSelect is left
4010296c7658Sdan ** pointing to the prepared version of the SQL statement.
4011296c7658Sdan */
sessionSelectRow(sqlite3 * db,const char * zTab,SessionApplyCtx * p)4012d5f0767cSdan static int sessionSelectRow(
4013d5f0767cSdan   sqlite3 *db,                    /* Database handle */
4014d5f0767cSdan   const char *zTab,               /* Table name */
40150c698471Sdan   SessionApplyCtx *p              /* Session changeset-apply context */
4016d5f0767cSdan ){
4017d7fb7d24Sdan   return sessionSelectStmt(
4018d7fb7d24Sdan       db, "main", zTab, p->nCol, p->azCol, p->abPK, &p->pSelect);
4019d5f0767cSdan }
4020d5f0767cSdan 
4021296c7658Sdan /*
4022296c7658Sdan ** Formulate and prepare an INSERT statement to add a record to table zTab.
4023296c7658Sdan ** For example:
4024296c7658Sdan **
4025296c7658Sdan **     INSERT INTO main."zTab" VALUES(?1, ?2, ?3 ...);
4026296c7658Sdan **
4027296c7658Sdan ** If successful, SQLITE_OK is returned and SessionApplyCtx.pInsert is left
4028296c7658Sdan ** pointing to the prepared version of the SQL statement.
4029296c7658Sdan */
sessionInsertRow(sqlite3 * db,const char * zTab,SessionApplyCtx * p)40300c698471Sdan static int sessionInsertRow(
40310c698471Sdan   sqlite3 *db,                    /* Database handle */
40320c698471Sdan   const char *zTab,               /* Table name */
40330c698471Sdan   SessionApplyCtx *p              /* Session changeset-apply context */
40340c698471Sdan ){
40350c698471Sdan   int rc = SQLITE_OK;
40360c698471Sdan   int i;
40370c698471Sdan   SessionBuffer buf = {0, 0, 0};
40380c698471Sdan 
40390c698471Sdan   sessionAppendStr(&buf, "INSERT INTO main.", &rc);
40400c698471Sdan   sessionAppendIdent(&buf, zTab, &rc);
4041ff677b20Sdan   sessionAppendStr(&buf, "(", &rc);
4042ff677b20Sdan   for(i=0; i<p->nCol; i++){
4043ff677b20Sdan     if( i!=0 ) sessionAppendStr(&buf, ", ", &rc);
4044ff677b20Sdan     sessionAppendIdent(&buf, p->azCol[i], &rc);
4045ff677b20Sdan   }
4046ff677b20Sdan 
4047ff677b20Sdan   sessionAppendStr(&buf, ") VALUES(?", &rc);
40480c698471Sdan   for(i=1; i<p->nCol; i++){
40490c698471Sdan     sessionAppendStr(&buf, ", ?", &rc);
40500c698471Sdan   }
40510c698471Sdan   sessionAppendStr(&buf, ")", &rc);
40520c698471Sdan 
40530c698471Sdan   if( rc==SQLITE_OK ){
40540c698471Sdan     rc = sqlite3_prepare_v2(db, (char *)buf.aBuf, buf.nBuf, &p->pInsert, 0);
40550c698471Sdan   }
40560c698471Sdan   sqlite3_free(buf.aBuf);
40570c698471Sdan   return rc;
40580c698471Sdan }
40590c698471Sdan 
sessionPrepare(sqlite3 * db,sqlite3_stmt ** pp,const char * zSql)40603739f298Sdan static int sessionPrepare(sqlite3 *db, sqlite3_stmt **pp, const char *zSql){
40613739f298Sdan   return sqlite3_prepare_v2(db, zSql, -1, pp, 0);
40623739f298Sdan }
40633739f298Sdan 
40643739f298Sdan /*
40653739f298Sdan ** Prepare statements for applying changes to the sqlite_stat1 table.
40663739f298Sdan ** These are similar to those created by sessionSelectRow(),
40673739f298Sdan ** sessionInsertRow(), sessionUpdateRow() and sessionDeleteRow() for
40683739f298Sdan ** other tables.
40693739f298Sdan */
sessionStat1Sql(sqlite3 * db,SessionApplyCtx * p)40703739f298Sdan static int sessionStat1Sql(sqlite3 *db, SessionApplyCtx *p){
40713739f298Sdan   int rc = sessionSelectRow(db, "sqlite_stat1", p);
40723739f298Sdan   if( rc==SQLITE_OK ){
40733739f298Sdan     rc = sessionPrepare(db, &p->pInsert,
40743739f298Sdan         "INSERT INTO main.sqlite_stat1 VALUES(?1, "
40753739f298Sdan         "CASE WHEN length(?2)=0 AND typeof(?2)='blob' THEN NULL ELSE ?2 END, "
40763739f298Sdan         "?3)"
40773739f298Sdan     );
40783739f298Sdan   }
40793739f298Sdan   if( rc==SQLITE_OK ){
40803739f298Sdan     rc = sessionPrepare(db, &p->pDelete,
40813739f298Sdan         "DELETE FROM main.sqlite_stat1 WHERE tbl=?1 AND idx IS "
40823739f298Sdan         "CASE WHEN length(?2)=0 AND typeof(?2)='blob' THEN NULL ELSE ?2 END "
40833739f298Sdan         "AND (?4 OR stat IS ?3)"
40843739f298Sdan     );
40853739f298Sdan   }
40863739f298Sdan   return rc;
40873739f298Sdan }
40883739f298Sdan 
4089296c7658Sdan /*
40907aa469cdSdan ** A wrapper around sqlite3_bind_value() that detects an extra problem.
40917aa469cdSdan ** See comments in the body of this function for details.
40927aa469cdSdan */
sessionBindValue(sqlite3_stmt * pStmt,int i,sqlite3_value * pVal)40937aa469cdSdan static int sessionBindValue(
40947aa469cdSdan   sqlite3_stmt *pStmt,            /* Statement to bind value to */
40957aa469cdSdan   int i,                          /* Parameter number to bind to */
40967aa469cdSdan   sqlite3_value *pVal             /* Value to bind */
40977aa469cdSdan ){
40985671ef69Sdrh   int eType = sqlite3_value_type(pVal);
4099082c96dfSdan   /* COVERAGE: The (pVal->z==0) branch is never true using current versions
4100082c96dfSdan   ** of SQLite. If a malloc fails in an sqlite3_value_xxx() function, either
4101082c96dfSdan   ** the (pVal->z) variable remains as it was or the type of the value is
4102082c96dfSdan   ** set to SQLITE_NULL.  */
41035671ef69Sdrh   if( (eType==SQLITE_TEXT || eType==SQLITE_BLOB) && pVal->z==0 ){
41047aa469cdSdan     /* This condition occurs when an earlier OOM in a call to
41057aa469cdSdan     ** sqlite3_value_text() or sqlite3_value_blob() (perhaps from within
4106082c96dfSdan     ** a conflict-handler) has zeroed the pVal->z pointer. Return NOMEM. */
41077aa469cdSdan     return SQLITE_NOMEM;
41087aa469cdSdan   }
41097aa469cdSdan   return sqlite3_bind_value(pStmt, i, pVal);
41107aa469cdSdan }
41117aa469cdSdan 
41127aa469cdSdan /*
4113db04571cSdan ** Iterator pIter must point to an SQLITE_INSERT entry. This function
4114db04571cSdan ** transfers new.* values from the current iterator entry to statement
4115db04571cSdan ** pStmt. The table being inserted into has nCol columns.
4116db04571cSdan **
4117d9151526Sdan ** New.* value $i from the iterator is bound to variable ($i+1) of
4118db04571cSdan ** statement pStmt. If parameter abPK is NULL, all values from 0 to (nCol-1)
4119db04571cSdan ** are transfered to the statement. Otherwise, if abPK is not NULL, it points
4120db04571cSdan ** to an array nCol elements in size. In this case only those values for
4121db04571cSdan ** which abPK[$i] is true are read from the iterator and bound to the
4122db04571cSdan ** statement.
4123db04571cSdan **
4124db04571cSdan ** An SQLite error code is returned if an error occurs. Otherwise, SQLITE_OK.
4125db04571cSdan */
sessionBindRow(sqlite3_changeset_iter * pIter,int (* xValue)(sqlite3_changeset_iter *,int,sqlite3_value **),int nCol,u8 * abPK,sqlite3_stmt * pStmt)41267aa469cdSdan static int sessionBindRow(
4127db04571cSdan   sqlite3_changeset_iter *pIter,  /* Iterator to read values from */
41287aa469cdSdan   int(*xValue)(sqlite3_changeset_iter *, int, sqlite3_value **),
4129db04571cSdan   int nCol,                       /* Number of columns */
4130db04571cSdan   u8 *abPK,                       /* If not NULL, bind only if true */
4131db04571cSdan   sqlite3_stmt *pStmt             /* Bind values to this statement */
4132db04571cSdan ){
4133db04571cSdan   int i;
4134db04571cSdan   int rc = SQLITE_OK;
41357aa469cdSdan 
41367aa469cdSdan   /* Neither sqlite3changeset_old or sqlite3changeset_new can fail if the
41377aa469cdSdan   ** argument iterator points to a suitable entry. Make sure that xValue
41387aa469cdSdan   ** is one of these to guarantee that it is safe to ignore the return
41397aa469cdSdan   ** in the code below. */
41407aa469cdSdan   assert( xValue==sqlite3changeset_old || xValue==sqlite3changeset_new );
41417aa469cdSdan 
4142db04571cSdan   for(i=0; rc==SQLITE_OK && i<nCol; i++){
4143db04571cSdan     if( !abPK || abPK[i] ){
414411a9ad56Sdrh       sqlite3_value *pVal = 0;
41457aa469cdSdan       (void)xValue(pIter, i, &pVal);
4146dd8a4af8Sdan       if( pVal==0 ){
4147dd8a4af8Sdan         /* The value in the changeset was "undefined". This indicates a
4148dd8a4af8Sdan         ** corrupt changeset blob.  */
4149e341ec69Sdan         rc = SQLITE_CORRUPT_BKPT;
4150dd8a4af8Sdan       }else{
41517aa469cdSdan         rc = sessionBindValue(pStmt, i+1, pVal);
4152db04571cSdan       }
4153db04571cSdan     }
4154dd8a4af8Sdan   }
4155db04571cSdan   return rc;
4156db04571cSdan }
4157db04571cSdan 
4158db04571cSdan /*
4159296c7658Sdan ** SQL statement pSelect is as generated by the sessionSelectRow() function.
4160296c7658Sdan ** This function binds the primary key values from the change that changeset
4161296c7658Sdan ** iterator pIter points to to the SELECT and attempts to seek to the table
4162296c7658Sdan ** entry. If a row is found, the SELECT statement left pointing at the row
4163296c7658Sdan ** and SQLITE_ROW is returned. Otherwise, if no row is found and no error
4164296c7658Sdan ** has occured, the statement is reset and SQLITE_OK is returned. If an
41657aa469cdSdan ** error occurs, the statement is reset and an SQLite error code is returned.
41667aa469cdSdan **
41677aa469cdSdan ** If this function returns SQLITE_ROW, the caller must eventually reset()
41687aa469cdSdan ** statement pSelect. If any other value is returned, the statement does
41697aa469cdSdan ** not require a reset().
4170296c7658Sdan **
4171296c7658Sdan ** If the iterator currently points to an INSERT record, bind values from the
4172db04571cSdan ** new.* record to the SELECT statement. Or, if it points to a DELETE or
4173db04571cSdan ** UPDATE, bind values from the old.* record.
4174296c7658Sdan */
sessionSeekToRow(sqlite3 * db,sqlite3_changeset_iter * pIter,u8 * abPK,sqlite3_stmt * pSelect)41750c698471Sdan static int sessionSeekToRow(
417637f133ecSdan   sqlite3 *db,                    /* Database handle */
417737f133ecSdan   sqlite3_changeset_iter *pIter,  /* Changeset iterator */
417837f133ecSdan   u8 *abPK,                       /* Primary key flags array */
41790c698471Sdan   sqlite3_stmt *pSelect           /* SELECT statement from sessionSelectRow() */
418037f133ecSdan ){
41817aa469cdSdan   int rc;                         /* Return code */
4182296c7658Sdan   int nCol;                       /* Number of columns in table */
4183296c7658Sdan   int op;                         /* Changset operation (SQLITE_UPDATE etc.) */
4184296c7658Sdan   const char *zDummy;             /* Unused */
418537f133ecSdan 
4186b4480e94Sdan   sqlite3changeset_op(pIter, &zDummy, &nCol, &op, 0);
41877aa469cdSdan   rc = sessionBindRow(pIter,
4188db04571cSdan       op==SQLITE_INSERT ? sqlite3changeset_new : sqlite3changeset_old,
4189db04571cSdan       nCol, abPK, pSelect
4190db04571cSdan   );
41910c698471Sdan 
41920c698471Sdan   if( rc==SQLITE_OK ){
41930c698471Sdan     rc = sqlite3_step(pSelect);
41940c698471Sdan     if( rc!=SQLITE_ROW ) rc = sqlite3_reset(pSelect);
41950c698471Sdan   }
41960c698471Sdan 
41970c698471Sdan   return rc;
41980c698471Sdan }
41990c698471Sdan 
4200b880a7b1Sdan /*
4201a920b209Sdrh ** This function is called from within sqlite3changeset_apply_v2() when
4202b880a7b1Sdan ** a conflict is encountered and resolved using conflict resolution
4203b880a7b1Sdan ** mode eType (either SQLITE_CHANGESET_OMIT or SQLITE_CHANGESET_REPLACE)..
4204b880a7b1Sdan ** It adds a conflict resolution record to the buffer in
4205b880a7b1Sdan ** SessionApplyCtx.rebase, which will eventually be returned to the caller
4206b880a7b1Sdan ** of apply_v2() as the "rebase" buffer.
4207b880a7b1Sdan **
4208b880a7b1Sdan ** Return SQLITE_OK if successful, or an SQLite error code otherwise.
4209b880a7b1Sdan */
sessionRebaseAdd(SessionApplyCtx * p,int eType,sqlite3_changeset_iter * pIter)4210a38e6c57Sdan static int sessionRebaseAdd(
4211b880a7b1Sdan   SessionApplyCtx *p,             /* Apply context */
4212b880a7b1Sdan   int eType,                      /* Conflict resolution (OMIT or REPLACE) */
4213b880a7b1Sdan   sqlite3_changeset_iter *pIter   /* Iterator pointing at current change */
4214a38e6c57Sdan ){
4215a38e6c57Sdan   int rc = SQLITE_OK;
4216dbe7d37aSdan   if( p->bRebase ){
4217a38e6c57Sdan     int i;
4218a38e6c57Sdan     int eOp = pIter->op;
4219a38e6c57Sdan     if( p->bRebaseStarted==0 ){
4220a38e6c57Sdan       /* Append a table-header to the rebase buffer */
4221a38e6c57Sdan       const char *zTab = pIter->zTab;
4222a38e6c57Sdan       sessionAppendByte(&p->rebase, 'T', &rc);
4223a38e6c57Sdan       sessionAppendVarint(&p->rebase, p->nCol, &rc);
4224a38e6c57Sdan       sessionAppendBlob(&p->rebase, p->abPK, p->nCol, &rc);
4225a38e6c57Sdan       sessionAppendBlob(&p->rebase, (u8*)zTab, (int)strlen(zTab)+1, &rc);
4226a38e6c57Sdan       p->bRebaseStarted = 1;
4227a38e6c57Sdan     }
4228a38e6c57Sdan 
4229a38e6c57Sdan     assert( eType==SQLITE_CHANGESET_REPLACE||eType==SQLITE_CHANGESET_OMIT );
4230a38e6c57Sdan     assert( eOp==SQLITE_DELETE || eOp==SQLITE_INSERT || eOp==SQLITE_UPDATE );
4231a38e6c57Sdan 
4232f01d3a7eSdan     sessionAppendByte(&p->rebase,
4233f01d3a7eSdan         (eOp==SQLITE_DELETE ? SQLITE_DELETE : SQLITE_INSERT), &rc
4234f01d3a7eSdan         );
4235f01d3a7eSdan     sessionAppendByte(&p->rebase, (eType==SQLITE_CHANGESET_REPLACE), &rc);
4236a38e6c57Sdan     for(i=0; i<p->nCol; i++){
4237a38e6c57Sdan       sqlite3_value *pVal = 0;
4238f01d3a7eSdan       if( eOp==SQLITE_DELETE || (eOp==SQLITE_UPDATE && p->abPK[i]) ){
4239a38e6c57Sdan         sqlite3changeset_old(pIter, i, &pVal);
4240a38e6c57Sdan       }else{
4241a38e6c57Sdan         sqlite3changeset_new(pIter, i, &pVal);
4242a38e6c57Sdan       }
4243a38e6c57Sdan       sessionAppendValue(&p->rebase, pVal, &rc);
4244a38e6c57Sdan     }
4245dbe7d37aSdan   }
4246a38e6c57Sdan   return rc;
4247a38e6c57Sdan }
4248a38e6c57Sdan 
4249296c7658Sdan /*
4250296c7658Sdan ** Invoke the conflict handler for the change that the changeset iterator
4251296c7658Sdan ** currently points to.
4252296c7658Sdan **
4253296c7658Sdan ** Argument eType must be either CHANGESET_DATA or CHANGESET_CONFLICT.
4254296c7658Sdan ** If argument pbReplace is NULL, then the type of conflict handler invoked
4255296c7658Sdan ** depends solely on eType, as follows:
4256296c7658Sdan **
4257296c7658Sdan **    eType value                 Value passed to xConflict
4258296c7658Sdan **    -------------------------------------------------
4259296c7658Sdan **    CHANGESET_DATA              CHANGESET_NOTFOUND
4260296c7658Sdan **    CHANGESET_CONFLICT          CHANGESET_CONSTRAINT
4261296c7658Sdan **
4262296c7658Sdan ** Or, if pbReplace is not NULL, then an attempt is made to find an existing
4263296c7658Sdan ** record with the same primary key as the record about to be deleted, updated
4264296c7658Sdan ** or inserted. If such a record can be found, it is available to the conflict
4265296c7658Sdan ** handler as the "conflicting" record. In this case the type of conflict
4266296c7658Sdan ** handler invoked is as follows:
4267296c7658Sdan **
4268296c7658Sdan **    eType value         PK Record found?   Value passed to xConflict
4269296c7658Sdan **    ----------------------------------------------------------------
4270296c7658Sdan **    CHANGESET_DATA      Yes                CHANGESET_DATA
4271296c7658Sdan **    CHANGESET_DATA      No                 CHANGESET_NOTFOUND
4272296c7658Sdan **    CHANGESET_CONFLICT  Yes                CHANGESET_CONFLICT
4273296c7658Sdan **    CHANGESET_CONFLICT  No                 CHANGESET_CONSTRAINT
4274296c7658Sdan **
4275296c7658Sdan ** If pbReplace is not NULL, and a record with a matching PK is found, and
4276296c7658Sdan ** the conflict handler function returns SQLITE_CHANGESET_REPLACE, *pbReplace
4277296c7658Sdan ** is set to non-zero before returning SQLITE_OK.
4278296c7658Sdan **
4279296c7658Sdan ** If the conflict handler returns SQLITE_CHANGESET_ABORT, SQLITE_ABORT is
4280296c7658Sdan ** returned. Or, if the conflict handler returns an invalid value,
4281296c7658Sdan ** SQLITE_MISUSE. If the conflict handler returns SQLITE_CHANGESET_OMIT,
4282296c7658Sdan ** this function returns SQLITE_OK.
4283296c7658Sdan */
sessionConflictHandler(int eType,SessionApplyCtx * p,sqlite3_changeset_iter * pIter,int (* xConflict)(void *,int,sqlite3_changeset_iter *),void * pCtx,int * pbReplace)42840c698471Sdan static int sessionConflictHandler(
4285296c7658Sdan   int eType,                      /* Either CHANGESET_DATA or CONFLICT */
4286296c7658Sdan   SessionApplyCtx *p,             /* changeset_apply() context */
42870c698471Sdan   sqlite3_changeset_iter *pIter,  /* Changeset iterator */
42880c698471Sdan   int(*xConflict)(void *, int, sqlite3_changeset_iter*),
4289296c7658Sdan   void *pCtx,                     /* First argument for conflict handler */
4290296c7658Sdan   int *pbReplace                  /* OUT: Set to true if PK row is found */
42910c698471Sdan ){
429274f598b6Smistachkin   int res = 0;                    /* Value returned by conflict handler */
42930c698471Sdan   int rc;
42940c698471Sdan   int nCol;
42950c698471Sdan   int op;
42960c698471Sdan   const char *zDummy;
42970c698471Sdan 
4298b4480e94Sdan   sqlite3changeset_op(pIter, &zDummy, &nCol, &op, 0);
42990c698471Sdan 
43000c698471Sdan   assert( eType==SQLITE_CHANGESET_CONFLICT || eType==SQLITE_CHANGESET_DATA );
43010c698471Sdan   assert( SQLITE_CHANGESET_CONFLICT+1==SQLITE_CHANGESET_CONSTRAINT );
43020c698471Sdan   assert( SQLITE_CHANGESET_DATA+1==SQLITE_CHANGESET_NOTFOUND );
430337f133ecSdan 
430437f133ecSdan   /* Bind the new.* PRIMARY KEY values to the SELECT statement. */
43050c698471Sdan   if( pbReplace ){
43060c698471Sdan     rc = sessionSeekToRow(p->db, pIter, p->abPK, p->pSelect);
43070c698471Sdan   }else{
4308db04571cSdan     rc = SQLITE_OK;
43090c698471Sdan   }
43100c698471Sdan 
43110c698471Sdan   if( rc==SQLITE_ROW ){
43120c698471Sdan     /* There exists another row with the new.* primary key. */
43130c698471Sdan     pIter->pConflict = p->pSelect;
43140c698471Sdan     res = xConflict(pCtx, eType, pIter);
43150c698471Sdan     pIter->pConflict = 0;
43160c698471Sdan     rc = sqlite3_reset(p->pSelect);
4317db04571cSdan   }else if( rc==SQLITE_OK ){
4318d9151526Sdan     if( p->bDeferConstraints && eType==SQLITE_CHANGESET_CONFLICT ){
4319d9151526Sdan       /* Instead of invoking the conflict handler, append the change blob
4320d9151526Sdan       ** to the SessionApplyCtx.constraints buffer. */
4321d9151526Sdan       u8 *aBlob = &pIter->in.aData[pIter->in.iCurrent];
4322d9151526Sdan       int nBlob = pIter->in.iNext - pIter->in.iCurrent;
4323d9151526Sdan       sessionAppendBlob(&p->constraints, aBlob, nBlob, &rc);
4324a38e6c57Sdan       return SQLITE_OK;
4325d9151526Sdan     }else{
43260c698471Sdan       /* No other row with the new.* primary key. */
43270c698471Sdan       res = xConflict(pCtx, eType+1, pIter);
43280c698471Sdan       if( res==SQLITE_CHANGESET_REPLACE ) rc = SQLITE_MISUSE;
432937f133ecSdan     }
4330d9151526Sdan   }
433137f133ecSdan 
433237f133ecSdan   if( rc==SQLITE_OK ){
43330c698471Sdan     switch( res ){
43340c698471Sdan       case SQLITE_CHANGESET_REPLACE:
4335f51e5f6cSdan         assert( pbReplace );
4336f51e5f6cSdan         *pbReplace = 1;
43370c698471Sdan         break;
43380c698471Sdan 
43390c698471Sdan       case SQLITE_CHANGESET_OMIT:
43400c698471Sdan         break;
43410c698471Sdan 
43420c698471Sdan       case SQLITE_CHANGESET_ABORT:
43430c698471Sdan         rc = SQLITE_ABORT;
43440c698471Sdan         break;
43450c698471Sdan 
43460c698471Sdan       default:
43470c698471Sdan         rc = SQLITE_MISUSE;
43480c698471Sdan         break;
43490c698471Sdan     }
4350a38e6c57Sdan     if( rc==SQLITE_OK ){
4351a38e6c57Sdan       rc = sessionRebaseAdd(p, res, pIter);
4352a38e6c57Sdan     }
43530c698471Sdan   }
43540c698471Sdan 
43550c698471Sdan   return rc;
43560c698471Sdan }
43570c698471Sdan 
4358296c7658Sdan /*
4359296c7658Sdan ** Attempt to apply the change that the iterator passed as the first argument
4360296c7658Sdan ** currently points to to the database. If a conflict is encountered, invoke
4361296c7658Sdan ** the conflict handler callback.
4362296c7658Sdan **
4363296c7658Sdan ** If argument pbRetry is NULL, then ignore any CHANGESET_DATA conflict. If
4364296c7658Sdan ** one is encountered, update or delete the row with the matching primary key
4365296c7658Sdan ** instead. Or, if pbRetry is not NULL and a CHANGESET_DATA conflict occurs,
4366296c7658Sdan ** invoke the conflict handler. If it returns CHANGESET_REPLACE, set *pbRetry
4367296c7658Sdan ** to true before returning. In this case the caller will invoke this function
4368296c7658Sdan ** again, this time with pbRetry set to NULL.
4369296c7658Sdan **
4370296c7658Sdan ** If argument pbReplace is NULL and a CHANGESET_CONFLICT conflict is
4371296c7658Sdan ** encountered invoke the conflict handler with CHANGESET_CONSTRAINT instead.
4372296c7658Sdan ** Or, if pbReplace is not NULL, invoke it with CHANGESET_CONFLICT. If such
4373296c7658Sdan ** an invocation returns SQLITE_CHANGESET_REPLACE, set *pbReplace to true
4374296c7658Sdan ** before retrying. In this case the caller attempts to remove the conflicting
4375296c7658Sdan ** row before invoking this function again, this time with pbReplace set
4376296c7658Sdan ** to NULL.
4377296c7658Sdan **
4378296c7658Sdan ** If any conflict handler returns SQLITE_CHANGESET_ABORT, this function
4379296c7658Sdan ** returns SQLITE_ABORT. Otherwise, if no error occurs, SQLITE_OK is
4380296c7658Sdan ** returned.
4381296c7658Sdan */
sessionApplyOneOp(sqlite3_changeset_iter * pIter,SessionApplyCtx * p,int (* xConflict)(void *,int,sqlite3_changeset_iter *),void * pCtx,int * pbReplace,int * pbRetry)43820c698471Sdan static int sessionApplyOneOp(
4383296c7658Sdan   sqlite3_changeset_iter *pIter,  /* Changeset iterator */
4384296c7658Sdan   SessionApplyCtx *p,             /* changeset_apply() context */
43850c698471Sdan   int(*xConflict)(void *, int, sqlite3_changeset_iter *),
4386296c7658Sdan   void *pCtx,                     /* First argument for the conflict handler */
4387296c7658Sdan   int *pbReplace,                 /* OUT: True to remove PK row and retry */
4388296c7658Sdan   int *pbRetry                    /* OUT: True to retry. */
43890c698471Sdan ){
43900c698471Sdan   const char *zDummy;
43910c698471Sdan   int op;
43920c698471Sdan   int nCol;
43930c698471Sdan   int rc = SQLITE_OK;
43940c698471Sdan 
4395e0d2096aSdan   assert( p->pDelete && p->pInsert && p->pSelect );
43960c698471Sdan   assert( p->azCol && p->abPK );
43970c698471Sdan   assert( !pbReplace || *pbReplace==0 );
43980c698471Sdan 
4399b4480e94Sdan   sqlite3changeset_op(pIter, &zDummy, &nCol, &op, 0);
44000c698471Sdan 
44010c698471Sdan   if( op==SQLITE_DELETE ){
44020c698471Sdan 
440373b3c055Sdan     /* Bind values to the DELETE statement. If conflict handling is required,
440473b3c055Sdan     ** bind values for all columns and set bound variable (nCol+1) to true.
440573b3c055Sdan     ** Or, if conflict handling is not required, bind just the PK column
440673b3c055Sdan     ** values and, if it exists, set (nCol+1) to false. Conflict handling
440773b3c055Sdan     ** is not required if:
440873b3c055Sdan     **
440973b3c055Sdan     **   * this is a patchset, or
441073b3c055Sdan     **   * (pbRetry==0), or
441173b3c055Sdan     **   * all columns of the table are PK columns (in this case there is
441273b3c055Sdan     **     no (nCol+1) variable to bind to).
441373b3c055Sdan     */
441473b3c055Sdan     u8 *abPK = (pIter->bPatchset ? p->abPK : 0);
441573b3c055Sdan     rc = sessionBindRow(pIter, sqlite3changeset_old, nCol, abPK, p->pDelete);
44167cf7df7dSdan     if( rc==SQLITE_OK && sqlite3_bind_parameter_count(p->pDelete)>nCol ){
441773b3c055Sdan       rc = sqlite3_bind_int(p->pDelete, nCol+1, (pbRetry==0 || abPK));
44187cf7df7dSdan     }
44190c698471Sdan     if( rc!=SQLITE_OK ) return rc;
44200c698471Sdan 
44210c698471Sdan     sqlite3_step(p->pDelete);
44220c698471Sdan     rc = sqlite3_reset(p->pDelete);
44230c698471Sdan     if( rc==SQLITE_OK && sqlite3_changes(p->db)==0 ){
44240c698471Sdan       rc = sessionConflictHandler(
44250c698471Sdan           SQLITE_CHANGESET_DATA, p, pIter, xConflict, pCtx, pbRetry
44260c698471Sdan       );
442735e2858eSdan     }else if( (rc&0xff)==SQLITE_CONSTRAINT ){
44280c698471Sdan       rc = sessionConflictHandler(
44290c698471Sdan           SQLITE_CHANGESET_CONFLICT, p, pIter, xConflict, pCtx, 0
44300c698471Sdan       );
44310c698471Sdan     }
44320c698471Sdan 
44330c698471Sdan   }else if( op==SQLITE_UPDATE ){
44340c698471Sdan     int i;
4435e0d2096aSdan     sqlite3_stmt *pUp = 0;
4436e0d2096aSdan     int bPatchset = (pbRetry==0 || pIter->bPatchset);
4437e0d2096aSdan 
4438e0d2096aSdan     rc = sessionUpdateFind(pIter, p, bPatchset, &pUp);
44390c698471Sdan 
44400c698471Sdan     /* Bind values to the UPDATE statement. */
44410c698471Sdan     for(i=0; rc==SQLITE_OK && i<nCol; i++){
44427aa469cdSdan       sqlite3_value *pOld = sessionChangesetOld(pIter, i);
44437aa469cdSdan       sqlite3_value *pNew = sessionChangesetNew(pIter, i);
4444e0d2096aSdan       if( p->abPK[i] || (bPatchset==0 && pOld) ){
4445e0d2096aSdan         rc = sessionBindValue(pUp, i*2+2, pOld);
44467aa469cdSdan       }
44477aa469cdSdan       if( rc==SQLITE_OK && pNew ){
4448e0d2096aSdan         rc = sessionBindValue(pUp, i*2+1, pNew);
44490c698471Sdan       }
44500c698471Sdan     }
44510c698471Sdan     if( rc!=SQLITE_OK ) return rc;
44520c698471Sdan 
44530c698471Sdan     /* Attempt the UPDATE. In the case of a NOTFOUND or DATA conflict,
44540c698471Sdan     ** the result will be SQLITE_OK with 0 rows modified. */
4455e0d2096aSdan     sqlite3_step(pUp);
4456e0d2096aSdan     rc = sqlite3_reset(pUp);
44570c698471Sdan 
44580c698471Sdan     if( rc==SQLITE_OK && sqlite3_changes(p->db)==0 ){
44590c698471Sdan       /* A NOTFOUND or DATA error. Search the table to see if it contains
44600c698471Sdan       ** a row with a matching primary key. If so, this is a DATA conflict.
44610c698471Sdan       ** Otherwise, if there is no primary key match, it is a NOTFOUND. */
44620c698471Sdan 
44630c698471Sdan       rc = sessionConflictHandler(
44640c698471Sdan           SQLITE_CHANGESET_DATA, p, pIter, xConflict, pCtx, pbRetry
44650c698471Sdan       );
44660c698471Sdan 
446735e2858eSdan     }else if( (rc&0xff)==SQLITE_CONSTRAINT ){
4468db04571cSdan       /* This is always a CONSTRAINT conflict. */
4469db04571cSdan       rc = sessionConflictHandler(
4470db04571cSdan           SQLITE_CHANGESET_CONFLICT, p, pIter, xConflict, pCtx, 0
44710c698471Sdan       );
44720c698471Sdan     }
44730c698471Sdan 
44740c698471Sdan   }else{
44750c698471Sdan     assert( op==SQLITE_INSERT );
4476d1cccf19Sdan     if( p->bStat1 ){
4477d1cccf19Sdan       /* Check if there is a conflicting row. For sqlite_stat1, this needs
4478d1cccf19Sdan       ** to be done using a SELECT, as there is no PRIMARY KEY in the
4479d1cccf19Sdan       ** database schema to throw an exception if a duplicate is inserted.  */
4480d1cccf19Sdan       rc = sessionSeekToRow(p->db, pIter, p->abPK, p->pSelect);
4481d1cccf19Sdan       if( rc==SQLITE_ROW ){
4482d1cccf19Sdan         rc = SQLITE_CONSTRAINT;
4483d1cccf19Sdan         sqlite3_reset(p->pSelect);
4484d1cccf19Sdan       }
4485d1cccf19Sdan     }
4486d1cccf19Sdan 
4487d1cccf19Sdan     if( rc==SQLITE_OK ){
44887aa469cdSdan       rc = sessionBindRow(pIter, sqlite3changeset_new, nCol, 0, p->pInsert);
44890c698471Sdan       if( rc!=SQLITE_OK ) return rc;
44900c698471Sdan 
44910c698471Sdan       sqlite3_step(p->pInsert);
44920c698471Sdan       rc = sqlite3_reset(p->pInsert);
4493d1cccf19Sdan     }
4494d1cccf19Sdan 
449535e2858eSdan     if( (rc&0xff)==SQLITE_CONSTRAINT ){
44960c698471Sdan       rc = sessionConflictHandler(
44970c698471Sdan           SQLITE_CHANGESET_CONFLICT, p, pIter, xConflict, pCtx, pbReplace
44980c698471Sdan       );
449937f133ecSdan     }
450037f133ecSdan   }
450137f133ecSdan 
450237f133ecSdan   return rc;
450337f133ecSdan }
450437f133ecSdan 
45055f5663dcSdan /*
45065f5663dcSdan ** Attempt to apply the change that the iterator passed as the first argument
45075f5663dcSdan ** currently points to to the database. If a conflict is encountered, invoke
45085f5663dcSdan ** the conflict handler callback.
45095f5663dcSdan **
45105f5663dcSdan ** The difference between this function and sessionApplyOne() is that this
45115f5663dcSdan ** function handles the case where the conflict-handler is invoked and
45125f5663dcSdan ** returns SQLITE_CHANGESET_REPLACE - indicating that the change should be
45135f5663dcSdan ** retried in some manner.
45145f5663dcSdan */
sessionApplyOneWithRetry(sqlite3 * db,sqlite3_changeset_iter * pIter,SessionApplyCtx * pApply,int (* xConflict)(void *,int,sqlite3_changeset_iter *),void * pCtx)4515d9151526Sdan static int sessionApplyOneWithRetry(
4516d9151526Sdan   sqlite3 *db,                    /* Apply change to "main" db of this handle */
4517d9151526Sdan   sqlite3_changeset_iter *pIter,  /* Changeset iterator to read change from */
4518d9151526Sdan   SessionApplyCtx *pApply,        /* Apply context */
4519d9151526Sdan   int(*xConflict)(void*, int, sqlite3_changeset_iter*),
4520d9151526Sdan   void *pCtx                      /* First argument passed to xConflict */
4521d9151526Sdan ){
4522d9151526Sdan   int bReplace = 0;
4523d9151526Sdan   int bRetry = 0;
4524d9151526Sdan   int rc;
4525d9151526Sdan 
4526d9151526Sdan   rc = sessionApplyOneOp(pIter, pApply, xConflict, pCtx, &bReplace, &bRetry);
4527a38e6c57Sdan   if( rc==SQLITE_OK ){
45285f5663dcSdan     /* If the bRetry flag is set, the change has not been applied due to an
45295f5663dcSdan     ** SQLITE_CHANGESET_DATA problem (i.e. this is an UPDATE or DELETE and
45305f5663dcSdan     ** a row with the correct PK is present in the db, but one or more other
45315f5663dcSdan     ** fields do not contain the expected values) and the conflict handler
45325f5663dcSdan     ** returned SQLITE_CHANGESET_REPLACE. In this case retry the operation,
45335f5663dcSdan     ** but pass NULL as the final argument so that sessionApplyOneOp() ignores
45345f5663dcSdan     ** the SQLITE_CHANGESET_DATA problem.  */
45355f5663dcSdan     if( bRetry ){
45365f5663dcSdan       assert( pIter->op==SQLITE_UPDATE || pIter->op==SQLITE_DELETE );
45375f5663dcSdan       rc = sessionApplyOneOp(pIter, pApply, xConflict, pCtx, 0, 0);
4538d9151526Sdan     }
4539d9151526Sdan 
45405f5663dcSdan     /* If the bReplace flag is set, the change is an INSERT that has not
45415f5663dcSdan     ** been performed because the database already contains a row with the
45425f5663dcSdan     ** specified primary key and the conflict handler returned
45435f5663dcSdan     ** SQLITE_CHANGESET_REPLACE. In this case remove the conflicting row
45445f5663dcSdan     ** before reattempting the INSERT.  */
45455f5663dcSdan     else if( bReplace ){
4546d9151526Sdan       assert( pIter->op==SQLITE_INSERT );
4547d9151526Sdan       rc = sqlite3_exec(db, "SAVEPOINT replace_op", 0, 0, 0);
4548d9151526Sdan       if( rc==SQLITE_OK ){
4549d9151526Sdan         rc = sessionBindRow(pIter,
4550d9151526Sdan             sqlite3changeset_new, pApply->nCol, pApply->abPK, pApply->pDelete);
4551d9151526Sdan         sqlite3_bind_int(pApply->pDelete, pApply->nCol+1, 1);
4552d9151526Sdan       }
4553d9151526Sdan       if( rc==SQLITE_OK ){
4554d9151526Sdan         sqlite3_step(pApply->pDelete);
4555d9151526Sdan         rc = sqlite3_reset(pApply->pDelete);
4556d9151526Sdan       }
4557d9151526Sdan       if( rc==SQLITE_OK ){
4558d9151526Sdan         rc = sessionApplyOneOp(pIter, pApply, xConflict, pCtx, 0, 0);
4559d9151526Sdan       }
4560d9151526Sdan       if( rc==SQLITE_OK ){
4561d9151526Sdan         rc = sqlite3_exec(db, "RELEASE replace_op", 0, 0, 0);
4562d9151526Sdan       }
4563d9151526Sdan     }
4564a38e6c57Sdan   }
4565d9151526Sdan 
4566d9151526Sdan   return rc;
4567d9151526Sdan }
4568d9151526Sdan 
4569d9151526Sdan /*
4570d9151526Sdan ** Retry the changes accumulated in the pApply->constraints buffer.
4571d9151526Sdan */
sessionRetryConstraints(sqlite3 * db,int bPatchset,const char * zTab,SessionApplyCtx * pApply,int (* xConflict)(void *,int,sqlite3_changeset_iter *),void * pCtx)4572d9151526Sdan static int sessionRetryConstraints(
4573d9151526Sdan   sqlite3 *db,
4574d9151526Sdan   int bPatchset,
4575d9151526Sdan   const char *zTab,
4576d9151526Sdan   SessionApplyCtx *pApply,
4577d9151526Sdan   int(*xConflict)(void*, int, sqlite3_changeset_iter*),
4578d9151526Sdan   void *pCtx                      /* First argument passed to xConflict */
4579d9151526Sdan ){
4580d9151526Sdan   int rc = SQLITE_OK;
4581d9151526Sdan 
4582d9151526Sdan   while( pApply->constraints.nBuf ){
4583d9151526Sdan     sqlite3_changeset_iter *pIter2 = 0;
4584d9151526Sdan     SessionBuffer cons = pApply->constraints;
4585d9151526Sdan     memset(&pApply->constraints, 0, sizeof(SessionBuffer));
4586d9151526Sdan 
45875d237bfaSdan     rc = sessionChangesetStart(
45881e25d20cSdan         &pIter2, 0, 0, cons.nBuf, cons.aBuf, pApply->bInvertConstraints, 1
45895d237bfaSdan     );
4590d9151526Sdan     if( rc==SQLITE_OK ){
4591f6ad201aSdrh       size_t nByte = 2*pApply->nCol*sizeof(sqlite3_value*);
4592d9151526Sdan       int rc2;
4593d9151526Sdan       pIter2->bPatchset = bPatchset;
4594d9151526Sdan       pIter2->zTab = (char*)zTab;
4595d9151526Sdan       pIter2->nCol = pApply->nCol;
4596d9151526Sdan       pIter2->abPK = pApply->abPK;
4597d9151526Sdan       sessionBufferGrow(&pIter2->tblhdr, nByte, &rc);
4598d9151526Sdan       pIter2->apValue = (sqlite3_value**)pIter2->tblhdr.aBuf;
4599d9151526Sdan       if( rc==SQLITE_OK ) memset(pIter2->apValue, 0, nByte);
4600d9151526Sdan 
4601d9151526Sdan       while( rc==SQLITE_OK && SQLITE_ROW==sqlite3changeset_next(pIter2) ){
4602d9151526Sdan         rc = sessionApplyOneWithRetry(db, pIter2, pApply, xConflict, pCtx);
4603d9151526Sdan       }
4604d9151526Sdan 
4605d9151526Sdan       rc2 = sqlite3changeset_finalize(pIter2);
46067e0765a9Sdrh       if( rc==SQLITE_OK ) rc = rc2;
4607d9151526Sdan     }
4608d9151526Sdan     assert( pApply->bDeferConstraints || pApply->constraints.nBuf==0 );
4609d9151526Sdan 
4610d9151526Sdan     sqlite3_free(cons.aBuf);
4611d9151526Sdan     if( rc!=SQLITE_OK ) break;
4612d9151526Sdan     if( pApply->constraints.nBuf>=cons.nBuf ){
4613d9151526Sdan       /* No progress was made on the last round. */
4614d9151526Sdan       pApply->bDeferConstraints = 0;
4615d9151526Sdan     }
4616d9151526Sdan   }
4617d9151526Sdan 
4618d9151526Sdan   return rc;
4619d9151526Sdan }
4620d9151526Sdan 
4621296c7658Sdan /*
46224757c658Sdan ** Argument pIter is a changeset iterator that has been initialized, but
46234757c658Sdan ** not yet passed to sqlite3changeset_next(). This function applies the
46244757c658Sdan ** changeset to the main database attached to handle "db". The supplied
46254757c658Sdan ** conflict handler callback is invoked to resolve any conflicts encountered
46264757c658Sdan ** while applying the change.
4627296c7658Sdan */
sessionChangesetApply(sqlite3 * db,sqlite3_changeset_iter * pIter,int (* xFilter)(void * pCtx,const char * zTab),int (* xConflict)(void * pCtx,int eConflict,sqlite3_changeset_iter * p),void * pCtx,void ** ppRebase,int * pnRebase,int flags)46284757c658Sdan static int sessionChangesetApply(
4629296c7658Sdan   sqlite3 *db,                    /* Apply change to "main" db of this handle */
46304757c658Sdan   sqlite3_changeset_iter *pIter,  /* Changeset to apply */
463140368988Sdan   int(*xFilter)(
463240368988Sdan     void *pCtx,                   /* Copy of sixth arg to _apply() */
463340368988Sdan     const char *zTab              /* Table name */
463440368988Sdan   ),
4635d5f0767cSdan   int(*xConflict)(
4636d5f0767cSdan     void *pCtx,                   /* Copy of fifth arg to _apply() */
4637d5f0767cSdan     int eConflict,                /* DATA, MISSING, CONFLICT, CONSTRAINT */
4638d5f0767cSdan     sqlite3_changeset_iter *p     /* Handle describing change and conflict */
4639d5f0767cSdan   ),
4640a38e6c57Sdan   void *pCtx,                     /* First argument passed to xConflict */
4641fe55da38Sdan   void **ppRebase, int *pnRebase, /* OUT: Rebase information */
4642fe55da38Sdan   int flags                       /* SESSION_APPLY_XXX flags */
4643d5f0767cSdan ){
4644ca62ad57Sdan   int schemaMismatch = 0;
4645fe55da38Sdan   int rc = SQLITE_OK;             /* Return code */
4646d5f0767cSdan   const char *zTab = 0;           /* Name of current table */
4647cfdbde21Sdrh   int nTab = 0;                   /* Result of sqlite3Strlen30(zTab) */
4648296c7658Sdan   SessionApplyCtx sApply;         /* changeset_apply() context object */
46495f5663dcSdan   int bPatchset;
4650d5f0767cSdan 
4651082c96dfSdan   assert( xConflict!=0 );
4652082c96dfSdan 
4653d9151526Sdan   pIter->in.bNoDiscard = 1;
46540c698471Sdan   memset(&sApply, 0, sizeof(sApply));
4655dbe7d37aSdan   sApply.bRebase = (ppRebase && pnRebase);
46565d237bfaSdan   sApply.bInvertConstraints = !!(flags & SQLITE_CHANGESETAPPLY_INVERT);
46574c220252Sdan   sqlite3_mutex_enter(sqlite3_db_mutex(db));
4658fe55da38Sdan   if( (flags & SQLITE_CHANGESETAPPLY_NOSAVEPOINT)==0 ){
46590c698471Sdan     rc = sqlite3_exec(db, "SAVEPOINT changeset_apply", 0, 0, 0);
4660fe55da38Sdan   }
4661cb3e4b79Sdan   if( rc==SQLITE_OK ){
4662cb3e4b79Sdan     rc = sqlite3_exec(db, "PRAGMA defer_foreign_keys = 1", 0, 0, 0);
4663cb3e4b79Sdan   }
46640c698471Sdan   while( rc==SQLITE_OK && SQLITE_ROW==sqlite3changeset_next(pIter) ){
46650c698471Sdan     int nCol;
4666d5f0767cSdan     int op;
46670c698471Sdan     const char *zNew;
4668ca62ad57Sdan 
4669b4480e94Sdan     sqlite3changeset_op(pIter, &zNew, &nCol, &op, 0);
4670d5f0767cSdan 
46710c698471Sdan     if( zTab==0 || sqlite3_strnicmp(zNew, zTab, nTab+1) ){
4672ca62ad57Sdan       u8 *abPK;
4673ca62ad57Sdan 
4674d9151526Sdan       rc = sessionRetryConstraints(
4675d9151526Sdan           db, pIter->bPatchset, zTab, &sApply, xConflict, pCtx
4676d9151526Sdan       );
4677d9151526Sdan       if( rc!=SQLITE_OK ) break;
4678d9151526Sdan 
4679e0d2096aSdan       sessionUpdateFree(&sApply);
4680cfdbde21Sdrh       sqlite3_free((char*)sApply.azCol);  /* cast works around VC++ bug */
46810c698471Sdan       sqlite3_finalize(sApply.pDelete);
46820c698471Sdan       sqlite3_finalize(sApply.pInsert);
46830c698471Sdan       sqlite3_finalize(sApply.pSelect);
46840c698471Sdan       sApply.db = db;
4685f1b40e83Sdan       sApply.pDelete = 0;
4686f1b40e83Sdan       sApply.pInsert = 0;
4687f1b40e83Sdan       sApply.pSelect = 0;
4688f1b40e83Sdan       sApply.nCol = 0;
4689f1b40e83Sdan       sApply.azCol = 0;
4690f1b40e83Sdan       sApply.abPK = 0;
4691f1b40e83Sdan       sApply.bStat1 = 0;
4692d9151526Sdan       sApply.bDeferConstraints = 1;
4693a38e6c57Sdan       sApply.bRebaseStarted = 0;
4694f1b40e83Sdan       memset(&sApply.constraints, 0, sizeof(SessionBuffer));
469537f133ecSdan 
469640368988Sdan       /* If an xFilter() callback was specified, invoke it now. If the
469740368988Sdan       ** xFilter callback returns zero, skip this table. If it returns
469840368988Sdan       ** non-zero, proceed. */
469940368988Sdan       schemaMismatch = (xFilter && (0==xFilter(pCtx, zNew)));
470040368988Sdan       if( schemaMismatch ){
470140368988Sdan         zTab = sqlite3_mprintf("%s", zNew);
4702f05ac112Sdan         if( zTab==0 ){
4703f05ac112Sdan           rc = SQLITE_NOMEM;
4704f05ac112Sdan           break;
4705f05ac112Sdan         }
47064f528042Sdan         nTab = (int)strlen(zTab);
470740368988Sdan         sApply.azCol = (const char **)zTab;
470840368988Sdan       }else{
4709ff677b20Sdan         int nMinCol = 0;
4710ff677b20Sdan         int i;
4711ff677b20Sdan 
4712ca62ad57Sdan         sqlite3changeset_pk(pIter, &abPK, 0);
47130cb735b9Sdan         rc = sessionTableInfo(0,
4714ca62ad57Sdan             db, "main", zNew, &sApply.nCol, &zTab, &sApply.azCol, &sApply.abPK
4715ca62ad57Sdan         );
4716ca62ad57Sdan         if( rc!=SQLITE_OK ) break;
4717ff677b20Sdan         for(i=0; i<sApply.nCol; i++){
4718ff677b20Sdan           if( sApply.abPK[i] ) nMinCol = i+1;
4719ff677b20Sdan         }
47200c698471Sdan 
4721ca62ad57Sdan         if( sApply.nCol==0 ){
4722ca62ad57Sdan           schemaMismatch = 1;
4723ca62ad57Sdan           sqlite3_log(SQLITE_SCHEMA,
4724ca62ad57Sdan               "sqlite3changeset_apply(): no such table: %s", zTab
4725ca62ad57Sdan           );
4726ca62ad57Sdan         }
4727ff677b20Sdan         else if( sApply.nCol<nCol ){
4728ca62ad57Sdan           schemaMismatch = 1;
4729ca62ad57Sdan           sqlite3_log(SQLITE_SCHEMA,
4730ff677b20Sdan               "sqlite3changeset_apply(): table %s has %d columns, "
4731ff677b20Sdan               "expected %d or more",
4732ca62ad57Sdan               zTab, sApply.nCol, nCol
4733ca62ad57Sdan           );
4734ca62ad57Sdan         }
4735ff677b20Sdan         else if( nCol<nMinCol || memcmp(sApply.abPK, abPK, nCol)!=0 ){
4736ca62ad57Sdan           schemaMismatch = 1;
473740368988Sdan           sqlite3_log(SQLITE_SCHEMA, "sqlite3changeset_apply(): "
473840368988Sdan               "primary key mismatch for table %s", zTab
4739ca62ad57Sdan           );
4740ca62ad57Sdan         }
4741ff677b20Sdan         else{
4742ff677b20Sdan           sApply.nCol = nCol;
47433739f298Sdan           if( 0==sqlite3_stricmp(zTab, "sqlite_stat1") ){
47443739f298Sdan             if( (rc = sessionStat1Sql(db, &sApply) ) ){
47453739f298Sdan               break;
47463739f298Sdan             }
4747d1cccf19Sdan             sApply.bStat1 = 1;
4748d1cccf19Sdan           }else{
4749ff677b20Sdan             if( (rc = sessionSelectRow(db, zTab, &sApply))
47500c698471Sdan              || (rc = sessionDeleteRow(db, zTab, &sApply))
47510c698471Sdan              || (rc = sessionInsertRow(db, zTab, &sApply))
475237f133ecSdan             ){
475337f133ecSdan               break;
475437f133ecSdan             }
4755d1cccf19Sdan             sApply.bStat1 = 0;
4756d1cccf19Sdan           }
4757ff677b20Sdan         }
4758cfdbde21Sdrh         nTab = sqlite3Strlen30(zTab);
4759d5f0767cSdan       }
476040368988Sdan     }
4761d5f0767cSdan 
4762ca62ad57Sdan     /* If there is a schema mismatch on the current table, proceed to the
4763ca62ad57Sdan     ** next change. A log message has already been issued. */
4764ca62ad57Sdan     if( schemaMismatch ) continue;
4765ca62ad57Sdan 
4766d9151526Sdan     rc = sessionApplyOneWithRetry(db, pIter, &sApply, xConflict, pCtx);
47670c698471Sdan   }
47680c698471Sdan 
47695f5663dcSdan   bPatchset = pIter->bPatchset;
4770296c7658Sdan   if( rc==SQLITE_OK ){
4771296c7658Sdan     rc = sqlite3changeset_finalize(pIter);
4772296c7658Sdan   }else{
4773296c7658Sdan     sqlite3changeset_finalize(pIter);
4774296c7658Sdan   }
4775d5f0767cSdan 
4776d5f0767cSdan   if( rc==SQLITE_OK ){
47775f5663dcSdan     rc = sessionRetryConstraints(db, bPatchset, zTab, &sApply, xConflict, pCtx);
47785f5663dcSdan   }
47795f5663dcSdan 
47805f5663dcSdan   if( rc==SQLITE_OK ){
478107001c45Sdrh     int nFk, notUsed;
478207001c45Sdrh     sqlite3_db_status(db, SQLITE_DBSTATUS_DEFERRED_FKS, &nFk, &notUsed, 0);
478307001c45Sdrh     if( nFk!=0 ){
4784cb3e4b79Sdan       int res = SQLITE_CHANGESET_ABORT;
4785cb3e4b79Sdan       sqlite3_changeset_iter sIter;
4786cb3e4b79Sdan       memset(&sIter, 0, sizeof(sIter));
4787cb3e4b79Sdan       sIter.nCol = nFk;
4788cb3e4b79Sdan       res = xConflict(pCtx, SQLITE_CHANGESET_FOREIGN_KEY, &sIter);
4789cb3e4b79Sdan       if( res!=SQLITE_CHANGESET_OMIT ){
4790cb3e4b79Sdan         rc = SQLITE_CONSTRAINT;
4791cb3e4b79Sdan       }
4792cb3e4b79Sdan     }
4793cb3e4b79Sdan   }
4794cb3e4b79Sdan   sqlite3_exec(db, "PRAGMA defer_foreign_keys = 0", 0, 0, 0);
4795cb3e4b79Sdan 
4796fe55da38Sdan   if( (flags & SQLITE_CHANGESETAPPLY_NOSAVEPOINT)==0 ){
4797cb3e4b79Sdan     if( rc==SQLITE_OK ){
4798d5f0767cSdan       rc = sqlite3_exec(db, "RELEASE changeset_apply", 0, 0, 0);
4799d5f0767cSdan     }else{
4800d5f0767cSdan       sqlite3_exec(db, "ROLLBACK TO changeset_apply", 0, 0, 0);
4801d5f0767cSdan       sqlite3_exec(db, "RELEASE changeset_apply", 0, 0, 0);
4802d5f0767cSdan     }
4803fe55da38Sdan   }
4804d5f0767cSdan 
4805dbe7d37aSdan   assert( sApply.bRebase || sApply.rebase.nBuf==0 );
4806dbe7d37aSdan   if( rc==SQLITE_OK && bPatchset==0 && sApply.bRebase ){
4807a38e6c57Sdan     *ppRebase = (void*)sApply.rebase.aBuf;
4808a38e6c57Sdan     *pnRebase = sApply.rebase.nBuf;
4809a38e6c57Sdan     sApply.rebase.aBuf = 0;
4810a38e6c57Sdan   }
4811e0d2096aSdan   sessionUpdateFree(&sApply);
48120c698471Sdan   sqlite3_finalize(sApply.pInsert);
48130c698471Sdan   sqlite3_finalize(sApply.pDelete);
48140c698471Sdan   sqlite3_finalize(sApply.pSelect);
4815cfdbde21Sdrh   sqlite3_free((char*)sApply.azCol);  /* cast works around VC++ bug */
4816d9151526Sdan   sqlite3_free((char*)sApply.constraints.aBuf);
4817a38e6c57Sdan   sqlite3_free((char*)sApply.rebase.aBuf);
48184c220252Sdan   sqlite3_mutex_leave(sqlite3_db_mutex(db));
4819d5f0767cSdan   return rc;
4820d5f0767cSdan }
482191ddd559Sdan 
4822b880a7b1Sdan /*
4823b880a7b1Sdan ** Apply the changeset passed via pChangeset/nChangeset to the main
4824b880a7b1Sdan ** database attached to handle "db".
4825b880a7b1Sdan */
sqlite3changeset_apply_v2(sqlite3 * db,int nChangeset,void * pChangeset,int (* xFilter)(void * pCtx,const char * zTab),int (* xConflict)(void * pCtx,int eConflict,sqlite3_changeset_iter * p),void * pCtx,void ** ppRebase,int * pnRebase,int flags)4826a38e6c57Sdan int sqlite3changeset_apply_v2(
4827a38e6c57Sdan   sqlite3 *db,                    /* Apply change to "main" db of this handle */
4828a38e6c57Sdan   int nChangeset,                 /* Size of changeset in bytes */
4829a38e6c57Sdan   void *pChangeset,               /* Changeset blob */
4830a38e6c57Sdan   int(*xFilter)(
4831a38e6c57Sdan     void *pCtx,                   /* Copy of sixth arg to _apply() */
4832a38e6c57Sdan     const char *zTab              /* Table name */
4833a38e6c57Sdan   ),
4834a38e6c57Sdan   int(*xConflict)(
4835a38e6c57Sdan     void *pCtx,                   /* Copy of sixth arg to _apply() */
4836a38e6c57Sdan     int eConflict,                /* DATA, MISSING, CONFLICT, CONSTRAINT */
4837a38e6c57Sdan     sqlite3_changeset_iter *p     /* Handle describing change and conflict */
4838a38e6c57Sdan   ),
4839a38e6c57Sdan   void *pCtx,                     /* First argument passed to xConflict */
4840fe55da38Sdan   void **ppRebase, int *pnRebase,
4841fe55da38Sdan   int flags
4842a38e6c57Sdan ){
4843a38e6c57Sdan   sqlite3_changeset_iter *pIter;  /* Iterator to skip through changeset */
48441e25d20cSdan   int bInv = !!(flags & SQLITE_CHANGESETAPPLY_INVERT);
48451e25d20cSdan   int rc = sessionChangesetStart(&pIter, 0, 0, nChangeset, pChangeset, bInv, 1);
4846a38e6c57Sdan   if( rc==SQLITE_OK ){
4847a38e6c57Sdan     rc = sessionChangesetApply(
4848fe55da38Sdan         db, pIter, xFilter, xConflict, pCtx, ppRebase, pnRebase, flags
4849a38e6c57Sdan     );
4850a38e6c57Sdan   }
4851a38e6c57Sdan   return rc;
4852a38e6c57Sdan }
4853a38e6c57Sdan 
485477fc1d5bSdan /*
48554757c658Sdan ** Apply the changeset passed via pChangeset/nChangeset to the main database
48564757c658Sdan ** attached to handle "db". Invoke the supplied conflict handler callback
48574757c658Sdan ** to resolve any conflicts encountered while applying the change.
48584757c658Sdan */
sqlite3changeset_apply(sqlite3 * db,int nChangeset,void * pChangeset,int (* xFilter)(void * pCtx,const char * zTab),int (* xConflict)(void * pCtx,int eConflict,sqlite3_changeset_iter * p),void * pCtx)48594757c658Sdan int sqlite3changeset_apply(
48604757c658Sdan   sqlite3 *db,                    /* Apply change to "main" db of this handle */
48614757c658Sdan   int nChangeset,                 /* Size of changeset in bytes */
48624757c658Sdan   void *pChangeset,               /* Changeset blob */
48634757c658Sdan   int(*xFilter)(
48644757c658Sdan     void *pCtx,                   /* Copy of sixth arg to _apply() */
48654757c658Sdan     const char *zTab              /* Table name */
48664757c658Sdan   ),
48674757c658Sdan   int(*xConflict)(
48684757c658Sdan     void *pCtx,                   /* Copy of fifth arg to _apply() */
48694757c658Sdan     int eConflict,                /* DATA, MISSING, CONFLICT, CONSTRAINT */
48704757c658Sdan     sqlite3_changeset_iter *p     /* Handle describing change and conflict */
48714757c658Sdan   ),
48724757c658Sdan   void *pCtx                      /* First argument passed to xConflict */
48734757c658Sdan ){
4874a38e6c57Sdan   return sqlite3changeset_apply_v2(
4875fe55da38Sdan       db, nChangeset, pChangeset, xFilter, xConflict, pCtx, 0, 0, 0
4876a38e6c57Sdan   );
48774757c658Sdan }
48784757c658Sdan 
48794757c658Sdan /*
48804757c658Sdan ** Apply the changeset passed via xInput/pIn to the main database
48814757c658Sdan ** attached to handle "db". Invoke the supplied conflict handler callback
48824757c658Sdan ** to resolve any conflicts encountered while applying the change.
48834757c658Sdan */
sqlite3changeset_apply_v2_strm(sqlite3 * db,int (* xInput)(void * pIn,void * pData,int * pnData),void * pIn,int (* xFilter)(void * pCtx,const char * zTab),int (* xConflict)(void * pCtx,int eConflict,sqlite3_changeset_iter * p),void * pCtx,void ** ppRebase,int * pnRebase,int flags)4884a38e6c57Sdan int sqlite3changeset_apply_v2_strm(
4885a38e6c57Sdan   sqlite3 *db,                    /* Apply change to "main" db of this handle */
4886a38e6c57Sdan   int (*xInput)(void *pIn, void *pData, int *pnData), /* Input function */
4887a38e6c57Sdan   void *pIn,                                          /* First arg for xInput */
4888a38e6c57Sdan   int(*xFilter)(
4889a38e6c57Sdan     void *pCtx,                   /* Copy of sixth arg to _apply() */
4890a38e6c57Sdan     const char *zTab              /* Table name */
4891a38e6c57Sdan   ),
4892a38e6c57Sdan   int(*xConflict)(
4893a38e6c57Sdan     void *pCtx,                   /* Copy of sixth arg to _apply() */
4894a38e6c57Sdan     int eConflict,                /* DATA, MISSING, CONFLICT, CONSTRAINT */
4895a38e6c57Sdan     sqlite3_changeset_iter *p     /* Handle describing change and conflict */
4896a38e6c57Sdan   ),
4897a38e6c57Sdan   void *pCtx,                     /* First argument passed to xConflict */
4898fe55da38Sdan   void **ppRebase, int *pnRebase,
4899fe55da38Sdan   int flags
4900a38e6c57Sdan ){
4901a38e6c57Sdan   sqlite3_changeset_iter *pIter;  /* Iterator to skip through changeset */
490244748f27Sdan   int bInverse = !!(flags & SQLITE_CHANGESETAPPLY_INVERT);
49031e25d20cSdan   int rc = sessionChangesetStart(&pIter, xInput, pIn, 0, 0, bInverse, 1);
4904a38e6c57Sdan   if( rc==SQLITE_OK ){
4905a38e6c57Sdan     rc = sessionChangesetApply(
4906fe55da38Sdan         db, pIter, xFilter, xConflict, pCtx, ppRebase, pnRebase, flags
4907a38e6c57Sdan     );
4908a38e6c57Sdan   }
4909a38e6c57Sdan   return rc;
4910a38e6c57Sdan }
sqlite3changeset_apply_strm(sqlite3 * db,int (* xInput)(void * pIn,void * pData,int * pnData),void * pIn,int (* xFilter)(void * pCtx,const char * zTab),int (* xConflict)(void * pCtx,int eConflict,sqlite3_changeset_iter * p),void * pCtx)4911f1a08ad8Sdrh int sqlite3changeset_apply_strm(
49124757c658Sdan   sqlite3 *db,                    /* Apply change to "main" db of this handle */
49134757c658Sdan   int (*xInput)(void *pIn, void *pData, int *pnData), /* Input function */
49144757c658Sdan   void *pIn,                                          /* First arg for xInput */
49154757c658Sdan   int(*xFilter)(
49164757c658Sdan     void *pCtx,                   /* Copy of sixth arg to _apply() */
49174757c658Sdan     const char *zTab              /* Table name */
49184757c658Sdan   ),
49194757c658Sdan   int(*xConflict)(
49204757c658Sdan     void *pCtx,                   /* Copy of sixth arg to _apply() */
49214757c658Sdan     int eConflict,                /* DATA, MISSING, CONFLICT, CONSTRAINT */
49224757c658Sdan     sqlite3_changeset_iter *p     /* Handle describing change and conflict */
49234757c658Sdan   ),
49244757c658Sdan   void *pCtx                      /* First argument passed to xConflict */
49254757c658Sdan ){
4926a38e6c57Sdan   return sqlite3changeset_apply_v2_strm(
4927fe55da38Sdan       db, xInput, pIn, xFilter, xConflict, pCtx, 0, 0, 0
4928a38e6c57Sdan   );
49294757c658Sdan }
49304757c658Sdan 
49314757c658Sdan /*
49325898ad69Sdan ** sqlite3_changegroup handle.
49335898ad69Sdan */
49345898ad69Sdan struct sqlite3_changegroup {
49355898ad69Sdan   int rc;                         /* Error code */
49365898ad69Sdan   int bPatch;                     /* True to accumulate patchsets */
49375898ad69Sdan   SessionTable *pList;            /* List of tables in current patch */
49385898ad69Sdan };
49395898ad69Sdan 
49405898ad69Sdan /*
494177fc1d5bSdan ** This function is called to merge two changes to the same row together as
494277fc1d5bSdan ** part of an sqlite3changeset_concat() operation. A new change object is
494377fc1d5bSdan ** allocated and a pointer to it stored in *ppNew.
494477fc1d5bSdan */
sessionChangeMerge(SessionTable * pTab,int bRebase,int bPatchset,SessionChange * pExist,int op2,int bIndirect,u8 * aRec,int nRec,SessionChange ** ppNew)49455d607a6eSdan static int sessionChangeMerge(
494677fc1d5bSdan   SessionTable *pTab,             /* Table structure */
4947c0a499eaSdan   int bRebase,                    /* True for a rebase hash-table */
494864277f4aSdan   int bPatchset,                  /* True for patchsets */
494977fc1d5bSdan   SessionChange *pExist,          /* Existing change */
495077fc1d5bSdan   int op2,                        /* Second change operation */
495177fc1d5bSdan   int bIndirect,                  /* True if second change is indirect */
495277fc1d5bSdan   u8 *aRec,                       /* Second change record */
495377fc1d5bSdan   int nRec,                       /* Number of bytes in aRec */
495477fc1d5bSdan   SessionChange **ppNew           /* OUT: Merged change */
49555d607a6eSdan ){
49565d607a6eSdan   SessionChange *pNew = 0;
495724a0c453Sdan   int rc = SQLITE_OK;
49585d607a6eSdan 
49595d607a6eSdan   if( !pExist ){
49602d77d80aSdrh     pNew = (SessionChange *)sqlite3_malloc64(sizeof(SessionChange) + nRec);
49615d607a6eSdan     if( !pNew ){
49625d607a6eSdan       return SQLITE_NOMEM;
49635d607a6eSdan     }
49645d607a6eSdan     memset(pNew, 0, sizeof(SessionChange));
4965798693b2Sdan     pNew->op = op2;
49665d607a6eSdan     pNew->bIndirect = bIndirect;
4967cbf6d2d2Sdan     pNew->aRecord = (u8*)&pNew[1];
496824a0c453Sdan     if( bIndirect==0 || bRebase==0 ){
496924a0c453Sdan       pNew->nRecord = nRec;
4970cbf6d2d2Sdan       memcpy(pNew->aRecord, aRec, nRec);
497124a0c453Sdan     }else{
497224a0c453Sdan       int i;
497324a0c453Sdan       u8 *pIn = aRec;
497424a0c453Sdan       u8 *pOut = pNew->aRecord;
497524a0c453Sdan       for(i=0; i<pTab->nCol; i++){
497624a0c453Sdan         int nIn = sessionSerialLen(pIn);
497724a0c453Sdan         if( *pIn==0 ){
497824a0c453Sdan           *pOut++ = 0;
497924a0c453Sdan         }else if( pTab->abPK[i]==0 ){
498024a0c453Sdan           *pOut++ = 0xFF;
498124a0c453Sdan         }else{
498224a0c453Sdan           memcpy(pOut, pIn, nIn);
498324a0c453Sdan           pOut += nIn;
498424a0c453Sdan         }
498524a0c453Sdan         pIn += nIn;
498624a0c453Sdan       }
498724a0c453Sdan       pNew->nRecord = pOut - pNew->aRecord;
498824a0c453Sdan     }
4989c0a499eaSdan   }else if( bRebase ){
499024a0c453Sdan     if( pExist->op==SQLITE_DELETE && pExist->bIndirect ){
499124a0c453Sdan       *ppNew = pExist;
499224a0c453Sdan     }else{
49932d77d80aSdrh       sqlite3_int64 nByte = nRec + pExist->nRecord + sizeof(SessionChange);
49942d77d80aSdrh       pNew = (SessionChange*)sqlite3_malloc64(nByte);
499524a0c453Sdan       if( pNew==0 ){
499624a0c453Sdan         rc = SQLITE_NOMEM;
499724a0c453Sdan       }else{
499824a0c453Sdan         int i;
499924a0c453Sdan         u8 *a1 = pExist->aRecord;
500024a0c453Sdan         u8 *a2 = aRec;
500124a0c453Sdan         u8 *pOut;
500224a0c453Sdan 
500324a0c453Sdan         memset(pNew, 0, nByte);
500424a0c453Sdan         pNew->bIndirect = bIndirect || pExist->bIndirect;
500524a0c453Sdan         pNew->op = op2;
500624a0c453Sdan         pOut = pNew->aRecord = (u8*)&pNew[1];
500724a0c453Sdan 
500824a0c453Sdan         for(i=0; i<pTab->nCol; i++){
500924a0c453Sdan           int n1 = sessionSerialLen(a1);
501024a0c453Sdan           int n2 = sessionSerialLen(a2);
5011f231e18cSdan           if( *a1==0xFF || (pTab->abPK[i]==0 && bIndirect) ){
501224a0c453Sdan             *pOut++ = 0xFF;
501324a0c453Sdan           }else if( *a2==0 ){
501424a0c453Sdan             memcpy(pOut, a1, n1);
501524a0c453Sdan             pOut += n1;
501624a0c453Sdan           }else{
501724a0c453Sdan             memcpy(pOut, a2, n2);
501824a0c453Sdan             pOut += n2;
501924a0c453Sdan           }
502024a0c453Sdan           a1 += n1;
502124a0c453Sdan           a2 += n2;
502224a0c453Sdan         }
502324a0c453Sdan         pNew->nRecord = pOut - pNew->aRecord;
502424a0c453Sdan       }
502524a0c453Sdan       sqlite3_free(pExist);
502624a0c453Sdan     }
50275d607a6eSdan   }else{
5028798693b2Sdan     int op1 = pExist->op;
50295d607a6eSdan 
50305d607a6eSdan     /*
50315d607a6eSdan     **   op1=INSERT, op2=INSERT      ->      Unsupported. Discard op2.
50325d607a6eSdan     **   op1=INSERT, op2=UPDATE      ->      INSERT.
50335d607a6eSdan     **   op1=INSERT, op2=DELETE      ->      (none)
50345d607a6eSdan     **
50355d607a6eSdan     **   op1=UPDATE, op2=INSERT      ->      Unsupported. Discard op2.
50365d607a6eSdan     **   op1=UPDATE, op2=UPDATE      ->      UPDATE.
50375d607a6eSdan     **   op1=UPDATE, op2=DELETE      ->      DELETE.
50385d607a6eSdan     **
50395d607a6eSdan     **   op1=DELETE, op2=INSERT      ->      UPDATE.
50405d607a6eSdan     **   op1=DELETE, op2=UPDATE      ->      Unsupported. Discard op2.
50415d607a6eSdan     **   op1=DELETE, op2=DELETE      ->      Unsupported. Discard op2.
50425d607a6eSdan     */
50435d607a6eSdan     if( (op1==SQLITE_INSERT && op2==SQLITE_INSERT)
50445d607a6eSdan      || (op1==SQLITE_UPDATE && op2==SQLITE_INSERT)
50455d607a6eSdan      || (op1==SQLITE_DELETE && op2==SQLITE_UPDATE)
50465d607a6eSdan      || (op1==SQLITE_DELETE && op2==SQLITE_DELETE)
50475d607a6eSdan     ){
50485d607a6eSdan       pNew = pExist;
50495d607a6eSdan     }else if( op1==SQLITE_INSERT && op2==SQLITE_DELETE ){
50505d607a6eSdan       sqlite3_free(pExist);
50515d607a6eSdan       assert( pNew==0 );
50525d607a6eSdan     }else{
505364277f4aSdan       u8 *aExist = pExist->aRecord;
50542d77d80aSdrh       sqlite3_int64 nByte;
50555d607a6eSdan       u8 *aCsr;
50565d607a6eSdan 
505764277f4aSdan       /* Allocate a new SessionChange object. Ensure that the aRecord[]
505864277f4aSdan       ** buffer of the new object is large enough to hold any record that
505964277f4aSdan       ** may be generated by combining the input records.  */
50605d607a6eSdan       nByte = sizeof(SessionChange) + pExist->nRecord + nRec;
50612d77d80aSdrh       pNew = (SessionChange *)sqlite3_malloc64(nByte);
50625d607a6eSdan       if( !pNew ){
50631756ae10Sdan         sqlite3_free(pExist);
50645d607a6eSdan         return SQLITE_NOMEM;
50655d607a6eSdan       }
50665d607a6eSdan       memset(pNew, 0, sizeof(SessionChange));
50675d607a6eSdan       pNew->bIndirect = (bIndirect && pExist->bIndirect);
50685d607a6eSdan       aCsr = pNew->aRecord = (u8 *)&pNew[1];
50695d607a6eSdan 
5070b08a1efaSdan       if( op1==SQLITE_INSERT ){             /* INSERT + UPDATE */
50715d607a6eSdan         u8 *a1 = aRec;
5072b08a1efaSdan         assert( op2==SQLITE_UPDATE );
5073798693b2Sdan         pNew->op = SQLITE_INSERT;
5074ef7a6304Sdan         if( bPatchset==0 ) sessionSkipRecord(&a1, pTab->nCol);
507564277f4aSdan         sessionMergeRecord(&aCsr, pTab->nCol, aExist, a1);
5076b08a1efaSdan       }else if( op1==SQLITE_DELETE ){       /* DELETE + INSERT */
5077b08a1efaSdan         assert( op2==SQLITE_INSERT );
5078798693b2Sdan         pNew->op = SQLITE_UPDATE;
5079fa29ecc4Sdan         if( bPatchset ){
5080fa29ecc4Sdan           memcpy(aCsr, aRec, nRec);
5081fa29ecc4Sdan           aCsr += nRec;
5082fa29ecc4Sdan         }else{
508364277f4aSdan           if( 0==sessionMergeUpdate(&aCsr, pTab, bPatchset, aExist, 0,aRec,0) ){
5084b08a1efaSdan             sqlite3_free(pNew);
5085b08a1efaSdan             pNew = 0;
50865d607a6eSdan           }
5087fa29ecc4Sdan         }
5088b08a1efaSdan       }else if( op2==SQLITE_UPDATE ){       /* UPDATE + UPDATE */
508964277f4aSdan         u8 *a1 = aExist;
50905d607a6eSdan         u8 *a2 = aRec;
5091cfec7eeeSdan         assert( op1==SQLITE_UPDATE );
509264277f4aSdan         if( bPatchset==0 ){
5093ef7a6304Sdan           sessionSkipRecord(&a1, pTab->nCol);
5094ef7a6304Sdan           sessionSkipRecord(&a2, pTab->nCol);
509564277f4aSdan         }
5096798693b2Sdan         pNew->op = SQLITE_UPDATE;
509764277f4aSdan         if( 0==sessionMergeUpdate(&aCsr, pTab, bPatchset, aRec, aExist,a1,a2) ){
50985d607a6eSdan           sqlite3_free(pNew);
50995d607a6eSdan           pNew = 0;
51005d607a6eSdan         }
5101b08a1efaSdan       }else{                                /* UPDATE + DELETE */
5102b08a1efaSdan         assert( op1==SQLITE_UPDATE && op2==SQLITE_DELETE );
5103798693b2Sdan         pNew->op = SQLITE_DELETE;
510464277f4aSdan         if( bPatchset ){
510564277f4aSdan           memcpy(aCsr, aRec, nRec);
510664277f4aSdan           aCsr += nRec;
510764277f4aSdan         }else{
510864277f4aSdan           sessionMergeRecord(&aCsr, pTab->nCol, aRec, aExist);
510964277f4aSdan         }
51105d607a6eSdan       }
51115d607a6eSdan 
51125d607a6eSdan       if( pNew ){
51134f528042Sdan         pNew->nRecord = (int)(aCsr - pNew->aRecord);
51145d607a6eSdan       }
51155d607a6eSdan       sqlite3_free(pExist);
51165d607a6eSdan     }
51175d607a6eSdan   }
51185d607a6eSdan 
51195d607a6eSdan   *ppNew = pNew;
512024a0c453Sdan   return rc;
51215d607a6eSdan }
51225d607a6eSdan 
512377fc1d5bSdan /*
51245898ad69Sdan ** Add all changes in the changeset traversed by the iterator passed as
51255898ad69Sdan ** the first argument to the changegroup hash tables.
512677fc1d5bSdan */
sessionChangesetToHash(sqlite3_changeset_iter * pIter,sqlite3_changegroup * pGrp,int bRebase)512716228167Sdan static int sessionChangesetToHash(
5128cbf6d2d2Sdan   sqlite3_changeset_iter *pIter,   /* Iterator to read from */
5129c0a499eaSdan   sqlite3_changegroup *pGrp,       /* Changegroup object to add changeset to */
5130c0a499eaSdan   int bRebase                      /* True if hash table is for rebasing */
51315d607a6eSdan ){
51325d607a6eSdan   u8 *aRec;
51335d607a6eSdan   int nRec;
5134cbf6d2d2Sdan   int rc = SQLITE_OK;
51355d607a6eSdan   SessionTable *pTab = 0;
51365d607a6eSdan 
5137c0a499eaSdan   while( SQLITE_ROW==sessionChangesetNext(pIter, &aRec, &nRec, 0) ){
51385d607a6eSdan     const char *zNew;
51395d607a6eSdan     int nCol;
51405d607a6eSdan     int op;
51415d607a6eSdan     int iHash;
51425d607a6eSdan     int bIndirect;
51435d607a6eSdan     SessionChange *pChange;
51445d607a6eSdan     SessionChange *pExist = 0;
51455d607a6eSdan     SessionChange **pp;
51465d607a6eSdan 
51475898ad69Sdan     if( pGrp->pList==0 ){
51485898ad69Sdan       pGrp->bPatch = pIter->bPatchset;
51495898ad69Sdan     }else if( pIter->bPatchset!=pGrp->bPatch ){
51505898ad69Sdan       rc = SQLITE_ERROR;
51515898ad69Sdan       break;
51525898ad69Sdan     }
51535898ad69Sdan 
51545d607a6eSdan     sqlite3changeset_op(pIter, &zNew, &nCol, &op, &bIndirect);
5155ef7a6304Sdan     if( !pTab || sqlite3_stricmp(zNew, pTab->zName) ){
51565d607a6eSdan       /* Search the list for a matching table */
51574f528042Sdan       int nNew = (int)strlen(zNew);
5158f29123b5Sdan       u8 *abPK;
5159f29123b5Sdan 
5160f29123b5Sdan       sqlite3changeset_pk(pIter, &abPK, 0);
51615898ad69Sdan       for(pTab = pGrp->pList; pTab; pTab=pTab->pNext){
51625d607a6eSdan         if( 0==sqlite3_strnicmp(pTab->zName, zNew, nNew+1) ) break;
51635d607a6eSdan       }
51645d607a6eSdan       if( !pTab ){
51656c39e6a8Sdan         SessionTable **ppTab;
51666c39e6a8Sdan 
51672d77d80aSdrh         pTab = sqlite3_malloc64(sizeof(SessionTable) + nCol + nNew+1);
5168f29123b5Sdan         if( !pTab ){
5169f29123b5Sdan           rc = SQLITE_NOMEM;
5170f29123b5Sdan           break;
5171f29123b5Sdan         }
51725d607a6eSdan         memset(pTab, 0, sizeof(SessionTable));
5173f29123b5Sdan         pTab->nCol = nCol;
5174ef7a6304Sdan         pTab->abPK = (u8*)&pTab[1];
5175ef7a6304Sdan         memcpy(pTab->abPK, abPK, nCol);
5176ef7a6304Sdan         pTab->zName = (char*)&pTab->abPK[nCol];
5177ef7a6304Sdan         memcpy(pTab->zName, zNew, nNew+1);
51786c39e6a8Sdan 
51796c39e6a8Sdan         /* The new object must be linked on to the end of the list, not
51806c39e6a8Sdan         ** simply added to the start of it. This is to ensure that the
51816c39e6a8Sdan         ** tables within the output of sqlite3changegroup_output() are in
51826c39e6a8Sdan         ** the right order.  */
51836c39e6a8Sdan         for(ppTab=&pGrp->pList; *ppTab; ppTab=&(*ppTab)->pNext);
51846c39e6a8Sdan         *ppTab = pTab;
5185f29123b5Sdan       }else if( pTab->nCol!=nCol || memcmp(pTab->abPK, abPK, nCol) ){
5186f29123b5Sdan         rc = SQLITE_SCHEMA;
5187f29123b5Sdan         break;
51885d607a6eSdan       }
51895d607a6eSdan     }
51905d607a6eSdan 
51910cb735b9Sdan     if( sessionGrowHash(0, pIter->bPatchset, pTab) ){
51921756ae10Sdan       rc = SQLITE_NOMEM;
51931756ae10Sdan       break;
51941756ae10Sdan     }
519564277f4aSdan     iHash = sessionChangeHash(
5196cbf6d2d2Sdan         pTab, (pIter->bPatchset && op==SQLITE_DELETE), aRec, pTab->nChange
519764277f4aSdan     );
51985d607a6eSdan 
51995d607a6eSdan     /* Search for existing entry. If found, remove it from the hash table.
52005d607a6eSdan     ** Code below may link it back in.
52015d607a6eSdan     */
52025d607a6eSdan     for(pp=&pTab->apChange[iHash]; *pp; pp=&(*pp)->pNext){
520364277f4aSdan       int bPkOnly1 = 0;
520464277f4aSdan       int bPkOnly2 = 0;
5205cbf6d2d2Sdan       if( pIter->bPatchset ){
520664277f4aSdan         bPkOnly1 = (*pp)->op==SQLITE_DELETE;
520764277f4aSdan         bPkOnly2 = op==SQLITE_DELETE;
520864277f4aSdan       }
520964277f4aSdan       if( sessionChangeEqual(pTab, bPkOnly1, (*pp)->aRecord, bPkOnly2, aRec) ){
52105d607a6eSdan         pExist = *pp;
52115d607a6eSdan         *pp = (*pp)->pNext;
52125d607a6eSdan         pTab->nEntry--;
52135d607a6eSdan         break;
52145d607a6eSdan       }
52155d607a6eSdan     }
52165d607a6eSdan 
5217c0a499eaSdan     rc = sessionChangeMerge(pTab, bRebase,
5218cbf6d2d2Sdan         pIter->bPatchset, pExist, op, bIndirect, aRec, nRec, &pChange
521964277f4aSdan     );
52205d607a6eSdan     if( rc ) break;
52215d607a6eSdan     if( pChange ){
52225d607a6eSdan       pChange->pNext = pTab->apChange[iHash];
52235d607a6eSdan       pTab->apChange[iHash] = pChange;
52245d607a6eSdan       pTab->nEntry++;
52255d607a6eSdan     }
52265d607a6eSdan   }
52275d607a6eSdan 
5228cbf6d2d2Sdan   if( rc==SQLITE_OK ) rc = pIter->rc;
52295d607a6eSdan   return rc;
52305d607a6eSdan }
52315d607a6eSdan 
52325d607a6eSdan /*
52335898ad69Sdan ** Serialize a changeset (or patchset) based on all changesets (or patchsets)
52345898ad69Sdan ** added to the changegroup object passed as the first argument.
52355d607a6eSdan **
52365898ad69Sdan ** If xOutput is not NULL, then the changeset/patchset is returned to the
52375898ad69Sdan ** user via one or more calls to xOutput, as with the other streaming
52385898ad69Sdan ** interfaces.
52395d607a6eSdan **
52405898ad69Sdan ** Or, if xOutput is NULL, then (*ppOut) is populated with a pointer to a
52415898ad69Sdan ** buffer containing the output changeset before this function returns. In
52425898ad69Sdan ** this case (*pnOut) is set to the size of the output buffer in bytes. It
52435898ad69Sdan ** is the responsibility of the caller to free the output buffer using
52445898ad69Sdan ** sqlite3_free() when it is no longer required.
52455898ad69Sdan **
52465898ad69Sdan ** If successful, SQLITE_OK is returned. Or, if an error occurs, an SQLite
52475898ad69Sdan ** error code. If an error occurs and xOutput is NULL, (*ppOut) and (*pnOut)
52485898ad69Sdan ** are both set to 0 before returning.
52495d607a6eSdan */
sessionChangegroupOutput(sqlite3_changegroup * pGrp,int (* xOutput)(void * pOut,const void * pData,int nData),void * pOut,int * pnOut,void ** ppOut)52505898ad69Sdan static int sessionChangegroupOutput(
52515898ad69Sdan   sqlite3_changegroup *pGrp,
5252cbf6d2d2Sdan   int (*xOutput)(void *pOut, const void *pData, int nData),
5253cbf6d2d2Sdan   void *pOut,
5254cbf6d2d2Sdan   int *pnOut,
5255cbf6d2d2Sdan   void **ppOut
52565d607a6eSdan ){
52575898ad69Sdan   int rc = SQLITE_OK;
5258e8fa8c96Sdan   SessionBuffer buf = {0, 0, 0};
52595898ad69Sdan   SessionTable *pTab;
5260cbf6d2d2Sdan   assert( xOutput==0 || (ppOut==0 && pnOut==0) );
52615d607a6eSdan 
52625d607a6eSdan   /* Create the serialized output changeset based on the contents of the
52635898ad69Sdan   ** hash tables attached to the SessionTable objects in list p->pList.
52645d607a6eSdan   */
52655898ad69Sdan   for(pTab=pGrp->pList; rc==SQLITE_OK && pTab; pTab=pTab->pNext){
52665d607a6eSdan     int i;
52675d607a6eSdan     if( pTab->nEntry==0 ) continue;
52685d607a6eSdan 
52695898ad69Sdan     sessionAppendTableHdr(&buf, pGrp->bPatch, pTab, &rc);
52705d607a6eSdan     for(i=0; i<pTab->nChange; i++){
52715d607a6eSdan       SessionChange *p;
52725d607a6eSdan       for(p=pTab->apChange[i]; p; p=p->pNext){
5273798693b2Sdan         sessionAppendByte(&buf, p->op, &rc);
52745d607a6eSdan         sessionAppendByte(&buf, p->bIndirect, &rc);
52755d607a6eSdan         sessionAppendBlob(&buf, p->aRecord, p->nRecord, &rc);
52761f48e67dSdan         if( rc==SQLITE_OK && xOutput && buf.nBuf>=sessions_strm_chunk_size ){
5277cbf6d2d2Sdan           rc = xOutput(pOut, buf.aBuf, buf.nBuf);
5278cbf6d2d2Sdan           buf.nBuf = 0;
5279cbf6d2d2Sdan         }
52805d607a6eSdan       }
5281f8248c42Sdan     }
5282f8248c42Sdan   }
52835d607a6eSdan 
52845d607a6eSdan   if( rc==SQLITE_OK ){
5285cbf6d2d2Sdan     if( xOutput ){
5286cbf6d2d2Sdan       if( buf.nBuf>0 ) rc = xOutput(pOut, buf.aBuf, buf.nBuf);
5287*7a3b4451Sdrh     }else if( ppOut ){
52885d607a6eSdan       *ppOut = buf.aBuf;
5289*7a3b4451Sdrh       if( pnOut ) *pnOut = buf.nBuf;
5290cbf6d2d2Sdan       buf.aBuf = 0;
52915d607a6eSdan     }
52925d607a6eSdan   }
5293cbf6d2d2Sdan   sqlite3_free(buf.aBuf);
52945d607a6eSdan 
52955d607a6eSdan   return rc;
52965d607a6eSdan }
52975d607a6eSdan 
5298cbf6d2d2Sdan /*
52995898ad69Sdan ** Allocate a new, empty, sqlite3_changegroup.
53005898ad69Sdan */
sqlite3changegroup_new(sqlite3_changegroup ** pp)53015898ad69Sdan int sqlite3changegroup_new(sqlite3_changegroup **pp){
53025898ad69Sdan   int rc = SQLITE_OK;             /* Return code */
53035898ad69Sdan   sqlite3_changegroup *p;         /* New object */
53045898ad69Sdan   p = (sqlite3_changegroup*)sqlite3_malloc(sizeof(sqlite3_changegroup));
53055898ad69Sdan   if( p==0 ){
53065898ad69Sdan     rc = SQLITE_NOMEM;
53075898ad69Sdan   }else{
53085898ad69Sdan     memset(p, 0, sizeof(sqlite3_changegroup));
53095898ad69Sdan   }
53105898ad69Sdan   *pp = p;
53115898ad69Sdan   return rc;
53125898ad69Sdan }
53135898ad69Sdan 
53145898ad69Sdan /*
53155898ad69Sdan ** Add the changeset currently stored in buffer pData, size nData bytes,
53165898ad69Sdan ** to changeset-group p.
53175898ad69Sdan */
sqlite3changegroup_add(sqlite3_changegroup * pGrp,int nData,void * pData)53185898ad69Sdan int sqlite3changegroup_add(sqlite3_changegroup *pGrp, int nData, void *pData){
53195898ad69Sdan   sqlite3_changeset_iter *pIter;  /* Iterator opened on pData/nData */
53205898ad69Sdan   int rc;                         /* Return code */
53215898ad69Sdan 
53225898ad69Sdan   rc = sqlite3changeset_start(&pIter, nData, pData);
53235898ad69Sdan   if( rc==SQLITE_OK ){
5324c0a499eaSdan     rc = sessionChangesetToHash(pIter, pGrp, 0);
53255898ad69Sdan   }
53265898ad69Sdan   sqlite3changeset_finalize(pIter);
53275898ad69Sdan   return rc;
53285898ad69Sdan }
53295898ad69Sdan 
53305898ad69Sdan /*
53315898ad69Sdan ** Obtain a buffer containing a changeset representing the concatenation
53325898ad69Sdan ** of all changesets added to the group so far.
53335898ad69Sdan */
sqlite3changegroup_output(sqlite3_changegroup * pGrp,int * pnData,void ** ppData)53345898ad69Sdan int sqlite3changegroup_output(
53355898ad69Sdan     sqlite3_changegroup *pGrp,
53365898ad69Sdan     int *pnData,
53375898ad69Sdan     void **ppData
53385898ad69Sdan ){
53395898ad69Sdan   return sessionChangegroupOutput(pGrp, 0, 0, pnData, ppData);
53405898ad69Sdan }
53415898ad69Sdan 
53425898ad69Sdan /*
53435898ad69Sdan ** Streaming versions of changegroup_add().
53445898ad69Sdan */
sqlite3changegroup_add_strm(sqlite3_changegroup * pGrp,int (* xInput)(void * pIn,void * pData,int * pnData),void * pIn)53455898ad69Sdan int sqlite3changegroup_add_strm(
53465898ad69Sdan   sqlite3_changegroup *pGrp,
53475898ad69Sdan   int (*xInput)(void *pIn, void *pData, int *pnData),
53485898ad69Sdan   void *pIn
53495898ad69Sdan ){
53505898ad69Sdan   sqlite3_changeset_iter *pIter;  /* Iterator opened on pData/nData */
53515898ad69Sdan   int rc;                         /* Return code */
53525898ad69Sdan 
53535898ad69Sdan   rc = sqlite3changeset_start_strm(&pIter, xInput, pIn);
53545898ad69Sdan   if( rc==SQLITE_OK ){
5355c0a499eaSdan     rc = sessionChangesetToHash(pIter, pGrp, 0);
53565898ad69Sdan   }
53575898ad69Sdan   sqlite3changeset_finalize(pIter);
53585898ad69Sdan   return rc;
53595898ad69Sdan }
53605898ad69Sdan 
53615898ad69Sdan /*
53625898ad69Sdan ** Streaming versions of changegroup_output().
53635898ad69Sdan */
sqlite3changegroup_output_strm(sqlite3_changegroup * pGrp,int (* xOutput)(void * pOut,const void * pData,int nData),void * pOut)53645898ad69Sdan int sqlite3changegroup_output_strm(
53655898ad69Sdan   sqlite3_changegroup *pGrp,
53665898ad69Sdan   int (*xOutput)(void *pOut, const void *pData, int nData),
53675898ad69Sdan   void *pOut
53685898ad69Sdan ){
53695898ad69Sdan   return sessionChangegroupOutput(pGrp, xOutput, pOut, 0, 0);
53705898ad69Sdan }
53715898ad69Sdan 
53725898ad69Sdan /*
53735898ad69Sdan ** Delete a changegroup object.
53745898ad69Sdan */
sqlite3changegroup_delete(sqlite3_changegroup * pGrp)53755898ad69Sdan void sqlite3changegroup_delete(sqlite3_changegroup *pGrp){
53765898ad69Sdan   if( pGrp ){
53770cb735b9Sdan     sessionDeleteTable(0, pGrp->pList);
53785898ad69Sdan     sqlite3_free(pGrp);
53795898ad69Sdan   }
53805898ad69Sdan }
53815898ad69Sdan 
53825898ad69Sdan /*
5383cbf6d2d2Sdan ** Combine two changesets together.
5384cbf6d2d2Sdan */
sqlite3changeset_concat(int nLeft,void * pLeft,int nRight,void * pRight,int * pnOut,void ** ppOut)5385cbf6d2d2Sdan int sqlite3changeset_concat(
5386cbf6d2d2Sdan   int nLeft,                      /* Number of bytes in lhs input */
5387cbf6d2d2Sdan   void *pLeft,                    /* Lhs input changeset */
5388cbf6d2d2Sdan   int nRight                      /* Number of bytes in rhs input */,
5389cbf6d2d2Sdan   void *pRight,                   /* Rhs input changeset */
5390cbf6d2d2Sdan   int *pnOut,                     /* OUT: Number of bytes in output changeset */
5391cbf6d2d2Sdan   void **ppOut                    /* OUT: changeset (left <concat> right) */
5392cbf6d2d2Sdan ){
53935898ad69Sdan   sqlite3_changegroup *pGrp;
5394cbf6d2d2Sdan   int rc;
5395cbf6d2d2Sdan 
53965898ad69Sdan   rc = sqlite3changegroup_new(&pGrp);
5397cbf6d2d2Sdan   if( rc==SQLITE_OK ){
53985898ad69Sdan     rc = sqlite3changegroup_add(pGrp, nLeft, pLeft);
5399cbf6d2d2Sdan   }
5400cbf6d2d2Sdan   if( rc==SQLITE_OK ){
54015898ad69Sdan     rc = sqlite3changegroup_add(pGrp, nRight, pRight);
5402cbf6d2d2Sdan   }
54035898ad69Sdan   if( rc==SQLITE_OK ){
54045898ad69Sdan     rc = sqlite3changegroup_output(pGrp, pnOut, ppOut);
54055898ad69Sdan   }
54065898ad69Sdan   sqlite3changegroup_delete(pGrp);
5407cbf6d2d2Sdan 
5408cbf6d2d2Sdan   return rc;
5409cbf6d2d2Sdan }
5410cbf6d2d2Sdan 
5411cbf6d2d2Sdan /*
5412cbf6d2d2Sdan ** Streaming version of sqlite3changeset_concat().
5413cbf6d2d2Sdan */
sqlite3changeset_concat_strm(int (* xInputA)(void * pIn,void * pData,int * pnData),void * pInA,int (* xInputB)(void * pIn,void * pData,int * pnData),void * pInB,int (* xOutput)(void * pOut,const void * pData,int nData),void * pOut)5414f1a08ad8Sdrh int sqlite3changeset_concat_strm(
5415cbf6d2d2Sdan   int (*xInputA)(void *pIn, void *pData, int *pnData),
5416cbf6d2d2Sdan   void *pInA,
5417cbf6d2d2Sdan   int (*xInputB)(void *pIn, void *pData, int *pnData),
5418cbf6d2d2Sdan   void *pInB,
5419cbf6d2d2Sdan   int (*xOutput)(void *pOut, const void *pData, int nData),
5420cbf6d2d2Sdan   void *pOut
5421cbf6d2d2Sdan ){
54225898ad69Sdan   sqlite3_changegroup *pGrp;
5423cbf6d2d2Sdan   int rc;
5424cbf6d2d2Sdan 
54255898ad69Sdan   rc = sqlite3changegroup_new(&pGrp);
5426cbf6d2d2Sdan   if( rc==SQLITE_OK ){
54275898ad69Sdan     rc = sqlite3changegroup_add_strm(pGrp, xInputA, pInA);
5428cbf6d2d2Sdan   }
5429cbf6d2d2Sdan   if( rc==SQLITE_OK ){
54305898ad69Sdan     rc = sqlite3changegroup_add_strm(pGrp, xInputB, pInB);
5431cbf6d2d2Sdan   }
54325898ad69Sdan   if( rc==SQLITE_OK ){
54335898ad69Sdan     rc = sqlite3changegroup_output_strm(pGrp, xOutput, pOut);
54345898ad69Sdan   }
54355898ad69Sdan   sqlite3changegroup_delete(pGrp);
5436cbf6d2d2Sdan 
5437cbf6d2d2Sdan   return rc;
5438cbf6d2d2Sdan }
5439cbf6d2d2Sdan 
54403fa5463cSdan /*
54413fa5463cSdan ** Changeset rebaser handle.
54423fa5463cSdan */
5443c0a499eaSdan struct sqlite3_rebaser {
5444c0a499eaSdan   sqlite3_changegroup grp;        /* Hash table */
5445c0a499eaSdan };
5446c0a499eaSdan 
5447c0a499eaSdan /*
5448c0a499eaSdan ** Buffers a1 and a2 must both contain a sessions module record nCol
5449c0a499eaSdan ** fields in size. This function appends an nCol sessions module
5450b880a7b1Sdan ** record to buffer pBuf that is a copy of a1, except that for
5451b880a7b1Sdan ** each field that is undefined in a1[], swap in the field from a2[].
5452c0a499eaSdan */
sessionAppendRecordMerge(SessionBuffer * pBuf,int nCol,u8 * a1,int n1,u8 * a2,int n2,int * pRc)5453c0a499eaSdan static void sessionAppendRecordMerge(
5454b880a7b1Sdan   SessionBuffer *pBuf,            /* Buffer to append to */
5455b880a7b1Sdan   int nCol,                       /* Number of columns in each record */
5456b880a7b1Sdan   u8 *a1, int n1,                 /* Record 1 */
5457b880a7b1Sdan   u8 *a2, int n2,                 /* Record 2 */
5458b880a7b1Sdan   int *pRc                        /* IN/OUT: error code */
5459c0a499eaSdan ){
5460c0a499eaSdan   sessionBufferGrow(pBuf, n1+n2, pRc);
5461c0a499eaSdan   if( *pRc==SQLITE_OK ){
5462c0a499eaSdan     int i;
5463c0a499eaSdan     u8 *pOut = &pBuf->aBuf[pBuf->nBuf];
5464c0a499eaSdan     for(i=0; i<nCol; i++){
5465c0a499eaSdan       int nn1 = sessionSerialLen(a1);
5466c0a499eaSdan       int nn2 = sessionSerialLen(a2);
546724a0c453Sdan       if( *a1==0 || *a1==0xFF ){
5468c0a499eaSdan         memcpy(pOut, a2, nn2);
5469c0a499eaSdan         pOut += nn2;
5470c0a499eaSdan       }else{
5471c0a499eaSdan         memcpy(pOut, a1, nn1);
5472c0a499eaSdan         pOut += nn1;
5473c0a499eaSdan       }
5474f1b40e83Sdan       a1 += nn1;
5475f1b40e83Sdan       a2 += nn2;
5476c0a499eaSdan     }
5477f1b40e83Sdan 
5478f1b40e83Sdan     pBuf->nBuf = pOut-pBuf->aBuf;
5479f1b40e83Sdan     assert( pBuf->nBuf<=pBuf->nAlloc );
5480c0a499eaSdan   }
5481c0a499eaSdan }
5482c0a499eaSdan 
54833fa5463cSdan /*
54843fa5463cSdan ** This function is called when rebasing a local UPDATE change against one
54853fa5463cSdan ** or more remote UPDATE changes. The aRec/nRec buffer contains the current
54863fa5463cSdan ** old.* and new.* records for the change. The rebase buffer (a single
54873fa5463cSdan ** record) is in aChange/nChange. The rebased change is appended to buffer
54883fa5463cSdan ** pBuf.
54893fa5463cSdan **
54903fa5463cSdan ** Rebasing the UPDATE involves:
54913fa5463cSdan **
54923fa5463cSdan **   * Removing any changes to fields for which the corresponding field
54933fa5463cSdan **     in the rebase buffer is set to "replaced" (type 0xFF). If this
54943fa5463cSdan **     means the UPDATE change updates no fields, nothing is appended
54953fa5463cSdan **     to the output buffer.
54963fa5463cSdan **
54973fa5463cSdan **   * For each field modified by the local change for which the
54983fa5463cSdan **     corresponding field in the rebase buffer is not "undefined" (0x00)
54993fa5463cSdan **     or "replaced" (0xFF), the old.* value is replaced by the value
55003fa5463cSdan **     in the rebase buffer.
55013fa5463cSdan */
sessionAppendPartialUpdate(SessionBuffer * pBuf,sqlite3_changeset_iter * pIter,u8 * aRec,int nRec,u8 * aChange,int nChange,int * pRc)5502f01d3a7eSdan static void sessionAppendPartialUpdate(
550324a0c453Sdan   SessionBuffer *pBuf,            /* Append record here */
550424a0c453Sdan   sqlite3_changeset_iter *pIter,  /* Iterator pointed at local change */
550524a0c453Sdan   u8 *aRec, int nRec,             /* Local change */
550624a0c453Sdan   u8 *aChange, int nChange,       /* Record to rebase against */
550724a0c453Sdan   int *pRc                        /* IN/OUT: Return Code */
5508f01d3a7eSdan ){
5509f01d3a7eSdan   sessionBufferGrow(pBuf, 2+nRec+nChange, pRc);
5510f01d3a7eSdan   if( *pRc==SQLITE_OK ){
5511f01d3a7eSdan     int bData = 0;
5512f01d3a7eSdan     u8 *pOut = &pBuf->aBuf[pBuf->nBuf];
5513f01d3a7eSdan     int i;
5514f01d3a7eSdan     u8 *a1 = aRec;
5515f01d3a7eSdan     u8 *a2 = aChange;
5516f01d3a7eSdan 
5517f01d3a7eSdan     *pOut++ = SQLITE_UPDATE;
5518f01d3a7eSdan     *pOut++ = pIter->bIndirect;
5519f01d3a7eSdan     for(i=0; i<pIter->nCol; i++){
5520f01d3a7eSdan       int n1 = sessionSerialLen(a1);
5521f01d3a7eSdan       int n2 = sessionSerialLen(a2);
5522f01d3a7eSdan       if( pIter->abPK[i] || a2[0]==0 ){
55231e25d20cSdan         if( !pIter->abPK[i] && a1[0] ) bData = 1;
5524f01d3a7eSdan         memcpy(pOut, a1, n1);
5525f01d3a7eSdan         pOut += n1;
552624a0c453Sdan       }else if( a2[0]!=0xFF ){
552724a0c453Sdan         bData = 1;
552824a0c453Sdan         memcpy(pOut, a2, n2);
552924a0c453Sdan         pOut += n2;
5530f01d3a7eSdan       }else{
5531f01d3a7eSdan         *pOut++ = '\0';
5532f01d3a7eSdan       }
5533f01d3a7eSdan       a1 += n1;
5534f01d3a7eSdan       a2 += n2;
5535f01d3a7eSdan     }
5536f01d3a7eSdan     if( bData ){
5537f01d3a7eSdan       a2 = aChange;
5538f01d3a7eSdan       for(i=0; i<pIter->nCol; i++){
5539f01d3a7eSdan         int n1 = sessionSerialLen(a1);
5540f01d3a7eSdan         int n2 = sessionSerialLen(a2);
554124a0c453Sdan         if( pIter->abPK[i] || a2[0]!=0xFF ){
5542f01d3a7eSdan           memcpy(pOut, a1, n1);
5543f01d3a7eSdan           pOut += n1;
5544f01d3a7eSdan         }else{
5545f01d3a7eSdan           *pOut++ = '\0';
5546f01d3a7eSdan         }
5547f01d3a7eSdan         a1 += n1;
5548f01d3a7eSdan         a2 += n2;
5549f01d3a7eSdan       }
5550f01d3a7eSdan       pBuf->nBuf = (pOut - pBuf->aBuf);
5551f01d3a7eSdan     }
5552f01d3a7eSdan   }
5553f01d3a7eSdan }
5554f01d3a7eSdan 
55553fa5463cSdan /*
55563fa5463cSdan ** pIter is configured to iterate through a changeset. This function rebases
55573fa5463cSdan ** that changeset according to the current configuration of the rebaser
55583fa5463cSdan ** object passed as the first argument. If no error occurs and argument xOutput
55593fa5463cSdan ** is not NULL, then the changeset is returned to the caller by invoking
55603fa5463cSdan ** xOutput zero or more times and SQLITE_OK returned. Or, if xOutput is NULL,
55613fa5463cSdan ** then (*ppOut) is set to point to a buffer containing the rebased changeset
55623fa5463cSdan ** before this function returns. In this case (*pnOut) is set to the size of
55633fa5463cSdan ** the buffer in bytes.  It is the responsibility of the caller to eventually
55643fa5463cSdan ** free the (*ppOut) buffer using sqlite3_free().
55653fa5463cSdan **
55663fa5463cSdan ** If an error occurs, an SQLite error code is returned. If ppOut and
55673fa5463cSdan ** pnOut are not NULL, then the two output parameters are set to 0 before
55683fa5463cSdan ** returning.
55693fa5463cSdan */
sessionRebase(sqlite3_rebaser * p,sqlite3_changeset_iter * pIter,int (* xOutput)(void * pOut,const void * pData,int nData),void * pOut,int * pnOut,void ** ppOut)5570c0a499eaSdan static int sessionRebase(
5571c0a499eaSdan   sqlite3_rebaser *p,             /* Rebaser hash table */
5572c0a499eaSdan   sqlite3_changeset_iter *pIter,  /* Input data */
5573c0a499eaSdan   int (*xOutput)(void *pOut, const void *pData, int nData),
5574c0a499eaSdan   void *pOut,                     /* Context for xOutput callback */
5575c0a499eaSdan   int *pnOut,                     /* OUT: Number of bytes in output changeset */
5576c0a499eaSdan   void **ppOut                    /* OUT: Inverse of pChangeset */
5577c0a499eaSdan ){
5578c0a499eaSdan   int rc = SQLITE_OK;
5579c0a499eaSdan   u8 *aRec = 0;
5580c0a499eaSdan   int nRec = 0;
5581c0a499eaSdan   int bNew = 0;
5582c0a499eaSdan   SessionTable *pTab = 0;
5583c0a499eaSdan   SessionBuffer sOut = {0,0,0};
5584c0a499eaSdan 
5585f1b40e83Sdan   while( SQLITE_ROW==sessionChangesetNext(pIter, &aRec, &nRec, &bNew) ){
5586c0a499eaSdan     SessionChange *pChange = 0;
5587f01d3a7eSdan     int bDone = 0;
5588c0a499eaSdan 
5589c0a499eaSdan     if( bNew ){
5590c0a499eaSdan       const char *zTab = pIter->zTab;
5591c0a499eaSdan       for(pTab=p->grp.pList; pTab; pTab=pTab->pNext){
5592c0a499eaSdan         if( 0==sqlite3_stricmp(pTab->zName, zTab) ) break;
5593c0a499eaSdan       }
5594c0a499eaSdan       bNew = 0;
5595c0a499eaSdan 
559695ccb6dcSdan       /* A patchset may not be rebased */
559795ccb6dcSdan       if( pIter->bPatchset ){
559895ccb6dcSdan         rc = SQLITE_ERROR;
559995ccb6dcSdan       }
560095ccb6dcSdan 
5601c0a499eaSdan       /* Append a table header to the output for this new table */
5602c0a499eaSdan       sessionAppendByte(&sOut, pIter->bPatchset ? 'P' : 'T', &rc);
5603c0a499eaSdan       sessionAppendVarint(&sOut, pIter->nCol, &rc);
5604c0a499eaSdan       sessionAppendBlob(&sOut, pIter->abPK, pIter->nCol, &rc);
56052c42c34dSmistachkin       sessionAppendBlob(&sOut,(u8*)pIter->zTab,(int)strlen(pIter->zTab)+1,&rc);
5606c0a499eaSdan     }
5607c0a499eaSdan 
56088cb83beeSdan     if( pTab && rc==SQLITE_OK ){
56098cb83beeSdan       int iHash = sessionChangeHash(pTab, 0, aRec, pTab->nChange);
5610c0a499eaSdan 
5611c0a499eaSdan       for(pChange=pTab->apChange[iHash]; pChange; pChange=pChange->pNext){
56128cb83beeSdan         if( sessionChangeEqual(pTab, 0, aRec, 0, pChange->aRecord) ){
5613c0a499eaSdan           break;
5614c0a499eaSdan         }
5615c0a499eaSdan       }
5616c0a499eaSdan     }
5617c0a499eaSdan 
5618c0a499eaSdan     if( pChange ){
5619c0a499eaSdan       assert( pChange->op==SQLITE_DELETE || pChange->op==SQLITE_INSERT );
5620f01d3a7eSdan       switch( pIter->op ){
5621f01d3a7eSdan         case SQLITE_INSERT:
5622f01d3a7eSdan           if( pChange->op==SQLITE_INSERT ){
5623f01d3a7eSdan             bDone = 1;
5624f01d3a7eSdan             if( pChange->bIndirect==0 ){
5625f01d3a7eSdan               sessionAppendByte(&sOut, SQLITE_UPDATE, &rc);
5626f01d3a7eSdan               sessionAppendByte(&sOut, pIter->bIndirect, &rc);
5627f01d3a7eSdan               sessionAppendBlob(&sOut, pChange->aRecord, pChange->nRecord, &rc);
5628f01d3a7eSdan               sessionAppendBlob(&sOut, aRec, nRec, &rc);
5629f01d3a7eSdan             }
5630f01d3a7eSdan           }
5631f01d3a7eSdan           break;
5632f01d3a7eSdan 
5633f01d3a7eSdan         case SQLITE_UPDATE:
5634f01d3a7eSdan           bDone = 1;
5635f01d3a7eSdan           if( pChange->op==SQLITE_DELETE ){
5636f01d3a7eSdan             if( pChange->bIndirect==0 ){
5637f01d3a7eSdan               u8 *pCsr = aRec;
5638f01d3a7eSdan               sessionSkipRecord(&pCsr, pIter->nCol);
5639f01d3a7eSdan               sessionAppendByte(&sOut, SQLITE_INSERT, &rc);
5640f01d3a7eSdan               sessionAppendByte(&sOut, pIter->bIndirect, &rc);
5641b880a7b1Sdan               sessionAppendRecordMerge(&sOut, pIter->nCol,
5642f01d3a7eSdan                   pCsr, nRec-(pCsr-aRec),
5643f01d3a7eSdan                   pChange->aRecord, pChange->nRecord, &rc
5644f01d3a7eSdan               );
5645f01d3a7eSdan             }
5646f01d3a7eSdan           }else{
5647f01d3a7eSdan             sessionAppendPartialUpdate(&sOut, pIter,
5648f01d3a7eSdan                 aRec, nRec, pChange->aRecord, pChange->nRecord, &rc
5649f01d3a7eSdan             );
5650f01d3a7eSdan           }
5651f01d3a7eSdan           break;
5652f01d3a7eSdan 
5653f01d3a7eSdan         default:
5654f01d3a7eSdan           assert( pIter->op==SQLITE_DELETE );
5655f01d3a7eSdan           bDone = 1;
5656f01d3a7eSdan           if( pChange->op==SQLITE_INSERT ){
5657f01d3a7eSdan             sessionAppendByte(&sOut, SQLITE_DELETE, &rc);
5658f01d3a7eSdan             sessionAppendByte(&sOut, pIter->bIndirect, &rc);
5659b880a7b1Sdan             sessionAppendRecordMerge(&sOut, pIter->nCol,
5660f01d3a7eSdan                 pChange->aRecord, pChange->nRecord, aRec, nRec, &rc
5661f01d3a7eSdan             );
5662f01d3a7eSdan           }
5663f01d3a7eSdan           break;
5664f01d3a7eSdan       }
5665f01d3a7eSdan     }
5666f01d3a7eSdan 
5667f01d3a7eSdan     if( bDone==0 ){
5668f01d3a7eSdan       sessionAppendByte(&sOut, pIter->op, &rc);
5669f01d3a7eSdan       sessionAppendByte(&sOut, pIter->bIndirect, &rc);
5670f01d3a7eSdan       sessionAppendBlob(&sOut, aRec, nRec, &rc);
5671f01d3a7eSdan     }
56721f48e67dSdan     if( rc==SQLITE_OK && xOutput && sOut.nBuf>sessions_strm_chunk_size ){
5673c0a499eaSdan       rc = xOutput(pOut, sOut.aBuf, sOut.nBuf);
5674c0a499eaSdan       sOut.nBuf = 0;
5675c0a499eaSdan     }
5676c0a499eaSdan     if( rc ) break;
5677c0a499eaSdan   }
5678c0a499eaSdan 
5679c0a499eaSdan   if( rc!=SQLITE_OK ){
5680c0a499eaSdan     sqlite3_free(sOut.aBuf);
5681c0a499eaSdan     memset(&sOut, 0, sizeof(sOut));
5682c0a499eaSdan   }
5683c0a499eaSdan 
5684c0a499eaSdan   if( rc==SQLITE_OK ){
5685c0a499eaSdan     if( xOutput ){
5686c0a499eaSdan       if( sOut.nBuf>0 ){
5687c0a499eaSdan         rc = xOutput(pOut, sOut.aBuf, sOut.nBuf);
5688c0a499eaSdan       }
5689*7a3b4451Sdrh     }else if( ppOut ){
5690c0a499eaSdan       *ppOut = (void*)sOut.aBuf;
5691c0a499eaSdan       *pnOut = sOut.nBuf;
5692c0a499eaSdan       sOut.aBuf = 0;
5693c0a499eaSdan     }
5694c0a499eaSdan   }
5695c0a499eaSdan   sqlite3_free(sOut.aBuf);
5696c0a499eaSdan   return rc;
5697c0a499eaSdan }
5698c0a499eaSdan 
5699c0a499eaSdan /*
5700c0a499eaSdan ** Create a new rebaser object.
5701c0a499eaSdan */
sqlite3rebaser_create(sqlite3_rebaser ** ppNew)5702c0a499eaSdan int sqlite3rebaser_create(sqlite3_rebaser **ppNew){
5703c0a499eaSdan   int rc = SQLITE_OK;
5704c0a499eaSdan   sqlite3_rebaser *pNew;
5705c0a499eaSdan 
5706c0a499eaSdan   pNew = sqlite3_malloc(sizeof(sqlite3_rebaser));
5707c0a499eaSdan   if( pNew==0 ){
5708c0a499eaSdan     rc = SQLITE_NOMEM;
5709f1b40e83Sdan   }else{
5710f1b40e83Sdan     memset(pNew, 0, sizeof(sqlite3_rebaser));
5711c0a499eaSdan   }
5712c0a499eaSdan   *ppNew = pNew;
5713c0a499eaSdan   return rc;
5714c0a499eaSdan }
5715c0a499eaSdan 
5716c0a499eaSdan /*
5717c0a499eaSdan ** Call this one or more times to configure a rebaser.
5718c0a499eaSdan */
sqlite3rebaser_configure(sqlite3_rebaser * p,int nRebase,const void * pRebase)5719c0a499eaSdan int sqlite3rebaser_configure(
5720c0a499eaSdan   sqlite3_rebaser *p,
5721c0a499eaSdan   int nRebase, const void *pRebase
5722c0a499eaSdan ){
5723c0a499eaSdan   sqlite3_changeset_iter *pIter = 0;   /* Iterator opened on pData/nData */
5724c0a499eaSdan   int rc;                              /* Return code */
5725c0a499eaSdan   rc = sqlite3changeset_start(&pIter, nRebase, (void*)pRebase);
5726c0a499eaSdan   if( rc==SQLITE_OK ){
5727c0a499eaSdan     rc = sessionChangesetToHash(pIter, &p->grp, 1);
5728c0a499eaSdan   }
5729c0a499eaSdan   sqlite3changeset_finalize(pIter);
5730c0a499eaSdan   return rc;
5731c0a499eaSdan }
5732c0a499eaSdan 
5733c0a499eaSdan /*
5734c0a499eaSdan ** Rebase a changeset according to current rebaser configuration
5735c0a499eaSdan */
sqlite3rebaser_rebase(sqlite3_rebaser * p,int nIn,const void * pIn,int * pnOut,void ** ppOut)5736c0a499eaSdan int sqlite3rebaser_rebase(
5737c0a499eaSdan   sqlite3_rebaser *p,
5738c0a499eaSdan   int nIn, const void *pIn,
5739c0a499eaSdan   int *pnOut, void **ppOut
5740c0a499eaSdan ){
5741c0a499eaSdan   sqlite3_changeset_iter *pIter = 0;   /* Iterator to skip through input */
5742c0a499eaSdan   int rc = sqlite3changeset_start(&pIter, nIn, (void*)pIn);
5743c0a499eaSdan 
5744c0a499eaSdan   if( rc==SQLITE_OK ){
5745c0a499eaSdan     rc = sessionRebase(p, pIter, 0, 0, pnOut, ppOut);
5746c0a499eaSdan     sqlite3changeset_finalize(pIter);
5747c0a499eaSdan   }
5748c0a499eaSdan 
5749c0a499eaSdan   return rc;
5750c0a499eaSdan }
5751c0a499eaSdan 
5752c0a499eaSdan /*
5753c0a499eaSdan ** Rebase a changeset according to current rebaser configuration
5754c0a499eaSdan */
sqlite3rebaser_rebase_strm(sqlite3_rebaser * p,int (* xInput)(void * pIn,void * pData,int * pnData),void * pIn,int (* xOutput)(void * pOut,const void * pData,int nData),void * pOut)5755c0a499eaSdan int sqlite3rebaser_rebase_strm(
5756c0a499eaSdan   sqlite3_rebaser *p,
5757c0a499eaSdan   int (*xInput)(void *pIn, void *pData, int *pnData),
5758c0a499eaSdan   void *pIn,
5759c0a499eaSdan   int (*xOutput)(void *pOut, const void *pData, int nData),
5760c0a499eaSdan   void *pOut
5761c0a499eaSdan ){
5762c0a499eaSdan   sqlite3_changeset_iter *pIter = 0;   /* Iterator to skip through input */
5763c0a499eaSdan   int rc = sqlite3changeset_start_strm(&pIter, xInput, pIn);
5764c0a499eaSdan 
5765c0a499eaSdan   if( rc==SQLITE_OK ){
5766c0a499eaSdan     rc = sessionRebase(p, pIter, xOutput, pOut, 0, 0);
5767c0a499eaSdan     sqlite3changeset_finalize(pIter);
5768c0a499eaSdan   }
5769c0a499eaSdan 
5770c0a499eaSdan   return rc;
5771c0a499eaSdan }
5772c0a499eaSdan 
5773c0a499eaSdan /*
5774c0a499eaSdan ** Destroy a rebaser object
5775c0a499eaSdan */
sqlite3rebaser_delete(sqlite3_rebaser * p)5776f1b40e83Sdan void sqlite3rebaser_delete(sqlite3_rebaser *p){
5777c0a499eaSdan   if( p ){
57780cb735b9Sdan     sessionDeleteTable(0, p->grp.pList);
5779c0a499eaSdan     sqlite3_free(p);
5780c0a499eaSdan   }
5781c0a499eaSdan }
5782c0a499eaSdan 
57831f48e67dSdan /*
57841f48e67dSdan ** Global configuration
57851f48e67dSdan */
sqlite3session_config(int op,void * pArg)57861f48e67dSdan int sqlite3session_config(int op, void *pArg){
57871f48e67dSdan   int rc = SQLITE_OK;
57881f48e67dSdan   switch( op ){
57891f48e67dSdan     case SQLITE_SESSION_CONFIG_STRMSIZE: {
57901f48e67dSdan       int *pInt = (int*)pArg;
57911f48e67dSdan       if( *pInt>0 ){
57921f48e67dSdan         sessions_strm_chunk_size = *pInt;
57931f48e67dSdan       }
57941f48e67dSdan       *pInt = sessions_strm_chunk_size;
57951f48e67dSdan       break;
57961f48e67dSdan     }
57971f48e67dSdan     default:
57981f48e67dSdan       rc = SQLITE_MISUSE;
57991f48e67dSdan       break;
58001f48e67dSdan   }
58011f48e67dSdan   return rc;
58021f48e67dSdan }
58031f48e67dSdan 
58049b1c62d4Sdrh #endif /* SQLITE_ENABLE_SESSION && SQLITE_ENABLE_PREUPDATE_HOOK */
5805