test_objects.py 52 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471
  1. # test_objects.py -- tests for objects.py
  2. # Copyright (C) 2007 James Westby <jw+debian@jameswestby.net>
  3. #
  4. # Dulwich is dual-licensed under the Apache License, Version 2.0 and the GNU
  5. # General Public License as public by the Free Software Foundation; version 2.0
  6. # or (at your option) any later version. You can redistribute it and/or
  7. # modify it under the terms of either of these two licenses.
  8. #
  9. # Unless required by applicable law or agreed to in writing, software
  10. # distributed under the License is distributed on an "AS IS" BASIS,
  11. # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  12. # See the License for the specific language governing permissions and
  13. # limitations under the License.
  14. #
  15. # You should have received a copy of the licenses; if not, see
  16. # <http://www.gnu.org/licenses/> for a copy of the GNU General Public License
  17. # and <http://www.apache.org/licenses/LICENSE-2.0> for a copy of the Apache
  18. # License, Version 2.0.
  19. #
  20. """Tests for git base objects."""
  21. # TODO: Round-trip parse-serialize-parse and serialize-parse-serialize tests.
  22. import datetime
  23. import os
  24. import stat
  25. from contextlib import contextmanager
  26. from io import BytesIO
  27. from itertools import permutations
  28. from dulwich.errors import ObjectFormatException
  29. from dulwich.objects import (
  30. MAX_TIME,
  31. Blob,
  32. Commit,
  33. ShaFile,
  34. Tag,
  35. Tree,
  36. TreeEntry,
  37. _parse_tree_py,
  38. _sorted_tree_items_py,
  39. check_hexsha,
  40. check_identity,
  41. format_timezone,
  42. hex_to_filename,
  43. hex_to_sha,
  44. object_class,
  45. parse_timezone,
  46. pretty_format_tree_entry,
  47. sha_to_hex,
  48. )
  49. try:
  50. from dulwich.objects import _parse_tree_rs, _sorted_tree_items_rs
  51. except ImportError:
  52. _sorted_tree_items_rs = _parse_tree_rs = None
  53. from dulwich.tests.utils import (
  54. ext_functest_builder,
  55. functest_builder,
  56. make_commit,
  57. make_object,
  58. )
  59. from . import TestCase
  60. a_sha = b"6f670c0fb53f9463760b7295fbb814e965fb20c8"
  61. b_sha = b"2969be3e8ee1c0222396a5611407e4769f14e54b"
  62. c_sha = b"954a536f7819d40e6f637f849ee187dd10066349"
  63. tree_sha = b"70c190eb48fa8bbb50ddc692a17b44cb781af7f6"
  64. tag_sha = b"71033db03a03c6a36721efcf1968dd8f8e0cf023"
  65. class TestHexToSha(TestCase):
  66. def test_simple(self):
  67. self.assertEqual(b"\xab\xcd" * 10, hex_to_sha(b"abcd" * 10))
  68. def test_reverse(self):
  69. self.assertEqual(b"abcd" * 10, sha_to_hex(b"\xab\xcd" * 10))
  70. class BlobReadTests(TestCase):
  71. """Test decompression of blobs."""
  72. def get_sha_file(self, cls, base, sha):
  73. dir = os.path.join(os.path.dirname(__file__), "..", "testdata", base)
  74. return cls.from_path(hex_to_filename(dir, sha))
  75. def get_blob(self, sha):
  76. """Return the blob named sha from the test data dir."""
  77. return self.get_sha_file(Blob, "blobs", sha)
  78. def get_tree(self, sha):
  79. return self.get_sha_file(Tree, "trees", sha)
  80. def get_tag(self, sha):
  81. return self.get_sha_file(Tag, "tags", sha)
  82. def commit(self, sha):
  83. return self.get_sha_file(Commit, "commits", sha)
  84. def test_decompress_simple_blob(self):
  85. b = self.get_blob(a_sha)
  86. self.assertEqual(b.data, b"test 1\n")
  87. self.assertEqual(b.sha().hexdigest().encode("ascii"), a_sha)
  88. def test_hash(self):
  89. b = self.get_blob(a_sha)
  90. self.assertEqual(hash(b.id), hash(b))
  91. def test_parse_empty_blob_object(self):
  92. sha = b"e69de29bb2d1d6434b8b29ae775ad8c2e48c5391"
  93. b = self.get_blob(sha)
  94. self.assertEqual(b.data, b"")
  95. self.assertEqual(b.id, sha)
  96. self.assertEqual(b.sha().hexdigest().encode("ascii"), sha)
  97. def test_create_blob_from_string(self):
  98. string = b"test 2\n"
  99. b = Blob.from_string(string)
  100. self.assertEqual(b.data, string)
  101. self.assertEqual(b.sha().hexdigest().encode("ascii"), b_sha)
  102. def test_legacy_from_file(self):
  103. b1 = Blob.from_string(b"foo")
  104. b_raw = b1.as_legacy_object()
  105. b2 = b1.from_file(BytesIO(b_raw))
  106. self.assertEqual(b1, b2)
  107. def test_legacy_from_file_compression_level(self):
  108. b1 = Blob.from_string(b"foo")
  109. b_raw = b1.as_legacy_object(compression_level=6)
  110. b2 = b1.from_file(BytesIO(b_raw))
  111. self.assertEqual(b1, b2)
  112. def test_chunks(self):
  113. string = b"test 5\n"
  114. b = Blob.from_string(string)
  115. self.assertEqual([string], b.chunked)
  116. def test_splitlines(self):
  117. for case in [
  118. [],
  119. [b"foo\nbar\n"],
  120. [b"bl\na", b"blie"],
  121. [b"bl\na", b"blie", b"bloe\n"],
  122. [b"", b"bl\na", b"blie", b"bloe\n"],
  123. [b"", b"", b"", b"bla\n"],
  124. [b"", b"", b"", b"bla\n", b""],
  125. [b"bl", b"", b"a\naaa"],
  126. [b"a\naaa", b"a"],
  127. ]:
  128. b = Blob()
  129. b.chunked = case
  130. self.assertEqual(b.data.splitlines(True), b.splitlines())
  131. def test_set_chunks(self):
  132. b = Blob()
  133. b.chunked = [b"te", b"st", b" 5\n"]
  134. self.assertEqual(b"test 5\n", b.data)
  135. b.chunked = [b"te", b"st", b" 6\n"]
  136. self.assertEqual(b"test 6\n", b.as_raw_string())
  137. self.assertEqual(b"test 6\n", bytes(b))
  138. def test_parse_legacy_blob(self):
  139. string = b"test 3\n"
  140. b = self.get_blob(c_sha)
  141. self.assertEqual(b.data, string)
  142. self.assertEqual(b.sha().hexdigest().encode("ascii"), c_sha)
  143. def test_eq(self):
  144. blob1 = self.get_blob(a_sha)
  145. blob2 = self.get_blob(a_sha)
  146. self.assertEqual(blob1, blob2)
  147. def test_read_tree_from_file(self):
  148. t = self.get_tree(tree_sha)
  149. self.assertEqual(t.items()[0], (b"a", 33188, a_sha))
  150. self.assertEqual(t.items()[1], (b"b", 33188, b_sha))
  151. def test_read_tree_from_file_parse_count(self):
  152. old_deserialize = Tree._deserialize
  153. def reset_deserialize():
  154. Tree._deserialize = old_deserialize
  155. self.addCleanup(reset_deserialize)
  156. self.deserialize_count = 0
  157. def counting_deserialize(*args, **kwargs):
  158. self.deserialize_count += 1
  159. return old_deserialize(*args, **kwargs)
  160. Tree._deserialize = counting_deserialize
  161. t = self.get_tree(tree_sha)
  162. self.assertEqual(t.items()[0], (b"a", 33188, a_sha))
  163. self.assertEqual(t.items()[1], (b"b", 33188, b_sha))
  164. self.assertEqual(self.deserialize_count, 1)
  165. def test_read_tag_from_file(self):
  166. t = self.get_tag(tag_sha)
  167. self.assertEqual(
  168. t.object, (Commit, b"51b668fd5bf7061b7d6fa525f88803e6cfadaa51")
  169. )
  170. self.assertEqual(t.name, b"signed")
  171. self.assertEqual(t.tagger, b"Ali Sabil <ali.sabil@gmail.com>")
  172. self.assertEqual(t.tag_time, 1231203091)
  173. self.assertEqual(t.message, b"This is a signed tag\n")
  174. self.assertEqual(
  175. t.signature,
  176. b"-----BEGIN PGP SIGNATURE-----\n"
  177. b"Version: GnuPG v1.4.9 (GNU/Linux)\n"
  178. b"\n"
  179. b"iEYEABECAAYFAkliqx8ACgkQqSMmLy9u/"
  180. b"kcx5ACfakZ9NnPl02tOyYP6pkBoEkU1\n"
  181. b"5EcAn0UFgokaSvS371Ym/4W9iJj6vh3h\n"
  182. b"=ql7y\n"
  183. b"-----END PGP SIGNATURE-----\n",
  184. )
  185. def test_read_commit_from_file(self):
  186. sha = b"60dacdc733de308bb77bb76ce0fb0f9b44c9769e"
  187. c = self.commit(sha)
  188. self.assertEqual(c.tree, tree_sha)
  189. self.assertEqual(c.parents, [b"0d89f20333fbb1d2f3a94da77f4981373d8f4310"])
  190. self.assertEqual(c.author, b"James Westby <jw+debian@jameswestby.net>")
  191. self.assertEqual(c.committer, b"James Westby <jw+debian@jameswestby.net>")
  192. self.assertEqual(c.commit_time, 1174759230)
  193. self.assertEqual(c.commit_timezone, 0)
  194. self.assertEqual(c.author_timezone, 0)
  195. self.assertEqual(c.message, b"Test commit\n")
  196. def test_read_commit_no_parents(self):
  197. sha = b"0d89f20333fbb1d2f3a94da77f4981373d8f4310"
  198. c = self.commit(sha)
  199. self.assertEqual(c.tree, b"90182552c4a85a45ec2a835cadc3451bebdfe870")
  200. self.assertEqual(c.parents, [])
  201. self.assertEqual(c.author, b"James Westby <jw+debian@jameswestby.net>")
  202. self.assertEqual(c.committer, b"James Westby <jw+debian@jameswestby.net>")
  203. self.assertEqual(c.commit_time, 1174758034)
  204. self.assertEqual(c.commit_timezone, 0)
  205. self.assertEqual(c.author_timezone, 0)
  206. self.assertEqual(c.message, b"Test commit\n")
  207. def test_read_commit_two_parents(self):
  208. sha = b"5dac377bdded4c9aeb8dff595f0faeebcc8498cc"
  209. c = self.commit(sha)
  210. self.assertEqual(c.tree, b"d80c186a03f423a81b39df39dc87fd269736ca86")
  211. self.assertEqual(
  212. c.parents,
  213. [
  214. b"ab64bbdcc51b170d21588e5c5d391ee5c0c96dfd",
  215. b"4cffe90e0a41ad3f5190079d7c8f036bde29cbe6",
  216. ],
  217. )
  218. self.assertEqual(c.author, b"James Westby <jw+debian@jameswestby.net>")
  219. self.assertEqual(c.committer, b"James Westby <jw+debian@jameswestby.net>")
  220. self.assertEqual(c.commit_time, 1174773719)
  221. self.assertEqual(c.commit_timezone, 0)
  222. self.assertEqual(c.author_timezone, 0)
  223. self.assertEqual(c.message, b"Merge ../b\n")
  224. def test_stub_sha(self):
  225. sha = b"5" * 40
  226. c = make_commit(id=sha, message=b"foo")
  227. self.assertIsInstance(c, Commit)
  228. self.assertEqual(sha, c.id)
  229. self.assertNotEqual(sha, c.sha())
  230. class ShaFileCheckTests(TestCase):
  231. def assertCheckFails(self, cls, data):
  232. obj = cls()
  233. def do_check():
  234. obj.set_raw_string(data)
  235. obj.check()
  236. self.assertRaises(ObjectFormatException, do_check)
  237. def assertCheckSucceeds(self, cls, data):
  238. obj = cls()
  239. obj.set_raw_string(data)
  240. self.assertEqual(None, obj.check())
  241. small_buffer_zlib_object = (
  242. b"\x48\x89\x15\xcc\x31\x0e\xc2\x30\x0c\x40\x51\xe6"
  243. b"\x9c\xc2\x3b\xaa\x64\x37\xc4\xc1\x12\x42\x5c\xc5"
  244. b"\x49\xac\x52\xd4\x92\xaa\x78\xe1\xf6\x94\xed\xeb"
  245. b"\x0d\xdf\x75\x02\xa2\x7c\xea\xe5\x65\xd5\x81\x8b"
  246. b"\x9a\x61\xba\xa0\xa9\x08\x36\xc9\x4c\x1a\xad\x88"
  247. b"\x16\xba\x46\xc4\xa8\x99\x6a\x64\xe1\xe0\xdf\xcd"
  248. b"\xa0\xf6\x75\x9d\x3d\xf8\xf1\xd0\x77\xdb\xfb\xdc"
  249. b"\x86\xa3\x87\xf1\x2f\x93\xed\x00\xb7\xc7\xd2\xab"
  250. b"\x2e\xcf\xfe\xf1\x3b\x50\xa4\x91\x53\x12\x24\x38"
  251. b"\x23\x21\x86\xf0\x03\x2f\x91\x24\x52"
  252. )
  253. class ShaFileTests(TestCase):
  254. def test_deflated_smaller_window_buffer(self):
  255. # zlib on some systems uses smaller buffers,
  256. # resulting in a different header.
  257. # See https://github.com/libgit2/libgit2/pull/464
  258. sf = ShaFile.from_file(BytesIO(small_buffer_zlib_object))
  259. self.assertEqual(sf.type_name, b"tag")
  260. self.assertEqual(sf.tagger, b" <@localhost>")
  261. class CommitSerializationTests(TestCase):
  262. def make_commit(self, **kwargs):
  263. attrs = {
  264. "tree": b"d80c186a03f423a81b39df39dc87fd269736ca86",
  265. "parents": [
  266. b"ab64bbdcc51b170d21588e5c5d391ee5c0c96dfd",
  267. b"4cffe90e0a41ad3f5190079d7c8f036bde29cbe6",
  268. ],
  269. "author": b"James Westby <jw+debian@jameswestby.net>",
  270. "committer": b"James Westby <jw+debian@jameswestby.net>",
  271. "commit_time": 1174773719,
  272. "author_time": 1174773719,
  273. "commit_timezone": 0,
  274. "author_timezone": 0,
  275. "message": b"Merge ../b\n",
  276. }
  277. attrs.update(kwargs)
  278. return make_commit(**attrs)
  279. def test_encoding(self):
  280. c = self.make_commit(encoding=b"iso8859-1")
  281. self.assertIn(b"encoding iso8859-1\n", c.as_raw_string())
  282. def test_short_timestamp(self):
  283. c = self.make_commit(commit_time=30)
  284. c1 = Commit()
  285. c1.set_raw_string(c.as_raw_string())
  286. self.assertEqual(30, c1.commit_time)
  287. def test_full_tree(self):
  288. c = self.make_commit(commit_time=30)
  289. t = Tree()
  290. t.add(b"data-x", 0o644, Blob().id)
  291. c.tree = t
  292. c1 = Commit()
  293. c1.set_raw_string(c.as_raw_string())
  294. self.assertEqual(t.id, c1.tree)
  295. self.assertEqual(c.as_raw_string(), c1.as_raw_string())
  296. def test_raw_length(self):
  297. c = self.make_commit()
  298. self.assertEqual(len(c.as_raw_string()), c.raw_length())
  299. def test_simple(self):
  300. c = self.make_commit()
  301. self.assertEqual(c.id, b"5dac377bdded4c9aeb8dff595f0faeebcc8498cc")
  302. self.assertEqual(
  303. b"tree d80c186a03f423a81b39df39dc87fd269736ca86\n"
  304. b"parent ab64bbdcc51b170d21588e5c5d391ee5c0c96dfd\n"
  305. b"parent 4cffe90e0a41ad3f5190079d7c8f036bde29cbe6\n"
  306. b"author James Westby <jw+debian@jameswestby.net> "
  307. b"1174773719 +0000\n"
  308. b"committer James Westby <jw+debian@jameswestby.net> "
  309. b"1174773719 +0000\n"
  310. b"\n"
  311. b"Merge ../b\n",
  312. c.as_raw_string(),
  313. )
  314. def test_timezone(self):
  315. c = self.make_commit(commit_timezone=(5 * 60))
  316. self.assertIn(b" +0005\n", c.as_raw_string())
  317. def test_neg_timezone(self):
  318. c = self.make_commit(commit_timezone=(-1 * 3600))
  319. self.assertIn(b" -0100\n", c.as_raw_string())
  320. def test_deserialize(self):
  321. c = self.make_commit()
  322. d = Commit()
  323. d._deserialize(c.as_raw_chunks())
  324. self.assertEqual(c, d)
  325. def test_serialize_gpgsig(self):
  326. commit = self.make_commit(
  327. gpgsig=b"""-----BEGIN PGP SIGNATURE-----
  328. Version: GnuPG v1
  329. iQIcBAABCgAGBQJULCdfAAoJEACAbyvXKaRXuKwP/RyP9PA49uAvu8tQVCC/uBa8
  330. vi975+xvO14R8Pp8k2nps7lSxCdtCd+xVT1VRHs0wNhOZo2YCVoU1HATkPejqSeV
  331. NScTHcxnk4/+bxyfk14xvJkNp7FlQ3npmBkA+lbV0Ubr33rvtIE5jiJPyz+SgWAg
  332. xdBG2TojV0squj00GoH/euK6aX7GgZtwdtpTv44haCQdSuPGDcI4TORqR6YSqvy3
  333. GPE+3ZqXPFFb+KILtimkxitdwB7CpwmNse2vE3rONSwTvi8nq3ZoQYNY73CQGkUy
  334. qoFU0pDtw87U3niFin1ZccDgH0bB6624sLViqrjcbYJeg815Htsu4rmzVaZADEVC
  335. XhIO4MThebusdk0AcNGjgpf3HRHk0DPMDDlIjm+Oao0cqovvF6VyYmcb0C+RmhJj
  336. dodLXMNmbqErwTk3zEkW0yZvNIYXH7m9SokPCZa4eeIM7be62X6h1mbt0/IU6Th+
  337. v18fS0iTMP/Viug5und+05C/v04kgDo0CPphAbXwWMnkE4B6Tl9sdyUYXtvQsL7x
  338. 0+WP1gL27ANqNZiI07Kz/BhbBAQI/+2TFT7oGr0AnFPQ5jHp+3GpUf6OKuT1wT3H
  339. ND189UFuRuubxb42vZhpcXRbqJVWnbECTKVUPsGZqat3enQUB63uM4i6/RdONDZA
  340. fDeF1m4qYs+cUXKNUZ03
  341. =X6RT
  342. -----END PGP SIGNATURE-----"""
  343. )
  344. self.maxDiff = None
  345. self.assertEqual(
  346. 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. gpgsig -----BEGIN PGP SIGNATURE-----
  353. Version: GnuPG v1
  354. iQIcBAABCgAGBQJULCdfAAoJEACAbyvXKaRXuKwP/RyP9PA49uAvu8tQVCC/uBa8
  355. vi975+xvO14R8Pp8k2nps7lSxCdtCd+xVT1VRHs0wNhOZo2YCVoU1HATkPejqSeV
  356. NScTHcxnk4/+bxyfk14xvJkNp7FlQ3npmBkA+lbV0Ubr33rvtIE5jiJPyz+SgWAg
  357. xdBG2TojV0squj00GoH/euK6aX7GgZtwdtpTv44haCQdSuPGDcI4TORqR6YSqvy3
  358. GPE+3ZqXPFFb+KILtimkxitdwB7CpwmNse2vE3rONSwTvi8nq3ZoQYNY73CQGkUy
  359. qoFU0pDtw87U3niFin1ZccDgH0bB6624sLViqrjcbYJeg815Htsu4rmzVaZADEVC
  360. XhIO4MThebusdk0AcNGjgpf3HRHk0DPMDDlIjm+Oao0cqovvF6VyYmcb0C+RmhJj
  361. dodLXMNmbqErwTk3zEkW0yZvNIYXH7m9SokPCZa4eeIM7be62X6h1mbt0/IU6Th+
  362. v18fS0iTMP/Viug5und+05C/v04kgDo0CPphAbXwWMnkE4B6Tl9sdyUYXtvQsL7x
  363. 0+WP1gL27ANqNZiI07Kz/BhbBAQI/+2TFT7oGr0AnFPQ5jHp+3GpUf6OKuT1wT3H
  364. ND189UFuRuubxb42vZhpcXRbqJVWnbECTKVUPsGZqat3enQUB63uM4i6/RdONDZA
  365. fDeF1m4qYs+cUXKNUZ03
  366. =X6RT
  367. -----END PGP SIGNATURE-----
  368. Merge ../b
  369. """,
  370. commit.as_raw_string(),
  371. )
  372. def test_serialize_mergetag(self):
  373. tag = make_object(
  374. Tag,
  375. object=(Commit, b"a38d6181ff27824c79fc7df825164a212eff6a3f"),
  376. object_type_name=b"commit",
  377. name=b"v2.6.22-rc7",
  378. tag_time=1183319674,
  379. tag_timezone=0,
  380. tagger=b"Linus Torvalds <torvalds@woody.linux-foundation.org>",
  381. message=default_message,
  382. )
  383. commit = self.make_commit(mergetag=[tag])
  384. self.assertEqual(
  385. b"""tree d80c186a03f423a81b39df39dc87fd269736ca86
  386. parent ab64bbdcc51b170d21588e5c5d391ee5c0c96dfd
  387. parent 4cffe90e0a41ad3f5190079d7c8f036bde29cbe6
  388. author James Westby <jw+debian@jameswestby.net> 1174773719 +0000
  389. committer James Westby <jw+debian@jameswestby.net> 1174773719 +0000
  390. mergetag object a38d6181ff27824c79fc7df825164a212eff6a3f
  391. type commit
  392. tag v2.6.22-rc7
  393. tagger Linus Torvalds <torvalds@woody.linux-foundation.org> 1183319674 +0000
  394. Linux 2.6.22-rc7
  395. -----BEGIN PGP SIGNATURE-----
  396. Version: GnuPG v1.4.7 (GNU/Linux)
  397. iD8DBQBGiAaAF3YsRnbiHLsRAitMAKCiLboJkQECM/jpYsY3WPfvUgLXkACgg3ql
  398. OK2XeQOiEeXtT76rV4t2WR4=
  399. =ivrA
  400. -----END PGP SIGNATURE-----
  401. Merge ../b
  402. """,
  403. commit.as_raw_string(),
  404. )
  405. def test_serialize_mergetags(self):
  406. tag = make_object(
  407. Tag,
  408. object=(Commit, b"a38d6181ff27824c79fc7df825164a212eff6a3f"),
  409. object_type_name=b"commit",
  410. name=b"v2.6.22-rc7",
  411. tag_time=1183319674,
  412. tag_timezone=0,
  413. tagger=b"Linus Torvalds <torvalds@woody.linux-foundation.org>",
  414. message=default_message,
  415. )
  416. commit = self.make_commit(mergetag=[tag, tag])
  417. self.assertEqual(
  418. b"""tree d80c186a03f423a81b39df39dc87fd269736ca86
  419. parent ab64bbdcc51b170d21588e5c5d391ee5c0c96dfd
  420. parent 4cffe90e0a41ad3f5190079d7c8f036bde29cbe6
  421. author James Westby <jw+debian@jameswestby.net> 1174773719 +0000
  422. committer James Westby <jw+debian@jameswestby.net> 1174773719 +0000
  423. mergetag object a38d6181ff27824c79fc7df825164a212eff6a3f
  424. type commit
  425. tag v2.6.22-rc7
  426. tagger Linus Torvalds <torvalds@woody.linux-foundation.org> 1183319674 +0000
  427. Linux 2.6.22-rc7
  428. -----BEGIN PGP SIGNATURE-----
  429. Version: GnuPG v1.4.7 (GNU/Linux)
  430. iD8DBQBGiAaAF3YsRnbiHLsRAitMAKCiLboJkQECM/jpYsY3WPfvUgLXkACgg3ql
  431. OK2XeQOiEeXtT76rV4t2WR4=
  432. =ivrA
  433. -----END PGP SIGNATURE-----
  434. mergetag object a38d6181ff27824c79fc7df825164a212eff6a3f
  435. type commit
  436. tag v2.6.22-rc7
  437. tagger Linus Torvalds <torvalds@woody.linux-foundation.org> 1183319674 +0000
  438. Linux 2.6.22-rc7
  439. -----BEGIN PGP SIGNATURE-----
  440. Version: GnuPG v1.4.7 (GNU/Linux)
  441. iD8DBQBGiAaAF3YsRnbiHLsRAitMAKCiLboJkQECM/jpYsY3WPfvUgLXkACgg3ql
  442. OK2XeQOiEeXtT76rV4t2WR4=
  443. =ivrA
  444. -----END PGP SIGNATURE-----
  445. Merge ../b
  446. """,
  447. commit.as_raw_string(),
  448. )
  449. def test_deserialize_mergetag(self):
  450. tag = make_object(
  451. Tag,
  452. object=(Commit, b"a38d6181ff27824c79fc7df825164a212eff6a3f"),
  453. object_type_name=b"commit",
  454. name=b"v2.6.22-rc7",
  455. tag_time=1183319674,
  456. tag_timezone=0,
  457. tagger=b"Linus Torvalds <torvalds@woody.linux-foundation.org>",
  458. message=default_message,
  459. )
  460. commit = self.make_commit(mergetag=[tag])
  461. d = Commit()
  462. d._deserialize(commit.as_raw_chunks())
  463. self.assertEqual(commit, d)
  464. def test_deserialize_mergetags(self):
  465. tag = make_object(
  466. Tag,
  467. object=(Commit, b"a38d6181ff27824c79fc7df825164a212eff6a3f"),
  468. object_type_name=b"commit",
  469. name=b"v2.6.22-rc7",
  470. tag_time=1183319674,
  471. tag_timezone=0,
  472. tagger=b"Linus Torvalds <torvalds@woody.linux-foundation.org>",
  473. message=default_message,
  474. )
  475. commit = self.make_commit(mergetag=[tag, tag])
  476. d = Commit()
  477. d._deserialize(commit.as_raw_chunks())
  478. self.assertEqual(commit, d)
  479. default_committer = b"James Westby <jw+debian@jameswestby.net> 1174773719 +0000"
  480. class CommitParseTests(ShaFileCheckTests):
  481. def make_commit_lines(
  482. self,
  483. tree=b"d80c186a03f423a81b39df39dc87fd269736ca86",
  484. parents=[
  485. b"ab64bbdcc51b170d21588e5c5d391ee5c0c96dfd",
  486. b"4cffe90e0a41ad3f5190079d7c8f036bde29cbe6",
  487. ],
  488. author=default_committer,
  489. committer=default_committer,
  490. encoding=None,
  491. message=b"Merge ../b\n",
  492. extra=None,
  493. ):
  494. lines = []
  495. if tree is not None:
  496. lines.append(b"tree " + tree)
  497. if parents is not None:
  498. lines.extend(b"parent " + p for p in parents)
  499. if author is not None:
  500. lines.append(b"author " + author)
  501. if committer is not None:
  502. lines.append(b"committer " + committer)
  503. if encoding is not None:
  504. lines.append(b"encoding " + encoding)
  505. if extra is not None:
  506. for name, value in sorted(extra.items()):
  507. lines.append(name + b" " + value)
  508. lines.append(b"")
  509. if message is not None:
  510. lines.append(message)
  511. return lines
  512. def make_commit_text(self, **kwargs):
  513. return b"\n".join(self.make_commit_lines(**kwargs))
  514. def test_simple(self):
  515. c = Commit.from_string(self.make_commit_text())
  516. self.assertEqual(b"Merge ../b\n", c.message)
  517. self.assertEqual(b"James Westby <jw+debian@jameswestby.net>", c.author)
  518. self.assertEqual(b"James Westby <jw+debian@jameswestby.net>", c.committer)
  519. self.assertEqual(b"d80c186a03f423a81b39df39dc87fd269736ca86", c.tree)
  520. self.assertEqual(
  521. [
  522. b"ab64bbdcc51b170d21588e5c5d391ee5c0c96dfd",
  523. b"4cffe90e0a41ad3f5190079d7c8f036bde29cbe6",
  524. ],
  525. c.parents,
  526. )
  527. expected_time = datetime.datetime(2007, 3, 24, 22, 1, 59)
  528. self.assertEqual(
  529. expected_time, datetime.datetime.utcfromtimestamp(c.commit_time)
  530. )
  531. self.assertEqual(0, c.commit_timezone)
  532. self.assertEqual(
  533. expected_time, datetime.datetime.utcfromtimestamp(c.author_time)
  534. )
  535. self.assertEqual(0, c.author_timezone)
  536. self.assertEqual(None, c.encoding)
  537. def test_custom(self):
  538. c = Commit.from_string(self.make_commit_text(extra={b"extra-field": b"data"}))
  539. self.assertEqual([(b"extra-field", b"data")], c._extra)
  540. def test_encoding(self):
  541. c = Commit.from_string(self.make_commit_text(encoding=b"UTF-8"))
  542. self.assertEqual(b"UTF-8", c.encoding)
  543. def test_check(self):
  544. self.assertCheckSucceeds(Commit, self.make_commit_text())
  545. self.assertCheckSucceeds(Commit, self.make_commit_text(parents=None))
  546. self.assertCheckSucceeds(Commit, self.make_commit_text(encoding=b"UTF-8"))
  547. self.assertCheckFails(Commit, self.make_commit_text(tree=b"xxx"))
  548. self.assertCheckFails(Commit, self.make_commit_text(parents=[a_sha, b"xxx"]))
  549. bad_committer = b"some guy without an email address 1174773719 +0000"
  550. self.assertCheckFails(Commit, self.make_commit_text(committer=bad_committer))
  551. self.assertCheckFails(Commit, self.make_commit_text(author=bad_committer))
  552. self.assertCheckFails(Commit, self.make_commit_text(author=None))
  553. self.assertCheckFails(Commit, self.make_commit_text(committer=None))
  554. self.assertCheckFails(
  555. Commit, self.make_commit_text(author=None, committer=None)
  556. )
  557. def test_check_duplicates(self):
  558. # duplicate each of the header fields
  559. for i in range(5):
  560. lines = self.make_commit_lines(parents=[a_sha], encoding=b"UTF-8")
  561. lines.insert(i, lines[i])
  562. text = b"\n".join(lines)
  563. if lines[i].startswith(b"parent"):
  564. # duplicate parents are ok for now
  565. self.assertCheckSucceeds(Commit, text)
  566. else:
  567. self.assertCheckFails(Commit, text)
  568. def test_check_order(self):
  569. lines = self.make_commit_lines(parents=[a_sha], encoding=b"UTF-8")
  570. headers = lines[:5]
  571. rest = lines[5:]
  572. # of all possible permutations, ensure only the original succeeds
  573. for perm in permutations(headers):
  574. perm = list(perm)
  575. text = b"\n".join(perm + rest)
  576. if perm == headers:
  577. self.assertCheckSucceeds(Commit, text)
  578. else:
  579. self.assertCheckFails(Commit, text)
  580. def test_check_commit_with_unparseable_time(self):
  581. identity_with_wrong_time = (
  582. b"Igor Sysoev <igor@sysoev.ru> 18446743887488505614+42707004"
  583. )
  584. # Those fail at reading time
  585. self.assertCheckFails(
  586. Commit,
  587. self.make_commit_text(
  588. author=default_committer, committer=identity_with_wrong_time
  589. ),
  590. )
  591. self.assertCheckFails(
  592. Commit,
  593. self.make_commit_text(
  594. author=identity_with_wrong_time, committer=default_committer
  595. ),
  596. )
  597. def test_check_commit_with_overflow_date(self):
  598. """Date with overflow should raise an ObjectFormatException when checked."""
  599. identity_with_wrong_time = (
  600. b"Igor Sysoev <igor@sysoev.ru> 18446743887488505614 +42707004"
  601. )
  602. commit0 = Commit.from_string(
  603. self.make_commit_text(
  604. author=identity_with_wrong_time, committer=default_committer
  605. )
  606. )
  607. commit1 = Commit.from_string(
  608. self.make_commit_text(
  609. author=default_committer, committer=identity_with_wrong_time
  610. )
  611. )
  612. # Those fails when triggering the check() method
  613. for commit in [commit0, commit1]:
  614. with self.assertRaises(ObjectFormatException):
  615. commit.check()
  616. def test_mangled_author_line(self):
  617. """Mangled author line should successfully parse."""
  618. author_line = (
  619. b'Karl MacMillan <kmacmill@redhat.com> <"Karl MacMillan '
  620. b'<kmacmill@redhat.com>"> 1197475547 -0500'
  621. )
  622. expected_identity = (
  623. b'Karl MacMillan <kmacmill@redhat.com> <"Karl MacMillan '
  624. b'<kmacmill@redhat.com>">'
  625. )
  626. commit = Commit.from_string(self.make_commit_text(author=author_line))
  627. # The commit parses properly
  628. self.assertEqual(commit.author, expected_identity)
  629. # But the check fails because the author identity is bogus
  630. with self.assertRaises(ObjectFormatException):
  631. commit.check()
  632. def test_parse_gpgsig(self):
  633. c = Commit.from_string(
  634. b"""tree aaff74984cccd156a469afa7d9ab10e4777beb24
  635. author Jelmer Vernooij <jelmer@samba.org> 1412179807 +0200
  636. committer Jelmer Vernooij <jelmer@samba.org> 1412179807 +0200
  637. gpgsig -----BEGIN PGP SIGNATURE-----
  638. Version: GnuPG v1
  639. iQIcBAABCgAGBQJULCdfAAoJEACAbyvXKaRXuKwP/RyP9PA49uAvu8tQVCC/uBa8
  640. vi975+xvO14R8Pp8k2nps7lSxCdtCd+xVT1VRHs0wNhOZo2YCVoU1HATkPejqSeV
  641. NScTHcxnk4/+bxyfk14xvJkNp7FlQ3npmBkA+lbV0Ubr33rvtIE5jiJPyz+SgWAg
  642. xdBG2TojV0squj00GoH/euK6aX7GgZtwdtpTv44haCQdSuPGDcI4TORqR6YSqvy3
  643. GPE+3ZqXPFFb+KILtimkxitdwB7CpwmNse2vE3rONSwTvi8nq3ZoQYNY73CQGkUy
  644. qoFU0pDtw87U3niFin1ZccDgH0bB6624sLViqrjcbYJeg815Htsu4rmzVaZADEVC
  645. XhIO4MThebusdk0AcNGjgpf3HRHk0DPMDDlIjm+Oao0cqovvF6VyYmcb0C+RmhJj
  646. dodLXMNmbqErwTk3zEkW0yZvNIYXH7m9SokPCZa4eeIM7be62X6h1mbt0/IU6Th+
  647. v18fS0iTMP/Viug5und+05C/v04kgDo0CPphAbXwWMnkE4B6Tl9sdyUYXtvQsL7x
  648. 0+WP1gL27ANqNZiI07Kz/BhbBAQI/+2TFT7oGr0AnFPQ5jHp+3GpUf6OKuT1wT3H
  649. ND189UFuRuubxb42vZhpcXRbqJVWnbECTKVUPsGZqat3enQUB63uM4i6/RdONDZA
  650. fDeF1m4qYs+cUXKNUZ03
  651. =X6RT
  652. -----END PGP SIGNATURE-----
  653. foo
  654. """
  655. )
  656. self.assertEqual(b"foo\n", c.message)
  657. self.assertEqual([], c._extra)
  658. self.assertEqual(
  659. b"""-----BEGIN PGP SIGNATURE-----
  660. Version: GnuPG v1
  661. iQIcBAABCgAGBQJULCdfAAoJEACAbyvXKaRXuKwP/RyP9PA49uAvu8tQVCC/uBa8
  662. vi975+xvO14R8Pp8k2nps7lSxCdtCd+xVT1VRHs0wNhOZo2YCVoU1HATkPejqSeV
  663. NScTHcxnk4/+bxyfk14xvJkNp7FlQ3npmBkA+lbV0Ubr33rvtIE5jiJPyz+SgWAg
  664. xdBG2TojV0squj00GoH/euK6aX7GgZtwdtpTv44haCQdSuPGDcI4TORqR6YSqvy3
  665. GPE+3ZqXPFFb+KILtimkxitdwB7CpwmNse2vE3rONSwTvi8nq3ZoQYNY73CQGkUy
  666. qoFU0pDtw87U3niFin1ZccDgH0bB6624sLViqrjcbYJeg815Htsu4rmzVaZADEVC
  667. XhIO4MThebusdk0AcNGjgpf3HRHk0DPMDDlIjm+Oao0cqovvF6VyYmcb0C+RmhJj
  668. dodLXMNmbqErwTk3zEkW0yZvNIYXH7m9SokPCZa4eeIM7be62X6h1mbt0/IU6Th+
  669. v18fS0iTMP/Viug5und+05C/v04kgDo0CPphAbXwWMnkE4B6Tl9sdyUYXtvQsL7x
  670. 0+WP1gL27ANqNZiI07Kz/BhbBAQI/+2TFT7oGr0AnFPQ5jHp+3GpUf6OKuT1wT3H
  671. ND189UFuRuubxb42vZhpcXRbqJVWnbECTKVUPsGZqat3enQUB63uM4i6/RdONDZA
  672. fDeF1m4qYs+cUXKNUZ03
  673. =X6RT
  674. -----END PGP SIGNATURE-----""",
  675. c.gpgsig,
  676. )
  677. def test_parse_header_trailing_newline(self):
  678. c = Commit.from_string(
  679. b"""\
  680. tree a7d6277f78d3ecd0230a1a5df6db00b1d9c521ac
  681. parent c09b6dec7a73760fbdb478383a3c926b18db8bbe
  682. author Neil Matatall <oreoshake@github.com> 1461964057 -1000
  683. committer Neil Matatall <oreoshake@github.com> 1461964057 -1000
  684. gpgsig -----BEGIN PGP SIGNATURE-----
  685. wsBcBAABCAAQBQJXI80ZCRA6pcNDcVZ70gAAarcIABs72xRX3FWeox349nh6ucJK
  686. CtwmBTusez2Zwmq895fQEbZK7jpaGO5TRO4OvjFxlRo0E08UFx3pxZHSpj6bsFeL
  687. hHsDXnCaotphLkbgKKRdGZo7tDqM84wuEDlh4MwNe7qlFC7bYLDyysc81ZX5lpMm
  688. 2MFF1TvjLAzSvkT7H1LPkuR3hSvfCYhikbPOUNnKOo0sYjeJeAJ/JdAVQ4mdJIM0
  689. gl3REp9+A+qBEpNQI7z94Pg5Bc5xenwuDh3SJgHvJV6zBWupWcdB3fAkVd4TPnEZ
  690. nHxksHfeNln9RKseIDcy4b2ATjhDNIJZARHNfr6oy4u3XPW4svRqtBsLoMiIeuI=
  691. =ms6q
  692. -----END PGP SIGNATURE-----
  693. 3.3.0 version bump and docs
  694. """
  695. )
  696. self.assertEqual([], c._extra)
  697. self.assertEqual(
  698. b"""\
  699. -----BEGIN PGP SIGNATURE-----
  700. wsBcBAABCAAQBQJXI80ZCRA6pcNDcVZ70gAAarcIABs72xRX3FWeox349nh6ucJK
  701. CtwmBTusez2Zwmq895fQEbZK7jpaGO5TRO4OvjFxlRo0E08UFx3pxZHSpj6bsFeL
  702. hHsDXnCaotphLkbgKKRdGZo7tDqM84wuEDlh4MwNe7qlFC7bYLDyysc81ZX5lpMm
  703. 2MFF1TvjLAzSvkT7H1LPkuR3hSvfCYhikbPOUNnKOo0sYjeJeAJ/JdAVQ4mdJIM0
  704. gl3REp9+A+qBEpNQI7z94Pg5Bc5xenwuDh3SJgHvJV6zBWupWcdB3fAkVd4TPnEZ
  705. nHxksHfeNln9RKseIDcy4b2ATjhDNIJZARHNfr6oy4u3XPW4svRqtBsLoMiIeuI=
  706. =ms6q
  707. -----END PGP SIGNATURE-----\n""",
  708. c.gpgsig,
  709. )
  710. _TREE_ITEMS = {
  711. b"a.c": (0o100755, b"d80c186a03f423a81b39df39dc87fd269736ca86"),
  712. b"a": (stat.S_IFDIR, b"d80c186a03f423a81b39df39dc87fd269736ca86"),
  713. b"a/c": (stat.S_IFDIR, b"d80c186a03f423a81b39df39dc87fd269736ca86"),
  714. }
  715. _SORTED_TREE_ITEMS = [
  716. TreeEntry(b"a.c", 0o100755, b"d80c186a03f423a81b39df39dc87fd269736ca86"),
  717. TreeEntry(b"a", stat.S_IFDIR, b"d80c186a03f423a81b39df39dc87fd269736ca86"),
  718. TreeEntry(b"a/c", stat.S_IFDIR, b"d80c186a03f423a81b39df39dc87fd269736ca86"),
  719. ]
  720. _TREE_ITEMS_BUG_1325 = {
  721. b'gamma': (0o100755, b'5944b31ff85b415573d1a43eb942e2dea30ab8be'),
  722. b'gamma-new': (0o100644, b'cf7a729ca69bfabd0995fc9b083e86a18215bd91'),
  723. }
  724. _SORTED_TREE_ITEMS_BUG_1325 = [
  725. TreeEntry(path=b'gamma', mode=0o100755, sha=b'5944b31ff85b415573d1a43eb942e2dea30ab8be'),
  726. TreeEntry(path=b'gamma-new', mode=0o100644, sha=b'cf7a729ca69bfabd0995fc9b083e86a18215bd91'),
  727. ]
  728. class TreeTests(ShaFileCheckTests):
  729. def test_add(self):
  730. myhexsha = b"d80c186a03f423a81b39df39dc87fd269736ca86"
  731. x = Tree()
  732. x.add(b"myname", 0o100755, myhexsha)
  733. self.assertEqual(x[b"myname"], (0o100755, myhexsha))
  734. self.assertEqual(b"100755 myname\0" + hex_to_sha(myhexsha), x.as_raw_string())
  735. def test_simple(self):
  736. myhexsha = b"d80c186a03f423a81b39df39dc87fd269736ca86"
  737. x = Tree()
  738. x[b"myname"] = (0o100755, myhexsha)
  739. self.assertEqual(b"100755 myname\0" + hex_to_sha(myhexsha), x.as_raw_string())
  740. self.assertEqual(b"100755 myname\0" + hex_to_sha(myhexsha), bytes(x))
  741. def test_tree_update_id(self):
  742. x = Tree()
  743. x[b"a.c"] = (0o100755, b"d80c186a03f423a81b39df39dc87fd269736ca86")
  744. self.assertEqual(b"0c5c6bc2c081accfbc250331b19e43b904ab9cdd", x.id)
  745. x[b"a.b"] = (stat.S_IFDIR, b"d80c186a03f423a81b39df39dc87fd269736ca86")
  746. self.assertEqual(b"07bfcb5f3ada15bbebdfa3bbb8fd858a363925c8", x.id)
  747. def test_tree_iteritems_dir_sort(self):
  748. x = Tree()
  749. for name, item in _TREE_ITEMS.items():
  750. x[name] = item
  751. self.assertEqual(_SORTED_TREE_ITEMS, x.items())
  752. def test_tree_items_dir_sort(self):
  753. x = Tree()
  754. for name, item in _TREE_ITEMS.items():
  755. x[name] = item
  756. self.assertEqual(_SORTED_TREE_ITEMS, x.items())
  757. def _do_test_parse_tree(self, parse_tree):
  758. dir = os.path.join(os.path.dirname(__file__), "..", "testdata", "trees")
  759. o = Tree.from_path(hex_to_filename(dir, tree_sha))
  760. self.assertEqual(
  761. [(b"a", 0o100644, a_sha), (b"b", 0o100644, b_sha)],
  762. list(parse_tree(o.as_raw_string())),
  763. )
  764. # test a broken tree that has a leading 0 on the file mode
  765. broken_tree = b"0100644 foo\0" + hex_to_sha(a_sha)
  766. def eval_parse_tree(*args, **kwargs):
  767. return list(parse_tree(*args, **kwargs))
  768. self.assertEqual([(b"foo", 0o100644, a_sha)], eval_parse_tree(broken_tree))
  769. self.assertRaises(
  770. ObjectFormatException, eval_parse_tree, broken_tree, strict=True
  771. )
  772. test_parse_tree = functest_builder(_do_test_parse_tree, _parse_tree_py)
  773. test_parse_tree_extension = ext_functest_builder(_do_test_parse_tree, _parse_tree_rs)
  774. def _do_test_sorted_tree_items(self, sorted_tree_items):
  775. def do_sort(entries, name_order):
  776. return list(sorted_tree_items(entries, name_order))
  777. actual = do_sort(_TREE_ITEMS, False)
  778. self.assertEqual(_SORTED_TREE_ITEMS, actual)
  779. self.assertIsInstance(actual[0], TreeEntry)
  780. actual = do_sort(_TREE_ITEMS_BUG_1325, True)
  781. self.assertEqual(_SORTED_TREE_ITEMS_BUG_1325, actual)
  782. self.assertIsInstance(actual[0], TreeEntry)
  783. # C/Python implementations may differ in specific error types, but
  784. # should all error on invalid inputs.
  785. # For example, the Rust implementation has stricter type checks, so may
  786. # raise TypeError where the Python implementation raises
  787. # AttributeError.
  788. errors = (TypeError, ValueError, AttributeError)
  789. self.assertRaises(errors, do_sort, b"foo", False)
  790. self.assertRaises(errors, do_sort, {b"foo": (1, 2, 3)}, False)
  791. myhexsha = b"d80c186a03f423a81b39df39dc87fd269736ca86"
  792. self.assertRaises(errors, do_sort, {b"foo": (b"xxx", myhexsha)}, False)
  793. self.assertRaises(errors, do_sort, {b"foo": (0o100755, 12345)}, False)
  794. test_sorted_tree_items = functest_builder(
  795. _do_test_sorted_tree_items, _sorted_tree_items_py
  796. )
  797. if _sorted_tree_items_rs is not None:
  798. test_sorted_tree_items_extension = ext_functest_builder(
  799. _do_test_sorted_tree_items, _sorted_tree_items_rs
  800. )
  801. def _do_test_sorted_tree_items_name_order(self, sorted_tree_items):
  802. self.assertEqual(
  803. [
  804. TreeEntry(
  805. b"a",
  806. stat.S_IFDIR,
  807. b"d80c186a03f423a81b39df39dc87fd269736ca86",
  808. ),
  809. TreeEntry(
  810. b"a.c",
  811. 0o100755,
  812. b"d80c186a03f423a81b39df39dc87fd269736ca86",
  813. ),
  814. TreeEntry(
  815. b"a/c",
  816. stat.S_IFDIR,
  817. b"d80c186a03f423a81b39df39dc87fd269736ca86",
  818. ),
  819. ],
  820. list(sorted_tree_items(_TREE_ITEMS, True)),
  821. )
  822. test_sorted_tree_items_name_order = functest_builder(
  823. _do_test_sorted_tree_items_name_order, _sorted_tree_items_py
  824. )
  825. if _sorted_tree_items_rs is not None:
  826. test_sorted_tree_items_name_order_extension = ext_functest_builder(
  827. _do_test_sorted_tree_items_name_order, _sorted_tree_items_rs
  828. )
  829. def test_check(self):
  830. t = Tree
  831. sha = hex_to_sha(a_sha)
  832. # filenames
  833. self.assertCheckSucceeds(t, b"100644 .a\0" + sha)
  834. self.assertCheckFails(t, b"100644 \0" + sha)
  835. self.assertCheckFails(t, b"100644 .\0" + sha)
  836. self.assertCheckFails(t, b"100644 a/a\0" + sha)
  837. self.assertCheckFails(t, b"100644 ..\0" + sha)
  838. self.assertCheckFails(t, b"100644 .git\0" + sha)
  839. # modes
  840. self.assertCheckSucceeds(t, b"100644 a\0" + sha)
  841. self.assertCheckSucceeds(t, b"100755 a\0" + sha)
  842. self.assertCheckSucceeds(t, b"160000 a\0" + sha)
  843. # TODO more whitelisted modes
  844. self.assertCheckFails(t, b"123456 a\0" + sha)
  845. self.assertCheckFails(t, b"123abc a\0" + sha)
  846. # should fail check, but parses ok
  847. self.assertCheckFails(t, b"0100644 foo\0" + sha)
  848. # shas
  849. self.assertCheckFails(t, b"100644 a\0" + (b"x" * 5))
  850. self.assertCheckFails(t, b"100644 a\0" + (b"x" * 18) + b"\0")
  851. self.assertCheckFails(t, b"100644 a\0" + (b"x" * 21) + b"\n100644 b\0" + sha)
  852. # ordering
  853. sha2 = hex_to_sha(b_sha)
  854. self.assertCheckSucceeds(t, b"100644 a\0" + sha + b"100644 b\0" + sha)
  855. self.assertCheckSucceeds(t, b"100644 a\0" + sha + b"100644 b\0" + sha2)
  856. self.assertCheckFails(t, b"100644 a\0" + sha + b"100755 a\0" + sha2)
  857. self.assertCheckFails(t, b"100644 b\0" + sha2 + b"100644 a\0" + sha)
  858. def test_iter(self):
  859. t = Tree()
  860. t[b"foo"] = (0o100644, a_sha)
  861. self.assertEqual({b"foo"}, set(t))
  862. class TagSerializeTests(TestCase):
  863. def test_serialize_simple(self):
  864. x = make_object(
  865. Tag,
  866. tagger=b"Jelmer Vernooij <jelmer@samba.org>",
  867. name=b"0.1",
  868. message=b"Tag 0.1",
  869. object=(Blob, b"d80c186a03f423a81b39df39dc87fd269736ca86"),
  870. tag_time=423423423,
  871. tag_timezone=0,
  872. )
  873. self.assertEqual(
  874. (
  875. b"object d80c186a03f423a81b39df39dc87fd269736ca86\n"
  876. b"type blob\n"
  877. b"tag 0.1\n"
  878. b"tagger Jelmer Vernooij <jelmer@samba.org> "
  879. b"423423423 +0000\n"
  880. b"\n"
  881. b"Tag 0.1"
  882. ),
  883. x.as_raw_string(),
  884. )
  885. def test_serialize_none_message(self):
  886. x = make_object(
  887. Tag,
  888. tagger=b"Jelmer Vernooij <jelmer@samba.org>",
  889. name=b"0.1",
  890. message=None,
  891. object=(Blob, b"d80c186a03f423a81b39df39dc87fd269736ca86"),
  892. tag_time=423423423,
  893. tag_timezone=0,
  894. )
  895. self.assertEqual(
  896. (
  897. b"object d80c186a03f423a81b39df39dc87fd269736ca86\n"
  898. b"type blob\n"
  899. b"tag 0.1\n"
  900. b"tagger Jelmer Vernooij <jelmer@samba.org> "
  901. b"423423423 +0000\n"
  902. ),
  903. x.as_raw_string(),
  904. )
  905. default_tagger = (
  906. b"Linus Torvalds <torvalds@woody.linux-foundation.org> " b"1183319674 -0700"
  907. )
  908. default_message = b"""Linux 2.6.22-rc7
  909. -----BEGIN PGP SIGNATURE-----
  910. Version: GnuPG v1.4.7 (GNU/Linux)
  911. iD8DBQBGiAaAF3YsRnbiHLsRAitMAKCiLboJkQECM/jpYsY3WPfvUgLXkACgg3ql
  912. OK2XeQOiEeXtT76rV4t2WR4=
  913. =ivrA
  914. -----END PGP SIGNATURE-----
  915. """
  916. class TagParseTests(ShaFileCheckTests):
  917. def make_tag_lines(
  918. self,
  919. object_sha=b"a38d6181ff27824c79fc7df825164a212eff6a3f",
  920. object_type_name=b"commit",
  921. name=b"v2.6.22-rc7",
  922. tagger=default_tagger,
  923. message=default_message,
  924. ):
  925. lines = []
  926. if object_sha is not None:
  927. lines.append(b"object " + object_sha)
  928. if object_type_name is not None:
  929. lines.append(b"type " + object_type_name)
  930. if name is not None:
  931. lines.append(b"tag " + name)
  932. if tagger is not None:
  933. lines.append(b"tagger " + tagger)
  934. if message is not None:
  935. lines.append(b"")
  936. lines.append(message)
  937. return lines
  938. def make_tag_text(self, **kwargs):
  939. return b"\n".join(self.make_tag_lines(**kwargs))
  940. def test_parse(self):
  941. x = Tag()
  942. x.set_raw_string(self.make_tag_text())
  943. self.assertEqual(
  944. b"Linus Torvalds <torvalds@woody.linux-foundation.org>", x.tagger
  945. )
  946. self.assertEqual(b"v2.6.22-rc7", x.name)
  947. object_type, object_sha = x.object
  948. self.assertEqual(b"a38d6181ff27824c79fc7df825164a212eff6a3f", object_sha)
  949. self.assertEqual(Commit, object_type)
  950. self.assertEqual(
  951. datetime.datetime.utcfromtimestamp(x.tag_time),
  952. datetime.datetime(2007, 7, 1, 19, 54, 34),
  953. )
  954. self.assertEqual(-25200, x.tag_timezone)
  955. def test_parse_no_tagger(self):
  956. x = Tag()
  957. x.set_raw_string(self.make_tag_text(tagger=None))
  958. self.assertEqual(None, x.tagger)
  959. self.assertEqual(b"v2.6.22-rc7", x.name)
  960. self.assertEqual(None, x.tag_time)
  961. def test_parse_no_message(self):
  962. x = Tag()
  963. x.set_raw_string(self.make_tag_text(message=None))
  964. self.assertEqual(None, x.message)
  965. self.assertEqual(
  966. b"Linus Torvalds <torvalds@woody.linux-foundation.org>", x.tagger
  967. )
  968. self.assertEqual(
  969. datetime.datetime.utcfromtimestamp(x.tag_time),
  970. datetime.datetime(2007, 7, 1, 19, 54, 34),
  971. )
  972. self.assertEqual(-25200, x.tag_timezone)
  973. self.assertEqual(b"v2.6.22-rc7", x.name)
  974. def test_check(self):
  975. self.assertCheckSucceeds(Tag, self.make_tag_text())
  976. self.assertCheckFails(Tag, self.make_tag_text(object_sha=None))
  977. self.assertCheckFails(Tag, self.make_tag_text(object_type_name=None))
  978. self.assertCheckFails(Tag, self.make_tag_text(name=None))
  979. self.assertCheckFails(Tag, self.make_tag_text(name=b""))
  980. self.assertCheckFails(Tag, self.make_tag_text(object_type_name=b"foobar"))
  981. self.assertCheckFails(
  982. Tag,
  983. self.make_tag_text(
  984. tagger=b"some guy without an email address 1183319674 -0700"
  985. ),
  986. )
  987. self.assertCheckFails(
  988. Tag,
  989. self.make_tag_text(
  990. tagger=(
  991. b"Linus Torvalds <torvalds@woody.linux-foundation.org> "
  992. b"Sun 7 Jul 2007 12:54:34 +0700"
  993. )
  994. ),
  995. )
  996. self.assertCheckFails(Tag, self.make_tag_text(object_sha=b"xxx"))
  997. def test_check_tag_with_unparseable_field(self):
  998. self.assertCheckFails(
  999. Tag,
  1000. self.make_tag_text(
  1001. tagger=(
  1002. b"Linus Torvalds <torvalds@woody.linux-foundation.org> "
  1003. b"423423+0000"
  1004. )
  1005. ),
  1006. )
  1007. def test_check_tag_with_overflow_time(self):
  1008. """Date with overflow should raise an ObjectFormatException when checked."""
  1009. author = f"Some Dude <some@dude.org> {MAX_TIME + 1} +0000"
  1010. tag = Tag.from_string(self.make_tag_text(tagger=(author.encode())))
  1011. with self.assertRaises(ObjectFormatException):
  1012. tag.check()
  1013. def test_check_duplicates(self):
  1014. # duplicate each of the header fields
  1015. for i in range(4):
  1016. lines = self.make_tag_lines()
  1017. lines.insert(i, lines[i])
  1018. self.assertCheckFails(Tag, b"\n".join(lines))
  1019. def test_check_order(self):
  1020. lines = self.make_tag_lines()
  1021. headers = lines[:4]
  1022. rest = lines[4:]
  1023. # of all possible permutations, ensure only the original succeeds
  1024. for perm in permutations(headers):
  1025. perm = list(perm)
  1026. text = b"\n".join(perm + rest)
  1027. if perm == headers:
  1028. self.assertCheckSucceeds(Tag, text)
  1029. else:
  1030. self.assertCheckFails(Tag, text)
  1031. def test_tree_copy_after_update(self):
  1032. """Check Tree.id is correctly updated when the tree is copied after updated."""
  1033. shas = []
  1034. tree = Tree()
  1035. shas.append(tree.id)
  1036. tree.add(b"data", 0o644, Blob().id)
  1037. copied = tree.copy()
  1038. shas.append(tree.id)
  1039. shas.append(copied.id)
  1040. self.assertNotIn(shas[0], shas[1:])
  1041. self.assertEqual(shas[1], shas[2])
  1042. class CheckTests(TestCase):
  1043. def test_check_hexsha(self):
  1044. check_hexsha(a_sha, "failed to check good sha")
  1045. self.assertRaises(
  1046. ObjectFormatException, check_hexsha, b"1" * 39, "sha too short"
  1047. )
  1048. self.assertRaises(
  1049. ObjectFormatException, check_hexsha, b"1" * 41, "sha too long"
  1050. )
  1051. self.assertRaises(
  1052. ObjectFormatException,
  1053. check_hexsha,
  1054. b"x" * 40,
  1055. "invalid characters",
  1056. )
  1057. def test_check_identity(self):
  1058. check_identity(
  1059. b"Dave Borowitz <dborowitz@google.com>",
  1060. "failed to check good identity",
  1061. )
  1062. check_identity(b" <dborowitz@google.com>", "failed to check good identity")
  1063. self.assertRaises(
  1064. ObjectFormatException,
  1065. check_identity,
  1066. b"<dborowitz@google.com>",
  1067. "no space before email",
  1068. )
  1069. self.assertRaises(
  1070. ObjectFormatException, check_identity, b"Dave Borowitz", "no email"
  1071. )
  1072. self.assertRaises(
  1073. ObjectFormatException,
  1074. check_identity,
  1075. b"Dave Borowitz <dborowitz",
  1076. "incomplete email",
  1077. )
  1078. self.assertRaises(
  1079. ObjectFormatException,
  1080. check_identity,
  1081. b"dborowitz@google.com>",
  1082. "incomplete email",
  1083. )
  1084. self.assertRaises(
  1085. ObjectFormatException,
  1086. check_identity,
  1087. b"Dave Borowitz <<dborowitz@google.com>",
  1088. "typo",
  1089. )
  1090. self.assertRaises(
  1091. ObjectFormatException,
  1092. check_identity,
  1093. b"Dave Borowitz <dborowitz@google.com>>",
  1094. "typo",
  1095. )
  1096. self.assertRaises(
  1097. ObjectFormatException,
  1098. check_identity,
  1099. b"Dave Borowitz <dborowitz@google.com>xxx",
  1100. "trailing characters",
  1101. )
  1102. self.assertRaises(
  1103. ObjectFormatException,
  1104. check_identity,
  1105. b"Dave Borowitz <dborowitz@google.com>xxx",
  1106. "trailing characters",
  1107. )
  1108. self.assertRaises(
  1109. ObjectFormatException,
  1110. check_identity,
  1111. b"Dave<Borowitz <dborowitz@google.com>",
  1112. "reserved byte in name",
  1113. )
  1114. self.assertRaises(
  1115. ObjectFormatException,
  1116. check_identity,
  1117. b"Dave>Borowitz <dborowitz@google.com>",
  1118. "reserved byte in name",
  1119. )
  1120. self.assertRaises(
  1121. ObjectFormatException,
  1122. check_identity,
  1123. b"Dave\0Borowitz <dborowitz@google.com>",
  1124. "null byte",
  1125. )
  1126. self.assertRaises(
  1127. ObjectFormatException,
  1128. check_identity,
  1129. b"Dave\nBorowitz <dborowitz@google.com>",
  1130. "newline byte",
  1131. )
  1132. class TimezoneTests(TestCase):
  1133. def test_parse_timezone_utc(self):
  1134. self.assertEqual((0, False), parse_timezone(b"+0000"))
  1135. def test_parse_timezone_utc_negative(self):
  1136. self.assertEqual((0, True), parse_timezone(b"-0000"))
  1137. def test_generate_timezone_utc(self):
  1138. self.assertEqual(b"+0000", format_timezone(0))
  1139. def test_generate_timezone_utc_negative(self):
  1140. self.assertEqual(b"-0000", format_timezone(0, True))
  1141. def test_parse_timezone_cet(self):
  1142. self.assertEqual((60 * 60, False), parse_timezone(b"+0100"))
  1143. def test_format_timezone_cet(self):
  1144. self.assertEqual(b"+0100", format_timezone(60 * 60))
  1145. def test_format_timezone_pdt(self):
  1146. self.assertEqual(b"-0400", format_timezone(-4 * 60 * 60))
  1147. def test_parse_timezone_pdt(self):
  1148. self.assertEqual((-4 * 60 * 60, False), parse_timezone(b"-0400"))
  1149. def test_format_timezone_pdt_half(self):
  1150. self.assertEqual(b"-0440", format_timezone(int(((-4 * 60) - 40) * 60)))
  1151. def test_format_timezone_double_negative(self):
  1152. self.assertEqual(b"--700", format_timezone(int((7 * 60) * 60), True))
  1153. def test_parse_timezone_pdt_half(self):
  1154. self.assertEqual((((-4 * 60) - 40) * 60, False), parse_timezone(b"-0440"))
  1155. def test_parse_timezone_double_negative(self):
  1156. self.assertEqual((int((7 * 60) * 60), False), parse_timezone(b"+700"))
  1157. self.assertEqual((int((7 * 60) * 60), True), parse_timezone(b"--700"))
  1158. class ShaFileCopyTests(TestCase):
  1159. def assert_copy(self, orig):
  1160. oclass = object_class(orig.type_num)
  1161. copy = orig.copy()
  1162. self.assertIsInstance(copy, oclass)
  1163. self.assertEqual(copy, orig)
  1164. self.assertIsNot(copy, orig)
  1165. def test_commit_copy(self):
  1166. attrs = {
  1167. "tree": b"d80c186a03f423a81b39df39dc87fd269736ca86",
  1168. "parents": [
  1169. b"ab64bbdcc51b170d21588e5c5d391ee5c0c96dfd",
  1170. b"4cffe90e0a41ad3f5190079d7c8f036bde29cbe6",
  1171. ],
  1172. "author": b"James Westby <jw+debian@jameswestby.net>",
  1173. "committer": b"James Westby <jw+debian@jameswestby.net>",
  1174. "commit_time": 1174773719,
  1175. "author_time": 1174773719,
  1176. "commit_timezone": 0,
  1177. "author_timezone": 0,
  1178. "message": b"Merge ../b\n",
  1179. }
  1180. commit = make_commit(**attrs)
  1181. self.assert_copy(commit)
  1182. def test_blob_copy(self):
  1183. blob = make_object(Blob, data=b"i am a blob")
  1184. self.assert_copy(blob)
  1185. def test_tree_copy(self):
  1186. blob = make_object(Blob, data=b"i am a blob")
  1187. tree = Tree()
  1188. tree[b"blob"] = (stat.S_IFREG, blob.id)
  1189. self.assert_copy(tree)
  1190. def test_tag_copy(self):
  1191. tag = make_object(
  1192. Tag,
  1193. name=b"tag",
  1194. message=b"",
  1195. tagger=b"Tagger <test@example.com>",
  1196. tag_time=12345,
  1197. tag_timezone=0,
  1198. object=(Commit, b"0" * 40),
  1199. )
  1200. self.assert_copy(tag)
  1201. class ShaFileSerializeTests(TestCase):
  1202. """`ShaFile` objects only gets serialized once if they haven't changed."""
  1203. @contextmanager
  1204. def assert_serialization_on_change(
  1205. self, obj, needs_serialization_after_change=True
  1206. ):
  1207. old_id = obj.id
  1208. self.assertFalse(obj._needs_serialization)
  1209. yield obj
  1210. if needs_serialization_after_change:
  1211. self.assertTrue(obj._needs_serialization)
  1212. else:
  1213. self.assertFalse(obj._needs_serialization)
  1214. new_id = obj.id
  1215. self.assertFalse(obj._needs_serialization)
  1216. self.assertNotEqual(old_id, new_id)
  1217. def test_commit_serialize(self):
  1218. attrs = {
  1219. "tree": b"d80c186a03f423a81b39df39dc87fd269736ca86",
  1220. "parents": [
  1221. b"ab64bbdcc51b170d21588e5c5d391ee5c0c96dfd",
  1222. b"4cffe90e0a41ad3f5190079d7c8f036bde29cbe6",
  1223. ],
  1224. "author": b"James Westby <jw+debian@jameswestby.net>",
  1225. "committer": b"James Westby <jw+debian@jameswestby.net>",
  1226. "commit_time": 1174773719,
  1227. "author_time": 1174773719,
  1228. "commit_timezone": 0,
  1229. "author_timezone": 0,
  1230. "message": b"Merge ../b\n",
  1231. }
  1232. commit = make_commit(**attrs)
  1233. with self.assert_serialization_on_change(commit):
  1234. commit.parents = [b"ab64bbdcc51b170d21588e5c5d391ee5c0c96dfd"]
  1235. def test_blob_serialize(self):
  1236. blob = make_object(Blob, data=b"i am a blob")
  1237. with self.assert_serialization_on_change(
  1238. blob, needs_serialization_after_change=False
  1239. ):
  1240. blob.data = b"i am another blob"
  1241. def test_tree_serialize(self):
  1242. blob = make_object(Blob, data=b"i am a blob")
  1243. tree = Tree()
  1244. tree[b"blob"] = (stat.S_IFREG, blob.id)
  1245. with self.assert_serialization_on_change(tree):
  1246. tree[b"blob2"] = (stat.S_IFREG, blob.id)
  1247. def test_tag_serialize(self):
  1248. tag = make_object(
  1249. Tag,
  1250. name=b"tag",
  1251. message=b"",
  1252. tagger=b"Tagger <test@example.com>",
  1253. tag_time=12345,
  1254. tag_timezone=0,
  1255. object=(Commit, b"0" * 40),
  1256. )
  1257. with self.assert_serialization_on_change(tag):
  1258. tag.message = b"new message"
  1259. def test_tag_serialize_time_error(self):
  1260. with self.assertRaises(ObjectFormatException):
  1261. tag = make_object(
  1262. Tag,
  1263. name=b"tag",
  1264. message=b"some message",
  1265. tagger=b"Tagger <test@example.com> 1174773719+0000",
  1266. object=(Commit, b"0" * 40),
  1267. )
  1268. tag._deserialize(tag._serialize())
  1269. class PrettyFormatTreeEntryTests(TestCase):
  1270. def test_format(self):
  1271. self.assertEqual(
  1272. "40000 tree 40820c38cfb182ce6c8b261555410d8382a5918b\tfoo\n",
  1273. pretty_format_tree_entry(
  1274. b"foo", 0o40000, b"40820c38cfb182ce6c8b261555410d8382a5918b"
  1275. ),
  1276. )