~bzr-pqm/bzr/bzr.dev

« back to all changes in this revision

Viewing changes to bzrlib/tests/test_store.py

(vila) Calling super() instead of mentioning the base class in setUp avoid
 mistakes. (Vincent Ladeuil)

Show diffs side-by-side

added added

removed removed

Lines of Context:
1
 
# Copyright (C) 2005 by Canonical Development Ltd
2
 
 
 
1
# Copyright (C) 2005-2009, 2011 Canonical Ltd
 
2
#
3
3
# This program is free software; you can redistribute it and/or modify
4
4
# it under the terms of the GNU General Public License as published by
5
5
# the Free Software Foundation; either version 2 of the License, or
6
6
# (at your option) any later version.
7
 
 
 
7
#
8
8
# This program is distributed in the hope that it will be useful,
9
9
# but WITHOUT ANY WARRANTY; without even the implied warranty of
10
10
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
11
11
# GNU General Public License for more details.
12
 
 
 
12
#
13
13
# You should have received a copy of the GNU General Public License
14
14
# along with this program; if not, write to the Free Software
15
 
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
 
15
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
16
16
 
17
17
"""Test Store implementations."""
18
18
 
21
21
import gzip
22
22
 
23
23
import bzrlib.errors as errors
24
 
from bzrlib.errors import BzrError, UnlistableStore, NoSuchFile
25
 
from bzrlib.transport.local import LocalTransport
 
24
from bzrlib.errors import BzrError
 
25
from bzrlib.store import TransportStore
26
26
from bzrlib.store.text import TextStore
 
27
from bzrlib.store.versioned import VersionedFileStore
27
28
from bzrlib.tests import TestCase, TestCaseInTempDir, TestCaseWithTransport
28
 
import bzrlib.store as store
29
29
import bzrlib.transactions as transactions
30
30
import bzrlib.transport as transport
31
31
from bzrlib.transport.memory import MemoryTransport
 
32
from bzrlib.weave import WeaveFile
32
33
 
33
34
 
34
35
class TestStores(object):
48
49
        """Test copying"""
49
50
        os.mkdir('a')
50
51
        store_a = self.get_store('a')
51
 
        store_a.add('foo', '1')
 
52
        store_a.add(StringIO('foo'), '1')
52
53
        os.mkdir('b')
53
54
        store_b = self.get_store('b')
54
55
        store_b.copy_all_ids(store_a)
63
64
    def test_get(self):
64
65
        store = self.get_store()
65
66
        self.fill_store(store)
66
 
    
 
67
 
67
68
        self.check_content(store, 'a', 'hello')
68
69
        self.check_content(store, 'b', 'other')
69
70
        self.check_content(store, 'c', 'something')
70
 
    
 
71
 
71
72
        # Make sure that requesting a non-existing file fails
72
73
        self.assertRaises(KeyError, self.check_content, store, 'd', None)
73
74
 
81
82
class TestCompressedTextStore(TestCaseInTempDir, TestStores):
82
83
 
83
84
    def get_store(self, path=u'.'):
84
 
        t = LocalTransport(path)
 
85
        t = transport.get_transport_from_path(path)
85
86
        return TextStore(t, compressed=True)
86
87
 
87
88
    def test_total_size(self):
91
92
        store.add(StringIO('goodbye2'), '123123', 'dsc')
92
93
        # these get gzipped - content should be stable
93
94
        self.assertEqual(store.total_size(), (2, 55))
94
 
        
 
95
 
95
96
    def test__relpath_suffixed(self):
96
97
        my_store = TextStore(MockTransport(),
97
98
                             prefixed=True, compressed=True)
100
101
 
101
102
 
102
103
class TestMemoryStore(TestCase):
103
 
    
 
104
 
104
105
    def get_store(self):
105
 
        return store.ImmutableMemoryStore()
106
 
    
107
 
    def test_imports(self):
108
 
        from bzrlib.store import ImmutableMemoryStore
 
106
        return TextStore(MemoryTransport())
109
107
 
110
108
    def test_add_and_retrieve(self):
111
109
        store = self.get_store()
118
116
 
119
117
    def test_missing_is_absent(self):
120
118
        store = self.get_store()
121
 
        self.failIf('aa' in store)
 
119
        self.assertFalse('aa' in store)
122
120
 
123
121
    def test_adding_fails_when_present(self):
124
122
        my_store = self.get_store()
141
139
class TestTextStore(TestCaseInTempDir, TestStores):
142
140
 
143
141
    def get_store(self, path=u'.'):
144
 
        t = LocalTransport(path)
 
142
        t = transport.get_transport_from_path(path)
145
143
        return TextStore(t, compressed=False)
146
144
 
147
145
    def test_total_size(self):
159
157
class TestMixedTextStore(TestCaseInTempDir, TestStores):
160
158
 
161
159
    def get_store(self, path=u'.', compressed=True):
162
 
        t = LocalTransport(path)
 
160
        t = transport.get_transport_from_path(path)
163
161
        return TextStore(t, compressed=compressed)
164
162
 
