signals.py 3.5 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788
  1. from __future__ import absolute_import, unicode_literals
  2. import six
  3. from django.core.signals import (
  4. request_started, request_finished, got_request_exception)
  5. from django.db.backends.signals import connection_created
  6. from django.db.models.signals import (
  7. class_prepared, pre_init, post_init, pre_save, post_save,
  8. pre_delete, post_delete, post_syncdb)
  9. from django.utils.translation import ugettext_lazy as _, ungettext
  10. from debug_toolbar.panels import Panel
  11. from debug_toolbar.compat import import_module, WEAKREF_TYPES
  12. class SignalsPanel(Panel):
  13. template = 'debug_toolbar/panels/signals.html'
  14. SIGNALS = {
  15. 'request_started': request_started,
  16. 'request_finished': request_finished,
  17. 'got_request_exception': got_request_exception,
  18. 'connection_created': connection_created,
  19. 'class_prepared': class_prepared,
  20. 'pre_init': pre_init,
  21. 'post_init': post_init,
  22. 'pre_save': pre_save,
  23. 'post_save': post_save,
  24. 'pre_delete': pre_delete,
  25. 'post_delete': post_delete,
  26. 'post_syncdb': post_syncdb,
  27. }
  28. def nav_subtitle(self):
  29. signals = self.get_stats()['signals']
  30. num_receivers = sum(len(s[2]) for s in signals)
  31. num_signals = len(signals)
  32. # here we have to handle a double count translation, hence the
  33. # hard coding of one signal
  34. if num_signals == 1:
  35. return ungettext("%(num_receivers)d receiver of 1 signal",
  36. "%(num_receivers)d receivers of 1 signal",
  37. num_receivers) % {'num_receivers': num_receivers}
  38. return ungettext("%(num_receivers)d receiver of %(num_signals)d signals",
  39. "%(num_receivers)d receivers of %(num_signals)d signals",
  40. num_receivers) % {'num_receivers': num_receivers,
  41. 'num_signals': num_signals}
  42. title = _("Signals")
  43. @property
  44. def signals(self):
  45. signals = self.SIGNALS.copy()
  46. for signal in self.toolbar.config['EXTRA_SIGNALS']:
  47. mod_path, signal_name = signal.rsplit('.', 1)
  48. signals_mod = import_module(mod_path)
  49. signals[signal_name] = getattr(signals_mod, signal_name)
  50. return signals
  51. def process_response(self, request, response):
  52. signals = []
  53. for name, signal in sorted(self.signals.items(), key=lambda x: x[0]):
  54. if signal is None:
  55. continue
  56. receivers = []
  57. for receiver in signal.receivers:
  58. receiver = receiver[1]
  59. if isinstance(receiver, WEAKREF_TYPES):
  60. receiver = receiver()
  61. if receiver is None:
  62. continue
  63. receiver = getattr(receiver, '__wraps__', receiver)
  64. receiver_name = getattr(receiver, '__name__', six.text_type(receiver))
  65. if getattr(receiver, '__self__', None) is not None:
  66. receiver_class_name = getattr(receiver.__self__, '__class__', type).__name__
  67. text = "%s.%s" % (receiver_class_name, receiver_name)
  68. elif getattr(receiver, 'im_class', None) is not None: # Python 2 only
  69. receiver_class_name = receiver.im_class.__name__
  70. text = "%s.%s" % (receiver_class_name, receiver_name)
  71. else:
  72. text = "%s" % receiver_name
  73. receivers.append(text)
  74. signals.append((name, signal, receivers))
  75. self.record_stats({'signals': signals})