add datacenter location awareness with pseudo-driver (cmd and file) #4
8 changed files with 135 additions and 24 deletions
28
README.md
28
README.md
|
@ -2,7 +2,28 @@
|
||||||
|
|
||||||
|
|
||||||
This project aims to create hardware automatically into Netbox based on standard tools (dmidecode, lldpd, parsing /sys/, etc).
|
This project aims to create hardware automatically into Netbox based on standard tools (dmidecode, lldpd, parsing /sys/, etc).
|
||||||
The goal is to generate an existing infrastructure on Netbox and have the ability to update it regularly.
|
|
||||||
|
The goal is to generate an existing infrastructure on Netbox and have the ability to update it regularly by executing the agent.
|
||||||
|
|
||||||
|
# Features
|
||||||
|
|
||||||
|
* Create servers, chassis and blade through standard tools (`dmidecode`)
|
||||||
|
* Create physical network interfaces with IPs
|
||||||
|
* Generic ability to guess datacenters through drivers (`cmd` and `file` and custom ones)
|
||||||
|
|
||||||
|
# Configuration
|
||||||
|
|
||||||
|
```netbox:
|
||||||
|
url: 'http://netbox.internal.company.com'
|
||||||
|
token: supersecrettoken
|
||||||
|
|
||||||
|
datacenter_location:
|
||||||
|
# driver_file: /opt/netbox_driver_dc.py
|
||||||
|
driver: file:/etc/qualification
|
||||||
|
regex: "datacenter: (?P<datacenter>[A-Za-z0-9]+)"
|
||||||
|
# driver: 'cmd:lldpctl'
|
||||||
|
# regex = 'SysName: .*\.(?P<datacenter>[A-Za-z0-9]+)'```
|
||||||
|
```
|
||||||
|
|
||||||
# Hardware
|
# Hardware
|
||||||
|
|
||||||
|
@ -40,14 +61,9 @@ Tested on:
|
||||||
|
|
||||||
# TODO
|
# TODO
|
||||||
|
|
||||||
- [ ] HP(E) servers support
|
|
||||||
- [ ] Handle blade moving
|
- [ ] Handle blade moving
|
||||||
- [ ] Handle network cards (MAC, IP addresses)
|
- [ ] Handle network cards (MAC, IP addresses)
|
||||||
- [ ] Handle switch <> NIC connections (using lldp)
|
- [ ] Handle switch <> NIC connections (using lldp)
|
||||||
- [ ] Handle blade and server local changes (new NIC, new RAM, etc) using somekind of diff
|
- [ ] Handle blade and server local changes (new NIC, new RAM, etc) using somekind of diff
|
||||||
|
|
||||||
# Ideas
|
|
||||||
|
|
||||||
- [ ] CPU, RAID Card(s), RAM, Disks in `Device`'s `Inventory`
|
- [ ] CPU, RAID Card(s), RAM, Disks in `Device`'s `Inventory`
|
||||||
- [ ] `CustomFields` support with firmware versions for Device (BIOS), RAID Cards and disks
|
- [ ] `CustomFields` support with firmware versions for Device (BIOS), RAID Cards and disks
|
||||||
- [ ] Handle custom business logic : datacenter guessing logic based on hostname/switch name
|
|
||||||
|
|
12
netbox_agent.yaml.example
Normal file
12
netbox_agent.yaml.example
Normal file
|
@ -0,0 +1,12 @@
|
||||||
|
netbox:
|
||||||
|
url: 'http://netbox.internal.company.com'
|
||||||
|
token: supersecrettoken
|
||||||
|
|
||||||
|
datacenter_location:
|
||||||
|
driver: "cmd:cat /etc/qualification | tr [a-z] [A-Z]"
|
||||||
|
regex: "DATACENTER: (?P<datacenter>[A-Za-z0-9]+)"
|
||||||
|
# driver: 'cmd:lldpctl'
|
||||||
|
# regex: 'SysName: .*\.([A-Za-z0-9]+)'
|
||||||
|
#
|
||||||
|
# driver: "file:/tmp/datacenter"
|
||||||
|
# regex: "(.*)"
|
16
netbox_agent/config.py
Normal file
16
netbox_agent/config.py
Normal file
|
@ -0,0 +1,16 @@
|
||||||
|
import pynetbox
|
||||||
|
import yaml
|
||||||
|
|
||||||
|
with open('/etc/netbox_agent.yaml', 'r') as ymlfile:
|
||||||
|
# FIXME: validate configuration file
|
||||||
|
config = yaml.load(ymlfile)
|
||||||
|
|
||||||
|
netbox_instance = pynetbox.api(
|
||||||
|
url=config['netbox']['url'],
|
||||||
|
token=config['netbox']['token']
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
DATACENTER_LOCATION_DRIVER_FILE = config.get('datacenter_location').get('driver_file')
|
||||||
|
DATACENTER_LOCATION = config.get('datacenter_location').get('driver')
|
||||||
|
DATACENTER_LOCATION_REGEX = config.get('datacenter_location').get('regex')
|
||||||
fixed in fixed in 9f0db28
for the sanity check, will address that in a PR around configuration file
|
46
netbox_agent/datacenter.py
Normal file
46
netbox_agent/datacenter.py
Normal file
|
@ -0,0 +1,46 @@
|
||||||
|
import importlib
|
||||||
|
import importlib.machinery
|
||||||
|
|
||||||
|
from netbox_agent.config import DATACENTER_LOCATION, DATACENTER_LOCATION_DRIVER_FILE, \
|
||||||
|
DATACENTER_LOCATION_REGEX
|
||||||
|
|
||||||
|
|
||||||
|
class Datacenter():
|
||||||
|
"""
|
||||||
|
This class is used to guess the datacenter in order to push the information
|
||||||
|
in Netbox for a `Device`
|
||||||
|
|
||||||
|
A driver takes a `value` and evaluates a regex with a `capture group`.
|
||||||
|
|
||||||
|
There's embeded drivers such as `file` or `cmd` which read a file or return the
|
||||||
|
output of a file.
|
||||||
|
|
||||||
|
There's also a support for an external driver file outside of this project in case
|
||||||
|
the logic isn't supported here.
|
||||||
|
"""
|
||||||
|
def __init__(self, *args, **kwargs):
|
||||||
|
self.driver = DATACENTER_LOCATION.split(':')[0]
|
||||||
|
self.driver_value = ':'.join(DATACENTER_LOCATION.split(':')[1:])
|
||||||
|
self.driver_file = DATACENTER_LOCATION_DRIVER_FILE
|
||||||
|
|
||||||
|
if self.driver_file:
|
||||||
|
try:
|
||||||
|
# FIXME: Works with Python 3.3+, support older version?
|
||||||
|
loader = importlib.machinery.SourceFileLoader('driver_file', self.driver_file)
|
||||||
|
self.driver = loader.load_module()
|
||||||
|
except ImportError:
|
||||||
|
raise ImportError("Couldn't import {} as a module".format(self.driver_file))
|
||||||
|
else:
|
||||||
|
try:
|
||||||
|
self.driver = importlib.import_module(
|
||||||
|
'netbox_agent.drivers.datacenter_{}'.format(self.driver)
|
||||||
|
)
|
||||||
|
except ImportError:
|
||||||
|
raise ImportError("Driver {} doesn't exists".format(self.driver))
|
||||||
|
|
||||||
|
def get(self):
|
||||||
|
if not hasattr(self.driver, 'get'):
|
||||||
|
raise Exception(
|
||||||
|
"Your driver {} doesn't have a get() function, please fix it".format(self.driver)
|
||||||
|
)
|
||||||
|
return getattr(self.driver, 'get')(self.driver_value, DATACENTER_LOCATION_REGEX)
|
||||||
We should properly check if the module has a We should properly check if the module has a `get` method to output a proper error message if a custom driver is not compliant instead of an ugly stacktrace.
Fixed in last commit Fixed in last commit
|
0
netbox_agent/drivers/__init__.py
Normal file
0
netbox_agent/drivers/__init__.py
Normal file
10
netbox_agent/drivers/datacenter_cmd.py
Normal file
10
netbox_agent/drivers/datacenter_cmd.py
Normal file
|
@ -0,0 +1,10 @@
|
||||||
|
import re
|
||||||
|
import subprocess
|
||||||
|
|
||||||
|
|
||||||
|
def get(value, regex):
|
||||||
|
output = subprocess.getoutput(value)
|
||||||
|
r = re.search(regex, output)
|
||||||
|
if r and len(r.groups()) > 0:
|
||||||
|
return r.groups()[0]
|
||||||
|
return None
|
9
netbox_agent/drivers/datacenter_file.py
Normal file
9
netbox_agent/drivers/datacenter_file.py
Normal file
|
@ -0,0 +1,9 @@
|
||||||
|
import re
|
||||||
|
|
||||||
|
|
||||||
|
def get(value, regex):
|
||||||
|
for line in open(value, 'r'):
|
||||||
|
r = re.search(regex, line)
|
||||||
|
if r and len(r.groups()) > 0:
|
||||||
|
return r.groups()[0]
|
||||||
|
return None
|
|
@ -3,6 +3,7 @@ import re
|
||||||
import socket
|
import socket
|
||||||
|
|
||||||
from netbox_agent.config import netbox_instance as nb
|
from netbox_agent.config import netbox_instance as nb
|
||||||
|
from netbox_agent.datacenter import Datacenter
|
||||||
import netbox_agent.dmidecode as dmidecode
|
import netbox_agent.dmidecode as dmidecode
|
||||||
|
|
||||||
# Regex to match base interface name
|
# Regex to match base interface name
|
||||||
|
@ -21,6 +22,16 @@ class ServerBase():
|
||||||
|
|
||||||
self.network_cards = []
|
self.network_cards = []
|
||||||
|
|
||||||
|
def get_datacenter(self):
|
||||||
|
dc = Datacenter()
|
||||||
|
return dc.get()
|
||||||
|
|
||||||
|
def get_netbox_datacenter(self):
|
||||||
|
datacenter = nb.dcim.sites.get(
|
||||||
|
slug=self.get_datacenter()
|
||||||
|
)
|
||||||
|
return datacenter
|
||||||
|
|
||||||
def get_product_name(self):
|
def get_product_name(self):
|
||||||
"""
|
"""
|
||||||
Return the Chassis Name from dmidecode info
|
Return the Chassis Name from dmidecode info
|
||||||
|
@ -66,7 +77,7 @@ class ServerBase():
|
||||||
nics.append(nic)
|
nics.append(nic)
|
||||||
return nics
|
return nics
|
||||||
|
|
||||||
def _netbox_create_blade_chassis(self):
|
def _netbox_create_blade_chassis(self, datacenter):
|
||||||
device_type = nb.dcim.device_types.get(
|
device_type = nb.dcim.device_types.get(
|
||||||
model=self.get_chassis(),
|
model=self.get_chassis(),
|
||||||
)
|
)
|
||||||
|
@ -75,39 +86,33 @@ class ServerBase():
|
||||||
device_role = nb.dcim.device_roles.get(
|
device_role = nb.dcim.device_roles.get(
|
||||||
name='Server Chassis',
|
name='Server Chassis',
|
||||||
)
|
)
|
||||||
datacenter = nb.dcim.sites.get(
|
|
||||||
name='DC3', # FIXME: datacenter support
|
|
||||||
)
|
|
||||||
new_chassis = nb.dcim.devices.create(
|
new_chassis = nb.dcim.devices.create(
|
||||||
name=''.format(),
|
name=''.format(),
|
||||||
device_type=device_type.id,
|
device_type=device_type.id,
|
||||||
serial=self.get_chassis_service_tag(),
|
serial=self.get_chassis_service_tag(),
|
||||||
device_role=device_role.id,
|
device_role=device_role.id,
|
||||||
site=datacenter.id,
|
site=datacenter.id if datacenter else None,
|
||||||
)
|
)
|
||||||
return new_chassis
|
return new_chassis
|
||||||
|
|
||||||
def _netbox_create_blade(self, chassis):
|
def _netbox_create_blade(self, chassis, datacenter):
|
||||||
device_role = nb.dcim.device_roles.get(
|
device_role = nb.dcim.device_roles.get(
|
||||||
name='Blade',
|
name='Blade',
|
||||||
)
|
)
|
||||||
device_type = nb.dcim.device_types.get(
|
device_type = nb.dcim.device_types.get(
|
||||||
model=self.get_product_name(),
|
model=self.get_product_name(),
|
||||||
)
|
)
|
||||||
datacenter = nb.dcim.sites.get(
|
|
||||||
name='DC3', # FIXME: datacenter support
|
|
||||||
)
|
|
||||||
new_blade = nb.dcim.devices.create(
|
new_blade = nb.dcim.devices.create(
|
||||||
name='{}'.format(socket.gethostname()),
|
name='{}'.format(socket.gethostname()),
|
||||||
serial=self.get_service_tag(),
|
serial=self.get_service_tag(),
|
||||||
device_role=device_role.id,
|
device_role=device_role.id,
|
||||||
device_type=device_type.id,
|
device_type=device_type.id,
|
||||||
parent_device=chassis.id,
|
parent_device=chassis.id,
|
||||||
site=datacenter.id,
|
site=datacenter.id if datacenter else None,
|
||||||
)
|
)
|
||||||
return new_blade
|
return new_blade
|
||||||
|
|
||||||
def _netbox_create_server(self):
|
def _netbox_create_server(self, datacenter):
|
||||||
device_role = nb.dcim.device_roles.get(
|
device_role = nb.dcim.device_roles.get(
|
||||||
name='Server',
|
name='Server',
|
||||||
)
|
)
|
||||||
|
@ -116,19 +121,17 @@ class ServerBase():
|
||||||
)
|
)
|
||||||
if not device_type:
|
if not device_type:
|
||||||
raise Exception('Chassis "{}" doesn\'t exist'.format(self.get_chassis()))
|
raise Exception('Chassis "{}" doesn\'t exist'.format(self.get_chassis()))
|
||||||
datacenter = nb.dcim.sites.get(
|
|
||||||
name='DC3' # FIXME: datacenter support
|
|
||||||
)
|
|
||||||
new_server = nb.dcim.devices.create(
|
new_server = nb.dcim.devices.create(
|
||||||
name='{}'.format(socket.gethostname()),
|
name='{}'.format(socket.gethostname()),
|
||||||
serial=self.get_service_tag(),
|
serial=self.get_service_tag(),
|
||||||
device_role=device_role.id,
|
device_role=device_role.id,
|
||||||
device_type=device_type.id,
|
device_type=device_type.id,
|
||||||
site=datacenter.id,
|
site=datacenter.id if datacenter else None,
|
||||||
)
|
)
|
||||||
return new_server
|
return new_server
|
||||||
|
|
||||||
def netbox_create(self):
|
def netbox_create(self):
|
||||||
|
datacenter = self.get_netbox_datacenter()
|
||||||
if self.is_blade():
|
if self.is_blade():
|
||||||
# let's find the blade
|
# let's find the blade
|
||||||
blade = nb.dcim.devices.get(serial=self.get_service_tag())
|
blade = nb.dcim.devices.get(serial=self.get_service_tag())
|
||||||
|
@ -138,9 +141,9 @@ class ServerBase():
|
||||||
# check if the chassis exist before
|
# check if the chassis exist before
|
||||||
# if it doesn't exist, create it
|
# if it doesn't exist, create it
|
||||||
if not chassis:
|
if not chassis:
|
||||||
chassis = self._netbox_create_blade_chassis()
|
chassis = self._netbox_create_blade_chassis(datacenter)
|
||||||
|
|
||||||
blade = self._netbox_create_blade(chassis)
|
blade = self._netbox_create_blade(chassis, datacenter)
|
||||||
|
|
||||||
# Find the slot and update it with our blade
|
# Find the slot and update it with our blade
|
||||||
device_bays = nb.dcim.device_bays.filter(
|
device_bays = nb.dcim.device_bays.filter(
|
||||||
|
@ -152,7 +155,6 @@ class ServerBase():
|
||||||
device_bay.installed_device = blade
|
device_bay.installed_device = blade
|
||||||
device_bay.save()
|
device_bay.save()
|
||||||
else:
|
else:
|
||||||
# FIXME : handle pizza box
|
|
||||||
server = nb.dcim.devices.get(serial=self.get_service_tag())
|
server = nb.dcim.devices.get(serial=self.get_service_tag())
|
||||||
if not server:
|
if not server:
|
||||||
self._netbox_create_server()
|
self._netbox_create_server()
|
||||||
|
|
Loading…
Reference in a new issue
Probably need some sanity check as the drivers (for now) require the regex to have a
datacenter
named group.I think there is 3 solutions :
datacenter
named group (not a big fan)