This repository was archived by the owner on Jun 30, 2021. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 5
/
Copy pathmanage.py
147 lines (122 loc) · 4.98 KB
/
manage.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
from flask import current_app
from flask.ext.script import Manager
from flask.ext.migrate import MigrateCommand
from decimal import Decimal
from collections import deque
from lincoln import create_app, db, coinserv
from lincoln.models import Block, Transaction, Output
import bitcoin.core.script as op
import bitcoin.core.serialize as serialize
import time
import datetime
manager = Manager(create_app)
manager.add_command('db', MigrateCommand)
@manager.command
def init_db():
db.session.commit()
db.drop_all()
db.create_all()
@manager.command
def sync():
# Get the most recent block in our database
highest = Block.query.order_by(Block.height.desc()).first()
if highest:
highest_hash = coinserv.getblockhash(highest.height)
# This means the coinserver and local index are on different chains
#if highest_hash != highest.hash:
server_height = coinserv.getinfo()['blocks']
server_hash = coinserv.getblockhash(server_height)
block_times = deque([], maxlen=1000)
while True:
t = time.time()
if not highest:
curr_height = 0
else:
curr_height = highest.height + 1
if curr_height > server_height:
break
else:
curr_hash = coinserv.getblockhash(curr_height)
block = coinserv.getblock(curr_hash)
block_obj = Block(hash=block.GetHash(),
height=curr_height,
ntime=datetime.datetime.utcfromtimestamp(block.nTime),
orphan=False,
total_in=0,
total_out=0,
difficulty=block.difficulty,
algo=current_app.config['algo']['display'],
currency=current_app.config['currency']['code'])
current_app.logger.info(
"Syncing block {}".format(block_obj))
db.session.add(block_obj)
# all TX's in block are connectable; index
for tx in block.vtx:
tx_obj = Transaction(block=block_obj,
txid=tx.GetHash(),
total_in=0,
total_out=0)
db.session.add(tx_obj)
current_app.logger.info("Found new tx {}".format(tx_obj))
for i, txout in enumerate(tx.vout):
out_dec = Decimal(txout.nValue) / 100000000
tx_obj.total_out += out_dec
# Sloppy as hell destination address checking
# ------------------------------------------------
scr = []
try:
scr = list(txout.scriptPubKey)
except op.CScriptTruncatedPushDataError:
pass
out = Output(origin_tx=tx_obj,
index=i,
amount=out_dec)
db.session.add(out)
# pay-to-pubkey-hash
if (len(scr) == 5 and
scr[0] == op.OP_DUP and
scr[1] == op.OP_HASH160 and
scr[3] == op.OP_EQUALVERIFY and
scr[4] == op.OP_CHECKSIG):
out.type = 1
out.dest_address = scr[2]
elif (len(scr) == 3 and
scr[0] == op.OP_HASH160 and
scr[2] == op.OP_EQUAL):
out.type = 0
out.dest_address = scr[1]
elif len(scr) == 2 and scr[1] == op.OP_CHECKSIG:
out.type = 2
out.dest_address = serialize.Hash160(scr[0])
else:
out.type = 3
current_app.logger.info("Unrecognized script {}"
.format(scr))
db.session.flush()
if not tx.is_coinbase():
for txin in tx.vin:
obj = Output.query.filter_by(
origin_tx_hash=txin.prevout.hash,
index=txin.prevout.n).one()
obj.spent_tx = tx_obj
tx_obj.total_in += obj.amount
else:
tx_obj.coinbase = True
block_obj.total_in += tx_obj.total_in
block_obj.total_out += tx_obj.total_out
highest = block_obj
db.session.commit()
block_times.append(time.time() - t)
# Display progress information
time_per = sum(block_times) / len(block_times)
time_remain = datetime.timedelta(
seconds=time_per * (server_height - curr_height))
current_app.logger.info(
"{:,}/{:,} {} estimated to catchup"
.format(curr_height, server_height, time_remain))
db.session.commit()
manager.add_option('-c', '--config', default='/config.yml')
manager.add_option('-l', '--log-level',
choices=['DEBUG', 'INFO', 'WARN', 'ERROR'], default='INFO')
if __name__ == "__main__":
manager.run()