Sentry being overkill and Djangodblog being deprecated, I rolled my own, cannibalising the necessary parts from both.
How it works is by catching the error signal. Then it uses Django's inbuilt exception reporter to generate the fancy 500 error page that Django displays when debugging is enabled. We store this in the DB and render it in the admin console.
Here's my implementation:
Model:
class Error(Model):
"""
Model for storing the individual errors.
"""
kind = CharField( _('type'),
null=True, blank=True, max_length=128, db_index=True
)
info = TextField(
null=False,
)
data = TextField(
blank=True, null=True
)
path = URLField(
null=True, blank=True, verify_exists=False,
)
when = DateTimeField(
null=False, auto_now_add=True, db_index=True,
)
html = TextField(
null=True, blank=True,
)
class Meta:
"""
Meta information for the model.
"""
verbose_name = _('Error')
verbose_name_plural = _('Errors')
def __unicode__(self):
"""
String representation of the object.
"""
return "%s: %s" % (self.kind, self.info)
Admin:
class ErrorAdmin(admin.ModelAdmin):
list_display = ('path', 'kind', 'info', 'when')
list_display_links = ('path',)
ordering = ('-id',)
search_fields = ('path', 'kind', 'info', 'data')
readonly_fields = ('path', 'kind', 'info', 'data', 'when', 'html',)
fieldsets = (
(None, {
'fields': ('kind', 'data', 'info')
}),
)
def has_delete_permission(self, request, obj=None):
"""
Disabling the delete permissions
"""
return False
def has_add_permission(self, request):
"""
Disabling the create permissions
"""
return False
def change_view(self, request, object_id, extra_context={}):
"""
The detail view of the error record.
"""
obj = self.get_object(request, unquote(object_id))
extra_context.update({
'instance': obj,
'error_body': mark_safe(obj.html),
})
return super(ErrorAdmin, self).change_view(request, object_id, extra_context)
admin.site.register(Error, ErrorAdmin)
Helper:
class LoggingExceptionHandler(object):
"""
The logging exception handler
"""
@staticmethod
def create_from_exception(sender, request=None, *args, **kwargs):
"""
Handles the exception upon receiving the signal.
"""
kind, info, data = sys.exc_info()
if not issubclass(kind, Http404):
error = Error.objects.create(
kind = kind.__name__,
html = ExceptionReporter(request, kind, info, data).get_traceback_html(),
path = request.build_absolute_uri(),
info = info,
data = '\n'.join(traceback.format_exception(kind, info, data)),
)
error.save()
Init:
from django.core.signals import got_request_exception
from modules.error.signals import LoggingExceptionHandler
got_request_exception.connect(LoggingExceptionHandler.create_from_exception)