forked from iceland2k14/kangaroo
-
Notifications
You must be signed in to change notification settings - Fork 5
/
kangaroo.py
141 lines (120 loc) · 6.06 KB
/
kangaroo.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
# -*- coding: utf-8 -*-
"""
Usage :
> python kangaroo.py -p 0352b1af31d67e6a83ec7931c148f56b0755ce40c836f20c6fe2b6da612c89cf3e -keyspace 935da71d7350734c3472fe305fef82ab8aca644fb:935da71d7350734c3472fe305fff82ab8aca644f9
> python kangaroo.py -p 02CEB6CBBCDBDF5EF7150682150F4CE2C6F4807B349827DCDBDD1F2EFA885A2630 -keyspace 800000000000000000000000000000:FFFFFFFFFFFFFFFFFFFFFFFFFFFFFF -rand -ncore 2
> python kangaroo.py -p 0425c2d005f3036c13070afcf139a18ce69355c3158e017cd99ae72d815d74c54fb3c7c0bc9f4089284cc2de737024d50328884282a8b9bbbaf989747198971669 -rand1
"""
import bit
import ctypes
import platform
import sys
import os
import random
import argparse
import signal
import logging
###############################################################################
parser = argparse.ArgumentParser(description='This tool use Kangaroo algo for searching 1 pubkey in the given range using multiple cpu',
epilog='Enjoy the program! :) ')
parser.version = '15112022'
parser.add_argument("-p", "--pubkey", help = "Public Key in hex format (compressed or uncompressed)", required=True)
parser.add_argument("-keyspace", help = "Keyspace Range ( hex ) to search from min:max. default=1:order of curve", action='store')
parser.add_argument("-ncore", help = "Number of CPU to use. default = Total-1", action='store')
parser.add_argument("-n", help = "Total range search in 1 loop. default=72057594037927935", action='store')
parser.add_argument("-rand", help = "Start from a random value in the given range from min:max and search 0XFFFFFFFFFFFFFF values then again take a new random", action="store_true")
parser.add_argument("-rand1", help = "First Start from a random value, then go fully sequential, in the given range from min:max", action="store_true")
if len(sys.argv) == 1:
parser.print_help()
sys.exit(1)
args = parser.parse_args()
###############################################################################
ss = args.keyspace if args.keyspace else '1:FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEBAAEDCE6AF48A03BBFD25E8CD0364140'
flag_random = True if args.rand else False
flag_random1 = True if args.rand1 else False
ncore = int(args.ncore) if args.ncore else os.cpu_count() - 1
increment = int(args.n) if args.n else 72057594037927935
public_key = args.pubkey # '02CEB6CBBCDBDF5EF7150682150F4CE2C6F4807B349827DCDBDD1F2EFA885A2630'
if flag_random1: flag_random = True
a, b = ss.split(':')
a = int(a, 16)
b = int(b, 16)
lastitem = 0
###############################################################################
if platform.system().lower().startswith('win'):
pathdll = os.path.realpath('Kangaroo_CPU.dll')
ice = ctypes.CDLL(pathdll)
elif platform.system().lower().startswith('lin'):
pathdll = os.path.realpath('Kangaroo_CPU.so')
ice = ctypes.CDLL(pathdll)
else:
logging.error('[-] Unsupported Platform currently for ctypes dll method. Only [Windows and Linux] is working')
sys.exit()
ice.run_cpu_kangaroo.argtypes = [ctypes.c_char_p, ctypes.c_char_p, ctypes.c_int, ctypes.c_int, ctypes.c_int, ctypes.c_char_p, ctypes.c_char_p] # st,en,dp,ncpu,mx,pvk,upub
ice.init_kangaroo_lib()
###############################################################################
def run_cpu_kangaroo(start_range_int, end_range_int, dp, ncpu, mx, upub_bytes):
st_hex = hex(start_range_int)[2:].encode('utf8')
en_hex = hex(end_range_int)[2:].encode('utf8')
res = (b'\x00') * 32
ice.run_cpu_kangaroo(st_hex, en_hex, dp, ncpu, mx, res, upub_bytes)
return res
def pub2upub(pub_hex):
x = int(pub_hex[2:66],16)
if len(pub_hex) < 70:
y = bit.format.x_to_y(x, int(pub_hex[:2],16)%2)
else:
y = int(pub_hex[66:],16)
return bytes.fromhex('04'+ hex(x)[2:].zfill(64) + hex(y)[2:].zfill(64))
def randk(a, b):
global lastitem
if flag_random:
random.seed(random.randint(1,2**256))
return random.SystemRandom().randint(a, b)
else:
if lastitem == 0:
return a
elif lastitem > b:
logging.info('[+] Range Finished')
exit()
else:
return lastitem + 1
def handler(signal_received, frame):
logging.info('\nSIGINT or CTRL-C detected. Exiting gracefully. BYE')
exit(0)
###############################################################################
logging.info('[+] Starting CPU Kangaroo.... Please Wait Version [%s]', parser.version)
dp = 10 # -1 for automatic value
mx = 2 # 0 for Endless
upub = pub2upub(public_key)
###############################################################################
if flag_random1:
logging.info('[+] Search Mode: Random Start then Continuous Range Search from it')
elif flag_random:
logging.info('[+] Search Mode: Random Start after every Range 0XFFFFFFFFFFFFFF key search')
else:
logging.info('[+] Search Mode: Range search Continuous in the given range')
###############################################################################
range_st = randk(a, b) # start from
range_en = range_st + increment
# Reset the flag after getting 1st Random Start Key
if flag_random1: flag_random = False
signal.signal(signal.SIGINT, handler) # Register the signal handler
logging.info('[+] ............................................')
while True:
logging.info('\r[+] Scanning Range %s : %s', hex(range_st), hex(range_en))
pvk_found = run_cpu_kangaroo(range_st, range_en, dp, ncore, mx, upub)
with open('scannedkeys.txt', 'a') as ab:
ab.write(pvk_found.hex() + '\n')
if int(pvk_found.hex(),16) != 0:
logging.info('\n============== KEYFOUND ==============')
logging.info('Kangaroo FOUND PrivateKey : 0x%s', pvk_found.hex())
logging.info('======================================')
with open('KEYFOUNDKEYFOUND.txt','a') as fw:
fw.write('Kangaroo FOUND PrivateKey : 0x'+pvk_found.hex()+'\n')
break
lastitem = range_en
range_st = randk(a, b)
range_en = range_st + increment
logging.info('')
logging.info('[+] Program Finished')