forked from shanhaicoder/XENGPUMiner
-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathrpc2.py
633 lines (498 loc) · 21.6 KB
/
rpc2.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
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
from flask import Flask, request, jsonify
import secrets
import sqlite3
from ethereum.transactions import Transaction
from ethereum.utils import decode_hex
import rlp
from rlp import encode
from web3 import Web3
import time
from flask_cors import cross_origin
app = Flask(__name__)
from flask_cors import CORS
CORS(app)
@app.route('/')
@cross_origin()
def hello_world():
return 'Hello, World!'
# Initialize eth_blockNumber
current_block_number = 0x2234
# Function to fetch balance from SQLite database
def validate_transaction(tx):
# Validate the transaction (this is a placeholder; your validation logic goes here)
return True
def broadcast_transaction(tx):
# Broadcast the transaction to the network (this is a placeholder; your broadcast logic goes here)
# Returning a fake hash for demonstration
return "0x" + tx.hash.hex()
def get_xuni_account_count(account_name):
# Initialize database connection
conn = sqlite3.connect('blocks.db')
cursor = conn.cursor()
try:
# Run the SQL query
cursor.execute("SELECT COUNT(*) as n FROM xuni WHERE account = ?", (account_name,))
data = cursor.fetchone()
# Close database connection
conn.close()
# Check if data was found
if data:
return data[0]
else:
return -1 # can use -1 or another indicator to show that account was not found
except sqlite3.Error as e:
print("Database error:", e)
# Close database connection in case of an error
conn.close()
return -2 # can use -2 or another indicator to show that there was a database error
# Function to update the account balances in the super_blocks table
def transfer(from_account, to_account, value):
try:
conn_cache = sqlite3.connect("cache.db")
cursor_cache = conn_cache.cursor()
conn = sqlite3.connect("blocks.db")
cursor = conn.cursor()
# Check if the value is negative
if value < 0:
print("Transfer value cannot be negative.")
return
# Debug print
print(f"Initiating transfer of {value} from {from_account} to {to_account}")
# Add '0x' prefix if not present and convert accounts to lower case
#from_account = from_account.lower() if from_account.startswith('0x') else '0x' + from_account.lower()
#to_account = to_account.lower() if to_account.startswith('0x') else '0x' + to_account.lower()
# Check balance of the sender account
#cursor.execute("SELECT super_block_count FROM super_blocks WHERE LOWER(account) = ?", (from_account,))
#row = cursor.fetchone()
#if row:
# current_balance = row[0]
# if current_balance < value:
# print(f"Insufficient balance in account {from_account}. Transfer aborted.")
# return
#else:
# print(f"Account {from_account} not found. Transfer aborted.")
# return
# Deduct the value from the sender account
#cursor.execute("""
# UPDATE super_blocks
# SET super_block_count = super_block_count - ?
# WHERE LOWER(account) = ?;
#""", (value, from_account))
#affected_rows = cursor.rowcount
#print(f"Deducted {value} from {from_account}. Rows affected: {affected_rows}")
# Add the value to the receiver account
#cursor.execute("""
# UPDATE super_blocks
# SET super_block_count = super_block_count + ?
# WHERE LOWER(account) = ?;
#""", (value, to_account))
#if cursor.rowcount == 0:
#print(f"Account {to_account} not found. Inserting a new row.")
#cursor.execute("INSERT INTO super_blocks (account, super_block_count) VALUES (?, ?)", (to_account, value))
# Commit and close
#conn.commit()
conn.close()
except sqlite3.Error as e:
print("SQLite error occurred:", e)
if conn:
conn.rollback()
print("Transaction rolled back.")
raise
def get_xblk_account_count(account):
try:
conn = sqlite3.connect("cache.db")
cursor_cache = conn.cursor()
cursor_cache.execute("SELECT super_blocks FROM cache_table WHERE LOWER(account) = LOWER(?)", (account,))
print ("Account: ", account)
row = cursor_cache.fetchone()
return row[0] if row else 0
except Exception as e:
print("Database error:", e)
return 0
finally:
if conn:
conn.close()
def get_balance_from_db(account):
try:
conn = sqlite3.connect("blocks.db")
cursor = conn.cursor()
conn_cache = sqlite3.connect("cache.db")
cursor_cache = conn_cache.cursor()
query = "SELECT super_block_count FROM super_blocks WHERE LOWER(account) = LOWER(?)"
#print(f"Executing SQL query: {query} with account: {account}") # Print the query and account to standard output
#cursor.execute("SELECT super_block_count FROM super_blocks WHERE LOWER(account) = LOWER(?)", (account,))
print ("Account: ", account)
cursor_cache.execute("SELECT total_blocks FROM cache_table WHERE LOWER(account) = ?", (account.lower(),))
row = cursor_cache.fetchone()
print ("Balance for ", account, row[0] * 10 if row else 0) # Modified this line to handle None
return row[0] * 10 if row else 0
except Exception as e:
print("Database error:", e)
return 0
finally:
if conn:
conn.close()
conn_cache.close()
def rlp_encode(input_string):
if len(input_string) == 1 and ord(input_string) < 0x80:
return input_string
elif len(input_string) <= 55:
return chr(0x80 + len(input_string)) + input_string
else:
length_of_length = len(str(len(input_string)))
return chr(0xb7 + length_of_length) + str(len(input_string)) + input_string
def handle_eth_call(data):
print("In handle_eth_call function: ", data)
# Check if 'data' key exists
if 'params' not in data or not isinstance(data['params'], list) or len(data['params']) == 0 or 'data' not in data['params'][0]:
print ("Data missing: ", data)
response = "0x123456"
return response
# Extracting the relevant details from the data
target_address = data['params'][0]['to'].lower()
# Default names and symbols for different contracts
contract_data = {
"0x999999cf1046e68e36e1aa2e0e07105eddd00002": {"name": "XUNI Token", "symbol": "XUNI"},
"0x999999cf1046e68e36e1aa2e0e07105eddd00001": {"name": "X.BLK Token", "symbol": "X.BLK"}
}
# If the contract address is not recognized
if target_address not in contract_data:
return {
"id": data['id'],
"jsonrpc": "2.0",
"error": {
"code": -32000,
"message": "Unsupported contract address"
}
}
function_data = data['params'][0]['data']
function_signature = function_data[:10]
address_queried = function_data[10:74]
print(f"Function Signature: {function_signature}")
if function_signature == '0x313ce567': # decimals function
print("RETURN DECIMALS")
decimals = 18
response = '0x' + hex(decimals)[2:].zfill(64)
elif function_signature == '0x06fdde03': # name function
token_name = contract_data[target_address]["name"]
length_in_hex = hex(len(token_name))[2:].zfill(64)
encoded_name = token_name.encode().hex().ljust(64, '0')
response = '0x' + '0000000000000000000000000000000000000000000000000000000000000020' + length_in_hex + encoded_name
print("RETURN NAME: ", token_name)
elif function_signature == '0x95d89b41': # symbol function
symbol = contract_data[target_address]["symbol"]
length_in_hex = hex(len(symbol))[2:].zfill(64)
encoded_symbol = symbol.encode().hex().ljust(64, '0')
response = '0x' + '0000000000000000000000000000000000000000000000000000000000000020' + length_in_hex + encoded_symbol
print("RETURN SYMBOL: ", symbol)
elif function_signature == '0x70a08231': # balanceOf function
address_queried = function_data[34:74].lower()
# Depending on the target address, call the appropriate function to get the balance
if target_address == "0x999999cf1046e68e36e1aa2e0e07105eddd00002":
balance = get_xuni_account_count('0x' + address_queried) * 1000000000000000000
elif target_address == "0x999999cf1046e68e36e1aa2e0e07105eddd00001":
balance = get_xblk_account_count('0x' + address_queried) * 1000000000000000000
else:
# Handle unknown contract address or give a default balance
balance = 0
response = '0x' + hex(balance)[2:].zfill(64)
print("RETURN BALANCE for: ", address_queried)
print("BALANCE is: ", balance)
else:
response = {
"id": data['id'],
"jsonrpc": "2.0",
"error": {
"code": -32000,
"message": "Invalid request"
}
}
print(response)
return response
def handle_eth_call2(data):
print("In handle_eth_call functioni: ", data)
# Initialize the response to None
response = None
# Check if 'data' key exists
if 'params' not in data or not isinstance(data['params'], list) or len(data['params']) == 0 or 'data' not in data['params'][0]:
print ("Data missing: ", data)
return {
"id": data['id'],
"jsonrpc": "2.0",
"result": "0x123456"
}
# Extracting the relevant details from the data
target_address = data['params'][0]['to'].lower()
# Checking if the target address matches
if target_address != "0x999999cf1046e68e36e1aa2e0e07105eddd00002":
return {
"id": data['id'],
"jsonrpc": "2.0",
"error": {
"code": -32000,
"message": "Unsupported contract address"
}
}
function_data = data['params'][0]['data']
function_signature = function_data[:10]
address = function_data[10:74]
print(f"Function Signature: {function_signature}")
if function_signature == '0x313ce567':
# decimals function
decimals = 18 # XUNI has 18 decimals
response = hex(decimals)
elif function_signature == '0x95d89b41':
# symbol function
symbol = "XUNI" # Ticker symbol for the token
response = '0x' + symbol.encode().hex()
elif function_signature in ['0x01ffc9a7', '0x70a08231']:
# balanceOf function
balance = 1000 # Fetch the balance for the provided address from your database or wherever you store it
response = '0x' + hex(balance)[2:].zfill(64) # Convert to hex and pad to 64 characters
@app.route('/', methods=['POST'])
@cross_origin()
def index():
data = request.get_json()
global current_block_number
print("Received data:", data) # Print received queries to the screen
if not data:
abort(400, description="No data provided")
if data['jsonrpc'] != '2.0':
response = {'jsonrpc': '2.0', 'error': {'code': -32600, 'message': 'Invalid Request'}, 'id': None}
print("Sending response:", response) # Print response to the screen
return jsonify(response), 400
# Initialize the result variable
result = None
if data['method'] == 'eth_blockNumber':
current_block_number += 1 # Increment block number
result = hex(current_block_number)
response = {
'jsonrpc': '2.0',
'id': data.get('id', None),
'result': result
}
print("Sending response:", response) # Print response to the screen
return jsonify(response)
elif data['method'] == 'eth_getBalance':
account = data['params'][0] # Convert account to lower case to ensure it matches
balance_decimal = get_balance_from_db(account) # Fetch balance from the database
result = hex(int(balance_decimal * 10**18)) # Convert to Wei
elif data['method'] == 'eth_estimateGas':
# Simulate gas estimation here. This is a simplified example and
# you would usually perform an actual gas estimation based on the
# transaction details.
result = '0x5208' # 21000 in hex, a common gas cost for simple transfers
#elif data['method'] == 'eth_call':
# result = '0x123456'
elif data['method'] == 'eth_call':
print ("ETH CALL RECEIVED: ")
print (data)
result = handle_eth_call(data)
elif data['method'] == 'eth_chainId':
result = '0x18705' # Mainnet
elif data['method'] == 'eth_getCode':
address = data['params'][0].lower() # Convert the address to lower case
block_number = data['params'][1] # The block number or tag; you may or may not use this depending on your implementation
# Example: Fetch the contract code from your database or some storage.
# In this example, it's hardcoded.
contract_code = {
'0xdadf7ac7d0622dedd7e58b4d85d3784cc0c9d0e7': '0x606060405260...',
'0x999999cf1046e68e36e1aa2e0e07105eddd00002': '0x00002',
'0x999999cf1046e68e36e1aa2e0e07105eddd00001': '0x00001'
}
result = contract_code.get(address, '0x') # return '0x' if the address is not a contract
elif data['method'] == 'eth_getTransactionReceipt':
tx_hash = data['params'][0]
print("Entering eth_getTransactionReceipt")
# Connect to SQLite database
conn = sqlite3.connect('blocks.db')
cursor = conn.cursor()
# Query the transactions table
cursor.execute("SELECT from_account, to_account, value FROM transactions WHERE tx_hash=?", (tx_hash,))
row = cursor.fetchone()
if row:
from_account, to_account, value = row
# Populate receipt with database values
receipt = {
'transactionHash': tx_hash,
'transactionIndex': '0x1',
'blockHash': '0x' + secrets.token_hex(32),
'blockNumber': '0x6',
'from': from_account,
'to': to_account,
'cumulativeGasUsed': '0xA',
'gasUsed': '0xA',
'contractAddress': None,
'logs': [],
'status': '0x1'
}
print("Sending: ", receipt)
result = receipt
else:
print(f"No transaction found for hash {tx_hash}")
conn.close()
elif data['method'] == 'eth_getTransactionCount':
address = data['params'][0].lower()
block_number = data['params'][1] # May or may not use this depending on your needs
# Fetch the nonce from your database using the get_nonce function
nonce = get_nonce(address)
print ("Returning nonce: ", nonce)
result = hex(nonce)
print ("Returning nonce hex: ", result)
elif data['method'] == '1eth_getTransactionCount':
address = data['params'][0].lower()
block_number = data['params'][1] # May or may not use this depending on your needs
# Fetch the nonce from your database or however you're storing nonces
# For demonstration, we'll just use a hardcoded example
nonces = {
"0xc855fd5aa2829799dde83b43ac33651e46f610bb": 10
}
result = hex(nonces.get(address, 0))
print ("Returning nonce hex: ", result)
elif data['method'] == 'Xeth_getBlockByNumber':
result = {
'number': '0x5BAD55',
'hash': '0x1234567890abcdef',
'transactions': []
}
elif data['method'] == 'eth_getBlockByNumber':
print("RETURN BLOCK DATA")
requested_block_number = data['params'][0]
current_timestamp = int(time.time()) # Assuming you've imported the time module
result = {
'number': requested_block_number,
'hash': '0x1234567890abcdef', # Mock hash, adjust as needed
'timestamp': hex(current_timestamp),
'transactions': [],
'parentHash': '0xabcdef1234567890', # Mock parent hash
'miner': '0xdeadbeefdeadbeef', # Mock miner address
# Add other block attributes if needed
}
elif data['method'] == 'net_version':
result = '1' # Mainnet
elif data['method'] == 'eth_getBlockByHash':
block_hash = data['params'][0]
full_tx = data['params'][1]
# Fetch block data by block hash.
# For demonstration purposes, let's use a mock block.
mock_block = {
'number': '0x1b4',
'hash': block_hash,
'transactions': ['0x123...', '0x124...'] if not full_tx else [
{
'hash': '0x123...',
'from': '0xabc...',
'to': '0xdef...',
'value': '0x1'
},
{
'hash': '0x124...',
'from': '0xghi...',
'to': '0xjkl...',
'value': '0x2'
}
]
# ... (other block fields)
}
result = mock_block
elif data['method'] == 'eth_gasPrice':
result = '0x3B9ACA00' # Example gas price, 1 Gwei in hexadecimal
elif data['method'] == 'eth_sendRawTransaction':
print (data)
raw_tx = data['params'][0]
print("Raw TX: ", raw_tx)
try:
handle_raw_transaction(raw_tx)
#tx_hash = broadcast_transaction(raw_tx)
tx_hash = get_transaction_hash(raw_tx)
print ("Returning results from eth_sendRawTransaction ", tx_hash)
result = tx_hash
except Exception as e:
result = {'jsonrpc': '2.0', 'error': {'code': -32000, 'message': f'Exception: {e}'}, 'id': data['id']}
print("Sending response:", result) # Print response to the screen
return result
else:
response = {'jsonrpc': '2.0', 'error': {'code': -32601, 'message': 'Method not found'}, 'id': data.get('id', None)}
print("Sending response:", response) # Print response to the screen
return jsonify(response), 400
response = {
'jsonrpc': '2.0',
'id': data.get('id', None),
'result': result
}
print("Sending Final response:", response) # Print response to the screen
return jsonify(response)
def broadcast_transaction(raw_tx):
# Generate a 64-character long hex string as a fake Ethereum transaction hash
fake_tx_hash = '0x' + secrets.token_hex(32)
return fake_tx_hash
from Crypto.Hash import keccak
import rlp
from coincurve import PublicKey
from web3 import Web3
def get_recovered_address(raw_tx):
w3 = Web3()
try:
# Recover the sender's address from the raw transaction
recovered_address = w3.eth.account.recover_transaction(raw_tx)
return recovered_address
except Exception as e:
print(f"An error occurred: {e}")
return None
def get_nonce(from_account):
conn = sqlite3.connect('blocks.db')
cursor = conn.cursor()
# Count the number of transactions for the from_account
cursor.execute("SELECT COUNT(*) FROM transactions WHERE from_account = ?", (from_account,))
count = cursor.fetchone()[0]
# If there's at least one transaction, increment the count to get the next nonce.
# If there are no transactions, the count will be 0 and that's the nonce you'll use.
return count + 1 if count else 0
def get_transaction_hash(raw_tx):
return Web3.keccak(hexstr=raw_tx).hex()
def handle_raw_transaction(raw_tx):
# Decode the raw transaction
conn = sqlite3.connect('blocks.db')
c = conn.cursor()
try:
# Decoding the hex-encoded transaction
decoded_tx = rlp.decode(bytes.fromhex(raw_tx[2:]), Transaction)
#tx_hash = "0x" + decoded_tx.hash().hex()
#tx_hash = broadcast_transaction(raw_tx)
tx_hash = get_transaction_hash(raw_tx)
print ("TX hash: ", tx_hash)
# Extract details
from_account = decoded_tx.sender # Or use another way to get sender address
to_account = decoded_tx.to
value = decoded_tx.value
gas = decoded_tx.startgas
gas_price = decoded_tx.gasprice
nonce = decoded_tx.nonce
input_data = decoded_tx.data
# Validate from address
validated_from = get_recovered_address(raw_tx)
from_account = validated_from
# Get the nonce value
nonce = get_nonce(from_account)
# Insert the new record into the transactions table
to_acc = "0x" + to_account.hex()
print ("Inserting: ", tx_hash, raw_tx, from_account, to_acc, value, nonce)
c.execute('''INSERT INTO transactions (tx_hash, raw_tx, from_account, to_account, value, nonce)
VALUES (?, ?, ?, ?, ?, ?)''', (tx_hash, raw_tx, from_account.lower(), to_acc.lower(), value, nonce))
# Commit the transaction
conn.commit()
# Close the connection
conn.close()
transfer (validated_from, to_account.hex(), value/1e18)
# Print the decoded transaction details
print(f"From Account: {validated_from}")
print(f"To Account: {to_account.hex() if to_account else 'Contract Creation'}")
print(f"Value: {value} wei")
print(f"Gas: {gas}")
print(f"Gas Price: {gas_price} wei")
print(f"Nonce: {nonce}")
print(f"Input Data: {input_data.hex()}")
except Exception as e:
print(f"An error occurred: {e}")
if __name__ == '__main__':
app.run(host='0.0.0.0', port=5555, debug=True)