test_objects.py 73 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011
  1. # test_objects.py -- tests for objects.py
  2. # Copyright (C) 2007 James Westby <jw+debian@jameswestby.net>
  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. """Tests for git base objects."""
  22. # TODO: Round-trip parse-serialize-parse and serialize-parse-serialize tests.
  23. import datetime
  24. import os
  25. import stat
  26. from contextlib import contextmanager
  27. from io import BytesIO
  28. from itertools import permutations
  29. from dulwich.errors import ObjectFormatException
  30. from dulwich.objects import (
  31. MAX_TIME,
  32. ZERO_SHA,
  33. Blob,
  34. Commit,
  35. ShaFile,
  36. Tag,
  37. Tree,
  38. TreeEntry,
  39. _parse_tree_py,
  40. _sorted_tree_items_py,
  41. check_hexsha,
  42. check_identity,
  43. format_timezone,
  44. hex_to_filename,
  45. hex_to_sha,
  46. key_entry,
  47. object_class,
  48. parse_timezone,
  49. pretty_format_tree_entry,
  50. sha_to_hex,
  51. )
  52. try:
  53. from dulwich.objects import _parse_tree_rs, _sorted_tree_items_rs
  54. except ImportError:
  55. _sorted_tree_items_rs = _parse_tree_rs = None
  56. from dulwich.tests.utils import (
  57. ext_functest_builder,
  58. functest_builder,
  59. make_commit,
  60. make_object,
  61. )
  62. from . import TestCase
  63. a_sha = b"6f670c0fb53f9463760b7295fbb814e965fb20c8"
  64. b_sha = b"2969be3e8ee1c0222396a5611407e4769f14e54b"
  65. c_sha = b"954a536f7819d40e6f637f849ee187dd10066349"
  66. tree_sha = b"70c190eb48fa8bbb50ddc692a17b44cb781af7f6"
  67. tag_sha = b"71033db03a03c6a36721efcf1968dd8f8e0cf023"
  68. class TestHexToSha(TestCase):
  69. def test_simple(self) -> None:
  70. self.assertEqual(b"\xab\xcd" * 10, hex_to_sha(b"abcd" * 10))
  71. def test_reverse(self) -> None:
  72. self.assertEqual(b"abcd" * 10, sha_to_hex(b"\xab\xcd" * 10))
  73. class BlobReadTests(TestCase):
  74. """Test decompression of blobs."""
  75. def get_sha_file(self, cls, base, sha):
  76. dir = os.path.join(os.path.dirname(__file__), "..", "testdata", base)
  77. return cls.from_path(hex_to_filename(dir, sha))
  78. def get_blob(self, sha):
  79. """Return the blob named sha from the test data dir."""
  80. return self.get_sha_file(Blob, "blobs", sha)
  81. def get_tree(self, sha):
  82. return self.get_sha_file(Tree, "trees", sha)
  83. def get_tag(self, sha):
  84. return self.get_sha_file(Tag, "tags", sha)
  85. def commit(self, sha):
  86. return self.get_sha_file(Commit, "commits", sha)
  87. def test_decompress_simple_blob(self) -> None:
  88. b = self.get_blob(a_sha)
  89. self.assertEqual(b.data, b"test 1\n")
  90. self.assertEqual(b.sha().hexdigest().encode("ascii"), a_sha)
  91. def test_hash(self) -> None:
  92. b = self.get_blob(a_sha)
  93. self.assertEqual(hash(b.id), hash(b))
  94. def test_parse_empty_blob_object(self) -> None:
  95. sha = b"e69de29bb2d1d6434b8b29ae775ad8c2e48c5391"
  96. b = self.get_blob(sha)
  97. self.assertEqual(b.data, b"")
  98. self.assertEqual(b.id, sha)
  99. self.assertEqual(b.sha().hexdigest().encode("ascii"), sha)
  100. def test_create_blob_from_string(self) -> None:
  101. string = b"test 2\n"
  102. b = Blob.from_string(string)
  103. self.assertEqual(b.data, string)
  104. self.assertEqual(b.sha().hexdigest().encode("ascii"), b_sha)
  105. def test_legacy_from_file(self) -> None:
  106. b1 = Blob.from_string(b"foo")
  107. b_raw = b1.as_legacy_object()
  108. b2 = b1.from_file(BytesIO(b_raw))
  109. self.assertEqual(b1, b2)
  110. def test_legacy_from_file_compression_level(self) -> None:
  111. b1 = Blob.from_string(b"foo")
  112. b_raw = b1.as_legacy_object(compression_level=6)
  113. b2 = b1.from_file(BytesIO(b_raw))
  114. self.assertEqual(b1, b2)
  115. def test_chunks(self) -> None:
  116. string = b"test 5\n"
  117. b = Blob.from_string(string)
  118. self.assertEqual([string], b.chunked)
  119. def test_splitlines(self) -> None:
  120. for case in [
  121. [],
  122. [b"foo\nbar\n"],
  123. [b"bl\na", b"blie"],
  124. [b"bl\na", b"blie", b"bloe\n"],
  125. [b"", b"bl\na", b"blie", b"bloe\n"],
  126. [b"", b"", b"", b"bla\n"],
  127. [b"", b"", b"", b"bla\n", b""],
  128. [b"bl", b"", b"a\naaa"],
  129. [b"a\naaa", b"a"],
  130. ]:
  131. b = Blob()
  132. b.chunked = case
  133. self.assertEqual(b.data.splitlines(True), b.splitlines())
  134. def test_set_chunks(self) -> None:
  135. b = Blob()
  136. b.chunked = [b"te", b"st", b" 5\n"]
  137. self.assertEqual(b"test 5\n", b.data)
  138. b.chunked = [b"te", b"st", b" 6\n"]
  139. self.assertEqual(b"test 6\n", b.as_raw_string())
  140. self.assertEqual(b"test 6\n", bytes(b))
  141. def test_parse_legacy_blob(self) -> None:
  142. string = b"test 3\n"
  143. b = self.get_blob(c_sha)
  144. self.assertEqual(b.data, string)
  145. self.assertEqual(b.sha().hexdigest().encode("ascii"), c_sha)
  146. def test_eq(self) -> None:
  147. blob1 = self.get_blob(a_sha)
  148. blob2 = self.get_blob(a_sha)
  149. self.assertEqual(blob1, blob2)
  150. def test_read_tree_from_file(self) -> None:
  151. t = self.get_tree(tree_sha)
  152. self.assertEqual(t.items()[0], (b"a", 33188, a_sha))
  153. self.assertEqual(t.items()[1], (b"b", 33188, b_sha))
  154. def test_read_tree_from_file_parse_count(self) -> None:
  155. old_deserialize = Tree._deserialize
  156. def reset_deserialize() -> None:
  157. Tree._deserialize = old_deserialize
  158. self.addCleanup(reset_deserialize)
  159. self.deserialize_count = 0
  160. def counting_deserialize(*args, **kwargs):
  161. self.deserialize_count += 1
  162. return old_deserialize(*args, **kwargs)
  163. Tree._deserialize = counting_deserialize
  164. t = self.get_tree(tree_sha)
  165. self.assertEqual(t.items()[0], (b"a", 33188, a_sha))
  166. self.assertEqual(t.items()[1], (b"b", 33188, b_sha))
  167. self.assertEqual(self.deserialize_count, 1)
  168. def test_read_tag_from_file(self) -> None:
  169. t = self.get_tag(tag_sha)
  170. self.assertEqual(
  171. t.object, (Commit, b"51b668fd5bf7061b7d6fa525f88803e6cfadaa51")
  172. )
  173. self.assertEqual(t.name, b"signed")
  174. self.assertEqual(t.tagger, b"Ali Sabil <ali.sabil@gmail.com>")
  175. self.assertEqual(t.tag_time, 1231203091)
  176. self.assertEqual(t.message, b"This is a signed tag\n")
  177. self.assertEqual(
  178. t.signature,
  179. b"-----BEGIN PGP SIGNATURE-----\n"
  180. b"Version: GnuPG v1.4.9 (GNU/Linux)\n"
  181. b"\n"
  182. b"iEYEABECAAYFAkliqx8ACgkQqSMmLy9u/"
  183. b"kcx5ACfakZ9NnPl02tOyYP6pkBoEkU1\n"
  184. b"5EcAn0UFgokaSvS371Ym/4W9iJj6vh3h\n"
  185. b"=ql7y\n"
  186. b"-----END PGP SIGNATURE-----\n",
  187. )
  188. self.assertEqual(t.raw_without_sig() + t.signature, bytes(t))
  189. def test_read_commit_from_file(self) -> None:
  190. sha = b"60dacdc733de308bb77bb76ce0fb0f9b44c9769e"
  191. c = self.commit(sha)
  192. self.assertEqual(c.tree, tree_sha)
  193. self.assertEqual(c.parents, [b"0d89f20333fbb1d2f3a94da77f4981373d8f4310"])
  194. self.assertEqual(c.author, b"James Westby <jw+debian@jameswestby.net>")
  195. self.assertEqual(c.committer, b"James Westby <jw+debian@jameswestby.net>")
  196. self.assertEqual(c.commit_time, 1174759230)
  197. self.assertEqual(c.commit_timezone, 0)
  198. self.assertEqual(c.author_timezone, 0)
  199. self.assertEqual(c.message, b"Test commit\n")
  200. def test_read_commit_no_parents(self) -> None:
  201. sha = b"0d89f20333fbb1d2f3a94da77f4981373d8f4310"
  202. c = self.commit(sha)
  203. self.assertEqual(c.tree, b"90182552c4a85a45ec2a835cadc3451bebdfe870")
  204. self.assertEqual(c.parents, [])
  205. self.assertEqual(c.author, b"James Westby <jw+debian@jameswestby.net>")
  206. self.assertEqual(c.committer, b"James Westby <jw+debian@jameswestby.net>")
  207. self.assertEqual(c.commit_time, 1174758034)
  208. self.assertEqual(c.commit_timezone, 0)
  209. self.assertEqual(c.author_timezone, 0)
  210. self.assertEqual(c.message, b"Test commit\n")
  211. def test_read_commit_two_parents(self) -> None:
  212. sha = b"5dac377bdded4c9aeb8dff595f0faeebcc8498cc"
  213. c = self.commit(sha)
  214. self.assertEqual(c.tree, b"d80c186a03f423a81b39df39dc87fd269736ca86")
  215. self.assertEqual(
  216. c.parents,
  217. [
  218. b"ab64bbdcc51b170d21588e5c5d391ee5c0c96dfd",
  219. b"4cffe90e0a41ad3f5190079d7c8f036bde29cbe6",
  220. ],
  221. )
  222. self.assertEqual(c.author, b"James Westby <jw+debian@jameswestby.net>")
  223. self.assertEqual(c.committer, b"James Westby <jw+debian@jameswestby.net>")
  224. self.assertEqual(c.commit_time, 1174773719)
  225. self.assertEqual(c.commit_timezone, 0)
  226. self.assertEqual(c.author_timezone, 0)
  227. self.assertEqual(c.message, b"Merge ../b\n")
  228. def test_stub_sha(self) -> None:
  229. sha = b"5" * 40
  230. c = make_commit(id=sha, message=b"foo")
  231. self.assertIsInstance(c, Commit)
  232. self.assertEqual(sha, c.id)
  233. self.assertNotEqual(sha, c.sha())
  234. class ShaFileCheckTests(TestCase):
  235. def assertCheckFails(self, cls, data) -> None:
  236. obj = cls()
  237. def do_check() -> None:
  238. obj.set_raw_string(data)
  239. obj.check()
  240. self.assertRaises(ObjectFormatException, do_check)
  241. def assertCheckSucceeds(self, cls, data) -> None:
  242. obj = cls()
  243. obj.set_raw_string(data)
  244. self.assertEqual(None, obj.check())
  245. small_buffer_zlib_object = (
  246. b"\x48\x89\x15\xcc\x31\x0e\xc2\x30\x0c\x40\x51\xe6"
  247. b"\x9c\xc2\x3b\xaa\x64\x37\xc4\xc1\x12\x42\x5c\xc5"
  248. b"\x49\xac\x52\xd4\x92\xaa\x78\xe1\xf6\x94\xed\xeb"
  249. b"\x0d\xdf\x75\x02\xa2\x7c\xea\xe5\x65\xd5\x81\x8b"
  250. b"\x9a\x61\xba\xa0\xa9\x08\x36\xc9\x4c\x1a\xad\x88"
  251. b"\x16\xba\x46\xc4\xa8\x99\x6a\x64\xe1\xe0\xdf\xcd"
  252. b"\xa0\xf6\x75\x9d\x3d\xf8\xf1\xd0\x77\xdb\xfb\xdc"
  253. b"\x86\xa3\x87\xf1\x2f\x93\xed\x00\xb7\xc7\xd2\xab"
  254. b"\x2e\xcf\xfe\xf1\x3b\x50\xa4\x91\x53\x12\x24\x38"
  255. b"\x23\x21\x86\xf0\x03\x2f\x91\x24\x52"
  256. )
  257. class ShaFileTests(TestCase):
  258. def test_deflated_smaller_window_buffer(self) -> None:
  259. # zlib on some systems uses smaller buffers,
  260. # resulting in a different header.
  261. # See https://github.com/libgit2/libgit2/pull/464
  262. sf = ShaFile.from_file(BytesIO(small_buffer_zlib_object))
  263. self.assertEqual(sf.type_name, b"tag")
  264. self.assertEqual(sf.tagger, b" <@localhost>")
  265. class CommitSerializationTests(TestCase):
  266. def make_commit(self, **kwargs):
  267. attrs = {
  268. "tree": b"d80c186a03f423a81b39df39dc87fd269736ca86",
  269. "parents": [
  270. b"ab64bbdcc51b170d21588e5c5d391ee5c0c96dfd",
  271. b"4cffe90e0a41ad3f5190079d7c8f036bde29cbe6",
  272. ],
  273. "author": b"James Westby <jw+debian@jameswestby.net>",
  274. "committer": b"James Westby <jw+debian@jameswestby.net>",
  275. "commit_time": 1174773719,
  276. "author_time": 1174773719,
  277. "commit_timezone": 0,
  278. "author_timezone": 0,
  279. "message": b"Merge ../b\n",
  280. }
  281. attrs.update(kwargs)
  282. return make_commit(**attrs)
  283. def test_encoding(self) -> None:
  284. c = self.make_commit(encoding=b"iso8859-1")
  285. self.assertIn(b"encoding iso8859-1\n", c.as_raw_string())
  286. def test_short_timestamp(self) -> None:
  287. c = self.make_commit(commit_time=30)
  288. c1 = Commit()
  289. c1.set_raw_string(c.as_raw_string())
  290. self.assertEqual(30, c1.commit_time)
  291. def test_full_tree(self) -> None:
  292. c = self.make_commit(commit_time=30)
  293. t = Tree()
  294. t.add(b"data-x", 0o644, Blob().id)
  295. c.tree = t
  296. c1 = Commit()
  297. c1.set_raw_string(c.as_raw_string())
  298. self.assertEqual(t.id, c1.tree)
  299. self.assertEqual(c.as_raw_string(), c1.as_raw_string())
  300. def test_raw_length(self) -> None:
  301. c = self.make_commit()
  302. self.assertEqual(len(c.as_raw_string()), c.raw_length())
  303. def test_simple(self) -> None:
  304. c = self.make_commit()
  305. self.assertEqual(c.id, b"5dac377bdded4c9aeb8dff595f0faeebcc8498cc")
  306. self.assertEqual(
  307. b"tree d80c186a03f423a81b39df39dc87fd269736ca86\n"
  308. b"parent ab64bbdcc51b170d21588e5c5d391ee5c0c96dfd\n"
  309. b"parent 4cffe90e0a41ad3f5190079d7c8f036bde29cbe6\n"
  310. b"author James Westby <jw+debian@jameswestby.net> "
  311. b"1174773719 +0000\n"
  312. b"committer James Westby <jw+debian@jameswestby.net> "
  313. b"1174773719 +0000\n"
  314. b"\n"
  315. b"Merge ../b\n",
  316. c.as_raw_string(),
  317. )
  318. def test_timezone(self) -> None:
  319. c = self.make_commit(commit_timezone=(5 * 60))
  320. self.assertIn(b" +0005\n", c.as_raw_string())
  321. def test_neg_timezone(self) -> None:
  322. c = self.make_commit(commit_timezone=(-1 * 3600))
  323. self.assertIn(b" -0100\n", c.as_raw_string())
  324. def test_deserialize(self) -> None:
  325. c = self.make_commit()
  326. d = Commit()
  327. d._deserialize(c.as_raw_chunks())
  328. self.assertEqual(c, d)
  329. def test_serialize_gpgsig(self) -> None:
  330. gpgsig = b"""-----BEGIN PGP SIGNATURE-----
  331. Version: GnuPG v1
  332. iQIcBAABCgAGBQJULCdfAAoJEACAbyvXKaRXuKwP/RyP9PA49uAvu8tQVCC/uBa8
  333. vi975+xvO14R8Pp8k2nps7lSxCdtCd+xVT1VRHs0wNhOZo2YCVoU1HATkPejqSeV
  334. NScTHcxnk4/+bxyfk14xvJkNp7FlQ3npmBkA+lbV0Ubr33rvtIE5jiJPyz+SgWAg
  335. xdBG2TojV0squj00GoH/euK6aX7GgZtwdtpTv44haCQdSuPGDcI4TORqR6YSqvy3
  336. GPE+3ZqXPFFb+KILtimkxitdwB7CpwmNse2vE3rONSwTvi8nq3ZoQYNY73CQGkUy
  337. qoFU0pDtw87U3niFin1ZccDgH0bB6624sLViqrjcbYJeg815Htsu4rmzVaZADEVC
  338. XhIO4MThebusdk0AcNGjgpf3HRHk0DPMDDlIjm+Oao0cqovvF6VyYmcb0C+RmhJj
  339. dodLXMNmbqErwTk3zEkW0yZvNIYXH7m9SokPCZa4eeIM7be62X6h1mbt0/IU6Th+
  340. v18fS0iTMP/Viug5und+05C/v04kgDo0CPphAbXwWMnkE4B6Tl9sdyUYXtvQsL7x
  341. 0+WP1gL27ANqNZiI07Kz/BhbBAQI/+2TFT7oGr0AnFPQ5jHp+3GpUf6OKuT1wT3H
  342. ND189UFuRuubxb42vZhpcXRbqJVWnbECTKVUPsGZqat3enQUB63uM4i6/RdONDZA
  343. fDeF1m4qYs+cUXKNUZ03
  344. =X6RT
  345. -----END PGP SIGNATURE-----"""
  346. pre_sig = b"""\
  347. tree d80c186a03f423a81b39df39dc87fd269736ca86
  348. parent ab64bbdcc51b170d21588e5c5d391ee5c0c96dfd
  349. parent 4cffe90e0a41ad3f5190079d7c8f036bde29cbe6
  350. author James Westby <jw+debian@jameswestby.net> 1174773719 +0000
  351. committer James Westby <jw+debian@jameswestby.net> 1174773719 +0000
  352. """
  353. git_sig = b"""\
  354. gpgsig -----BEGIN PGP SIGNATURE-----
  355. Version: GnuPG v1
  356. iQIcBAABCgAGBQJULCdfAAoJEACAbyvXKaRXuKwP/RyP9PA49uAvu8tQVCC/uBa8
  357. vi975+xvO14R8Pp8k2nps7lSxCdtCd+xVT1VRHs0wNhOZo2YCVoU1HATkPejqSeV
  358. NScTHcxnk4/+bxyfk14xvJkNp7FlQ3npmBkA+lbV0Ubr33rvtIE5jiJPyz+SgWAg
  359. xdBG2TojV0squj00GoH/euK6aX7GgZtwdtpTv44haCQdSuPGDcI4TORqR6YSqvy3
  360. GPE+3ZqXPFFb+KILtimkxitdwB7CpwmNse2vE3rONSwTvi8nq3ZoQYNY73CQGkUy
  361. qoFU0pDtw87U3niFin1ZccDgH0bB6624sLViqrjcbYJeg815Htsu4rmzVaZADEVC
  362. XhIO4MThebusdk0AcNGjgpf3HRHk0DPMDDlIjm+Oao0cqovvF6VyYmcb0C+RmhJj
  363. dodLXMNmbqErwTk3zEkW0yZvNIYXH7m9SokPCZa4eeIM7be62X6h1mbt0/IU6Th+
  364. v18fS0iTMP/Viug5und+05C/v04kgDo0CPphAbXwWMnkE4B6Tl9sdyUYXtvQsL7x
  365. 0+WP1gL27ANqNZiI07Kz/BhbBAQI/+2TFT7oGr0AnFPQ5jHp+3GpUf6OKuT1wT3H
  366. ND189UFuRuubxb42vZhpcXRbqJVWnbECTKVUPsGZqat3enQUB63uM4i6/RdONDZA
  367. fDeF1m4qYs+cUXKNUZ03
  368. =X6RT
  369. -----END PGP SIGNATURE-----
  370. """
  371. post_sig = b"""\
  372. Merge ../b
  373. """
  374. commit = self.make_commit(gpgsig=gpgsig)
  375. self.maxDiff = None
  376. self.assertEqual(pre_sig + git_sig + post_sig, commit.as_raw_string())
  377. self.assertEqual(pre_sig + post_sig, commit.raw_without_sig())
  378. self.assertEqual(gpgsig, commit.gpgsig)
  379. self.assertEqual(b"Merge ../b\n", commit.message)
  380. def test_serialize_mergetag(self) -> None:
  381. tag = make_object(
  382. Tag,
  383. object=(Commit, b"a38d6181ff27824c79fc7df825164a212eff6a3f"),
  384. object_type_name=b"commit",
  385. name=b"v2.6.22-rc7",
  386. tag_time=1183319674,
  387. tag_timezone=0,
  388. tagger=b"Linus Torvalds <torvalds@woody.linux-foundation.org>",
  389. message=default_message,
  390. )
  391. commit = self.make_commit(mergetag=[tag])
  392. self.assertEqual(
  393. b"""tree d80c186a03f423a81b39df39dc87fd269736ca86
  394. parent ab64bbdcc51b170d21588e5c5d391ee5c0c96dfd
  395. parent 4cffe90e0a41ad3f5190079d7c8f036bde29cbe6
  396. author James Westby <jw+debian@jameswestby.net> 1174773719 +0000
  397. committer James Westby <jw+debian@jameswestby.net> 1174773719 +0000
  398. mergetag object a38d6181ff27824c79fc7df825164a212eff6a3f
  399. type commit
  400. tag v2.6.22-rc7
  401. tagger Linus Torvalds <torvalds@woody.linux-foundation.org> 1183319674 +0000
  402. Linux 2.6.22-rc7
  403. -----BEGIN PGP SIGNATURE-----
  404. Version: GnuPG v1.4.7 (GNU/Linux)
  405. iD8DBQBGiAaAF3YsRnbiHLsRAitMAKCiLboJkQECM/jpYsY3WPfvUgLXkACgg3ql
  406. OK2XeQOiEeXtT76rV4t2WR4=
  407. =ivrA
  408. -----END PGP SIGNATURE-----
  409. Merge ../b
  410. """,
  411. commit.as_raw_string(),
  412. )
  413. def test_serialize_mergetags(self) -> None:
  414. tag = make_object(
  415. Tag,
  416. object=(Commit, b"a38d6181ff27824c79fc7df825164a212eff6a3f"),
  417. object_type_name=b"commit",
  418. name=b"v2.6.22-rc7",
  419. tag_time=1183319674,
  420. tag_timezone=0,
  421. tagger=b"Linus Torvalds <torvalds@woody.linux-foundation.org>",
  422. message=default_message,
  423. )
  424. commit = self.make_commit(mergetag=[tag, tag])
  425. self.assertEqual(
  426. b"""tree d80c186a03f423a81b39df39dc87fd269736ca86
  427. parent ab64bbdcc51b170d21588e5c5d391ee5c0c96dfd
  428. parent 4cffe90e0a41ad3f5190079d7c8f036bde29cbe6
  429. author James Westby <jw+debian@jameswestby.net> 1174773719 +0000
  430. committer James Westby <jw+debian@jameswestby.net> 1174773719 +0000
  431. mergetag object a38d6181ff27824c79fc7df825164a212eff6a3f
  432. type commit
  433. tag v2.6.22-rc7
  434. tagger Linus Torvalds <torvalds@woody.linux-foundation.org> 1183319674 +0000
  435. Linux 2.6.22-rc7
  436. -----BEGIN PGP SIGNATURE-----
  437. Version: GnuPG v1.4.7 (GNU/Linux)
  438. iD8DBQBGiAaAF3YsRnbiHLsRAitMAKCiLboJkQECM/jpYsY3WPfvUgLXkACgg3ql
  439. OK2XeQOiEeXtT76rV4t2WR4=
  440. =ivrA
  441. -----END PGP SIGNATURE-----
  442. mergetag object a38d6181ff27824c79fc7df825164a212eff6a3f
  443. type commit
  444. tag v2.6.22-rc7
  445. tagger Linus Torvalds <torvalds@woody.linux-foundation.org> 1183319674 +0000
  446. Linux 2.6.22-rc7
  447. -----BEGIN PGP SIGNATURE-----
  448. Version: GnuPG v1.4.7 (GNU/Linux)
  449. iD8DBQBGiAaAF3YsRnbiHLsRAitMAKCiLboJkQECM/jpYsY3WPfvUgLXkACgg3ql
  450. OK2XeQOiEeXtT76rV4t2WR4=
  451. =ivrA
  452. -----END PGP SIGNATURE-----
  453. Merge ../b
  454. """,
  455. commit.as_raw_string(),
  456. )
  457. def test_deserialize_mergetag(self) -> None:
  458. tag = make_object(
  459. Tag,
  460. object=(Commit, b"a38d6181ff27824c79fc7df825164a212eff6a3f"),
  461. object_type_name=b"commit",
  462. name=b"v2.6.22-rc7",
  463. tag_time=1183319674,
  464. tag_timezone=0,
  465. tagger=b"Linus Torvalds <torvalds@woody.linux-foundation.org>",
  466. message=default_message,
  467. )
  468. commit = self.make_commit(mergetag=[tag])
  469. d = Commit()
  470. d._deserialize(commit.as_raw_chunks())
  471. self.assertEqual(commit, d)
  472. def test_deserialize_mergetags(self) -> None:
  473. tag = make_object(
  474. Tag,
  475. object=(Commit, b"a38d6181ff27824c79fc7df825164a212eff6a3f"),
  476. object_type_name=b"commit",
  477. name=b"v2.6.22-rc7",
  478. tag_time=1183319674,
  479. tag_timezone=0,
  480. tagger=b"Linus Torvalds <torvalds@woody.linux-foundation.org>",
  481. message=default_message,
  482. )
  483. commit = self.make_commit(mergetag=[tag, tag])
  484. d = Commit()
  485. d._deserialize(commit.as_raw_chunks())
  486. self.assertEqual(commit, d)
  487. default_committer = b"James Westby <jw+debian@jameswestby.net> 1174773719 +0000"
  488. class CommitParseTests(ShaFileCheckTests):
  489. def make_commit_lines(
  490. self,
  491. tree=b"d80c186a03f423a81b39df39dc87fd269736ca86",
  492. parents=[
  493. b"ab64bbdcc51b170d21588e5c5d391ee5c0c96dfd",
  494. b"4cffe90e0a41ad3f5190079d7c8f036bde29cbe6",
  495. ],
  496. author=default_committer,
  497. committer=default_committer,
  498. encoding=None,
  499. message=b"Merge ../b\n",
  500. extra=None,
  501. ):
  502. lines = []
  503. if tree is not None:
  504. lines.append(b"tree " + tree)
  505. if parents is not None:
  506. lines.extend(b"parent " + p for p in parents)
  507. if author is not None:
  508. lines.append(b"author " + author)
  509. if committer is not None:
  510. lines.append(b"committer " + committer)
  511. if encoding is not None:
  512. lines.append(b"encoding " + encoding)
  513. if extra is not None:
  514. for name, value in sorted(extra.items()):
  515. lines.append(name + b" " + value)
  516. lines.append(b"")
  517. if message is not None:
  518. lines.append(message)
  519. return lines
  520. def make_commit_text(self, **kwargs):
  521. return b"\n".join(self.make_commit_lines(**kwargs))
  522. def test_simple(self) -> None:
  523. c = Commit.from_string(self.make_commit_text())
  524. self.assertEqual(b"Merge ../b\n", c.message)
  525. self.assertEqual(b"James Westby <jw+debian@jameswestby.net>", c.author)
  526. self.assertEqual(b"James Westby <jw+debian@jameswestby.net>", c.committer)
  527. self.assertEqual(b"d80c186a03f423a81b39df39dc87fd269736ca86", c.tree)
  528. self.assertEqual(
  529. [
  530. b"ab64bbdcc51b170d21588e5c5d391ee5c0c96dfd",
  531. b"4cffe90e0a41ad3f5190079d7c8f036bde29cbe6",
  532. ],
  533. c.parents,
  534. )
  535. expected_time = datetime.datetime(2007, 3, 24, 22, 1, 59)
  536. self.assertEqual(
  537. expected_time,
  538. datetime.datetime.fromtimestamp(
  539. c.commit_time, datetime.timezone.utc
  540. ).replace(tzinfo=None),
  541. )
  542. self.assertEqual(0, c.commit_timezone)
  543. self.assertEqual(
  544. expected_time,
  545. datetime.datetime.fromtimestamp(
  546. c.author_time, datetime.timezone.utc
  547. ).replace(tzinfo=None),
  548. )
  549. self.assertEqual(0, c.author_timezone)
  550. self.assertEqual(None, c.encoding)
  551. def test_custom(self) -> None:
  552. c = Commit.from_string(self.make_commit_text(extra={b"extra-field": b"data"}))
  553. self.assertEqual([(b"extra-field", b"data")], c._extra)
  554. def test_encoding(self) -> None:
  555. c = Commit.from_string(self.make_commit_text(encoding=b"UTF-8"))
  556. self.assertEqual(b"UTF-8", c.encoding)
  557. def test_check(self) -> None:
  558. self.assertCheckSucceeds(Commit, self.make_commit_text())
  559. self.assertCheckSucceeds(Commit, self.make_commit_text(parents=None))
  560. self.assertCheckSucceeds(Commit, self.make_commit_text(encoding=b"UTF-8"))
  561. self.assertCheckFails(Commit, self.make_commit_text(tree=b"xxx"))
  562. self.assertCheckFails(Commit, self.make_commit_text(parents=[a_sha, b"xxx"]))
  563. bad_committer = b"some guy without an email address 1174773719 +0000"
  564. self.assertCheckFails(Commit, self.make_commit_text(committer=bad_committer))
  565. self.assertCheckFails(Commit, self.make_commit_text(author=bad_committer))
  566. self.assertCheckFails(Commit, self.make_commit_text(author=None))
  567. self.assertCheckFails(Commit, self.make_commit_text(committer=None))
  568. self.assertCheckFails(
  569. Commit, self.make_commit_text(author=None, committer=None)
  570. )
  571. def test_check_duplicates(self) -> None:
  572. # duplicate each of the header fields
  573. for i in range(5):
  574. lines = self.make_commit_lines(parents=[a_sha], encoding=b"UTF-8")
  575. lines.insert(i, lines[i])
  576. text = b"\n".join(lines)
  577. if lines[i].startswith(b"parent"):
  578. # duplicate parents are ok for now
  579. self.assertCheckSucceeds(Commit, text)
  580. else:
  581. self.assertCheckFails(Commit, text)
  582. def test_check_order(self) -> None:
  583. lines = self.make_commit_lines(parents=[a_sha], encoding=b"UTF-8")
  584. headers = lines[:5]
  585. rest = lines[5:]
  586. # of all possible permutations, ensure only the original succeeds
  587. for perm in permutations(headers):
  588. perm = list(perm)
  589. text = b"\n".join(perm + rest)
  590. if perm == headers:
  591. self.assertCheckSucceeds(Commit, text)
  592. else:
  593. self.assertCheckFails(Commit, text)
  594. def test_check_commit_with_unparseable_time(self) -> None:
  595. identity_with_wrong_time = (
  596. b"Igor Sysoev <igor@sysoev.ru> 18446743887488505614+42707004"
  597. )
  598. # Those fail at reading time
  599. self.assertCheckFails(
  600. Commit,
  601. self.make_commit_text(
  602. author=default_committer, committer=identity_with_wrong_time
  603. ),
  604. )
  605. self.assertCheckFails(
  606. Commit,
  607. self.make_commit_text(
  608. author=identity_with_wrong_time, committer=default_committer
  609. ),
  610. )
  611. def test_check_commit_with_overflow_date(self) -> None:
  612. """Date with overflow should raise an ObjectFormatException when checked."""
  613. identity_with_wrong_time = (
  614. b"Igor Sysoev <igor@sysoev.ru> 18446743887488505614 +42707004"
  615. )
  616. commit0 = Commit.from_string(
  617. self.make_commit_text(
  618. author=identity_with_wrong_time, committer=default_committer
  619. )
  620. )
  621. commit1 = Commit.from_string(
  622. self.make_commit_text(
  623. author=default_committer, committer=identity_with_wrong_time
  624. )
  625. )
  626. # Those fails when triggering the check() method
  627. for commit in [commit0, commit1]:
  628. with self.assertRaises(ObjectFormatException):
  629. commit.check()
  630. def test_mangled_author_line(self) -> None:
  631. """Mangled author line should successfully parse."""
  632. author_line = (
  633. b'Karl MacMillan <kmacmill@redhat.com> <"Karl MacMillan '
  634. b'<kmacmill@redhat.com>"> 1197475547 -0500'
  635. )
  636. expected_identity = (
  637. b'Karl MacMillan <kmacmill@redhat.com> <"Karl MacMillan '
  638. b'<kmacmill@redhat.com>">'
  639. )
  640. commit = Commit.from_string(self.make_commit_text(author=author_line))
  641. # The commit parses properly
  642. self.assertEqual(commit.author, expected_identity)
  643. # But the check fails because the author identity is bogus
  644. with self.assertRaises(ObjectFormatException):
  645. commit.check()
  646. def test_parse_gpgsig(self) -> None:
  647. pre_sig = b"""tree aaff74984cccd156a469afa7d9ab10e4777beb24
  648. author Jelmer Vernooij <jelmer@samba.org> 1412179807 +0200
  649. committer Jelmer Vernooij <jelmer@samba.org> 1412179807 +0200
  650. """
  651. git_sig = b"""\
  652. gpgsig -----BEGIN PGP SIGNATURE-----
  653. Version: GnuPG v1
  654. iQIcBAABCgAGBQJULCdfAAoJEACAbyvXKaRXuKwP/RyP9PA49uAvu8tQVCC/uBa8
  655. vi975+xvO14R8Pp8k2nps7lSxCdtCd+xVT1VRHs0wNhOZo2YCVoU1HATkPejqSeV
  656. NScTHcxnk4/+bxyfk14xvJkNp7FlQ3npmBkA+lbV0Ubr33rvtIE5jiJPyz+SgWAg
  657. xdBG2TojV0squj00GoH/euK6aX7GgZtwdtpTv44haCQdSuPGDcI4TORqR6YSqvy3
  658. GPE+3ZqXPFFb+KILtimkxitdwB7CpwmNse2vE3rONSwTvi8nq3ZoQYNY73CQGkUy
  659. qoFU0pDtw87U3niFin1ZccDgH0bB6624sLViqrjcbYJeg815Htsu4rmzVaZADEVC
  660. XhIO4MThebusdk0AcNGjgpf3HRHk0DPMDDlIjm+Oao0cqovvF6VyYmcb0C+RmhJj
  661. dodLXMNmbqErwTk3zEkW0yZvNIYXH7m9SokPCZa4eeIM7be62X6h1mbt0/IU6Th+
  662. v18fS0iTMP/Viug5und+05C/v04kgDo0CPphAbXwWMnkE4B6Tl9sdyUYXtvQsL7x
  663. 0+WP1gL27ANqNZiI07Kz/BhbBAQI/+2TFT7oGr0AnFPQ5jHp+3GpUf6OKuT1wT3H
  664. ND189UFuRuubxb42vZhpcXRbqJVWnbECTKVUPsGZqat3enQUB63uM4i6/RdONDZA
  665. fDeF1m4qYs+cUXKNUZ03
  666. =X6RT
  667. -----END PGP SIGNATURE-----
  668. """
  669. post_sig = b"""\
  670. foo
  671. """
  672. c = Commit.from_string(pre_sig + git_sig + post_sig)
  673. self.assertEqual(pre_sig + post_sig, c.raw_without_sig())
  674. self.assertEqual(pre_sig + git_sig + post_sig, bytes(c))
  675. self.assertEqual(b"foo\n", c.message)
  676. self.assertEqual([], c._extra)
  677. self.assertEqual(
  678. b"""-----BEGIN PGP SIGNATURE-----
  679. Version: GnuPG v1
  680. iQIcBAABCgAGBQJULCdfAAoJEACAbyvXKaRXuKwP/RyP9PA49uAvu8tQVCC/uBa8
  681. vi975+xvO14R8Pp8k2nps7lSxCdtCd+xVT1VRHs0wNhOZo2YCVoU1HATkPejqSeV
  682. NScTHcxnk4/+bxyfk14xvJkNp7FlQ3npmBkA+lbV0Ubr33rvtIE5jiJPyz+SgWAg
  683. xdBG2TojV0squj00GoH/euK6aX7GgZtwdtpTv44haCQdSuPGDcI4TORqR6YSqvy3
  684. GPE+3ZqXPFFb+KILtimkxitdwB7CpwmNse2vE3rONSwTvi8nq3ZoQYNY73CQGkUy
  685. qoFU0pDtw87U3niFin1ZccDgH0bB6624sLViqrjcbYJeg815Htsu4rmzVaZADEVC
  686. XhIO4MThebusdk0AcNGjgpf3HRHk0DPMDDlIjm+Oao0cqovvF6VyYmcb0C+RmhJj
  687. dodLXMNmbqErwTk3zEkW0yZvNIYXH7m9SokPCZa4eeIM7be62X6h1mbt0/IU6Th+
  688. v18fS0iTMP/Viug5und+05C/v04kgDo0CPphAbXwWMnkE4B6Tl9sdyUYXtvQsL7x
  689. 0+WP1gL27ANqNZiI07Kz/BhbBAQI/+2TFT7oGr0AnFPQ5jHp+3GpUf6OKuT1wT3H
  690. ND189UFuRuubxb42vZhpcXRbqJVWnbECTKVUPsGZqat3enQUB63uM4i6/RdONDZA
  691. fDeF1m4qYs+cUXKNUZ03
  692. =X6RT
  693. -----END PGP SIGNATURE-----""",
  694. c.gpgsig,
  695. )
  696. def test_parse_header_trailing_newline(self) -> None:
  697. pre_sig = b"""\
  698. tree a7d6277f78d3ecd0230a1a5df6db00b1d9c521ac
  699. parent c09b6dec7a73760fbdb478383a3c926b18db8bbe
  700. author Neil Matatall <oreoshake@github.com> 1461964057 -1000
  701. committer Neil Matatall <oreoshake@github.com> 1461964057 -1000
  702. """
  703. git_sig = b"""\
  704. gpgsig -----BEGIN PGP SIGNATURE-----
  705. wsBcBAABCAAQBQJXI80ZCRA6pcNDcVZ70gAAarcIABs72xRX3FWeox349nh6ucJK
  706. CtwmBTusez2Zwmq895fQEbZK7jpaGO5TRO4OvjFxlRo0E08UFx3pxZHSpj6bsFeL
  707. hHsDXnCaotphLkbgKKRdGZo7tDqM84wuEDlh4MwNe7qlFC7bYLDyysc81ZX5lpMm
  708. 2MFF1TvjLAzSvkT7H1LPkuR3hSvfCYhikbPOUNnKOo0sYjeJeAJ/JdAVQ4mdJIM0
  709. gl3REp9+A+qBEpNQI7z94Pg5Bc5xenwuDh3SJgHvJV6zBWupWcdB3fAkVd4TPnEZ
  710. nHxksHfeNln9RKseIDcy4b2ATjhDNIJZARHNfr6oy4u3XPW4svRqtBsLoMiIeuI=
  711. =ms6q
  712. -----END PGP SIGNATURE-----
  713. """
  714. post_sig = b"""\
  715. 3.3.0 version bump and docs
  716. """
  717. gpgsig = b"""\
  718. -----BEGIN PGP SIGNATURE-----
  719. wsBcBAABCAAQBQJXI80ZCRA6pcNDcVZ70gAAarcIABs72xRX3FWeox349nh6ucJK
  720. CtwmBTusez2Zwmq895fQEbZK7jpaGO5TRO4OvjFxlRo0E08UFx3pxZHSpj6bsFeL
  721. hHsDXnCaotphLkbgKKRdGZo7tDqM84wuEDlh4MwNe7qlFC7bYLDyysc81ZX5lpMm
  722. 2MFF1TvjLAzSvkT7H1LPkuR3hSvfCYhikbPOUNnKOo0sYjeJeAJ/JdAVQ4mdJIM0
  723. gl3REp9+A+qBEpNQI7z94Pg5Bc5xenwuDh3SJgHvJV6zBWupWcdB3fAkVd4TPnEZ
  724. nHxksHfeNln9RKseIDcy4b2ATjhDNIJZARHNfr6oy4u3XPW4svRqtBsLoMiIeuI=
  725. =ms6q
  726. -----END PGP SIGNATURE-----\n"""
  727. c = Commit.from_string(pre_sig + git_sig + post_sig)
  728. self.assertEqual([], c._extra)
  729. self.assertEqual(pre_sig + git_sig + post_sig, c.as_raw_string())
  730. self.assertEqual(pre_sig + post_sig, c.raw_without_sig())
  731. self.assertEqual(gpgsig, c.gpgsig)
  732. self.assertEqual(b"3.3.0 version bump and docs\n", c.message)
  733. def test_commit_extract_signature_pgp(self) -> None:
  734. from dulwich.objects import SIGNATURE_PGP
  735. gpgsig = b"""-----BEGIN PGP SIGNATURE-----
  736. Version: GnuPG v1
  737. iQIcBAABCgAGBQJULCdfAAoJEACAbyvXKaRXuKwP/RyP9PA49uAvu8tQVCC/uBa8
  738. vi975+xvO14R8Pp8k2nps7lSxCdtCd+xVT1VRHs0wNhOZo2YCVoU1HATkPejqSeV
  739. NScTHcxnk4/+bxyfk14xvJkNp7FlQ3npmBkA+lbV0Ubr33rvtIE5jiJPyz+SgWAg
  740. -----END PGP SIGNATURE-----"""
  741. c = Commit()
  742. c.tree = b"d80c186a03f423a81b39df39dc87fd269736ca86"
  743. c.parents = [
  744. b"ab64bbdcc51b170d21588e5c5d391ee5c0c96dfd",
  745. b"4cffe90e0a41ad3f5190079d7c8f036bde29cbe6",
  746. ]
  747. c.author = c.committer = b"James Westby <jw+debian@jameswestby.net>"
  748. c.commit_time = c.author_time = 1174773719
  749. c.commit_timezone = c.author_timezone = 0
  750. c.message = b"Merge ../b\n"
  751. c.gpgsig = gpgsig
  752. payload, signature, sig_type = c.extract_signature()
  753. self.assertEqual(payload, c.raw_without_sig())
  754. self.assertEqual(signature, gpgsig)
  755. self.assertEqual(sig_type, SIGNATURE_PGP)
  756. def test_commit_extract_signature_ssh(self) -> None:
  757. from dulwich.objects import SIGNATURE_SSH
  758. ssh_sig = b"""-----BEGIN SSH SIGNATURE-----
  759. U1NIU0lHAAAAAQAAADMAAAALc3NoLWVkMjU1MTkAAAAgJwKO3yOmR5JlXCyN5bys
  760. ZTpDKBGsVP6ydcKdZxAvJlUAAAAEZmlsZQAAAAAAAAAGc2hhNTEyAAAAUwAAAAtz
  761. -----END SSH SIGNATURE-----"""
  762. c = Commit()
  763. c.tree = b"d80c186a03f423a81b39df39dc87fd269736ca86"
  764. c.parents = []
  765. c.author = c.committer = b"Test User <test@example.com>"
  766. c.commit_time = c.author_time = 1234567890
  767. c.commit_timezone = c.author_timezone = 0
  768. c.message = b"Test commit with SSH signature\n"
  769. c.gpgsig = ssh_sig
  770. payload, signature, sig_type = c.extract_signature()
  771. self.assertEqual(payload, c.raw_without_sig())
  772. self.assertEqual(signature, ssh_sig)
  773. self.assertEqual(sig_type, SIGNATURE_SSH)
  774. def test_commit_extract_signature_none(self) -> None:
  775. c = Commit()
  776. c.tree = b"d80c186a03f423a81b39df39dc87fd269736ca86"
  777. c.parents = []
  778. c.author = c.committer = b"Test User <test@example.com>"
  779. c.commit_time = c.author_time = 1234567890
  780. c.commit_timezone = c.author_timezone = 0
  781. c.message = b"Test commit without signature\n"
  782. payload, signature, sig_type = c.extract_signature()
  783. self.assertEqual(payload, c.as_raw_string())
  784. self.assertIsNone(signature)
  785. self.assertIsNone(sig_type)
  786. def test_commit_extract_signature_unknown(self) -> None:
  787. from dulwich.objects import ObjectFormatException
  788. unknown_sig = b"UNKNOWN SIGNATURE FORMAT DATA"
  789. c = Commit()
  790. c.tree = b"d80c186a03f423a81b39df39dc87fd269736ca86"
  791. c.parents = []
  792. c.author = c.committer = b"Test User <test@example.com>"
  793. c.commit_time = c.author_time = 1234567890
  794. c.commit_timezone = c.author_timezone = 0
  795. c.message = b"Test commit with unknown signature\n"
  796. c.gpgsig = unknown_sig
  797. # Unknown signature format should raise an exception
  798. with self.assertRaises(ObjectFormatException):
  799. c.extract_signature()
  800. def test_parse_time_entry_broken_negative_date(self) -> None:
  801. from dulwich.objects import parse_time_entry_broken
  802. author_line = b"Jane Doe <jdoe@example.org> -12345 +0100"
  803. expected_identity = b"Jane Doe <jdoe@example.org>"
  804. expected_time = -12345
  805. expected_timezone = +1 * 60 * 60
  806. person, time, (timezone, timezone_neg_utc) = parse_time_entry_broken(
  807. author_line
  808. )
  809. self.assertEqual(person, expected_identity)
  810. self.assertEqual(time, expected_time)
  811. self.assertEqual(timezone, expected_timezone)
  812. self.assertFalse(timezone_neg_utc)
  813. def test_parse_time_entry_broken_double_negative_timezone(self) -> None:
  814. from dulwich.objects import parse_time_entry_broken
  815. author_line = b"Jane Doe <jdoe@example.org> 12345 --700"
  816. expected_identity = b"Jane Doe <jdoe@example.org>"
  817. expected_time = 12345
  818. expected_timezone = +7 * 60 * 60
  819. person, time, (timezone, timezone_neg_utc) = parse_time_entry_broken(
  820. author_line
  821. )
  822. self.assertEqual(person, expected_identity)
  823. self.assertEqual(time, expected_time)
  824. self.assertEqual(timezone, expected_timezone)
  825. self.assertTrue(timezone_neg_utc)
  826. def test_parse_time_entry_broken_long_timezone(self) -> None:
  827. from dulwich.objects import parse_time_entry_broken
  828. author_line = (
  829. b"Geoff Cant <nem@lisp.geek.nz> 1170648114 -72000" # codespell:ignore
  830. )
  831. expected_identity = b"Geoff Cant <nem@lisp.geek.nz>" # codespell:ignore
  832. expected_time = 1170648114
  833. expected_timezone = -720 * 60 * 60
  834. person, time, (timezone, _timezone_neg_utc) = parse_time_entry_broken(
  835. author_line
  836. )
  837. self.assertEqual(person, expected_identity)
  838. self.assertEqual(time, expected_time)
  839. self.assertEqual(timezone, expected_timezone)
  840. def test_parse_time_entry_broken_short_timezone(self) -> None:
  841. from dulwich.objects import parse_time_entry_broken
  842. author_line = (
  843. b"Pl\xc3\xa1cidoMonteiro <Pl\xc3\xa1cidoMonteiro@.(none)> 1380083482 +02"
  844. )
  845. expected_identity = b"Pl\xc3\xa1cidoMonteiro <Pl\xc3\xa1cidoMonteiro@.(none)>"
  846. expected_time = 1380083482
  847. expected_timezone = +2 * 60
  848. person, time, (timezone, _timezone_neg_utc) = parse_time_entry_broken(
  849. author_line
  850. )
  851. self.assertEqual(person, expected_identity)
  852. self.assertEqual(time, expected_time)
  853. self.assertEqual(timezone, expected_timezone)
  854. def test_parse_time_entry_broken_unsigned_timezone(self) -> None:
  855. from dulwich.objects import parse_time_entry_broken
  856. author_line = (
  857. b"applehq <applehq@203d044e-caa7-11dc-91ec-67e1038599e7> 1205785941 0000"
  858. )
  859. expected_identity = b"applehq <applehq@203d044e-caa7-11dc-91ec-67e1038599e7>"
  860. expected_time = 1205785941
  861. expected_timezone = 0
  862. person, time, (timezone, _timezone_neg_utc) = parse_time_entry_broken(
  863. author_line
  864. )
  865. self.assertEqual(person, expected_identity)
  866. self.assertEqual(time, expected_time)
  867. self.assertEqual(timezone, expected_timezone)
  868. def test_parse_time_entry_broken_nonsensical_timezone(self) -> None:
  869. """Timezone is 'UTC + 5 hours and 75 minutes'."""
  870. from dulwich.objects import parse_time_entry_broken
  871. author_line = b"acpmasquerade <d@picovico.com> 1460127297 +0575"
  872. expected_identity = b"acpmasquerade <d@picovico.com>"
  873. expected_time = 1460127297
  874. expected_timezone = +6 * 60 * 60 + 15 * 60
  875. person, time, (timezone, _timezone_neg_utc) = parse_time_entry_broken(
  876. author_line
  877. )
  878. self.assertEqual(person, expected_identity)
  879. self.assertEqual(time, expected_time)
  880. self.assertEqual(timezone, expected_timezone)
  881. def test_parse_time_entry_broken_missing_brackets(self) -> None:
  882. from dulwich.objects import parse_time_entry_broken
  883. author_line = b"kapil.foss@gmail.com 1297013737 -0500"
  884. expected_identity = b"kapil.foss@gmail.com"
  885. expected_time = 1297013737
  886. expected_timezone = -5 * 60 * 60
  887. person, time, (timezone, _timezone_neg_utc) = parse_time_entry_broken(
  888. author_line
  889. )
  890. self.assertEqual(person, expected_identity)
  891. self.assertEqual(time, expected_time)
  892. self.assertEqual(timezone, expected_timezone)
  893. class BrokenCommitParseTests(TestCase):
  894. """Tests for parsing commits with broken author/committer lines using parse_commit_broken."""
  895. def make_commit_text(
  896. self,
  897. tree=b"d80c186a03f423a81b39df39dc87fd269736ca86",
  898. parents=None,
  899. author=b"Test User <test@example.com> 1234567890 +0000",
  900. committer=b"Test User <test@example.com> 1234567890 +0000",
  901. encoding=None,
  902. message=b"Test commit\n",
  903. extra=None,
  904. ):
  905. lines = []
  906. if tree is not None:
  907. lines.append(b"tree " + tree)
  908. if parents is not None:
  909. lines.extend(b"parent " + p for p in parents)
  910. if author is not None:
  911. lines.append(b"author " + author)
  912. if committer is not None:
  913. lines.append(b"committer " + committer)
  914. if encoding is not None:
  915. lines.append(b"encoding " + encoding)
  916. if extra is not None:
  917. for name, value in sorted(extra.items()):
  918. lines.append(name + b" " + value)
  919. lines.append(b"")
  920. if message is not None:
  921. lines.append(message)
  922. return b"\n".join(lines)
  923. def test_negative_timestamp(self) -> None:
  924. from dulwich.objects import parse_commit_broken
  925. author_line = b"Jane Doe <jdoe@example.org> -12345 +0100"
  926. commit_text = self.make_commit_text(author=author_line, committer=author_line)
  927. commit = parse_commit_broken(commit_text)
  928. self.assertEqual(commit.author, b"Jane Doe <jdoe@example.org>")
  929. self.assertEqual(commit.author_time, -12345)
  930. self.assertEqual(commit.author_timezone, +1 * 60 * 60)
  931. def test_double_negative_timezone(self) -> None:
  932. from dulwich.objects import parse_commit_broken
  933. author_line = b"Jane Doe <jdoe@example.org> 12345 --700"
  934. commit_text = self.make_commit_text(author=author_line, committer=author_line)
  935. commit = parse_commit_broken(commit_text)
  936. self.assertEqual(commit.author, b"Jane Doe <jdoe@example.org>")
  937. self.assertEqual(commit.author_time, 12345)
  938. self.assertEqual(commit.author_timezone, +7 * 60 * 60)
  939. self.assertTrue(commit._author_timezone_neg_utc)
  940. def test_long_timezone(self) -> None:
  941. from dulwich.objects import parse_commit_broken
  942. # Real example from https://github.com/lisp/geek-nz
  943. author_line = (
  944. b"Geoff Cant <nem@lisp.geek.nz> 1170648114 -72000" # codespell:ignore
  945. )
  946. commit_text = self.make_commit_text(author=author_line, committer=author_line)
  947. commit = parse_commit_broken(commit_text)
  948. self.assertEqual(
  949. commit.author,
  950. b"Geoff Cant <nem@lisp.geek.nz>", # codespell:ignore
  951. )
  952. self.assertEqual(commit.author_time, 1170648114)
  953. self.assertEqual(commit.author_timezone, -720 * 60 * 60)
  954. def test_short_timezone(self) -> None:
  955. from dulwich.objects import parse_commit_broken
  956. author_line = (
  957. b"Pl\xc3\xa1cidoMonteiro <Pl\xc3\xa1cidoMonteiro@.(none)> 1380083482 +02"
  958. )
  959. commit_text = self.make_commit_text(author=author_line, committer=author_line)
  960. commit = parse_commit_broken(commit_text)
  961. self.assertEqual(
  962. commit.author, b"Pl\xc3\xa1cidoMonteiro <Pl\xc3\xa1cidoMonteiro@.(none)>"
  963. )
  964. self.assertEqual(commit.author_time, 1380083482)
  965. self.assertEqual(commit.author_timezone, +2 * 60)
  966. def test_unsigned_timezone(self) -> None:
  967. from dulwich.objects import parse_commit_broken
  968. author_line = (
  969. b"applehq <applehq@203d044e-caa7-11dc-91ec-67e1038599e7> 1205785941 0000"
  970. )
  971. commit_text = self.make_commit_text(author=author_line, committer=author_line)
  972. commit = parse_commit_broken(commit_text)
  973. self.assertEqual(
  974. commit.author, b"applehq <applehq@203d044e-caa7-11dc-91ec-67e1038599e7>"
  975. )
  976. self.assertEqual(commit.author_time, 1205785941)
  977. self.assertEqual(commit.author_timezone, 0)
  978. def test_nonsensical_timezone(self) -> None:
  979. from dulwich.objects import parse_commit_broken
  980. # Timezone is 'UTC + 5 hours and 75 minutes'
  981. author_line = b"acpmasquerade <d@picovico.com> 1460127297 +0575"
  982. commit_text = self.make_commit_text(author=author_line, committer=author_line)
  983. commit = parse_commit_broken(commit_text)
  984. self.assertEqual(commit.author, b"acpmasquerade <d@picovico.com>")
  985. self.assertEqual(commit.author_time, 1460127297)
  986. self.assertEqual(commit.author_timezone, +6 * 60 * 60 + 15 * 60)
  987. def test_missing_angle_brackets(self) -> None:
  988. from dulwich.objects import parse_commit_broken
  989. # Real example from https://github.com/noderabbit-team/tasks
  990. author_line = b"kapil.foss@gmail.com 1297013737 -0500"
  991. commit_text = self.make_commit_text(author=author_line, committer=author_line)
  992. commit = parse_commit_broken(commit_text)
  993. self.assertEqual(commit.author, b"kapil.foss@gmail.com")
  994. self.assertEqual(commit.author_time, 1297013737)
  995. self.assertEqual(commit.author_timezone, -5 * 60 * 60)
  996. _TREE_ITEMS = {
  997. b"a-c": (0o100755, b"d80c186a03f423a81b39df39dc87fd269736ca86"),
  998. b"a.c": (0o100755, b"d80c186a03f423a81b39df39dc87fd269736ca86"),
  999. b"aoc": (0o100755, b"d80c186a03f423a81b39df39dc87fd269736ca86"),
  1000. b"a": (stat.S_IFDIR, b"d80c186a03f423a81b39df39dc87fd269736ca86"),
  1001. b"a/c": (stat.S_IFDIR, b"d80c186a03f423a81b39df39dc87fd269736ca86"),
  1002. }
  1003. _SORTED_TREE_ITEMS = [
  1004. TreeEntry(b"a-c", 0o100755, b"d80c186a03f423a81b39df39dc87fd269736ca86"),
  1005. TreeEntry(b"a.c", 0o100755, b"d80c186a03f423a81b39df39dc87fd269736ca86"),
  1006. TreeEntry(b"a", stat.S_IFDIR, b"d80c186a03f423a81b39df39dc87fd269736ca86"),
  1007. TreeEntry(b"a/c", stat.S_IFDIR, b"d80c186a03f423a81b39df39dc87fd269736ca86"),
  1008. TreeEntry(b"aoc", 0o100755, b"d80c186a03f423a81b39df39dc87fd269736ca86"),
  1009. ]
  1010. _TREE_ITEMS_BUG_1325 = {
  1011. b"dir": (stat.S_IFDIR | 0o644, b"5944b31ff85b415573d1a43eb942e2dea30ab8be"),
  1012. b"dira": (0o100644, b"cf7a729ca69bfabd0995fc9b083e86a18215bd91"),
  1013. }
  1014. _SORTED_TREE_ITEMS_BUG_1325 = [
  1015. TreeEntry(
  1016. path=b"dir",
  1017. mode=stat.S_IFDIR | 0o644,
  1018. sha=b"5944b31ff85b415573d1a43eb942e2dea30ab8be",
  1019. ),
  1020. TreeEntry(
  1021. path=b"dira", mode=0o100644, sha=b"cf7a729ca69bfabd0995fc9b083e86a18215bd91"
  1022. ),
  1023. ]
  1024. class TreeTests(ShaFileCheckTests):
  1025. def test_add(self) -> None:
  1026. myhexsha = b"d80c186a03f423a81b39df39dc87fd269736ca86"
  1027. x = Tree()
  1028. x.add(b"myname", 0o100755, myhexsha)
  1029. self.assertEqual(x[b"myname"], (0o100755, myhexsha))
  1030. self.assertEqual(b"100755 myname\0" + hex_to_sha(myhexsha), x.as_raw_string())
  1031. def test_simple(self) -> None:
  1032. myhexsha = b"d80c186a03f423a81b39df39dc87fd269736ca86"
  1033. x = Tree()
  1034. x[b"myname"] = (0o100755, myhexsha)
  1035. self.assertEqual(b"100755 myname\0" + hex_to_sha(myhexsha), x.as_raw_string())
  1036. self.assertEqual(b"100755 myname\0" + hex_to_sha(myhexsha), bytes(x))
  1037. def test_tree_update_id(self) -> None:
  1038. x = Tree()
  1039. x[b"a.c"] = (0o100755, b"d80c186a03f423a81b39df39dc87fd269736ca86")
  1040. self.assertEqual(b"0c5c6bc2c081accfbc250331b19e43b904ab9cdd", x.id)
  1041. x[b"a.b"] = (stat.S_IFDIR, b"d80c186a03f423a81b39df39dc87fd269736ca86")
  1042. self.assertEqual(b"07bfcb5f3ada15bbebdfa3bbb8fd858a363925c8", x.id)
  1043. def test_tree_iteritems_dir_sort(self) -> None:
  1044. x = Tree()
  1045. for name, item in _TREE_ITEMS.items():
  1046. x[name] = item
  1047. self.assertEqual(_SORTED_TREE_ITEMS, x.items())
  1048. def test_tree_items_dir_sort(self) -> None:
  1049. x = Tree()
  1050. for name, item in _TREE_ITEMS.items():
  1051. x[name] = item
  1052. self.assertEqual(_SORTED_TREE_ITEMS, x.items())
  1053. def _do_test_parse_tree(self, parse_tree) -> None:
  1054. dir = os.path.join(os.path.dirname(__file__), "..", "testdata", "trees")
  1055. o = Tree.from_path(hex_to_filename(dir, tree_sha))
  1056. self.assertEqual(
  1057. [(b"a", 0o100644, a_sha), (b"b", 0o100644, b_sha)],
  1058. list(parse_tree(o.as_raw_string(), 20)),
  1059. )
  1060. # test a broken tree that has a leading 0 on the file mode
  1061. broken_tree = b"0100644 foo\0" + hex_to_sha(a_sha)
  1062. def eval_parse_tree(*args, **kwargs):
  1063. return list(parse_tree(*args, **kwargs))
  1064. self.assertEqual([(b"foo", 0o100644, a_sha)], eval_parse_tree(broken_tree, 20))
  1065. self.assertRaises(
  1066. ObjectFormatException, eval_parse_tree, broken_tree, 20, strict=True
  1067. )
  1068. test_parse_tree = functest_builder(_do_test_parse_tree, _parse_tree_py)
  1069. test_parse_tree_extension = ext_functest_builder(
  1070. _do_test_parse_tree, _parse_tree_rs
  1071. )
  1072. def _do_test_sorted_tree_items(self, sorted_tree_items) -> None:
  1073. def do_sort(entries, name_order):
  1074. return list(sorted_tree_items(entries, name_order))
  1075. actual = do_sort(_TREE_ITEMS, False)
  1076. self.assertEqual(_SORTED_TREE_ITEMS, actual)
  1077. self.assertIsInstance(actual[0], TreeEntry)
  1078. actual = do_sort(_TREE_ITEMS_BUG_1325, False)
  1079. self.assertEqual(
  1080. key_entry((b"a", (0o40644, b"cf7a729ca69bfabd0995fc9b083e86a18215bd91"))),
  1081. b"a/",
  1082. )
  1083. self.assertEqual(_SORTED_TREE_ITEMS_BUG_1325, actual)
  1084. self.assertIsInstance(actual[0], TreeEntry)
  1085. # C/Python implementations may differ in specific error types, but
  1086. # should all error on invalid inputs.
  1087. # For example, the Rust implementation has stricter type checks, so may
  1088. # raise TypeError where the Python implementation raises
  1089. # AttributeError.
  1090. errors = (TypeError, ValueError, AttributeError)
  1091. self.assertRaises(errors, do_sort, b"foo", False)
  1092. self.assertRaises(errors, do_sort, {b"foo": (1, 2, 3)}, False)
  1093. myhexsha = b"d80c186a03f423a81b39df39dc87fd269736ca86"
  1094. self.assertRaises(errors, do_sort, {b"foo": (b"xxx", myhexsha)}, False)
  1095. self.assertRaises(errors, do_sort, {b"foo": (0o100755, 12345)}, False)
  1096. test_sorted_tree_items = functest_builder(
  1097. _do_test_sorted_tree_items, _sorted_tree_items_py
  1098. )
  1099. if _sorted_tree_items_rs is not None:
  1100. assert _sorted_tree_items_rs != _sorted_tree_items_py
  1101. test_sorted_tree_items_extension = ext_functest_builder(
  1102. _do_test_sorted_tree_items, _sorted_tree_items_rs
  1103. )
  1104. def _do_test_sorted_tree_items_name_order(self, sorted_tree_items) -> None:
  1105. self.assertEqual(
  1106. [
  1107. TreeEntry(
  1108. b"a",
  1109. stat.S_IFDIR,
  1110. b"d80c186a03f423a81b39df39dc87fd269736ca86",
  1111. ),
  1112. TreeEntry(
  1113. b"a-c",
  1114. 0o100755,
  1115. b"d80c186a03f423a81b39df39dc87fd269736ca86",
  1116. ),
  1117. TreeEntry(
  1118. b"a.c",
  1119. 0o100755,
  1120. b"d80c186a03f423a81b39df39dc87fd269736ca86",
  1121. ),
  1122. TreeEntry(
  1123. b"a/c",
  1124. stat.S_IFDIR,
  1125. b"d80c186a03f423a81b39df39dc87fd269736ca86",
  1126. ),
  1127. TreeEntry(
  1128. b"aoc",
  1129. 0o100755,
  1130. b"d80c186a03f423a81b39df39dc87fd269736ca86",
  1131. ),
  1132. ],
  1133. list(sorted_tree_items(_TREE_ITEMS, True)),
  1134. )
  1135. test_sorted_tree_items_name_order = functest_builder(
  1136. _do_test_sorted_tree_items_name_order, _sorted_tree_items_py
  1137. )
  1138. if _sorted_tree_items_rs is not None:
  1139. test_sorted_tree_items_name_order_extension = ext_functest_builder(
  1140. _do_test_sorted_tree_items_name_order, _sorted_tree_items_rs
  1141. )
  1142. def _do_test_sorted_tree_items_issue_1325(self, sorted_tree_items) -> None:
  1143. """Test case to reproduce issue #1325: submodules incorrectly sorted as directories.
  1144. The bug: Rust uses (mode & 0o40000 != 0) which incorrectly matches
  1145. submodules (0o160000) since 0o160000 & 0o40000 = 0o40000
  1146. """
  1147. # Test case 1: Minimal test - submodule vs file
  1148. entries = {
  1149. b"sub": (
  1150. 0o160000,
  1151. b"a03f423a81b39df39dc87fd269736ca86d80c186",
  1152. ), # submodule
  1153. b"sub.txt": (0o100644, b"81b39df39dc87fd269736ca86d80c186a03f423a"), # file
  1154. }
  1155. result = list(sorted_tree_items(entries, False))
  1156. paths = [entry.path for entry in result]
  1157. # Submodules should sort as regular files, not directories
  1158. # Expected order: sub, sub.txt
  1159. # Bug causes: sub.txt, sub (because sub is treated as sub/)
  1160. self.assertEqual([b"sub", b"sub.txt"], paths)
  1161. # Test case 2: Scenario from issue - file rename + submodule
  1162. # This simulates the "gamma" scenario mentioned in the issue
  1163. entries2 = {
  1164. b"alpha": (0o100644, b"a03f423a81b39df39dc87fd269736ca86d80c186"),
  1165. b"beta": (0o100644, b"81b39df39dc87fd269736ca86d80c186a03f423a"),
  1166. b"gamma": (
  1167. 0o160000,
  1168. b"d80c186a03f423a81b39df39dc87fd269736ca86",
  1169. ), # submodule (was file)
  1170. b"delta": (0o100644, b"cf7a729ca69bfabd0995fc9b083e86a18215bd91"),
  1171. }
  1172. result2 = list(sorted_tree_items(entries2, False))
  1173. paths2 = [entry.path for entry in result2]
  1174. # All entries should sort in alphabetical order since none are directories
  1175. self.assertEqual([b"alpha", b"beta", b"delta", b"gamma"], paths2)
  1176. test_sorted_tree_items_issue_1325 = functest_builder(
  1177. _do_test_sorted_tree_items_issue_1325, _sorted_tree_items_py
  1178. )
  1179. if _sorted_tree_items_rs is not None:
  1180. test_sorted_tree_items_issue_1325_extension = ext_functest_builder(
  1181. _do_test_sorted_tree_items_issue_1325, _sorted_tree_items_rs
  1182. )
  1183. def test_sorted_tree_items_issue_1325_comparison(self) -> None:
  1184. """Direct comparison test to show the difference between Python and Rust implementations."""
  1185. if _sorted_tree_items_rs is None:
  1186. self.skipTest("Rust extension not available")
  1187. # Minimal test case: submodule vs file
  1188. entries = {
  1189. b"sub": (
  1190. 0o160000,
  1191. b"a03f423a81b39df39dc87fd269736ca86d80c186",
  1192. ), # submodule
  1193. b"sub.txt": (0o100644, b"81b39df39dc87fd269736ca86d80c186a03f423a"), # file
  1194. }
  1195. # Get results from both implementations
  1196. py_result = list(_sorted_tree_items_py(entries, False))
  1197. rs_result = list(_sorted_tree_items_rs(entries, False))
  1198. # Show the actual ordering from each
  1199. py_paths = [entry.path for entry in py_result]
  1200. rs_paths = [entry.path for entry in rs_result]
  1201. # This test shows the bug: Rust treats submodules as directories
  1202. self.assertEqual(
  1203. py_paths, rs_paths, "Bug: Rust treats submodules (0o160000) as directories"
  1204. )
  1205. def test_check(self) -> None:
  1206. t = Tree
  1207. sha = hex_to_sha(a_sha)
  1208. # filenames
  1209. self.assertCheckSucceeds(t, b"100644 .a\0" + sha)
  1210. self.assertCheckFails(t, b"100644 \0" + sha)
  1211. self.assertCheckFails(t, b"100644 .\0" + sha)
  1212. self.assertCheckFails(t, b"100644 a/a\0" + sha)
  1213. self.assertCheckFails(t, b"100644 ..\0" + sha)
  1214. self.assertCheckFails(t, b"100644 .git\0" + sha)
  1215. # modes
  1216. self.assertCheckSucceeds(t, b"100644 a\0" + sha)
  1217. self.assertCheckSucceeds(t, b"100755 a\0" + sha)
  1218. self.assertCheckSucceeds(t, b"160000 a\0" + sha)
  1219. # TODO more whitelisted modes
  1220. self.assertCheckFails(t, b"123456 a\0" + sha)
  1221. self.assertCheckFails(t, b"123abc a\0" + sha)
  1222. # should fail check, but parses ok
  1223. self.assertCheckFails(t, b"0100644 foo\0" + sha)
  1224. # shas
  1225. self.assertCheckFails(t, b"100644 a\0" + (b"x" * 5))
  1226. self.assertCheckFails(t, b"100644 a\0" + (b"x" * 18) + b"\0")
  1227. self.assertCheckFails(t, b"100644 a\0" + (b"x" * 21) + b"\n100644 b\0" + sha)
  1228. # ordering
  1229. sha2 = hex_to_sha(b_sha)
  1230. self.assertCheckSucceeds(t, b"100644 a\0" + sha + b"100644 b\0" + sha)
  1231. self.assertCheckSucceeds(t, b"100644 a\0" + sha + b"100644 b\0" + sha2)
  1232. self.assertCheckFails(t, b"100644 a\0" + sha + b"100755 a\0" + sha2)
  1233. self.assertCheckFails(t, b"100644 b\0" + sha2 + b"100644 a\0" + sha)
  1234. def test_iter(self) -> None:
  1235. t = Tree()
  1236. t[b"foo"] = (0o100644, a_sha)
  1237. self.assertEqual({b"foo"}, set(t))
  1238. class TagSerializeTests(TestCase):
  1239. def test_serialize_simple(self) -> None:
  1240. x = make_object(
  1241. Tag,
  1242. tagger=b"Jelmer Vernooij <jelmer@samba.org>",
  1243. name=b"0.1",
  1244. message=b"Tag 0.1",
  1245. object=(Blob, b"d80c186a03f423a81b39df39dc87fd269736ca86"),
  1246. tag_time=423423423,
  1247. tag_timezone=0,
  1248. )
  1249. self.assertEqual(
  1250. (
  1251. b"object d80c186a03f423a81b39df39dc87fd269736ca86\n"
  1252. b"type blob\n"
  1253. b"tag 0.1\n"
  1254. b"tagger Jelmer Vernooij <jelmer@samba.org> "
  1255. b"423423423 +0000\n"
  1256. b"\n"
  1257. b"Tag 0.1"
  1258. ),
  1259. x.as_raw_string(),
  1260. )
  1261. def test_serialize_none_message(self) -> None:
  1262. x = make_object(
  1263. Tag,
  1264. tagger=b"Jelmer Vernooij <jelmer@samba.org>",
  1265. name=b"0.1",
  1266. message=None,
  1267. object=(Blob, b"d80c186a03f423a81b39df39dc87fd269736ca86"),
  1268. tag_time=423423423,
  1269. tag_timezone=0,
  1270. )
  1271. self.assertEqual(
  1272. (
  1273. b"object d80c186a03f423a81b39df39dc87fd269736ca86\n"
  1274. b"type blob\n"
  1275. b"tag 0.1\n"
  1276. b"tagger Jelmer Vernooij <jelmer@samba.org> "
  1277. b"423423423 +0000\n\n"
  1278. ),
  1279. x.as_raw_string(),
  1280. )
  1281. default_tagger = (
  1282. b"Linus Torvalds <torvalds@woody.linux-foundation.org> 1183319674 -0700"
  1283. )
  1284. default_message = b"""Linux 2.6.22-rc7
  1285. -----BEGIN PGP SIGNATURE-----
  1286. Version: GnuPG v1.4.7 (GNU/Linux)
  1287. iD8DBQBGiAaAF3YsRnbiHLsRAitMAKCiLboJkQECM/jpYsY3WPfvUgLXkACgg3ql
  1288. OK2XeQOiEeXtT76rV4t2WR4=
  1289. =ivrA
  1290. -----END PGP SIGNATURE-----
  1291. """
  1292. class TagParseTests(ShaFileCheckTests):
  1293. def make_tag_lines(
  1294. self,
  1295. object_sha=b"a38d6181ff27824c79fc7df825164a212eff6a3f",
  1296. object_type_name=b"commit",
  1297. name=b"v2.6.22-rc7",
  1298. tagger=default_tagger,
  1299. message=default_message,
  1300. ):
  1301. lines = []
  1302. if object_sha is not None:
  1303. lines.append(b"object " + object_sha)
  1304. if object_type_name is not None:
  1305. lines.append(b"type " + object_type_name)
  1306. if name is not None:
  1307. lines.append(b"tag " + name)
  1308. if tagger is not None:
  1309. lines.append(b"tagger " + tagger)
  1310. if message is not None:
  1311. lines.append(b"")
  1312. lines.append(message)
  1313. return lines
  1314. def make_tag_text(self, **kwargs):
  1315. return b"\n".join(self.make_tag_lines(**kwargs))
  1316. def test_parse(self) -> None:
  1317. x = Tag()
  1318. x.set_raw_string(self.make_tag_text())
  1319. self.assertEqual(
  1320. b"Linus Torvalds <torvalds@woody.linux-foundation.org>", x.tagger
  1321. )
  1322. self.assertEqual(b"v2.6.22-rc7", x.name)
  1323. object_type, object_sha = x.object
  1324. self.assertEqual(b"a38d6181ff27824c79fc7df825164a212eff6a3f", object_sha)
  1325. self.assertEqual(Commit, object_type)
  1326. self.assertEqual(
  1327. datetime.datetime.fromtimestamp(x.tag_time, datetime.timezone.utc).replace(
  1328. tzinfo=None
  1329. ),
  1330. datetime.datetime(2007, 7, 1, 19, 54, 34),
  1331. )
  1332. self.assertEqual(-25200, x.tag_timezone)
  1333. def test_parse_no_tagger(self) -> None:
  1334. x = Tag()
  1335. x.set_raw_string(self.make_tag_text(tagger=None))
  1336. self.assertEqual(None, x.tagger)
  1337. self.assertEqual(b"v2.6.22-rc7", x.name)
  1338. self.assertEqual(None, x.tag_time)
  1339. def test_parse_no_message(self) -> None:
  1340. x = Tag()
  1341. x.set_raw_string(self.make_tag_text(message=None))
  1342. self.assertEqual(None, x.message)
  1343. self.assertEqual(
  1344. b"Linus Torvalds <torvalds@woody.linux-foundation.org>", x.tagger
  1345. )
  1346. self.assertEqual(
  1347. datetime.datetime.fromtimestamp(x.tag_time, datetime.timezone.utc).replace(
  1348. tzinfo=None
  1349. ),
  1350. datetime.datetime(2007, 7, 1, 19, 54, 34),
  1351. )
  1352. self.assertEqual(-25200, x.tag_timezone)
  1353. self.assertEqual(b"v2.6.22-rc7", x.name)
  1354. def test_check(self) -> None:
  1355. self.assertCheckSucceeds(Tag, self.make_tag_text())
  1356. self.assertCheckFails(Tag, self.make_tag_text(object_sha=None))
  1357. self.assertCheckFails(Tag, self.make_tag_text(object_type_name=None))
  1358. self.assertCheckFails(Tag, self.make_tag_text(name=None))
  1359. self.assertCheckFails(Tag, self.make_tag_text(name=b""))
  1360. self.assertCheckFails(Tag, self.make_tag_text(object_type_name=b"foobar"))
  1361. self.assertCheckFails(
  1362. Tag,
  1363. self.make_tag_text(
  1364. tagger=b"some guy without an email address 1183319674 -0700"
  1365. ),
  1366. )
  1367. self.assertCheckFails(
  1368. Tag,
  1369. self.make_tag_text(
  1370. tagger=(
  1371. b"Linus Torvalds <torvalds@woody.linux-foundation.org> "
  1372. b"Sun 7 Jul 2007 12:54:34 +0700"
  1373. )
  1374. ),
  1375. )
  1376. self.assertCheckFails(Tag, self.make_tag_text(object_sha=b"xxx"))
  1377. def test_check_tag_with_unparseable_field(self) -> None:
  1378. self.assertCheckFails(
  1379. Tag,
  1380. self.make_tag_text(
  1381. tagger=(
  1382. b"Linus Torvalds <torvalds@woody.linux-foundation.org> 423423+0000"
  1383. )
  1384. ),
  1385. )
  1386. def test_check_tag_with_overflow_time(self) -> None:
  1387. """Date with overflow should raise an ObjectFormatException when checked."""
  1388. author = f"Some Dude <some@dude.org> {MAX_TIME + 1} +0000"
  1389. tag = Tag.from_string(self.make_tag_text(tagger=(author.encode())))
  1390. with self.assertRaises(ObjectFormatException):
  1391. tag.check()
  1392. def test_check_duplicates(self) -> None:
  1393. # duplicate each of the header fields
  1394. for i in range(4):
  1395. lines = self.make_tag_lines()
  1396. lines.insert(i, lines[i])
  1397. self.assertCheckFails(Tag, b"\n".join(lines))
  1398. def test_check_order(self) -> None:
  1399. lines = self.make_tag_lines()
  1400. headers = lines[:4]
  1401. rest = lines[4:]
  1402. # of all possible permutations, ensure only the original succeeds
  1403. for perm in permutations(headers):
  1404. perm = list(perm)
  1405. text = b"\n".join(perm + rest)
  1406. if perm == headers:
  1407. self.assertCheckSucceeds(Tag, text)
  1408. else:
  1409. self.assertCheckFails(Tag, text)
  1410. def test_tree_copy_after_update(self) -> None:
  1411. """Check Tree.id is correctly updated when the tree is copied after updated."""
  1412. shas = []
  1413. tree = Tree()
  1414. shas.append(tree.id)
  1415. tree.add(b"data", 0o644, Blob().id)
  1416. copied = tree.copy()
  1417. shas.append(tree.id)
  1418. shas.append(copied.id)
  1419. self.assertNotIn(shas[0], shas[1:])
  1420. self.assertEqual(shas[1], shas[2])
  1421. def test_tag_withough_sig(self) -> None:
  1422. x = Tag()
  1423. x.set_raw_string(self.make_tag_text())
  1424. self.assertEqual(bytes(x), x.raw_without_sig() + x.signature)
  1425. self.assertEqual(
  1426. b"""\
  1427. -----BEGIN PGP SIGNATURE-----
  1428. Version: GnuPG v1.4.7 (GNU/Linux)
  1429. iD8DBQBGiAaAF3YsRnbiHLsRAitMAKCiLboJkQECM/jpYsY3WPfvUgLXkACgg3ql
  1430. OK2XeQOiEeXtT76rV4t2WR4=
  1431. =ivrA
  1432. -----END PGP SIGNATURE-----
  1433. """,
  1434. x.signature,
  1435. )
  1436. def test_tag_extract_signature_pgp(self) -> None:
  1437. from dulwich.objects import SIGNATURE_PGP
  1438. x = Tag()
  1439. x.set_raw_string(self.make_tag_text())
  1440. payload, signature, sig_type = x.extract_signature()
  1441. self.assertEqual(payload, x.raw_without_sig())
  1442. self.assertEqual(signature, x.signature)
  1443. self.assertEqual(sig_type, SIGNATURE_PGP)
  1444. def test_tag_extract_signature_ssh(self) -> None:
  1445. from dulwich.objects import SIGNATURE_SSH
  1446. tag_text_lines = self.make_tag_lines()
  1447. # Replace PGP signature with SSH signature
  1448. tag_text_lines[-1] = b"""\
  1449. -----BEGIN SSH SIGNATURE-----
  1450. U1NIU0lHAAAAAQAAADMAAAALc3NoLWVkMjU1MTkAAAAgJwKO3yOmR5JlXCyN5bys
  1451. ZTpDKBGsVP6ydcKdZxAvJlUAAAAEZmlsZQAAAAAAAAAGc2hhNTEyAAAAUwAAAAtz
  1452. -----END SSH SIGNATURE-----
  1453. """
  1454. tag_text = b"\n".join(tag_text_lines)
  1455. x = Tag()
  1456. x.set_raw_string(tag_text)
  1457. payload, signature, sig_type = x.extract_signature()
  1458. self.assertEqual(payload, x.raw_without_sig())
  1459. self.assertEqual(signature, x.signature)
  1460. self.assertEqual(sig_type, SIGNATURE_SSH)
  1461. def test_tag_extract_signature_none(self) -> None:
  1462. tag_lines = self.make_tag_lines(message=b"Test tag\n")
  1463. x = Tag()
  1464. x.set_raw_string(b"\n".join(tag_lines))
  1465. payload, signature, sig_type = x.extract_signature()
  1466. self.assertEqual(payload, bytes(x))
  1467. self.assertIsNone(signature)
  1468. self.assertIsNone(sig_type)
  1469. def test_tag_extract_signature_unknown(self) -> None:
  1470. from dulwich.objects import ObjectFormatException
  1471. # Create a tag with a signature that has an unknown format
  1472. # It needs to look like a signature to be detected but not be PGP or SSH
  1473. tag_text = b"""object a38d6181ff27824c79fc7df825164a212eff6a3f
  1474. type commit
  1475. tag v2.6.22-rc7
  1476. tagger Linus Torvalds <torvalds@woody.linux-foundation.org> 1183319674 +0000
  1477. Linux 2.6.22-rc7
  1478. -----BEGIN UNKNOWN SIGNATURE-----
  1479. Some unknown signature format
  1480. -----END UNKNOWN SIGNATURE-----
  1481. """
  1482. x = Tag()
  1483. # First we need to manually set the signature to test the extract_signature method
  1484. x.set_raw_string(tag_text[: tag_text.index(b"-----BEGIN")])
  1485. x._signature = b"-----BEGIN UNKNOWN SIGNATURE-----\nSome unknown signature format\n-----END UNKNOWN SIGNATURE-----\n"
  1486. x._needs_serialization = False
  1487. # Unknown signature format should raise an exception
  1488. with self.assertRaises(ObjectFormatException):
  1489. x.extract_signature()
  1490. class CheckTests(TestCase):
  1491. def test_check_hexsha(self) -> None:
  1492. check_hexsha(a_sha, "failed to check good sha")
  1493. self.assertRaises(
  1494. ObjectFormatException, check_hexsha, b"1" * 39, "sha too short"
  1495. )
  1496. self.assertRaises(
  1497. ObjectFormatException, check_hexsha, b"1" * 41, "sha too long"
  1498. )
  1499. self.assertRaises(
  1500. ObjectFormatException,
  1501. check_hexsha,
  1502. b"x" * 40,
  1503. "invalid characters",
  1504. )
  1505. def test_check_identity(self) -> None:
  1506. check_identity(
  1507. b"Dave Borowitz <dborowitz@google.com>",
  1508. "failed to check good identity",
  1509. )
  1510. check_identity(b" <dborowitz@google.com>", "failed to check good identity")
  1511. self.assertRaises(
  1512. ObjectFormatException,
  1513. check_identity,
  1514. b"<dborowitz@google.com>",
  1515. "no space before email",
  1516. )
  1517. self.assertRaises(
  1518. ObjectFormatException, check_identity, b"Dave Borowitz", "no email"
  1519. )
  1520. self.assertRaises(
  1521. ObjectFormatException,
  1522. check_identity,
  1523. b"Dave Borowitz <dborowitz",
  1524. "incomplete email",
  1525. )
  1526. self.assertRaises(
  1527. ObjectFormatException,
  1528. check_identity,
  1529. b"dborowitz@google.com>",
  1530. "incomplete email",
  1531. )
  1532. self.assertRaises(
  1533. ObjectFormatException,
  1534. check_identity,
  1535. b"Dave Borowitz <<dborowitz@google.com>",
  1536. "typo",
  1537. )
  1538. self.assertRaises(
  1539. ObjectFormatException,
  1540. check_identity,
  1541. b"Dave Borowitz <dborowitz@google.com>>",
  1542. "typo",
  1543. )
  1544. self.assertRaises(
  1545. ObjectFormatException,
  1546. check_identity,
  1547. b"Dave Borowitz <dborowitz@google.com>xxx",
  1548. "trailing characters",
  1549. )
  1550. self.assertRaises(
  1551. ObjectFormatException,
  1552. check_identity,
  1553. b"Dave Borowitz <dborowitz@google.com>xxx",
  1554. "trailing characters",
  1555. )
  1556. self.assertRaises(
  1557. ObjectFormatException,
  1558. check_identity,
  1559. b"Dave<Borowitz <dborowitz@google.com>",
  1560. "reserved byte in name",
  1561. )
  1562. self.assertRaises(
  1563. ObjectFormatException,
  1564. check_identity,
  1565. b"Dave>Borowitz <dborowitz@google.com>",
  1566. "reserved byte in name",
  1567. )
  1568. self.assertRaises(
  1569. ObjectFormatException,
  1570. check_identity,
  1571. b"Dave\0Borowitz <dborowitz@google.com>",
  1572. "null byte",
  1573. )
  1574. self.assertRaises(
  1575. ObjectFormatException,
  1576. check_identity,
  1577. b"Dave\nBorowitz <dborowitz@google.com>",
  1578. "newline byte",
  1579. )
  1580. class TimezoneTests(TestCase):
  1581. def test_parse_timezone_utc(self) -> None:
  1582. self.assertEqual((0, False), parse_timezone(b"+0000"))
  1583. def test_parse_timezone_utc_negative(self) -> None:
  1584. self.assertEqual((0, True), parse_timezone(b"-0000"))
  1585. def test_generate_timezone_utc(self) -> None:
  1586. self.assertEqual(b"+0000", format_timezone(0))
  1587. def test_generate_timezone_utc_negative(self) -> None:
  1588. self.assertEqual(b"-0000", format_timezone(0, True))
  1589. def test_parse_timezone_cet(self) -> None:
  1590. self.assertEqual((60 * 60, False), parse_timezone(b"+0100"))
  1591. def test_format_timezone_cet(self) -> None:
  1592. self.assertEqual(b"+0100", format_timezone(60 * 60))
  1593. def test_format_timezone_pdt(self) -> None:
  1594. self.assertEqual(b"-0400", format_timezone(-4 * 60 * 60))
  1595. def test_parse_timezone_pdt(self) -> None:
  1596. self.assertEqual((-4 * 60 * 60, False), parse_timezone(b"-0400"))
  1597. def test_format_timezone_pdt_half(self) -> None:
  1598. self.assertEqual(b"-0440", format_timezone(((-4 * 60) - 40) * 60))
  1599. def test_format_timezone_double_negative(self) -> None:
  1600. self.assertEqual(b"--700", format_timezone(((7 * 60) * 60), True))
  1601. def test_parse_timezone_pdt_half(self) -> None:
  1602. self.assertEqual((((-4 * 60) - 40) * 60, False), parse_timezone(b"-0440"))
  1603. def test_parse_timezone_double_negative(self) -> None:
  1604. self.assertEqual((((7 * 60) * 60), False), parse_timezone(b"+700"))
  1605. self.assertEqual((((7 * 60) * 60), True), parse_timezone(b"--700"))
  1606. class ShaFileCopyTests(TestCase):
  1607. def assert_copy(self, orig) -> None:
  1608. oclass = object_class(orig.type_num)
  1609. copy = orig.copy()
  1610. self.assertIsInstance(copy, oclass)
  1611. self.assertEqual(copy, orig)
  1612. self.assertIsNot(copy, orig)
  1613. def test_commit_copy(self) -> None:
  1614. attrs = {
  1615. "tree": b"d80c186a03f423a81b39df39dc87fd269736ca86",
  1616. "parents": [
  1617. b"ab64bbdcc51b170d21588e5c5d391ee5c0c96dfd",
  1618. b"4cffe90e0a41ad3f5190079d7c8f036bde29cbe6",
  1619. ],
  1620. "author": b"James Westby <jw+debian@jameswestby.net>",
  1621. "committer": b"James Westby <jw+debian@jameswestby.net>",
  1622. "commit_time": 1174773719,
  1623. "author_time": 1174773719,
  1624. "commit_timezone": 0,
  1625. "author_timezone": 0,
  1626. "message": b"Merge ../b\n",
  1627. }
  1628. commit = make_commit(**attrs)
  1629. self.assert_copy(commit)
  1630. def test_blob_copy(self) -> None:
  1631. blob = make_object(Blob, data=b"i am a blob")
  1632. self.assert_copy(blob)
  1633. def test_tree_copy(self) -> None:
  1634. blob = make_object(Blob, data=b"i am a blob")
  1635. tree = Tree()
  1636. tree[b"blob"] = (stat.S_IFREG, blob.id)
  1637. self.assert_copy(tree)
  1638. def test_tag_copy(self) -> None:
  1639. tag = make_object(
  1640. Tag,
  1641. name=b"tag",
  1642. message=b"",
  1643. tagger=b"Tagger <test@example.com>",
  1644. tag_time=12345,
  1645. tag_timezone=0,
  1646. object=(Commit, ZERO_SHA),
  1647. )
  1648. self.assert_copy(tag)
  1649. class ShaFileSerializeTests(TestCase):
  1650. """`ShaFile` objects only gets serialized once if they haven't changed."""
  1651. @contextmanager
  1652. def assert_serialization_on_change(
  1653. self, obj, needs_serialization_after_change=True
  1654. ):
  1655. old_id = obj.id
  1656. self.assertFalse(obj._needs_serialization)
  1657. yield obj
  1658. if needs_serialization_after_change:
  1659. self.assertTrue(obj._needs_serialization)
  1660. else:
  1661. self.assertFalse(obj._needs_serialization)
  1662. new_id = obj.id
  1663. self.assertFalse(obj._needs_serialization)
  1664. self.assertNotEqual(old_id, new_id)
  1665. def test_commit_serialize(self) -> None:
  1666. attrs = {
  1667. "tree": b"d80c186a03f423a81b39df39dc87fd269736ca86",
  1668. "parents": [
  1669. b"ab64bbdcc51b170d21588e5c5d391ee5c0c96dfd",
  1670. b"4cffe90e0a41ad3f5190079d7c8f036bde29cbe6",
  1671. ],
  1672. "author": b"James Westby <jw+debian@jameswestby.net>",
  1673. "committer": b"James Westby <jw+debian@jameswestby.net>",
  1674. "commit_time": 1174773719,
  1675. "author_time": 1174773719,
  1676. "commit_timezone": 0,
  1677. "author_timezone": 0,
  1678. "message": b"Merge ../b\n",
  1679. }
  1680. commit = make_commit(**attrs)
  1681. with self.assert_serialization_on_change(commit):
  1682. commit.parents = [b"ab64bbdcc51b170d21588e5c5d391ee5c0c96dfd"]
  1683. def test_blob_serialize(self) -> None:
  1684. blob = make_object(Blob, data=b"i am a blob")
  1685. with self.assert_serialization_on_change(
  1686. blob, needs_serialization_after_change=False
  1687. ):
  1688. blob.data = b"i am another blob"
  1689. def test_tree_serialize(self) -> None:
  1690. blob = make_object(Blob, data=b"i am a blob")
  1691. tree = Tree()
  1692. tree[b"blob"] = (stat.S_IFREG, blob.id)
  1693. with self.assert_serialization_on_change(tree):
  1694. tree[b"blob2"] = (stat.S_IFREG, blob.id)
  1695. def test_tag_serialize(self) -> None:
  1696. tag = make_object(
  1697. Tag,
  1698. name=b"tag",
  1699. message=b"",
  1700. tagger=b"Tagger <test@example.com>",
  1701. tag_time=12345,
  1702. tag_timezone=0,
  1703. object=(Commit, ZERO_SHA),
  1704. )
  1705. with self.assert_serialization_on_change(tag):
  1706. tag.message = b"new message"
  1707. def test_tag_serialize_time_error(self) -> None:
  1708. with self.assertRaises(ObjectFormatException):
  1709. tag = make_object(
  1710. Tag,
  1711. name=b"tag",
  1712. message=b"some message",
  1713. tagger=b"Tagger <test@example.com> 1174773719+0000",
  1714. object=(Commit, ZERO_SHA),
  1715. )
  1716. tag._deserialize(tag._serialize())
  1717. class PrettyFormatTreeEntryTests(TestCase):
  1718. def test_format(self) -> None:
  1719. self.assertEqual(
  1720. "40000 tree 40820c38cfb182ce6c8b261555410d8382a5918b\tfoo\n",
  1721. pretty_format_tree_entry(
  1722. b"foo", 0o40000, b"40820c38cfb182ce6c8b261555410d8382a5918b"
  1723. ),
  1724. )