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
|
# Copyright 2011 Canonical Ltd. This software is licensed under the
# GNU Affero General Public License version 3 (see the file LICENSE).
"""Tests for lp.testing.fixture."""
__metaclass__ = type
import sys
import oops_amqp
import psycopg2
from storm.exceptions import DisconnectionError
from zope.component import (
adapts,
ComponentLookupError,
getGlobalSiteManager,
queryAdapter,
)
from zope.interface import (
implements,
Interface,
)
from zope.sendmail.interfaces import IMailDelivery
from canonical.config import (
config,
dbconfig,
)
from lp.services.database.lpstorm import IMasterStore
from canonical.launchpad.webapp.errorlog import (
globalErrorUtility,
notify_publisher,
)
from canonical.testing.layers import (
BaseLayer,
DatabaseLayer,
LaunchpadLayer,
LaunchpadZopelessLayer,
)
from lp.registry.model.person import Person
from lp.services.messaging import rabbit
from lp.testing import TestCase
from lp.testing.fixture import (
CaptureOops,
PGBouncerFixture,
ZopeAdapterFixture,
ZopeUtilityFixture,
)
class IFoo(Interface):
pass
class IBar(Interface):
pass
class Foo:
implements(IFoo)
class Bar:
implements(IBar)
class FooToBar:
adapts(IFoo)
implements(IBar)
def __init__(self, foo):
self.foo = foo
class TestZopeAdapterFixture(TestCase):
layer = BaseLayer
def test_register_and_unregister(self):
# Entering ZopeAdapterFixture's context registers the given adapter,
# and exiting the context unregisters the adapter again.
context = Foo()
# No adapter from Foo to Bar is registered.
self.assertIs(None, queryAdapter(context, IBar))
with ZopeAdapterFixture(FooToBar):
# Now there is an adapter from Foo to Bar.
adapter = queryAdapter(context, IBar)
self.assertIsNot(None, adapter)
self.assertIsInstance(adapter, FooToBar)
# The adapter is no longer registered.
self.assertIs(None, queryAdapter(context, IBar))
class DummyMailer(object):
implements(IMailDelivery)
class TestZopeUtilityFixture(TestCase):
layer = BaseLayer
def test_fixture(self):
def get_mailer():
return getGlobalSiteManager().getUtility(
IMailDelivery, 'Mail')
fake = DummyMailer()
# In BaseLayer there should be no mailer by default.
self.assertRaises(ComponentLookupError, get_mailer)
with ZopeUtilityFixture(fake, IMailDelivery, 'Mail'):
self.assertEquals(get_mailer(), fake)
self.assertRaises(ComponentLookupError, get_mailer)
class TestPGBouncerFixtureWithCA(TestCase):
"""PGBouncerFixture reconnect tests for Component Architecture layers.
Registered Storm Stores should be reconnected through pgbouncer.
"""
layer = LaunchpadZopelessLayer
def is_connected(self):
# First rollback any existing transaction to ensure we attempt
# to reconnect. We currently rollback the store explicitely
# rather than call transaction.abort() due to Bug #819282.
store = IMasterStore(Person)
store.rollback()
try:
store.find(Person).first()
return True
except DisconnectionError:
return False
def test_stop_and_start(self):
# Database is working.
assert self.is_connected()
# And database with the fixture is working too.
pgbouncer = PGBouncerFixture()
with PGBouncerFixture() as pgbouncer:
assert self.is_connected()
# pgbouncer is transparant. To confirm we are connecting via
# pgbouncer, we need to shut it down and confirm our
# connections are dropped.
pgbouncer.stop()
assert not self.is_connected()
# If we restart it, things should be back to normal.
pgbouncer.start()
assert self.is_connected()
# Database is still working.
assert self.is_connected()
def test_stop_no_start(self):
# Database is working.
assert self.is_connected()
# And database with the fixture is working too.
with PGBouncerFixture() as pgbouncer:
assert self.is_connected()
# pgbouncer is transparant. To confirm we are connecting via
# pgbouncer, we need to shut it down and confirm our
# connections are dropped.
pgbouncer.stop()
assert not self.is_connected()
# Database is working again.
assert self.is_connected()
class TestPGBouncerFixtureWithoutCA(TestCase):
"""PGBouncerFixture tests for non-Component Architecture layers."""
layer = DatabaseLayer
def is_db_available(self):
# Direct connection to the DB.
con_str = dbconfig.rw_main_master + ' user=launchpad_main'
try:
con = psycopg2.connect(con_str)
cur = con.cursor()
cur.execute("SELECT id FROM Person LIMIT 1")
con.close()
return True
except psycopg2.OperationalError:
return False
def test_install_fixture(self):
self.assert_(self.is_db_available())
with PGBouncerFixture() as pgbouncer:
self.assertTrue(self.is_db_available())
pgbouncer.stop()
self.assertFalse(self.is_db_available())
# This confirms that we are again connecting directly to the
# database, as the pgbouncer process was shutdown.
self.assertTrue(self.is_db_available())
def test_install_fixture_with_restart(self):
self.assert_(self.is_db_available())
with PGBouncerFixture() as pgbouncer:
self.assertTrue(self.is_db_available())
pgbouncer.stop()
self.assertFalse(self.is_db_available())
pgbouncer.start()
self.assertTrue(self.is_db_available())
# Note that because pgbouncer was left running, we can't confirm
# that we are now connecting directly to the database.
self.assertTrue(self.is_db_available())
class TestCaptureOopsNoRabbit(TestCase):
# Need CA for subscription.
layer = BaseLayer
def test_subscribes_to_events(self):
capture = self.useFixture(CaptureOops())
publishers = globalErrorUtility._oops_config.publishers[:]
try:
globalErrorUtility._oops_config.publishers[:] = [notify_publisher]
id = globalErrorUtility.raising(sys.exc_info())['id']
self.assertEqual(id, capture.oopses[0]['id'])
self.assertEqual(1, len(capture.oopses))
finally:
globalErrorUtility._oops_config.publishers[:] = publishers
class TestCaptureOopsRabbit(TestCase):
# Has rabbit + CA.
layer = LaunchpadLayer
def test_no_oopses_no_hang_on_sync(self):
capture = self.useFixture(CaptureOops())
capture.sync()
def test_sync_grabs_pending_oopses(self):
factory = rabbit.connect
exchange = config.error_reports.error_exchange
routing_key = config.error_reports.error_queue_key
capture = self.useFixture(CaptureOops())
amqp_publisher = oops_amqp.Publisher(
factory, exchange, routing_key, inherit_id=True)
oops = {'id': 'fnor', 'foo': 'dr'}
self.assertEqual('fnor', amqp_publisher(oops))
oops2 = {'id': 'quux', 'foo': 'strangelove'}
self.assertEqual('quux', amqp_publisher(oops2))
capture.sync()
self.assertEqual([oops, oops2], capture.oopses)
def test_sync_twice_works(self):
capture = self.useFixture(CaptureOops())
capture.sync()
capture.sync()
|