91
100
pb = ui.ui_factory.nested_progress_bar()
92
101
pb.show_pct = pb.show_count = False
94
pb.update("Finding revisions", 0, 2)
95
search = self._revids_to_fetch()
103
pb.update(gettext("Finding revisions"), 0, 2)
104
search_result = self._revids_to_fetch()
105
mutter('fetching: %s', search_result)
106
if search_result.is_empty():
98
pb.update("Fetching revisions", 1, 2)
99
self._fetch_everything_for_search(search)
108
pb.update(gettext("Fetching revisions"), 1, 2)
109
self._fetch_everything_for_search(search_result)
151
158
"""Determines the exact revisions needed from self.from_repository to
152
159
install self._last_revision in self.to_repository.
154
If no revisions need to be fetched, then this just returns None.
161
:returns: A SearchResult of some sort. (Possibly a
162
PendingAncestryResult, EmptySearchResult, etc.)
156
164
if self._fetch_spec is not None:
165
# The fetch spec is already a concrete search result.
157
166
return self._fetch_spec
158
mutter('fetch up to rev {%s}', self._last_revision)
159
if self._last_revision is NULL_REVISION:
167
elif self._last_revision == NULL_REVISION:
168
# fetch_spec is None + last_revision is null => empty fetch.
160
169
# explicit limit of no revisions needed
162
return self.to_repository.search_missing_revision_ids(
163
self.from_repository, self._last_revision,
164
find_ghosts=self.find_ghosts)
166
def _parent_inventories(self, revision_ids):
167
# Find all the parent revisions referenced by the stream, but
168
# not present in the stream, and make sure we send their
170
parent_maps = self.to_repository.get_parent_map(revision_ids)
172
map(parents.update, parent_maps.itervalues())
173
parents.discard(NULL_REVISION)
174
parents.difference_update(revision_ids)
175
missing_keys = set(('inventories', rev_id) for rev_id in parents)
170
return vf_search.EmptySearchResult()
171
elif self._last_revision is not None:
172
return vf_search.NotInOtherForRevs(self.to_repository,
173
self.from_repository, [self._last_revision],
174
find_ghosts=self.find_ghosts).execute()
175
else: # self._last_revision is None:
176
return vf_search.EverythingNotInOther(self.to_repository,
177
self.from_repository,
178
find_ghosts=self.find_ghosts).execute()
179
181
class Inter1and2Helper(object):
203
208
revs = list(revs)
205
210
for tree in self.source.revision_trees(revs[:100]):
206
if tree.inventory.revision_id is None:
207
tree.inventory.revision_id = tree.get_revision_id()
211
if tree.root_inventory.revision_id is None:
212
tree.root_inventory.revision_id = tree.get_revision_id()
209
214
revs = revs[100:]
211
216
def _find_root_ids(self, revs, parent_map, graph):
212
217
revision_root = {}
213
218
for tree in self.iter_rev_trees(revs):
214
revision_id = tree.inventory.root.revision
215
219
root_id = tree.get_root_id()
220
revision_id = tree.get_file_revision(root_id, u"")
216
221
revision_root[revision_id] = root_id
217
222
# Find out which parents we don't already know root ids for
219
for revision_parents in parent_map.itervalues():
220
parents.update(revision_parents)
221
parents.difference_update(revision_root.keys() + [NULL_REVISION])
223
parents = set(viewvalues(parent_map))
224
parents.difference_update(revision_root)
225
parents.discard(NULL_REVISION)
222
226
# Limit to revisions present in the versionedfile
223
parents = graph.get_parent_map(parents).keys()
227
parents = graph.get_parent_map(parents)
224
228
for tree in self.iter_rev_trees(parents):
225
229
root_id = tree.get_root_id()
226
230
revision_root[tree.get_revision_id()] = root_id
243
247
# yet, and are unlikely to in non-rich-root environments anyway.
244
248
root_id_order.sort(key=operator.itemgetter(0))
245
249
# Create a record stream containing the roots to create.
247
# XXX: not covered by tests, should have a flag to always run
248
# this. -- mbp 20100129
249
graph = self.source_repo.get_known_graph_ancestry(revs)
250
if len(revs) > self.known_graph_threshold:
251
graph = self.source.get_known_graph_ancestry(revs)
250
252
new_roots_stream = _new_root_data_stream(
251
253
root_id_order, rev_id_to_root_id, parent_map, self.source, graph)
252
254
return [('texts', new_roots_stream)]
255
def _get_rich_root_heads_graph(source_repo, revision_ids):
256
"""Get a Graph object suitable for asking heads() for new rich roots."""
260
257
def _new_root_data_stream(
261
258
root_keys_to_create, rev_id_to_root_id_map, parent_map, repo, graph=None):
262
259
"""Generate a texts substream of synthesised root entries.
340
337
selected_ids.append(parent_id)
341
338
parent_keys = [(root_id, parent_id) for parent_id in selected_ids]
342
339
return parent_keys
342
class TargetRepoKinds(object):
343
"""An enum-like set of constants.
345
They are the possible values of FetchSpecFactory.target_repo_kinds.
348
PREEXISTING = 'preexisting'
353
class FetchSpecFactory(object):
354
"""A helper for building the best fetch spec for a sprout call.
356
Factors that go into determining the sort of fetch to perform:
357
* did the caller specify any revision IDs?
358
* did the caller specify a source branch (need to fetch its
359
heads_to_fetch(), usually the tip + tags)
360
* is there an existing target repo (don't need to refetch revs it
362
* target is stacked? (similar to pre-existing target repo: even if
363
the target itself is new don't want to refetch existing revs)
365
:ivar source_branch: the source branch if one specified, else None.
366
:ivar source_branch_stop_revision_id: fetch up to this revision of
367
source_branch, rather than its tip.
368
:ivar source_repo: the source repository if one found, else None.
369
:ivar target_repo: the target repository acquired by sprout.
370
:ivar target_repo_kind: one of the TargetRepoKinds constants.
374
self._explicit_rev_ids = set()
375
self.source_branch = None
376
self.source_branch_stop_revision_id = None
377
self.source_repo = None
378
self.target_repo = None
379
self.target_repo_kind = None
382
def add_revision_ids(self, revision_ids):
383
"""Add revision_ids to the set of revision_ids to be fetched."""
384
self._explicit_rev_ids.update(revision_ids)
386
def make_fetch_spec(self):
387
"""Build a SearchResult or PendingAncestryResult or etc."""
388
if self.target_repo_kind is None or self.source_repo is None:
389
raise AssertionError(
390
'Incomplete FetchSpecFactory: %r' % (self.__dict__,))
391
if len(self._explicit_rev_ids) == 0 and self.source_branch is None:
392
if self.limit is not None:
393
raise NotImplementedError(
394
"limit is only supported with a source branch set")
395
# Caller hasn't specified any revisions or source branch
396
if self.target_repo_kind == TargetRepoKinds.EMPTY:
397
return vf_search.EverythingResult(self.source_repo)
399
# We want everything not already in the target (or target's
401
return vf_search.EverythingNotInOther(
402
self.target_repo, self.source_repo).execute()
403
heads_to_fetch = set(self._explicit_rev_ids)
404
if self.source_branch is not None:
405
must_fetch, if_present_fetch = self.source_branch.heads_to_fetch()
406
if self.source_branch_stop_revision_id is not None:
407
# Replace the tip rev from must_fetch with the stop revision
408
# XXX: this might be wrong if the tip rev is also in the
409
# must_fetch set for other reasons (e.g. it's the tip of
410
# multiple loom threads?), but then it's pretty unclear what it
411
# should mean to specify a stop_revision in that case anyway.
412
must_fetch.discard(self.source_branch.last_revision())
413
must_fetch.add(self.source_branch_stop_revision_id)
414
heads_to_fetch.update(must_fetch)
416
if_present_fetch = set()
417
if self.target_repo_kind == TargetRepoKinds.EMPTY:
418
# PendingAncestryResult does not raise errors if a requested head
419
# is absent. Ideally it would support the
420
# required_ids/if_present_ids distinction, but in practice
421
# heads_to_fetch will almost certainly be present so this doesn't
423
all_heads = heads_to_fetch.union(if_present_fetch)
424
ret = vf_search.PendingAncestryResult(all_heads, self.source_repo)
425
if self.limit is not None:
426
graph = self.source_repo.get_graph()
427
topo_order = list(graph.iter_topo_order(ret.get_keys()))
428
result_set = topo_order[:self.limit]
429
ret = self.source_repo.revision_ids_to_search_result(result_set)
432
return vf_search.NotInOtherForRevs(self.target_repo, self.source_repo,
433
required_ids=heads_to_fetch, if_present_ids=if_present_fetch,
434
limit=self.limit).execute()