2020-10-14 12:39:40 +02:00
|
|
|
import logging
|
2021-07-09 11:10:43 +02:00
|
|
|
import subprocess
|
|
|
|
|
2020-10-14 12:39:40 +02:00
|
|
|
from netbox_agent.misc import is_tool
|
2019-08-26 11:05:41 +02:00
|
|
|
|
|
|
|
|
|
|
|
class LLDP():
|
2020-02-02 20:08:56 +01:00
|
|
|
def __init__(self, output=None):
|
2020-10-14 12:39:40 +02:00
|
|
|
if not is_tool('lldpctl'):
|
|
|
|
logging.debug('lldpd package seems to be missing or daemon not running.')
|
2020-02-02 20:08:56 +01:00
|
|
|
if output:
|
|
|
|
self.output = output
|
|
|
|
else:
|
|
|
|
self.output = subprocess.getoutput('lldpctl -f keyvalue')
|
2019-08-26 11:05:41 +02:00
|
|
|
self.data = self.parse()
|
|
|
|
|
|
|
|
def parse(self):
|
|
|
|
output_dict = {}
|
2019-09-09 15:32:14 +02:00
|
|
|
vlans = {}
|
|
|
|
vid = None
|
2019-08-26 11:05:41 +02:00
|
|
|
for entry in self.output.splitlines():
|
|
|
|
if '=' not in entry:
|
|
|
|
continue
|
|
|
|
path, value = entry.strip().split("=", 1)
|
2019-09-09 15:32:14 +02:00
|
|
|
split_path = path.split(".")
|
|
|
|
interface = split_path[1]
|
|
|
|
path_components, final = split_path[:-1], split_path[-1]
|
2019-08-26 11:05:41 +02:00
|
|
|
current_dict = output_dict
|
2019-09-09 15:32:14 +02:00
|
|
|
|
|
|
|
if vlans.get(interface) is None:
|
|
|
|
vlans[interface] = {}
|
|
|
|
|
2019-08-26 11:05:41 +02:00
|
|
|
for path_component in path_components:
|
|
|
|
current_dict[path_component] = current_dict.get(path_component, {})
|
|
|
|
current_dict = current_dict[path_component]
|
2019-09-09 15:32:14 +02:00
|
|
|
if 'vlan-id' in path:
|
|
|
|
vid = value
|
|
|
|
vlans[interface][value] = vlans[interface].get(vid, {})
|
|
|
|
elif path.endswith('vlan'):
|
Fix KeyError: 'pvid' in lldp.py
```
lldp.eth0.vlan.vlan-id=300
lldp.eth0.vlan.pvid=yes
lldp.eth0.vlan=VLAN300
```
```
Version: lldpd 1.0.11
```
```
{'300': {'pvid': True}, 'VLAN300': {}}
Traceback (most recent call last):
File "/root/.local/bin/netbox_agent", line 8, in <module>
sys.exit(main())
File "/root/.local/lib/python3.10/site-packages/netbox_agent/cli.py", line 44, in main
return run(config)
File "/root/.local/lib/python3.10/site-packages/netbox_agent/cli.py", line 39, in run
server.netbox_create_or_update(config)
File "/root/.local/lib/python3.10/site-packages/netbox_agent/server.py", line 292, in netbox_create_or_update
self.network.create_or_update_netbox_network_cards()
File "/root/.local/lib/python3.10/site-packages/netbox_agent/network.py", line 417, in create_or_update_netbox_network_cards
ret, interface = self.reset_vlan_on_interface(nic, interface)
File "/root/.local/lib/python3.10/site-packages/netbox_agent/network.py", line 234, in reset_vlan_on_interface
pvid_vlan = [key for (key, value) in lldp_vlan.items() if value['pvid']]
File "/root/.local/lib/python3.10/site-packages/netbox_agent/network.py", line 234, in <listcomp>
pvid_vlan = [key for (key, value) in lldp_vlan.items() if value['pvid']]
KeyError: 'pvid'
```
2022-04-26 12:25:28 +02:00
|
|
|
vid = value.replace('vlan-', '').replace('VLAN', '')
|
2019-09-09 15:32:14 +02:00
|
|
|
vlans[interface][vid] = vlans[interface].get(vid, {})
|
|
|
|
elif 'pvid' in path:
|
|
|
|
vlans[interface][vid]['pvid'] = True
|
|
|
|
if 'vlan' not in path:
|
|
|
|
current_dict[final] = value
|
|
|
|
for interface, vlan in vlans.items():
|
|
|
|
output_dict['lldp'][interface]['vlan'] = vlan
|
2020-10-14 12:39:40 +02:00
|
|
|
if not output_dict:
|
|
|
|
logging.debug('No LLDP output, please check your network config.')
|
2019-08-26 11:05:41 +02:00
|
|
|
return output_dict
|
|
|
|
|
|
|
|
def get_switch_ip(self, interface):
|
|
|
|
# lldp.eth0.chassis.mgmt-ip=100.66.7.222
|
|
|
|
if self.data['lldp'].get(interface) is None:
|
|
|
|
return None
|
|
|
|
return self.data['lldp'][interface]['chassis']['mgmt-ip']
|
|
|
|
|
|
|
|
def get_switch_port(self, interface):
|
|
|
|
# lldp.eth0.port.descr=GigabitEthernet1/0/1
|
|
|
|
if self.data['lldp'].get(interface) is None:
|
|
|
|
return None
|
2020-02-02 20:08:56 +01:00
|
|
|
if self.data['lldp'][interface]['port'].get('ifname'):
|
|
|
|
return self.data['lldp'][interface]['port']['ifname']
|
2019-08-26 11:05:41 +02:00
|
|
|
return self.data['lldp'][interface]['port']['descr']
|
|
|
|
|
|
|
|
def get_switch_vlan(self, interface):
|
|
|
|
# lldp.eth0.vlan.vlan-id=296
|
|
|
|
if self.data['lldp'].get(interface) is None:
|
|
|
|
return None
|
2019-09-09 15:32:14 +02:00
|
|
|
return self.data['lldp'][interface]['vlan']
|