aboutsummaryrefslogtreecommitdiffstats
path: root/debug_toolbar
diff options
context:
space:
mode:
authorVladislav Poluhin2013-04-24 15:54:17 +0800
committerVladislav Poluhin2013-04-24 15:54:17 +0800
commit4c1c4f7a8873a80f843ba683240b9d0ff0f49e4b (patch)
tree4840de7ec2ec41d1ef7cc362803c2d0ace58d663 /debug_toolbar
parent3d0467d9a4394c4b994a802e6e861ff2562dbb2b (diff)
parent6f79f0b0ada69e1d95d97066ba78e0d18aeab3ba (diff)
downloaddjango-debug-toolbar-4c1c4f7a8873a80f843ba683240b9d0ff0f49e4b.tar.bz2
Merge branch 'master' of https://github.com/django-debug-toolbar/django-debug-toolbar into sql-panel-refactor
Conflicts: debug_toolbar/static/debug_toolbar/css/toolbar.min.css debug_toolbar/views.py
Diffstat (limited to 'debug_toolbar')
-rw-r--r--debug_toolbar/middleware.py2
-rw-r--r--debug_toolbar/panels/template.py6
-rw-r--r--debug_toolbar/static/debug_toolbar/css/toolbar.css43
-rw-r--r--debug_toolbar/urls.py5
-rw-r--r--debug_toolbar/utils/__init__.py10
-rw-r--r--debug_toolbar/utils/tracking/db.py41
-rw-r--r--debug_toolbar/views.py9
7 files changed, 79 insertions, 37 deletions
diff --git a/debug_toolbar/middleware.py b/debug_toolbar/middleware.py
index 1979e1d..2c36a19 100644
--- a/debug_toolbar/middleware.py
+++ b/debug_toolbar/middleware.py
@@ -119,7 +119,7 @@ class DebugToolbarMiddleware(object):
if not toolbar or request.is_ajax():
return response
if isinstance(response, HttpResponseRedirect):
- if not toolbar.config['INTERCEPT_REDIRECTS']:
+ if not toolbar.config['INTERCEPT_REDIRECTS'] or request.is_ajax():
return response
redirect_to = response.get('Location', None)
if redirect_to:
diff --git a/debug_toolbar/panels/template.py b/debug_toolbar/panels/template.py
index 48225a0..390c1e1 100644
--- a/debug_toolbar/panels/template.py
+++ b/debug_toolbar/panels/template.py
@@ -6,7 +6,7 @@ from django.conf import settings
from django.template.context import get_standard_processors
from django.test.signals import template_rendered
from django.utils.translation import ugettext_lazy as _
-from django.db.models.query import QuerySet
+from django.db.models.query import QuerySet, RawQuerySet
from debug_toolbar.panels import DebugPanel
from debug_toolbar.utils.tracking.db import recording, SQLQueryTriggered
@@ -76,9 +76,9 @@ class TemplateDebugPanel(DebugPanel):
elif key == 'LANGUAGES' and isinstance(value, tuple):
temp_layer[key] = '<<languages>>'
# QuerySet would trigger the database: user can run the query from SQL Panel
- elif isinstance(value, QuerySet):
+ elif isinstance(value, (QuerySet, RawQuerySet)):
model_name = "%s.%s" % (value.model._meta.app_label, value.model.__name__)
- temp_layer[key] = '<<queryset of %s>>' % model_name
+ temp_layer[key] = '<<%s of %s>>' % (value.__class__.__name__.lower(), model_name)
else:
try:
recording(False)
diff --git a/debug_toolbar/static/debug_toolbar/css/toolbar.css b/debug_toolbar/static/debug_toolbar/css/toolbar.css
index 06ed173..e3cd15d 100644
--- a/debug_toolbar/static/debug_toolbar/css/toolbar.css
+++ b/debug_toolbar/static/debug_toolbar/css/toolbar.css
@@ -1,9 +1,9 @@
/* http://www.positioniseverything.net/easyclearing.html */
#djDebug .clearfix:after {
- content: ".";
- display: block;
- height: 0;
- clear: both;
+ content: ".";
+ display: block;
+ height: 0;
+ clear: both;
visibility: hidden;
}
#djDebug .clearfix {display: inline-block;}
@@ -31,13 +31,13 @@
line-height:1.5em;
color:#000;
vertical-align:baseline;
- background:transparent;
+ background-color:transparent;
font-family:sans-serif;
text-align:left;
}
#djDebug #djDebugToolbar {
- background:#111;
+ background-color:#111;
width:200px;
z-index:100000000;
position:fixed;
@@ -86,7 +86,8 @@
}
#djDebug #djDebugToolbar li.active {
- background: #333 url() no-repeat left center;
+ background: #333 no-repeat left center;
+ background-image: url("");
padding-left:10px;
}
@@ -105,7 +106,7 @@
#djDebug #djDebugToolbarHandle {
position:fixed;
- background:#fff;
+ background-color:#fff;
border:1px solid #111;
top:30px;
right:0;
@@ -127,7 +128,8 @@
text-decoration:none;
text-align:center;
text-indent:-999999px;
- background:#000 url() no-repeat left center;
+ background: #000 no-repeat left center;
+ background-image: url("");
opacity:0.5;
}
@@ -291,13 +293,13 @@
#djDebug .panelContent table td {
padding:5px 10px;
font-size:14px;
- background:#fff;
+ background-color:#fff;
color:#000;
vertical-align:top;
border:0;
}
#djDebug .panelContent table tr.djDebugOdd td {
- background:#eee;
+ background-color:#eee;
}
*/
@@ -309,19 +311,20 @@
right:15px;
height:40px;
width:40px;
- background: url() no-repeat center center;
+ background: no-repeat center center;
+ background-image: url("");
}
#djDebug .panelContent .djDebugClose:hover {
- background-image: url();
+ background-image: url("");
}
#djDebug .panelContent .djDebugClose.djDebugBack {
- background-image: url();
+ background-image: url("");
}
#djDebug .panelContent .djDebugClose.djDebugBack:hover {
- background-image: url();
+ background-image: url("");
}
#djDebug .panelContent dt, #djDebug .panelContent dd {
@@ -339,16 +342,18 @@
#djDebug a.toggleTemplate {
padding:4px;
background-color:#bbb;
- -moz-border-radius:3px;
-webkit-border-radius:3px;
+ -moz-border-radius:3px;
+ border-radius:3px;
}
#djDebug a.toggleTemplate:hover {
padding:4px;
background-color:#444;
color:#ffe761;
- -moz-border-radius:3px;
-webkit-border-radius:3px;
+ -moz-border-radius:3px;
+ border-radius:3px;
}
@@ -478,7 +483,7 @@
#djDebug .panelContent ul.stats {
- position: relative;
+ position: relative;
}
#djDebug .panelContent ul.stats li {
width: 30%;
@@ -543,10 +548,10 @@
margin-top:0.8em;
}
#djDebug pre {
- white-space: pre-wrap; /* CSS-3 */
white-space: -moz-pre-wrap; /* Mozilla, since 1999 */
white-space: -pre-wrap; /* Opera 4-6 */
white-space: -o-pre-wrap; /* Opera 7 */
+ white-space: pre-wrap; /* CSS-3 */
word-wrap: break-word; /* Internet Explorer 5.5+ */
color: #555;
border:1px solid #ccc;
diff --git a/debug_toolbar/urls.py b/debug_toolbar/urls.py
index 60d6424..b37bf2d 100644
--- a/debug_toolbar/urls.py
+++ b/debug_toolbar/urls.py
@@ -4,7 +4,10 @@ URLpatterns for the debug toolbar.
These should not be loaded explicitly; the debug toolbar middleware will patch
this into the urlconf for the request.
"""
-from django.conf.urls.defaults import *
+try:
+ from django.conf.urls import patterns, url
+except ImportError: # django < 1.4
+ from django.conf.urls.defaults import patterns, url
_PREFIX = '__debug__'
diff --git a/debug_toolbar/utils/__init__.py b/debug_toolbar/utils/__init__.py
index bfb485c..2d2ff10 100644
--- a/debug_toolbar/utils/__init__.py
+++ b/debug_toolbar/utils/__init__.py
@@ -56,9 +56,15 @@ def render_stacktrace(trace):
stacktrace = []
for frame in trace:
params = map(escape, frame[0].rsplit(os.path.sep, 1) + list(frame[1:]))
+ params_dict = dict((unicode(idx), v) for idx, v in enumerate(params))
try:
- stacktrace.append(u'<span class="path">{0}/</span><span class="file">{1}</span> in <span class="func">{3}</span>(<span class="lineno">{2}</span>)\n <span class="code">{4}</span>'.format(*params))
- except IndexError:
+ stacktrace.append(u'<span class="path">%(0)s/</span>'
+ u'<span class="file">%(1)s</span>'
+ u' in <span class="func">%(3)s</span>'
+ u'(<span class="lineno">%(2)s</span>)\n'
+ u' <span class="code">%(4)s</span>'
+ % params_dict)
+ except KeyError:
# This frame doesn't have the expected format, so skip it and move on to the next one
continue
return mark_safe('\n'.join(stacktrace))
diff --git a/debug_toolbar/utils/tracking/db.py b/debug_toolbar/utils/tracking/db.py
index 68408c3..8a51c36 100644
--- a/debug_toolbar/utils/tracking/db.py
+++ b/debug_toolbar/utils/tracking/db.py
@@ -5,7 +5,6 @@ from threading import local
from django.conf import settings
from django.template import Node
-from django.utils import simplejson
from django.utils.encoding import force_unicode, smart_str
from debug_toolbar.utils import ms_from_timedelta, tidy_stacktrace, \
@@ -13,8 +12,13 @@ from debug_toolbar.utils import ms_from_timedelta, tidy_stacktrace, \
from debug_toolbar.utils.compat.db import connections
try:
+ import json
+except ImportError: # python < 2.6
+ from django.utils import simplejson as json
+
+try:
from hashlib import sha1
-except ImportError:
+except ImportError: # python < 2.5
from django.utils.hashcompat import sha_constructor as sha1
# TODO:This should be set in the toolbar loader as a default and panels should
@@ -87,12 +91,25 @@ class NormalCursorWrapper(object):
for key, value in params.iteritems())
return map(self._quote_expr, params)
+ def _decode(self, param):
+ try:
+ return force_unicode(param, strings_only=True)
+ except UnicodeDecodeError:
+ return '(encoded string)'
+
def execute(self, sql, params=()):
- __traceback_hide__ = True
start = datetime.now()
try:
return self.cursor.execute(sql, params)
finally:
+ # FIXME: Sometimes connections which are not in the connections
+ # dict are used (for example in test database destroying).
+ # The code below (at least get_transaction_id(alias) needs to have
+ # the connection in the connections dict. It would be good to
+ # not have this requirement at all, but for now lets just skip
+ # these connections.
+ if self.db.alias not in connections:
+ return
stop = datetime.now()
duration = ms_from_timedelta(stop - start)
enable_stacktraces = getattr(settings,
@@ -103,10 +120,8 @@ class NormalCursorWrapper(object):
stacktrace = []
_params = ''
try:
- _params = simplejson.dumps(
- [force_unicode(x, strings_only=True) for x in params]
- )
- except TypeError:
+ _params = json.dumps(map(self._decode, params))
+ except Exception:
pass # object not JSON serializable
template_info = None
@@ -124,7 +139,7 @@ class NormalCursorWrapper(object):
del cur_frame
alias = getattr(self.db, 'alias', 'default')
- conn = connections[alias].connection
+ conn = self.db.connection
# HACK: avoid imports
if conn:
engine = conn.__class__.__module__.split('.', 1)[0]
@@ -148,11 +163,17 @@ class NormalCursorWrapper(object):
}
if engine == 'psycopg2':
- from psycopg2.extensions import TRANSACTION_STATUS_INERROR
+ # If an erroneous query was ran on the connection, it might
+ # be in a state where checking isolation_level raises an
+ # exception.
+ try:
+ iso_level = conn.isolation_level
+ except conn.InternalError:
+ iso_level = 'unknown'
params.update({
'trans_id': self.logger.get_transaction_id(alias),
'trans_status': conn.get_transaction_status(),
- 'iso_level': conn.isolation_level if not conn.get_transaction_status() == TRANSACTION_STATUS_INERROR else "",
+ 'iso_level': iso_level,
'encoding': conn.encoding,
})
diff --git a/debug_toolbar/views.py b/debug_toolbar/views.py
index 13a36b1..ee650ba 100644
--- a/debug_toolbar/views.py
+++ b/debug_toolbar/views.py
@@ -9,8 +9,13 @@ from django.shortcuts import render_to_response
from django.views.decorators.csrf import csrf_exempt
try:
+ import json
+except ImportError: # python < 2.6
+ from django.utils import simplejson as json
+
+try:
from hashlib import sha1
-except ImportError:
+except ImportError: # python < 2.5
from django.utils.hashcompat import sha_constructor as sha1
from debug_toolbar.forms import SQLSelectForm
@@ -58,6 +63,8 @@ def sql_explain(request):
# EXPLAIN QUERY PLAN dumps a more human-readable summary
# See http://www.sqlite.org/lang_explain.html for details
cursor.execute("EXPLAIN QUERY PLAN %s" % (sql,), params)
+ elif engine == "psycopg2":
+ cursor.execute("EXPLAIN ANALYZE %s" % (sql,), params)
else:
cursor.execute("EXPLAIN %s" % (sql,), params)