test_fastexport.py 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308
  1. # test_fastexport.py -- Fast export/import functionality
  2. # Copyright (C) 2010 Jelmer Vernooij <jelmer@jelmer.uk>
  3. #
  4. # SPDX-License-Identifier: Apache-2.0 OR GPL-2.0-or-later
  5. # Dulwich is dual-licensed under the Apache License, Version 2.0 and the GNU
  6. # General Public License as published by the Free Software Foundation; version 2.0
  7. # or (at your option) any later version. You can redistribute it and/or
  8. # modify it under the terms of either of these two licenses.
  9. #
  10. # Unless required by applicable law or agreed to in writing, software
  11. # distributed under the License is distributed on an "AS IS" BASIS,
  12. # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13. # See the License for the specific language governing permissions and
  14. # limitations under the License.
  15. #
  16. # You should have received a copy of the licenses; if not, see
  17. # <http://www.gnu.org/licenses/> for a copy of the GNU General Public License
  18. # and <http://www.apache.org/licenses/LICENSE-2.0> for a copy of the Apache
  19. # License, Version 2.0.
  20. #
  21. import stat
  22. from io import BytesIO
  23. from dulwich.object_store import MemoryObjectStore
  24. from dulwich.objects import ZERO_SHA, Blob, Commit, Tree
  25. from dulwich.repo import MemoryRepo
  26. from dulwich.tests.utils import build_commit_graph, make_commit
  27. from . import DependencyMissing, TestCase
  28. class GitFastExporterTests(TestCase):
  29. """Tests for the GitFastExporter tests."""
  30. def setUp(self) -> None:
  31. super().setUp()
  32. self.store = MemoryObjectStore()
  33. self.stream = BytesIO()
  34. try:
  35. from dulwich.fastexport import GitFastExporter
  36. except ImportError as exc:
  37. raise DependencyMissing("python-fastimport") from exc
  38. self.fastexporter = GitFastExporter(self.stream, self.store)
  39. def test_emit_blob(self) -> None:
  40. b = Blob()
  41. b.data = b"fooBAR"
  42. self.fastexporter.emit_blob(b)
  43. self.assertEqual(b"blob\nmark :1\ndata 6\nfooBAR\n", self.stream.getvalue())
  44. def test_emit_commit(self) -> None:
  45. b = Blob()
  46. b.data = b"FOO"
  47. t = Tree()
  48. t.add(b"foo", stat.S_IFREG | 0o644, b.id)
  49. c = make_commit(
  50. author=b"Jelmer <jelmer@host>",
  51. committer=b"Jelmer <jelmer@host>",
  52. author_time=1271345553,
  53. commit_time=1271345553,
  54. author_timezone=0,
  55. commit_timezone=0,
  56. message=b"msg",
  57. tree=t.id,
  58. )
  59. self.store.add_objects([(b, None), (t, None), (c, None)])
  60. self.fastexporter.emit_commit(c, b"refs/heads/master")
  61. self.assertEqual(
  62. b"""blob
  63. mark :1
  64. data 3
  65. FOO
  66. commit refs/heads/master
  67. mark :2
  68. author Jelmer <jelmer@host> 1271345553 +0000
  69. committer Jelmer <jelmer@host> 1271345553 +0000
  70. data 3
  71. msg
  72. M 644 :1 foo
  73. """,
  74. self.stream.getvalue(),
  75. )
  76. class GitImportProcessorTests(TestCase):
  77. """Tests for the GitImportProcessor tests."""
  78. def setUp(self) -> None:
  79. super().setUp()
  80. self.repo = MemoryRepo()
  81. try:
  82. from dulwich.fastexport import GitImportProcessor
  83. except ImportError as exc:
  84. raise DependencyMissing("python-fastimport") from exc
  85. self.processor = GitImportProcessor(self.repo)
  86. def test_reset_handler(self) -> None:
  87. from fastimport import commands
  88. [c1] = build_commit_graph(self.repo.object_store, [[1]])
  89. cmd = commands.ResetCommand(b"refs/heads/foo", c1.id)
  90. self.processor.reset_handler(cmd)
  91. self.assertEqual(c1.id, self.repo.get_refs()[b"refs/heads/foo"])
  92. self.assertEqual(c1.id, self.processor.last_commit)
  93. def test_reset_handler_marker(self) -> None:
  94. from fastimport import commands
  95. [c1, _c2] = build_commit_graph(self.repo.object_store, [[1], [2]])
  96. self.processor.markers[b"10"] = c1.id
  97. cmd = commands.ResetCommand(b"refs/heads/foo", b":10")
  98. self.processor.reset_handler(cmd)
  99. self.assertEqual(c1.id, self.repo.get_refs()[b"refs/heads/foo"])
  100. def test_reset_handler_default(self) -> None:
  101. from fastimport import commands
  102. [_c1, _c2] = build_commit_graph(self.repo.object_store, [[1], [2]])
  103. cmd = commands.ResetCommand(b"refs/heads/foo", None)
  104. self.processor.reset_handler(cmd)
  105. self.assertEqual(ZERO_SHA, self.repo.get_refs()[b"refs/heads/foo"])
  106. def test_commit_handler(self) -> None:
  107. from fastimport import commands
  108. cmd = commands.CommitCommand(
  109. b"refs/heads/foo",
  110. b"mrkr",
  111. (b"Jelmer", b"jelmer@samba.org", 432432432.0, 3600),
  112. (b"Jelmer", b"jelmer@samba.org", 432432432.0, 3600),
  113. b"FOO",
  114. None,
  115. [],
  116. [],
  117. )
  118. self.processor.commit_handler(cmd)
  119. commit = self.repo[self.processor.last_commit]
  120. self.assertEqual(b"Jelmer <jelmer@samba.org>", commit.author)
  121. self.assertEqual(b"Jelmer <jelmer@samba.org>", commit.committer)
  122. self.assertEqual(b"FOO", commit.message)
  123. self.assertEqual([], commit.parents)
  124. self.assertEqual(432432432.0, commit.commit_time)
  125. self.assertEqual(432432432.0, commit.author_time)
  126. self.assertEqual(3600, commit.commit_timezone)
  127. self.assertEqual(3600, commit.author_timezone)
  128. self.assertEqual(commit, self.repo[b"refs/heads/foo"])
  129. def test_commit_handler_markers(self) -> None:
  130. from fastimport import commands
  131. [c1, c2, c3] = build_commit_graph(self.repo.object_store, [[1], [2], [3]])
  132. self.processor.markers[b"10"] = c1.id
  133. self.processor.markers[b"42"] = c2.id
  134. self.processor.markers[b"98"] = c3.id
  135. cmd = commands.CommitCommand(
  136. b"refs/heads/foo",
  137. b"mrkr",
  138. (b"Jelmer", b"jelmer@samba.org", 432432432.0, 3600),
  139. (b"Jelmer", b"jelmer@samba.org", 432432432.0, 3600),
  140. b"FOO",
  141. b":10",
  142. [b":42", b":98"],
  143. [],
  144. )
  145. self.processor.commit_handler(cmd)
  146. commit = self.repo[self.processor.last_commit]
  147. self.assertEqual(c1.id, commit.parents[0])
  148. self.assertEqual(c2.id, commit.parents[1])
  149. self.assertEqual(c3.id, commit.parents[2])
  150. def test_import_stream(self) -> None:
  151. markers = self.processor.import_stream(
  152. BytesIO(
  153. b"""blob
  154. mark :1
  155. data 11
  156. text for a
  157. commit refs/heads/master
  158. mark :2
  159. committer Joe Foo <joe@foo.com> 1288287382 +0000
  160. data 20
  161. <The commit message>
  162. M 100644 :1 a
  163. """
  164. )
  165. )
  166. self.assertEqual(2, len(markers))
  167. self.assertIsInstance(self.repo[markers[b"1"]], Blob)
  168. self.assertIsInstance(self.repo[markers[b"2"]], Commit)
  169. def test_file_add(self) -> None:
  170. from fastimport import commands
  171. cmd = commands.BlobCommand(b"23", b"data")
  172. self.processor.blob_handler(cmd)
  173. cmd = commands.CommitCommand(
  174. b"refs/heads/foo",
  175. b"mrkr",
  176. (b"Jelmer", b"jelmer@samba.org", 432432432.0, 3600),
  177. (b"Jelmer", b"jelmer@samba.org", 432432432.0, 3600),
  178. b"FOO",
  179. None,
  180. [],
  181. [commands.FileModifyCommand(b"path", 0o100644, b":23", None)],
  182. )
  183. self.processor.commit_handler(cmd)
  184. commit = self.repo[self.processor.last_commit]
  185. self.assertEqual(
  186. [(b"path", 0o100644, b"6320cd248dd8aeaab759d5871f8781b5c0505172")],
  187. self.repo[commit.tree].items(),
  188. )
  189. def simple_commit(self):
  190. from fastimport import commands
  191. cmd = commands.BlobCommand(b"23", b"data")
  192. self.processor.blob_handler(cmd)
  193. cmd = commands.CommitCommand(
  194. b"refs/heads/foo",
  195. b"mrkr",
  196. (b"Jelmer", b"jelmer@samba.org", 432432432.0, 3600),
  197. (b"Jelmer", b"jelmer@samba.org", 432432432.0, 3600),
  198. b"FOO",
  199. None,
  200. [],
  201. [commands.FileModifyCommand(b"path", 0o100644, b":23", None)],
  202. )
  203. self.processor.commit_handler(cmd)
  204. commit = self.repo[self.processor.last_commit]
  205. return commit
  206. def make_file_commit(self, file_cmds):
  207. """Create a trivial commit with the specified file commands.
  208. Args:
  209. file_cmds: File commands to run.
  210. Returns: The created commit object
  211. """
  212. from fastimport import commands
  213. cmd = commands.CommitCommand(
  214. b"refs/heads/foo",
  215. b"mrkr",
  216. (b"Jelmer", b"jelmer@samba.org", 432432432.0, 3600),
  217. (b"Jelmer", b"jelmer@samba.org", 432432432.0, 3600),
  218. b"FOO",
  219. None,
  220. [],
  221. file_cmds,
  222. )
  223. self.processor.commit_handler(cmd)
  224. return self.repo[self.processor.last_commit]
  225. def test_file_copy(self) -> None:
  226. from fastimport import commands
  227. self.simple_commit()
  228. commit = self.make_file_commit([commands.FileCopyCommand(b"path", b"new_path")])
  229. self.assertEqual(
  230. [
  231. (
  232. b"new_path",
  233. 0o100644,
  234. b"6320cd248dd8aeaab759d5871f8781b5c0505172",
  235. ),
  236. (
  237. b"path",
  238. 0o100644,
  239. b"6320cd248dd8aeaab759d5871f8781b5c0505172",
  240. ),
  241. ],
  242. self.repo[commit.tree].items(),
  243. )
  244. def test_file_move(self) -> None:
  245. from fastimport import commands
  246. self.simple_commit()
  247. commit = self.make_file_commit(
  248. [commands.FileRenameCommand(b"path", b"new_path")]
  249. )
  250. self.assertEqual(
  251. [
  252. (
  253. b"new_path",
  254. 0o100644,
  255. b"6320cd248dd8aeaab759d5871f8781b5c0505172",
  256. ),
  257. ],
  258. self.repo[commit.tree].items(),
  259. )
  260. def test_file_delete(self) -> None:
  261. from fastimport import commands
  262. self.simple_commit()
  263. commit = self.make_file_commit([commands.FileDeleteCommand(b"path")])
  264. self.assertEqual([], self.repo[commit.tree].items())
  265. def test_file_deleteall(self) -> None:
  266. from fastimport import commands
  267. self.simple_commit()
  268. commit = self.make_file_commit([commands.FileDeleteAllCommand()])
  269. self.assertEqual([], self.repo[commit.tree].items())