165
163
    def test_get_mixed(self):
167
165
        s = self.get_store(u'.', compressed=False)
168
166
        cs.add(StringIO('hello there'), 'a')
169
167
 
170
 
        self.failUnlessExists('a.gz')
171
 
        self.failIf(os.path.lexists('a'))
 
168
        self.assertPathExists('a.gz')
 
169
        self.assertFalse(os.path.lexists('a'))
172
170
 
173
171
        self.assertEquals(gzip.GzipFile('a.gz').read(), 'hello there')
174
172
 
176
174
        self.assertEquals(s.has_id('a'), True)
177
175
        self.assertEquals(cs.get('a').read(), 'hello there')
178
176
        self.assertEquals(s.get('a').read(), 'hello there')
179
 
        
 
177
 
180
178
        self.assertRaises(BzrError, s.add, StringIO('goodbye'), 'a')
181
179
 
182
180
        s.add(StringIO('goodbye'), 'b')
183
 
        self.failUnlessExists('b')
184
 
        self.failIf(os.path.lexists('b.gz'))
 
181
        self.assertPathExists('b')
 
182
        self.assertFalse(os.path.lexists('b.gz'))
185
183
        self.assertEquals(open('b').read(), 'goodbye')
186
184
 
187
185
        self.assertEquals(cs.has_id('b'), True)
188
186
        self.assertEquals(s.has_id('b'), True)
189
187
        self.assertEquals(cs.get('b').read(), 'goodbye')
190
188
        self.assertEquals(s.get('b').read(), 'goodbye')
191
 
        
 
189
 
192
190
        self.assertRaises(BzrError, cs.add, StringIO('again'), 'b')
193
191
 
194
192
class MockTransport(transport.Transport):
206
204
        return
207
205
 
208
206
 
209
 
class InstrumentedTransportStore(store.TransportStore):
 
