• Home
  • Features
  • Pricing
  • Docs
  • Announcements
  • Sign In

MerginMaps / geodiff / 26440904450

26 May 2026 08:20AM UTC coverage: 87.996% (-0.1%) from 88.114%
26440904450

Pull #252

github

web-flow
Merge c14e0f642 into 0a94b5ba4
Pull Request #252: Allow schema changes in diffs

1056 of 1166 new or added lines in 13 files covered. (90.57%)

13 existing lines in 3 files now uncovered.

4237 of 4815 relevant lines covered (88.0%)

610.96 hits per line

Source File
Press 'n' to go to next uncovered line, 'b' for previous

93.29
/geodiff/src/changesetreader.cpp
1
/*
2
 GEODIFF - MIT License
3
 Copyright (C) 2020 Martin Dobias
4
*/
5

6
#include "changesetreader.h"
7

8
#include "changeset.h"
9
#include "geodiffutils.hpp"
10
#include "changesetgetvarint.h"
11
#include "portableendian.h"
12
#include "sqliteutils.h"
13
#include "tableschema.h"
14

15
#include <assert.h>
16
#include <memory.h>
17

18
#include <sstream>
19

20

21
ChangesetReader::ChangesetReader() = default;
887✔
22

23
ChangesetReader::~ChangesetReader() = default;
887✔
24

25

26
bool ChangesetReader::open( const std::string &filename )
888✔
27
{
28
  try
29
  {
30
    mBuffer.reset( new Buffer );
888✔
31
    mBuffer->read( filename );
888✔
32
  }
33
  catch ( const GeoDiffException & )
10✔
34
  {
35
    return false;
10✔
36
  }
10✔
37

38
  return true;
878✔
39
}
40

41
bool ChangesetReader::nextEntry( ChangesetEntry &entry )
2,446✔
42
{
43
  while ( 1 )
44
  {
45
    if ( mOffset >= mBuffer->size() )
3,354✔
46
      break;   // EOF
758✔
47

48
    ChangesetEntryType type = static_cast<ChangesetEntryType>( readByte() );
2,596✔
49
    if ( type == ChangesetEntryType::OpTableRecord )
2,596✔
50
    {
51
      readTableRecord();
908✔
52
      // and now continue reading, we want an entry
53
    }
54
    else if ( type == ChangesetEntryType::OpInsert || type == ChangesetEntryType::OpUpdate || type == ChangesetEntryType::OpDelete )
1,688✔
55
    {
56
      entry = readDataEntry( type );
1,648✔
57
      return true;  // we're done!
1,648✔
58
    }
59
    else if ( type == ChangesetEntryType::OpCreateTable )
40✔
60
    {
61
      entry = readCreateTableEntry();
11✔
62
      return true;
11✔
63
    }
64
    else if ( type == ChangesetEntryType::OpDropTable )
29✔
65
    {
66
      entry = readDropTableEntry();
8✔
67
      return true;
8✔
68
    }
69
    else if ( type == ChangesetEntryType::OpAddColumn )
21✔
70
    {
71
      entry = readAddColumnEntry();
11✔
72
      return true;
11✔
73
    }
74
    else if ( type == ChangesetEntryType::OpDropColumn )
10✔
75
    {
76
      entry = readDropColumnEntry();
8✔
77
      return true;
8✔
78
    }
79
    else
80
    {
81
      throwReaderError( "Unknown entry type " + std::to_string( static_cast<int>( type ) ) );
6✔
82
    }
83
  }
908✔
84
  return false;
758✔
85
}
86

87
bool ChangesetReader::isEmpty() const
393✔
88
{
89
  return mBuffer->size() == 0;
393✔
90
}
91

92
void ChangesetReader::rewind()
64✔
93
{
94
  mOffset = 0;
64✔
95
  mCurrentTable = {};
64✔
96
}
64✔
97

98
char ChangesetReader::readByte()
15,199✔
99
{
100
  if ( mOffset >= mBuffer->size() )
15,199✔
101
    throwReaderError( "readByte: at the end of buffer" );
×
102
  const char *ptr = mBuffer->c_buf() + mOffset;
15,199✔
103
  ++mOffset;
15,199✔
104
  return *ptr;
15,199✔
105
}
106

