netbox-agent/netbox_agent/server.py

306 lines
10 KiB
Python
Raw Normal View History

import logging
2019-08-04 15:14:36 +02:00
import socket
import subprocess
2020-02-02 20:08:56 +01:00
from pprint import pprint
import netbox_agent.dmidecode as dmidecode
2020-02-02 20:08:56 +01:00
from netbox_agent.config import config
from netbox_agent.config import netbox_instance as nb
from netbox_agent.inventory import Inventory
2020-02-02 20:08:56 +01:00
from netbox_agent.location import Datacenter, Rack
from netbox_agent.network import ServerNetwork
2019-09-05 13:47:10 +02:00
from netbox_agent.power import PowerSupply
2019-08-04 15:14:36 +02:00
def get_device_role(role):
device_role = nb.dcim.device_roles.get(
name=role
)
if device_role is None:
raise Exception('DeviceRole "{}" does not exist, please create it'.format(role))
return device_role
def get_device_type(type):
device_type = nb.dcim.device_types.get(
model=type
)
if device_type is None:
raise Exception('DeviceType "{}" does not exist, please create it'.format(type))
return device_type
class ServerBase():
def __init__(self, dmi=None):
if dmi:
self.dmi = dmi
else:
self.dmi = dmidecode.parse()
2020-02-02 20:08:56 +01:00
self.baseboard = dmidecode.get_by_type(self.dmi, 'Baseboard')
self.bios = dmidecode.get_by_type(self.dmi, 'BIOS')
self.chassis = dmidecode.get_by_type(self.dmi, 'Chassis')
self.system = dmidecode.get_by_type(self.dmi, 'System')
self.network = None
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
2019-09-03 13:16:37 +02:00
def update_netbox_location(self, server):
dc = self.get_datacenter()
rack = self.get_rack()
nb_rack = self.get_netbox_rack()
nb_dc = self.get_netbox_datacenter()
update = False
if dc and server.site and server.site.slug != nb_dc.slug:
2019-09-03 13:16:37 +02:00
logging.info('Datacenter location has changed from {} to {}, updating'.format(
server.site.slug,
nb_dc.slug,
))
update = True
server.site = nb_dc.id
if rack and server.rack and server.rack.id != nb_rack.id:
2019-09-03 13:16:37 +02:00
logging.info('Rack location has changed from {} to {}, updating'.format(
server.rack,
nb_rack,
))
update = True
server.rack = nb_rack
if nb_rack is None:
server.face = None
server.position = None
return update, server
2019-08-05 11:54:06 +02:00
def get_rack(self):
rack = Rack()
return rack.get()
def get_netbox_rack(self):
rack = nb.dcim.racks.get(
name=self.get_rack(),
2019-09-11 17:35:27 +02:00
site_id=self.get_netbox_datacenter().id,
2019-08-05 11:54:06 +02:00
)
return rack
def get_product_name(self):
"""
Return the Chassis Name from dmidecode info
"""
2019-09-10 14:46:04 +02:00
return self.system[0]['Product Name'].strip()
def get_service_tag(self):
"""
Return the Service Tag from dmidecode info
"""
return self.system[0]['Serial Number'].strip()
def get_hostname(self):
if config.hostname_cmd is None:
return '{}'.format(socket.gethostname())
return subprocess.getoutput(config.hostname_cmd)
def is_blade(self):
raise NotImplementedError
def get_blade_slot(self):
raise NotImplementedError
def get_chassis(self):
raise NotImplementedError
2019-08-08 14:37:02 +02:00
def get_chassis_name(self):
raise NotImplementedError
def get_chassis_service_tag(self):
raise NotImplementedError
def get_bios_version(self):
raise NotImplementedError
def get_bios_version_attr(self):
raise NotImplementedError
def get_bios_release_date(self):
raise NotImplementedError
2019-09-05 13:47:10 +02:00
def get_power_consumption(self):
raise NotImplementedError
def _netbox_create_chassis(self, datacenter, rack):
device_type = get_device_type(self.get_chassis())
device_role = get_device_role('Server Chassis')
2019-08-06 17:59:46 +02:00
serial = self.get_chassis_service_tag()
logging.info('Creating chassis blade (serial: {serial})'.format(
2019-08-06 17:59:46 +02:00
serial=serial))
new_chassis = nb.dcim.devices.create(
2019-08-08 14:37:02 +02:00
name=self.get_chassis_name(),
device_type=device_type.id,
2019-08-06 17:59:46 +02:00
serial=serial,
device_role=device_role.id,
site=datacenter.id if datacenter else None,
rack=rack.id if rack else None,
)
return new_chassis
def _netbox_create_blade(self, chassis, datacenter, rack):
device_role = get_device_role('Blade')
device_type = get_device_type(self.get_product_name())
2019-08-06 17:59:46 +02:00
serial = self.get_service_tag()
hostname = self.get_hostname()
logging.info(
2019-08-06 17:59:46 +02:00
'Creating blade (serial: {serial}) {hostname} on chassis {chassis_serial}'.format(
serial=serial, hostname=hostname, chassis_serial=chassis.serial
))
new_blade = nb.dcim.devices.create(
2019-08-06 17:59:46 +02:00
name=hostname,
serial=serial,
device_role=device_role.id,
device_type=device_type.id,
parent_device=chassis.id,
site=datacenter.id if datacenter else None,
rack=rack.id if rack else None,
)
return new_blade
def _netbox_create_server(self, datacenter, rack):
device_role = get_device_role('Server')
device_type = get_device_type(self.get_product_name())
2019-08-03 15:57:06 +02:00
if not device_type:
raise Exception('Chassis "{}" doesn\'t exist'.format(self.get_chassis()))
2019-08-06 17:59:46 +02:00
serial = self.get_service_tag()
hostname = self.get_hostname()
logging.info('Creating server (serial: {serial}) {hostname}'.format(
2019-08-06 17:59:46 +02:00
serial=serial, hostname=hostname))
2019-08-03 15:57:06 +02:00
new_server = nb.dcim.devices.create(
2019-08-06 17:59:46 +02:00
name=hostname,
serial=serial,
2019-08-03 15:57:06 +02:00
device_role=device_role.id,
device_type=device_type.id,
site=datacenter.id if datacenter else None,
rack=rack.id if rack else None,
2019-08-03 15:57:06 +02:00
)
return new_server
2019-08-04 15:14:36 +02:00
def get_netbox_server(self):
return nb.dcim.devices.get(serial=self.get_service_tag())
def _netbox_set_or_update_blade_slot(self, server, chassis, datacenter):
# before everything check if right chassis
actual_device_bay = server.parent_device.device_bay if server.parent_device else None
actual_chassis = actual_device_bay.device if actual_device_bay else None
slot = self.get_blade_slot()
if actual_chassis and \
actual_chassis.serial == chassis.serial and \
actual_device_bay.name == slot:
return
real_device_bays = nb.dcim.device_bays.filter(
device_id=chassis.id,
name=slot,
)
if len(real_device_bays) > 0:
logging.info(
'Setting device ({serial}) new slot on {slot} '
'(Chassis {chassis_serial})..'.format(
serial=server.serial, slot=slot, chassis_serial=chassis.serial
))
# reset actual device bay if set
if actual_device_bay:
actual_device_bay.installed_device = None
actual_device_bay.save()
# setup new device bay
real_device_bay = real_device_bays[0]
real_device_bay.installed_device = server
real_device_bay.save()
else:
logging.error('Could not find slot {slot} for chassis'.format(
slot=slot
))
def netbox_create_or_update(self, config):
"""
Netbox method to update info about our server/blade
Handle:
* new chasis for a blade
* new slot for a bblade
* hostname update
* new network infos
"""
datacenter = self.get_netbox_datacenter()
rack = self.get_netbox_rack()
2019-09-03 13:16:37 +02:00
2019-08-05 12:02:42 +02:00
if self.is_blade():
chassis = nb.dcim.devices.get(
serial=self.get_chassis_service_tag()
)
# Chassis does not exist
if not chassis:
chassis = self._netbox_create_chassis(datacenter, rack)
server = nb.dcim.devices.get(serial=self.get_service_tag())
if not server:
server = self._netbox_create_blade(chassis, datacenter, rack)
2019-09-03 13:16:37 +02:00
# Set slot for blade
self._netbox_set_or_update_blade_slot(server, chassis, datacenter)
else:
server = nb.dcim.devices.get(serial=self.get_service_tag())
if not server:
self._netbox_create_server(datacenter, rack)
2019-09-03 13:16:37 +02:00
logging.debug('Updating Server...')
2019-08-05 12:02:42 +02:00
# check network cards
if config.register or config.update_all or config.update_network:
2020-04-12 20:43:25 +02:00
self.network = ServerNetwork(server=self)
2020-04-13 00:39:38 +02:00
self.network.create_or_update_netbox_network_cards()
# update inventory if feature is enabled
if config.inventory and (config.register or config.update_all or config.update_inventory):
2019-09-03 13:16:37 +02:00
self.inventory = Inventory(server=self)
self.inventory.create_or_update()
2019-09-05 13:47:10 +02:00
# update psu
if config.register or config.update_all or config.update_psu:
2019-09-05 13:47:10 +02:00
self.power = PowerSupply(server=self)
self.power.create_or_update_power_supply()
self.power.report_power_consumption()
update = 0
# for every other specs
# check hostname
if server.name != self.get_hostname():
update += 1
server.name = self.get_hostname()
if config.update_all or config.update_location:
ret, server = self.update_netbox_location(server)
update += ret
if update:
server.save()
logging.debug('Finished updating Server!')
2019-08-04 15:14:36 +02:00
def print_debug(self):
2020-04-12 20:43:25 +02:00
self.network = ServerNetwork(server=self)
print('Datacenter:', self.get_datacenter())
print('Netbox Datacenter:', self.get_netbox_datacenter())
2019-08-05 11:54:06 +02:00
print('Rack:', self.get_rack())
print('Netbox Rack:', self.get_netbox_rack())
print('Is blade:', self.is_blade())
print('Product Name:', self.get_product_name())
print('Chassis:', self.get_chassis())
print('Chassis service tag:', self.get_chassis_service_tag())
print('Service tag:', self.get_service_tag())
2019-08-04 14:37:51 +02:00
print('NIC:',)
pprint(self.network.get_network_cards())
pass