-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathopenl2m.py
executable file
·279 lines (225 loc) · 8.97 KB
/
openl2m.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
#
# This file is part of Open Layer 2 Management (OpenL2M).
#
# OpenL2M is free software: you can redistribute it and/or modify it under
# the terms of the GNU General Public License version 3 as published by
# the Free Software Foundation.
#
# This program is distributed in the hope that it will be useful, but WITHOUT
# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
# FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
# more details. You should have received a copy of the GNU General Public
# License along with OpenL2M. If not, see <http://www.gnu.org/licenses/>.
#
#
# This implements a Python 3 client class for OpenL2M's REST API.
#
import requests
class Server:
"""Object that implements the OpenL2M client side of the REST API on the server"""
def __init__(self, url: str, token: str):
self.url = url # the base URL where the REST queries will be made.
self.token = token # the API token
self.headers = {
"Authorization": f"Token {token}",
"Content-Type": "application/json", # we use JSON posting format.
}
self.status = (
0 # will contain the HTTP response status, or -1 if Exception caught.
)
self.error = "" # will contain string value of last caught error.
self.response = (
False # will contain the response from the last .get() or .post()
)
self.verbose = 0 # debug output level. If > 0, will print debug info to stdout
self.verify = True # verify SSL cert chain ?
def set_verify(self, verify: bool) -> None:
"""
Enable or disable SSL verification.
Args:
verify (bool): Verify SSL (True) or not (False).
Returns:
n/a
"""
self.verify = verify
def set_verbose(self, verbose: int):
"""
Set the verbosity level.
Args:
verbose (int): the verbosity level.
Returns:
n/a
"""
self.verbose = verbose
def debug(self, text: str, level: int = 1):
"""Show debug text if verbosity is at or above level given.
Args:
text (str): text to print to output.
level (int): the level at which to print the output.
Returns:
n/a
"""
if self.verbose >= level:
print(f"DEBUG{level}: {text}")
def execute(self, endpoint: str, data: dict = {}) -> bool:
"""Execute a GET or a POST, depending on data given (POST) or not (GET).
Will set self.response, self.status and self.error as applicable.
Args:
endpoint (str): the API endpoint, to add to the host url.
data (dict): if given, a dictionary of POST data that will be sent as JSON.
Returns:
bool: True if successfull, False if not.
"""
self.debug(f"execute(endpoint={endpoint})")
self.status = 0
self.error = ""
self.response = False
try:
url = f"{self.url}{endpoint}"
if data:
self.debug(f" POST: {url}")
response = requests.post(
url=url, json=data, headers=self.headers, verify=self.verify
)
else:
self.debug(f" GET: {url}")
response = requests.get(
url=f"{self.url}{endpoint}",
headers=self.headers,
verify=self.verify,
)
self.debug(f" Response code {response.status_code}")
self.debug(f" Body:\n{response.content}\n", level=2)
except Exception as e:
self.debug(f" ERROR: {e}")
self.status = -1
self.error = f"{e}"
return False
self.response = response
self.status = response.status_code
if response.status_code != requests.codes.ok: # HTTP returned error code!
return False
return True
def devices(self) -> bool:
"""Get the list of devices available to this token.
Args:
n/a
Returns:
bool: True if successfull, False if not.
"""
self.debug("devices()")
return self.execute(endpoint="api/switches/")
def menu(self) -> bool:
return self.devices()
def switches(self) -> bool:
return self.devices()
def device(self, device_url: str):
"""Get an object to a device.
Args:
device_url (str): proper URL path to this device, likely retrieved from get_menu().
Returns:
bool: True if successfull, False if not.
"""
self.debug(f"device(device_url={device_url})")
return Device(device_url=device_url, token=self.token)
def stats(self) -> bool:
"""Get usage statistics of the OpenL2M server.
Args:
n/a
Returns:
bool: True if successfull, False if not.
"""
self.debug("stats()")
return self.execute(endpoint="api/stats/")
def environment(self) -> bool:
"""Get information about the runtime environment of the OpenL2M server.
Args:
n/a
Returns:
bool: True if successfull, False if not.
"""
self.debug("environment()")
return self.execute(endpoint="api/environment/")
#
# Device() implements access to a specific device
#
class Device(Server):
"""This class allow access to a specific device
that is accessible/manageable from the OpenL2M Server.
"""
def __init__(self, device_url: str, token: str):
super().__init__(url=device_url, token=token)
def get(self, view=""):
"""Get the device/switch data for the current switch.
args:
view (str): if "details" API will return the ARP/Ethernet/LLDP details data.
any other value will return regular interface data.
Returns:
bool: True if successfull, False if not.
"""
self.debug(f"read_switch(view={view})")
endpoint = ""
if view == "details":
endpoint = "details/"
else:
endpoint = ""
return self.execute(endpoint=endpoint)
def set_interface_state(self, interface_id: str, state: bool) -> bool:
"""Set the interface state.
Args:
interface_id (str): the identifier of the requested interface, as retrieved from read_switch().
state (bool): True to enable or False to disable the requested interface.
Returns:
bool: True if successfull, False if not.
"""
self.debug(f"set_interface_state(interface_id={interface_id}, state={state})")
self.error_string = ""
if state:
data = {"state": "on"}
else:
data = {"state": "off"}
endpoint = f"interface/{interface_id}/state/"
return self.execute(endpoint=endpoint, data=data)
def set_interface_poe_state(self, interface_id: str, poe_state: bool) -> bool:
"""Set the interface PoE status.
Args:
interface_id (str): the identifier of the requested interface, as retrieved from read_switch().
poe_state (bool): True to enable or False to disable PoE on the requested interface.
Returns:
bool: True if successfull, False if not.
"""
self.debug(
f"set_interface_poe_status(interface_id={interface_id}, poe_state={poe_state})"
)
self.error_string = ""
data = {"poe_state": poe_state}
endpoint = f"interface/{interface_id}/poe_state/"
return self.execute(endpoint=endpoint, data=data)
def set_interface_description(self, interface_id: str, description: str) -> bool:
"""Set the interface description.
Args:
interface_id (str): the identifier of the requested interface, as retrieved from read_switch().
description (str): the new description for the given interface.
Returns:
bool: True if successfull, False if not.
"""
self.debug(
f"set_interface_description(interface_id={interface_id}, description={description})"
)
data = {"description": description}
endpoint = f"interface/{interface_id}/description/"
return self.execute(endpoint=endpoint, data=data)
def set_interface_vlan(self, interface_id: str, vlan_id: int) -> bool:
"""Set the interface untagged vlan id.
Args:
interface_id (str): the identifier of the requested interface, as retrieved from read_switch().
vlan_id (int): the new untagged vlan id.
Returns:
bool: True if successfull, False if not.
"""
self.debug(
f"set_interface_vlan(interface_id={interface_id}, vlan_id={vlan_id})"
)
data = {"vlan": vlan_id}
endpoint = f"interface/{interface_id}/vlan/"
return self.execute(endpoint=endpoint, data=data)