diff options
Diffstat (limited to 'ansible/netbox/sync-unifi.py')
-rw-r--r-- | ansible/netbox/sync-unifi.py | 160 |
1 files changed, 130 insertions, 30 deletions
diff --git a/ansible/netbox/sync-unifi.py b/ansible/netbox/sync-unifi.py index 620e48d..7995866 100644 --- a/ansible/netbox/sync-unifi.py +++ b/ansible/netbox/sync-unifi.py @@ -2,6 +2,9 @@ import os import sys from unifi_controller_api import UnifiController from unifi_controller_api.exceptions import UnifiAuthenticationError, UnifiAPIError +from pprint import pprint + +import pynetbox class Db(): def __init__(self): @@ -9,12 +12,86 @@ class Db(): self.interfaces = [] self.ips = [] -def controller_login() -> UnifiController: +class NetboxCache(): + def __init__(self, nb): + self.nb = nb + self.device_roles = {} + self.device_types = {} + + def get_device_role(self, slug): + dt = self.device_roles.get(slug) + + if dt is not None: + return dt + + dt = self.nb.dcim.device_roles.get(slug=slug) + + if dt is None: + raise Exception(f"No such device type: {slug}") + + self.device_roles[slug] = dt + return dt + + def get_device_type(self, slug): + dt = self.device_types.get(slug) + + if dt is not None: + return dt + + dt = self.nb.dcim.device_types.get(slug=slug) + + if dt is None: + raise Exception(f"No such device type: {slug}") + + self.device_types[slug] = dt + return dt + +def process_switch(d, db: Db, nb: NetboxCache, site, vrf): + db.devices.append({ + "name": d.name, + "device_type": nb.get_device_type("ubiquiti-us-8-150w").id, + "role": nb.get_device_role("switch").id, + "mac": d.mac, + "serial": d.serial, + "site": site.id, + }) + +def main(): + unifi_url=os.getenv("UNIFI_URL") + unifi_username=os.getenv("UNIFI_USERNAME") + unifi_password=os.getenv("UNIFI_PASSWORD") + unifi_site=os.getenv("UNIFI_SITE") + + netbox_url=os.getenv("NETBOX_URL") + netbox_token=os.getenv("NETBOX_TOKEN") + netbox_vrf_name=os.getenv("NETBOX_VRF") + netbox_site_name=os.getenv("NETBOX_SITE") + + controller = controller_login(unifi_url, unifi_username, unifi_password) + + (nb, netbox_site, netbox_vrf) = netbox_login(netbox_url, netbox_token, netbox_site_name, netbox_vrf_name) + status = nb.status() + print(f"NetBox status: {status}") + + devices = collect_devices(controller, unifi_site) + + nb_cache = NetboxCache(nb) + db = Db() + for d in devices: + pprint(d) + if d.model == "US8P150": + process_switch(d, db, nb_cache, netbox_site, netbox_vrf) + + print_db(db) + + sync_db(nb, db) + +def controller_login(url, username, password) -> UnifiController: # try: controller = UnifiController( - controller_url=os.getenv("UNIFI_URL"), - username=os.getenv("UNIFI_USERNAME"), - password=os.getenv("UNIFI_PASSWORD"), + controller_url=url, + username=username, + password=password, is_udm_pro=False, verify_ssl=True, ) @@ -30,16 +107,9 @@ def controller_login() -> UnifiController: # except Exception as e: # print(f"An unexpected error occurred: {e}") -site_name=os.getenv("UNIFI_SITE") -vrf=os.getenv("VRF") - -def collect_devices(controller: UnifiController) -> Db: - db = Db() - - devices = [] - +def collect_devices(controller: UnifiController, site_name: str) -> list[any]: try: - devices = controller.get_unifi_site_device(site_name=site_name, detailed=True, raw=False) + return controller.get_unifi_site_device(site_name=site_name, detailed=True, raw=False) except UnifiAPIError as e: print(f"Error fetching device information: {e}") except Exception as e: @@ -49,19 +119,20 @@ def collect_devices(controller: UnifiController) -> Db: for d in devices: print(f"{d.name}", file=sys.stderr) print(f"{d}", file=sys.stderr) - db.devices.append({ - "name": d.name, - "mac": d.mac, - "serial": d.serial, - }) - db.ips.append({ - "address": d.ip, - "status": "active", - "vrf": vrf, - "device": d.name, - "interface": "", - "is_primary": "true", - }) + db.devices = devices +# db.devices.append({ +# "name": d.name, +# "mac": d.mac, +# "serial": d.serial, +# }) +# db.ips.append({ +# "address": d.ip, +# "status": "active", +# "vrf": None, +# "device": d.name, +# "interface": "", +# "is_primary": "true", +# }) # db.interfaces.append({ # }) @@ -78,11 +149,40 @@ def print_db(db: Db): p(sorted(db.interfaces, key=lambda d: d["address"])) p(sorted(db.ips, key=lambda d: d["address"])) +def sync_db(nb, db: Db): + print("Synchronizing") + for d in db.devices: + pprint(d) + remote = nb.dcim.devices.get(name = d["name"]) + if remote is None: + print(f"Creating new for {d['name']}") + remote = nb.dcim.devices.create(d) + pprint(remote.serialize()) + else: + pprint(remote.serialize()) + remote.update(d) -def main(): - controller = controller_login() - db = collect_devices(controller) - print_db(db) +def netbox_login(url: str, token: str, site_name: str, vrf_name: str) -> pynetbox.core.api.Api: + nb = pynetbox.api(url, token=token) + + site = nb.dcim.sites.get(name=site_name) + if site is None: + site = nb.dcim.sites.get(slug=site_name) + if site is None: + print(f"Could not look up site by name or slug: {site_name}", file=sys.stderr) + exit(1) + print(f"NetBox site {site.name}", file=sys.stderr) + + vrf = None + vrf_id = None + if vrf_name is not None: + vrf = nb.ipam.vrfs.get(site=site, name=vrf_name) + if vrf is None: + print(f"Could not look up VRF by slug: {vrf_name}", file=sys.stderr) + exit(1) + vrf_id = vrf.id + + return nb, site, vrf if __name__ == "__main__": main() |