-
Notifications
You must be signed in to change notification settings - Fork 199
/
Copy pathsetssidvlanid.py
234 lines (180 loc) · 8.6 KB
/
setssidvlanid.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
# This is a script to set the VLAN on an SSID. Usage:
# python setssidvlanid.py -k <api_key> -o <org_name> -n <ssid_name> -v <vlan_id> [-t <net_tag>]
#
# Mandatory arguments:
# -k <api_key> Your Meraki Dashboard API key
# -o <org_name> Name of the organization you want to be processed
# -n <ssid_name> Name of the SSID to be processed
# -v <vlan_id> New VLAN ID (number) for the SSID
# Optional argument:
# -t <net_tag> Only process networks tagged <net_tag>
#
# Example:
# python setssidvlanid.py -k 1234 -o "Meraki Inc" -n "Meraki corp" -v 10
#
# The script requires the Requests module. To install it via pip:
# pip install requests
#
# To pass parameters containing spaces in Windows, use double quotes "".
#
# This file was last modified on 2018-03-23
import sys, getopt, requests, json, time
from datetime import datetime
#Used for time.sleep(API_EXEC_DELAY). Delay added to avoid hitting dashboard API max request rate
API_EXEC_DELAY = 0.21
#connect and read timeouts for the Requests module
REQUESTS_CONNECT_TIMEOUT = 30
REQUESTS_READ_TIMEOUT = 30
#used by merakirequestthrottler(). DO NOT MODIFY
LAST_MERAKI_REQUEST = datetime.now()
def merakirequestthrottler():
#makes sure there is enough time between API requests to Dashboard not to hit shaper
global LAST_MERAKI_REQUEST
if (datetime.now()-LAST_MERAKI_REQUEST).total_seconds() < API_EXEC_DELAY:
time.sleep(API_EXEC_DELAY)
LAST_MERAKI_REQUEST = datetime.now()
return
def printusertext(p_message):
#prints a line of text that is meant for the user to read
#do not process these lines when chaining scripts
print('@ %s' % p_message)
def printhelp():
#prints help text
printusertext('This is a script to set the VLAN on an SSID. Usage:')
printusertext(' python setssidvlanid.py -k <api_key> -o <org_name> -n <ssid_name> -v <vlan_id> [-t <net_tag>]')
printusertext('')
printusertext('Example:')
printusertext(' python setssidvlanid.py -k 1234 -o "Meraki Inc" -n "Meraki corp" -v 10')
printusertext('')
printusertext('To pass parameters containing spaces in Windows, use double quotes "".')
def getorgid(p_apikey, p_orgname):
#looks up org id for a specific org name
#on failure returns 'null'
merakirequestthrottler()
try:
r = requests.get('https://api.meraki.com/api/v0/organizations', headers={'X-Cisco-Meraki-API-Key': p_apikey, 'Content-Type': 'application/json'}, timeout=(REQUESTS_CONNECT_TIMEOUT, REQUESTS_READ_TIMEOUT))
except:
printusertext('ERROR 01: Unable to contact Meraki cloud')
sys.exit(2)
if r.status_code != requests.codes.ok:
return 'null'
rjson = r.json()
for record in rjson:
if record['name'] == p_orgname:
return record['id']
return('null')
def getnwlist(p_apikey, p_shardhost, p_orgid):
#returns a list of all networks in an organization
#on failure returns a single record with 'null' name and id
merakirequestthrottler()
try:
r = requests.get('https://%s/api/v0/organizations/%s/networks' % (p_shardhost, p_orgid), headers={'X-Cisco-Meraki-API-Key': p_apikey, 'Content-Type': 'application/json'}, timeout=(REQUESTS_CONNECT_TIMEOUT, REQUESTS_READ_TIMEOUT) )
except:
printusertext('ERROR 02: Unable to contact Meraki cloud')
sys.exit(2)
returnvalue = []
if r.status_code != requests.codes.ok:
returnvalue.append({'name': 'null', 'id': 'null'})
return(returnvalue)
return(r.json())
def getssids(p_apikey, p_shardhost, p_netid):
#returns a list of all MR SSIDs in a network
merakirequestthrottler()
try:
r = requests.get('https://%s/api/v0/networks/%s/ssids' % (p_shardhost, p_netid), headers={'X-Cisco-Meraki-API-Key': p_apikey, 'Content-Type': 'application/json'}, timeout=(REQUESTS_CONNECT_TIMEOUT, REQUESTS_READ_TIMEOUT) )
except:
printusertext('ERROR 03: Unable to contact Meraki cloud')
sys.exit(2)
returnvalue = []
if r.status_code != requests.codes.ok:
returnvalue.append({'number': 'null'})
return(returnvalue)
return(r.json())
def setssidattribute(p_apikey, p_shardhost, p_netid, p_ssidnum, p_attribute, p_value):
#writes one attribute to one SSID
merakirequestthrottler()
try:
r = requests.put('https://%s/api/v0/networks/%s/ssids/%s' % (p_shardhost, p_netid, p_ssidnum), data=json.dumps({p_attribute: p_value}), headers={'X-Cisco-Meraki-API-Key': p_apikey, 'Content-Type': 'application/json'}, timeout=(REQUESTS_CONNECT_TIMEOUT, REQUESTS_READ_TIMEOUT) )
except:
printusertext('ERROR 04: Unable to contact Meraki cloud')
sys.exit(2)
if r.status_code != requests.codes.ok:
return ('null')
return('ok')
def main(argv):
printusertext('INFO: Script started at %s' % datetime.now())
# python setssidvlanid.py -k <api_key> -o <org_name> -n <ssid_name> -v <vlan_id> [-t <net_tag>]
#initialize variables for command line arguments
arg_apikey = ''
arg_orgname = ''
arg_ssidname = ''
arg_vlanid = ''
arg_nettag = ''
#get command line arguments
try:
opts, args = getopt.getopt(argv, 'hk:o:n:v:t:')
except getopt.GetoptError:
printhelp()
sys.exit(2)
for opt, arg in opts:
if opt == '-h':
printhelp()
sys.exit()
elif opt == '-k':
arg_apikey = arg
elif opt == '-o':
arg_orgname = arg
elif opt == '-n':
arg_ssidname = arg
elif opt == '-v':
arg_vlanid = arg
elif opt == '-t':
arg_nettag = arg
#check if all required parameters have been given
if arg_apikey == '' or arg_orgname == '' or arg_ssidname == '' or arg_vlanid == '':
printhelp()
sys.exit(2)
#set operational mode flags (True/False)
flag_modematchtag = (arg_nettag != '')
#resolve orgid
orgid = getorgid(arg_apikey, arg_orgname)
if orgid == 'null':
printusertext('ERROR 05: Unable to find org named "%s"' % arg_orgname)
sys.exit(2)
#get networks' list for orgid
netlist = getnwlist(arg_apikey, 'api.meraki.com', orgid)
if netlist[0]['id'] == 'null':
printusertext('ERROR 06: Error retrieving net list for org id "%s"' % orgid)
sys.exit(2)
for net in netlist:
#check that network has required tag, if user has given one
flag_tagmatchsuccess = True
if flag_modematchtag:
if net['tags'] is None:
flag_tagmatchsuccess = False
else:
if net['tags'].find(arg_nettag) == -1:
flag_tagmatchsuccess = False
if flag_tagmatchsuccess:
#get SSID list
ssidlist = getssids(arg_apikey, 'api.meraki.com', net['id'])
if ssidlist[0]['number'] == 'null':
printusertext('WARNING: Skipping network "%s": No MR config' % net['name'])
else:
flag_ssidnotfound = True
#get SSID number corresponding to SSID name
for ssid in ssidlist:
if ssid['name'] == arg_ssidname:
printusertext('INFO: Setting VLAN ID "%s" for net "%s", SSID %s "%s"' % (arg_vlanid, net['name'], ssid['number'], ssid['name']))
#set VLAN for correct SSID number
if ssid['ipAssignmentMode'] in ['Layer 3 roaming with a concentrator', 'VPN']:
flag_ssidnotfound = False
setssidattribute(arg_apikey, 'api.meraki.com', net['id'], ssid['number'], 'vlanId', arg_vlanid)
break #SSID name is unique, so no need to check the rest
if flag_ssidnotfound:
printusertext('WARNING: Skipping network "%s": no SSID "%s" or wrong addressing mode' % (net['name'], arg_ssidname))
else:
printusertext('WARNING: Skipping network "%s": No tag "%s"' % (net['name'], arg_nettag))
printusertext('INFO: Reached end of script at %s' % datetime.now())
if __name__ == '__main__':
main(sys.argv[1:])