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
"""Roundtripping support."""
17
"""Roundtripping support.
19
Bazaar stores more data than Git, which means that in order to preserve
20
a commit when it is pushed from Bazaar into Git we have to stash
21
that extra metadata somewhere.
23
There are two kinds of metadata relevant here:
24
* per-file metadata (stored by revision+path)
25
- usually stored per tree
26
* per-revision metadata (stored by git commit id)
28
Bazaar revisions have the following information that is not
29
present in Git commits:
37
* path last changed revisions [1]
39
[1] path last changed revision information can usually
40
be induced from the existing history, unless
41
ghost revisions are involved.
43
This extra metadata is stored in so-called "supplements":
48
from bzrlib import osutils
20
50
from cStringIO import StringIO
23
class BzrGitRevisionMetadata(object):
24
"""Metadata for a Bazaar revision roundtripped into Git.
53
class CommitSupplement(object):
54
"""Supplement for a Bazaar revision roundtripped into Git.
26
56
:ivar revision_id: Revision id, as string
27
57
:ivar properties: Revision properties, as dictionary
39
69
self.properties = {}
41
71
def __nonzero__(self):
42
return bool(self.revision_id or self.properties)
72
return bool(self.revision_id or self.properties or self.explicit_parent_ids)
75
class TreeSupplement(object):
76
"""Supplement for a Bazaar tree roundtripped into Git.
78
This provides file ids (if they are different from the mapping default)
79
and can provide text revisions.
45
84
def parse_roundtripping_metadata(text):
46
85
"""Parse Bazaar roundtripping metadata."""
47
ret = BzrGitRevisionMetadata()
86
ret = CommitSupplement()
49
88
for l in f.readlines():
50
89
(key, value) = l.split(":", 1)
55
94
elif key == "testament3-sha1":
56
95
ret.verifiers["testament3-sha1"] = value.strip()
57
96
elif key.startswith("property-"):
58
ret.properties[key[len("property-"):]] = value[1:].rstrip("\n")
97
name = key[len("property-"):]
98
if not name in ret.properties:
99
ret.properties[name] = value[1:].rstrip("\n")
101
ret.properties[name] += "\n" + value[1:].rstrip("\n")
73
116
if metadata.explicit_parent_ids:
74
117
lines.append("parent-ids: %s\n" % " ".join(metadata.explicit_parent_ids))
75
118
for key in sorted(metadata.properties.keys()):
76
lines.append("property-%s: %s\n" % (key.encode(encoding), metadata.properties[key].encode(encoding)))
119
for l in metadata.properties[key].split("\n"):
120
lines.append("property-%s: %s\n" % (key.encode(encoding), osutils.safe_utf8(l)))
77
121
if "testament3-sha1" in metadata.verifiers:
78
122
lines.append("testament3-sha1: %s\n" %
79
123
metadata.verifiers["testament3-sha1"])
92
136
return split[0], parse_roundtripping_metadata(split[1])
95
def inject_bzr_metadata(message, metadata, encoding):
139
def inject_bzr_metadata(message, commit_supplement, encoding):
140
if not commit_supplement:
98
rt_data = generate_roundtripping_metadata(metadata, encoding)
142
rt_data = generate_roundtripping_metadata(commit_supplement, encoding)
101
145
assert type(rt_data) == str