forked from litepresence/Graphene-Metanode
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathgraphene_sql.py
493 lines (481 loc) · 13.7 KB
/
graphene_sql.py
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
#!/usr/bin/env python
# DISABLE SELECT PYLINT TESTS
# pylint: disable=broad-except
# pylint: disable=bad-continuation, too-many-branches
# import pdb; pdb.set_trace()
r"""
╔════════════════════════════════════════════════════╗
║ ╔═╗╦═╗╔═╗╔═╗╦ ╦╔═╗╔╗╔╔═╗ ╔╦╗╔═╗╔╦╗╔═╗╔╗╔╔═╗╔╦╗╔═╗ ║
║ ║ ╦╠╦╝╠═╣╠═╝╠═╣║╣ ║║║║╣ ║║║║╣ ║ ╠═╣║║║║ ║ ║║║╣ ║
║ ╚═╝╩╚═╩ ╩╩ ╩ ╩╚═╝╝╚╝╚═╝ ╩ ╩╚═╝ ╩ ╩ ╩╝╚╝╚═╝═╩╝╚═╝ ║
╚════════════════════════════════════════════════════╝
~
CREATE THE METANODE SQL DATABASE AND PROVIDE A SAFE READ / WRITE WRAPPER
~
mavens are streaming windowed lists of json data
colleced directly from public api nodes user whitelist as "mostly trustworthy"
regenerative multiprcocessing prevents failed sockets from hanging main process
the statiscial mode of these lists is moved to the respective base table
note in some cases a REAL or INTEGER may be a TEXT as maven, eg.
maven.account.fees.cancel = "[0.2, 0.2, 0.2, 0.1]" ->
account.fees.cancel = 0.2
"""
# STANDARD MODULES
import json
import os
import time
from sqlite3 import OperationalError, Row, connect
# GRAPHENE MODULES
# ~ *soon* from hummingbot.connector.exchange.graphene.
from graphene_constants import GrapheneConstants
from graphene_utils import it, jprint
# GLOBAL CONSTANTS
DEV = False
PATH = os.path.dirname(os.path.abspath(__file__)) + "/database"
CREATES = [
"""
CREATE TABLE chain (
name TEXT PRIMARY KEY,
id TEXT UNIQUE
)
""",
"""
CREATE TABLE account (
name TEXT PRIMARY KEY,
id TEXT UNIQUE,
fees_account TEXT,
ltm INT,
cancels TEXT
)
""",
"""
CREATE TABLE nodes (
url TEXT PRIMARY KEY,
ping REAL,
handshake REAL,
blocktime INT,
code INT,
status TEXT
)
""",
"""
CREATE TABLE objects (
id TEXT PRIMARY KEY,
name TEXT,
precision TEXT
)
""",
"""
CREATE TABLE timing (
name TEXT,
blocknum INT,
blocktime INT,
server REAL,
ping REAL,
handshake REAL,
read REAL,
begin REAL
)
""",
"""
CREATE TABLE assets (
name TEXT PRIMARY KEY,
id TEXT UNIQUE,
precision INT,
supply REAL,
fees_asset TEXT,
balance TEXT
)
""",
"""
CREATE TABLE pairs (
name TEXT PRIMARY KEY,
id TEXT UNIQUE,
invert_pair TEXT UNIQUE,
invert_id TEXT UNIQUE,
ops TEXT,
last REAL,
book TEXT,
history TEXT,
opens TEXT,
fills TEXT
)
""",
"""
CREATE TABLE maven_account (
name TEXT PRIMARY KEY,
fees_account TEXT,
ltm TEXT,
cancels TEXT
)
""",
"""
CREATE TABLE maven_assets (
name TEXT PRIMARY KEY,
supply TEXT,
fees_asset TEXT,
balance TEXT
)
""",
"""
CREATE TABLE maven_pairs (
name TEXT PRIMARY KEY,
ops TEXT,
last TEXT,
book TEXT,
history TEXT,
opens TEXT,
fills TEXT
)
""",
"""
CREATE TABLE maven_timing (
name TEXT,
blocknum TEXT,
blocktime TEXT,
read TEXT
)
""",
]
SELECTS = [
"""
SELECT * FROM chain
""",
"""
SELECT * FROM nodes
""",
"""
SELECT * FROM objects
""",
"""
SELECT * FROM account
""",
"""
SELECT * FROM maven_account
""",
"""
SELECT * FROM timing
""",
"""
SELECT * FROM maven_timing
""",
"""
SELECT * FROM assets
""",
"""
SELECT * FROM maven_assets
""",
"""
SELECT * FROM pairs
""",
"""
SELECT * FROM maven_pairs
""",
]
UPDATES = [
(
"""
UPDATE nodes SET ping=?, code=?, status=?
""",
("999.9", "1000", "INITIALIZING"),
),
(
"""
UPDATE account SET fees_account=?, ltm=?, cancels=?
""",
("{}", "0", "[]"),
),
(
"""
UPDATE assets SET precision=?, supply=?, fees_asset=?, balance=?
""",
("0", "0.0", "{}", "{}"),
),
(
"""
UPDATE pairs SET last=?, book=?, history=?, opens=?, fills=?, ops=?
""",
("0", "{}", "[]", "[]", "[]", "[]"),
),
(
"""
UPDATE maven_account SET fees_account=?, ltm=?, cancels=?
""",
("[]", "[]", "[]"),
),
(
"""
UPDATE maven_assets SET supply=?, fees_asset=?, balance=?
""",
("[]", "[]", "[]"),
),
(
"""
UPDATE maven_pairs SET last=?, book=?, history=?, opens=?, fills=?, ops=?
""",
("[]", "[]", "[]", "[]", "[]", "[]"),
),
(
"""
UPDATE maven_timing SET blocknum=?, blocktime=?, read=?
""",
("[]", "[]", "[]"),
),
]
class Sql:
"""
creation of hummingbot graphene database and execution of queries
"""
def __init__(self, constants):
self.constants = constants
def restart(self):
"""
delete any existing db and initialize new SQL db
"""
# create database folder
os.makedirs(PATH, exist_ok=True)
# user input w/ warning
print("\033c")
print(
it("red", "WARNING THIS SCRIPT WILL RESTART DATABASE AND ERASE ALL DATA\n")
)
# erase the database
command = f"rm {self.constants.chain.DATABASE}"
print("\033c", it("red", command), "\n")
os.system(command)
print("creating sqlite3:", it("green", self.constants.chain.DATABASE), "\n")
# initialize insert operations with chain specific configuration
inserts = [
(
"""
INSERT INTO chain (name, id) VALUES (?,?)
""",
(
self.constants.chain.NAME,
self.constants.chain.ID,
),
),
(
"""
INSERT INTO timing
(name, ping, handshake, blocktime, blocknum, read, begin)
VALUES (?,?,?,?,?,?,?)
""",
(
self.constants.chain.ACCOUNT,
9999,
9999,
0,
0,
9999,
self.constants.metanode.BEGIN,
),
),
(
"""
INSERT INTO maven_timing (name, blocktime, blocknum, read)
VALUES (?,?,?,?)
""",
(
self.constants.chain.ACCOUNT,
"[]",
"[]",
"[]",
),
),
(
"""
INSERT INTO account (name) VALUES (?)
""",
(self.constants.chain.ACCOUNT,),
),
(
"""
INSERT INTO objects (id, name) VALUES (?,?)
""",
(
"1.3.0",
self.constants.chain.CORE,
),
),
(
"""
INSERT INTO maven_account (name) VALUES (?)
""",
(self.constants.chain.ACCOUNT,),
),
]
for asset in self.constants.chain.ASSETS:
inserts.append(
(
"""
INSERT INTO assets (name) VALUES (?)
""",
(asset,),
)
)
inserts.append(
(
"""
INSERT INTO maven_assets (name) VALUES (?)
""",
(asset,),
)
)
for pair in self.constants.chain.PAIRS:
inserts.append(
(
"""
INSERT INTO pairs (name) VALUES (?)
""",
(pair,),
)
)
inserts.append(
(
"""
INSERT INTO maven_pairs (name) VALUES (?)
""",
(pair,),
)
)
for node in self.constants.chain.NODES:
inserts.append(
(
"""
INSERT INTO nodes (url) VALUES (?)
""",
(node,),
)
)
# new table creation
queries = []
for query in CREATES:
dml = {"query": query, "values": tuple()}
queries.append(dml)
self.execute(queries)
# row creation in each table
queries = []
for insert in inserts:
dml = {"query": insert[0], "values": insert[1]}
queries.append(dml)
self.execute(queries)
# default column data in each row
queries = []
for update in UPDATES:
dml = {"query": update[0], "values": update[1]}
queries.append(dml)
self.execute(queries)
# print
if DEV:
for query in SELECTS:
jprint(self.execute(query))
def execute(self, query, values=()):
"""
execute discrete sql queries, handle race condition gracefully
if query is a string, assume values is a
else, query can be a list of dicts with keys ["query","values"]
While True:
Try:
con = connect(DB)
cur = con.cursor()
cur.execute(query, values)
ret = cur.fetchall()
con.commit()
con.close()
break
Except:
continue
:return ret:
"""
queries = []
# handle both single query and multiple queries
if isinstance(query, str):
queries.append({"query": query, "values": values})
else:
queries = query
# strip double spaces and new lines in each query
for idx, dml in enumerate(queries):
queries[idx]["query"] = " ".join(dml["query"].replace("\n", " ").split())
# print sql except when...
for dml in queries:
if DEV:
print(it("yellow", f"'query': {dml['query']}"))
print(it("green", f"'values': {dml['values']}\n"))
# attempt to update database until satisfied
pause = -1
curfetchall = None
while True:
try:
pause += 1
# only allow batched write queries
if len(queries) > 1:
for dml in queries:
if "SELECT" in dml["query"]:
raise ValueError("batch queries must be write only")
# ======================================================================
# SQL CONNECT
# ======================================================================
con = connect(self.constants.chain.DATABASE)
for dml in queries:
con.row_factory = Row
cur = con.cursor()
cur.execute(dml["query"], dml["values"])
curfetchall = cur.fetchall()
con.commit()
con.close()
# ======================================================================
# SQL CLOSE
# ======================================================================
data = [dict(i) for i in curfetchall]
for idx, row in enumerate(data):
for key, val in row.items():
# these are sql REAL, but TEXT when maven_
if (
key
in [
"ltm",
"supply",
"last",
]
and "maven_" in dml["query"]
):
data[idx][key] = json.loads(val)
# these are valid json sql REAL
elif key in [
"fees_account",
"fees_asset",
"balance",
"book",
"history",
"ops",
"opens",
"fills",
]:
data[idx][key] = json.loads(val)
return data
except OperationalError:
if DEV:
print("Race condition at", int(time.time()))
# ascending pause here prevents excess cpu on corruption of database
# and allows for decreased load during race condition
time.sleep(min(5, 1.01 ** pause - 1))
continue
def unit_test():
"""
initialize the database
"""
print("\033c")
constants = GrapheneConstants()
dispatch = {str(idx): chain for idx, chain in enumerate(constants.core.CHAINS)}
for key, value in dispatch.items():
if "testnet" not in value:
print(key + ": " + it("blue", value))
else:
print(key + ": " + it("purple", value))
chain = dispatch[input("Enter choice: ")]
constants = GrapheneConstants(chain)
sql = Sql(constants)
sql.restart()
if __name__ == "__main__":
unit_test()