1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94 | """
$Id: SQLLogMiddleware.py 306 2007-10-22 14:55:47Z tguettler $
This middleware
in settings.py you need to set
DEBUG=True
DEBUG_SQL=True
# Since you can't see the output if the page results in a redirect,
# you can log the result into a directory:
# DEBUG_SQL='/mypath/...'
MIDDLEWARE_CLASSES = (
'YOURPATH.SQLLogMiddleware.SQLLogMiddleware',
'django.middleware.transaction.TransactionMiddleware',
...)
"""
# Python
import os
import time
import datetime
# Django
from django.conf import settings
from django.db import connection
from django.template import Template, Context
class SQLLogMiddleware:
start=None
def process_request(self, request):
self.start=time.time()
def process_response (self, request, response):
# self.start is empty if an append slash redirect happened.
debug_sql=getattr(settings, "DEBUG_SQL", False)
if (not self.start) or not (settings.DEBUG and debug_sql):
return response
timesql=0.0
for q in connection.queries:
timesql+=float(q['time'])
seen={}
duplicate=0
for q in connection.queries:
sql=q["sql"]
c=seen.get(sql, 0)
if c:
duplicate+=1
q["seen"]=c
seen[sql]=c+1
t = Template('''
<p>
<em>request.path:</em> {{ request.path|escape }}<br />
<em>Total query count:</em> {{ queries|length }}<br/>
<em>Total duplicate query count:</em> {{ duplicate }}<br/>
<em>Total SQL execution time:</em> {{ timesql }}<br/>
<em>Total Request execution time:</em> {{ timerequest }}<br/>
</p>
<table class="sqllog">
<tr>
<th>Time</th>
<th>Seen</th>
<th>SQL</th>
</tr>
{% for sql in queries %}
<tr>
<td>{{ sql.time }}</td>
<td align="right">{{ sql.seen }}</td>
<td>{{ sql.sql }}</td>
</tr>
{% endfor %}
</table>
''')
timerequest=round(time.time()-self.start, 3)
queries=connection.queries
html=t.render(Context(locals()))
if debug_sql==True:
if response.get("content-type", "").startswith("text/html"):
response.write(html)
return response
assert os.path.isdir(debug_sql), debug_sql
outfile=os.path.join(debug_sql, "%s.html" % datetime.datetime.now().isoformat())
fd=open(outfile, "wt")
fd.write('''<html><head><title>SQL Log %s</title></head><body>%s</body></html>''' % (
request.path, html))
fd.close()
return response
|
Comments
Hmmm...As I know middleware instance shared between working threads, so
startattribute may not be correspond to this request. I think thatstartmust be assigned to request object.#
Yes, you are right. Saving state information in the middleware objects is not good. This flag should be set on the request objects. I will update this snippet sometime.
#