1
# Copyright (C) 2005-2010 Canonical Ltd
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.
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.
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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
15
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
17
# TODO: Some kind of command-line display of revision properties:
17
# TODO: Some kind of command-line display of revision properties:
18
18
# perhaps show them in log -v and allow them as options to the commit command.
21
from bzrlib.lazy_import import lazy_import
22
lazy_import(globals(), """
23
from bzrlib import deprecated_graph
24
from bzrlib import bugtracker
22
import bzrlib.errors as errors
23
from bzrlib.graph import node_distances, select_farthest, all_descendants, Graph
30
24
from bzrlib.osutils import contains_whitespace
25
from bzrlib.progress import DummyProgress
26
from bzrlib.symbol_versioning import *
32
28
NULL_REVISION="null:"
33
CURRENT_REVISION="current:"
36
30
class Revision(object):
37
31
"""Single revision on a branch.
114
104
def get_summary(self):
115
105
"""Get the first line of the log message for this revision.
117
Return an empty string if message is None.
120
return self.message.lstrip().split('\n', 1)[0]
124
@symbol_versioning.deprecated_method(symbol_versioning.deprecated_in((1, 13, 0)))
125
def get_apparent_author(self):
126
"""Return the apparent author of this revision.
128
This method is deprecated in favour of get_apparent_authors.
130
If the revision properties contain any author names,
131
return the first. Otherwise return the committer name.
133
authors = self.get_apparent_authors()
139
def get_apparent_authors(self):
140
"""Return the apparent authors of this revision.
142
If the revision properties contain the names of the authors,
143
return them. Otherwise return the committer name.
145
The return value will be a list containing at least one element.
147
authors = self.properties.get('authors', None)
149
author = self.properties.get('author', self.committer)
154
return authors.split("\n")
157
"""Iterate over the bugs associated with this revision."""
158
bug_property = self.properties.get('bugs', None)
159
if bug_property is None:
161
for line in bug_property.splitlines():
163
url, status = line.split(None, 2)
165
raise errors.InvalidLineInBugsProperty(line)
166
if status not in bugtracker.ALLOWED_BUG_STATUSES:
167
raise errors.InvalidBugStatus(status)
107
return self.message.split('\n', 1)[0]
110
def is_ancestor(revision_id, candidate_id, branch):
111
"""Return true if candidate_id is an ancestor of revision_id.
113
A false negative will be returned if any intermediate descendent of
114
candidate_id is not present in any of the revision_sources.
116
revisions_source is an object supporting a get_revision operation that
117
behaves like Branch's.
119
return candidate_id in branch.repository.get_ancestry(revision_id)
171
122
def iter_ancestors(revision_id, revision_source, only_present=False):
194
145
"""Return the ancestors of a revision present in a branch.
196
147
It's possible that a branch won't have the complete ancestry of
197
one of its revisions.
148
one of its revisions.
200
151
found_ancestors = {}
201
152
anc_iter = enumerate(iter_ancestors(revision_id, revision_source,
202
153
only_present=True))
203
154
for anc_order, (anc_id, anc_distance) in anc_iter:
204
if anc_id not in found_ancestors:
155
if not found_ancestors.has_key(anc_id):
205
156
found_ancestors[anc_id] = (anc_order, anc_distance)
206
157
return found_ancestors
209
160
def __get_closest(intersection):
210
161
intersection.sort()
212
163
for entry in intersection:
213
164
if entry[0] == intersection[0][0]:
214
165
matches.append(entry[2])
218
def is_reserved_id(revision_id):
219
"""Determine whether a revision id is reserved
221
:return: True if the revision is reserved, False otherwise
169
def revision_graph(revision, revision_source):
170
"""Produce a graph of the ancestry of the specified revision.
172
:return: root, ancestors map, descendants map
223
return isinstance(revision_id, basestring) and revision_id.endswith(':')
226
def check_not_reserved_id(revision_id):
227
"""Raise ReservedId if the supplied revision_id is reserved"""
228
if is_reserved_id(revision_id):
229
raise errors.ReservedId(revision_id)
232
def ensure_null(revision_id):
233
"""Ensure only NULL_REVISION is used to represent the null revision"""
234
if revision_id is None:
235
symbol_versioning.warn('NULL_REVISION should be used for the null'
236
' revision instead of None, as of bzr 0.91.',
237
DeprecationWarning, stacklevel=2)
243
def is_null(revision_id):
244
if revision_id is None:
245
symbol_versioning.warn('NULL_REVISION should be used for the null'
246
' revision instead of None, as of bzr 0.90.',
247
DeprecationWarning, stacklevel=2)
248
return revision_id in (None, NULL_REVISION)
174
revision_source.lock_read()
176
return _revision_graph(revision, revision_source)
178
revision_source.unlock()
181
def _revision_graph(revision, revision_source):
182
"""See revision_graph."""
183
from bzrlib.tsort import topo_sort
184
graph = revision_source.get_revision_graph(revision)
185
# mark all no-parent revisions as being NULL_REVISION parentage.
186
for node, parents in graph.items():
187
if len(parents) == 0:
188
graph[node] = [NULL_REVISION]
189
# add NULL_REVISION to the graph
190
graph[NULL_REVISION] = []
192
# pick a root. If there are multiple roots
193
# this could pick a random one.
194
topo_order = topo_sort(graph.items())
200
# map the descendants of the graph.
201
# and setup our set based return graph.
202
for node in graph.keys():
203
descendants[node] = {}
204
for node, parents in graph.items():
205
for parent in parents:
206
descendants[parent][node] = 1
207
ancestors[node] = set(parents)
209
assert root not in descendants[root]
210
assert root not in ancestors[root]
211
return root, ancestors, descendants
214
def combined_graph(revision_a, revision_b, revision_source):
215
"""Produce a combined ancestry graph.
216
Return graph root, ancestors map, descendants map, set of common nodes"""
217
root, ancestors, descendants = revision_graph(
218
revision_a, revision_source)
219
root_b, ancestors_b, descendants_b = revision_graph(
220
revision_b, revision_source)
222
raise bzrlib.errors.NoCommonRoot(revision_a, revision_b)
224
for node, node_anc in ancestors_b.iteritems():
225
if node in ancestors:
228
ancestors[node] = set()
229
ancestors[node].update(node_anc)
230
for node, node_dec in descendants_b.iteritems():
231
if node not in descendants:
232
descendants[node] = {}
233
descendants[node].update(node_dec)
234
return root, ancestors, descendants, common
237
def common_ancestor(revision_a, revision_b, revision_source,
239
if None in (revision_a, revision_b):
241
# trivial optimisation
242
if revision_a == revision_b:
246
pb.update('Picking ancestor', 1, 3)
247
graph = revision_source.get_revision_graph_with_ghosts(
248
[revision_a, revision_b])
249
# convert to a NULL_REVISION based graph.
250
ancestors = graph.get_ancestors()
251
descendants = graph.get_descendants()
252
common = set(graph.get_ancestry(revision_a)).intersection(
253
set(graph.get_ancestry(revision_b)))
254
descendants[NULL_REVISION] = {}
255
ancestors[NULL_REVISION] = []
256
for root in graph.roots:
257
descendants[NULL_REVISION][root] = 1
258
ancestors[root].append(NULL_REVISION)
259
for ghost in graph.ghosts:
260
# ghosts act as roots for the purpose of finding
261
# the longest paths from the root: any ghost *might*
262
# be directly attached to the root, so we treat them
264
# ghost now descends from NULL
265
descendants[NULL_REVISION][ghost] = 1
266
# that is it has an ancestor of NULL
267
ancestors[ghost] = [NULL_REVISION]
268
# ghost is common if any of ghosts descendants are common:
269
for ghost_descendant in descendants[ghost]:
270
if ghost_descendant in common:
274
common.add(NULL_REVISION)
275
except bzrlib.errors.NoCommonRoot:
276
raise bzrlib.errors.NoCommonAncestor(revision_a, revision_b)
278
pb.update('Picking ancestor', 2, 3)
279
distances = node_distances (descendants, ancestors, root)
280
pb.update('Picking ancestor', 3, 2)
281
farthest = select_farthest(distances, common)
282
if farthest is None or farthest == NULL_REVISION:
283
raise bzrlib.errors.NoCommonAncestor(revision_a, revision_b)
289
class MultipleRevisionSources(object):
290
"""Proxy that looks in multiple branches for revisions."""
291
def __init__(self, *args):
292
object.__init__(self)
293
assert len(args) != 0
294
self._revision_sources = args
296
def revision_parents(self, revision_id):
297
for source in self._revision_sources:
299
return source.revision_parents(revision_id)
300
except (errors.WeaveRevisionNotPresent, errors.NoSuchRevision), e:
304
def get_revision(self, revision_id):
305
for source in self._revision_sources:
307
return source.get_revision(revision_id)
308
except bzrlib.errors.NoSuchRevision, e:
312
def get_revision_graph(self, revision_id):
313
# we could probe incrementally until the pending
314
# ghosts list stop growing, but its cheaper for now
315
# to just ask for the complete graph for each repository.
317
for source in self._revision_sources:
318
ghost_graph = source.get_revision_graph_with_ghosts()
319
graphs.append(ghost_graph)
322
if not revision_id in graph.get_ancestors():
324
if absent == len(graphs):
325
raise errors.NoSuchRevision(self._revision_sources[0], revision_id)
329
pending = set([revision_id])
330
def find_parents(node_id):
331
"""find the parents for node_id."""
333
ancestors = graph.get_ancestors()
335
return ancestors[node_id]
338
raise errors.NoSuchRevision(self._revision_sources[0], node_id)
340
# all the graphs should have identical parent lists
341
node_id = pending.pop()
343
result[node_id] = find_parents(node_id)
344
for parent_node in result[node_id]:
345
if not parent_node in result:
346
pending.add(parent_node)
347
except errors.NoSuchRevision:
352
def get_revision_graph_with_ghosts(self, revision_ids):
353
# query all the sources for their entire graphs
354
# and then build a combined graph for just
357
for source in self._revision_sources:
358
ghost_graph = source.get_revision_graph_with_ghosts()
359
graphs.append(ghost_graph.get_ancestors())
360
for revision_id in revision_ids:
363
if not revision_id in graph:
365
if absent == len(graphs):
366
raise errors.NoSuchRevision(self._revision_sources[0],
371
pending = set(revision_ids)
373
def find_parents(node_id):
374
"""find the parents for node_id."""
377
return graph[node_id]
380
raise errors.NoSuchRevision(self._revision_sources[0], node_id)
382
# all the graphs should have identical parent lists
383
node_id = pending.pop()
385
parents = find_parents(node_id)
386
for parent_node in parents:
388
if (parent_node not in pending and
389
parent_node not in done):
391
pending.add(parent_node)
392
result.add_node(node_id, parents)
394
except errors.NoSuchRevision:
396
result.add_ghost(node_id)
401
for source in self._revision_sources:
405
for source in self._revision_sources: