117 lines
3.7 KiB
Python
117 lines
3.7 KiB
Python
import json
|
|
|
|
from django.http.request import RawPostDataException
|
|
from django.template.loader import render_to_string
|
|
from django.templatetags.static import static
|
|
from django.urls import path
|
|
from django.utils import timezone
|
|
from django.utils.translation import gettext_lazy as _
|
|
|
|
from debug_toolbar.panels import Panel
|
|
from debug_toolbar.panels.history import views
|
|
from debug_toolbar.panels.history.forms import HistoryStoreForm
|
|
|
|
|
|
class HistoryPanel(Panel):
|
|
"""A panel to display History"""
|
|
|
|
title = _("History")
|
|
nav_title = _("History")
|
|
template = "debug_toolbar/panels/history.html"
|
|
|
|
def get_headers(self, request):
|
|
headers = super().get_headers(request)
|
|
observe_request = self.toolbar.get_observe_request()
|
|
store_id = getattr(self.toolbar, "store_id")
|
|
if store_id and observe_request(request):
|
|
headers["djdt-store-id"] = store_id
|
|
return headers
|
|
|
|
@property
|
|
def enabled(self):
|
|
# Do not show the history panel if the panels are rendered on request
|
|
# rather than loaded via ajax.
|
|
return super().enabled and not self.toolbar.should_render_panels()
|
|
|
|
@property
|
|
def is_historical(self):
|
|
"""The HistoryPanel should not be included in the historical panels."""
|
|
return False
|
|
|
|
@classmethod
|
|
def get_urls(cls):
|
|
return [
|
|
path("history_sidebar/", views.history_sidebar, name="history_sidebar"),
|
|
path("history_refresh/", views.history_refresh, name="history_refresh"),
|
|
]
|
|
|
|
@property
|
|
def nav_subtitle(self):
|
|
return self.get_stats().get("request_url", "")
|
|
|
|
def generate_stats(self, request, response):
|
|
try:
|
|
if request.method == "GET":
|
|
data = request.GET.copy()
|
|
else:
|
|
data = request.POST.copy()
|
|
# GraphQL tends to not be populated in POST. If the request seems
|
|
# empty, check if it's a JSON request.
|
|
if (
|
|
not data
|
|
and request.body
|
|
and request.headers.get("content-type") == "application/json"
|
|
):
|
|
try:
|
|
data = json.loads(request.body)
|
|
except ValueError:
|
|
pass
|
|
except RawPostDataException:
|
|
# It is not guaranteed that we may read the request data (again).
|
|
data = None
|
|
|
|
self.record_stats(
|
|
{
|
|
"request_url": request.get_full_path(),
|
|
"request_method": request.method,
|
|
"status_code": response.status_code,
|
|
"data": data,
|
|
"time": timezone.now(),
|
|
}
|
|
)
|
|
|
|
@property
|
|
def content(self):
|
|
"""Content of the panel when it's displayed in full screen.
|
|
|
|
Fetch every store for the toolbar and include it in the template.
|
|
"""
|
|
stores = {}
|
|
for id, toolbar in reversed(self.toolbar._store.items()):
|
|
stores[id] = {
|
|
"toolbar": toolbar,
|
|
"form": HistoryStoreForm(
|
|
initial={"store_id": id, "exclude_history": True}
|
|
),
|
|
}
|
|
|
|
return render_to_string(
|
|
self.template,
|
|
{
|
|
"current_store_id": self.toolbar.store_id,
|
|
"stores": stores,
|
|
"refresh_form": HistoryStoreForm(
|
|
initial={
|
|
"store_id": self.toolbar.store_id,
|
|
"exclude_history": True,
|
|
}
|
|
),
|
|
},
|
|
)
|
|
|
|
@property
|
|
def scripts(self):
|
|
scripts = super().scripts
|
|
scripts.append(static("debug_toolbar/js/history.js"))
|
|
return scripts
|