signals.py 4.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139
  1. from django.conf import settings
  2. from django.contrib.contenttypes.models import ContentType
  3. from django.db.models.signals import m2m_changed, post_save, pre_delete
  4. from django.dispatch import receiver
  5. from django_prometheus.models import model_deletes, model_inserts, model_updates
  6. from prometheus_client import Counter
  7. from netbox.signals import post_clean
  8. from .choices import ObjectChangeActionChoices
  9. from .models import CustomField, ObjectChange
  10. from .webhooks import enqueue_object, get_snapshots, serialize_for_webhook
  11. #
  12. # Change logging/webhooks
  13. #
  14. def _handle_changed_object(request, webhook_queue, sender, instance, **kwargs):
  15. """
  16. Fires when an object is created or updated.
  17. """
  18. def is_same_object(instance, webhook_data):
  19. return (
  20. ContentType.objects.get_for_model(instance) == webhook_data['content_type'] and
  21. instance.pk == webhook_data['object_id'] and
  22. request.id == webhook_data['request_id']
  23. )
  24. if not hasattr(instance, 'to_objectchange'):
  25. return
  26. m2m_changed = False
  27. # Determine the type of change being made
  28. if kwargs.get('created'):
  29. action = ObjectChangeActionChoices.ACTION_CREATE
  30. elif 'created' in kwargs:
  31. action = ObjectChangeActionChoices.ACTION_UPDATE
  32. elif kwargs.get('action') in ['post_add', 'post_remove'] and kwargs['pk_set']:
  33. # m2m_changed with objects added or removed
  34. m2m_changed = True
  35. action = ObjectChangeActionChoices.ACTION_UPDATE
  36. else:
  37. return
  38. # Record an ObjectChange if applicable
  39. if hasattr(instance, 'to_objectchange'):
  40. if m2m_changed:
  41. ObjectChange.objects.filter(
  42. changed_object_type=ContentType.objects.get_for_model(instance),
  43. changed_object_id=instance.pk,
  44. request_id=request.id
  45. ).update(
  46. postchange_data=instance.to_objectchange(action).postchange_data
  47. )
  48. else:
  49. objectchange = instance.to_objectchange(action)
  50. objectchange.user = request.user
  51. objectchange.request_id = request.id
  52. objectchange.save()
  53. # If this is an M2M change, update the previously queued webhook (from post_save)
  54. if m2m_changed and webhook_queue and is_same_object(instance, webhook_queue[-1]):
  55. instance.refresh_from_db() # Ensure that we're working with fresh M2M assignments
  56. webhook_queue[-1]['data'] = serialize_for_webhook(instance)
  57. webhook_queue[-1]['snapshots']['postchange'] = get_snapshots(instance, action)['postchange']
  58. else:
  59. enqueue_object(webhook_queue, instance, request.user, request.id, action)
  60. # Increment metric counters
  61. if action == ObjectChangeActionChoices.ACTION_CREATE:
  62. model_inserts.labels(instance._meta.model_name).inc()
  63. elif action == ObjectChangeActionChoices.ACTION_UPDATE:
  64. model_updates.labels(instance._meta.model_name).inc()
  65. def _handle_deleted_object(request, webhook_queue, sender, instance, **kwargs):
  66. """
  67. Fires when an object is deleted.
  68. """
  69. if not hasattr(instance, 'to_objectchange'):
  70. return
  71. # Record an ObjectChange if applicable
  72. if hasattr(instance, 'to_objectchange'):
  73. objectchange = instance.to_objectchange(ObjectChangeActionChoices.ACTION_DELETE)
  74. objectchange.user = request.user
  75. objectchange.request_id = request.id
  76. objectchange.save()
  77. # Enqueue webhooks
  78. enqueue_object(webhook_queue, instance, request.user, request.id, ObjectChangeActionChoices.ACTION_DELETE)
  79. # Increment metric counters
  80. model_deletes.labels(instance._meta.model_name).inc()
  81. #
  82. # Custom fields
  83. #
  84. def handle_cf_removed_obj_types(instance, action, pk_set, **kwargs):
  85. """
  86. Handle the cleanup of old custom field data when a CustomField is removed from one or more ContentTypes.
  87. """
  88. if action == 'post_remove':
  89. instance.remove_stale_data(ContentType.objects.filter(pk__in=pk_set))
  90. def handle_cf_renamed(instance, created, **kwargs):
  91. """
  92. Handle the renaming of custom field data on objects when a CustomField is renamed.
  93. """
  94. if not created and instance.name != instance._name:
  95. instance.rename_object_data(old_name=instance._name, new_name=instance.name)
  96. def handle_cf_deleted(instance, **kwargs):
  97. """
  98. Handle the cleanup of old custom field data when a CustomField is deleted.
  99. """
  100. instance.remove_stale_data(instance.content_types.all())
  101. m2m_changed.connect(handle_cf_removed_obj_types, sender=CustomField.content_types.through)
  102. post_save.connect(handle_cf_renamed, sender=CustomField)
  103. pre_delete.connect(handle_cf_deleted, sender=CustomField)
  104. #
  105. # Custom validation
  106. #
  107. @receiver(post_clean)
  108. def run_custom_validators(sender, instance, **kwargs):
  109. model_name = f'{sender._meta.app_label}.{sender._meta.model_name}'
  110. validators = settings.CUSTOM_VALIDATORS.get(model_name, [])
  111. for validator in validators:
  112. validator(instance)