signals.py 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101
  1. import logging
  2. from django.db.models.signals import post_save, pre_delete
  3. from django.dispatch import receiver
  4. from .choices import CableStatusChoices
  5. from .models import Cable, Device, VirtualChassis
  6. @receiver(post_save, sender=VirtualChassis)
  7. def assign_virtualchassis_master(instance, created, **kwargs):
  8. """
  9. When a VirtualChassis is created, automatically assign its master device to the VC.
  10. """
  11. if created:
  12. devices = Device.objects.filter(pk=instance.master.pk)
  13. for device in devices:
  14. device.virtual_chassis = instance
  15. device.vc_position = None
  16. device.save()
  17. @receiver(pre_delete, sender=VirtualChassis)
  18. def clear_virtualchassis_members(instance, **kwargs):
  19. """
  20. When a VirtualChassis is deleted, nullify the vc_position and vc_priority fields of its prior members.
  21. """
  22. devices = Device.objects.filter(virtual_chassis=instance.pk)
  23. for device in devices:
  24. device.vc_position = None
  25. device.vc_priority = None
  26. device.save()
  27. @receiver(post_save, sender=Cable)
  28. def update_connected_endpoints(instance, **kwargs):
  29. """
  30. When a Cable is saved, check for and update its two connected endpoints
  31. """
  32. logger = logging.getLogger('netbox.dcim.cable')
  33. # Cache the Cable on its two termination points
  34. if instance.termination_a.cable != instance:
  35. logger.debug("Updating termination A for cable {}".format(instance))
  36. instance.termination_a.cable = instance
  37. instance.termination_a.save()
  38. if instance.termination_b.cable != instance:
  39. logger.debug("Updating termination B for cable {}".format(instance))
  40. instance.termination_b.cable = instance
  41. instance.termination_b.save()
  42. # Update any endpoints for this Cable.
  43. endpoints = instance.termination_a.get_path_endpoints() + instance.termination_b.get_path_endpoints()
  44. for endpoint in endpoints:
  45. path, split_ends = endpoint.trace()
  46. # Determine overall path status (connected or planned)
  47. path_status = True
  48. for segment in path:
  49. if segment[1] is None or segment[1].status != CableStatusChoices.STATUS_CONNECTED:
  50. path_status = False
  51. break
  52. endpoint_a = path[0][0]
  53. endpoint_b = path[-1][2]
  54. if getattr(endpoint_a, 'is_path_endpoint', False) and getattr(endpoint_b, 'is_path_endpoint', False):
  55. logger.debug("Updating path endpoints: {} <---> {}".format(endpoint_a, endpoint_b))
  56. endpoint_a.connected_endpoint = endpoint_b
  57. endpoint_a.connection_status = path_status
  58. endpoint_a.save()
  59. endpoint_b.connected_endpoint = endpoint_a
  60. endpoint_b.connection_status = path_status
  61. endpoint_b.save()
  62. @receiver(pre_delete, sender=Cable)
  63. def nullify_connected_endpoints(instance, **kwargs):
  64. """
  65. When a Cable is deleted, check for and update its two connected endpoints
  66. """
  67. logger = logging.getLogger('netbox.dcim.cable')
  68. endpoints = instance.termination_a.get_path_endpoints() + instance.termination_b.get_path_endpoints()
  69. # Disassociate the Cable from its termination points
  70. if instance.termination_a is not None:
  71. logger.debug("Nullifying termination A for cable {}".format(instance))
  72. instance.termination_a.cable = None
  73. instance.termination_a.save()
  74. if instance.termination_b is not None:
  75. logger.debug("Nullifying termination B for cable {}".format(instance))
  76. instance.termination_b.cable = None
  77. instance.termination_b.save()
  78. # If this Cable was part of any complete end-to-end paths, tear them down.
  79. for endpoint in endpoints:
  80. logger.debug(f"Removing path information for {endpoint}")
  81. if hasattr(endpoint, 'connected_endpoint'):
  82. endpoint.connected_endpoint = None
  83. endpoint.connection_status = None
  84. endpoint.save()