18
18
"""Support for committing in native Git working trees."""
21
from dulwich.index import (
26
26
from bzrlib.repository import (
30
30
from dulwich.objects import (
38
from bzrlib.plugins.git.mapping import (
43
36
class GitCommitBuilder(CommitBuilder):
45
38
def __init__(self, *args, **kwargs):
46
39
super(GitCommitBuilder, self).__init__(*args, **kwargs)
47
self.store = self.repository._git.object_store
50
def record_entry_contents(self, ie, parent_invs, path, tree,
52
raise NotImplementedError(self.record_entry_contents)
42
def _new_tree(self, path):
44
# FIXME: Inherit children from the base revision
45
self._trees[path] = newtree
48
def _add_tree(self, path):
49
if path in self._trees:
50
return self._trees[path]
52
return self._new_tree("")
53
dirname, basename = osutils.split(path)
54
t = self._add_tree(dirname)
55
assert isinstance(basename, str)
57
newtree = self._new_tree(path)
58
t[basename] = (stat.S_IFDIR, newtree.id)
61
return self.repository._git.object_store[t[basename][1]]
63
def _change_blob(self, path, value):
64
assert isinstance(path, str)
65
dirname, basename = osutils.split(path)
66
t = self._add_tree(dirname)
54
69
def record_delete(self, path, file_id):
55
self._blobs[path] = None
56
self._any_changes = True
70
dirname, basename = osutils.split(path)
71
t = self._add_tree(dirname)
58
74
def record_iter_changes(self, workingtree, basis_revid, iter_changes):
59
index = getattr(workingtree, "index", None)
61
def index_sha1(path, file_id):
62
return index.get_sha1(path.encode("utf-8"))
63
text_sha1 = link_sha1 = index_sha1
65
def link_sha1(path, file_id):
67
blob.data = workingtree.get_symlink_target(file_id)
69
def text_sha1(path, file_id):
71
blob.data = workingtree.get_file_text(file_id, path)
73
75
for (file_id, path, changed_content, versioned, parent, name, kind,
74
76
executable) in iter_changes:
75
77
if kind[1] in ("directory",):
76
78
if kind[0] in ("file", "symlink"):
77
79
self.record_delete(path[0], file_id)
80
self.record_delete(path[0], file_id)
83
82
mode = stat.S_IFREG
84
sha = text_sha1(path[1], file_id)
85
elif kind[1] == "symlink":
86
84
mode = stat.S_IFLNK
87
sha = link_sha1(path[1], file_id)
88
elif kind[1] == "tree-reference":
92
raise AssertionError("Unknown kind %r" % kind[1])
95
self._any_changes = True
96
self._blobs[path[1].encode("utf-8")] = (mode, sha)
87
self._change_blob(path[1].encode("utf-8"), (mode, workingtree.index.get_sha1(path[1].encode("utf-8"))))
97
88
yield file_id, path, (None, None)
98
# Fill in entries that were not changed
99
basis_tree = workingtree.basis_tree()
100
assert basis_tree.get_revision_id() == basis_revid
101
for path, entry in basis_tree.iter_entries_by_dir():
102
if entry.kind not in ("file", "symlink"):
104
if not path in self._blobs:
106
if entry.kind == "symlink":
107
blob.data = basis_tree.get_symlink_target(entry.file_id)
109
blob.data = basis_tree.get_file_text(entry.file_id)
110
self._blobs[path.encode("utf-8")] = (entry_mode(entry), blob.id)
112
def finish_inventory(self):
113
# eliminate blobs that were removed
114
for path, entry in self._blobs.iteritems():
116
del self._blobs[path]
118
90
def commit(self, message):
91
# FIXME: Eliminate any empty trees recursively
92
# Write any tree objects to disk
93
for path in sorted(self._trees.keys(), reverse=True):
94
self.repository._git.object_store.add_object(self._trees[path])
120
c.parents = [self.repository.lookup_git_revid(revid)[0] for revid in self.parents]
121
c.tree = commit_tree(self.store,
122
[(path, sha, mode) for (path, (mode, sha)) in self._blobs.iteritems()])
96
root_tree = self._add_tree("")
123
98
c.committer = self._committer
124
99
c.author = self._revprops.get('author', self._committer)
125
c.commit_time = int(self._timestamp)
126
c.author_time = int(self._timestamp)
100
c.commit_timestamp = self._timestamp
101
c.author_timestamp = self._timestamp
127
102
c.commit_timezone = self._timezone
128
103
c.author_timezone = self._timezone
129
104
c.message = message.encode("utf-8")
130
self.store.add_object(c)
131
return self.repository.get_mapping().revision_id_foreign_to_bzr(c.id)
105
self.repository._git.object_store.add_object(c)