SpyCoder77 commited on
Commit
1d370b0
·
verified ·
1 Parent(s): 9f93bbc

Create app.py

Browse files
Files changed (1) hide show
  1. app.py +262 -0
app.py ADDED
@@ -0,0 +1,262 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ #!/usr/bin/python
2
+ #
3
+ # Copyright 2011 Jeff Garzik
4
+ #
5
+ # This program is free software; you can redistribute it and/or modify
6
+ # it under the terms of the GNU General Public License as published by
7
+ # the Free Software Foundation.
8
+ #
9
+ # This program is distributed in the hope that it will be useful,
10
+ # but WITHOUT ANY WARRANTY; without even the implied warranty of
11
+ # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12
+ # GNU General Public License for more details.
13
+ #
14
+ # You should have received a copy of the GNU General Public License
15
+ # along with this program; see the file COPYING. If not, write to
16
+ # the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA.
17
+ #
18
+
19
+ import time
20
+ import json
21
+ import pprint
22
+ import hashlib
23
+ import struct
24
+ import re
25
+ import base64
26
+ import httplib
27
+ import sys
28
+ from multiprocessing import Process
29
+
30
+ ERR_SLEEP = 15
31
+ MAX_NONCE = 1000000L
32
+
33
+ settings = {}
34
+ pp = pprint.PrettyPrinter(indent=4)
35
+
36
+ class BitcoinRPC:
37
+ OBJID = 1
38
+
39
+ def __init__(self, host, port, username, password):
40
+ authpair = "%s:%s" % (username, password)
41
+ self.authhdr = "Basic %s" % (base64.b64encode(authpair))
42
+ self.conn = httplib.HTTPConnection(host, port, False, 30)
43
+ def rpc(self, method, params=None):
44
+ self.OBJID += 1
45
+ obj = { 'version' : '1.1',
46
+ 'method' : method,
47
+ 'id' : self.OBJID }
48
+ if params is None:
49
+ obj['params'] = []
50
+ else:
51
+ obj['params'] = params
52
+ self.conn.request('POST', '/', json.dumps(obj),
53
+ { 'Authorization' : self.authhdr,
54
+ 'Content-type' : 'application/json' })
55
+
56
+ resp = self.conn.getresponse()
57
+ if resp is None:
58
+ print "JSON-RPC: no response"
59
+ return None
60
+
61
+ body = resp.read()
62
+ resp_obj = json.loads(body)
63
+ if resp_obj is None:
64
+ print "JSON-RPC: cannot JSON-decode body"
65
+ return None
66
+ if 'error' in resp_obj and resp_obj['error'] != None:
67
+ return resp_obj['error']
68
+ if 'result' not in resp_obj:
69
+ print "JSON-RPC: no result in object"
70
+ return None
71
+
72
+ return resp_obj['result']
73
+ def getblockcount(self):
74
+ return self.rpc('getblockcount')
75
+ def getwork(self, data=None):
76
+ return self.rpc('getwork', data)
77
+
78
+ def uint32(x):
79
+ return x & 0xffffffffL
80
+
81
+ def bytereverse(x):
82
+ return uint32(( ((x) << 24) | (((x) << 8) & 0x00ff0000) |
83
+ (((x) >> 8) & 0x0000ff00) | ((x) >> 24) ))
84
+
85
+ def bufreverse(in_buf):
86
+ out_words = []
87
+ for i in range(0, len(in_buf), 4):
88
+ word = struct.unpack('@I', in_buf[i:i+4])[0]
89
+ out_words.append(struct.pack('@I', bytereverse(word)))
90
+ return ''.join(out_words)
91
+
92
+ def wordreverse(in_buf):
93
+ out_words = []
94
+ for i in range(0, len(in_buf), 4):
95
+ out_words.append(in_buf[i:i+4])
96
+ out_words.reverse()
97
+ return ''.join(out_words)
98
+
99
+ class Miner:
100
+ def __init__(self, id):
101
+ self.id = id
102
+ self.max_nonce = MAX_NONCE
103
+
104
+ def work(self, datastr, targetstr):
105
+ # decode work data hex string to binary
106
+ static_data = datastr.decode('hex')
107
+ static_data = bufreverse(static_data)
108
+
109
+ # the first 76b of 80b do not change
110
+ blk_hdr = static_data[:76]
111
+
112
+ # decode 256-bit target value
113
+ targetbin = targetstr.decode('hex')
114
+ targetbin = targetbin[::-1] # byte-swap and dword-swap
115
+ targetbin_str = targetbin.encode('hex')
116
+ target = long(targetbin_str, 16)
117
+
118
+ # pre-hash first 76b of block header
119
+ static_hash = hashlib.sha256()
120
+ static_hash.update(blk_hdr)
121
+
122
+ for nonce in xrange(self.max_nonce):
123
+
124
+ # encode 32-bit nonce value
125
+ nonce_bin = struct.pack("<I", nonce)
126
+
127
+ # hash final 4b, the nonce value
128
+ hash1_o = static_hash.copy()
129
+ hash1_o.update(nonce_bin)
130
+ hash1 = hash1_o.digest()
131
+
132
+ # sha256 hash of sha256 hash
133
+ hash_o = hashlib.sha256()
134
+ hash_o.update(hash1)
135
+ hash = hash_o.digest()
136
+
137
+ # quick test for winning solution: high 32 bits zero?
138
+ if hash[-4:] != '\0\0\0\0':
139
+ continue
140
+
141
+ # convert binary hash to 256-bit Python long
142
+ hash = bufreverse(hash)
143
+ hash = wordreverse(hash)
144
+
145
+ hash_str = hash.encode('hex')
146
+ l = long(hash_str, 16)
147
+
148
+ # proof-of-work test: hash < target
149
+ if l < target:
150
+ print time.asctime(), "PROOF-OF-WORK found: %064x" % (l,)
151
+ return (nonce + 1, nonce_bin)
152
+ else:
153
+ print time.asctime(), "PROOF-OF-WORK false positive %064x" % (l,)
154
+ # return (nonce + 1, nonce_bin)
155
+
156
+ return (nonce + 1, None)
157
+
158
+ def submit_work(self, rpc, original_data, nonce_bin):
159
+ nonce_bin = bufreverse(nonce_bin)
160
+ nonce = nonce_bin.encode('hex')
161
+ solution = original_data[:152] + nonce + original_data[160:256]
162
+ param_arr = [ solution ]
163
+ result = rpc.getwork(param_arr)
164
+ print time.asctime(), "--> Upstream RPC result:", result
165
+
166
+ def iterate(self, rpc):
167
+ work = rpc.getwork()
168
+ if work is None:
169
+ time.sleep(ERR_SLEEP)
170
+ return
171
+ if 'data' not in work or 'target' not in work:
172
+ time.sleep(ERR_SLEEP)
173
+ return
174
+
175
+ time_start = time.time()
176
+
177
+ (hashes_done, nonce_bin) = self.work(work['data'],
178
+ work['target'])
179
+
180
+ time_end = time.time()
181
+ time_diff = time_end - time_start
182
+
183
+ self.max_nonce = long(
184
+ (hashes_done * settings['scantime']) / time_diff)
185
+ if self.max_nonce > 0xfffffffaL:
186
+ self.max_nonce = 0xfffffffaL
187
+
188
+ if settings['hashmeter']:
189
+ print "HashMeter(%d): %d hashes, %.2f Khash/sec" % (
190
+ self.id, hashes_done,
191
+ (hashes_done / 1000.0) / time_diff)
192
+
193
+ if nonce_bin is not None:
194
+ self.submit_work(rpc, work['data'], nonce_bin)
195
+
196
+ def loop(self):
197
+ rpc = BitcoinRPC(settings['host'], settings['port'],
198
+ settings['rpcuser'], settings['rpcpass'])
199
+ if rpc is None:
200
+ return
201
+
202
+ while True:
203
+ self.iterate(rpc)
204
+
205
+ def miner_thread(id):
206
+ miner = Miner(id)
207
+ miner.loop()
208
+
209
+ if __name__ == '__main__':
210
+ if len(sys.argv) != 2:
211
+ print "Usage: pyminer.py CONFIG-FILE"
212
+ sys.exit(1)
213
+
214
+ f = open(sys.argv[1])
215
+ for line in f:
216
+ # skip comment lines
217
+ m = re.search('^\s*#', line)
218
+ if m:
219
+ continue
220
+
221
+ # parse key=value lines
222
+ m = re.search('^(\w+)\s*=\s*(\S.*)$', line)
223
+ if m is None:
224
+ continue
225
+ settings[m.group(1)] = m.group(2)
226
+ f.close()
227
+
228
+ if 'host' not in settings:
229
+ settings['host'] = '127.0.0.1'
230
+ if 'port' not in settings:
231
+ settings['port'] = 8332
232
+ if 'threads' not in settings:
233
+ settings['threads'] = 1
234
+ if 'hashmeter' not in settings:
235
+ settings['hashmeter'] = 0
236
+ if 'scantime' not in settings:
237
+ settings['scantime'] = 30L
238
+ if 'rpcuser' not in settings or 'rpcpass' not in settings:
239
+ print "Missing username and/or password in cfg file"
240
+ sys.exit(1)
241
+
242
+ settings['port'] = int(settings['port'])
243
+ settings['threads'] = int(settings['threads'])
244
+ settings['hashmeter'] = int(settings['hashmeter'])
245
+ settings['scantime'] = long(settings['scantime'])
246
+
247
+ thr_list = []
248
+ for thr_id in range(settings['threads']):
249
+ p = Process(target=miner_thread, args=(thr_id,))
250
+ p.start()
251
+ thr_list.append(p)
252
+ time.sleep(1) # stagger threads
253
+
254
+ print settings['threads'], "mining threads started"
255
+
256
+ print time.asctime(), "Miner Starts - %s:%s" % (settings['host'], settings['port'])
257
+ try:
258
+ for thr_proc in thr_list:
259
+ thr_proc.join()
260
+ except KeyboardInterrupt:
261
+ pass
262
+ print time.asctime(), "Miner Stops - %s:%s" % (settings['host'], settings['port'])