1
# Copyright (C) 2006 Canonical Ltd
2
# Authors: Robert Collins <robert.collins@canonical.com>
4
# This program is free software; you can redistribute it and/or modify
5
# it under the terms of the GNU General Public License as published by
6
# the Free Software Foundation; either version 2 of the License, or
7
# (at your option) any later version.
9
# This program is distributed in the hope that it will be useful,
10
# but WITHOUT ANY WARRANTY; without even the implied warranty of
11
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12
# GNU General Public License for more details.
14
# You should have received a copy of the GNU General Public License
15
# along with this program; if not, write to the Free Software
16
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
18
"""Tests for the MemoryTree class."""
21
from ..memorytree import MemoryTree
22
from . import TestCaseWithTransport
23
from ..treebuilder import TreeBuilder
26
class TestMemoryTree(TestCaseWithTransport):
28
def test_create_on_branch(self):
29
"""Creating a mutable tree on a trivial branch works."""
30
branch = self.make_branch('branch')
31
tree = MemoryTree.create_on_branch(branch)
32
self.assertEqual(branch.controldir, tree.controldir)
33
self.assertEqual(branch, tree.branch)
34
self.assertEqual([], tree.get_parent_ids())
36
def test_create_on_branch_with_content(self):
37
"""Creating a mutable tree on a non-trivial branch works."""
38
branch = self.make_branch('branch')
39
tree = MemoryTree.create_on_branch(branch)
42
builder = TreeBuilder()
43
builder.start_tree(tree)
44
builder.build(['foo'])
46
rev_id = tree.commit('first post')
48
tree = MemoryTree.create_on_branch(branch)
50
self.assertEqual([rev_id], tree.get_parent_ids())
51
with tree.get_file('foo') as f:
52
self.assertEqual(b'contents of foo\n', f.read())
55
def test_get_root_id(self):
56
branch = self.make_branch('branch')
57
tree = MemoryTree.create_on_branch(branch)
61
self.assertIsNot(None, tree.get_root_id())
65
def test_lock_tree_write(self):
66
"""Check we can lock_tree_write and unlock MemoryTrees."""
67
branch = self.make_branch('branch')
68
tree = MemoryTree.create_on_branch(branch)
69
tree.lock_tree_write()
72
def test_lock_tree_write_after_read_fails(self):
73
"""Check that we error when trying to upgrade a read lock to write."""
74
branch = self.make_branch('branch')
75
tree = MemoryTree.create_on_branch(branch)
77
self.assertRaises(errors.ReadOnlyError, tree.lock_tree_write)
80
def test_lock_write(self):
81
"""Check we can lock_write and unlock MemoryTrees."""
82
branch = self.make_branch('branch')
83
tree = MemoryTree.create_on_branch(branch)
87
def test_lock_write_after_read_fails(self):
88
"""Check that we error when trying to upgrade a read lock to write."""
89
branch = self.make_branch('branch')
90
tree = MemoryTree.create_on_branch(branch)
92
self.assertRaises(errors.ReadOnlyError, tree.lock_write)
95
def test_add_with_kind(self):
96
branch = self.make_branch('branch')
97
tree = MemoryTree.create_on_branch(branch)
99
tree.add(['', 'afile', 'adir'], None,
100
['directory', 'file', 'directory'])
101
self.assertEqual('afile', tree.id2path(tree.path2id('afile')))
102
self.assertEqual('adir', tree.id2path(tree.path2id('adir')))
103
self.assertFalse(tree.has_filename('afile'))
104
self.assertFalse(tree.has_filename('adir'))
107
def test_put_new_file(self):
108
branch = self.make_branch('branch')
109
tree = MemoryTree.create_on_branch(branch)
111
tree.add(['', 'foo'], ids=[b'root-id', b'foo-id'],
112
kinds=['directory', 'file'])
113
tree.put_file_bytes_non_atomic('foo', b'barshoom')
114
self.assertEqual(b'barshoom', tree.get_file('foo').read())
117
def test_put_existing_file(self):
118
branch = self.make_branch('branch')
119
tree = MemoryTree.create_on_branch(branch)
121
tree.add(['', 'foo'], ids=[b'root-id', b'foo-id'],
122
kinds=['directory', 'file'])
123
tree.put_file_bytes_non_atomic('foo', b'first-content')
124
tree.put_file_bytes_non_atomic('foo', b'barshoom')
125
self.assertEqual(b'barshoom', tree.get_file('foo').read())
128
def test_add_in_subdir(self):
129
branch = self.make_branch('branch')
130
tree = MemoryTree.create_on_branch(branch)
132
self.addCleanup(tree.unlock)
133
tree.add([''], [b'root-id'], ['directory'])
134
# Unfortunately, the only way to 'mkdir' is to call 'tree.mkdir', but
135
# that *always* adds the directory as well. So if you want to create a
136
# file in a subdirectory, you have to split out the 'mkdir()' calls
137
# from the add and put_file_bytes_non_atomic calls. :(
138
tree.mkdir('adir', b'dir-id')
139
tree.add(['adir/afile'], [b'file-id'], ['file'])
140
self.assertEqual('adir/afile', tree.id2path(b'file-id'))
141
self.assertEqual('adir', tree.id2path(b'dir-id'))
142
tree.put_file_bytes_non_atomic('adir/afile', b'barshoom')
144
def test_commit_trivial(self):
145
"""Smoke test for commit on a MemoryTree.
147
Becamse of commits design and layering, if this works, all commit
148
logic should work quite reliably.
150
branch = self.make_branch('branch')
151
tree = MemoryTree.create_on_branch(branch)
153
tree.add(['', 'foo'], ids=[b'root-id', b'foo-id'],
154
kinds=['directory', 'file'])
155
tree.put_file_bytes_non_atomic('foo', b'barshoom')
156
revision_id = tree.commit('message baby')
157
# the parents list for the tree should have changed.
158
self.assertEqual([revision_id], tree.get_parent_ids())
160
# and we should have a revision that is accessible outside the tree lock
161
revtree = tree.branch.repository.revision_tree(revision_id)
163
self.addCleanup(revtree.unlock)
164
with revtree.get_file('foo') as f:
165
self.assertEqual(b'barshoom', f.read())
167
def test_unversion(self):
168
"""Some test for unversion of a memory tree."""
169
branch = self.make_branch('branch')
170
tree = MemoryTree.create_on_branch(branch)
172
tree.add(['', 'foo'], ids=[b'root-id', b'foo-id'],
173
kinds=['directory', 'file'])
174
tree.unversion(['foo'])
175
self.assertFalse(tree.is_versioned('foo'))
176
self.assertFalse(tree.has_id(b'foo-id'))
179
def test_last_revision(self):
180
"""There should be a last revision method we can call."""
181
tree = self.make_branch_and_memory_tree('branch')
184
rev_id = tree.commit('first post')
186
self.assertEqual(rev_id, tree.last_revision())
188
def test_rename_file(self):
189
tree = self.make_branch_and_memory_tree('branch')
191
self.addCleanup(tree.unlock)
192
tree.add(['', 'foo'], [b'root-id', b'foo-id'], ['directory', 'file'])
193
tree.put_file_bytes_non_atomic('foo', b'content\n')
194
tree.commit('one', rev_id=b'rev-one')
195
tree.rename_one('foo', 'bar')
196
self.assertEqual('bar', tree.id2path(b'foo-id'))
197
self.assertEqual(b'content\n', tree._file_transport.get_bytes('bar'))
198
self.assertRaises(errors.NoSuchFile,
199
tree._file_transport.get_bytes, 'foo')
200
tree.commit('two', rev_id=b'rev-two')
201
self.assertEqual(b'content\n', tree._file_transport.get_bytes('bar'))
202
self.assertRaises(errors.NoSuchFile,
203
tree._file_transport.get_bytes, 'foo')
205
rev_tree2 = tree.branch.repository.revision_tree(b'rev-two')
206
self.assertEqual('bar', rev_tree2.id2path(b'foo-id'))
207
self.assertEqual(b'content\n', rev_tree2.get_file_text('bar'))
209
def test_rename_file_to_subdir(self):
210
tree = self.make_branch_and_memory_tree('branch')
212
self.addCleanup(tree.unlock)
214
tree.mkdir('subdir', b'subdir-id')
215
tree.add('foo', b'foo-id', 'file')
216
tree.put_file_bytes_non_atomic('foo', b'content\n')
217
tree.commit('one', rev_id=b'rev-one')
219
tree.rename_one('foo', 'subdir/bar')
220
self.assertEqual('subdir/bar', tree.id2path(b'foo-id'))
221
self.assertEqual(b'content\n',
222
tree._file_transport.get_bytes('subdir/bar'))
223
tree.commit('two', rev_id=b'rev-two')
224
rev_tree2 = tree.branch.repository.revision_tree(b'rev-two')
225
self.assertEqual('subdir/bar', rev_tree2.id2path(b'foo-id'))