14
14
# along with this program; if not, write to the Free Software
15
15
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
17
from bzrlib import osutils, ui, urlutils
18
from bzrlib.errors import InvalidRevisionId
19
from bzrlib.inventory import Inventory
20
from bzrlib.repository import InterRepository
21
from bzrlib.trace import info
22
from bzrlib.tsort import topo_sort
24
from bzrlib.plugins.git import git
17
from dulwich.objects import (
22
from dulwich.object_store import (
35
from bzrlib.errors import (
39
from bzrlib.inventory import (
46
from bzrlib.lru_cache import (
49
from bzrlib.repository import (
52
from bzrlib.revision import (
55
from bzrlib.tsort import (
58
from bzrlib.versionedfile import (
59
FulltextContentFactory,
62
from bzrlib.plugins.git.mapping import (
64
inventory_to_tree_and_blobs,
69
from bzrlib.plugins.git.object_store import (
72
from bzrlib.plugins.git.remote import (
25
75
from bzrlib.plugins.git.repository import (
30
from bzrlib.plugins.git.remote import RemoteGitRepository
32
from dulwich.objects import Commit
34
from cStringIO import StringIO
37
class BzrFetchGraphWalker(object):
39
def __init__(self, repository, mapping):
40
self.repository = repository
41
self.mapping = mapping
43
self.heads = set(repository.all_revision_ids())
47
revid = self.mapping.revision_id_foreign_to_bzr(sha)
50
def remove(self, revid):
53
self.heads.remove(revid)
54
if revid in self.parents:
55
for p in self.parents[revid]:
60
ret = self.heads.pop()
61
ps = self.repository.get_parent_map([ret])[ret]
62
self.parents[ret] = ps
63
self.heads.update([p for p in ps if not p in self.done])
66
return self.mapping.revision_id_bzr_to_foreign(ret)
67
except InvalidRevisionId:
72
def import_git_blob(repo, mapping, path, blob, inv, parent_invs, executable):
82
def import_git_blob(texts, mapping, path, hexsha, base_inv, base_ie, parent_id,
83
revision_id, parent_invs, shagitmap, lookup_object, executable, symlink):
73
84
"""Import a git blob object into a bzr repository.
75
:param repo: bzr repository
86
:param texts: VersionedFiles to add to
76
87
:param path: Path in the tree
77
88
:param blob: A git blob
89
:return: Inventory delta for this file
79
91
file_id = mapping.generate_file_id(path)
80
text_revision = inv.revision_id
81
repo.texts.add_lines((file_id, text_revision),
82
[(file_id, p[file_id].revision) for p in parent_invs if file_id in p],
83
osutils.split_lines(blob.data))
84
ie = inv.add_path(path, "file", file_id)
85
ie.revision = text_revision
86
ie.text_size = len(blob.data)
87
ie.text_sha1 = osutils.sha_string(blob.data)
96
# We just have to hope this is indeed utf-8:
97
ie = cls(file_id, urlutils.basename(path).decode("utf-8"), parent_id)
88
98
ie.executable = executable
91
def import_git_tree(repo, mapping, path, tree, inv, parent_invs, lookup_object):
99
# See if this has changed at all
104
base_sha = shagitmap.lookup_blob(file_id, base_ie.revision)
108
if (base_sha == hexsha and base_ie.executable == ie.executable
109
and base_ie.kind == ie.kind):
110
# If nothing has changed since the base revision, we're done
112
if base_sha == hexsha and base_ie.kind == ie.kind:
113
ie.text_size = base_ie.text_size
114
ie.text_sha1 = base_ie.text_sha1
115
ie.symlink_target = base_ie.symlink_target
116
if ie.executable == base_ie.executable:
117
ie.revision = base_ie.revision
119
blob = lookup_object(hexsha)
121
blob = lookup_object(hexsha)
122
if ie.kind == "symlink":
124
ie.symlink_target = blob.data
128
ie.text_size = len(blob.data)
129
ie.text_sha1 = osutils.sha_string(blob.data)
130
# Check what revision we should store
132
for pinv in parent_invs:
133
if pinv.revision_id == base_inv.revision_id:
142
if pie.text_sha1 == ie.text_sha1 and pie.executable == ie.executable and pie.symlink_target == ie.symlink_target:
143
# found a revision in one of the parents to use
144
ie.revision = pie.revision
146
parent_keys.append((file_id, pie.revision))
147
if ie.revision is None:
148
# Need to store a new revision
149
ie.revision = revision_id
150
assert file_id is not None
151
assert ie.revision is not None
152
texts.insert_record_stream([FulltextContentFactory((file_id, ie.revision), tuple(parent_keys), ie.text_sha1, blob.data)])
153
shamap = [(hexsha, "blob", (ie.file_id, ie.revision))]
157
if base_ie is not None:
158
old_path = base_inv.id2path(file_id)
159
if base_ie.kind == "directory":
160
invdelta.extend(remove_disappeared_children(old_path, base_ie.children, []))
163
invdelta.append((old_path, path, file_id, ie))
164
return (invdelta, shamap)
167
class SubmodulesRequireSubtrees(BzrError):
168
_fmt = """The repository you are fetching from contains submodules. Please run 'bzr upgrade --development-subtree'."""
172
def import_git_submodule(texts, mapping, path, hexsha, base_inv, base_ie,
173
parent_id, revision_id, parent_invs, shagitmap, lookup_object):
174
file_id = mapping.generate_file_id(path)
175
ie = TreeReference(file_id, urlutils.basename(path.decode("utf-8")),
177
ie.revision = revision_id
182
if base_ie.kind == ie.kind and base_ie.reference_revision == ie.reference_revision:
183
ie.revision = base_ie.revision
184
ie.reference_revision = mapping.revision_id_foreign_to_bzr(hexsha)
185
texts.insert_record_stream([FulltextContentFactory((file_id, ie.revision), (), None, "")])
186
invdelta = [(oldpath, path, file_id, ie)]
187
return invdelta, {}, {}
190
def remove_disappeared_children(path, base_children, existing_children):
192
deletable = [(osutils.pathjoin(path, k), v) for k,v in base_children.iteritems() if k not in existing_children]
194
(path, ie) = deletable.pop()
195
ret.append((path, None, ie.file_id, None))
196
if ie.kind == "directory":
197
for name, child_ie in ie.children.iteritems():
198
deletable.append((osutils.pathjoin(path, name), child_ie))
202
def import_git_tree(texts, mapping, path, hexsha, base_inv, base_ie, parent_id,
203
revision_id, parent_invs, shagitmap, lookup_object):
92
204
"""Import a git tree object into a bzr repository.
94
:param repo: A Bzr repository object
206
:param texts: VersionedFiles object to add to
95
207
:param path: Path in the tree
96
208
:param tree: A git tree object
97
:param inv: Inventory object
209
:param base_inv: Base inventory against which to return inventory delta
210
:return: Inventory delta for this subtree
99
213
file_id = mapping.generate_file_id(path)
100
text_revision = inv.revision_id
101
repo.texts.add_lines((file_id, text_revision),
102
[(file_id, p[file_id].revision) for p in parent_invs if file_id in p],
104
ie = inv.add_path(path, "directory", file_id)
105
ie.revision = text_revision
106
for mode, name, hexsha in tree.entries():
107
entry_kind = (mode & 0700000) / 0100000
214
# We just have to hope this is indeed utf-8:
215
ie = InventoryDirectory(file_id, urlutils.basename(path.decode("utf-8")),
218
# Newly appeared here
219
ie.revision = revision_id
220
texts.insert_record_stream([FulltextContentFactory((file_id, ie.revision), (), None, "")])
221
invdelta.append((None, path, file_id, ie))
223
# See if this has changed at all
225
base_sha = shagitmap.lookup_tree(file_id, base_inv.revision_id)
229
if base_sha == hexsha:
230
# If nothing has changed since the base revision, we're done
232
if base_ie.kind != "directory":
233
ie.revision = revision_id
234
texts.insert_record_stream([FulltextContentFactory((ie.file_id, ie.revision), (), None, "")])
235
invdelta.append((base_inv.id2path(ie.file_id), path, ie.file_id, ie))
236
if base_ie is not None and base_ie.kind == "directory":
237
base_children = base_ie.children
240
# Remember for next time
241
existing_children = set()
244
tree = lookup_object(hexsha)
245
for mode, name, child_hexsha in tree.entries():
108
246
basename = name.decode("utf-8")
112
child_path = urlutils.join(path, name)
114
tree = lookup_object(hexsha)
115
import_git_tree(repo, mapping, child_path, tree, inv, parent_invs, lookup_object)
116
elif entry_kind == 1:
117
blob = lookup_object(hexsha)
118
fs_mode = mode & 0777
119
import_git_blob(repo, mapping, child_path, blob, inv, parent_invs, bool(fs_mode & 0111))
121
raise AssertionError("Unknown blob kind, perms=%r." % (mode,))
124
def import_git_objects(repo, mapping, object_iter, pb=None):
247
existing_children.add(basename)
248
child_path = osutils.pathjoin(path, name)
249
if stat.S_ISDIR(mode):
250
subinvdelta, grandchildmodes, subshamap = import_git_tree(
251
texts, mapping, child_path, child_hexsha, base_inv,
252
base_children.get(basename), file_id, revision_id, parent_invs, shagitmap,
254
invdelta.extend(subinvdelta)
255
child_modes.update(grandchildmodes)
256
shamap.extend(subshamap)
257
elif S_ISGITLINK(mode): # submodule
258
subinvdelta, grandchildmodes, subshamap = import_git_submodule(
259
texts, mapping, child_path, child_hexsha, base_inv, base_children.get(basename),
260
file_id, revision_id, parent_invs, shagitmap, lookup_object)
261
invdelta.extend(subinvdelta)
262
child_modes.update(grandchildmodes)
263
shamap.extend(subshamap)
265
subinvdelta, subshamap = import_git_blob(texts, mapping,
266
child_path, child_hexsha, base_inv, base_children.get(basename), file_id,
267
revision_id, parent_invs, shagitmap, lookup_object,
268
mode_is_executable(mode), stat.S_ISLNK(mode))
269
invdelta.extend(subinvdelta)
270
shamap.extend(subshamap)
271
if mode not in (stat.S_IFDIR, DEFAULT_FILE_MODE,
272
stat.S_IFLNK, DEFAULT_FILE_MODE|0111):
273
child_modes[child_path] = mode
274
# Remove any children that have disappeared
275
if base_ie is not None and base_ie.kind == "directory":
276
invdelta.extend(remove_disappeared_children(base_inv.id2path(file_id),
277
base_children, existing_children))
278
shamap.append((hexsha, "tree", (file_id, revision_id)))
279
return invdelta, child_modes, shamap
282
def import_git_objects(repo, mapping, object_iter, target_git_object_retriever,
125
284
"""Import a set of git objects into a bzr repository.
127
:param repo: Bazaar repository
286
:param repo: Target Bazaar repository
128
287
:param mapping: Mapping to use
129
288
:param object_iter: Iterator over Git objects.
290
def lookup_object(sha):
292
return object_iter[sha]
294
return target_git_object_retriever[sha]
131
295
# TODO: a more (memory-)efficient implementation of this
133
for i, o in enumerate(object_iter):
135
pb.update("fetching objects", i)
301
parent_invs_cache = LRUCache(50)
140
302
# Find and convert commit objects
141
for o in objects.itervalues():
305
pb.update("finding revisions to fetch", len(graph), None)
307
assert isinstance(head, str)
309
o = lookup_object(head)
311
trace.mutter('missing head %s', head)
142
313
if isinstance(o, Commit):
143
314
rev = mapping.import_commit(o)
144
root_trees[rev.revision_id] = objects[o.tree]
315
if repo.has_revision(rev.revision_id):
317
squash_revision(repo, rev)
318
root_trees[rev.revision_id] = o.tree
145
319
revisions[rev.revision_id] = rev
146
320
graph.append((rev.revision_id, rev.parent_ids))
321
target_git_object_retriever._idmap.add_entry(o.id, "commit",
322
(rev.revision_id, o.tree))
323
heads.extend([p for p in o.parents if p not in checked])
324
elif isinstance(o, Tag):
325
heads.append(o.object[1])
327
trace.warning("Unable to import head object %r" % o)
147
329
# Order the revisions
148
330
# Create the inventory objects
149
331
for i, revid in enumerate(topo_sort(graph)):
150
332
if pb is not None:
151
333
pb.update("fetching revisions", i, len(graph))
152
root_tree = root_trees[revid]
153
334
rev = revisions[revid]
154
# We have to do this here, since we have to walk the tree and
155
# we need to make sure to import the blobs / trees with the riht
335
# We have to do this here, since we have to walk the tree and
336
# we need to make sure to import the blobs / trees with the right
156
337
# path; this may involve adding them more than once.
158
inv.revision_id = rev.revision_id
159
def lookup_object(sha):
162
return reconstruct_git_object(repo, mapping, sha)
163
parent_invs = [repo.get_inventory(r) for r in rev.parent_ids]
164
import_git_tree(repo, mapping, "", root_tree, inv, parent_invs, lookup_object)
165
repo.add_revision(rev.revision_id, rev, inv)
168
def reconstruct_git_commit(repo, rev):
169
raise NotImplementedError(self.reconstruct_git_commit)
172
def reconstruct_git_object(repo, mapping, sha):
174
revid = mapping.revision_id_foreign_to_bzr(sha)
176
rev = repo.get_revision(revid)
177
except NoSuchRevision:
180
return reconstruct_git_commit(rev)
184
raise KeyError("No such object %s" % sha)
339
for parent_id in rev.parent_ids:
341
parent_invs.append(parent_invs_cache[parent_id])
343
parent_inv = repo.get_inventory(parent_id)
344
parent_invs.append(parent_inv)
345
parent_invs_cache[parent_id] = parent_inv
346
if parent_invs == []:
347
base_inv = Inventory(root_id=None)
350
base_inv = parent_invs[0]
351
base_ie = base_inv.root
352
inv_delta, unusual_modes, shamap = import_git_tree(repo.texts,
353
mapping, "", root_trees[revid], base_inv, base_ie, None, revid,
354
parent_invs, target_git_object_retriever._idmap, lookup_object)
355
target_git_object_retriever._idmap.add_entries(shamap)
356
if unusual_modes != {}:
357
for path, mode in unusual_modes.iteritems():
358
warn_unusual_mode(rev.foreign_revid, path, mode)
359
mapping.import_unusual_file_modes(rev, unusual_modes)
361
basis_id = rev.parent_ids[0]
363
basis_id = NULL_REVISION
365
rev.inventory_sha1, inv = repo.add_inventory_by_delta(basis_id,
366
inv_delta, rev.revision_id, rev.parent_ids,
368
parent_invs_cache[rev.revision_id] = inv
369
repo.add_revision(rev.revision_id, rev)
370
if "verify" in debug.debug_flags:
371
new_unusual_modes = mapping.export_unusual_file_modes(rev)
372
if new_unusual_modes != unusual_modes:
373
raise AssertionError("unusual modes don't match: %r != %r" % (unusual_modes, new_unusual_modes))
374
objs = inventory_to_tree_and_blobs(inv, repo.texts, mapping, unusual_modes)
375
for sha1, newobj, path in objs:
376
assert path is not None
377
oldobj = tree_lookup_path(lookup_object, root_trees[revid], path)
379
raise AssertionError("%r != %r in %s" % (oldobj, newobj, path))
381
target_git_object_retriever._idmap.commit()
187
384
class InterGitRepository(InterRepository):
189
_matching_repo_format = GitFormat()
386
_matching_repo_format = GitRepositoryFormat()
192
389
def _get_repo_format_to_test():
196
393
"""See InterRepository.copy_content."""
197
394
self.fetch(revision_id, pb, find_ghosts=False)
199
def fetch(self, revision_id=None, pb=None, find_ghosts=False,
396
def fetch(self, revision_id=None, pb=None, find_ghosts=False, mapping=None,
398
self.fetch_refs(revision_id=revision_id, pb=pb, find_ghosts=find_ghosts,
399
mapping=mapping, fetch_spec=fetch_spec)
402
class InterGitNonGitRepository(InterGitRepository):
403
"""Base InterRepository that copies revisions from a Git into a non-Git
406
def fetch_refs(self, revision_id=None, pb=None, find_ghosts=False,
407
mapping=None, fetch_spec=None):
201
408
if mapping is None:
202
409
mapping = self.source.get_mapping()
204
pb.note("git: %s", text)
205
def determine_wants(heads):
206
if revision_id is None:
410
if revision_id is not None:
411
interesting_heads = [revision_id]
412
elif fetch_spec is not None:
413
interesting_heads = fetch_spec.heads
415
interesting_heads = None
417
def determine_wants(refs):
419
if interesting_heads is None:
420
ret = [sha for (ref, sha) in refs.iteritems() if not ref.endswith("^{}")]
209
ret = [mapping.revision_id_bzr_to_foreign(revision_id)]
422
ret = [mapping.revision_id_bzr_to_foreign(revid)[0] for revid in interesting_heads if revid not in (None, NULL_REVISION)]
210
423
return [rev for rev in ret if not self.target.has_revision(mapping.revision_id_foreign_to_bzr(rev))]
211
graph_walker = BzrFetchGraphWalker(self.target, mapping)
424
pack_hint = self.fetch_objects(determine_wants, mapping, pb)
425
if pack_hint is not None and self.target._format.pack_compresses:
426
self.target.pack(hint=pack_hint)
427
if interesting_heads is not None:
428
present_interesting_heads = self.target.has_revisions(interesting_heads)
429
missing_interesting_heads = set(interesting_heads) - present_interesting_heads
430
if missing_interesting_heads:
431
raise AssertionError("Missing interesting heads: %r" % missing_interesting_heads)
435
_GIT_PROGRESS_RE = re.compile(r"(.*?): +(\d+)% \((\d+)/(\d+)\)")
436
def report_git_progress(pb, text):
437
text = text.rstrip("\r\n")
438
g = _GIT_PROGRESS_RE.match(text)
440
(text, pct, current, total) = g.groups()
441
pb.update(text, int(current), int(total))
443
pb.update(text, 0, 0)
446
class InterRemoteGitNonGitRepository(InterGitNonGitRepository):
447
"""InterRepository that copies revisions from a remote Git into a non-Git
450
def get_target_heads(self):
451
# FIXME: This should be more efficient
452
all_revs = self.target.all_revision_ids()
453
parent_map = self.target.get_parent_map(all_revs)
455
map(all_parents.update, parent_map.itervalues())
456
return set(all_revs) - all_parents
458
def fetch_objects(self, determine_wants, mapping, pb=None):
460
report_git_progress(pb, text)
461
store = BazaarObjectStore(self.target, mapping)
462
self.target.lock_write()
464
heads = self.get_target_heads()
465
graph_walker = store.get_graph_walker(
466
[store._lookup_revision_sha1(head) for head in heads])
469
def record_determine_wants(heads):
470
wants = determine_wants(heads)
471
recorded_wants.extend(wants)
476
create_pb = pb = ui.ui_factory.nested_progress_bar()
478
self.target.start_write_group()
480
objects_iter = self.source.fetch_objects(
481
record_determine_wants, graph_walker,
482
store.get_raw, progress)
483
import_git_objects(self.target, mapping, objects_iter,
484
store, recorded_wants, pb)
486
pack_hint = self.target.commit_write_group()
495
def is_compatible(source, target):
496
"""Be compatible with GitRepository."""
497
# FIXME: Also check target uses VersionedFile
498
return (isinstance(source, RemoteGitRepository) and
499
target.supports_rich_root() and
500
not isinstance(target, GitRepository))
503
class InterLocalGitNonGitRepository(InterGitNonGitRepository):
504
"""InterRepository that copies revisions from a local Git into a non-Git
507
def fetch_objects(self, determine_wants, mapping, pb=None):
508
wants = determine_wants(self.source._git.get_refs())
214
511
create_pb = pb = ui.ui_factory.nested_progress_bar()
512
target_git_object_retriever = BazaarObjectStore(self.target, mapping)
216
514
self.target.lock_write()
218
516
self.target.start_write_group()
220
518
import_git_objects(self.target, mapping,
221
iter(self.source.fetch_objects(determine_wants, graph_walker,
519
self.source._git.object_store,
520
target_git_object_retriever, wants, pb)
224
self.target.commit_write_group()
522
pack_hint = self.target.commit_write_group()
226
525
self.target.unlock()