-
Notifications
You must be signed in to change notification settings - Fork 1
/
readyfor.py
executable file
·266 lines (245 loc) · 11 KB
/
readyfor.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
#!/usr/bin/python3
import sys
try:
import argparse, datetime, hashlib, json, netifaces, os, platform, qrcode
import random, re, shlex, ssl, string, subprocess, time
from http.server import BaseHTTPRequestHandler, HTTPServer
from cryptography import x509
from cryptography.x509.oid import NameOID
from cryptography.hazmat.primitives import hashes
from cryptography.hazmat.primitives import serialization
from cryptography.hazmat.primitives.asymmetric import rsa
except ImportError as err:
print(f"Unable to load {err.name} - have you installed library dependencies?")
sys.exit(1)
VER_STRING = "1.6.60"
EXPIRY_PERIOD = 60
RAND_VALID_CHARS = string.ascii_uppercase + string.ascii_lowercase + string.digits
HTTP_PORT = 9833
XFREERDP_INSTRUCTIONS = "https://github.com/FreeRDP/FreeRDP/wiki/PreBuilds"
XFREERDP_RE = re.compile(r"\d+")
XFREERDP_COMMAND = "{} /v:{} /cert:ignore /size:{} /u:{} /p:{}"
DEFAULT_CONFIG_FILE = "settings.json"
BASE_CONFIG = {
"verbose": 0,
"no_check_freerdp": False,
"freerdp_path": "/opt/freerdp-nightly/bin/xfreerdp",
"resolution": "1280x720",
"username": False, # Is set to a string if a custom username/password is used
"password": False
}
parser = argparse.ArgumentParser()
parser.add_argument("-v", "--verbose", help="Show debugging messages", action="count", default=0)
parser.add_argument("--no-check-freerdp", help="Skip freerdp presence & version checks", action="store_true")
parser.add_argument("--freerdp-path", help="Specify an alternative path for freerdp")
parser.add_argument("-r", "--resolution", help="Set custom resolution for RDP connection")
parser.add_argument("-c", "--config", help="Specify a config file", default=DEFAULT_CONFIG_FILE)
args = parser.parse_args()
# This will take the base config and override options provided in overlay
def overlay_config(base, overlay):
if type(overlay) is not dict:
overlay = vars(overlay)
# Copy dict - don't modify dict passed in
updated_config = base.copy()
for k, v in overlay.items():
if ((k == "verbose" and v > updated_config[k]) or (k not in ("config", "verbose", "no_check_freerdp")) or (k == "no_check_freerdp" and v == True)) and v is not None:
if args.verbose >= 2 or updated_config["verbose"] >= 2:
print(f"[ReadyForPy] Changing {k}: {updated_config[k]} -> {v}")
updated_config[k] = v
return updated_config
def load_config(args):
try:
with open(args.config, 'r') as f:
print(f"[ReadyForPy] Using config file \"{args.config}\"")
return json.loads(f.read())
except FileNotFoundError as err:
if args.config != DEFAULT_CONFIG_FILE:
print(f"[ReadyForPy] Unable to find config file \"{args.config}\"")
sys.exit(3)
else:
return BASE_CONFIG
except PermissionError as err:
print(f"[ReadyForPy] Unable to open \"{args.config}\" - are the permissions set correctly?")
sys.exit(3)
except json.decoder.JSONDecodeError as err:
print(f"[ReadyForPy] Syntax error in config file \"{args.config}\":")
print("[ReadyForPy]", err)
sys.exit(3)
config = overlay_config(overlay_config(BASE_CONFIG, load_config(args)), args)
if config["verbose"] >= 1:
print("[ReadyForPy] Configuration:", config)
def check_freerdp():
if platform.system() == 'Linux':
# Check if a xfreerdp is installed
try:
xfreerdp_result = subprocess.run(
shlex.split(f'{config["freerdp_path"]} --version'),
stdout = subprocess.PIPE,
stderr = subprocess.DEVNULL,
universal_newlines=True
)
version = XFREERDP_RE.findall(xfreerdp_result.stdout)
if (int(version[0]) < 3):
print(f"[ReadyForPy] The provided version of xfreerdp is too old ({'.'.join(version[:3])}). Please update to 3.0.0 or newer")
print("[ReadyForPy] (if this was a mistake, run with --no-check-freerdp)")
sys.exit(1)
elif config["verbose"] >= 1:
print(f"[ReadyForPy] Detected freerdp version {'.'.join(version[:3])}")
except FileNotFoundError as err:
print("[ReadyForPy] The nightly release of xfreerdp does not appear to be",
f"installed. Please see {XFREERDP_INSTRUCTIONS} for details",
"on installing")
sys.exit(1)
elif platform.system() == 'Windows':
print("Windows has not yet been tested - rerun with the --no-check-freerdp flag and specify a location for freerdp with --freerdp-path")
sys.exit(2)
else:
print("You are currently not on a tested platform. You will need freerdp",
"(3.0.0 or newer). To test, rerun with --no-check-freerdp and specify",
"the location of freerdp with --freerdp-path")
sys.exit(2)
# Generates a new certificate
def generate_cert():
X509 = dict()
# Generate key
key = rsa.generate_private_key(
public_exponent=65537,
key_size=2048
)
key_data = key.private_bytes(
encoding=serialization.Encoding.PEM,
format=serialization.PrivateFormat.PKCS8,
encryption_algorithm=serialization.NoEncryption()
)
# Must be written to file for Python's SSL library
with open("selfsigned.key", "wb") as f:
f.write(key_data)
X509['key'] = key_data.decode()
if config["verbose"] >= 2:
print(X509['key'])
subject = issuer = x509.Name([
x509.NameAttribute(NameOID.COUNTRY_NAME, u"CN"),
x509.NameAttribute(NameOID.ORGANIZATIONAL_UNIT_NAME, u"MBG"),
x509.NameAttribute(NameOID.ORGANIZATION_NAME, u"Lenovo"),
x509.NameAttribute(NameOID.COMMON_NAME, u"Lenovo"),
])
cert = x509.CertificateBuilder().subject_name(subject).issuer_name(issuer).public_key(key.public_key()).serial_number(x509.random_serial_number()).not_valid_before(datetime.datetime.utcnow() - datetime.timedelta(days=365)).not_valid_after(datetime.datetime.utcnow() + datetime.timedelta(days=365)).sign(key, hashes.SHA256())
# Must be written to file for Python's SSL library
with open("selfsigned.cert", "wb") as f:
f.write(cert.public_bytes(serialization.Encoding.PEM))
X509['cert'] = cert.public_bytes(serialization.Encoding.PEM).decode()
if config["verbose"] >= 2:
print(X509['cert'])
X509['fp'] = ''.join(f'{b:02x}' for b in cert.fingerprint(hashes.SHA256()))
if config["verbose"] >= 2:
print(X509['fp'])
return X509
# Gets list of IP addresses available to machine
# Ignores localhost and non-IPv4 addresses
def get_ip_addresses():
addresses = []
for iface in netifaces.interfaces():
for k,v in netifaces.ifaddresses(iface).items():
if v[0]['addr'] != '127.0.0.1' and len(v[0]['addr'].split('.')) == 4:
# Valid address!
addresses.append(v[0]['addr'])
# Return first available IP address
if len(addresses) == 0:
print("Unable to find valid IP address. Are you connected and have an",
"IPv4 address?")
sys.exit(3)
if config["verbose"] >= 1:
print("[ReadyForPy] IP addresses: ", ", ".join(addresses))
return addresses
# Generates the host info dictionary that will be processed for the qr code
def generate_host_info(keycert):
host_info = {}
host_info['fp'] = keycert['fp']
host_info['authLevel'] = 2
host_info['sn'] = 0
host_info['ips'] = get_ip_addresses();
host_info['timestamp'] = int(time.time())
host_info['user'] = ''.join(random.choices(RAND_VALID_CHARS, k=16))
host_info['pass'] = ''.join(random.choices(RAND_VALID_CHARS, k=16))
host_info['version'] = VER_STRING
content = ("[email protected]"
+ VER_STRING
+ str(host_info['timestamp'])
+ host_info['user']
+ host_info['pass']
+ str(EXPIRY_PERIOD) + "[\"" + host_info['ips'][0] + "\"]")
host_info['token'] = hashlib.sha256(content.encode()).hexdigest()[:16]
return host_info
def generate_qr(host_info):
qr_content = "motorolardpconnection" + json.dumps(host_info, separators=(',', ':'))
if config["verbose"] >= 1:
print("[ReadyForPy] QR content:", qr_content)
qr = qrcode.make(qr_content)
qr.show()
class ReadyForHandler(BaseHTTPRequestHandler):
def do_POST(self):
if self.path == "/rdp/connect":
raw_data = self.rfile.read().decode('utf8')
if config["verbose"] >= 2:
print("[ReadyForPy] Payload:", raw_data)
phone_info = {}
for kv_pair in raw_data.split('&'):
split = kv_pair.split('=')
phone_info[split[0]] = split[1]
if config["verbose"] >= 1:
print("[ReadyForPy]", split[0] + ": " + split[1])
if phone_info['token'] == host_info['token']:
print("*** You can close the QR code window now ***")
self.protocol_version = "HTTP/1.1"
self.send_response(200)
self.send_header("Content-Length", len("success"))
self.end_headers()
self.wfile.write(bytes("success", "utf8"))
if config["verbose"] >= 2:
subprocess.run(
shlex.split(XFREERDP_COMMAND.format(
config["freerdp_path"],
phone_info["phoneIp"],
config["resolution"],
host_info["user"],
host_info["pass"]
)),
universal_newlines=True
)
else:
# Suppress output
subprocess.run(
shlex.split(XFREERDP_COMMAND.format(
config["freerdp_path"],
phone_info['phoneIp'],
config["resolution"],
host_info['user'],
host_info['pass']
)),
stdout = subprocess.DEVNULL,
stderr = subprocess.DEVNULL,
universal_newlines=True
)
sys.exit(0)
elif self.path == "/rdp/connect/success":
print("Success!")
self.protocol_version = "HTTP/1.1"
self.send_response(200)
self.send_header("Content-Length", len("success"))
self.end_headers()
self.wfile.write(bytes("success", "utf8"))
if not config["no_check_freerdp"]:
check_freerdp()
keycert = generate_cert()
host_info = generate_host_info(keycert)
generate_qr(host_info)
try:
httpd = HTTPServer(('', HTTP_PORT), ReadyForHandler)
# TODO: ssl.wrap_socket is deprecated with Py 3.7+
httpd.socket = ssl.wrap_socket(httpd.socket,
keyfile="selfsigned.key",
certfile="selfsigned.cert",
server_side=True)
httpd.serve_forever()
except KeyboardInterrupt as interrupt:
print("\nQuitting...")