aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--.travis.yml5
-rw-r--r--MANIFEST.in3
-rw-r--r--Makefile10
-rw-r--r--README.rst161
-rw-r--r--debug_toolbar/media/debug_toolbar/css/toolbar.min.css1
-rw-r--r--debug_toolbar/middleware.py8
-rw-r--r--debug_toolbar/panels/sql.py3
-rw-r--r--debug_toolbar/panels/template.py6
-rw-r--r--debug_toolbar/static/debug_toolbar/css/toolbar.css (renamed from debug_toolbar/media/debug_toolbar/css/toolbar.css)54
-rw-r--r--debug_toolbar/static/debug_toolbar/css/toolbar.min.css1
-rw-r--r--debug_toolbar/static/debug_toolbar/img/back.png (renamed from debug_toolbar/media/debug_toolbar/img/back.png)bin1039 -> 1039 bytes
-rw-r--r--debug_toolbar/static/debug_toolbar/img/back_hover.png (renamed from debug_toolbar/media/debug_toolbar/img/back_hover.png)bin1030 -> 1030 bytes
-rw-r--r--debug_toolbar/static/debug_toolbar/img/close.png (renamed from debug_toolbar/media/debug_toolbar/img/close.png)bin1045 -> 1045 bytes
-rw-r--r--debug_toolbar/static/debug_toolbar/img/close_hover.png (renamed from debug_toolbar/media/debug_toolbar/img/close_hover.png)bin1155 -> 1155 bytes
-rw-r--r--debug_toolbar/static/debug_toolbar/img/djdt_vertical.png (renamed from debug_toolbar/media/debug_toolbar/img/djdt_vertical.png)bin1349 -> 1349 bytes
-rw-r--r--debug_toolbar/static/debug_toolbar/img/dot.gif (renamed from debug_toolbar/media/debug_toolbar/img/dot.gif)bin62 -> 62 bytes
-rw-r--r--debug_toolbar/static/debug_toolbar/img/indicator.png (renamed from debug_toolbar/media/debug_toolbar/img/indicator.png)bin607 -> 607 bytes
-rw-r--r--debug_toolbar/static/debug_toolbar/img/panel_bg.png (renamed from debug_toolbar/media/debug_toolbar/img/panel_bg.png)bin110 -> 110 bytes
-rw-r--r--debug_toolbar/static/debug_toolbar/js/jquery.cookie.js (renamed from debug_toolbar/media/debug_toolbar/js/jquery.cookie.js)0
-rw-r--r--debug_toolbar/static/debug_toolbar/js/jquery.js (renamed from debug_toolbar/media/debug_toolbar/js/jquery.js)0
-rw-r--r--debug_toolbar/static/debug_toolbar/js/toolbar.js (renamed from debug_toolbar/media/debug_toolbar/js/toolbar.js)0
-rw-r--r--debug_toolbar/static/debug_toolbar/js/toolbar.min.js (renamed from debug_toolbar/media/debug_toolbar/js/toolbar.min.js)0
-rw-r--r--debug_toolbar/templates/debug_toolbar/base.html4
-rw-r--r--debug_toolbar/templates/debug_toolbar/panels/logger.html2
-rw-r--r--debug_toolbar/templates/debug_toolbar/panels/templates.html6
-rw-r--r--debug_toolbar/toolbar/loader.py5
-rw-r--r--debug_toolbar/urls.py6
-rw-r--r--debug_toolbar/utils/__init__.py10
-rw-r--r--debug_toolbar/utils/tracking/db.py47
-rw-r--r--debug_toolbar/views.py34
-rw-r--r--example/settings.py2
-rw-r--r--example/static/js/jquery.js (renamed from example/media/js/jquery.js)0
-rw-r--r--example/static/js/mootools.js (renamed from example/media/js/mootools.js)0
-rw-r--r--example/static/js/prototype.js (renamed from example/media/js/prototype.js)0
-rw-r--r--example/templates/jquery/index.html2
-rw-r--r--example/templates/mootools/index.html2
-rw-r--r--example/templates/prototype/index.html2
-rw-r--r--example/urls.py10
-rw-r--r--runtests.py5
-rw-r--r--test_pgsql.py28
-rw-r--r--tests/tests.py22
-rw-r--r--tests/urls.py7
42 files changed, 278 insertions, 168 deletions
diff --git a/.travis.yml b/.travis.yml
index 2e5676f..6e0da0a 100644
--- a/.travis.yml
+++ b/.travis.yml
@@ -3,8 +3,9 @@ python:
- "2.6"
- "2.7"
env:
- - DJANGO_VERSION=1.3.1
- - DJANGO_VERSION=1.4
+ - DJANGO_VERSION=1.3.7
+ - DJANGO_VERSION=1.4.5
+ - DJANGO_VERSION=1.5
install:
- pip install Django==$DJANGO_VERSION
- python setup.py install
diff --git a/MANIFEST.in b/MANIFEST.in
index 487183f..08be1cc 100644
--- a/MANIFEST.in
+++ b/MANIFEST.in
@@ -1,6 +1,5 @@
include AUTHORS
include LICENSE
include README.rst
-recursive-include debug_toolbar/media *
+recursive-include debug_toolbar/static *
recursive-include debug_toolbar/templates *
-prune example \ No newline at end of file
diff --git a/Makefile b/Makefile
index 01f598b..96eddf8 100644
--- a/Makefile
+++ b/Makefile
@@ -1,9 +1,13 @@
# Make file to compress and join all JS files
all: compress_js compress_css
+test:
+ pip install Django
+ python runtests.py
+
compress_js:
- java -jar ~/bin/yuicompressor.jar debug_toolbar/media/debug_toolbar/js/jquery.js > debug_toolbar/media/debug_toolbar/js/toolbar.min.js
- java -jar ~/bin/yuicompressor.jar debug_toolbar/media/debug_toolbar/js/toolbar.js >> debug_toolbar/media/debug_toolbar/js/toolbar.min.js
+ java -jar ~/bin/yuicompressor.jar debug_toolbar/static/debug_toolbar/js/jquery.js > debug_toolbar/static/debug_toolbar/js/toolbar.min.js
+ java -jar ~/bin/yuicompressor.jar debug_toolbar/static/debug_toolbar/js/toolbar.js >> debug_toolbar/static/debug_toolbar/js/toolbar.min.js
compress_css:
- java -jar ~/bin/yuicompressor.jar --type css debug_toolbar/media/debug_toolbar/css/toolbar.css > debug_toolbar/media/debug_toolbar/css/toolbar.min.css
+ java -jar ~/bin/yuicompressor.jar --type css debug_toolbar/static/debug_toolbar/css/toolbar.css > debug_toolbar/static/debug_toolbar/css/toolbar.min.css
diff --git a/README.rst b/README.rst
index 7c54f40..ed71e07 100644
--- a/README.rst
+++ b/README.rst
@@ -2,6 +2,10 @@
Django Debug Toolbar
====================
+.. image:: https://secure.travis-ci.org/django-debug-toolbar/django-debug-toolbar.png
+ :alt: Build Status
+ :target: http://travis-ci.org/django-debug-toolbar/django-debug-toolbar
+
The Django Debug Toolbar is a configurable set of panels that display various
debug information about the current request/response and when clicked, display
more details about the panel's content.
@@ -20,132 +24,153 @@ Currently, the following panels have been written and are working:
There is also one Django management command currently:
-- `debugsqlshell`: Outputs the SQL that gets executed as you work in the Python
- interactive shell. (See example below)
+- ``debugsqlshell``: Outputs the SQL that gets executed as you work in the
+ Python interactive shell. (See example below)
If you have ideas for other panels please let us know.
-* Note: The Debug Toolbar only works on Django 1.2 and newer.
+* Note: The Debug Toolbar only works on Django 1.3 and newer.
Installation
============
-#. Add the `debug_toolbar` directory to your Python path.
+#. Add the ``debug_toolbar`` directory to your Python path.
-#. Add the following middleware to your project's `settings.py` file:
+#. Add the following middleware to your project's ``settings.py`` file::
- ``'debug_toolbar.middleware.DebugToolbarMiddleware',``
+ MIDDLEWARE_CLASSES = (
+ # ...
+ 'debug_toolbar.middleware.DebugToolbarMiddleware',
+ # ...
+ )
Tying into middleware allows each panel to be instantiated on request and
rendering to happen on response.
- The order of MIDDLEWARE_CLASSES is important: the Debug Toolbar middleware
- must come after any other middleware that encodes the response's content
- (such as GZipMiddleware).
+ The order of ``MIDDLEWARE_CLASSES`` is important: the Debug Toolbar
+ middleware must come after any other middleware that encodes the
+ response's content (such as GZipMiddleware).
- Note: The debug toolbar will only display itself if the mimetype of the
- response is either `text/html` or `application/xhtml+xml` and contains a
- closing `</body>` tag.
+ **Note**: The debug toolbar will only display itself if the mimetype of the
+ response is either ``text/html`` or ``application/xhtml+xml`` and contains a
+ closing ``</body>`` tag.
- Note: Be aware of middleware ordering and other middleware that may
- intercept requests and return responses. Putting the debug toolbar
+ **Note**: Be aware of middleware ordering and other middleware that may
+ intercept requests and return responses. Putting the debug toolbar
middleware *after* the Flatpage middleware, for example, means the
toolbar will not show up on flatpages.
-#. Make sure your IP is listed in the `INTERNAL_IPS` setting. If you are
- working locally this will be:
+#. Make sure your IP is listed in the ``INTERNAL_IPS`` setting. If you are
+ working locally this will be::
- INTERNAL_IPS = ('127.0.0.1',)
+ INTERNAL_IPS = ('127.0.0.1',)
- Note: This is required because of the built-in requirements of the
- `show_toolbar` method. See below for how to define a method to determine
+ **Note**: This is required because of the built-in requirements of the
+ ``show_toolbar`` method. See below for how to define a method to determine
your own logic for displaying the toolbar.
-#. Add `debug_toolbar` to your `INSTALLED_APPS` setting so Django can find the
- template files associated with the Debug Toolbar.
+#. Add ``debug_toolbar`` to your ``INSTALLED_APPS`` setting so Django can
+ find the template files associated with the Debug Toolbar::
- Alternatively, add the path to the debug toolbar templates
- (``'path/to/debug_toolbar/templates'`` to your ``TEMPLATE_DIRS`` setting.)
+ INSTALLED_APPS = (
+ ...
+ 'debug_toolbar',
+ )
Configuration
=============
-The debug toolbar has two settings that can be set in `settings.py`:
+The debug toolbar has two settings that can be set in ``settings.py``:
-#. Optional: Add a tuple called `DEBUG_TOOLBAR_PANELS` to your ``settings.py``
+#. Optional: Add a tuple called ``DEBUG_TOOLBAR_PANELS`` to your ``settings.py``
file that specifies the full Python path to the panel that you want included
- in the Toolbar. This setting looks very much like the `MIDDLEWARE_CLASSES`
- setting. For example::
-
- DEBUG_TOOLBAR_PANELS = (
- 'debug_toolbar.panels.version.VersionDebugPanel',
- 'debug_toolbar.panels.timer.TimerDebugPanel',
- 'debug_toolbar.panels.settings_vars.SettingsVarsDebugPanel',
- 'debug_toolbar.panels.headers.HeaderDebugPanel',
- 'debug_toolbar.panels.request_vars.RequestVarsDebugPanel',
- 'debug_toolbar.panels.template.TemplateDebugPanel',
- 'debug_toolbar.panels.sql.SQLDebugPanel',
- 'debug_toolbar.panels.signals.SignalDebugPanel',
- 'debug_toolbar.panels.logger.LoggingPanel',
- )
+ in the Toolbar. This setting looks very much like the ``MIDDLEWARE_CLASSES``
+ setting. For example::
+
+ DEBUG_TOOLBAR_PANELS = (
+ 'debug_toolbar.panels.version.VersionDebugPanel',
+ 'debug_toolbar.panels.timer.TimerDebugPanel',
+ 'debug_toolbar.panels.settings_vars.SettingsVarsDebugPanel',
+ 'debug_toolbar.panels.headers.HeaderDebugPanel',
+ 'debug_toolbar.panels.request_vars.RequestVarsDebugPanel',
+ 'debug_toolbar.panels.template.TemplateDebugPanel',
+ 'debug_toolbar.panels.sql.SQLDebugPanel',
+ 'debug_toolbar.panels.signals.SignalDebugPanel',
+ 'debug_toolbar.panels.logger.LoggingPanel',
+ )
You can change the ordering of this tuple to customize the order of the
- panels you want to display, or add/remove panels. If you have custom panels
+ panels you want to display, or add/remove panels. If you have custom panels
you can include them in this way -- just provide the full Python path to
your panel.
#. Optional: There are a few configuration options to the debug toolbar that
- can be placed in a dictionary:
+ can be placed in a dictionary called ``DEBUG_TOOLBAR_CONFIG``:
+
+ * ``INTERCEPT_REDIRECTS``
- * `INTERCEPT_REDIRECTS`: If set to True (default), the debug toolbar will
+ If set to True (default), the debug toolbar will
show an intermediate page upon redirect so you can view any debug
- information prior to redirecting. This page will provide a link to the
- redirect destination you can follow when ready. If set to False, redirects
+ information prior to redirecting. This page will provide a link to the
+ redirect destination you can follow when ready. If set to False, redirects
will proceed as normal.
- * `SHOW_TOOLBAR_CALLBACK`: If not set or set to None, the debug_toolbar
+ * ``SHOW_TOOLBAR_CALLBACK``
+
+ If not set or set to None, the debug_toolbar
middleware will use its built-in show_toolbar method for determining whether
- the toolbar should show or not. The default checks are that DEBUG must be
- set to True or the IP of the request must be in INTERNAL_IPS. You can
+ the toolbar should show or not. The default checks are that DEBUG must be
+ set to True and the IP of the request must be in INTERNAL_IPS. You can
provide your own method for displaying the toolbar which contains your
- custom logic. This method should return True or False.
+ custom logic. This method should return True or False.
- * `EXTRA_SIGNALS`: An array of custom signals that might be in your project,
+ * ``EXTRA_SIGNALS``
+
+ An array of custom signals that might be in your project,
defined as the python path to the signal.
- * `HIDE_DJANGO_SQL`: If set to True (the default) then code in Django itself
+ * ``HIDE_DJANGO_SQL``
+
+ If set to True (the default) then code in Django itself
won't be shown in SQL stacktraces.
- * `SHOW_TEMPLATE_CONTEXT`: If set to True (the default) then a template's
- context will be included with it in the Template debug panel. Turning this
+ * ``SHOW_TEMPLATE_CONTEXT``
+
+ If set to True (the default) then a template's
+ context will be included with it in the Template debug panel. Turning this
off is useful when you have large template contexts, or you have template
contexts with lazy datastructures that you don't want to be evaluated.
- * `TAG`: If set, this will be the tag to which debug_toolbar will attach the
+ * ``TAG``
+
+ If set, this will be the tag to which debug_toolbar will attach the
debug toolbar. Defaults to 'body'.
- * `ENABLE_STACKTRACES`: If set, this will show stacktraces for SQL queries
+ * ``ENABLE_STACKTRACES``
+
+ If set, this will show stacktraces for SQL queries
and cache calls. Enabling stacktraces can increase the CPU time used when
executing queries. Defaults to True.
Example configuration::
- def custom_show_toolbar(request):
- return True # Always show toolbar, for example purposes only.
-
- DEBUG_TOOLBAR_CONFIG = {
- 'INTERCEPT_REDIRECTS': False,
- 'SHOW_TOOLBAR_CALLBACK': custom_show_toolbar,
- 'EXTRA_SIGNALS': ['myproject.signals.MySignal'],
- 'HIDE_DJANGO_SQL': False,
- 'TAG': 'div',
- 'ENABLE_STACKTRACES' : True,
- }
+ def custom_show_toolbar(request):
+ return True # Always show toolbar, for example purposes only.
+
+ DEBUG_TOOLBAR_CONFIG = {
+ 'INTERCEPT_REDIRECTS': False,
+ 'SHOW_TOOLBAR_CALLBACK': custom_show_toolbar,
+ 'EXTRA_SIGNALS': ['myproject.signals.MySignal'],
+ 'HIDE_DJANGO_SQL': False,
+ 'TAG': 'div',
+ 'ENABLE_STACKTRACES' : True,
+ }
+
+``debugsqlshell``
+=================
-`debugsqlshell`
-===============
The following is sample output from running the `debugsqlshell` management
-command. Each ORM call that results in a database query will be beautifully
+command. Each ORM call that results in a database query will be beautifully
output in the shell::
$ ./manage.py debugsqlshell
diff --git a/debug_toolbar/media/debug_toolbar/css/toolbar.min.css b/debug_toolbar/media/debug_toolbar/css/toolbar.min.css
deleted file mode 100644
index cdb92ba..0000000
--- a/debug_toolbar/media/debug_toolbar/css/toolbar.min.css
+++ /dev/null
@@ -1 +0,0 @@
-#djDebug .clearfix:after{content:".";display:block;height:0;clear:both;visibility:hidden}#djDebug .clearfix{display:inline-block}/*\*/#djDebug .clearfix{display:block}* html #djDebug .clearfix{height:1%}/**/#djDebug{color:#000;background:#FFF}#djDebug,#djDebug div,#djDebug span,#djDebug applet,#djDebug object,#djDebug iframe,#djDebug h1,#djDebug h2,#djDebug h3,#djDebug h4,#djDebug h5,#djDebug h6,#djDebug p,#djDebug blockquote,#djDebug pre,#djDebug a,#djDebug abbr,#djDebug acronym,#djDebug address,#djDebug big,#djDebug cite,#djDebug code,#djDebug del,#djDebug dfn,#djDebug em,#djDebug font,#djDebug img,#djDebug ins,#djDebug kbd,#djDebug q,#djDebug s,#djDebug samp,#djDebug small,#djDebug strike,#djDebug strong,#djDebug sub,#djDebug sup,#djDebug tt,#djDebug var,#djDebug b,#djDebug u,#djDebug i,#djDebug center,#djDebug dl,#djDebug dt,#djDebug dd,#djDebug ol,#djDebug ul,#djDebug li,#djDebug fieldset,#djDebug form,#djDebug label,#djDebug legend,#djDebug table,#djDebug caption,#djDebug tbody,#djDebug tfoot,#djDebug thead,#djDebug tr,#djDebug th,#djDebug td{margin:0;padding:0;border:0;outline:0;font-size:12px;line-height:1.5em;color:#000;vertical-align:baseline;background:transparent;font-family:sans-serif;text-align:left}#djDebug #djDebugToolbar{background:#111;width:200px;z-index:100000000;position:fixed;top:0;bottom:0;right:0;opacity:.9}#djDebug #djDebugToolbar small{color:#999}#djDebug #djDebugToolbar ul{margin:0;padding:0;list-style:none}#djDebug #djDebugToolbar li{border-bottom:1px solid #222;color:#fff;display:block;font-weight:bold;float:none;margin:0;padding:0;position:relative;width:auto}#djDebug #djDebugToolbar li>a,#djDebug #djDebugToolbar li>div.contentless{font-weight:normal;font-style:normal;text-decoration:none;display:block;font-size:16px;padding:10px 10px 5px 25px;color:#fff}#djDebug #djDebugToolbar li a:hover{color:#111;background-color:#ffc}#djDebug #djDebugToolbar li.active{background-image:url(../img/indicator.png);background-image:url();background-repeat:no-repeat;background-position:left center;background-color:#333;padding-left:10px}#djDebug #djDebugToolbar li.active a:hover{color:#b36a60;background-color:transparent}#djDebug #djDebugToolbar li small{font-size:12px;color:#999;font-style:normal;text-decoration:none;font-variant:small-caps}#djDebug #djDebugToolbarHandle{position:fixed;background:#fff;border:1px solid #111;top:30px;right:0;z-index:100000000;opacity:.75}#djDebug a#djShowToolBarButton{display:block;height:75px;width:30px;border-right:0;border-bottom:4px solid #fff;border-top:4px solid #fff;border-left:4px solid #fff;color:#fff;font-size:10px;font-weight:bold;text-decoration:none;text-align:center;text-indent:-999999px;background:#000 url(../img/djdt_vertical.png) no-repeat left center;background-image:url();opacity:.5}#djDebug a#djShowToolBarButton:hover{background-color:#111;padding-right:6px;border-top-color:#ffe761;border-left-color:#ffe761;border-bottom-color:#ffe761;opacity:1.0}#djDebug code{display:block;font-family:Consolas,Monaco,"Bitstream Vera Sans Mono","Lucida Console",monospace;white-space:pre;overflow:auto}#djDebug tr.djDebugOdd{background-color:#f5f5f5}#djDebug .panelContent{display:none;position:fixed;margin:0;top:0;right:200px;bottom:0;left:0;background-color:#eee;color:#666;z-index:100000000}#djDebug .panelContent>div{border-bottom:1px solid #ddd}#djDebug .djDebugPanelTitle{position:absolute;background-color:#ffc;color:#666;padding-left:20px;top:0;right:0;left:0;height:50px}#djDebug .djDebugPanelTitle code{display:inline;font-size:inherit}#djDebug .djDebugPanelContent{position:absolute;top:50px;right:0;bottom:0;left:0;height:auto;padding:5px 0 0 20px}#djDebug .djDebugPanelContent .scroll{height:100%;overflow:auto;display:block;padding:0 10px 0 0}#djDebug h3{font-size:24px;font-weight:normal;line-height:50px}#djDebug h4{font-size:20px;font-weight:bold;margin-top:.8em}#djDebug .panelContent table{border:1px solid #ccc;border-collapse:collapse;width:100%;background-color:#fff;display:block;margin-top:.8em;overflow:auto}#djDebug .panelContent tbody td,#djDebug .panelContent tbody th{vertical-align:top;padding:2px 3px}#djDebug .panelContent thead th{padding:1px 6px 1px 3px;text-align:left;font-weight:bold;font-size:14px}#djDebug .panelContent tbody th{width:12em;text-align:right;color:#666;padding-right:.5em}#djDebug .djTemplateHideContextDiv{background-color:#fff}#djDebug .panelContent .djDebugClose{text-indent:-9999999px;display:block;position:absolute;top:4px;right:15px;height:40px;width:40px;background:url(../img/close.png) no-repeat center center;background-image:url()}#djDebug .panelContent .djDebugClose:hover{background-image:url(../img/close_hover.png);background-image:url()}#djDebug .panelContent .djDebugClose.djDebugBack{background-image:url(../img/back.png);background-image:url()}#djDebug .panelContent .djDebugClose.djDebugBack:hover{background-image:url(../img/back_hover.png);background-image:url()}#djDebug .panelContent dt,#djDebug .panelContent dd{display:block}#djDebug .panelContent dt{margin-top:.75em}#djDebug .panelContent dd{margin-left:10px}#djDebug a.toggleTemplate{padding:4px;background-color:#bbb;-moz-border-radius:3px;-webkit-border-radius:3px}#djDebug a.toggleTemplate:hover{padding:4px;background-color:#444;color:#ffe761;-moz-border-radius:3px;-webkit-border-radius:3px}#djDebug a.djTemplateShowContext,#djDebug a.djTemplateShowContext span.toggleArrow{color:#999}#djDebug a.djTemplateShowContext:hover,#djDebug a.djTemplateShowContext:hover span.toggleArrow{color:#000;cursor:pointer}#djDebug .djDebugSqlWrap{position:relative}#djDebug .djDebugCollapsed{display:none;text-decoration:none;color:#333}#djDebug .djDebugUncollapsed{color:#333;text-decoration:none}#djDebug .djUnselected{display:none}#djDebug tr.djHiddenByDefault{display:none}#djDebug tr.djSelected{display:table-row}#djDebug .djDebugSql{z-index:100000002}#djDebug .djSQLDetailsDiv tbody th{text-align:left}#djDebug .djSqlExplain td{white-space:pre}#djDebug span.djDebugLineChart{background-color:#777;height:3px;position:absolute;bottom:0;top:0;left:0;display:block;z-index:1000000001}#djDebug span.djDebugLineChartWarning{background-color:#900}#djDebug .highlight{color:#000}#djDebug .highlight .err{color:#000}#djDebug .highlight .g{color:#000}#djDebug .highlight .k{color:#000;font-weight:bold}#djDebug .highlight .o{color:#000}#djDebug .highlight .n{color:#000}#djDebug .highlight .mi{color:#000;font-weight:bold}#djDebug .highlight .l{color:#000}#djDebug .highlight .x{color:#000}#djDebug .highlight .p{color:#000}#djDebug .highlight .m{color:#000;font-weight:bold}#djDebug .highlight .s{color:#333}#djDebug .highlight .w{color:#888}#djDebug .highlight .il{color:#000;font-weight:bold}#djDebug .highlight .na{color:#333}#djDebug .highlight .nt{color:#000;font-weight:bold}#djDebug .highlight .nv{color:#333}#djDebug .highlight .s2{color:#333}#djDebug .highlight .cp{color:#333}#djDebug .timeline{width:30%}#djDebug .djDebugTimeline{position:relative;height:100%;min-height:100%}#djDebug div.djDebugLineChart{position:absolute;left:0;right:0;top:0;bottom:0;vertical-align:middle}#djDebug div.djDebugLineChart strong{text-indent:-10000em;display:block;font-weight:normal;vertical-align:middle;background-color:#ccc}#djDebug div.djDebugLineChartWarning strong{background-color:#900}#djDebug .djDebugInTransaction div.djDebugLineChart strong{background-color:#d3ff82}#djDebug .djDebugStartTransaction div.djDebugLineChart strong{border-left:1px solid #94b24d}#djDebug .djDebugEndTransaction div.djDebugLineChart strong{border-right:1px solid #94b24d}#djDebug .djDebugHover div.djDebugLineChart strong{background-color:#000}#djDebug .djDebugInTransaction.djDebugHover div.djDebugLineChart strong{background-color:#94b24d}#djDebug .panelContent ul.stats{position:relative}#djDebug .panelContent ul.stats li{width:30%;float:left}#djDebug .panelContent ul.stats li strong.label{display:block}#djDebug .panelContent ul.stats li span.color{height:12px;width:3px;display:inline-block}#djDebug .panelContent ul.stats li span.info{display:block;padding-left:5px}#djDebug .panelcontent thead th{white-space:nowrap}#djDebug .djDebugRowWarning .time{color:red}#djdebug .panelcontent table .toggle{width:14px;padding-top:3px}#djdebug .panelcontent table .actions{min-width:70px}#djdebug .panelcontent table .color{width:3px}#djdebug .panelcontent table .color span{width:3px;height:12px;overflow:hidden;padding:0}#djDebug .djToggleSwitch{text-decoration:none;border:1px solid #999;height:12px;width:12px;line-height:12px;text-align:center;color:#777;display:inline-block;filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#FFF',endColorstr='#DCDCDC');background:-webkit-gradient(linear,left top,left bottom,from(#FFF),to(#dcdcdc));background:-moz-linear-gradient(center top,#fff 0,#dcdcdc 100%) repeat scroll 0 0 transparent}#djDebug .djNoToggleSwitch{height:14px;width:14px;display:inline-block}#djDebug .djSQLDetailsDiv{margin-top:.8em}#djDebug pre{white-space:pre-wrap;white-space:-moz-pre-wrap;white-space:-pre-wrap;white-space:-o-pre-wrap;word-wrap:break-word;color:#555;border:1px solid #ccc;border-collapse:collapse;background-color:#fff;display:block;overflow:auto;padding:2px 3px;margin-bottom:3px;font-family:Consolas,Monaco,"Bitstream Vera Sans Mono","Lucida Console",monospace}#djDebug .stack span{color:#000;font-weight:bold}#djDebug .stack span.path{color:#777;font-weight:normal}#djDebug .stack span.code{font-weight:normal}@media print{#djDebug{display:none}} \ No newline at end of file
diff --git a/debug_toolbar/middleware.py b/debug_toolbar/middleware.py
index 14cbb95..2c36a19 100644
--- a/debug_toolbar/middleware.py
+++ b/debug_toolbar/middleware.py
@@ -105,14 +105,18 @@ class DebugToolbarMiddleware(object):
toolbar = self.__class__.debug_toolbars.get(thread.get_ident())
if not toolbar:
return
+ result = None
for panel in toolbar.panels:
- panel.process_view(request, view_func, view_args, view_kwargs)
+ response = panel.process_view(request, view_func, view_args, view_kwargs)
+ if response:
+ result = response
+ return result
def process_response(self, request, response):
__traceback_hide__ = True
ident = thread.get_ident()
toolbar = self.__class__.debug_toolbars.get(ident)
- if not toolbar:
+ if not toolbar or request.is_ajax():
return response
if isinstance(response, HttpResponseRedirect):
if not toolbar.config['INTERCEPT_REDIRECTS'] or request.is_ajax():
diff --git a/debug_toolbar/panels/sql.py b/debug_toolbar/panels/sql.py
index fb34e51..a492888 100644
--- a/debug_toolbar/panels/sql.py
+++ b/debug_toolbar/panels/sql.py
@@ -170,7 +170,8 @@ class SQLDebugPanel(DebugPanel):
query['iso_level'] = get_isolation_level_display(query['engine'], query['iso_level'])
if 'trans_status' in query:
query['trans_status'] = get_transaction_status_display(query['engine'], query['trans_status'])
- query['sql'] = reformat_sql(query['sql'])
+ if query['sql']:
+ query['sql'] = reformat_sql(query['sql'])
query['rgb_color'] = self._databases[alias]['rgb_color']
try:
query['width_ratio'] = (query['duration'] / self._sql_time) * 100
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/media/debug_toolbar/css/toolbar.css b/debug_toolbar/static/debug_toolbar/css/toolbar.css
index 65e7c92..b29b2aa 100644
--- a/debug_toolbar/media/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,12 +86,9 @@
}
#djDebug #djDebugToolbar li.active {
- background-image:url(../img/indicator.png);
- background-image:url();
- background-repeat:no-repeat;
- background-position:left center;
- background-color:#333;
- padding-left:10px;
+ background: #333 no-repeat left center;
+ background-image: url("");
+ padding-left:10px;
}
#djDebug #djDebugToolbar li.active a:hover {
@@ -109,7 +106,7 @@
#djDebug #djDebugToolbarHandle {
position:fixed;
- background:#fff;
+ background-color:#fff;
border:1px solid #111;
top:30px;
right:0;
@@ -131,8 +128,8 @@
text-decoration:none;
text-align:center;
text-indent:-999999px;
- background:#000 url(../img/djdt_vertical.png) no-repeat left center;
- background-image:url();
+ background: #000 no-repeat left center;
+ background-image: url("");
opacity:0.5;
}
@@ -291,13 +288,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,23 +306,20 @@
right:15px;
height:40px;
width:40px;
- background:url(../img/close.png) no-repeat center center;
- background-image: url();
+ background: no-repeat center center;
+ background-image: url("");
}
#djDebug .panelContent .djDebugClose:hover {
- background-image:url(../img/close_hover.png);
- background-image: url();
+ background-image: url("");
}
#djDebug .panelContent .djDebugClose.djDebugBack {
- background-image:url(../img/back.png);
- background-image: url();
+ background-image: url("");
}
#djDebug .panelContent .djDebugClose.djDebugBack:hover {
- background-image:url(../img/back_hover.png);
- background-image: url();
+ background-image: url("");
}
#djDebug .panelContent dt, #djDebug .panelContent dd {
@@ -343,16 +337,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;
}
@@ -482,7 +478,7 @@
#djDebug .panelContent ul.stats {
- position: relative;
+ position: relative;
}
#djDebug .panelContent ul.stats li {
width: 30%;
@@ -546,10 +542,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/static/debug_toolbar/css/toolbar.min.css b/debug_toolbar/static/debug_toolbar/css/toolbar.min.css
new file mode 100644
index 0000000..ab235cb
--- /dev/null
+++ b/debug_toolbar/static/debug_toolbar/css/toolbar.min.css
@@ -0,0 +1 @@
+#djDebug .clearfix:after{content:".";display:block;height:0;clear:both;visibility:hidden}#djDebug .clearfix{display:inline-block}/*\*/#djDebug .clearfix{display:block}* html #djDebug .clearfix{height:1%}/**/#djDebug{color:#000;background:#FFF}#djDebug,#djDebug div,#djDebug span,#djDebug applet,#djDebug object,#djDebug iframe,#djDebug h1,#djDebug h2,#djDebug h3,#djDebug h4,#djDebug h5,#djDebug h6,#djDebug p,#djDebug blockquote,#djDebug pre,#djDebug a,#djDebug abbr,#djDebug acronym,#djDebug address,#djDebug big,#djDebug cite,#djDebug code,#djDebug del,#djDebug dfn,#djDebug em,#djDebug font,#djDebug img,#djDebug ins,#djDebug kbd,#djDebug q,#djDebug s,#djDebug samp,#djDebug small,#djDebug strike,#djDebug strong,#djDebug sub,#djDebug sup,#djDebug tt,#djDebug var,#djDebug b,#djDebug u,#djDebug i,#djDebug center,#djDebug dl,#djDebug dt,#djDebug dd,#djDebug ol,#djDebug ul,#djDebug li,#djDebug fieldset,#djDebug form,#djDebug label,#djDebug legend,#djDebug table,#djDebug caption,#djDebug tbody,#djDebug tfoot,#djDebug thead,#djDebug tr,#djDebug th,#djDebug td{margin:0;padding:0;border:0;outline:0;font-size:12px;line-height:1.5em;color:#000;vertical-align:baseline;background-color:transparent;font-family:sans-serif;text-align:left}#djDebug #djDebugToolbar{background-color:#111;width:200px;z-index:100000000;position:fixed;top:0;bottom:0;right:0;opacity:.9}#djDebug #djDebugToolbar small{color:#999}#djDebug #djDebugToolbar ul{margin:0;padding:0;list-style:none}#djDebug #djDebugToolbar li{border-bottom:1px solid #222;color:#fff;display:block;font-weight:bold;float:none;margin:0;padding:0;position:relative;width:auto}#djDebug #djDebugToolbar li>a,#djDebug #djDebugToolbar li>div.contentless{font-weight:normal;font-style:normal;text-decoration:none;display:block;font-size:16px;padding:10px 10px 5px 25px;color:#fff}#djDebug #djDebugToolbar li a:hover{color:#111;background-color:#ffc}#djDebug #djDebugToolbar li.active{background:#333 no-repeat left center;background-image:url("");padding-left:10px}#djDebug #djDebugToolbar li.active a:hover{color:#b36a60;background-color:transparent}#djDebug #djDebugToolbar li small{font-size:12px;color:#999;font-style:normal;text-decoration:none;font-variant:small-caps}#djDebug #djDebugToolbarHandle{position:fixed;background-color:#fff;border:1px solid #111;top:30px;right:0;z-index:100000000;opacity:.75}#djDebug a#djShowToolBarButton{display:block;height:75px;width:30px;border-right:0;border-bottom:4px solid #fff;border-top:4px solid #fff;border-left:4px solid #fff;color:#fff;font-size:10px;font-weight:bold;text-decoration:none;text-align:center;text-indent:-999999px;background:#000 no-repeat left center;background-image:url("");opacity:.5}#djDebug a#djShowToolBarButton:hover{background-color:#111;padding-right:6px;border-top-color:#ffe761;border-left-color:#ffe761;border-bottom-color:#ffe761;opacity:1.0}#djDebug code{display:block;font-family:Consolas,Monaco,"Bitstream Vera Sans Mono","Lucida Console",monospace;white-space:pre;overflow:auto}#djDebug tr.djDebugOdd{background-color:#f5f5f5}#djDebug .panelContent{display:none;position:fixed;margin:0;top:0;right:200px;bottom:0;left:0;background-color:#eee;color:#666;z-index:100000000}#djDebug .panelContent>div{border-bottom:1px solid #ddd}#djDebug .djDebugPanelTitle{position:absolute;background-color:#ffc;color:#666;padding-left:20px;top:0;right:0;left:0;height:50px}#djDebug .djDebugPanelTitle code{display:inline;font-size:inherit}#djDebug .djDebugPanelContent{position:absolute;top:50px;right:0;bottom:0;left:0;height:auto;padding:5px 0 0 20px}#djDebug .djDebugPanelContent .scroll{height:100%;overflow:auto;display:block;padding:0 10px 0 0}#djDebug h3{font-size:24px;font-weight:normal;line-height:50px}#djDebug h4{font-size:20px;font-weight:bold;margin-top:.8em}#djDebug .panelContent table{border:1px solid #ccc;border-collapse:collapse;width:100%;background-color:#fff;display:block;margin-top:.8em;overflow:auto}#djDebug .panelContent tbody td,#djDebug .panelContent tbody th{vertical-align:top;padding:2px 3px}#djDebug .panelContent thead th{padding:1px 6px 1px 3px;text-align:left;font-weight:bold;font-size:14px}#djDebug .panelContent tbody th{width:12em;text-align:right;color:#666;padding-right:.5em}#djDebug .djTemplateHideContextDiv{background-color:#fff}#djDebug .panelContent .djDebugClose{text-indent:-9999999px;display:block;position:absolute;top:4px;right:15px;height:40px;width:40px;background:no-repeat center center;background-image:url("")}#djDebug .panelContent .djDebugClose:hover{background-image:url("")}#djDebug .panelContent .djDebugClose.djDebugBack{background-image:url("")}#djDebug .panelContent .djDebugClose.djDebugBack:hover{background-image:url("")}#djDebug .panelContent dt,#djDebug .panelContent dd{display:block}#djDebug .panelContent dt{margin-top:.75em}#djDebug .panelContent dd{margin-left:10px}#djDebug a.toggleTemplate{padding:4px;background-color:#bbb;-webkit-border-radius:3px;-moz-border-radius:3px;border-radius:3px}#djDebug a.toggleTemplate:hover{padding:4px;background-color:#444;color:#ffe761;-webkit-border-radius:3px;-moz-border-radius:3px;border-radius:3px}#djDebug a.djTemplateShowContext,#djDebug a.djTemplateShowContext span.toggleArrow{color:#999}#djDebug a.djTemplateShowContext:hover,#djDebug a.djTemplateShowContext:hover span.toggleArrow{color:#000;cursor:pointer}#djDebug .djDebugSqlWrap{position:relative}#djDebug .djDebugCollapsed{display:none;text-decoration:none;color:#333}#djDebug .djDebugUncollapsed{color:#333;text-decoration:none}#djDebug .djUnselected{display:none}#djDebug tr.djHiddenByDefault{display:none}#djDebug tr.djSelected{display:table-row}#djDebug .djDebugSql{z-index:100000002}#djDebug .djSQLDetailsDiv tbody th{text-align:left}#djDebug .djSqlExplain td{white-space:pre}#djDebug span.djDebugLineChart{background-color:#777;height:3px;position:absolute;bottom:0;top:0;left:0;display:block;z-index:1000000001}#djDebug span.djDebugLineChartWarning{background-color:#900}#djDebug .highlight{color:#000}#djDebug .highlight .err{color:#000}#djDebug .highlight .g{color:#000}#djDebug .highlight .k{color:#000;font-weight:bold}#djDebug .highlight .o{color:#000}#djDebug .highlight .n{color:#000}#djDebug .highlight .mi{color:#000;font-weight:bold}#djDebug .highlight .l{color:#000}#djDebug .highlight .x{color:#000}#djDebug .highlight .p{color:#000}#djDebug .highlight .m{color:#000;font-weight:bold}#djDebug .highlight .s{color:#333}#djDebug .highlight .w{color:#888}#djDebug .highlight .il{color:#000;font-weight:bold}#djDebug .highlight .na{color:#333}#djDebug .highlight .nt{color:#000;font-weight:bold}#djDebug .highlight .nv{color:#333}#djDebug .highlight .s2{color:#333}#djDebug .highlight .cp{color:#333}#djDebug .timeline{width:30%}#djDebug .djDebugTimeline{position:relative;height:100%;min-height:100%}#djDebug div.djDebugLineChart{position:absolute;left:0;right:0;top:0;bottom:0;vertical-align:middle}#djDebug div.djDebugLineChart strong{text-indent:-10000em;display:block;font-weight:normal;vertical-align:middle;background-color:#ccc}#djDebug div.djDebugLineChartWarning strong{background-color:#900}#djDebug .djDebugInTransaction div.djDebugLineChart strong{background-color:#d3ff82}#djDebug .djDebugStartTransaction div.djDebugLineChart strong{border-left:1px solid #94b24d}#djDebug .djDebugEndTransaction div.djDebugLineChart strong{border-right:1px solid #94b24d}#djDebug .djDebugHover div.djDebugLineChart strong{background-color:#000}#djDebug .djDebugInTransaction.djDebugHover div.djDebugLineChart strong{background-color:#94b24d}#djDebug .panelContent ul.stats{position:relative}#djDebug .panelContent ul.stats li{width:30%;float:left}#djDebug .panelContent ul.stats li strong.label{display:block}#djDebug .panelContent ul.stats li span.color{height:12px;width:3px;display:inline-block}#djDebug .panelContent ul.stats li span.info{display:block;padding-left:5px}#djDebug .panelcontent thead th{white-space:nowrap}#djDebug .djDebugRowWarning .time{color:red}#djdebug .panelcontent table .toggle{width:14px;padding-top:3px}#djdebug .panelcontent table .actions{min-width:70px}#djdebug .panelcontent table .color{width:3px}#djdebug .panelcontent table .color span{width:3px;height:12px;overflow:hidden;padding:0}#djDebug .djToggleSwitch{text-decoration:none;border:1px solid #999;height:12px;width:12px;line-height:12px;text-align:center;color:#777;display:inline-block;filter:progid:DXImageTransform.Microsoft.gradient(startColorstr='#FFF',endColorstr='#DCDCDC');background:-webkit-gradient(linear,left top,left bottom,from(#FFF),to(#dcdcdc));background:-moz-linear-gradient(center top,#fff 0,#dcdcdc 100%) repeat scroll 0 0 transparent}#djDebug .djNoToggleSwitch{height:14px;width:14px;display:inline-block}#djDebug .djSQLDetailsDiv{margin-top:.8em}#djDebug pre{white-space:-moz-pre-wrap;white-space:-pre-wrap;white-space:-o-pre-wrap;white-space:pre-wrap;word-wrap:break-word;color:#555;border:1px solid #ccc;border-collapse:collapse;background-color:#fff;display:block;overflow:auto;padding:2px 3px;margin-bottom:3px;font-family:Consolas,Monaco,"Bitstream Vera Sans Mono","Lucida Console",monospace}#djDebug .stack span{color:#000;font-weight:bold}#djDebug .stack span.path{color:#777;font-weight:normal}#djDebug .stack span.code{font-weight:normal}@media print{#djDebug{display:none}}
diff --git a/debug_toolbar/media/debug_toolbar/img/back.png b/debug_toolbar/static/debug_toolbar/img/back.png
index 6ac8a52..6ac8a52 100644
--- a/debug_toolbar/media/debug_toolbar/img/back.png
+++ b/debug_toolbar/static/debug_toolbar/img/back.png
Binary files differ
diff --git a/debug_toolbar/media/debug_toolbar/img/back_hover.png b/debug_toolbar/static/debug_toolbar/img/back_hover.png
index 452b673..452b673 100644
--- a/debug_toolbar/media/debug_toolbar/img/back_hover.png
+++ b/debug_toolbar/static/debug_toolbar/img/back_hover.png
Binary files differ
diff --git a/debug_toolbar/media/debug_toolbar/img/close.png b/debug_toolbar/static/debug_toolbar/img/close.png
index c0e8135..c0e8135 100644
--- a/debug_toolbar/media/debug_toolbar/img/close.png
+++ b/debug_toolbar/static/debug_toolbar/img/close.png
Binary files differ
diff --git a/debug_toolbar/media/debug_toolbar/img/close_hover.png b/debug_toolbar/static/debug_toolbar/img/close_hover.png
index 5b2c812..5b2c812 100644
--- a/debug_toolbar/media/debug_toolbar/img/close_hover.png
+++ b/debug_toolbar/static/debug_toolbar/img/close_hover.png
Binary files differ
diff --git a/debug_toolbar/media/debug_toolbar/img/djdt_vertical.png b/debug_toolbar/static/debug_toolbar/img/djdt_vertical.png
index 000c60f..000c60f 100644
--- a/debug_toolbar/media/debug_toolbar/img/djdt_vertical.png
+++ b/debug_toolbar/static/debug_toolbar/img/djdt_vertical.png
Binary files differ
diff --git a/debug_toolbar/media/debug_toolbar/img/dot.gif b/debug_toolbar/static/debug_toolbar/img/dot.gif
index 9009f29..9009f29 100644
--- a/debug_toolbar/media/debug_toolbar/img/dot.gif
+++ b/debug_toolbar/static/debug_toolbar/img/dot.gif
Binary files differ
diff --git a/debug_toolbar/media/debug_toolbar/img/indicator.png b/debug_toolbar/static/debug_toolbar/img/indicator.png
index 1a2c578..1a2c578 100644
--- a/debug_toolbar/media/debug_toolbar/img/indicator.png
+++ b/debug_toolbar/static/debug_toolbar/img/indicator.png
Binary files differ
diff --git a/debug_toolbar/media/debug_toolbar/img/panel_bg.png b/debug_toolbar/static/debug_toolbar/img/panel_bg.png
index 73add17..73add17 100644
--- a/debug_toolbar/media/debug_toolbar/img/panel_bg.png
+++ b/debug_toolbar/static/debug_toolbar/img/panel_bg.png
Binary files differ
diff --git a/debug_toolbar/media/debug_toolbar/js/jquery.cookie.js b/debug_toolbar/static/debug_toolbar/js/jquery.cookie.js
index 6df1fac..6df1fac 100644
--- a/debug_toolbar/media/debug_toolbar/js/jquery.cookie.js
+++ b/debug_toolbar/static/debug_toolbar/js/jquery.cookie.js
diff --git a/debug_toolbar/media/debug_toolbar/js/jquery.js b/debug_toolbar/static/debug_toolbar/js/jquery.js
index 74ce411..74ce411 100644
--- a/debug_toolbar/media/debug_toolbar/js/jquery.js
+++ b/debug_toolbar/static/debug_toolbar/js/jquery.js
diff --git a/debug_toolbar/media/debug_toolbar/js/toolbar.js b/debug_toolbar/static/debug_toolbar/js/toolbar.js
index 6ec7218..6ec7218 100644
--- a/debug_toolbar/media/debug_toolbar/js/toolbar.js
+++ b/debug_toolbar/static/debug_toolbar/js/toolbar.js
diff --git a/debug_toolbar/media/debug_toolbar/js/toolbar.min.js b/debug_toolbar/static/debug_toolbar/js/toolbar.min.js
index 1a1ca35..1a1ca35 100644
--- a/debug_toolbar/media/debug_toolbar/js/toolbar.min.js
+++ b/debug_toolbar/static/debug_toolbar/js/toolbar.min.js
diff --git a/debug_toolbar/templates/debug_toolbar/base.html b/debug_toolbar/templates/debug_toolbar/base.html
index b6ec842..a790bd1 100644
--- a/debug_toolbar/templates/debug_toolbar/base.html
+++ b/debug_toolbar/templates/debug_toolbar/base.html
@@ -1,9 +1,9 @@
{% load i18n %}
<style type="text/css">
@media print { #djDebug {display:none;}}
-{{ css }}
</style>
-<script type="text/javascript">{{ js }}</script>
+<link rel="stylesheet" href="{{ STATIC_URL }}debug_toolbar/css/toolbar.min.css" type="text/css">
+<script type="text/javascript" src="{{ STATIC_URL }}debug_toolbar/js/toolbar.min.js"></script>
<div id="djDebug" style="display:none;" dir="ltr">
<div style="display:none;" id="djDebugToolbar">
<ul id="djDebugPanelList">
diff --git a/debug_toolbar/templates/debug_toolbar/panels/logger.html b/debug_toolbar/templates/debug_toolbar/panels/logger.html
index c41749e..3908f4f 100644
--- a/debug_toolbar/templates/debug_toolbar/panels/logger.html
+++ b/debug_toolbar/templates/debug_toolbar/panels/logger.html
@@ -16,7 +16,7 @@
<td>{{ record.level }}</td>
<td>{{ record.time|date:"h:i:s m/d/Y" }}</td>
<td>{{ record.channel|default:"-" }}</td>
- <td>{{ record.message }}</td>
+ <td>{{ record.message|linebreaksbr }}</td>
<td>{{ record.file }}:{{ record.line }}</td>
</tr>
{% endfor %}
diff --git a/debug_toolbar/templates/debug_toolbar/panels/templates.html b/debug_toolbar/templates/debug_toolbar/panels/templates.html
index 8061053..bde09a9 100644
--- a/debug_toolbar/templates/debug_toolbar/panels/templates.html
+++ b/debug_toolbar/templates/debug_toolbar/panels/templates.html
@@ -1,5 +1,5 @@
{% load i18n %}
-<h4>{% blocktrans count template_count=template_dirs|length %}Template path{% plural %}Template paths{% endblocktrans %}</h4>
+<h4>{% blocktrans count template_dirs|length as template_count %}Template path{% plural %}Template paths{% endblocktrans %}</h4>
{% if template_dirs %}
<ol>
{% for template in template_dirs %}
@@ -10,7 +10,7 @@
<p>{% trans "None" %}</p>
{% endif %}
-<h4>{% blocktrans count template_count=templates|length %}Template{% plural %}Templates{% endblocktrans %}</h4>
+<h4>{% blocktrans count templates|length as template_count %}Template{% plural %}Templates{% endblocktrans %}</h4>
{% if templates %}
<dl>
{% for template in templates %}
@@ -28,7 +28,7 @@
<p>{% trans 'None' %}</p>
{% endif %}
-<h4>{% blocktrans count context_processors_count=context_processors|length %}Context processor{% plural %}Context processors{% endblocktrans %}</h4>
+<h4>{% blocktrans count context_processors|length as context_processors_count %}Context processor{% plural %}Context processors{% endblocktrans %}</h4>
{% if context_processors %}
<dl>
{% for key, value in context_processors.iteritems %}
diff --git a/debug_toolbar/toolbar/loader.py b/debug_toolbar/toolbar/loader.py
index 02a558d..5701118 100644
--- a/debug_toolbar/toolbar/loader.py
+++ b/debug_toolbar/toolbar/loader.py
@@ -26,6 +26,7 @@ class DebugToolbar(object):
self.template_context = {
'BASE_URL': base_url, # for backwards compatibility
'DEBUG_TOOLBAR_MEDIA_URL': self.config.get('MEDIA_URL'),
+ 'STATIC_URL': settings.STATIC_URL,
}
self.load_panels()
@@ -55,13 +56,9 @@ class DebugToolbar(object):
"""
Renders the overall Toolbar with panels inside.
"""
- media_path = os.path.join(os.path.dirname(__file__), os.pardir, 'media', 'debug_toolbar')
-
context = self.template_context.copy()
context.update({
'panels': self.panels,
- 'js': mark_safe(open(os.path.join(media_path, 'js', 'toolbar.min.js'), 'r').read()),
- 'css': mark_safe(open(os.path.join(media_path, 'css', 'toolbar.min.css'), 'r').read()),
})
return render_to_string('debug_toolbar/base.html', context)
diff --git a/debug_toolbar/urls.py b/debug_toolbar/urls.py
index 986b78e..0053ae7 100644
--- a/debug_toolbar/urls.py
+++ b/debug_toolbar/urls.py
@@ -4,12 +4,14 @@ 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__'
urlpatterns = patterns('',
- url(r'^%s/m/(.*)$' % _PREFIX, 'debug_toolbar.views.debug_media'),
url(r'^%s/sql_select/$' % _PREFIX, 'debug_toolbar.views.sql_select', name='sql_select'),
url(r'^%s/sql_explain/$' % _PREFIX, 'debug_toolbar.views.sql_explain', name='sql_explain'),
url(r'^%s/sql_profile/$' % _PREFIX, 'debug_toolbar.views.sql_profile', name='sql_profile'),
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 4d87090..0ff3359 100644
--- a/debug_toolbar/utils/tracking/db.py
+++ b/debug_toolbar/utils/tracking/db.py
@@ -5,13 +5,22 @@ 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 django.utils.hashcompat import sha_constructor
from debug_toolbar.utils import ms_from_timedelta, tidy_stacktrace, \
get_template_info, get_stack
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: # 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
# get a copy of the toolbar object with access to its config dictionary
SQL_WARNING_THRESHOLD = getattr(settings, 'DEBUG_TOOLBAR_CONFIG', {}) \
@@ -82,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,
@@ -98,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
@@ -119,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]
@@ -134,7 +154,7 @@ class NormalCursorWrapper(object):
'duration': duration,
'raw_sql': sql,
'params': _params,
- 'hash': sha_constructor(settings.SECRET_KEY \
+ 'hash': sha1(settings.SECRET_KEY \
+ smart_str(sql) \
+ _params).hexdigest(),
'stacktrace': stacktrace,
@@ -146,10 +166,17 @@ class NormalCursorWrapper(object):
}
if engine == 'psycopg2':
+ # 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,
+ 'iso_level': iso_level,
'encoding': conn.encoding,
})
diff --git a/debug_toolbar/views.py b/debug_toolbar/views.py
index 3fa809a..4b4ebc9 100644
--- a/debug_toolbar/views.py
+++ b/debug_toolbar/views.py
@@ -9,11 +9,19 @@ import django.views.static
from django.conf import settings
from django.http import HttpResponseBadRequest
from django.shortcuts import render_to_response
-from django.utils import simplejson
-from django.utils.hashcompat import sha_constructor
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: # python < 2.5
+ from django.utils.hashcompat import sha_constructor as sha1
+
class InvalidSQLError(Exception):
def __init__(self, value):
@@ -23,14 +31,6 @@ class InvalidSQLError(Exception):
return repr(self.value)
-def debug_media(request, path):
- root = getattr(settings, 'DEBUG_TOOLBAR_MEDIA_ROOT', None)
- if root is None:
- parent = os.path.abspath(os.path.dirname(__file__))
- root = os.path.join(parent, 'media', 'debug_toolbar')
- return django.views.static.serve(request, path, root)
-
-
def sql_select(request):
"""
Returns the output of the SQL SELECT statement.
@@ -45,11 +45,11 @@ def sql_select(request):
sql = request.GET.get('sql', '')
params = request.GET.get('params', '')
alias = request.GET.get('alias', 'default')
- hash = sha_constructor(settings.SECRET_KEY + sql + params).hexdigest()
+ hash = sha1(settings.SECRET_KEY + sql + params).hexdigest()
if hash != request.GET.get('hash', ''):
return HttpResponseBadRequest('Tamper alert') # SQL Tampering alert
if sql.lower().strip().startswith('select'):
- params = simplejson.loads(params)
+ params = json.loads(params)
cursor = connections[alias].cursor()
cursor.execute(sql, params)
headers = [d[0] for d in cursor.description]
@@ -80,11 +80,11 @@ def sql_explain(request):
sql = request.GET.get('sql', '')
params = request.GET.get('params', '')
alias = request.GET.get('alias', 'default')
- hash = sha_constructor(settings.SECRET_KEY + sql + params).hexdigest()
+ hash = sha1(settings.SECRET_KEY + sql + params).hexdigest()
if hash != request.GET.get('hash', ''):
return HttpResponseBadRequest('Tamper alert') # SQL Tampering alert
if sql.lower().strip().startswith('select'):
- params = simplejson.loads(params)
+ params = json.loads(params)
cursor = connections[alias].cursor()
conn = connections[alias].connection
@@ -95,6 +95,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)
@@ -126,11 +128,11 @@ def sql_profile(request):
sql = request.GET.get('sql', '')
params = request.GET.get('params', '')
alias = request.GET.get('alias', 'default')
- hash = sha_constructor(settings.SECRET_KEY + sql + params).hexdigest()
+ hash = sha1(settings.SECRET_KEY + sql + params).hexdigest()
if hash != request.GET.get('hash', ''):
return HttpResponseBadRequest('Tamper alert') # SQL Tampering alert
if sql.lower().strip().startswith('select'):
- params = simplejson.loads(params)
+ params = json.loads(params)
cursor = connections[alias].cursor()
result = None
headers = None
diff --git a/example/settings.py b/example/settings.py
index c421a62..6fa3e56 100644
--- a/example/settings.py
+++ b/example/settings.py
@@ -17,6 +17,8 @@ INSTALLED_APPS = (
INTERNAL_IPS = ('127.0.0.1',)
MEDIA_ROOT = os.path.join(PROJECT_PATH, 'media')
MEDIA_URL = '/media'
+STATIC_ROOT = os.path.join(HOME_ROOT, 'staticfiles')
+STATIC_URL = '/static/'
MIDDLEWARE_CLASSES = (
'django.middleware.common.CommonMiddleware',
'django.contrib.sessions.middleware.SessionMiddleware',
diff --git a/example/media/js/jquery.js b/example/static/js/jquery.js
index fa5132b..fa5132b 100644
--- a/example/media/js/jquery.js
+++ b/example/static/js/jquery.js
diff --git a/example/media/js/mootools.js b/example/static/js/mootools.js
index 0098571..0098571 100644
--- a/example/media/js/mootools.js
+++ b/example/static/js/mootools.js
diff --git a/example/media/js/prototype.js b/example/static/js/prototype.js
index 9fe6e12..9fe6e12 100644
--- a/example/media/js/prototype.js
+++ b/example/static/js/prototype.js
diff --git a/example/templates/jquery/index.html b/example/templates/jquery/index.html
index 61b3cbe..9cab242 100644
--- a/example/templates/jquery/index.html
+++ b/example/templates/jquery/index.html
@@ -6,7 +6,7 @@
.hide {display:none;}
#v {font-weight:bold;}
</style>
- <script type="text/javascript" charset="utf-8" src="{{ MEDIA_URL }}/js/jquery.js"></script>
+ <script type="text/javascript" charset="utf-8" src="{{ STATIC_URL }}/js/jquery.js"></script>
<script type="text/javascript">
$(document).ready(function() {
$('p.hide').show();
diff --git a/example/templates/mootools/index.html b/example/templates/mootools/index.html
index fa12abe..dde019d 100644
--- a/example/templates/mootools/index.html
+++ b/example/templates/mootools/index.html
@@ -5,7 +5,7 @@
<style>
.hide {display:none;}
</style>
- <script type="text/javascript" charset="utf-8" src="{{ MEDIA_URL }}/js/mootools.js"></script>
+ <script type="text/javascript" charset="utf-8" src="{{ STATIC_URL }}/js/mootools.js"></script>
<script type="text/javascript">
window.addEvent('domready', function() {
$$('p.hide').setStyle('display', 'block');
diff --git a/example/templates/prototype/index.html b/example/templates/prototype/index.html
index bcc3141..bf3d0a5 100644
--- a/example/templates/prototype/index.html
+++ b/example/templates/prototype/index.html
@@ -5,7 +5,7 @@
<style>
.hide {display:none;}
</style>
- <script type="text/javascript" charset="utf-8" src="{{ MEDIA_URL }}/js/prototype.js"></script>
+ <script type="text/javascript" charset="utf-8" src="{{ STATIC_URL }}/js/prototype.js"></script>
<script type="text/javascript">
document.observe('dom:loaded', function() {
$('showme').removeClassName('hide');
diff --git a/example/urls.py b/example/urls.py
index 25acc31..d8bed27 100644
--- a/example/urls.py
+++ b/example/urls.py
@@ -1,15 +1,15 @@
from django.conf import settings
from django.conf.urls.defaults import *
from django.contrib import admin
-from django.views.generic.simple import direct_to_template
+from django.views.generic import TemplateView
admin.autodiscover()
urlpatterns = patterns('',
- (r'^$', direct_to_template, {'template': 'index.html'}),
- (r'^jquery/index/$', direct_to_template, {'template': 'jquery/index.html'}),
- (r'^mootools/index/$', direct_to_template, {'template': 'mootools/index.html'}),
- (r'^prototype/index/$', direct_to_template, {'template': 'prototype/index.html'}),
+ (r'^$', TemplateView.as_view(template_name='index.html')),
+ (r'^jquery/index/$', TemplateView.as_view(template_name='jquery/index.html')),
+ (r'^mootools/index/$', TemplateView.as_view(template_name='mootools/index.html')),
+ (r'^prototype/index/$', TemplateView.as_view(template_name='prototype/index.html')),
(r'^admin/', include(admin.site.urls)),
)
diff --git a/runtests.py b/runtests.py
index 3432f8b..ed49824 100644
--- a/runtests.py
+++ b/runtests.py
@@ -1,11 +1,14 @@
#!/usr/bin/env python
import sys
+import os
from os.path import dirname, abspath
from optparse import OptionParser
from django.conf import settings, global_settings
-if not settings.configured:
+# For convenience configure settings if they are not pre-configured or if we
+# haven't been provided settings to use by environment variable.
+if not settings.configured and not os.environ.get('DJANGO_SETTINGS_MODULE'):
settings.configure(
DATABASES={
'default': {
diff --git a/test_pgsql.py b/test_pgsql.py
new file mode 100644
index 0000000..28c0178
--- /dev/null
+++ b/test_pgsql.py
@@ -0,0 +1,28 @@
+from django.conf import global_settings
+DATABASES={
+ 'default': {
+ 'ENGINE': 'django.db.backends.postgresql_psycopg2',
+ # Edit the below settings before use...
+ 'USER': '',
+ 'NAME': '',
+ 'HOST': '',
+ 'PASSWORD': '',
+ }
+}
+INSTALLED_APPS=[
+ 'django.contrib.auth',
+ 'django.contrib.admin',
+ 'django.contrib.contenttypes',
+ 'django.contrib.sessions',
+ 'django.contrib.sites',
+
+ 'debug_toolbar',
+
+ 'tests',
+]
+MIDDLEWARE_CLASSES=global_settings.MIDDLEWARE_CLASSES + (
+ 'debug_toolbar.middleware.DebugToolbarMiddleware',
+)
+ROOT_URLCONF=''
+DEBUG=False
+SITE_ID=1
diff --git a/tests/tests.py b/tests/tests.py
index 32ea9d9..f76f1ab 100644
--- a/tests/tests.py
+++ b/tests/tests.py
@@ -1,10 +1,13 @@
+from __future__ import with_statement
import thread
from django.conf import settings
from django.contrib.auth.models import User
+from django.db import connection
from django.http import HttpResponse
from django.test import TestCase, RequestFactory
from django.template import Template, Context
+from django.utils import unittest
from debug_toolbar.middleware import DebugToolbarMiddleware
from debug_toolbar.panels.sql import SQLDebugPanel
@@ -103,8 +106,6 @@ class DebugToolbarTestCase(BaseTestCase):
self.assertFalse(isinstance(request.urlconf, basestring))
- self.assertTrue(hasattr(request.urlconf.urlpatterns[0], '_callback_str'))
- self.assertEquals(request.urlconf.urlpatterns[0]._callback_str, 'debug_toolbar.views.debug_media')
self.assertTrue(hasattr(request.urlconf.urlpatterns[1], '_callback_str'))
self.assertEquals(request.urlconf.urlpatterns[-1]._callback_str, 'tests.views.execute_sql')
@@ -120,8 +121,6 @@ class DebugToolbarTestCase(BaseTestCase):
self.assertFalse(isinstance(request.urlconf, basestring))
- self.assertTrue(hasattr(request.urlconf.urlpatterns[0], '_callback_str'))
- self.assertEquals(request.urlconf.urlpatterns[0]._callback_str, 'debug_toolbar.views.debug_media')
self.assertTrue(hasattr(request.urlconf.urlpatterns[1], '_callback_str'))
self.assertEquals(request.urlconf.urlpatterns[-1]._callback_str, 'tests.views.execute_sql')
@@ -135,8 +134,6 @@ class DebugToolbarTestCase(BaseTestCase):
self.assertFalse(isinstance(request.urlconf, basestring))
- self.assertTrue(hasattr(request.urlconf.urlpatterns[0], '_callback_str'))
- self.assertEquals(request.urlconf.urlpatterns[0]._callback_str, 'debug_toolbar.views.debug_media')
self.assertTrue(hasattr(request.urlconf.urlpatterns[1], '_callback_str'))
self.assertEquals(request.urlconf.urlpatterns[-1]._callback_str, 'tests.views.execute_sql')
@@ -220,6 +217,19 @@ class SQLPanelTestCase(BaseTestCase):
# ensure the stacktrace is populated
self.assertTrue(len(query[1]['stacktrace']) > 0)
+ @unittest.skipUnless(connection.vendor=='postgresql',
+ 'Test valid only on PostgreSQL')
+ def test_erroneous_query(self):
+ """
+ Test that an error in the query isn't swallowed by the middleware.
+ """
+ from django.db import connection
+ from django.db.utils import DatabaseError
+ try:
+ connection.cursor().execute("erroneous query")
+ except DatabaseError as e:
+ self.assertTrue('erroneous query' in str(e))
+
def test_disable_stacktraces(self):
panel = self.toolbar.get_panel(SQLDebugPanel)
self.assertEquals(len(panel._queries), 0)
diff --git a/tests/urls.py b/tests/urls.py
index a556703..778f417 100644
--- a/tests/urls.py
+++ b/tests/urls.py
@@ -4,14 +4,17 @@ 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 *
from django.contrib import admin
+try:
+ from django.conf.urls import patterns, url
+except ImportError: # django < 1.4
+ from django.conf.urls.defaults import patterns, url
admin.autodiscover()
urlpatterns = patterns('',
# This pattern should be last to ensure tests still work
- url(r'^resolving1/(.+)/(.+)/$', 'tests.views.resolving_view', name = 'positional-resolving'),
+ url(r'^resolving1/(.+)/(.+)/$', 'tests.views.resolving_view', name='positional-resolving'),
url(r'^resolving2/(?P<arg1>.+)/(?P<arg2>.+)/$', 'tests.views.resolving_view'),
url(r'^resolving3/(.+)/$', 'tests.views.resolving_view', { 'arg2' : 'default' }),
url(r'^execute_sql/$', 'tests.views.execute_sql'),