176 lines
6.4 KiB
Python
176 lines
6.4 KiB
Python
from urllib.parse import quote_plus
|
|
|
|
from dcim.models import Device
|
|
from django.contrib import messages
|
|
from django.core.cache import cache
|
|
from django.db import transaction
|
|
from django.http import Http404
|
|
from django.shortcuts import get_object_or_404, redirect
|
|
from django.urls import reverse
|
|
from django.views import View
|
|
from ipam.models import VLAN, VLANGroup
|
|
|
|
from netbox_librenms_plugin.views.mixins import (
|
|
CacheMixin,
|
|
LibreNMSAPIMixin,
|
|
LibreNMSPermissionMixin,
|
|
NetBoxObjectPermissionMixin,
|
|
)
|
|
|
|
|
|
class SyncVLANsView(LibreNMSPermissionMixin, NetBoxObjectPermissionMixin, LibreNMSAPIMixin, CacheMixin, View):
|
|
"""
|
|
Handle POST requests to create/update VLANs in NetBox from LibreNMS data.
|
|
"""
|
|
|
|
required_object_permissions = {
|
|
"POST": [
|
|
("add", VLAN),
|
|
("change", VLAN),
|
|
],
|
|
}
|
|
|
|
def post(self, request, object_type: str, object_id: int):
|
|
"""
|
|
Process sync request.
|
|
|
|
Expected POST data:
|
|
- action: 'create_vlans'
|
|
- select: List of VLAN IDs to create
|
|
- vlan_group_{vid}: Per-row VLAN group selection
|
|
"""
|
|
# Check both plugin write and NetBox object permissions
|
|
if error := self.require_all_permissions("POST"):
|
|
return error
|
|
|
|
# Read server_key from POST so we use the exact server the user was viewing
|
|
self._post_server_key = request.POST.get("server_key") or self.librenms_api.server_key
|
|
|
|
obj = self.get_object(object_type, object_id)
|
|
action = request.POST.get("action", "")
|
|
|
|
if action == "create_vlans":
|
|
return self._handle_create_vlans(request, obj, object_type, object_id)
|
|
else:
|
|
messages.error(request, "Invalid action specified.")
|
|
return self._redirect(object_type, object_id)
|
|
|
|
def get_object(self, object_type: str, object_id: int):
|
|
"""Get the target object (Device or VM)."""
|
|
if object_type == "device":
|
|
return get_object_or_404(Device, pk=object_id)
|
|
raise Http404("Invalid object type.")
|
|
|
|
def _redirect(self, object_type: str, object_id: int):
|
|
"""Redirect back to sync page with VLAN tab active."""
|
|
url_name = (
|
|
"dcim:device_librenms_sync"
|
|
if object_type == "device"
|
|
else "plugins:netbox_librenms_plugin:vm_librenms_sync"
|
|
)
|
|
server_key = getattr(self, "_post_server_key", None) or self.librenms_api.server_key
|
|
url = reverse(url_name, kwargs={"pk": object_id}) + "?tab=vlans"
|
|
if server_key:
|
|
url += f"&server_key={quote_plus(server_key)}"
|
|
return redirect(url)
|
|
|
|
def _handle_create_vlans(self, request, obj, object_type, object_id):
|
|
"""
|
|
Handle creating selected VLANs in NetBox.
|
|
|
|
Reads per-row VLAN group selections from form fields named 'vlan_group_{vid}'.
|
|
"""
|
|
selected_vlans = request.POST.getlist("select")
|
|
|
|
if not selected_vlans:
|
|
messages.error(request, "No VLANs selected for creation.")
|
|
return self._redirect(object_type, object_id)
|
|
|
|
# Get cached VLAN data
|
|
cached_vlans = cache.get(self.get_cache_key(obj, "vlans", self._post_server_key))
|
|
if not cached_vlans:
|
|
messages.error(request, "No cached VLAN data. Please refresh VLANs first.")
|
|
return self._redirect(object_type, object_id)
|
|
|
|
# Build lookup of LibreNMS VLANs by VID
|
|
librenms_vlans = {str(v["vlan_vlan"]): v for v in cached_vlans}
|
|
|
|
created_count = 0
|
|
updated_count = 0
|
|
skipped_count = 0
|
|
|
|
with transaction.atomic():
|
|
for vid_str in selected_vlans:
|
|
try:
|
|
vid = int(vid_str)
|
|
except ValueError:
|
|
continue
|
|
|
|
vlan_data = librenms_vlans.get(vid_str)
|
|
if not vlan_data:
|
|
continue
|
|
|
|
# Get per-row VLAN group selection
|
|
group_id_str = request.POST.get(f"vlan_group_{vid}", "")
|
|
row_vlan_group = None
|
|
if group_id_str:
|
|
try:
|
|
row_vlan_group = VLANGroup.objects.get(pk=int(group_id_str))
|
|
except (ValueError, VLANGroup.DoesNotExist):
|
|
pass # Fall back to global VLAN (no group)
|
|
|
|
librenms_name = vlan_data.get("vlan_name", f"VLAN {vid}")
|
|
|
|
if row_vlan_group:
|
|
# Grouped VLAN: match by VID (unique constraint within group)
|
|
vlan, created = VLAN.objects.get_or_create(
|
|
vid=vid,
|
|
group=row_vlan_group,
|
|
defaults={
|
|
"name": librenms_name,
|
|
"status": "active",
|
|
},
|
|
)
|
|
if created:
|
|
created_count += 1
|
|
elif vlan.name != librenms_name:
|
|
vlan.name = librenms_name
|
|
vlan.save()
|
|
updated_count += 1
|
|
else:
|
|
skipped_count += 1
|
|
else:
|
|
# Global VLAN: match by VID only (unique constraint with group=NULL)
|
|
vlan, created = VLAN.objects.get_or_create(
|
|
vid=vid,
|
|
group=None,
|
|
defaults={
|
|
"name": librenms_name,
|
|
"status": "active",
|
|
},
|
|
)
|
|
if created:
|
|
created_count += 1
|
|
elif vlan.name != librenms_name:
|
|
vlan.name = librenms_name
|
|
vlan.save()
|
|
updated_count += 1
|
|
else:
|
|
skipped_count += 1
|
|
|
|
# Build summary message
|
|
parts = []
|
|
if created_count > 0:
|
|
parts.append(f"{created_count} created")
|
|
if updated_count > 0:
|
|
parts.append(f"{updated_count} updated")
|
|
if skipped_count > 0:
|
|
parts.append(f"{skipped_count} unchanged")
|
|
|
|
if parts:
|
|
messages.success(request, f"VLANs synced: {', '.join(parts)}.")
|
|
else:
|
|
messages.warning(request, "No VLANs were created or updated.")
|
|
|
|
return self._redirect(object_type, object_id)
|