-
Notifications
You must be signed in to change notification settings - Fork 6
/
metadata_writer_sqlite.h
346 lines (285 loc) · 15.7 KB
/
metadata_writer_sqlite.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
/* Copyright (c) 2015, Celerway, Kristian Evensen <kristrev@celerway.com>
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are met:
*
* 1. Redistributions of source code must retain the above copyright notice,
* this list of conditions and the following disclaimer.
*
* 2. Redistributions in binary form must reproduce the above copyright notice,
* this list of conditions and the following disclaimer in the documentation
* and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
* ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
* LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
* POSSIBILITY OF SUCH DAMAGE.
*/
#pragma once
#include <sys/time.h>
#include <sqlite3.h>
#include "metadata_exporter.h"
#define DEFAULT_TIMEOUT 5000
#define TIMEOUT_FILE 1000
#define EVENT_LIMIT 10
#define MAX_PATH_LEN 128
#define FAKE_UPDATE_LIMIT 120
#define CREATE_SQL "CREATE TABLE IF NOT EXISTS NetworkEvent(" \
"NodeId INTEGER NOT NULL," \
"SessionId INTEGER NOT NULL," \
"SessionIdMultip INTEGER NOT NULL," \
"SimCardIccid TEXT NOT NULL," \
"SimCardImsi TEXT NOT NULL," \
"Timestamp INTEGER NOT NULL," \
"Sequence INTEGER NOT NULL," \
"L3SessionId INTEGER NOT NULL," \
"L4SessionId INTEGER NOT NULL DEFAULT 0," \
"EventType INTEGER NOT NULL," \
"EventParam INTEGER NOT NULL," \
"EventValue INTEGER," \
"HasIp INTEGER," \
"Connectivity INTEGER," \
"ConnectionMode INTEGER," \
"Quality INTEGER," \
"InterfaceType INTEGER NOT NULL," \
"InterfaceIdType INTEGER NOT NULL," \
"InterfaceId TEXT NOT NULL," \
"NetworkProvider INT," \
"NetworkAddressFamily INTEGER NOT NULL," \
"NetworkAddress TEXT NOT NULL," \
"PRIMARY KEY(SessionId,SessionIdMultip,"\
"SimCardIccid,SimCardImsi,InterfaceId,Timestamp,"\
"Sequence))"
#define CREATE_UPDATE_SQL "CREATE TABLE IF NOT EXISTS NetworkUpdates(" \
"NodeId INTEGER NOT NULL," \
"SessionId INTEGER NOT NULL," \
"SessionIdMultip INTEGER NOT NULL," \
"SimCardIccid TEXT NOT NULL," \
"SimCardImsi TEXT NOT NULL," \
"Timestamp INTEGER NOT NULL," \
"Sequence INTEGER NOT NULL," \
"L3SessionId INTEGER NOT NULL," \
"L4SessionId INTEGER NOT NULL DEFAULT 0," \
"EventType INTEGER NOT NULL," \
"EventParam INTEGER NOT NULL," \
"HasIp INTEGER," \
"Connectivity INTEGER," \
"ConnectionMode INTEGER," \
"Quality INTEGER," \
"InterfaceType INTEGER NOT NULL," \
"InterfaceId TEXT NOT NULL," \
"NetworkAddressFamily INTEGER NOT NULL," \
"NetworkAddress TEXT NOT NULL," \
"NetworkProvider INT," \
"EventValueStr TEXT, " \
"PRIMARY KEY(SessionId,SessionIdMultip,"\
"SimCardIccid,SimCardImsi,"\
"L3SessionId,L4SessionId,InterfaceId,"\
"NetworkAddressFamily,NetworkAddress))"
#define CREATE_GPS_SQL "CREATE TABLE IF NOT EXISTS GpsUpdate(" \
"NodeId INTEGER NOT NULL," \
"BootCount INTEGER,"\
"BootMultiplier INTEGER,"\
"Timestamp INTEGER NOT NULL," \
"Sequence INTEGER NOT NULL," \
"EventType INTEGER NOT NULL," \
"EventParam INTEGER NOT NULL," \
"Latitude REAL NOT NULL," \
"Longitude REAL NOT NULL," \
"Altitude REAL," \
"Speed REAL," \
"SatelliteCount INTEGER," \
"PRIMARY KEY(NodeId) ON CONFLICT REPLACE)"
#define CREATE_MONITOR_SQL "CREATE TABLE IF NOT EXISTS MonitorEvents(" \
"NodeId INTEGER NOT NULL," \
"Timestamp INTEGER NOT NULL," \
"Sequence INTEGER NOT NULL," \
"Boottime INTEGER NOT NULL," \
"PRIMARY KEY(NodeId,Timestamp,Sequence))"
#define CREATE_USAGE_SQL "CREATE TABLE IF NOT EXISTS DataUse(" \
"NodeId INTEGER NOT NULL," \
"DeviceId TEXT NOT NULL," \
"NetworkAddressFamily INTEGER NOT NULL," \
"EventType INTEGER NOT NULL," \
"EventParam INTEGER NOT NULL," \
"SimCardIccid TEXT NOT NULL," \
"SimCardImsi TEXT NOT NULL," \
"Timestamp INTEGER NOT NULL," \
"RxData INTEGER NOT NULL,"\
"TxData INTEGER NOT NULL," \
"PRIMARY KEY(DeviceId,NetworkAddressFamily,SimCardIccid,SimCardImsi,Timestamp))"
#define CREATE_REBOOT_SQL "CREATE TABLE IF NOT EXISTS RebootEvent(" \
"NodeId INTEGER NOT NULL," \
"BootCount INTEGER," \
"BootMultiplier INTEGER," \
"Timestamp INTEGER NOT NULL," \
"Sequence INTEGER NOT NULL," \
"EventType INTEGER NOT NULL," \
"DeviceId TEXT NOT NULL," \
"PRIMARY KEY(BootCount,BootMultiplier,Timestamp,Sequence))"
#define INSERT_EVENT "INSERT INTO NetworkEvent(NodeId,SessionId,"\
"SessionIdMultip,SimCardIccid,SimCardImsi,Timestamp,Sequence,L3SessionId,"\
"L4SessionId,EventType,EventParam,EventValue,"\
"HasIp,Connectivity,ConnectionMode,Quality,InterfaceType,"\
"InterfaceIdType,InterfaceId,NetworkProvider,NetworkAddressFamily,"\
"NetworkAddress) " \
"VALUES (?,?,?,?,?,?,?,?,?,?,?,?,?,?,?,?,?,?,?,?,?,?)"
#define INSERT_UPDATE "INSERT INTO NetworkUpdates(NodeId,SessionId,"\
"SessionIdMultip,SimCardIccid,SimCardImsi,Timestamp,Sequence,L3SessionId,"\
"L4SessionId,EventType,EventParam,HasIp,Connectivity,ConnectionMode,Quality"\
",InterfaceType,InterfaceId,NetworkAddressFamily,NetworkAddress,NetworkProvider,EventValueStr) " \
"VALUES (?,?,?,?,?,?,?,?,?,?,?,?,?,?,?,?,?,?,?,?,?)"
#define INSERT_GPS_EVENT "INSERT INTO GpsUpdate(NodeId,BootCount" \
",BootMultiplier,Timestamp" \
",Sequence,EventType,EventParam,Latitude,Longitude" \
",Altitude,Speed,SatelliteCount) " \
"VALUES (?,?,?,?,?,?,?,?,?,?,?,?)"
#define INSERT_MONITOR_EVENT "INSERT INTO MonitorEvents(NodeId,Timestamp" \
",Sequence,Boottime) " \
"VALUES (?,?,?,?)"
#define INSERT_USAGE "INSERT INTO DataUse(NodeId, DeviceId,NetworkAddressFamily,EventType,EventParam" \
",SimCardIccid" \
",SimCardImsi,Timestamp,RxData,TxData) " \
"VALUES (?, ?,?,?,?,?,?,?,?,?)"
#define INSERT_REBOOT_EVENT "INSERT INTO RebootEvent(NodeId, BootCount," \
"BootMultiplier, Timestamp, Sequence, EventType, DeviceId)"\
"VALUES (?,?,?,?,?,16,?)"
#define SELECT_LAST_UPDATE "SELECT HasIp,Connectivity,ConnectionMode,Quality "\
" FROM NetworkUpdates WHERE "\
"L3SessionId=? AND "\
"L4SessionId=? AND InterfaceId=? "\
"AND SimCardIccid=? AND SimCardImsi=? "\
"AND NetworkAddressFamily=? AND "\
"NetworkAddress=? ORDER BY Timestamp DESC LIMIT 1;"
//The reason NetworkAddress is used here, is that we first identify by L3/L4
//session. However, it might be that multiple addresses on same interface has
//the same L3/L4 ID (since we use timestamp). (L3 ID,address) is unique on one
//interface.
//
//We can have the same L3/L4 IDs on multiple nodes. On one node, the same
//address can have multiple L3/L4 IDs (two mf823 for example). However, same
//address (+prefix) and same interface is guaranteed to have unique L3/L4
#define UPDATE_UPDATE "UPDATE NetworkUpdates SET " \
"Timestamp=?,HasIp=?,Connectivity=?,ConnectionMode=?," \
"Quality=?,EventValueStr=? " \
"WHERE "\
"L3SessionId=? AND L4SessionId=? " \
"AND SimCardIccid=? AND SimCardImsi=?"\
"AND NetworkAddressFamily=? AND "\
"NetworkAddress=? AND InterfaceId=?"
#define UPDATE_USAGE "UPDATE DataUse SET " \
"RxData = RxData + ?, TxData = TxData + ? " \
"WHERE " \
"DeviceId=? AND NetworkAddressFamily=? AND SimCardIccid=? AND SimCardImsi=? AND Timestamp=?"
#define UPDATE_EVENT_ID "UPDATE NetworkEvent SET " \
"NodeId=? "\
"WHERE NodeId=0"
#define UPDATE_UPDATES_ID "UPDATE NetworkUpdates SET " \
"NodeId=? "\
"WHERE NodeId=0"
#define UPDATE_SYSTEM_ID "UPDATE RebootEvent SET " \
"NodeId=? "\
"WHERE NodeId=0"
#define UPDATE_GPS_ID "UPDATE GpsUpdate SET " \
"NodeId=? "\
"WHERE NodeId=0"
#define UPDATE_USAGE_ID "UPDATE DataUse SET " \
"NodeId=? " \
"WHERE NodeId=0"
#define UPDATE_EVENT_TSTAMP "UPDATE NetworkEvent SET " \
"Timestamp = (Timestamp - ?) + ? "\
"WHERE Timestamp < ?"
#define UPDATE_UPDATES_TSTAMP "UPDATE NetworkUpdates SET " \
"Timestamp = (Timestamp - ?) + ? "\
"WHERE Timestamp < ?"
#define UPDATE_SYSTEM_TSTAMP "UPDATE RebootEvent SET " \
"Timestamp = (Timestamp - ?) + ? "\
"WHERE Timestamp < ?"
#define UPDATE_GPS_TSTAMP "UPDATE GpsUpdate SET " \
"Timestamp = (Timestamp - ?) + ? "\
"WHERE Timestamp < ?"
#define UPDATE_EVENT_SESSION_ID "UPDATE NetworkEvent SET "\
"SessionId=?,SessionIdMultip=? "\
"WHERE SessionId = 0"
#define UPDATE_UPDATES_SESSION_ID "UPDATE NetworkUpdates SET "\
"SessionId=?,SessionIdMultip=? "\
"WHERE SessionId = 0"
#define UPDATE_SYSTEM_SESSION_ID "UPDATE RebootEvent SET "\
"BootCount=?,BootMultiplier=? "\
"WHERE BootCount = 0"
#define UPDATE_GPS_SESSION_ID "UPDATE GpsUpdate SET "\
"BootCount=?,BootMultiplier=? "\
"WHERE BootCount = 0"
#define DELETE_TABLE "DELETE FROM NetworkEvent"
#define DELETE_NW_UPDATE "DELETE FROM NetworkUpdates WHERE Timestamp < ?"
#define DELETE_GPS_TABLE "DELETE FROM GpsUpdate"
#define DELETE_MONITOR_TABLE "DELETE FROM MonitorEvents"
#define DELETE_USAGE_TABLE "DELETE FROM DataUse"
#define DELETE_SYSTEM_TABLE "DELETE FROM RebootEvent"
//Define statements for JSON export
#define DUMP_EVENTS_JSON "SELECT * FROM NetworkEvent WHERE Timestamp>=? ORDER BY TimeStamp"
#define DUMP_UPDATES_JSON "SELECT * FROM NetworkUpdates WHERE Timestamp>=? ORDER BY TimeStamp"
#define DUMP_GPS_JSON "SELECT * FROM GpsUpdate ORDER BY Timestamp"
#define DUMP_MONITOR_JSON "SELECT * FROM MonitorEvents ORDER BY Timestamp"
#define DUMP_USAGE_JSON "SELECT * FROM DataUse"
#define DUMP_SYSTEM_JSON "SELECT * FROM RebootEvent"
struct md_event;
struct md_writer;
struct backend_timeout_handle;
struct md_writer_sqlite {
MD_WRITER;
uint64_t dump_tstamp;
uint64_t last_msg_tstamp;
uint64_t last_gps_insert;
uint64_t orig_boot_time;
uint64_t orig_uptime;
uint64_t orig_raw_time;
//TODO: Consider moving this to the generic writer struct if need be
//These values keep track of the unique session id (and multiplier), which
//are normally assumed to be the boot counter (+ multiplier)
uint64_t session_id;
uint64_t session_id_multip;
float gps_speed;
size_t meta_prefix_len, gps_prefix_len, monitor_prefix_len,
usage_prefix_len, system_prefix_len;
sqlite3 *db_handle;
sqlite3_stmt *insert_event, *insert_update;
sqlite3_stmt *update_update, *dump_update;
sqlite3_stmt *delete_table, *dump_table;
sqlite3_stmt *last_update;
sqlite3_stmt *insert_gps, *delete_gps, *dump_gps;
sqlite3_stmt *insert_monitor, *delete_monitor, *dump_monitor;
sqlite3_stmt *insert_usage, *update_usage, *dump_usage, *delete_usage;
sqlite3_stmt *insert_system, *dump_system, *delete_system;
char *session_id_file;
char *node_id_file;
const char *last_conn_tstamp_path;
struct backend_timeout_handle *timeout_handle;
struct timeval first_fake_update;
uint32_t node_id;
uint32_t db_interval;
uint32_t db_events;
uint32_t num_conn_events;
uint32_t num_gps_events;
uint32_t num_munin_events;
uint32_t num_usage_events;
uint32_t num_system_events;
uint8_t timeout_added;
uint8_t file_failed;
uint8_t do_fake_updates;
uint8_t valid_timestamp;
char meta_prefix[128], gps_prefix[128], monitor_prefix[128],
usage_prefix[128], system_prefix[128], ntp_fix_file[128];
uint8_t api_version;
uint8_t delete_conn_update;
};
void md_sqlite_usage();
void md_sqlite_setup(struct md_exporter *mde, struct md_writer_sqlite* mws);