forked from litepresence/Graphene-Metanode
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathgraphene_auth.py
551 lines (538 loc) · 25.4 KB
/
graphene_auth.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
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
#!/usr/bin/env python
# DISABLE SELECT PYLINT TESTS
# pylint: disable=bad-continuation, broad-except, too-many-locals, too-many-statements
# pylint: disable=import-error, too-many-branches, line-too-long
r"""
╔════════════════════════════════════════════════════╗
║ ╔═╗╦═╗╔═╗╔═╗╦ ╦╔═╗╔╗╔╔═╗ ╔╦╗╔═╗╔╦╗╔═╗╔╗╔╔═╗╔╦╗╔═╗ ║
║ ║ ╦╠╦╝╠═╣╠═╝╠═╣║╣ ║║║║╣ ║║║║╣ ║ ╠═╣║║║║ ║ ║║║╣ ║
║ ╚═╝╩╚═╩ ╩╩ ╩ ╩╚═╝╝╚╝╚═╝ ╩ ╩╚═╝ ╩ ╩ ╩╝╚╝╚═╝═╩╝╚═╝ ║
╚════════════════════════════════════════════════════╝
~
AUTHENTICATE, CREATE ORDER HEADERS, BUILD TRANSACTIONS IN GRAPHENE TERMS
THEN FROM HIGH LEVEL, SERIALIZE, SIGN, VERIFY, AND BROADCAST
"""
# STANDARD MODULES
import json
import time
from binascii import unhexlify # hexidecimal to binary text
from collections import OrderedDict
from decimal import Decimal as decimal
from multiprocessing import Process, Value
from struct import unpack_from # convert back to PY variable
# GRAPHENE MODULES
# ~ *soon* from hummingbot.connector.exchange.graphene.
from graphene_metanode_server import GrapheneTrustlessClient
from graphene_rpc import RemoteProcedureCall
from graphene_signing import (ObjectId, PrivateKey, serialize_transaction,
sign_transaction, verify_transaction)
from graphene_utils import it, jprint, to_iso_date, trace
DEV = False
class GrapheneAuth:
"""
Auth class required by Graphene DEXs
"""
def __init__(self, constants: str, wif: str):
self.constants = constants
self.wif = wif
self.rpc = None
self.metanode = GrapheneTrustlessClient(self.constants)
def prototype_order(self, trading_pair=None, client_order_id=1):
"""
creates an auto formatted empty prototype order in json format
you will add your ['edicts'] and ['wif']
metaNODE handles everything else
usage
from manualSIGNING import prototype_order
order = json_loads(prototype_order())
order['header']['wif'] = wif
order['edicts'] = edicts
broker(order)
"""
proto = {}
# ==============================================================================
whitelist = self.metanode.whitelist # DISCRETE SQL QUERY
account = self.metanode.account # DISCRETE SQL QUERY
objects = self.metanode.objects # DISCRETE SQL QUERY
assets = self.metanode.assets # DISCRETE SQL QUERY
# ==============================================================================
core_precision = int(objects["1.3.0"]["precision"])
create = int(account["fees_account"]["create_graphene"])
cancel = int(account["fees_account"]["cancel_graphene"])
print(core_precision)
print(create, cancel)
proto["op"] = ""
proto["nodes"] = whitelist
proto["header"] = {
"wif": self.wif,
"account_id": account["id"],
"account_name": account["name"],
"client_order_id": client_order_id,
}
if trading_pair is not None:
asset, currency = trading_pair.split("-")
proto["header"].update(
{
"asset_id": assets[asset]["id"],
"currency_id": assets[currency]["id"],
"asset_name": asset,
"currency_name": currency,
"asset_precision": assets[asset]["precision"],
"currency_precision": assets[currency]["precision"],
"pair": asset + "-" + currency,
"fees": {
"create": create,
"cancel": cancel,
},
}
)
return json.dumps(proto)
def broker(self, order):
"""
"broker(order) --> _execute(signal, order)"
# insistent timed multiprocess wrapper for authorized ops
# covers all incoming buy/sell/cancel authenticated requests
# if command does not _execute in time: terminate and respawn
# serves to force disconnect websockets if hung
"up to self.constants.signing.ATTEMPTS chances;
each self.constants.signing.PROCESS_TIMEOUT long: else abort"
# signal is switched to 0 after execution to end the process
"""
self.rpc = RemoteProcedureCall(self.constants)
signal = Value("i", 0)
auth = Value("i", 0)
iteration = 0
while (iteration < self.constants.signing.ATTEMPTS) and not signal.value:
iteration += 1
print(
"\nmanualSIGNING authentication attempt:", iteration, time.ctime(), "\n"
)
child = Process(target=self._execute, args=(signal, auth, order))
child.daemon = False
child.start()
# if (
# self.constants.signing.JOIN
# ): # means main script will not continue till child done
child.join(self.constants.signing.PROCESS_TIMEOUT)
return bool(auth.value)
def _execute(
self,
signal,
auth,
order,
):
"""
build tx, serialize, sign, verify, broadcast
"""
def transact(order, auth):
trx = self._build_transaction(order)
# if there are any orders, perform ecdsa on serialized transaction
if trx["operations"]:
trx, message = serialize_transaction(trx, self.constants, self.rpc)
signed_tx = sign_transaction(
trx, message, wif, self.constants.chain.PREFIX
)
signed_tx = verify_transaction(signed_tx, wif, self.constants)
print(
self.rpc.broadcast_transaction(
signed_tx, order["header"]["client_order_id"]
)
)
auth.value = 1
msg = it("green", "EXECUTED ORDER")
else:
msg = it("red", "REJECTED ORDER")
return msg
wif = order["header"]["wif"]
start = time.time()
# if this is just an authentication test,
# then there is no serialization / signing
# just check that the private key references to the account id in question
msg = it("red", "FAILED FOR UNKNOWN REASON")
# do not allow mixed create/cancel edicts
if len(order["edicts"]) > 1:
for edict in order["edicts"]:
if edict["op"] == "cancel":
raise ValueError("batch edicts must not be cancel operations")
if order["edicts"][0]["op"] == "login":
msg = it("red", "LOGIN FAILED")
# instantitate a PrivateKey object
private_key = PrivateKey(wif, self.constants.chain.PREFIX)
print("PrivateKey init")
# which contains an Address object
address = private_key.address
# which contains str(PREFIX) and a Base58(pubkey)
# from these two, build a human terms "public key"
public_key = address.prefix + str(address.pubkey)
# get a key reference from that public key to 1.2.x account id
print(public_key)
key_reference_id = self.rpc.key_reference(public_key)
print(key_reference_id)
key_reference_id = key_reference_id[0][0]
# extract the account id in the metanode
account_id = order["header"]["account_id"]
print("wif account id", key_reference_id)
print("order account id", account_id)
# if they match we're authenticated
if account_id == key_reference_id:
auth.value = 1
msg = it("green", "AUTHENTICATED")
else:
try:
########################################################################
# "CANCEL ALL ONE MARKET"
########################################################################
if (
order["edicts"][0]["op"] == "cancel"
and "1.7.X" in order["edicts"][0]["ids"]
):
msg = it("red", "NO OPEN ORDERS")
open_orders = True
while (
len(
self.rpc.open_order_ids(
{
"currency_id": order["header"]["currency_id"],
"asset_id": order["header"]["asset_id"],
}
)
)
> 0
):
# ==============================================================
open_orders = self.metanode.pairs[order["header"]["pair"]][
"opens"
] # DISCRETE SQL QUERY
# ==============================================================
ids = order["edicts"][0]["ids"] = [
order["order_number"] for order in open_orders
]
if ids:
msg = transact(order, auth)
time.sleep(10)
########################################################################
# "CANCEL SOME ORDERS"
########################################################################
elif (
order["edicts"][0]["op"] == "cancel"
and "1.7.X" not in order["edicts"][0]["ids"]
):
jprint(order)
msg = transact(order, auth)
time.sleep(5)
########################################################################
# "BUY / SELL"
########################################################################
else:
msg = transact(order, auth)
except Exception as error:
print(trace(error))
stars = it("yellow", "*" * (len(msg) - 1))
msg = "manualSIGNING " + msg
print(stars + "\n " + msg + "\n" + stars)
print("process elapsed: %.3f sec" % (time.time() - start), "\n\n")
signal.value = 1
def login(self):
"""
:return bool(): True = authenticated
"""
order = self.prototype_order()
order["edicts"] = [{"op": "login"}]
return self.broker(order)
def _build_transaction(self, order):
"""
# this performs incoming limit order api conversion
# from human terms to graphene terms
# humans speak:
"account name, asset name, order number"
"decimal amounts, rounded is just fine"
"buy/sell/cancel"
"amount of assets"
"price in currency"
# graphene speaks:
"1.2.x, 1.3.x, 1.7.x"
"only in integers"
"create/cancel"
"min_to_receive/10^receiving_precision"
"amount_to_sell/10^selling_precision"
# _build_transaction speaks:
"list of buy/sell/cancel human terms edicts any order in"
"validated data request"
"autoscale amounts if out of budget"
"autoscale amounts if spending last bitshare"
"bundled cancel/buy/sell transactions out; cancel first"
"prevent inadvertent huge number of orders"
"do not place orders for dust amounts"
"""
# VALIDATE INCOMING DATA
if not isinstance(order["edicts"], list):
raise ValueError("order parameter must be list: %s" % order["edicts"])
if not isinstance(order["nodes"], list):
raise ValueError("order parameter must be list: %s" % order["nodes"])
if not isinstance(order["header"], dict):
raise ValueError("order parameter must be list: %s" % order["header"])
# the location of the decimal place must be provided by order
asset_precision = int(order["header"]["asset_precision"])
asset_id = str(order["header"]["asset_id"])
asset_name = str(order["header"]["asset_name"])
currency_name = str(order["header"]["currency_name"])
account_id = str(order["header"]["account_id"])
checks = [account_id, asset_id]
pair = str(order["header"]["pair"])
fees = dict(order["header"]["fees"])
# perform checks on currency for limit and call orders
if order["edicts"][0]["op"] in ["buy", "sell"]:
currency_precision = int(order["header"]["currency_precision"])
currency_id = str(order["header"]["currency_id"])
checks.append(currency_id)
# validate a.b.c identifiers of account id and asset ids
for check in checks:
ObjectId(check)
# GATHER TRANSACTION HEADER DATA
# fetch block data via websocket request
block = self.rpc.block_number_raw()
ref_block_num = block["head_block_number"] & 0xFFFF
ref_block_prefix = unpack_from("<I", unhexlify(block["head_block_id"]), 4)[0]
# establish transaction expiration
tx_expiration = to_iso_date(int(time.time() + 120))
# initialize tx_operations list
tx_operations = []
# SORT INCOMING EDICTS BY TYPE AND CONVERT TO DECIMAL
buy_edicts = []
sell_edicts = []
cancel_edicts = []
for edict in order["edicts"]:
if edict["op"] == "cancel":
print(it("yellow", str({k: str(v) for k, v in edict.items()})))
cancel_edicts.append(edict)
elif edict["op"] == "buy":
print(it("yellow", str({k: str(v) for k, v in edict.items()})))
buy_edicts.append(edict)
elif edict["op"] == "sell":
print(it("yellow", str({k: str(v) for k, v in edict.items()})))
sell_edicts.append(edict)
for idx, _ in enumerate(buy_edicts):
buy_edicts[idx]["amount"] = decimal(buy_edicts[idx]["amount"])
buy_edicts[idx]["price"] = decimal(buy_edicts[idx]["price"])
for idx, _ in enumerate(sell_edicts):
sell_edicts[idx]["amount"] = decimal(sell_edicts[idx]["amount"])
sell_edicts[idx]["price"] = decimal(sell_edicts[idx]["price"])
# TRANSLATE CANCEL ORDERS TO GRAPHENE
for edict in cancel_edicts:
if "ids" not in edict.keys():
edict["ids"] = ["1.7.X"]
if "1.7.X" in edict["ids"]: # the "cancel all" signal
# for cancel all op, we collect all open orders in 1 market
# FIXME
metanode_pairs = self.metanode.pairs
edict["ids"] = [order["id"] for order in metanode_pairs[pair]["opens"]]
print(it("yellow", str(edict)))
for order_id in edict["ids"]:
# confirm it is good 1.7.x format:
# FIXME this check should use ObjectId class instead of duplicate code
order_id = str(order_id)
aaa, bbb, ccc = order_id.split(".", 2)
assert int(aaa) == float(aaa) == 1
assert int(bbb) == float(bbb) == 7
assert int(ccc) == float(ccc) > 0
# create cancel fee ordered dictionary
fee = OrderedDict([("amount", fees["cancel"]), ("asset_id", "1.3.0")])
# create ordered operation dicitonary for this edict
operation = [
2, # two means "Limit_order_cancel"
OrderedDict(
[
("fee", fee),
("fee_paying_account", account_id),
("order", order_id),
("extensions", []),
]
),
]
# append the ordered dict to the trx operations list
tx_operations.append(operation)
# SCALE ORDER SIZE TO FUNDS ON HAND
if self.constants.signing.AUTOSCALE or self.constants.signing.CORE_FEES:
# ==========================================================================
metanode_assets = self.metanode.assets # DISCRETE SQL QUERY
metanode_objects = self.metanode.objects # DISCRETE SQL QUERY
# ==========================================================================
metanode_currency = metanode_assets[currency_name]
metanode_asset = metanode_assets[asset_name]
metanode_core = metanode_assets[metanode_objects["1.3.0"]["name"]]
currency_balance = metanode_currency["balance"]["free"]
asset_balance = metanode_asset["balance"]["free"]
core_balance = metanode_core["balance"]["free"]
if self.constants.signing.AUTOSCALE and buy_edicts + sell_edicts:
# autoscale buy edicts
if buy_edicts:
currency_value = 0
# calculate total value of each amount in the order
for idx, _ in enumerate(buy_edicts):
currency_value += (
buy_edicts[idx]["amount"] * buy_edicts[idx]["price"]
)
# scale the order amounts to means
scale = (
self.constants.core.DECIMAL_SIXSIG
* decimal(currency_balance)
/ (currency_value + self.constants.core.DECIMAL_SATOSHI)
)
if scale < 1:
print(
it(
"yellow",
"ALERT: scaling buy edicts to means: %.3f" % scale,
)
)
for idx, _ in enumerate(buy_edicts):
buy_edicts[idx]["amount"] *= scale
# autoscale sell edicts
if sell_edicts:
asset_total = 0
# calculate total amount in the order
for idx, _ in enumerate(sell_edicts):
asset_total += sell_edicts[idx]["amount"]
scale = (
self.constants.core.DECIMAL_SIXSIG
* decimal(asset_balance)
/ (asset_total + self.constants.core.DECIMAL_SATOSHI)
)
# scale the order amounts to means
if scale < 1:
print(
it(
"yellow",
"ALERT: scaling sell edicts to means: %.3f" % scale,
)
)
for idx, _ in enumerate(sell_edicts):
sell_edicts[idx]["amount"] *= scale
# ALWAYS SAVE LAST 2 BITSHARES FOR FEES
if self.constants.signing.CORE_FEES and (
buy_edicts + sell_edicts and ("1.3.0" in [asset_id, currency_id])
):
# print(bitshares, 'BTS balance')
# when BTS is the currency don't spend the last 2
if currency_id == "1.3.0" and buy_edicts:
bts_value = 0
# calculate total bts value of each amount in the order
for idx, _ in enumerate(buy_edicts):
bts_value += (
buy_edicts[idx]["amount"] * buy_edicts[idx]["price"]
)
# scale the order amounts to save last two bitshares
scale = (
self.constants.core.DECIMAL_SIXSIG
* max(0, (core_balance - 2))
/ (bts_value + self.constants.core.DECIMAL_SATOSHI)
)
if scale < 1:
print(
it(
"yellow",
"ALERT: scaling buy edicts for fees: %.4f" % scale,
)
)
for idx, _ in enumerate(buy_edicts):
buy_edicts[idx]["amount"] *= scale
# when BTS is the asset don't sell the last 2
if asset_id == "1.3.0" and sell_edicts:
bts_total = 0
# calculate total of each bts amount in the order
for idx, _ in enumerate(sell_edicts):
bts_total += sell_edicts[idx]["amount"]
scale = (
self.constants.core.DECIMAL_SIXSIG
* decimal(max(0, (core_balance - 2)))
/ (bts_total + self.constants.core.DECIMAL_SATOSHI)
)
# scale the order amounts to save last two bitshares
if scale < 1:
print(
it(
"yellow",
"ALERT: scaling sell edicts for fees: %.4f" % scale,
)
)
for idx, _ in enumerate(sell_edicts):
sell_edicts[idx]["amount"] *= scale
# after scaling recombine buy and sell
create_edicts = buy_edicts + sell_edicts
# REMOVE DUST EDICTS
if self.constants.signing.DUST and create_edicts:
create_edicts2 = []
dust = self.constants.signing.DUST * 100000 / 10 ** asset_precision
for idx, _ in enumerate(create_edicts):
if create_edicts[idx]["amount"] > dust:
create_edicts2.append(create_edicts[idx])
else:
print(
it("red", "WARN: removing dust threshold %s order" % dust),
create_edicts[idx],
)
create_edicts = create_edicts2[:] # copy as new list
del create_edicts2
# TRANSLATE LIMIT ORDERS TO GRAPHENE
for idx, _ in enumerate(create_edicts):
price = create_edicts[idx]["price"]
amount = create_edicts[idx]["amount"]
op_exp = int(create_edicts[idx]["expiration"])
# convert zero expiration flag to "really far in future"
if op_exp == 0:
op_exp = self.constants.core.END_OF_TIME
op_expiration = to_iso_date(op_exp)
# we'll use ordered dicts and put items in api specific order
min_to_receive = OrderedDict({})
amount_to_sell = OrderedDict({})
# derive min_to_receive & amount_to_sell from price & amount
# means SELLING currency RECEIVING assets
if create_edicts[idx]["op"] == "buy":
min_to_receive["amount"] = int(amount * 10 ** asset_precision)
min_to_receive["asset_id"] = asset_id
amount_to_sell["amount"] = int(
amount * price * 10 ** currency_precision
)
amount_to_sell["asset_id"] = currency_id
# means SELLING assets RECEIVING currency
if create_edicts[idx]["op"] == "sell":
min_to_receive["amount"] = int(
amount * price * 10 ** currency_precision
)
min_to_receive["asset_id"] = currency_id
amount_to_sell["amount"] = int(amount * 10 ** asset_precision)
amount_to_sell["asset_id"] = asset_id
# Limit_order_create fee ordered dictionary
fee = OrderedDict([("amount", fees["create"]), ("asset_id", "1.3.0")])
# create ordered dicitonary from each buy/sell operation
operation = [
1,
OrderedDict(
[
("fee", fee), # OrderedDict
("seller", account_id), # "a.b.c"
("amount_to_sell", amount_to_sell), # OrderedDict
("min_to_receive", min_to_receive), # OrderedDict
("expiration", op_expiration), # self.constants.core.ISO8601
("fill_or_kill", self.constants.signing.KILL_OR_FILL), # bool
(
"extensions",
[],
), # always empty list for our purpose
]
),
]
tx_operations.append(operation)
# prevent inadvertent huge number of orders
tx_operations = tx_operations[: self.constants.signing.LIMIT]
# the trx is just a regular dictionary we will convert to json later
# the operations themselves must still be an OrderedDict
trx = {
"ref_block_num": ref_block_num,
"ref_block_prefix": ref_block_prefix,
"expiration": tx_expiration,
"operations": tx_operations,
"signatures": [],
"extensions": [],
}
return trx