signals.py 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142
  1. import random
  2. from datetime import timedelta
  3. from cacheops.signals import cache_invalidated, cache_read
  4. from django.conf import settings
  5. from django.contrib.contenttypes.models import ContentType
  6. from django.db import DEFAULT_DB_ALIAS
  7. from django.db.models.signals import m2m_changed, post_save, pre_delete
  8. from django.utils import timezone
  9. from django_prometheus.models import model_deletes, model_inserts, model_updates
  10. from prometheus_client import Counter
  11. from .choices import ObjectChangeActionChoices
  12. from .models import CustomField, ObjectChange
  13. from .webhooks import enqueue_webhooks
  14. #
  15. # Change logging/webhooks
  16. #
  17. def _handle_changed_object(request, sender, instance, **kwargs):
  18. """
  19. Fires when an object is created or updated.
  20. """
  21. m2m_changed = False
  22. # Determine the type of change being made
  23. if kwargs.get('created'):
  24. action = ObjectChangeActionChoices.ACTION_CREATE
  25. elif 'created' in kwargs:
  26. action = ObjectChangeActionChoices.ACTION_UPDATE
  27. elif kwargs.get('action') in ['post_add', 'post_remove'] and kwargs['pk_set']:
  28. # m2m_changed with objects added or removed
  29. m2m_changed = True
  30. action = ObjectChangeActionChoices.ACTION_UPDATE
  31. else:
  32. return
  33. # Record an ObjectChange if applicable
  34. if hasattr(instance, 'to_objectchange'):
  35. if m2m_changed:
  36. ObjectChange.objects.filter(
  37. changed_object_type=ContentType.objects.get_for_model(instance),
  38. changed_object_id=instance.pk,
  39. request_id=request.id
  40. ).update(
  41. postchange_data=instance.to_objectchange(action).postchange_data
  42. )
  43. else:
  44. objectchange = instance.to_objectchange(action)
  45. objectchange.user = request.user
  46. objectchange.request_id = request.id
  47. objectchange.save()
  48. # Enqueue webhooks
  49. enqueue_webhooks(instance, request.user, request.id, action)
  50. # Increment metric counters
  51. if action == ObjectChangeActionChoices.ACTION_CREATE:
  52. model_inserts.labels(instance._meta.model_name).inc()
  53. elif action == ObjectChangeActionChoices.ACTION_UPDATE:
  54. model_updates.labels(instance._meta.model_name).inc()
  55. # Housekeeping: 0.1% chance of clearing out expired ObjectChanges
  56. if settings.CHANGELOG_RETENTION and random.randint(1, 1000) == 1:
  57. cutoff = timezone.now() - timedelta(days=settings.CHANGELOG_RETENTION)
  58. ObjectChange.objects.filter(time__lt=cutoff)._raw_delete(using=DEFAULT_DB_ALIAS)
  59. def _handle_deleted_object(request, sender, instance, **kwargs):
  60. """
  61. Fires when an object is deleted.
  62. """
  63. # Record an ObjectChange if applicable
  64. if hasattr(instance, 'to_objectchange'):
  65. objectchange = instance.to_objectchange(ObjectChangeActionChoices.ACTION_DELETE)
  66. objectchange.user = request.user
  67. objectchange.request_id = request.id
  68. objectchange.save()
  69. # Enqueue webhooks
  70. enqueue_webhooks(instance, request.user, request.id, ObjectChangeActionChoices.ACTION_DELETE)
  71. # Increment metric counters
  72. model_deletes.labels(instance._meta.model_name).inc()
  73. #
  74. # Custom fields
  75. #
  76. def handle_cf_removed_obj_types(instance, action, pk_set, **kwargs):
  77. """
  78. Handle the cleanup of old custom field data when a CustomField is removed from one or more ContentTypes.
  79. """
  80. if action == 'post_remove':
  81. instance.remove_stale_data(ContentType.objects.filter(pk__in=pk_set))
  82. def handle_cf_renamed(instance, created, **kwargs):
  83. """
  84. Handle the renaming of custom field data on objects when a CustomField is renamed.
  85. """
  86. if not created and instance.name != instance._name:
  87. instance.rename_object_data(old_name=instance._name, new_name=instance.name)
  88. def handle_cf_deleted(instance, **kwargs):
  89. """
  90. Handle the cleanup of old custom field data when a CustomField is deleted.
  91. """
  92. instance.remove_stale_data(instance.content_types.all())
  93. m2m_changed.connect(handle_cf_removed_obj_types, sender=CustomField.content_types.through)
  94. post_save.connect(handle_cf_renamed, sender=CustomField)
  95. pre_delete.connect(handle_cf_deleted, sender=CustomField)
  96. #
  97. # Caching
  98. #
  99. cacheops_cache_hit = Counter('cacheops_cache_hit', 'Number of cache hits')
  100. cacheops_cache_miss = Counter('cacheops_cache_miss', 'Number of cache misses')
  101. cacheops_cache_invalidated = Counter('cacheops_cache_invalidated', 'Number of cache invalidations')
  102. def cache_read_collector(sender, func, hit, **kwargs):
  103. if hit:
  104. cacheops_cache_hit.inc()
  105. else:
  106. cacheops_cache_miss.inc()
  107. def cache_invalidated_collector(sender, obj_dict, **kwargs):
  108. cacheops_cache_invalidated.inc()
  109. cache_read.connect(cache_read_collector)
  110. cache_invalidated.connect(cache_invalidated_collector)