-
Notifications
You must be signed in to change notification settings - Fork 98
/
Copy pathsqlite_migrations.js
677 lines (667 loc) · 31.6 KB
/
sqlite_migrations.js
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
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
/*jslint node: true */
"use strict";
var eventBus = require('./event_bus.js');
var constants = require("./constants.js");
var conf = require("./conf.js");
var VERSION = 46;
var async = require('async');
var bCordova = (typeof window === 'object' && window.cordova);
function migrateDb(connection, onDone){
connection.db[bCordova ? 'query' : 'all']("PRAGMA user_version", function(err, result){
if (err)
throw Error("PRAGMA user_version failed: "+err);
var rows = bCordova ? result.rows : result;
if (rows.length !== 1)
throw Error("PRAGMA user_version returned "+rows.length+" rows");
var version = rows[0].user_version;
console.log("db version "+version+", software version "+VERSION);
if (version > VERSION)
throw Error("user version "+version+" > "+VERSION+": looks like you are using a new database with an old client");
if (version === VERSION)
return onDone();
var bLongUpgrade = (version < 31 && !conf.bLight);
eventBus.emit('started_db_upgrade', bLongUpgrade);
if (typeof window === 'undefined'){
var message = bLongUpgrade ? "=== will upgrade the database, it will take several hours" : "=== will upgrade the database, it can take some time";
console.error(message);
console.log(message);
}
var arrQueries = [];
let units_sql;
async.series([
function (cb) {
connection.query("SELECT sql from sqlite_master WHERE type='table' AND name='units'", ([{ sql }]) => {
units_sql = sql;
cb();
});
},
function(cb){
if (version < 1){
connection.addQuery(arrQueries, "CREATE INDEX IF NOT EXISTS unitAuthorsIndexByAddressDefinitionChash ON unit_authors(address, definition_chash)");
connection.addQuery(arrQueries, "CREATE INDEX IF NOT EXISTS outputsIsSerial ON outputs(is_serial)");
connection.addQuery(arrQueries, "CREATE INDEX IF NOT EXISTS bySequence ON units(sequence)");
}
if (version < 2){
connection.addQuery(arrQueries, "CREATE UNIQUE INDEX IF NOT EXISTS hcobyAddressMci ON headers_commission_outputs(address, main_chain_index)");
connection.addQuery(arrQueries, "CREATE UNIQUE INDEX IF NOT EXISTS byWitnessAddressMci ON witnessing_outputs(address, main_chain_index)");
connection.addQuery(arrQueries, "CREATE INDEX IF NOT EXISTS inputsIndexByAddressTypeToMci ON inputs(address, type, to_main_chain_index)");
connection.addQuery(arrQueries, "DELETE FROM known_bad_joints");
}
if (version < 5){
connection.addQuery(arrQueries, "CREATE TABLE IF NOT EXISTS push_registrations (registrationId TEXT, device_address TEXT NOT NULL, PRIMARY KEY (device_address))");
}
if (version < 6){
connection.addQuery(arrQueries, "CREATE TABLE IF NOT EXISTS chat_messages ( \n\
id INTEGER PRIMARY KEY, \n\
correspondent_address CHAR(33) NOT NULL, \n\
message LONGTEXT NOT NULL, \n\
creation_date TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP, \n\
is_incoming INTEGER(1) NOT NULL, \n\
type CHAR(15) NOT NULL DEFAULT 'text', \n\
FOREIGN KEY (correspondent_address) REFERENCES correspondent_devices(device_address) \n\
)");
connection.addQuery(arrQueries, "CREATE INDEX IF NOT EXISTS chatMessagesIndexByDeviceAddress ON chat_messages(correspondent_address, id)");
connection.addQuery(arrQueries, "ALTER TABLE correspondent_devices ADD COLUMN my_record_pref INTEGER DEFAULT 1");
connection.addQuery(arrQueries, "ALTER TABLE correspondent_devices ADD COLUMN peer_record_pref INTEGER DEFAULT 1");
connection.addQuery(arrQueries, "DELETE FROM known_bad_joints");
}
if (version < 8) {
connection.addQuery(arrQueries, "CREATE INDEX IF NOT EXISTS bySequence ON units(sequence)");
connection.addQuery(arrQueries, "DELETE FROM known_bad_joints");
}
if(version < 9){
connection.addQuery(arrQueries, "CREATE TABLE IF NOT EXISTS watched_light_units (peer VARCHAR(100) NOT NULL, unit CHAR(44) NOT NULL, creation_date TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP, PRIMARY KEY (peer, unit))");
connection.addQuery(arrQueries, "CREATE INDEX IF NOT EXISTS wlabyUnit ON watched_light_units(unit)");
}
if(version < 10){
connection.addQuery(arrQueries, "BEGIN TRANSACTION");
connection.addQuery(arrQueries, "ALTER TABLE chat_messages RENAME TO chat_messages_old");
connection.addQuery(arrQueries, "CREATE TABLE IF NOT EXISTS chat_messages ( \n\
id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT, \n\
correspondent_address CHAR(33) NOT NULL, \n\
message LONGTEXT NOT NULL, \n\
creation_date TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP, \n\
is_incoming INTEGER(1) NOT NULL, \n\
type CHAR(15) NOT NULL DEFAULT 'text', \n\
FOREIGN KEY (correspondent_address) REFERENCES correspondent_devices(device_address) ON DELETE CASCADE \n\
)");
connection.addQuery(arrQueries, "INSERT INTO chat_messages SELECT * FROM chat_messages_old");
connection.addQuery(arrQueries, "DROP TABLE chat_messages_old");
connection.addQuery(arrQueries, "CREATE INDEX chatMessagesIndexByDeviceAddress ON chat_messages(correspondent_address, id);");
connection.addQuery(arrQueries, "COMMIT");
connection.addQuery(arrQueries, "DELETE FROM known_bad_joints");
connection.addQuery(arrQueries, "DELETE FROM unhandled_joints");
connection.addQuery(arrQueries, "DELETE FROM dependencies");
connection.addQuery(arrQueries, "DELETE FROM hash_tree_balls");
connection.addQuery(arrQueries, "DELETE FROM catchup_chain_balls");
}
if (version < 11) {
connection.addQuery(arrQueries, "CREATE TABLE IF NOT EXISTS bots ( \n\
id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT, \n\
rank INTEGER NOT NULL DEFAULT 0, \n\
name VARCHAR(100) NOT NULL UNIQUE, \n\
pairing_code VARCHAR(200) NOT NULL, \n\
description LONGTEXT NOT NULL \n\
);");
}
if (version < 12)
connection.addQuery(arrQueries, "DELETE FROM known_bad_joints");
if (version < 13){
connection.addQuery(arrQueries, "ALTER TABLE unit_authors ADD COLUMN _mci INT NULL");
connection.addQuery(arrQueries, "PRAGMA user_version=13");
}
if (version < 14){
connection.addQuery(arrQueries, "UPDATE unit_authors SET _mci=(SELECT main_chain_index FROM units WHERE units.unit=unit_authors.unit)");
connection.addQuery(arrQueries, "CREATE INDEX IF NOT EXISTS unitAuthorsIndexByAddressMci ON unit_authors(address, _mci)");
}
if (version < 15){
connection.addQuery(arrQueries, "CREATE TABLE IF NOT EXISTS asset_metadata ( \n\
asset CHAR(44) NOT NULL PRIMARY KEY, \n\
metadata_unit CHAR(44) NOT NULL, \n\
registry_address CHAR(32) NULL, \n\
suffix VARCHAR(20) NULL, \n\
name VARCHAR(20) NULL, \n\
decimals TINYINT NULL, \n\
creation_date TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP, \n\
UNIQUE (name, registry_address), \n\
FOREIGN KEY (asset) REFERENCES assets(unit), \n\
FOREIGN KEY (metadata_unit) REFERENCES units(unit) \n\
)");
}
if (version < 16){
connection.addQuery(arrQueries, "CREATE TABLE IF NOT EXISTS sent_mnemonics ( \n\
unit CHAR(44) NOT NULL, \n\
address CHAR(32) NOT NULL, \n\
mnemonic VARCHAR(107) NOT NULL, \n\
textAddress VARCHAR(120) NOT NULL, \n\
creation_date TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP, \n\
FOREIGN KEY (unit) REFERENCES units(unit) \n\
)");
connection.addQuery(arrQueries, "CREATE INDEX IF NOT EXISTS sentByAddress ON sent_mnemonics(address)");
connection.addQuery(arrQueries, "CREATE INDEX IF NOT EXISTS sentByUnit ON sent_mnemonics(unit)");
connection.addQuery(arrQueries, "DELETE FROM known_bad_joints");
}
if (version < 17){
connection.addQuery(arrQueries, "CREATE TABLE IF NOT EXISTS private_profiles ( \n\
private_profile_id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT, \n\
unit CHAR(44) NOT NULL, \n\
payload_hash CHAR(44) NOT NULL, \n\
attestor_address CHAR(32) NOT NULL, \n\
address CHAR(32) NOT NULL, \n\
src_profile TEXT NOT NULL, \n\
creation_date TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP, \n\
FOREIGN KEY (unit) REFERENCES units(unit) \n\
)");
connection.addQuery(arrQueries, "CREATE TABLE IF NOT EXISTS private_profile_fields ( \n\
private_profile_id INTEGER NOT NULL , \n\
`field` VARCHAR(50) NOT NULL, \n\
`value` VARCHAR(50) NOT NULL, \n\
blinding CHAR(16) NOT NULL, \n\
creation_date TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP, \n\
UNIQUE (private_profile_id, `field`), \n\
FOREIGN KEY (private_profile_id) REFERENCES private_profiles(private_profile_id) \n\
)");
connection.addQuery(arrQueries, "CREATE INDEX IF NOT EXISTS ppfByField ON private_profile_fields(`field`)");
}
cb();
},
function(cb){
if (version < 18){
connection.addQuery(arrQueries, "CREATE TABLE IF NOT EXISTS attested_fields ( \n\
unit CHAR(44) NOT NULL, \n\
message_index TINYINT NOT NULL, \n\
attestor_address CHAR(32) NOT NULL, \n\
address CHAR(32) NOT NULL, \n\
`field` VARCHAR(50) NOT NULL, \n\
`value` VARCHAR(100) NOT NULL, \n\
PRIMARY KEY (unit, message_index, `field`), \n\
"+(conf.bLight ? '' : "CONSTRAINT attestationsByAttestorAddress FOREIGN KEY (attestor_address) REFERENCES addresses(address),")+" \n\
FOREIGN KEY (unit) REFERENCES units(unit) \n\
)");
connection.addQuery(arrQueries,
"CREATE INDEX IF NOT EXISTS attestedFieldsByAttestorFieldValue ON attested_fields(attestor_address, `field`, `value`)");
connection.addQuery(arrQueries, "CREATE INDEX IF NOT EXISTS attestedFieldsByAddressField ON attested_fields(address, `field`)");
connection.addQuery(arrQueries, "CREATE TABLE IF NOT EXISTS original_addresses ( \n\
unit CHAR(44) NOT NULL, \n\
address CHAR(32) NOT NULL, \n\
original_address VARCHAR(100) NOT NULL, \n\
PRIMARY KEY (unit, address), \n\
FOREIGN KEY (unit) REFERENCES units(unit) \n\
)");
connection.query(
"SELECT unit, message_index, attestor_address, address, payload FROM attestations CROSS JOIN messages USING(unit, message_index)",
function(rows){
rows.forEach(function(row){
var attestation = JSON.parse(row.payload);
if (attestation.address !== row.address)
throw Error("attestation.address !== row.address");
for (var field in attestation.profile){
var value = attestation.profile[field];
if (field.length <= constants.MAX_PROFILE_FIELD_LENGTH && typeof value === 'string' && value.length <= constants.MAX_PROFILE_VALUE_LENGTH){
connection.addQuery(arrQueries,
"INSERT "+connection.getIgnore()+" INTO attested_fields \n\
(unit, message_index, attestor_address, address, field, value) VALUES(?,?, ?,?, ?,?)",
[row.unit, row.message_index, row.attestor_address, row.address, field, value]);
}
}
});
cb();
}
);
}
else
cb();
},
function(cb){
if (version < 19)
connection.addQuery(arrQueries, "CREATE INDEX IF NOT EXISTS outputsIsSerial ON outputs(is_serial)");
if (version < 20)
connection.addQuery(arrQueries, "DELETE FROM known_bad_joints");
if (version < 21)
connection.addQuery(arrQueries, "ALTER TABLE push_registrations ADD COLUMN platform TEXT NOT NULL DEFAULT 'android'");
if (version < 22)
connection.addQuery(arrQueries, "CREATE INDEX IF NOT EXISTS sharedAddressSigningPathsByDeviceAddress ON shared_address_signing_paths(device_address);");
if (version < 23){
connection.addQuery(arrQueries, "CREATE TABLE IF NOT EXISTS peer_addresses ( \n\
address CHAR(32) NOT NULL, \n\
signing_paths VARCHAR(255) NULL, \n\
device_address CHAR(33) NOT NULL, \n\
definition TEXT NULL, \n\
creation_date TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP, \n\
PRIMARY KEY (address), \n\
FOREIGN KEY (device_address) REFERENCES correspondent_devices(device_address) \n\
)");
connection.addQuery(arrQueries, "CREATE TABLE IF NOT EXISTS prosaic_contracts ( \n\
hash CHAR(44) NOT NULL PRIMARY KEY, \n\
peer_address CHAR(32) NOT NULL, \n\
peer_device_address CHAR(33) NOT NULL, \n\
my_address CHAR(32) NOT NULL, \n\
is_incoming TINYINT NOT NULL, \n\
creation_date TIMESTAMP NOT NULL, \n\
ttl INT NOT NULL DEFAULT 168, -- 168 hours = 24 * 7 = 1 week \n\
status TEXT CHECK (status IN('pending', 'revoked', 'accepted', 'declined')) NOT NULL DEFAULT 'active', \n\
title VARCHAR(1000) NOT NULL, \n\
`text` TEXT NOT NULL, \n\
shared_address CHAR(32), \n\
unit CHAR(44), \n\
cosigners VARCHAR(1500), \n\
FOREIGN KEY (my_address) REFERENCES my_addresses(address) \n\
)");
}
if (version < 24){
connection.addQuery(arrQueries, "BEGIN TRANSACTION");
connection.addQuery(arrQueries, "CREATE TABLE asset_attestors_new ( \n\
unit CHAR(44) NOT NULL, \n\
message_index TINYINT NOT NULL, \n\
asset CHAR(44) NOT NULL, -- in the initial attestor list: same as unit \n\
attestor_address CHAR(32) NOT NULL, \n\
PRIMARY KEY (unit, message_index, attestor_address), \n\
UNIQUE (asset, attestor_address, unit), \n\
FOREIGN KEY (unit) REFERENCES units(unit), \n\
CONSTRAINT assetAttestorsByAsset FOREIGN KEY (asset) REFERENCES assets(unit) \n\
)");
connection.addQuery(arrQueries, "INSERT INTO asset_attestors_new SELECT * FROM asset_attestors");
connection.addQuery(arrQueries, "DROP TABLE asset_attestors");
connection.addQuery(arrQueries, "ALTER TABLE asset_attestors_new RENAME TO asset_attestors");
connection.addQuery(arrQueries, "COMMIT");
}
if (version < 25)
connection.addQuery(arrQueries, "ALTER TABLE correspondent_devices ADD COLUMN is_blackhole TINYINT NOT NULL DEFAULT 0");
if (version < 26){
connection.addQuery(arrQueries, "ALTER TABLE correspondent_devices ADD COLUMN push_enabled TINYINT NOT NULL DEFAULT 1");
connection.addQuery(arrQueries, "CREATE TABLE IF NOT EXISTS correspondent_settings ( \n\
device_address CHAR(33) NOT NULL, \n\
correspondent_address CHAR(33) NOT NULL, \n\
push_enabled TINYINT NOT NULL, \n\
creation_date TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP, \n\
PRIMARY KEY (device_address, correspondent_address) \n\
)");
connection.addQuery(arrQueries, "PRAGMA user_version=26");
}
if (version < 27){
connection.addQuery(arrQueries, "CREATE UNIQUE INDEX IF NOT EXISTS unqPayloadHash ON private_profiles(payload_hash)");
}
if (version < 28){
connection.addQuery(arrQueries, "ALTER TABLE units ADD COLUMN timestamp INT NOT NULL DEFAULT 0");
connection.addQuery(arrQueries, "PRAGMA user_version=28");
}
if (version < 29)
connection.addQuery(arrQueries, "DELETE FROM known_bad_joints");
if (version < 30) {
connection.addQuery(arrQueries, "CREATE TABLE IF NOT EXISTS joints ( \n\
unit CHAR(44) NOT NULL PRIMARY KEY, \n\
json TEXT NOT NULL, \n\
creation_date TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP \n\
)");
connection.addQuery(arrQueries, "CREATE TABLE IF NOT EXISTS aa_addresses ( \n\
address CHAR(32) NOT NULL PRIMARY KEY, \n\
unit CHAR(44) NOT NULL, -- where it is first defined. No index for better speed \n\
mci INT NOT NULL, -- it is available since this mci (mci of the above unit) \n\
definition TEXT NOT NULL, \n\
creation_date TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP \n\
)");
connection.addQuery(arrQueries, "CREATE TABLE IF NOT EXISTS aa_triggers ( \n\
mci INT NOT NULL, \n\
unit CHAR(44) NOT NULL, \n\
address CHAR(32) NOT NULL, \n\
creation_date TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP, \n\
PRIMARY KEY (mci, unit, address), \n\
FOREIGN KEY (address) REFERENCES aa_addresses(address) \n\
)");
connection.addQuery(arrQueries, "CREATE TABLE IF NOT EXISTS aa_balances ( \n\
address CHAR(32) NOT NULL, \n\
asset CHAR(44) NOT NULL, -- 'base' for bytes (NULL would not work for uniqueness of primary key) \n\
balance BIGINT NOT NULL, \n\
creation_date TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP, \n\
PRIMARY KEY (address, asset), \n\
FOREIGN KEY (address) REFERENCES aa_addresses(address) \n\
-- FOREIGN KEY (asset) REFERENCES assets(unit) \n\
)");
connection.addQuery(arrQueries, "CREATE TABLE IF NOT EXISTS aa_responses ( \n\
aa_response_id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT, \n\
mci INT NOT NULL, -- mci of the trigger unit \n\
trigger_address CHAR(32) NOT NULL, -- trigger address \n\
aa_address CHAR(32) NOT NULL, \n\
trigger_unit CHAR(44) NOT NULL, \n\
bounced TINYINT NOT NULL, \n\
response_unit CHAR(44) NULL UNIQUE, \n\
response TEXT NULL, -- json \n\
creation_date TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP, \n\
UNIQUE (trigger_unit, aa_address), \n\
"+(conf.bLight ? "" : "FOREIGN KEY (aa_address) REFERENCES aa_addresses(address),")+" \n\
FOREIGN KEY (trigger_unit) REFERENCES units(unit) \n\
-- FOREIGN KEY (response_unit) REFERENCES units(unit) \n\
)");
connection.addQuery(arrQueries, "CREATE INDEX IF NOT EXISTS aaResponsesByTriggerAddress ON aa_responses(trigger_address)");
connection.addQuery(arrQueries, "CREATE INDEX IF NOT EXISTS aaResponsesByAAAddress ON aa_responses(aa_address)");
connection.addQuery(arrQueries, "CREATE INDEX IF NOT EXISTS aaResponsesByMci ON aa_responses(mci)");
connection.addQuery(arrQueries, "PRAGMA user_version=30");
}
cb();
},
function(cb){
if (version < 31) {
async.series(arrQueries, function () {
require('./migrate_to_kv.js')(connection, function () {
arrQueries = [];
cb();
});
});
}
else
cb();
},
function(cb){
if (version < 32)
connection.addQuery(arrQueries, "CREATE TABLE IF NOT EXISTS my_watched_addresses (\n\
address CHAR(32) NOT NULL PRIMARY KEY,\n\
creation_date TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP\n\
)");
if (version < 33) {
connection.addQuery(arrQueries, "ALTER TABLE aa_addresses ADD COLUMN storage_size INT NOT NULL DEFAULT 0");
connection.addQuery(arrQueries, "PRAGMA user_version=33");
}
cb();
},
function (cb) {
if (version < 34)
initStorageSizes(connection, arrQueries, cb);
else
cb();
},
function (cb) {
if (version < 35)
connection.addQuery(arrQueries, "REPLACE INTO aa_balances (address, asset, balance) \n\
SELECT address, IFNULL(asset, 'base'), SUM(amount) AS balance \n\
FROM aa_addresses \n\
CROSS JOIN outputs USING(address) \n\
CROSS JOIN units ON outputs.unit=units.unit \n\
WHERE is_spent=0 AND ( \n\
is_stable=1 \n\
OR EXISTS (SELECT 1 FROM unit_authors CROSS JOIN aa_addresses USING(address) WHERE unit_authors.unit=outputs.unit) \n\
) \n\
GROUP BY address, asset");
if (version < 36) {
connection.addQuery(arrQueries, "CREATE TABLE IF NOT EXISTS watched_light_aas ( \n\
peer VARCHAR(100) NOT NULL, \n\
aa CHAR(32) NOT NULL, \n\
address CHAR(32) NULL, \n\
creation_date TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP, \n\
PRIMARY KEY (peer, aa, address) \n\
)");
connection.addQuery(arrQueries, "CREATE INDEX IF NOT EXISTS wlaabyAA ON watched_light_aas(aa)");
}
if (version < 37) {
connection.addQuery(arrQueries, "ALTER TABLE aa_addresses ADD COLUMN base_aa CHAR(32) NULL" + (conf.bLight ? "" : " CONSTRAINT aaAddressesByBaseAA REFERENCES aa_addresses(address)"));
connection.addQuery(arrQueries, "PRAGMA user_version=37");
}
if (version < 38)
connection.addQuery(arrQueries, "CREATE INDEX IF NOT EXISTS byBaseAA ON aa_addresses(base_aa)");
cb();
},
function (cb) {
if (version < 39)
addTypesToStateVars(cb);
else
cb();
},
function (cb) {
if (version < 40) {
connection.addQuery(arrQueries, "ALTER TABLE aa_addresses ADD COLUMN getters TEXT NULL");
connection.addQuery(arrQueries, "PRAGMA user_version=40");
}
if (version < 41)
connection.addQuery(arrQueries, "DELETE FROM known_bad_joints");
if (version < 42 && conf.bLight) {
connection.addQuery(arrQueries, "CREATE TABLE IF NOT EXISTS unprocessed_addresses (\n\
address CHAR(32) NOT NULL PRIMARY KEY,\n\
creation_date TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP\n\
);");
}
if (version < 43) {
connection.addQuery(arrQueries, "CREATE TABLE IF NOT EXISTS arbiter_locations ( \n\
arbiter_address CHAR(32) NOT NULL PRIMARY KEY, \n\
arbstore_address CHAR(32) NOT NULL, \n\
unit CHAR(44) NULL \n\
);");
connection.addQuery(arrQueries, "CREATE TABLE IF NOT EXISTS wallet_arbiters ( \n\
arbiter_address CHAR(32) NOT NULL PRIMARY KEY, \n\
real_name VARCHAR(250) NULL, \n\
device_pub_key VARCHAR(44) NULL \n\
);");
connection.addQuery(arrQueries, "CREATE TABLE IF NOT EXISTS wallet_arbiter_contracts ( \n\
hash CHAR(44) NOT NULL PRIMARY KEY, \n\
peer_address CHAR(32) NOT NULL, \n\
peer_device_address CHAR(33) NOT NULL, \n\
my_address CHAR(32) NOT NULL, \n\
arbiter_address CHAR(32) NOT NULL, \n\
me_is_payer TINYINT NOT NULL, \n\
amount BIGINT NULL, \n\
asset CHAR(44) NULL, \n\
is_incoming TINYINT NOT NULL, \n\
me_is_cosigner TINYINT NULL, \n\
creation_date TIMESTAMP NOT NULL, \n\
ttl INT NOT NULL DEFAULT 168, -- 168 hours = 24 * 7 = 1 week \n\
status VARCHAR CHECK (status IN('pending', 'revoked', 'accepted', 'signed', 'declined', 'paid', 'in_dispute', 'dispute_resolved', 'in_appeal', 'appeal_approved', 'appeal_declined', 'cancelled', 'completed')) NOT NULL DEFAULT 'pending', \n\
title VARCHAR(1000) NOT NULL, \n\
text TEXT NOT NULL, \n\
my_contact_info TEXT NULL, \n\
peer_contact_info TEXT NULL, \n\
peer_pairing_code VARCHAR(200) NULL, \n\
shared_address CHAR(32) NULL UNIQUE, \n\
unit CHAR(44) NULL, \n\
cosigners VARCHAR(1500), \n\
resolution_unit CHAR(44) NULL, \n\
arbstore_address CHAR(32) NULL, \n\
arbstore_device_address CHAR(33) NULL, \n\
FOREIGN KEY (my_address) REFERENCES my_addresses(address) \n\
)");
connection.addQuery(arrQueries, "CREATE INDEX IF NOT EXISTS wacStatus ON wallet_arbiter_contracts(status)");
connection.addQuery(arrQueries, "CREATE INDEX IF NOT EXISTS wacArbiterAddress ON wallet_arbiter_contracts(arbiter_address)");
connection.addQuery(arrQueries, "CREATE INDEX IF NOT EXISTS wacPeerAddress ON wallet_arbiter_contracts(peer_address)");
connection.addQuery(arrQueries, "CREATE TABLE IF NOT EXISTS arbiter_disputes (\n\
contract_hash CHAR(44) NOT NULL PRIMARY KEY,\n\
plaintiff_address CHAR(32) NOT NULL,\n\
respondent_address CHAR(32) NOT NULL,\n\
plaintiff_is_payer TINYINT(1) NOT NULL,\n\
plaintiff_pairing_code VARCHAR(200) NOT NULL,\n\
respondent_pairing_code VARCHAR(200) NOT NULL,\n\
contract_content TEXT NOT NULL,\n\
contract_unit CHAR(44) NOT NULL,\n\
amount BIGINT NOT NULL,\n\
asset CHAR(44) NULL,\n\
arbiter_address CHAR(32) NOT NULL,\n\
service_fee_asset CHAR(44) NULL,\n\
arbstore_device_address CHAR(33) NOT NULL,\n\
status VARCHAR(40) CHECK (status IN('pending', 'resolved')) NOT NULL DEFAULT 'pending',\n\
creation_date TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP,\n\
plaintiff_contact_info TEXT NULL,\n\
respondent_contact_info TEXT NULL,\n\
FOREIGN KEY (arbstore_device_address) REFERENCES correspondent_devices(device_address)\n\
)");
connection.addQuery(arrQueries, "DROP TABLE IF EXISTS asset_metadata");
connection.addQuery(arrQueries, "CREATE TABLE IF NOT EXISTS asset_metadata ( \n\
asset CHAR(44) NOT NULL PRIMARY KEY, \n\
metadata_unit CHAR(44) NOT NULL, \n\
registry_address CHAR(32) NULL, \n\
suffix VARCHAR(20) NULL, \n\
name VARCHAR(20) NULL, \n\
decimals TINYINT NULL, \n\
creation_date TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP, \n\
UNIQUE (name, registry_address), \n\
FOREIGN KEY (asset) REFERENCES assets(unit), \n\
FOREIGN KEY (metadata_unit) REFERENCES units(unit) \n\
)");
}
if (version < 44 && !conf.bLight && constants.bTestnet)
connection.addQuery(arrQueries, "REPLACE INTO aa_balances (address, asset, balance) \n\
SELECT address, IFNULL(asset, 'base'), SUM(amount) AS balance \n\
FROM aa_addresses \n\
CROSS JOIN outputs USING(address) \n\
CROSS JOIN units ON outputs.unit=units.unit \n\
WHERE is_spent=0 AND address='SLBA27JAT5UJBMQGDQLAT3FQ467XDOGF' AND ( \n\
is_stable=1 \n\
OR EXISTS (SELECT 1 FROM unit_authors CROSS JOIN aa_addresses USING(address) WHERE unit_authors.unit=outputs.unit) \n\
) \n\
GROUP BY address, asset");
if (version < 45) {
connection.addQuery(arrQueries, "ALTER TABLE wallet_arbiter_contracts ADD COLUMN my_party_name VARCHAR(100) NULL");
connection.addQuery(arrQueries, "ALTER TABLE wallet_arbiter_contracts ADD COLUMN peer_party_name VARCHAR(100) NULL");
connection.addQuery(arrQueries, "PRAGMA user_version=45");
}
if (version < 46) {
if (!conf.bLight) {
connection.addQuery(arrQueries, `CREATE TABLE IF NOT EXISTS system_votes (
unit CHAR(44) NOT NULL,
address CHAR(32) NOT NULL,
subject VARCHAR(50) NOT NULL,
value TEXT NOT NULL,
timestamp INT NOT NULL,
creation_date TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP,
PRIMARY KEY (unit, address, subject)
-- FOREIGN KEY (unit) REFERENCES units(unit)
)`);
connection.addQuery(arrQueries, `CREATE INDEX IF NOT EXISTS bySysVotesAddress ON system_votes(address)`);
connection.addQuery(arrQueries, `CREATE INDEX IF NOT EXISTS bySysVotesSubjectAddress ON system_votes(subject, address)`);
connection.addQuery(arrQueries, `CREATE INDEX IF NOT EXISTS bySysVotesSubjectTimestamp ON system_votes(subject, timestamp)`);
connection.addQuery(arrQueries, `CREATE TABLE IF NOT EXISTS op_votes (
unit CHAR(44) NOT NULL,
address CHAR(32) NOT NULL,
op_address CHAR(32) NOT NULL,
timestamp INT NOT NULL,
creation_date TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP,
PRIMARY KEY (address, op_address)
-- FOREIGN KEY (unit) REFERENCES units(unit)
)`);
connection.addQuery(arrQueries, `CREATE INDEX IF NOT EXISTS byOpVotesTs ON op_votes(timestamp)`);
connection.addQuery(arrQueries, `CREATE TABLE IF NOT EXISTS numerical_votes (
unit CHAR(44) NOT NULL,
address CHAR(32) NOT NULL,
subject VARCHAR(50) NOT NULL,
value DOUBLE NOT NULL,
timestamp INT NOT NULL,
creation_date TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP,
PRIMARY KEY (address, subject)
-- FOREIGN KEY (unit) REFERENCES units(unit)
)`);
connection.addQuery(arrQueries, `CREATE INDEX IF NOT EXISTS byNumericalVotesSubjectTs ON numerical_votes(subject, timestamp)`);
connection.addQuery(arrQueries, `CREATE TABLE IF NOT EXISTS system_vars (
subject VARCHAR(50) NOT NULL,
value TEXT NOT NULL,
vote_count_mci INT NOT NULL, -- applies since the next mci
is_emergency TINYINT NOT NULL DEFAULT 0,
creation_date TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP,
PRIMARY KEY (subject, vote_count_mci DESC)
)`);
connection.addQuery(arrQueries, `CREATE TABLE IF NOT EXISTS tps_fees_balances (
address CHAR(32) NOT NULL,
mci INT NOT NULL,
tps_fees_balance INT NOT NULL DEFAULT 0, -- can be negative
creation_date TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP,
PRIMARY KEY (address, mci DESC)
)`);
connection.addQuery(arrQueries, `CREATE TABLE IF NOT EXISTS node_vars (
name VARCHAR(30) NOT NULL PRIMARY KEY,
value TEXT NOT NULL,
last_update TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP
)`);
connection.addQuery(arrQueries, `INSERT OR IGNORE INTO node_vars (name, value) VALUES ('last_temp_data_purge_mci', ?)`, [constants.v4UpgradeMci]);
}
if (!units_sql.includes('oversize_fee')) {
console.log('no oversize_fee column yet');
connection.addQuery(arrQueries, "ALTER TABLE units ADD COLUMN oversize_fee INT NULL");
connection.addQuery(arrQueries, "ALTER TABLE units ADD COLUMN tps_fee INT NULL");
connection.addQuery(arrQueries, "ALTER TABLE units ADD COLUMN actual_tps_fee INT NULL");
connection.addQuery(arrQueries, "ALTER TABLE units ADD COLUMN burn_fee INT NULL");
connection.addQuery(arrQueries, "ALTER TABLE units ADD COLUMN max_aa_responses INT NULL");
connection.addQuery(arrQueries, "ALTER TABLE units ADD COLUMN count_aa_responses INT NULL");
connection.addQuery(arrQueries, "ALTER TABLE units ADD COLUMN is_aa_response TINYINT NULL");
connection.addQuery(arrQueries, "ALTER TABLE units ADD COLUMN count_primary_aa_triggers TINYINT NULL");
}
else
console.log('already have oversize_fee column');
connection.addQuery(arrQueries, `UPDATE units SET is_aa_response=1 WHERE unit IN (SELECT response_unit FROM aa_responses)`);
connection.addQuery(arrQueries, `UPDATE units
SET count_primary_aa_triggers=(SELECT COUNT(*) FROM aa_responses WHERE trigger_unit=unit)
WHERE is_aa_response!=1 AND unit IN (SELECT trigger_unit FROM aa_responses)
`);
}
cb();
},
],
function(){
connection.addQuery(arrQueries, "PRAGMA user_version="+VERSION);
async.series(arrQueries, function(){
eventBus.emit('finished_db_upgrade');
if (typeof window === 'undefined'){
console.error("=== db upgrade finished");
console.log("=== db upgrade finished");
}
onDone();
});
});
});
}
function initStorageSizes(connection, arrQueries, cb){
if (bCordova)
return cb();
var options = {};
options.gte = "st\n";
options.lte = "st\n\uFFFF";
var assocSizes = {};
var handleData = function (data) {
var address = data.key.substr(3, 32);
var var_name = data.key.substr(36);
if (!assocSizes[address])
assocSizes[address] = 0;
assocSizes[address] += var_name.length + data.value.length;
}
var kvstore = require('./kvstore.js');
var stream = kvstore.createReadStream(options);
stream.on('data', handleData)
.on('end', function(){
for (var address in assocSizes)
connection.addQuery(arrQueries, "UPDATE aa_addresses SET storage_size=? WHERE address=?", [assocSizes[address], address]);
cb();
})
.on('error', function(error){
throw Error('error from data stream: '+error);
});
}
function addTypesToStateVars(cb){
if (bCordova || conf.bLight)
return cb();
var string_utils = require("./string_utils.js");
var kvstore = require('./kvstore.js');
var batch = kvstore.batch();
var options = {};
options.gte = "st\n";
options.lte = "st\n\uFFFF";
var bOldFormat = false;
var handleData = function (data) {
if (data.value.split("\n", 2).length < 2) // check if already upgraded
bOldFormat = true; // if at least one non-upgraded value found, then we didn't upgrade yet
var f = string_utils.getNumericFeedValue(data.value); // use old rules to convert strings to numbers
var type = (f !== null) ? 'n' : 's';
batch.put(data.key, type + "\n" + data.value);
}
var stream = kvstore.createReadStream(options);
stream.on('data', handleData)
.on('end', function () {
if (!bOldFormat) {
console.log("state vars already upgraded");
batch.clear();
return cb();
}
batch.write(function(err){
if (err)
throw Error("writer: batch write failed: " + err);
console.log("done upgrading state vars");
cb();
});
})
.on('error', function(error){
throw Error('error from data stream: ' + error);
});
}
exports.migrateDb = migrateDb;