-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathcalcPayments.py
202 lines (152 loc) · 5.98 KB
/
calcPayments.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
import requests
#import json
import os
import hyperjson
myLeases = {}
myCanceledLeases = {}
myForgedBlocks = []
payments = {}
totalfee = 0
with open('config.json') as json_file:
config = hyperjson.load(json_file)
def getAllBlocks():
startblock = 1
steps = 100
blocks = []
#determine endBlock
if config['endBlock'] == 0:
endBlock = requests.get(config['node'] + '/blocks/height').json()['height'] - 1
else:
endBlock = config['endBlock']
#try to load previous processed blocks
try:
with open(config['blockStorage'], 'r') as f:
#blocks = json.load(f)
blocks = hyperjson.load(f)
startblock = blocks[len(blocks) - 1]['height'] + 1
print('retrieved blocks from ' + str(blocks[0]['height']) + ' to ' + str(startblock - 1))
except Exception as e:
print('no previous blocks file found')
#retrieve blocks
while (startblock < endBlock):
if (startblock + (steps - 1) < endBlock):
print('getting blocks from ' + str(startblock) + ' to ' + str(startblock + (steps - 1)))
blocksJSON = requests.get(config['node'] + '/blocks/seq/' + str(startblock) + '/' + str(startblock + (steps - 1))).json()
else:
print('getting blocks from ' + str(startblock) + ' to ' + str(endBlock))
blocksJSON = requests.get(config['node'] + '/blocks/seq/' + str(startblock) + '/' + str(endBlock)).json()
blocks += blocksJSON
if (startblock + steps < endBlock):
startblock += steps
else:
startblock = endBlock
return blocks
def prepareDataStructure(blocks):
global myLeases
global myCanceledLeases
global myForgedBlocks
prevBlock = []
for block in blocks:
fee = 0
if block['generator'] == config['address']:
myForgedBlocks.append(block)
for tx in block['transactions']:
if (tx['type'] == 8) and (tx['recipient'] == config['address'] or tx['recipient'] == 'address:' + config['address']):
tx['block'] = block['height']
myLeases[tx['id']] = tx
elif (tx['type'] == 9) and (tx['leaseId'] in myLeases):
tx['block'] = block['height']
myCanceledLeases[tx['leaseId']] = tx
if len(prevBlock) > 0:
block['previousBlockFees'] = prevBlock['fees']
block['fees'] = block['fee']
prevBlock = block
return blocks
def getActiveLeasesAtBlock(block):
global myLeases
global myCanceledLeases
activeleases = []
totalLeased = 0
activeLeasesPerAddress = {}
for key in myLeases:
currentLease = myLeases[key]
if not(key in myCanceledLeases) or myCanceledLeases[key]['block'] > block['height']:
activeleases.append(currentLease)
for lease in activeleases:
if block['height'] > lease['block'] + 1000:
if lease['sender'] in activeLeasesPerAddress:
activeLeasesPerAddress[lease['sender']] += lease['amount']
else:
activeLeasesPerAddress[lease['sender']] = lease['amount']
totalLeased += lease['amount']
return {'totalLeased': totalLeased, 'activeLeases': activeLeasesPerAddress}
def distribute(activeLeases, amountTotalLeased, block):
global payments
global totalfee
fee = block['fees'] * 0.4 + block['previousBlockFees'] * 0.6
totalfee += fee
for address in activeLeases:
share = activeLeases[address] / amountTotalLeased
amount = fee * share
if address in payments:
payments[address] += amount * (config['percentageOfFeesToDistribute'] / 100)
else:
payments[address] = amount * (config['percentageOfFeesToDistribute'] / 100)
def checkTotalDistributableAmount(payments):
total = 0
for address in payments:
amount = payments[address]
total += amount
return total
def createPayment():
global payments
tx = []
for address in payments:
if round(payments[address]) > config['minAmounttoPay']:
paytx = {'recipient': address, 'amount': round(payments[address])}
tx.append(paytx)
with open(config['paymentStorage'], 'w') as outfile:
hyperjson.dump(tx, outfile)
print('payments written to ' + config['paymentStorage'])
def main():
global payments
global myForgedBlocks
global myLeases
global myCanceledLeases
print('getting blocks...')
blocks = getAllBlocks()
print('preparing datastructures...')
blocks = prepareDataStructure(blocks)
#clear payed tx
for block in blocks:
txs = []
if block['height'] < config['startBlock']:
for tx in block['transactions']:
if tx['type'] == 8 or tx['type'] == 9:
txs.append(tx)
else:
txs = block['transactions']
block['transactions'] = txs
#save current blocks
print('saving blockfile...')
with open(config['blockStorage'], 'w') as outfile:
hyperjson.dump(blocks, outfile)
print('preparing payments...')
if config['endBlock'] == 0:
endBlock = requests.get(config['node'] + '/blocks/height').json()['height'] - 1
else:
endBlock = config['endBlock']
for block in myForgedBlocks:
if block['height'] >= config['startBlock'] and block['height'] <= endBlock:
blockLeaseData = getActiveLeasesAtBlock(block)
activeLeasesForBlock = blockLeaseData['activeLeases']
amountTotalLeased = blockLeaseData['totalLeased']
distribute(activeLeasesForBlock, amountTotalLeased, block)
total = checkTotalDistributableAmount(payments)
createPayment()
print('forged blocks: ' + str(len(myForgedBlocks)))
print('number of payments: ' + str(len(payments)))
print('total payment: ' + str(total / pow(10, 8)))
print('number of leases: ' + str(len(myLeases)))
print('number of cancelled leases: ' + str(len(myCanceledLeases)))
main()