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
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
|
#!${buildout:executable} -S
##############################################################################
#
# Copyright (c) 2004 Zope Corporation and Contributors.
# All Rights Reserved.
#
# This software is subject to the provisions of the Zope Public License,
# Version 2.1 (ZPL). A copy of the ZPL should accompany this distribution.
# THIS SOFTWARE IS PROVIDED "AS IS" AND ANY AND ALL EXPRESS OR IMPLIED
# WARRANTIES ARE DISCLAIMED, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
# WARRANTIES OF TITLE, MERCHANTABILITY, AGAINST INFRINGEMENT, AND FITNESS
# FOR A PARTICULAR PURPOSE.
#
##############################################################################
"""Test script
"""
# NOTE: This is a generated file. The original is in
# buildout-templates/bin/test.in
import logging, os, re, sys, time, warnings
# Initialize our paths.
${python-relative-path-setup}
import sys
sys.path.insert(0, ${scripts:parts-directory|path-repr})
import site
# Fix doctest so that it can handle mixed unicode and encoded output.
import doctest
_RealSpoofOut = doctest._SpoofOut
class _SpoofOut(doctest._SpoofOut):
def write(self, value):
if isinstance(value, unicode):
value = value.encode('utf8')
_RealSpoofOut.write(self, value)
doctest._SpoofOut = _SpoofOut
BUILD_DIR = ${buildout:directory|path-repr}
CUSTOM_SITE_DIR = ${scripts:parts-directory|path-repr}
# Make tests run in a timezone no launchpad developers live in.
# Our tests need to run in any timezone.
# (This is no longer actually required, as PQM does this.)
os.environ['TZ'] = 'Asia/Calcutta'
time.tzset()
# Storm's C extensions should already be enabled from lp_sitecustomize.py,
# which our custom sitecustomize.py ran.
assert os.environ['STORM_CEXTENSIONS'] == '1'
# Make sure our site.py is the one that subprocesses use.
os.environ['PYTHONPATH'] = CUSTOM_SITE_DIR
# Set a flag if this is the main testrunner process
if len(sys.argv) > 1 and sys.argv[1] == '--resume-layer':
main_process = False
else:
main_process = True
# Install the import fascist import hook and atexit handler.
from lp.scripts.utilities import importfascist
importfascist.install_import_fascist()
# Install the warning handler hook and atexit handler.
from lp.scripts.utilities import warninghandler
warninghandler.install_warning_handler()
# Ensure that atexit handlers are executed on TERM.
import signal
def exit_with_atexit_handlers(*ignored):
sys.exit(-1 * signal.SIGTERM)
signal.signal(signal.SIGTERM, exit_with_atexit_handlers)
# Tell canonical.config to use the testrunner config instance.
from canonical.config import config
config.setInstance('testrunner')
config.generate_overrides()
# Remove this module's directory from path, so that zope.testbrowser
# can import pystone from test:
sys.path[:] = [p for p in sys.path if os.path.abspath(p) != BUILD_DIR]
# Turn on psycopg debugging wrapper
#import canonical.database.debug
#canonical.database.debug.install()
# Unset the http_proxy environment variable, because we're going to make
# requests to localhost and we don't wand this to be proxied.
try:
os.environ.pop('http_proxy')
except KeyError:
pass
# Suppress accessability warning because the test runner does not have UI.
os.environ['GTK_MODULES'] = ''
# Silence spurious warnings. Note that this does not propagate to subprocesses
# so this is not always as easy as it seems. Warnings caused by our code that
# need to be silenced should have an accompanied Bug reference.
#
warnings.filterwarnings(
'ignore', 'PyCrypto', RuntimeWarning, 'twisted[.]conch[.]ssh',
)
warnings.filterwarnings(
'ignore', 'twisted.python.plugin', DeprecationWarning,
)
warnings.filterwarnings(
'ignore', 'zope.testing.doctest', DeprecationWarning,
)
warnings.filterwarnings(
'ignore', 'bzrlib.*was deprecated', DeprecationWarning,
)
# XXX: 2010-04-26, Salgado, bug=570246: Silence python2.6 deprecation
# warnings.
# We cannot narrow this warnings filter to just twisted because
# warnings.warn_explicit() sees this import as coming from importfascist, not
# from twisted. It makes no sense to put module='importfascist' here though
# because /everything/ gets imported through it. So, sad as it is, until
# twisted doesn't produce warnings under Python 2.6, just ignore all these
# deprecations.
warnings.filterwarnings(
'ignore', '.*(md5|sha|sets)', DeprecationWarning,
)
# The next one is caused by a lamosity in python-openid. The following change
# to openid/server/server.py would make the warning filter unnecessary:
# 978c974,974
# > try:
# > namespace = request.message.getOpenIDNamespace()
# > except AttributeError:
# > namespace = request.namespace
# > self.fields = Message(namespace)
# ---
# < self.fields = Message(request.namespace)
warnings.filterwarnings(
'ignore',
(r'The \"namespace\" attribute of CheckIDRequest objects is deprecated.\s+'
r'Use \"message.getOpenIDNamespace\(\)\" instead'),
DeprecationWarning
)
# This warning will be triggered if the beforeTraversal hook fails. We
# want to ensure it is not raised as an error, as this will mask the real
# problem.
warnings.filterwarnings(
'always',
re.escape('clear_request_started() called outside of a request'),
UserWarning
)
# Unicode warnings are always fatal
warnings.filterwarnings('error', category=UnicodeWarning)
# shortlist() raises an error when it is misused.
warnings.filterwarnings('error', r'shortlist\(\)')
from lp.testing import pgsql
# If this is removed, make sure lp.testing.pgsql is updated
# because the test harness there relies on the Connection wrapper being
# installed.
pgsql.installFakeConnect()
from zope.testing import testrunner
from zope.testing.testrunner import options
defaults = {
# Find tests in the tests and ftests directories
'tests_pattern': '^f?tests$',
'test_path': [${buildout:directory/lib|path-repr}],
'package': ['canonical', 'lp', 'devscripts', 'launchpad_loggerhead'],
'layer': ['!(MailmanLayer|WindmillLayer)'],
}
# Monkey-patch os.listdir to randomise the results
original_listdir = os.listdir
import random
def listdir(path):
"""Randomise the results of os.listdir.
It uses random.suffle to randomise os.listdir results, this way tests
relying on unstable ordering will have a higher chance to fail in the
development environment.
"""
directory_contents = original_listdir(path)
random.shuffle(directory_contents)
return directory_contents
os.listdir = listdir
from canonical.testing.customresult import filter_tests, patch_find_tests
if __name__ == '__main__':
# Extract arguments so we can see them too. We need to strip
# --resume-layer and --default stuff if found as get_options can't
# handle it.
if len(sys.argv) > 1 and sys.argv[1] == '--resume-layer':
args = list(sys.argv)
args.pop(1) # --resume-layer
args.pop(1) # The layer name
args.pop(1) # The resume number
while len(args) > 1 and args[1] == '--default':
args.pop(1) # --default
args.pop(1) # The default value
args.insert(0, sys.argv[0])
else:
args = sys.argv
# thunk across to parallel support if needed.
if '--parallel' in sys.argv and '--list-tests' not in sys.argv:
# thunk over to parallel testing.
from canonical.testing.parallel import main
sys.exit(main(sys.argv))
def load_list(option, opt_str, list_name, parser):
patch_find_tests(filter_tests(list_name))
options.parser.add_option(
'--load-list', type=str, action='callback', callback=load_list)
options.parser.add_option(
'--parallel', action='store_true',
help='Run tests in parallel processes. '
'Poorly isolated tests will break.')
# tests_pattern is a regexp, so the parsed value is hard to compare
# with the default value in the loop below.
options.parser.defaults['tests_pattern'] = defaults['tests_pattern']
local_options = options.get_options(args=args)
# Set our default options, if the options aren't specified.
for name, value in defaults.items():
parsed_option = getattr(local_options, name)
if ((parsed_option == []) or
(parsed_option == options.parser.defaults.get(name))):
# The option probably wasn't specified on the command line,
# let's replace it with our default value. It could be that
# the real default (as specified in
# zope.testing.testrunner.options) was specified, and we
# shouldn't replace it with our default, but it's such and
# edge case, so we don't have to care about it.
options.parser.defaults[name] = value
# Turn on Layer profiling if requested.
from canonical.testing import profiled
if local_options.verbose >= 3 and main_process:
profiled.setup_profiling()
# The working directory change is just so that the test script
# can be invoked from places other than the root of the source
# tree. This is very useful for IDE integration, so an IDE can
# e.g. run the test that you are currently editing.
try:
try:
there = os.getcwd()
os.chdir(BUILD_DIR)
testrunner.run([])
except SystemExit:
# Print Layer profiling report if requested.
if main_process and local_options.verbose >= 3:
profiled.report_profile_stats()
raise
finally:
os.chdir(there)
|