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
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
|
# Copyright 2009 Canonical Ltd. This software is licensed under the
# GNU Affero General Public License version 3 (see the file LICENSE).
"""Handlers for warnings, to be installed when testing."""
__metaclass__ = type
import atexit
import inspect
import StringIO
import sys
import warnings
class WarningReport:
def __init__(self, message, info):
self.message = message
self.info = info
def __str__(self):
info = str(self.info)
if info:
return info
else:
return self.message
class ImportantInfo:
def __init__(self, expressiontext, viewclassname, templatefilename,
requesturl, viewclassfilename, viewclasslineno, viewclassfunc,
doctestname, doctestline):
self.expressiontext = expressiontext
self.viewclassname = viewclassname
self.viewclassfilename = viewclassfilename
self.viewclasslineno = viewclasslineno
self.templatefilename = templatefilename
self.requesturl = requesturl
self.viewclassfunc = viewclassfunc
self.doctestname = doctestname
self.doctestline = doctestline
def __str__(self):
L = []
if self.expressiontext:
L.append('The expression: %s in %s' % (
self.expressiontext, self.templatefilename))
if self.viewclassname:
L.append('The method %s.%s' % (
self.viewclassname, self.viewclassfunc))
#L.append('at line %s of file %s' % (
# self.viewclasslineno, self.viewclassfilename)
if self.doctestname:
L.append("The doctest %s, at the line:" % self.doctestname)
L.append(" >>> %s" % self.doctestline)
if self.requesturl:
L.append('request url: %s' % self.requesturl)
return '\n'.join(L)
# PageTemplateFile has .filename.
from z3c.ptcompat import PageTemplateFile, ViewPageTemplateFile
# PythonExpr has .text, the text of the expression.
from zope.tales.pythonexpr import PythonExpr
# TrustedZopeContext has self.contexts, a dict with template, view, context,
# request, etc.
from zope.app.pagetemplate.engine import TrustedZopeContext
# TALInterpreter has self.sourceFile, a filename of a page template.
from zope.tal.talinterpreter import TALInterpreter
from zope.app.pagetemplate.simpleviewclass import simple
def find_important_info():
stack = inspect.stack()
try:
important_classes = set([
PythonExpr,
TrustedZopeContext,
TALInterpreter,
ViewPageTemplateFile,
simple
])
important_objects = {}
metadata = {} # cls -> (filename, lineno, funcname)
for frame, filename, lineno, func_name, context, lineidx in stack:
try:
if (filename.startswith('<doctest ') and
"doctest" not in important_objects):
# Very fragile inspection of the state of the doctest
# runner. So, enclosed in a try-except so it will at
# least fail gracefully if it fails.
try:
line = frame.f_back.f_locals['example'].source
except KeyboardInterrupt:
pass
except Exception:
line = "# cannot get line of code"
important_objects["doctest"] = (filename, line)
metadata["doctest"] = (filename, lineno, func_name)
if 'self' in frame.f_locals:
fself = frame.f_locals['self']
ftype = type(fself)
for cls in list(important_classes):
if isinstance(fself, cls):
important_objects[cls] = fself
metadata[cls] = (filename, lineno, func_name)
important_classes.remove(cls)
finally:
del frame
finally:
del stack
expressiontext = ''
if PythonExpr in important_objects:
expressiontext = important_objects[PythonExpr].text
viewclassname = ''
viewclassfilename = ''
viewclasslineno = ''
viewclassfunc = ''
doctestname = ''
doctestline = ''
if simple in important_objects:
cls = important_objects[simple].__class__
if cls is not simple:
viewclassname = cls.__mro__[1].__name__
viewclassfilename, viewclasslineno, viewclassfunc = (
metadata[simple])
templatefilename = ''
if ViewPageTemplateFile in important_objects:
templatefilename = important_objects[ViewPageTemplateFile].filename
templatefilename = templatefilename.split('/')[-1]
requesturl = ''
if TrustedZopeContext in important_objects:
ptcontexts = important_objects[TrustedZopeContext].contexts
requesturl = ptcontexts['request'].getURL()
if "doctest" in important_objects:
doctestname, doctestline = important_objects["doctest"]
return ImportantInfo(expressiontext, viewclassname, templatefilename,
requesturl, viewclassfilename, viewclasslineno, viewclassfunc,
doctestname, doctestline)
need_page_titles = []
no_order_by = []
# Maps (category, filename, lineno) to WarningReport
other_warnings = {}
old_show_warning = warnings.showwarning
def launchpad_showwarning(message, category, filename, lineno, file=None,
line=None):
if file is None:
file = sys.stderr
stream = StringIO.StringIO()
old_show_warning(message, category, filename, lineno, stream, line=line)
warning_message = stream.getvalue()
important_info = find_important_info()
if isinstance(message, UserWarning):
args = message.args
if args:
arg = args[0]
if arg.startswith('No page title in '):
global need_page_titles
need_page_titles.append(arg)
return
if arg == 'Getting a slice of an unordered set is unpredictable.':
# find the page template and view class, if any
# show these, plus the request.
global no_order_by
no_order_by.append(
WarningReport(warning_message, important_info)
)
return
other_warnings[(category, filename, lineno)] = WarningReport(
warning_message, important_info)
def report_need_page_titles():
global need_page_titles
if need_page_titles:
print
print "The following pages need titles."
for message in need_page_titles:
print " ", message
def report_no_order_by():
global no_order_by
if no_order_by:
print
print ("The following code has issues with"
" ambiguous select results ordering.")
for report in no_order_by:
print
print report
def report_other_warnings():
global other_warnings
if other_warnings:
print
print "General warnings."
for warninginfo in other_warnings.itervalues():
print
print warninginfo
def report_warnings():
report_need_page_titles()
report_no_order_by()
report_other_warnings()
def install_warning_handler():
warnings.showwarning = launchpad_showwarning
atexit.register(report_warnings)
|