207
class InstrumentedTransportStore(TransportStore):
210
208
    """An instrumented TransportStore.
211
209
 
212
210
    Here we replace template method worker methods with calls that record the
232
230
class TestMockTransport(TestCase):
233
231
 
234
232
    def test_isinstance(self):
235
 
        self.failUnless(isinstance(MockTransport(), transport.Transport))
 
233
        self.assertIsInstance(MockTransport(), transport.Transport)
236
234
 
237
235
    def test_has(self):
238
236
        self.assertEqual(False, MockTransport().has('foo'))
242
240
 
243
241
 
244
242
class TestTransportStore(TestCase):
245
 
    
 
243
 
246
244
    def test__relpath_invalid(self):
247
 
        my_store = store.TransportStore(MockTransport())
 
245
        my_store = TransportStore(MockTransport())
248
246
        self.assertRaises(ValueError, my_store._relpath, '/foo')
249
247
        self.assertRaises(ValueError, my_store._relpath, 'foo/')
250
248
 
251
249
    def test_register_invalid_suffixes(self):
252
 
        my_store = store.TransportStore(MockTransport())
 
250
        my_store = TransportStore(MockTransport())
253
251
        self.assertRaises(ValueError, my_store.register_suffix, '/')
254
252
        self.assertRaises(ValueError, my_store.register_suffix, '.gz/bar')
255
253
 
256
254
    def test__relpath_unregister_suffixes(self):
257
 
        my_store = store.TransportStore(MockTransport())
 
255
        my_store = TransportStore(MockTransport())
258
256
        self.assertRaises(ValueError, my_store._relpath, 'foo', ['gz'])
259
257
        self.assertRaises(ValueError, my_store._relpath, 'foo', ['dsc', 'gz'])
260
258
 
261
259
    def test__relpath_simple(self):
262
 
        my_store = store.TransportStore(MockTransport())
 
260
        my_store = TransportStore(MockTransport())
263
261
        self.assertEqual("foo", my_store._relpath('foo'))
264
262
 
265
263
    def test__relpath_prefixed(self):
266
 
        my_store = store.TransportStore(MockTransport(), True)
 
264
        my_store = TransportStore(MockTransport(), True)
267
265
        self.assertEqual('45/foo', my_store._relpath('foo'))
268
266
 
269
267
    def test__relpath_simple_suffixed(self):
270
 
        my_store = store.TransportStore(MockTransport())
 
268
        my_store = TransportStore(MockTransport())
271
269
        my_store.register_suffix('bar')
272
270
        my_store.register_suffix('baz')
273
271
        self.assertEqual('foo.baz', my_store._relpath('foo', ['baz']))
274
272
        self.assertEqual('foo.bar.baz', my_store._relpath('foo', ['bar', 'baz']))
275
273
 
276
274
    def test__relpath_prefixed_suffixed(self):
277
 
        my_store = store.TransportStore(MockTransport(), True)
 
275
        my_store = TransportStore(MockTransport(), True)
278
276
        my_store.register_suffix('bar')
279
277
        my_store.register_suffix('baz')
280
278
        self.assertEqual('45/foo.baz', my_store._relpath('foo', ['baz']))
299
297
        my_store.register_suffix('dsc')
300
298
        my_store.add(stream, "foo", 'dsc')
301
299
        self.assertEqual([("_add", "foo.dsc", stream)], my_store._calls)
302
 
        
 
300
 
303
301
    def test_add_simple_suffixed(self):
304
302
        stream = StringIO("content")
305
303
        my_store = InstrumentedTransportStore(MockTransport(), True)
319
317
        stream = StringIO("signature for missing base")
320
318
        my_store.add(stream, "missing", 'sig')
321
319
        return my_store
322
 
        
 
320
 
323
321
    def test_has_simple(self):
324
322
        my_store = self.get_populated_store()
325
323
        self.assertEqual(True, my_store.has_id('foo'))
396
394
        self.assertRaises(KeyError, to_store.get, 'missing', 'sig')
397
395
 
398
396
    def test_relpath_escaped(self):
399
 
        my_store = store.TransportStore(MemoryTransport())
 
397
        my_store = TransportStore(MemoryTransport())
400
398
        self.assertEqual('%25', my_store._relpath('%'))
401
399
 
 
400
    def test_escaped_uppercase(self):
 
401
        """Uppercase letters are escaped for safety on Windows"""
 
402
        my_store = TransportStore(MemoryTransport(), prefixed=True,
 
403
            escaped=True)
 
404
        # a particularly perverse file-id! :-)
 
405
        self.assertEquals(my_store._relpath('C:<>'), 'be/%2543%253a%253c%253e')
 
406
 
402
407
 
403
408
class TestVersionFileStore(TestCaseWithTransport):
404
409
 
 
410
    def get_scope(self):
 
411
        return self._transaction
 
412
 
405
413
    def setUp(self):
406
414
        super(TestVersionFileStore, self).setUp()
407
 
        self.vfstore = store.versioned.VersionedFileStore(MemoryTransport())
 
415
        self.vfstore = VersionedFileStore(MemoryTransport(),
 
416
            versionedfile_class=WeaveFile)
 
417
        self.vfstore.get_scope = self.get_scope
 
418
        self._transaction = None
408
419
 
409
420
    def test_get_weave_registers_dirty_in_write(self):
410
 
        transaction = transactions.WriteTransaction()
411
 
        vf = self.vfstore.get_weave_or_empty('id', transaction)
412
 
        transaction.finish()
413
 
        self.assertRaises(errors.OutSideTransaction, vf.add_lines, 'b', [], [])
414
 
        transaction = transactions.WriteTransaction()
415
 
        vf = self.vfstore.get_weave('id', transaction)
416
 
        transaction.finish()
417
 
        self.assertRaises(errors.OutSideTransaction, vf.add_lines, 'b', [], [])
418
 
 
419
 
    def test_get_weave_or_empty_readonly_fails(self):
420
 
        transaction = transactions.ReadOnlyTransaction()
421
 
        vf = self.assertRaises(errors.ReadOnlyError,
422
 
                               self.vfstore.get_weave_or_empty,
423
 
                               'id',
424
 
                               transaction)
 
421
        self._transaction = transactions.WriteTransaction()
 
422
        vf = self.vfstore.get_weave_or_empty('id', self._transaction)
 
423
        self._transaction.finish()
 
424
        self._transaction = None
 
425
        self.assertRaises(errors.OutSideTransaction, vf.add_lines, 'b', [], [])
 
426
        self._transaction = transactions.WriteTransaction()
 
427
        vf = self.vfstore.get_weave('id', self._transaction)
 
428
        self._transaction.finish()
 
429
        self._transaction = None
 
430
        self.assertRaises(errors.OutSideTransaction, vf.add_lines, 'b', [], [])
425
431
 
426
432
    def test_get_weave_readonly_cant_write(self):
427
 
        transaction = transactions.WriteTransaction()
428
 
        vf = self.vfstore.get_weave_or_empty('id', transaction)
429
 
        transaction.finish()
430
 
        transaction = transactions.ReadOnlyTransaction()
431
 
        vf = self.vfstore.get_weave_or_empty('id', transaction)
 
433
        self._transaction = transactions.WriteTransaction()
 
434
        vf = self.vfstore.get_weave_or_empty('id', self._transaction)
 
435
        self._transaction.finish()
 
436
        self._transaction = transactions.ReadOnlyTransaction()
 
437
        vf = self.vfstore.get_weave_or_empty('id', self._transaction)
432
438
        self.assertRaises(errors.ReadOnlyError, vf.add_lines, 'b', [], [])
433
439
 
 
440
    def test___iter__escaped(self):
 
441
        self.vfstore = VersionedFileStore(MemoryTransport(),
 
442
            prefixed=True, escaped=True, versionedfile_class=WeaveFile)
 
443
        self.vfstore.get_scope = self.get_scope
 
444
        self._transaction = transactions.WriteTransaction()
 
445
        vf = self.vfstore.get_weave_or_empty(' ', self._transaction)
 
446
        vf.add_lines('a', [], [])
 
447
        del vf
 
448
        self._transaction.finish()
 
449
        self.assertEqual([' '], list(self.vfstore))