-
Notifications
You must be signed in to change notification settings - Fork 0
/
cisco-macaddr.py
421 lines (369 loc) · 14.1 KB
/
cisco-macaddr.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
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
"""
References:
https://stackoverflow.com/questions/6545023/how-to-sort-ip-addresses-stored-in-dictionary-in-python/6545090#6545090
https://stackoverflow.com/questions/20944483/python-3-sort-a-dict-by-its-values
https://docs.python.org/3.3/tutorial/datastructures.html
https://www.quora.com/How-do-I-write-a-dictionary-to-a-file-in-Python
https://www.programiz.com/python-programming/break-continue
https://www.ascii-art-generator.org/
read mac-addr.txt containing the output of
show mac add int g1/0/1 | i Gi
show mac add int g1/0/2 | i Gi
show mac add int g1/0/3 | i Gi
show mac add int g1/0/4 | i Gi
and create a list of Vlan, Mac Address, interface and manufacturer.
test-switch#show mac add int g1/0/1 | i Gi
10 8434.97a7.708b DYNAMIC Gi1/0/1
test-switch#show mac add int g1/0/2 | i Gi
test-switch#show mac add int g1/0/3 | i Gi
10 0c4d.e9c1.4a0d DYNAMIC Gi1/0/3
test-switch#show mac add int g1/0/4 | i Gi
Output
Number Entries: 65
Vlan MAC Address Interface
10 8434.97a7.708b Gi1/0/1 HewlettP
--------------------------------------------------
10 0c4d.e9c1.4a0d Gi1/0/3 Apple
--------------------------------------------------
Uses the Parser library for Wireshark's OUI database from
https://github.com/coolbho3k/manuf to convert the MAC to a manufacture.
The database needs to be updated occasionally using:
python3 manuf.py -u
Changelog
March 7, 2018
Added code to read Mac2IP.json and use it as a dictionary of IP to MAC.
Mac2IP.json is created by running arp.py against the output
"show ip arp" or "sh ip arp vlan x" on a core switch
if Mac2IP.json is found in the same directory as macaddr.py it adds the
IP address to the output.
if Mac2IP.json is not found the IP address is not added
Vlan MAC Address Interface IP Vendor
20 f8b1.56d2.3c13 Gi1/0/3 10.129.20.70 Dell
----------------------------------------------------------------
20 0011.431b.b291 Gi1/0/16 10.129.20.174 Dell
----------------------------------------------------------------
March 24, 2018
Added an MD5 hash function to the list of MAC addresses. This gives a
quick comparison of the before
and after is some cables got swapped but are on the correct vlan.
Added a sorted output of the MAC addresses. If there are differences
before and after you can save the list of MACs and use MELD or Notepad++
(with the compare plugin) to see what is different.
Hash of all the MAC addresses
6449620420f0d67bffd26b65e9a824a4
Sorted list of MAC Addresses
0018.c840.1295
0018.c840.12a8
0027.0dbd.9f6e
April 7, 2018
Added output for PingInfoView (nirsoft.net)
PingInfo Data
10.56.238.150 b499.ba01.bc82
10.56.239.240 0026.5535.7b7a
April 30, 2018
Added better error trapping for the json and mac-addr.txt.
Stopped stripping DYNAMIC and STATIC from the input.
May 15, 2018
Fixed a bug in the IP address selection loop. I wasn't clearing IP_Data at the
end of the loop so is an interface didn't have an IP address in the json file
it would use the last IP address.
Clear the IP address in case the next interface has a MAC but no IP address
IP_Data = ''
"""
import argparse
import hashlib
import json
import os
import re
import sys
from icecream import ic
import manuf
# ic.enable()
ic.disable()
__author__ = "Michael Hubbard"
__author_email__ = "[email protected]"
__copyright__ = ""
__license__ = "Unlicense"
vernum = "1.1"
AsciiArt = """
__ __ _ ____ ____ __ __ __ _
| \/ | / \ / ___| |___ \ | \/ | __ _ _ __ _ _ / _| __ _ ___| |_
| |\/| | / _ \| | __) | | |\/| |/ _` | '_ \| | | | |_ / _` |/ __| __|
| | | |/ ___ \ |___ / __/ | | | | (_| | | | | |_| | _| (_| | (__| |_
|_| |_/_/ \_\____| |_____| |_| |_|\__,_|_| |_|\__,_|_| \__,_|\___|\__|
"""
def version():
"""
This function prints the version of this program. It doesn't allow
any argument.
"""
print(AsciiArt)
print("+----------------------------------------------------------------------+")
print(
"| "
+ sys.argv[0]
+ " Version "
+ vernum
+ " |"
)
print("| This program is free software; you can redistribute it and/or modify |")
print("| it in any way you want. If you improve it please send me a copy at |")
print("| the email address below. |")
print("| |")
print("| Author: Michael Hubbard, [email protected] |")
print("| mwhubbard.blogspot.com |")
print("| @rikosintie |")
print("+----------------------------------------------------------------------+")
def remove_empty_lines(filename):
if not os.path.isfile(filename):
print("{} does not exist ".format(filename))
return
with open(filename) as filehandle:
lines = filehandle.readlines()
with open(filename, "w") as filehandle:
lines = filter(lambda x: x.strip(), lines)
filehandle.writelines(lines)
def create_filename(sub_dir1: str, extension: str = "", sub_dir2="") -> str:
"""
returns a valid path regardless of the OS
Args:
sub_dir1 (str): name of the sub directory off the cwd required
extension (str, optional): string appended after hostname - ex. -interface.txt
sub_dir2 (str, optional): if a nested sub_dir is used Defaults to "".
Returns:
str: full pathname of the file to be written
"""
current_path = os.getcwd()
extension = hostname + extension
int_report = os.path.join(current_path, sub_dir1, sub_dir2, extension)
return int_report
def get_current_path():
current_path = os.getcwd()
return current_path
parser = argparse.ArgumentParser(
description="-s site, -c core hostname in a Core/IDF deployment"
)
parser.add_argument(
"-s",
"--site",
help="Site name - ex. HQ",
)
parser.add_argument(
"-c",
"--coreswitch", # Optional (but recommended) long version
default="",
help="Coreswitch hostname",
)
args = parser.parse_args()
site = args.site
core = args.coreswitch
if site is None:
print("-s site name is a required argument")
sys.exit()
else:
dev_inv_file = "device-inventory-" + site + ".csv"
ic(dev_inv_file)
# check if site's device inventory file exists
if not os.path.isfile(dev_inv_file):
print("{} doesn't exist ".format(dev_inv_file))
sys.exit()
remove_empty_lines(dev_inv_file)
with open(dev_inv_file) as devices_file:
fabric = devices_file.readlines()
print("-" * (len(dev_inv_file) + 23))
print(f"Reading devices from: {dev_inv_file}")
print("-" * (len(dev_inv_file) + 23))
uptime = []
for line in fabric:
line = line.strip("\n")
vendor = line.split(",")[1]
hostname = line.split(",")[2]
mac_file = create_filename("port-maps", "-mac-address.txt", "data")
ic(mac_file)
print()
# create a blank list to accept each line in the file
# data1 = []
# try:
# f = open(mac_file, 'r')
# except FileNotFoundError:
# print(f'{mac_file} does not exist')
# else:
# print()
# MAC addresses are expressed differently depending on the manufacture and even model device
# the formats that this script can parse are:
# 0a:0a:0a:0a:0a:0a, 0a-0a-0a-0a-0a-0a, 0a0a0a.0a0a0a0 and 0a0a0a-0a0a0a
# this should cover most Cisco and HP devices.
#
# create an empty dictionary to hold the mac-IP data
Mac_IP = {}
IP_Data = ""
device_name = ""
# create an empty list to hold MAC addresses for hashing
hash_list = []
# open the json created by arp.py if it exists
if core:
temp = hostname
hostname = core
my_json_file = create_filename("port-maps", "-Mac2IP.json")
hostname = temp
else:
my_json_file = create_filename("port-maps", "-Mac2IP.json")
# "JC-core-Mac2IP.json"
# my_json_file = hostname + "-Mac2IP.json"
try:
with open(my_json_file) as f:
Mac_IP = json.load(f)
except FileNotFoundError as fnf_error:
print(fnf_error)
print("IP Addresses will not be included because Mac2IP.json is not available")
my_json_file = None
p = manuf.MacParser()
# create a blank list to accept each line in the file
data = []
# mydatafile = '01_' + hostname + '-mac-address.txt'
try:
f = open(mac_file, "r")
except FileNotFoundError as fnf_error:
print(fnf_error)
sys.exit(0)
else:
for line in f:
match_PC = re.search(r"([0-9A-F]{2}[-:]){5}([0-9A-F]{2})", line, re.I)
match_Cisco = re.search(r"([0-9A-F]{4}[.]){2}([0-9A-F]{4})", line, re.I)
match_HP = re.search(r"([0-9A-F]{6}[-])([0-9A-F]{6})", line, re.I)
# strip out lines without a mac address
if match_PC or match_Cisco or match_HP:
data.append(line)
device_name_loc = line.find("#")
if device_name_loc != -1:
device_name = line[0:device_name_loc]
device_name = device_name.strip()
ic(device_name)
f.close
ct = len(data) - 1
counter = 0
IPs = []
print()
print("Device Name: %s " % device_name)
print("PingInfo Data")
while counter <= ct:
IP = data[counter]
# Remove newline at end
IP = IP.strip("\n")
# The Nexus line adds an * and spaces to the front of the line
IP = IP.strip("* ")
# The Nexus line includes additional fields that need to be stripped
IP = IP.replace(" 0 F F ", "")
IP = IP.replace(" ~~~ F F ", "")
# extract MAC Address and save to hash_list for hashing
L = str.split(IP)
Vlan = L[0]
Mac = L[1]
Mac_Type = L[2]
Interface_Num = L[3]
ic(Mac)
# The interface isn't in the same location in the output on all switches
# This loop seaches for a / in the value before picking the interface.
# Old method *******************************************
# if Interface_Num.find('/') == -1:
# Interface_Num = L[5]
# ******************************************************
ct2 = len(L)
count2 = 2
while count2 < ct2:
Interface_Num = L[count2]
if Interface_Num.find("/") == -1:
count2 += 1
# print(Interface_Num)
else:
break
# continue
temp = hash_list.append(Mac)
if Mac in Mac_IP:
IP_Data = Mac_IP[Mac]
else:
IP_Data = "No-Match"
# print the pinginfo data
print(IP_Data, Mac)
# pull the manufacturer with manuf
manufacture = p.get_manuf(Mac)
# Pad with spaces for output alignment
Front_Pad = 4 - len(Vlan)
Pad = 7 - len(Vlan) - Front_Pad
Vlan = (" " * Front_Pad) + Vlan + (" " * Pad)
# Pad MAC Address Field
Pad = 18 - len(Mac)
Mac = Mac + (" " * Pad)
# Pad type field
Pad = 11 - len(Mac_Type)
Mac_Type = Mac_Type + (" " * Pad)
# Pad Interface
Pad = 12 - len(Interface_Num)
Interface_Num = Interface_Num + (" " * Pad)
# Pad IP address field if the json file exists
if my_json_file:
Pad = 17 - len(IP_Data)
IP_Data = IP_Data + (" " * Pad)
# create the separator at 80 characters
Pad = "--" * 40
else:
# if not create the separator at 60 characters since there won't be IPs
Pad = "--" * 30
# Build the string
IP = Vlan + IP_Data + Mac + Mac_Type + Interface_Num + str(manufacture)
IPs.append(str(IP))
# IPs.append('--' * 40)
IPs.append(Pad)
# Clear the IP address in case the next interface has a MAC but no IP address
IP_Data = ""
counter = counter + 1
d = int(len(IPs) / 2)
# output_file = hostname + '-ports.txt' # Specify the filename for the output file
# loc = get_current_path()
# loc = loc + "/Final/"
# output_file = (
# loc + hostname + "-ports.txt"
# ) # Specify the filename for the output file
output_file = create_filename("port-maps", "-ports.txt", "Final")
ic(output_file)
with open(output_file, "w") as f:
sys.stdout = f
# Redirect print statements to the file
version()
print()
print(f"Number of Entries: {d}")
print()
print(f"Device Name: {device_name}")
if my_json_file:
header = "Vlan IP Address MAC Address Interface Vendor"
print("Vlan IP Address MAC Address Interface Vendor")
print("--" * 40)
else:
header = "Vlan MAC Address Interface Vendor"
print("Vlan MAC Address Interface Vendor")
print("--" * 30)
for IP in IPs:
print(IP)
# Reset print function to default
sys.stdout = sys.__stdout__
"""
hash the string of all macs. This gives a quick way to compare the
before and after MACS
"""
hash_list_str = str(hash_list)
# convert the string to bytes
b = hash_list_str.encode()
hash_object = hashlib.md5(b)
print()
print("Hash of all the MAC addresses")
print(hash_object.hexdigest())
print()
"""
print out the MAC Addresses sorted for review.
This is useful if the patch cables got mixed up during replacement
"""
print("Sorted list of MAC Addresses")
# print(hash_list)
for x in sorted(hash_list):
print(x)
print("End of output")