-
Notifications
You must be signed in to change notification settings - Fork 5
/
Copy pathList_Crestron_Devices.py
246 lines (209 loc) · 10.3 KB
/
List_Crestron_Devices.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
#!/usr/bin/env python
# -*- coding: UTF-8 -*-
"""
Copyright © 2017 by Stephen Genusa
Redistribution and use in source and binary forms, with or without
modification, are permitted provided that the following conditions are met:
1. Redistributions of source code must retain the above copyright notice,
this list of conditions and the following disclaimer.
2. Redistributions in binary form must reproduce the above copyright notice,
this list of conditions and the following disclaimer in the documentation
and/or other materials provided with the distribution.
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
POSSIBILITY OF SUCH DAMAGE.
"""
from __future__ import print_function
import argparse
import os
import re
import socket
import subprocess
import sys
from time import sleep
#
import netifaces
MAX_RETRIES = 3
BUFF_SIZE = 2000
CIP_PORT = 41794
BROADCAST_IP = '255.255.255.255'
FORMATTING = " " * 5
UDP_MSG = \
"\x14\x00\x00\x00\x01\x04\x00\x03\x00\x00\x66\x65\x65\x64" + \
("\x00" * 252)
class CrestronDeviceFinder(object):
def __init__(self, args):
"""
initialize internal properties
"""
self.active_ips_to_check = []
self.args = args
self.crestron_devices = {}
def initialize_run_variables(self):
self.console_prompt = ""
# Print iterations progress
def print_progress(self, iteration, total, prefix='', suffix='', decimals=1, bar_length=100):
"""
From https://gist.github.com/aubricus/f91fb55dc6ba5557fbab06119420dd6a w/mod
Call in a loop to create terminal progress bar
@params:
iteration - Required : current iteration (Int)
total - Required : total iterations (Int)
prefix - Optional : prefix string (Str)
suffix - Optional : suffix string (Str)
decimals - Optional : positive number of decimals in percent complete (Int)
bar_length - Optional : character length of bar (Int)
"""
str_format = "{0:." + str(decimals) + "f}"
percents = str_format.format(100 * (iteration / float(total)))
filled_length = int(round(bar_length * iteration / float(total)))
bar = chr(178) * filled_length + '-' * (bar_length - filled_length)
sys.stdout.write('\r%s |%s| %s%s %s' % (prefix, bar, percents, '%', suffix)),
if iteration == total:
sys.stdout.write('\n')
sys.stdout.flush()
def open_device_connection(self):
"""
Open the device connection, attempting port 41795
"""
try:
self.sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
server_address = (self.device_ip_address, 41795)
self.sock.settimeout(2.0)
self.sock.connect(server_address)
return True
except:
#print ("exception thrown:", sys.exc_info()[0])
pass
return False
def close_device_connection(self):
"""
Close the socket
"""
try:
self.sock.close()
except:
pass
def get_console_prompt(self):
"""
Determine the device console prompt
"""
data = ""
for _unused in range(0, MAX_RETRIES):
try:
self.sock.settimeout(30.0)
for cr in ["\r", "\r\n"]:
self.sock.sendall(cr)
sleep(.25)
data += self.sock.recv(BUFF_SIZE)
search = re.findall("[\n\r]([\w-]{3,30})>", data, re.MULTILINE)
if search:
self.console_prompt = search[0]
return True
except Exception as e:
pass
return False
def ip_responding_to_ping(self, ip_addr, attempts, limbo):
wait_time = 100
for _ in range(0, attempts):
if not subprocess.Popen(["ping", "-n", "1", "-w", str(wait_time), ip_addr], stdout=limbo, stderr=limbo).wait():
return True
wait_time += 100
def show_devices_using_icmp(self, subnet):
"""
Build a list of devices that respond to ping for a /24 subnet like 17.1.6.{1}:
"""
print ("Building list of active IP addresses on subnet {0}.0/24\nPlease wait. This will take about a few minutes depending on how many devices are found...".format(subnet))
with open(os.devnull, "wb") as limbo:
for last_octet in xrange(1, 255):
ip = "{0}.{1}".format(subnet, last_octet)
if self.ip_responding_to_ping(ip, 3, limbo):
self.active_ips_to_check.append(ip)
self.print_progress (last_octet, 254, bar_length = 70)
if self.active_ips_to_check:
print("\nLocated {0} active IPs on subnet. Now testing for console on each IP.".format(len(self.active_ips_to_check)))
self.active_ips_len = len(self.active_ips_to_check)
for index, self.device_ip_address in enumerate(self.active_ips_to_check):
item_pos = "(" + str(index + 1) + "/" + str(self.active_ips_len) + ")"
if self.open_device_connection():
if self.get_console_prompt():
print(FORMATTING + self.console_prompt.ljust(30) + str(" located at " + self.device_ip_address).ljust(25), item_pos)
self.close_device_connection()
else:
print(FORMATTING + "Console not found on", self.device_ip_address)
else:
print(FORMATTING + "N/A - No CIP".ljust(30) + str(" located at " + self.device_ip_address).ljust(25), item_pos)
sys.stdout.flush()
def show_devices_using_udp(self):
for iface in netifaces.interfaces():
# if interface has ipv4 address
if netifaces.AF_INET in netifaces.ifaddresses(iface):
# if addr attribute in interface dictionary
if 'addr' in netifaces.ifaddresses(iface)[netifaces.AF_INET][0]:
# get the ipv4 address
cur_ip = netifaces.ifaddresses(iface)[netifaces.AF_INET][0]['addr']
# tell user what we are testing
print("Testing IP subnet", cur_ip)
# set the UDP test up
sock = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
# bind the socket to the local IP:CIP_PORT
sock.bind((cur_ip, CIP_PORT))
sock.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
sock.setsockopt(socket.SOL_SOCKET, socket.SO_BROADCAST, 1)
# send the UDP message to the broadcast IP:CIP_PORT
sock.sendto(UDP_MSG, (BROADCAST_IP, CIP_PORT))
# we want an exception to end the while True: loop below,
# based on the following timeout
sock.settimeout(3.0)
try:
while True:
# get a UDP response packet and store in buffer
data, addr = sock.recvfrom(4096)
# find the device hostname in the response buffer
search = re.findall ('\x00([a-zA-Z0-9-]{2,30})\x00', data[9:40])
if search:
dev_name = search[0]
# get the ipv4 address received from sock.recvfrom()
dev_ip = addr[0]
# find if ver info is part of UDP packet
firmware_info = ""
search = re.findall ('\x00([\w].{10,80})\x00', data[265:350])
if search:
firmware_info = search[0]
# add only new devices and skip our own packet
if dev_name not in self.crestron_devices and dev_name != "feed":
print((FORMATTING + dev_name + " located at " + dev_ip).ljust(35) + \
"\n" + (FORMATTING * 2) + firmware_info)
# save the device to a dictionary so we don't repeat it
self.crestron_devices[dev_name] = dev_ip
except:
pass
print()
print("\nLocated a total of", len(self.crestron_devices), "Crestron devices")
def find_devices(self):
if self.args.autolocatecrestron:
self.show_devices_using_udp()
elif self.args.autolocateactiveips:
self.show_devices_using_icmp(self.args.autolocateactiveips)
if __name__ == "__main__":
# pylint: disable-msg=C0103
print("\nStephen Genusa's Crestron Device Locator\n")
parser = argparse.ArgumentParser()
parser.add_argument("-ala", "--autolocateactiveips", default="", type=str,
help="Automatically locate active IPs on a subnet and look for Crestron devices. \n Example: 174.209.101 as an argument will check 174.209.101.0/24")
parser.add_argument("-alc", "--autolocatecrestron", action="store_true",
help="Automatically locate Crestron devices on all connected subnets and build documentation")
args = parser.parse_args()
if not args.autolocatecrestron and not args.autolocateactiveips:
parser.print_help()
exit()
documenter = CrestronDeviceFinder(args)
documenter.find_devices()