107
int ChangesetReader::readVarint()
3,628✔
108
{
109
  u32 value;
110
  const unsigned char *ptr = reinterpret_cast<const unsigned char *>( mBuffer->c_buf() ) + mOffset;
3,628✔
111
  int nBytes = getVarint32( ptr, value );
3,628✔
112
  mOffset += nBytes;
3,628✔
113
  return value;
3,628✔
114
}
115

116
std::string ChangesetReader::readNullTerminatedString()
1,098✔
117
{
118
  const char *ptr = mBuffer->c_buf() + mOffset;
1,098✔
119
  int count = 0;
1,098✔
120
  while ( mOffset + count < mBuffer->size() && ptr[count] )
7,925✔
121
    ++count;
6,827✔
122

123
  if ( mOffset + count >= mBuffer->size() )
1,098✔
124
    throwReaderError( "readNullTerminatedString: at the end of buffer" );
×
125

126
  mOffset += count + 1;
1,098✔
127
  return std::string( ptr, count );
2,196✔
128
}
129

130
void ChangesetReader::readRowValues( std::vector<Value> &values )
2,042✔
131
{
132
  // let's ensure we have the right size of array
133
  if ( values.size() != mCurrentTable->columnCount() )
2,042✔
134
  {
135
    values.resize( mCurrentTable->columnCount() );
2,042✔
136
  }
137

138
  for ( size_t i = 0; i < mCurrentTable->columnCount(); ++i )
9,552✔
139
  {
140
    int type = readByte();
7,510✔
141
    if ( type == Value::TypeInt ) // 0x01
7,510✔
142
    {
143
      // 64-bit int (big endian)
144
      int64_t v;
145
      uint64_t x;
146
      memcpy( &x, mBuffer->c_buf() + mOffset, 8 );
2,719✔
147
      mOffset += 8;
2,719✔
148
      x = be64toh( x ); // convert big endian to host
2,719✔
149
      memcpy( &v, &x, 8 );
2,719✔
150
      values[i].setInt( v );
2,719✔
151
    }
152
    else if ( type == Value::TypeDouble ) // 0x02
4,791✔
153
    {
154
      // 64-bit double (big endian)
155
      double v;
156
      uint64_t x;
157
      memcpy( &x, mBuffer->c_buf() + mOffset, 8 );
61✔
158
      mOffset += 8;
61✔
159
      x = be64toh( x ); // convert big endian to host
61✔
160
      memcpy( &v, &x, 8 );
61✔
161
      values[i].setDouble( v );
61✔
162
    }
163
    else if ( type == Value::TypeText || type == Value::TypeBlob ) // 0x03 or 0x04
4,730✔
164
    {
165
      int len = readVarint();
2,625✔
166
      if ( mOffset + len > mBuffer->size() )
2,625✔
167
        throwReaderError( "readRowValues: text/blob: at the end of buffer" );
×
168
      values[i].setString( type == Value::TypeText ? Value::TypeText : Value::TypeBlob, mBuffer->c_buf() + mOffset, len );
2,625✔
169
      mOffset += len;
2,625✔
170
    }
2,625✔
171
    else if ( type == Value::TypeNull ) // 0x05
2,105✔
172
    {
173
      values[i].setNull();
513✔
174
    }
175
    else if ( type == Value::TypeUndefined )  // undefined value  (different from NULL)
1,592✔
176
    {
177
      values[i].setUndefined();
1,592✔
178
    }
179
    else
180
    {
181
      throwReaderError( "readRowValues: unexpected entry type" );
×
182
    }
183
  }
184
}
2,042✔
185

186
void ChangesetReader::readTableRecord()
908✔
187
{
188
  /* A 'table' record consists of:
189
  **
190
  **   * A constant 'T' character,
191
  **   * Number of columns in said table (a varint),
192
  **   * An array of nCol bytes (sPK),
193
  **   * A nul-terminated table name.
194
  */
195

196
  int nCol = readVarint();
908✔
197
  if ( nCol < 0 || nCol > 65536 )
908✔
198
    throwReaderError( "readByte: unexpected number of columns" );
×
199

200
  mCurrentTable = std::make_shared<ChangesetTable>();
908✔
201
  mCurrentTable->primaryKeys.clear();
908✔
202

203
  for ( int i = 0; i < nCol; ++i )
4,201✔
204
  {
205
    mCurrentTable->primaryKeys.push_back( readByte() );
3,293✔
206
  }
207

208
  mCurrentTable->name = readNullTerminatedString();
908✔
209
}
908✔
210

