1
# Copyright (C) 2007-2011 Canonical Ltd
3
# This program is free software; you can redistribute it and/or modify
4
# it under the terms of the GNU General Public License as published by
5
# the Free Software Foundation; either version 2 of the License, or
6
# (at your option) any later version.
8
# This program is distributed in the hope that it will be useful,
9
# but WITHOUT ANY WARRANTY; without even the implied warranty of
10
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11
# GNU General Public License for more details.
13
# You should have received a copy of the GNU General Public License
14
# along with this program; if not, write to the Free Software
15
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
24
from ..revision import NULL_REVISION
25
from .test_graph import TestGraphBase
38
ancestry_1 = {'rev1': [NULL_REVISION], 'rev2a': ['rev1'], 'rev2b': ['rev1'],
39
'rev3': ['rev2a'], 'rev4': ['rev3', 'rev2b']}
52
ancestry_2 = {'rev1a': [NULL_REVISION], 'rev2a': ['rev1a'],
53
'rev1b': [NULL_REVISION], 'rev3a': ['rev2a'], 'rev4a': ['rev3a']}
56
# Extended history shortcut
68
extended_history_shortcut = {'a': [NULL_REVISION],
77
class TestSearchResultRefine(tests.TestCase):
79
def make_graph(self, ancestors):
80
return _mod_graph.Graph(_mod_graph.DictParentsProvider(ancestors))
82
def test_refine(self):
83
# Used when pulling from a stacked repository, so test some revisions
84
# being satisfied from the stacking branch.
86
{"tip":["mid"], "mid":["base"], "tag":["base"],
87
"base":[NULL_REVISION], NULL_REVISION:[]})
88
result = vf_search.SearchResult({'tip', 'tag'},
89
{NULL_REVISION}, 4, {'tip', 'mid', 'tag', 'base'})
90
result = result.refine({'tip'}, {'mid'})
91
recipe = result.get_recipe()
92
# We should be starting from tag (original head) and mid (seen ref)
93
self.assertEqual({'mid', 'tag'}, recipe[1])
94
# We should be stopping at NULL (original stop) and tip (seen head)
95
self.assertEqual({NULL_REVISION, 'tip'}, recipe[2])
96
self.assertEqual(3, recipe[3])
97
result = result.refine({'mid', 'tag', 'base'},
99
recipe = result.get_recipe()
100
# We should be starting from nothing (NULL was known as a cut point)
101
self.assertEqual(set([]), recipe[1])
102
# We should be stopping at NULL (original stop) and tip (seen head) and
103
# tag (seen head) and mid(seen mid-point head). We could come back and
104
# define this as not including mid, for minimal results, but it is
105
# still 'correct' to include mid, and simpler/easier.
106
self.assertEqual({NULL_REVISION, 'tip', 'tag', 'mid'}, recipe[2])
107
self.assertEqual(0, recipe[3])
108
self.assertTrue(result.is_empty())
111
class TestSearchResultFromParentMap(TestGraphBase):
113
def assertSearchResult(self, start_keys, stop_keys, key_count, parent_map,
115
(start, stop, count) = vf_search.search_result_from_parent_map(
116
parent_map, missing_keys)
117
self.assertEqual((sorted(start_keys), sorted(stop_keys), key_count),
118
(sorted(start), sorted(stop), count))
120
def test_no_parents(self):
121
self.assertSearchResult([], [], 0, {})
122
self.assertSearchResult([], [], 0, None)
124
def test_ancestry_1(self):
125
self.assertSearchResult(['rev4'], [NULL_REVISION], len(ancestry_1),
128
def test_ancestry_2(self):
129
self.assertSearchResult(['rev1b', 'rev4a'], [NULL_REVISION],
130
len(ancestry_2), ancestry_2)
131
self.assertSearchResult(['rev1b', 'rev4a'], [],
132
len(ancestry_2)+1, ancestry_2,
133
missing_keys=[NULL_REVISION])
135
def test_partial_search(self):
136
parent_map = dict((k, extended_history_shortcut[k])
138
self.assertSearchResult(['e', 'f'], ['d', 'a'], 2,
140
parent_map.update((k, extended_history_shortcut[k])
142
self.assertSearchResult(['e', 'f'], ['c', NULL_REVISION], 4,
144
parent_map['c'] = extended_history_shortcut['c']
145
self.assertSearchResult(['e', 'f'], ['b'], 6,
146
parent_map, missing_keys=[NULL_REVISION])
147
parent_map['b'] = extended_history_shortcut['b']
148
self.assertSearchResult(['e', 'f'], [], 7,
149
parent_map, missing_keys=[NULL_REVISION])
152
class TestLimitedSearchResultFromParentMap(TestGraphBase):
154
def assertSearchResult(self, start_keys, stop_keys, key_count, parent_map,
155
missing_keys, tip_keys, depth):
156
(start, stop, count) = vf_search.limited_search_result_from_parent_map(
157
parent_map, missing_keys, tip_keys, depth)
158
self.assertEqual((sorted(start_keys), sorted(stop_keys), key_count),
159
(sorted(start), sorted(stop), count))
161
def test_empty_ancestry(self):
162
self.assertSearchResult([], [], 0, {}, (), ['tip-rev-id'], 10)
164
def test_ancestry_1(self):
165
self.assertSearchResult(['rev4'], ['rev1'], 4,
166
ancestry_1, (), ['rev1'], 10)
167
self.assertSearchResult(['rev2a', 'rev2b'], ['rev1'], 2,
168
ancestry_1, (), ['rev1'], 1)
171
def test_multiple_heads(self):
172
self.assertSearchResult(['e', 'f'], ['a'], 5,
173
extended_history_shortcut, (), ['a'], 10)
174
# Note that even though we only take 1 step back, we find 'f', which
175
# means the described search will still find d and c.
176
self.assertSearchResult(['f'], ['a'], 4,
177
extended_history_shortcut, (), ['a'], 1)
178
self.assertSearchResult(['f'], ['a'], 4,
179
extended_history_shortcut, (), ['a'], 2)
182
class TestPendingAncestryResultRefine(tests.TestCase):
184
def make_graph(self, ancestors):
185
return _mod_graph.Graph(_mod_graph.DictParentsProvider(ancestors))
187
def test_refine(self):
188
# Used when pulling from a stacked repository, so test some revisions
189
# being satisfied from the stacking branch.
191
{"tip":["mid"], "mid":["base"], "tag":["base"],
192
"base":[NULL_REVISION], NULL_REVISION:[]})
193
result = vf_search.PendingAncestryResult(['tip', 'tag'], None)
194
result = result.refine({'tip'}, {'mid'})
195
self.assertEqual({'mid', 'tag'}, result.heads)
196
result = result.refine({'mid', 'tag', 'base'},
198
self.assertEqual({NULL_REVISION}, result.heads)
199
self.assertTrue(result.is_empty())
202
class TestPendingAncestryResultGetKeys(tests.TestCaseWithMemoryTransport):
203
"""Tests for breezy.graph.PendingAncestryResult."""
205
def test_get_keys(self):
206
builder = self.make_branch_builder('b')
207
builder.start_series()
208
builder.build_snapshot(None, [
209
('add', ('', b'root-id', 'directory', ''))],
210
revision_id=b'rev-1')
211
builder.build_snapshot([b'rev-1'], [], revision_id=b'rev-2')
212
builder.finish_series()
213
repo = builder.get_branch().repository
215
self.addCleanup(repo.unlock)
216
result = vf_search.PendingAncestryResult([b'rev-2'], repo)
217
self.assertEqual({b'rev-1', b'rev-2'}, set(result.get_keys()))
219
def test_get_keys_excludes_ghosts(self):
220
builder = self.make_branch_builder('b')
221
builder.start_series()
222
builder.build_snapshot(None, [
223
('add', ('', b'root-id', 'directory', ''))],
224
revision_id=b'rev-1')
225
builder.build_snapshot([b'rev-1', b'ghost'], [], revision_id=b'rev-2')
226
builder.finish_series()
227
repo = builder.get_branch().repository
229
self.addCleanup(repo.unlock)
230
result = vf_search.PendingAncestryResult([b'rev-2'], repo)
231
self.assertEqual(sorted([b'rev-1', b'rev-2']), sorted(result.get_keys()))
233
def test_get_keys_excludes_null(self):
234
# Make a 'graph' with an iter_ancestry that returns NULL_REVISION
235
# somewhere other than the last element, which can happen in real
237
class StubGraph(object):
238
def iter_ancestry(self, keys):
239
return [(NULL_REVISION, ()), ('foo', (NULL_REVISION,))]
240
result = vf_search.PendingAncestryResult(['rev-3'], None)
241
result_keys = result._get_keys(StubGraph())
242
# Only the non-null keys from the ancestry appear.
243
self.assertEqual({'foo'}, set(result_keys))