211
ChangesetDataEntry ChangesetReader::readDataEntry( ChangesetEntryType type )
1,648✔
212
{
213
  ChangesetDataEntry entry;
1,648✔
214
  readByte();
1,648✔
215
  if ( type != ChangesetEntryType::OpInsert )
1,648✔
216
    readRowValues( entry.oldValues );
661✔
217
  else
218
    entry.oldValues.erase( entry.oldValues.begin(), entry.oldValues.end() );
987✔
219
  if ( type != ChangesetEntryType::OpDelete )
1,648✔
220
    readRowValues( entry.newValues );
1,381✔
221
  else
222
    entry.newValues.erase( entry.newValues.begin(), entry.newValues.end() );
267✔
223

224
  entry.op = static_cast<ChangesetDataEntry::OperationType>( type );
1,648✔
225
  entry.table = mCurrentTable;
1,648✔
226
  return entry;
1,648✔
NEW
227
}
×
228

229
TableColumnInfo ChangesetReader::readColumnInfo()
76✔
230
{
231
  TableColumnInfo column;
76✔
232
  column.name = readNullTerminatedString();
76✔
233
  column.type.baseType = static_cast<TableColumnType::BaseType>( readByte() );
76✔
234
  column.type.dbType = column.type.baseTypeToString( column.type.baseType );
76✔
235
  char flags = readByte();
76✔
236
  column.isPrimaryKey = flags & 1;
76✔
237
  column.isNotNull = flags & ( 1 << 1 );
76✔
238
  column.isAutoIncrement = flags & ( 1 << 2 );
76✔
239
  column.isGeometry = flags & ( 1 << 3 );
76✔
240
  column.geomHasZ = flags & ( 1 << 4 );
76✔
241
  column.geomHasM = flags & ( 1 << 5 );
76✔
242
  column.geomType = readNullTerminatedString();
76✔
243
  column.geomSrsId = readVarint();
76✔
244
  return column;
76✔
UNCOV
245
}
×
246

247
ChangesetCreateTableEntry ChangesetReader::readCreateTableEntry()
11✔
248
{
249
  ChangesetCreateTableEntry entry;
11✔
250
  entry.tableName = readNullTerminatedString();
11✔
251
  int columnCount = readVarint();
11✔
252
  entry.columns.resize( columnCount );
11✔
253
  for ( size_t i = 0; i < entry.columns.size(); i++ )
44✔
254
  {
255
    entry.columns[i] = readColumnInfo();
33✔
256
  }
257
  return entry;
11✔
NEW
258
}
×
259

260
ChangesetDropTableEntry ChangesetReader::readDropTableEntry()
8✔
261
{
262
  ChangesetDropTableEntry entry;
8✔
263
  entry.tableName = readNullTerminatedString();
8✔
264
  int columnCount = readVarint();
8✔
265
  entry.columns.resize( columnCount );
8✔
266
  for ( size_t i = 0; i < entry.columns.size(); i++ )
32✔
267
  {
268
    entry.columns[i] = readColumnInfo();
24✔
269
  }
270
  return entry;
8✔
NEW
271
}
×
272

273
ChangesetAddColumnEntry ChangesetReader::readAddColumnEntry()
11✔
274
{
275
  ChangesetAddColumnEntry entry;
11✔
276
  entry.tableName = readNullTerminatedString();
11✔
277
  entry.column = readColumnInfo();
11✔
278
  return entry;
11✔
NEW
279
}
×
280

281
ChangesetDropColumnEntry ChangesetReader::readDropColumnEntry()
8✔
282
{
283
  ChangesetDropColumnEntry entry;
8✔
284
  entry.tableName = readNullTerminatedString();
8✔
285
  entry.column = readColumnInfo();
8✔
286
  return entry;
8✔
NEW
287
}
×
288

289
void ChangesetReader::throwReaderError( const std::string &message ) const
2✔
290
{
291
  std::ostringstream stringStream;
2✔
292
  stringStream << "Reader error at offset " << mOffset << ":\n" << message;
2✔
293
  std::string str = stringStream.str();
2✔
294
  throw GeoDiffException( str );
2✔
295
}
4✔
STATUS · Troubleshooting · Open an Issue · Sales · Support · CAREERS · ENTERPRISE · START FREE · SCHEDULE DEMO
ANNOUNCEMENTS · TWITTER · TOS & SLA · Supported CI Services · What's a CI service? · Automated Testing

© 2026 Coveralls, Inc