test_pack.py 46 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300
  1. # test_pack.py -- Tests for the handling of git packs.
  2. # Copyright (C) 2007 James Westby <jw+debian@jameswestby.net>
  3. # Copyright (C) 2008 Jelmer Vernooij <jelmer@jelmer.uk>
  4. #
  5. # Dulwich is dual-licensed under the Apache License, Version 2.0 and the GNU
  6. # General Public License as public 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 Dulwich packs."""
  22. import os
  23. import shutil
  24. import sys
  25. import tempfile
  26. import zlib
  27. from hashlib import sha1
  28. from io import BytesIO
  29. from typing import Set
  30. from dulwich.errors import ApplyDeltaError, ChecksumMismatch
  31. from dulwich.file import GitFile
  32. from dulwich.object_store import MemoryObjectStore
  33. from dulwich.objects import Blob, Commit, Tree, hex_to_sha, sha_to_hex
  34. from dulwich.pack import (
  35. OFS_DELTA,
  36. REF_DELTA,
  37. DeltaChainIterator,
  38. MemoryPackIndex,
  39. Pack,
  40. PackData,
  41. PackStreamReader,
  42. UnpackedObject,
  43. UnresolvedDeltas,
  44. _delta_encode_size,
  45. _encode_copy_operation,
  46. apply_delta,
  47. compute_file_sha,
  48. create_delta,
  49. deltify_pack_objects,
  50. load_pack_index,
  51. read_zlib_chunks,
  52. unpack_object,
  53. write_pack,
  54. write_pack_header,
  55. write_pack_index_v1,
  56. write_pack_index_v2,
  57. write_pack_object,
  58. )
  59. from dulwich.tests.utils import build_pack, make_object
  60. from . import TestCase
  61. pack1_sha = b"bc63ddad95e7321ee734ea11a7a62d314e0d7481"
  62. a_sha = b"6f670c0fb53f9463760b7295fbb814e965fb20c8"
  63. tree_sha = b"b2a2766a2879c209ab1176e7e778b81ae422eeaa"
  64. commit_sha = b"f18faa16531ac570a3fdc8c7ca16682548dafd12"
  65. indexmode = "0o100644" if sys.platform != "win32" else "0o100666"
  66. class PackTests(TestCase):
  67. """Base class for testing packs."""
  68. def setUp(self):
  69. super().setUp()
  70. self.tempdir = tempfile.mkdtemp()
  71. self.addCleanup(shutil.rmtree, self.tempdir)
  72. datadir = os.path.abspath(
  73. os.path.join(os.path.dirname(__file__), "../testdata/packs")
  74. )
  75. def get_pack_index(self, sha):
  76. """Returns a PackIndex from the datadir with the given sha."""
  77. return load_pack_index(
  78. os.path.join(self.datadir, "pack-{}.idx".format(sha.decode("ascii")))
  79. )
  80. def get_pack_data(self, sha):
  81. """Returns a PackData object from the datadir with the given sha."""
  82. return PackData(
  83. os.path.join(self.datadir, "pack-{}.pack".format(sha.decode("ascii")))
  84. )
  85. def get_pack(self, sha):
  86. return Pack(os.path.join(self.datadir, "pack-{}".format(sha.decode("ascii"))))
  87. def assertSucceeds(self, func, *args, **kwargs):
  88. try:
  89. func(*args, **kwargs)
  90. except ChecksumMismatch as e:
  91. self.fail(e)
  92. class PackIndexTests(PackTests):
  93. """Class that tests the index of packfiles."""
  94. def test_object_offset(self):
  95. """Tests that the correct object offset is returned from the index."""
  96. p = self.get_pack_index(pack1_sha)
  97. self.assertRaises(KeyError, p.object_offset, pack1_sha)
  98. self.assertEqual(p.object_offset(a_sha), 178)
  99. self.assertEqual(p.object_offset(tree_sha), 138)
  100. self.assertEqual(p.object_offset(commit_sha), 12)
  101. def test_object_sha1(self):
  102. """Tests that the correct object offset is returned from the index."""
  103. p = self.get_pack_index(pack1_sha)
  104. self.assertRaises(KeyError, p.object_sha1, 876)
  105. self.assertEqual(p.object_sha1(178), hex_to_sha(a_sha))
  106. self.assertEqual(p.object_sha1(138), hex_to_sha(tree_sha))
  107. self.assertEqual(p.object_sha1(12), hex_to_sha(commit_sha))
  108. def test_index_len(self):
  109. p = self.get_pack_index(pack1_sha)
  110. self.assertEqual(3, len(p))
  111. def test_get_stored_checksum(self):
  112. p = self.get_pack_index(pack1_sha)
  113. self.assertEqual(
  114. b"f2848e2ad16f329ae1c92e3b95e91888daa5bd01",
  115. sha_to_hex(p.get_stored_checksum()),
  116. )
  117. self.assertEqual(
  118. b"721980e866af9a5f93ad674144e1459b8ba3e7b7",
  119. sha_to_hex(p.get_pack_checksum()),
  120. )
  121. def test_index_check(self):
  122. p = self.get_pack_index(pack1_sha)
  123. self.assertSucceeds(p.check)
  124. def test_iterentries(self):
  125. p = self.get_pack_index(pack1_sha)
  126. entries = [(sha_to_hex(s), o, c) for s, o, c in p.iterentries()]
  127. self.assertEqual(
  128. [
  129. (b"6f670c0fb53f9463760b7295fbb814e965fb20c8", 178, None),
  130. (b"b2a2766a2879c209ab1176e7e778b81ae422eeaa", 138, None),
  131. (b"f18faa16531ac570a3fdc8c7ca16682548dafd12", 12, None),
  132. ],
  133. entries,
  134. )
  135. def test_iter(self):
  136. p = self.get_pack_index(pack1_sha)
  137. self.assertEqual({tree_sha, commit_sha, a_sha}, set(p))
  138. class TestPackDeltas(TestCase):
  139. test_string1 = b"The answer was flailing in the wind"
  140. test_string2 = b"The answer was falling down the pipe"
  141. test_string3 = b"zzzzz"
  142. test_string_empty = b""
  143. test_string_big = b"Z" * 8192
  144. test_string_huge = b"Z" * 100000
  145. def _test_roundtrip(self, base, target):
  146. self.assertEqual(
  147. target, b"".join(apply_delta(base, list(create_delta(base, target))))
  148. )
  149. def test_nochange(self):
  150. self._test_roundtrip(self.test_string1, self.test_string1)
  151. def test_nochange_huge(self):
  152. self._test_roundtrip(self.test_string_huge, self.test_string_huge)
  153. def test_change(self):
  154. self._test_roundtrip(self.test_string1, self.test_string2)
  155. def test_rewrite(self):
  156. self._test_roundtrip(self.test_string1, self.test_string3)
  157. def test_empty_to_big(self):
  158. self._test_roundtrip(self.test_string_empty, self.test_string_big)
  159. def test_empty_to_huge(self):
  160. self._test_roundtrip(self.test_string_empty, self.test_string_huge)
  161. def test_huge_copy(self):
  162. self._test_roundtrip(
  163. self.test_string_huge + self.test_string1,
  164. self.test_string_huge + self.test_string2,
  165. )
  166. def test_dest_overflow(self):
  167. self.assertRaises(
  168. ApplyDeltaError,
  169. apply_delta,
  170. b"a" * 0x10000,
  171. b"\x80\x80\x04\x80\x80\x04\x80" + b"a" * 0x10000,
  172. )
  173. self.assertRaises(
  174. ApplyDeltaError, apply_delta, b"", b"\x00\x80\x02\xb0\x11\x11"
  175. )
  176. def test_pypy_issue(self):
  177. # Test for https://github.com/jelmer/dulwich/issues/509 /
  178. # https://bitbucket.org/pypy/pypy/issues/2499/cpyext-pystring_asstring-doesnt-work
  179. chunks = [
  180. b"tree 03207ccf58880a748188836155ceed72f03d65d6\n"
  181. b"parent 408fbab530fd4abe49249a636a10f10f44d07a21\n"
  182. b"author Victor Stinner <victor.stinner@gmail.com> "
  183. b"1421355207 +0100\n"
  184. b"committer Victor Stinner <victor.stinner@gmail.com> "
  185. b"1421355207 +0100\n"
  186. b"\n"
  187. b"Backout changeset 3a06020af8cf\n"
  188. b"\nStreamWriter: close() now clears the reference to the "
  189. b"transport\n"
  190. b"\nStreamWriter now raises an exception if it is closed: "
  191. b"write(), writelines(),\n"
  192. b"write_eof(), can_write_eof(), get_extra_info(), drain().\n"
  193. ]
  194. delta = [
  195. b"\xcd\x03\xad\x03]tree ff3c181a393d5a7270cddc01ea863818a8621ca8\n"
  196. b"parent 20a103cc90135494162e819f98d0edfc1f1fba6b\x91]7\x0510738"
  197. b"\x91\x99@\x0b10738 +0100\x93\x04\x01\xc9"
  198. ]
  199. res = apply_delta(chunks, delta)
  200. expected = [
  201. b"tree ff3c181a393d5a7270cddc01ea863818a8621ca8\n"
  202. b"parent 20a103cc90135494162e819f98d0edfc1f1fba6b",
  203. b"\nauthor Victor Stinner <victor.stinner@gmail.com> 14213",
  204. b"10738",
  205. b" +0100\ncommitter Victor Stinner <victor.stinner@gmail.com> " b"14213",
  206. b"10738 +0100",
  207. b"\n\nStreamWriter: close() now clears the reference to the "
  208. b"transport\n\n"
  209. b"StreamWriter now raises an exception if it is closed: "
  210. b"write(), writelines(),\n"
  211. b"write_eof(), can_write_eof(), get_extra_info(), drain().\n",
  212. ]
  213. self.assertEqual(b"".join(expected), b"".join(res))
  214. class TestPackData(PackTests):
  215. """Tests getting the data from the packfile."""
  216. def test_create_pack(self):
  217. self.get_pack_data(pack1_sha).close()
  218. def test_from_file(self):
  219. path = os.path.join(self.datadir, "pack-{}.pack".format(pack1_sha.decode("ascii")))
  220. with open(path, "rb") as f:
  221. PackData.from_file(f, os.path.getsize(path))
  222. def test_pack_len(self):
  223. with self.get_pack_data(pack1_sha) as p:
  224. self.assertEqual(3, len(p))
  225. def test_index_check(self):
  226. with self.get_pack_data(pack1_sha) as p:
  227. self.assertSucceeds(p.check)
  228. def test_iter_unpacked(self):
  229. with self.get_pack_data(pack1_sha) as p:
  230. commit_data = (
  231. b"tree b2a2766a2879c209ab1176e7e778b81ae422eeaa\n"
  232. b"author James Westby <jw+debian@jameswestby.net> "
  233. b"1174945067 +0100\n"
  234. b"committer James Westby <jw+debian@jameswestby.net> "
  235. b"1174945067 +0100\n"
  236. b"\n"
  237. b"Test commit\n"
  238. )
  239. blob_sha = b"6f670c0fb53f9463760b7295fbb814e965fb20c8"
  240. tree_data = b"100644 a\0" + hex_to_sha(blob_sha)
  241. actual = list(p.iter_unpacked())
  242. self.assertEqual(
  243. [
  244. UnpackedObject(
  245. offset=12,
  246. pack_type_num=1,
  247. decomp_chunks=[commit_data],
  248. crc32=None,
  249. ),
  250. UnpackedObject(
  251. offset=138,
  252. pack_type_num=2,
  253. decomp_chunks=[tree_data],
  254. crc32=None,
  255. ),
  256. UnpackedObject(
  257. offset=178,
  258. pack_type_num=3,
  259. decomp_chunks=[b"test 1\n"],
  260. crc32=None,
  261. ),
  262. ],
  263. actual,
  264. )
  265. def test_iterentries(self):
  266. with self.get_pack_data(pack1_sha) as p:
  267. entries = {(sha_to_hex(s), o, c) for s, o, c in p.iterentries()}
  268. self.assertEqual(
  269. {
  270. (
  271. b"6f670c0fb53f9463760b7295fbb814e965fb20c8",
  272. 178,
  273. 1373561701,
  274. ),
  275. (
  276. b"b2a2766a2879c209ab1176e7e778b81ae422eeaa",
  277. 138,
  278. 912998690,
  279. ),
  280. (
  281. b"f18faa16531ac570a3fdc8c7ca16682548dafd12",
  282. 12,
  283. 3775879613,
  284. ),
  285. },
  286. entries,
  287. )
  288. def test_create_index_v1(self):
  289. with self.get_pack_data(pack1_sha) as p:
  290. filename = os.path.join(self.tempdir, "v1test.idx")
  291. p.create_index_v1(filename)
  292. idx1 = load_pack_index(filename)
  293. idx2 = self.get_pack_index(pack1_sha)
  294. self.assertEqual(oct(os.stat(filename).st_mode), indexmode)
  295. self.assertEqual(idx1, idx2)
  296. def test_create_index_v2(self):
  297. with self.get_pack_data(pack1_sha) as p:
  298. filename = os.path.join(self.tempdir, "v2test.idx")
  299. p.create_index_v2(filename)
  300. idx1 = load_pack_index(filename)
  301. idx2 = self.get_pack_index(pack1_sha)
  302. self.assertEqual(oct(os.stat(filename).st_mode), indexmode)
  303. self.assertEqual(idx1, idx2)
  304. def test_compute_file_sha(self):
  305. f = BytesIO(b"abcd1234wxyz")
  306. self.assertEqual(
  307. sha1(b"abcd1234wxyz").hexdigest(), compute_file_sha(f).hexdigest()
  308. )
  309. self.assertEqual(
  310. sha1(b"abcd1234wxyz").hexdigest(),
  311. compute_file_sha(f, buffer_size=5).hexdigest(),
  312. )
  313. self.assertEqual(
  314. sha1(b"abcd1234").hexdigest(),
  315. compute_file_sha(f, end_ofs=-4).hexdigest(),
  316. )
  317. self.assertEqual(
  318. sha1(b"1234wxyz").hexdigest(),
  319. compute_file_sha(f, start_ofs=4).hexdigest(),
  320. )
  321. self.assertEqual(
  322. sha1(b"1234").hexdigest(),
  323. compute_file_sha(f, start_ofs=4, end_ofs=-4).hexdigest(),
  324. )
  325. def test_compute_file_sha_short_file(self):
  326. f = BytesIO(b"abcd1234wxyz")
  327. self.assertRaises(AssertionError, compute_file_sha, f, end_ofs=-20)
  328. self.assertRaises(AssertionError, compute_file_sha, f, end_ofs=20)
  329. self.assertRaises(
  330. AssertionError, compute_file_sha, f, start_ofs=10, end_ofs=-12
  331. )
  332. class TestPack(PackTests):
  333. def test_len(self):
  334. with self.get_pack(pack1_sha) as p:
  335. self.assertEqual(3, len(p))
  336. def test_contains(self):
  337. with self.get_pack(pack1_sha) as p:
  338. self.assertIn(tree_sha, p)
  339. def test_get(self):
  340. with self.get_pack(pack1_sha) as p:
  341. self.assertEqual(type(p[tree_sha]), Tree)
  342. def test_iter(self):
  343. with self.get_pack(pack1_sha) as p:
  344. self.assertEqual({tree_sha, commit_sha, a_sha}, set(p))
  345. def test_iterobjects(self):
  346. with self.get_pack(pack1_sha) as p:
  347. expected = {p[s] for s in [commit_sha, tree_sha, a_sha]}
  348. self.assertEqual(expected, set(list(p.iterobjects())))
  349. def test_pack_tuples(self):
  350. with self.get_pack(pack1_sha) as p:
  351. tuples = p.pack_tuples()
  352. expected = {(p[s], None) for s in [commit_sha, tree_sha, a_sha]}
  353. self.assertEqual(expected, set(list(tuples)))
  354. self.assertEqual(expected, set(list(tuples)))
  355. self.assertEqual(3, len(tuples))
  356. def test_get_object_at(self):
  357. """Tests random access for non-delta objects."""
  358. with self.get_pack(pack1_sha) as p:
  359. obj = p[a_sha]
  360. self.assertEqual(obj.type_name, b"blob")
  361. self.assertEqual(obj.sha().hexdigest().encode("ascii"), a_sha)
  362. obj = p[tree_sha]
  363. self.assertEqual(obj.type_name, b"tree")
  364. self.assertEqual(obj.sha().hexdigest().encode("ascii"), tree_sha)
  365. obj = p[commit_sha]
  366. self.assertEqual(obj.type_name, b"commit")
  367. self.assertEqual(obj.sha().hexdigest().encode("ascii"), commit_sha)
  368. def test_copy(self):
  369. with self.get_pack(pack1_sha) as origpack:
  370. self.assertSucceeds(origpack.index.check)
  371. basename = os.path.join(self.tempdir, "Elch")
  372. write_pack(basename, origpack.pack_tuples())
  373. with Pack(basename) as newpack:
  374. self.assertEqual(origpack, newpack)
  375. self.assertSucceeds(newpack.index.check)
  376. self.assertEqual(origpack.name(), newpack.name())
  377. self.assertEqual(
  378. origpack.index.get_pack_checksum(),
  379. newpack.index.get_pack_checksum(),
  380. )
  381. wrong_version = origpack.index.version != newpack.index.version
  382. orig_checksum = origpack.index.get_stored_checksum()
  383. new_checksum = newpack.index.get_stored_checksum()
  384. self.assertTrue(wrong_version or orig_checksum == new_checksum)
  385. def test_commit_obj(self):
  386. with self.get_pack(pack1_sha) as p:
  387. commit = p[commit_sha]
  388. self.assertEqual(b"James Westby <jw+debian@jameswestby.net>", commit.author)
  389. self.assertEqual([], commit.parents)
  390. def _copy_pack(self, origpack):
  391. basename = os.path.join(self.tempdir, "somepack")
  392. write_pack(basename, origpack.pack_tuples())
  393. return Pack(basename)
  394. def test_keep_no_message(self):
  395. with self.get_pack(pack1_sha) as p:
  396. p = self._copy_pack(p)
  397. with p:
  398. keepfile_name = p.keep()
  399. # file should exist
  400. self.assertTrue(os.path.exists(keepfile_name))
  401. with open(keepfile_name) as f:
  402. buf = f.read()
  403. self.assertEqual("", buf)
  404. def test_keep_message(self):
  405. with self.get_pack(pack1_sha) as p:
  406. p = self._copy_pack(p)
  407. msg = b"some message"
  408. with p:
  409. keepfile_name = p.keep(msg)
  410. # file should exist
  411. self.assertTrue(os.path.exists(keepfile_name))
  412. # and contain the right message, with a linefeed
  413. with open(keepfile_name, "rb") as f:
  414. buf = f.read()
  415. self.assertEqual(msg + b"\n", buf)
  416. def test_name(self):
  417. with self.get_pack(pack1_sha) as p:
  418. self.assertEqual(pack1_sha, p.name())
  419. def test_length_mismatch(self):
  420. with self.get_pack_data(pack1_sha) as data:
  421. index = self.get_pack_index(pack1_sha)
  422. Pack.from_objects(data, index).check_length_and_checksum()
  423. data._file.seek(12)
  424. bad_file = BytesIO()
  425. write_pack_header(bad_file.write, 9999)
  426. bad_file.write(data._file.read())
  427. bad_file = BytesIO(bad_file.getvalue())
  428. bad_data = PackData("", file=bad_file)
  429. bad_pack = Pack.from_lazy_objects(lambda: bad_data, lambda: index)
  430. self.assertRaises(AssertionError, lambda: bad_pack.data)
  431. self.assertRaises(AssertionError, bad_pack.check_length_and_checksum)
  432. def test_checksum_mismatch(self):
  433. with self.get_pack_data(pack1_sha) as data:
  434. index = self.get_pack_index(pack1_sha)
  435. Pack.from_objects(data, index).check_length_and_checksum()
  436. data._file.seek(0)
  437. bad_file = BytesIO(data._file.read()[:-20] + (b"\xff" * 20))
  438. bad_data = PackData("", file=bad_file)
  439. bad_pack = Pack.from_lazy_objects(lambda: bad_data, lambda: index)
  440. self.assertRaises(ChecksumMismatch, lambda: bad_pack.data)
  441. self.assertRaises(ChecksumMismatch, bad_pack.check_length_and_checksum)
  442. def test_iterobjects_2(self):
  443. with self.get_pack(pack1_sha) as p:
  444. objs = {o.id: o for o in p.iterobjects()}
  445. self.assertEqual(3, len(objs))
  446. self.assertEqual(sorted(objs), sorted(p.index))
  447. self.assertIsInstance(objs[a_sha], Blob)
  448. self.assertIsInstance(objs[tree_sha], Tree)
  449. self.assertIsInstance(objs[commit_sha], Commit)
  450. def test_iterobjects_subset(self):
  451. with self.get_pack(pack1_sha) as p:
  452. objs = {o.id: o for o in p.iterobjects_subset([commit_sha])}
  453. self.assertEqual(1, len(objs))
  454. self.assertIsInstance(objs[commit_sha], Commit)
  455. class TestThinPack(PackTests):
  456. def setUp(self):
  457. super().setUp()
  458. self.store = MemoryObjectStore()
  459. self.blobs = {}
  460. for blob in (b"foo", b"bar", b"foo1234", b"bar2468"):
  461. self.blobs[blob] = make_object(Blob, data=blob)
  462. self.store.add_object(self.blobs[b"foo"])
  463. self.store.add_object(self.blobs[b"bar"])
  464. # Build a thin pack. 'foo' is as an external reference, 'bar' an
  465. # internal reference.
  466. self.pack_dir = tempfile.mkdtemp()
  467. self.addCleanup(shutil.rmtree, self.pack_dir)
  468. self.pack_prefix = os.path.join(self.pack_dir, "pack")
  469. with open(self.pack_prefix + ".pack", "wb") as f:
  470. build_pack(
  471. f,
  472. [
  473. (REF_DELTA, (self.blobs[b"foo"].id, b"foo1234")),
  474. (Blob.type_num, b"bar"),
  475. (REF_DELTA, (self.blobs[b"bar"].id, b"bar2468")),
  476. ],
  477. store=self.store,
  478. )
  479. # Index the new pack.
  480. with self.make_pack(True) as pack:
  481. with PackData(pack._data_path) as data:
  482. data.create_index(
  483. self.pack_prefix + ".idx", resolve_ext_ref=pack.resolve_ext_ref
  484. )
  485. del self.store[self.blobs[b"bar"].id]
  486. def make_pack(self, resolve_ext_ref):
  487. return Pack(
  488. self.pack_prefix,
  489. resolve_ext_ref=self.store.get_raw if resolve_ext_ref else None,
  490. )
  491. def test_get_raw(self):
  492. with self.make_pack(False) as p:
  493. self.assertRaises(KeyError, p.get_raw, self.blobs[b"foo1234"].id)
  494. with self.make_pack(True) as p:
  495. self.assertEqual((3, b"foo1234"), p.get_raw(self.blobs[b"foo1234"].id))
  496. def test_get_unpacked_object(self):
  497. self.maxDiff = None
  498. with self.make_pack(False) as p:
  499. expected = UnpackedObject(
  500. 7,
  501. delta_base=b"\x19\x10(\x15f=#\xf8\xb7ZG\xe7\xa0\x19e\xdc\xdc\x96F\x8c",
  502. decomp_chunks=[b"\x03\x07\x90\x03\x041234"],
  503. )
  504. expected.offset = 12
  505. got = p.get_unpacked_object(self.blobs[b"foo1234"].id)
  506. self.assertEqual(expected, got)
  507. with self.make_pack(True) as p:
  508. expected = UnpackedObject(
  509. 7,
  510. delta_base=b"\x19\x10(\x15f=#\xf8\xb7ZG\xe7\xa0\x19e\xdc\xdc\x96F\x8c",
  511. decomp_chunks=[b"\x03\x07\x90\x03\x041234"],
  512. )
  513. expected.offset = 12
  514. got = p.get_unpacked_object(self.blobs[b"foo1234"].id)
  515. self.assertEqual(
  516. expected,
  517. got,
  518. )
  519. def test_iterobjects(self):
  520. with self.make_pack(False) as p:
  521. self.assertRaises(UnresolvedDeltas, list, p.iterobjects())
  522. with self.make_pack(True) as p:
  523. self.assertEqual(
  524. sorted(
  525. [
  526. self.blobs[b"foo1234"].id,
  527. self.blobs[b"bar"].id,
  528. self.blobs[b"bar2468"].id,
  529. ]
  530. ),
  531. sorted(o.id for o in p.iterobjects()),
  532. )
  533. class WritePackTests(TestCase):
  534. def test_write_pack_header(self):
  535. f = BytesIO()
  536. write_pack_header(f.write, 42)
  537. self.assertEqual(b"PACK\x00\x00\x00\x02\x00\x00\x00*", f.getvalue())
  538. def test_write_pack_object(self):
  539. f = BytesIO()
  540. f.write(b"header")
  541. offset = f.tell()
  542. crc32 = write_pack_object(f.write, Blob.type_num, b"blob")
  543. self.assertEqual(crc32, zlib.crc32(f.getvalue()[6:]) & 0xFFFFFFFF)
  544. f.write(b"x") # unpack_object needs extra trailing data.
  545. f.seek(offset)
  546. unpacked, unused = unpack_object(f.read, compute_crc32=True)
  547. self.assertEqual(Blob.type_num, unpacked.pack_type_num)
  548. self.assertEqual(Blob.type_num, unpacked.obj_type_num)
  549. self.assertEqual([b"blob"], unpacked.decomp_chunks)
  550. self.assertEqual(crc32, unpacked.crc32)
  551. self.assertEqual(b"x", unused)
  552. def test_write_pack_object_sha(self):
  553. f = BytesIO()
  554. f.write(b"header")
  555. offset = f.tell()
  556. sha_a = sha1(b"foo")
  557. sha_b = sha_a.copy()
  558. write_pack_object(f.write, Blob.type_num, b"blob", sha=sha_a)
  559. self.assertNotEqual(sha_a.digest(), sha_b.digest())
  560. sha_b.update(f.getvalue()[offset:])
  561. self.assertEqual(sha_a.digest(), sha_b.digest())
  562. def test_write_pack_object_compression_level(self):
  563. f = BytesIO()
  564. f.write(b"header")
  565. offset = f.tell()
  566. sha_a = sha1(b"foo")
  567. sha_b = sha_a.copy()
  568. write_pack_object(
  569. f.write, Blob.type_num, b"blob", sha=sha_a, compression_level=6
  570. )
  571. self.assertNotEqual(sha_a.digest(), sha_b.digest())
  572. sha_b.update(f.getvalue()[offset:])
  573. self.assertEqual(sha_a.digest(), sha_b.digest())
  574. pack_checksum = hex_to_sha("721980e866af9a5f93ad674144e1459b8ba3e7b7")
  575. class BaseTestPackIndexWriting:
  576. def assertSucceeds(self, func, *args, **kwargs):
  577. try:
  578. func(*args, **kwargs)
  579. except ChecksumMismatch as e:
  580. self.fail(e)
  581. def index(self, filename, entries, pack_checksum):
  582. raise NotImplementedError(self.index)
  583. def test_empty(self):
  584. idx = self.index("empty.idx", [], pack_checksum)
  585. self.assertEqual(idx.get_pack_checksum(), pack_checksum)
  586. self.assertEqual(0, len(idx))
  587. def test_large(self):
  588. entry1_sha = hex_to_sha("4e6388232ec39792661e2e75db8fb117fc869ce6")
  589. entry2_sha = hex_to_sha("e98f071751bd77f59967bfa671cd2caebdccc9a2")
  590. entries = [
  591. (entry1_sha, 0xF2972D0830529B87, 24),
  592. (entry2_sha, (~0xF2972D0830529B87) & (2**64 - 1), 92),
  593. ]
  594. if not self._supports_large:
  595. self.assertRaises(
  596. TypeError, self.index, "single.idx", entries, pack_checksum
  597. )
  598. return
  599. idx = self.index("single.idx", entries, pack_checksum)
  600. self.assertEqual(idx.get_pack_checksum(), pack_checksum)
  601. self.assertEqual(2, len(idx))
  602. actual_entries = list(idx.iterentries())
  603. self.assertEqual(len(entries), len(actual_entries))
  604. for mine, actual in zip(entries, actual_entries):
  605. my_sha, my_offset, my_crc = mine
  606. actual_sha, actual_offset, actual_crc = actual
  607. self.assertEqual(my_sha, actual_sha)
  608. self.assertEqual(my_offset, actual_offset)
  609. if self._has_crc32_checksum:
  610. self.assertEqual(my_crc, actual_crc)
  611. else:
  612. self.assertIsNone(actual_crc)
  613. def test_single(self):
  614. entry_sha = hex_to_sha("6f670c0fb53f9463760b7295fbb814e965fb20c8")
  615. my_entries = [(entry_sha, 178, 42)]
  616. idx = self.index("single.idx", my_entries, pack_checksum)
  617. self.assertEqual(idx.get_pack_checksum(), pack_checksum)
  618. self.assertEqual(1, len(idx))
  619. actual_entries = list(idx.iterentries())
  620. self.assertEqual(len(my_entries), len(actual_entries))
  621. for mine, actual in zip(my_entries, actual_entries):
  622. my_sha, my_offset, my_crc = mine
  623. actual_sha, actual_offset, actual_crc = actual
  624. self.assertEqual(my_sha, actual_sha)
  625. self.assertEqual(my_offset, actual_offset)
  626. if self._has_crc32_checksum:
  627. self.assertEqual(my_crc, actual_crc)
  628. else:
  629. self.assertIsNone(actual_crc)
  630. class BaseTestFilePackIndexWriting(BaseTestPackIndexWriting):
  631. def setUp(self):
  632. self.tempdir = tempfile.mkdtemp()
  633. def tearDown(self):
  634. shutil.rmtree(self.tempdir)
  635. def index(self, filename, entries, pack_checksum):
  636. path = os.path.join(self.tempdir, filename)
  637. self.writeIndex(path, entries, pack_checksum)
  638. idx = load_pack_index(path)
  639. self.assertSucceeds(idx.check)
  640. self.assertEqual(idx.version, self._expected_version)
  641. return idx
  642. def writeIndex(self, filename, entries, pack_checksum):
  643. # FIXME: Write to BytesIO instead rather than hitting disk ?
  644. with GitFile(filename, "wb") as f:
  645. self._write_fn(f, entries, pack_checksum)
  646. class TestMemoryIndexWriting(TestCase, BaseTestPackIndexWriting):
  647. def setUp(self):
  648. TestCase.setUp(self)
  649. self._has_crc32_checksum = True
  650. self._supports_large = True
  651. def index(self, filename, entries, pack_checksum):
  652. return MemoryPackIndex(entries, pack_checksum)
  653. def tearDown(self):
  654. TestCase.tearDown(self)
  655. class TestPackIndexWritingv1(TestCase, BaseTestFilePackIndexWriting):
  656. def setUp(self):
  657. TestCase.setUp(self)
  658. BaseTestFilePackIndexWriting.setUp(self)
  659. self._has_crc32_checksum = False
  660. self._expected_version = 1
  661. self._supports_large = False
  662. self._write_fn = write_pack_index_v1
  663. def tearDown(self):
  664. TestCase.tearDown(self)
  665. BaseTestFilePackIndexWriting.tearDown(self)
  666. class TestPackIndexWritingv2(TestCase, BaseTestFilePackIndexWriting):
  667. def setUp(self):
  668. TestCase.setUp(self)
  669. BaseTestFilePackIndexWriting.setUp(self)
  670. self._has_crc32_checksum = True
  671. self._supports_large = True
  672. self._expected_version = 2
  673. self._write_fn = write_pack_index_v2
  674. def tearDown(self):
  675. TestCase.tearDown(self)
  676. BaseTestFilePackIndexWriting.tearDown(self)
  677. class ReadZlibTests(TestCase):
  678. decomp = (
  679. b"tree 4ada885c9196b6b6fa08744b5862bf92896fc002\n"
  680. b"parent None\n"
  681. b"author Jelmer Vernooij <jelmer@samba.org> 1228980214 +0000\n"
  682. b"committer Jelmer Vernooij <jelmer@samba.org> 1228980214 +0000\n"
  683. b"\n"
  684. b"Provide replacement for mmap()'s offset argument."
  685. )
  686. comp = zlib.compress(decomp)
  687. extra = b"nextobject"
  688. def setUp(self):
  689. super().setUp()
  690. self.read = BytesIO(self.comp + self.extra).read
  691. self.unpacked = UnpackedObject(
  692. Tree.type_num, decomp_len=len(self.decomp), crc32=0
  693. )
  694. def test_decompress_size(self):
  695. good_decomp_len = len(self.decomp)
  696. self.unpacked.decomp_len = -1
  697. self.assertRaises(ValueError, read_zlib_chunks, self.read, self.unpacked)
  698. self.unpacked.decomp_len = good_decomp_len - 1
  699. self.assertRaises(zlib.error, read_zlib_chunks, self.read, self.unpacked)
  700. self.unpacked.decomp_len = good_decomp_len + 1
  701. self.assertRaises(zlib.error, read_zlib_chunks, self.read, self.unpacked)
  702. def test_decompress_truncated(self):
  703. read = BytesIO(self.comp[:10]).read
  704. self.assertRaises(zlib.error, read_zlib_chunks, read, self.unpacked)
  705. read = BytesIO(self.comp).read
  706. self.assertRaises(zlib.error, read_zlib_chunks, read, self.unpacked)
  707. def test_decompress_empty(self):
  708. unpacked = UnpackedObject(Tree.type_num, decomp_len=0)
  709. comp = zlib.compress(b"")
  710. read = BytesIO(comp + self.extra).read
  711. unused = read_zlib_chunks(read, unpacked)
  712. self.assertEqual(b"", b"".join(unpacked.decomp_chunks))
  713. self.assertNotEqual(b"", unused)
  714. self.assertEqual(self.extra, unused + read())
  715. def test_decompress_no_crc32(self):
  716. self.unpacked.crc32 = None
  717. read_zlib_chunks(self.read, self.unpacked)
  718. self.assertEqual(None, self.unpacked.crc32)
  719. def _do_decompress_test(self, buffer_size, **kwargs):
  720. unused = read_zlib_chunks(
  721. self.read, self.unpacked, buffer_size=buffer_size, **kwargs
  722. )
  723. self.assertEqual(self.decomp, b"".join(self.unpacked.decomp_chunks))
  724. self.assertEqual(zlib.crc32(self.comp), self.unpacked.crc32)
  725. self.assertNotEqual(b"", unused)
  726. self.assertEqual(self.extra, unused + self.read())
  727. def test_simple_decompress(self):
  728. self._do_decompress_test(4096)
  729. self.assertEqual(None, self.unpacked.comp_chunks)
  730. # These buffer sizes are not intended to be realistic, but rather simulate
  731. # larger buffer sizes that may end at various places.
  732. def test_decompress_buffer_size_1(self):
  733. self._do_decompress_test(1)
  734. def test_decompress_buffer_size_2(self):
  735. self._do_decompress_test(2)
  736. def test_decompress_buffer_size_3(self):
  737. self._do_decompress_test(3)
  738. def test_decompress_buffer_size_4(self):
  739. self._do_decompress_test(4)
  740. def test_decompress_include_comp(self):
  741. self._do_decompress_test(4096, include_comp=True)
  742. self.assertEqual(self.comp, b"".join(self.unpacked.comp_chunks))
  743. class DeltifyTests(TestCase):
  744. def test_empty(self):
  745. self.assertEqual([], list(deltify_pack_objects([])))
  746. def test_single(self):
  747. b = Blob.from_string(b"foo")
  748. self.assertEqual(
  749. [
  750. UnpackedObject(
  751. b.type_num,
  752. sha=b.sha().digest(),
  753. delta_base=None,
  754. decomp_chunks=b.as_raw_chunks(),
  755. )
  756. ],
  757. list(deltify_pack_objects([(b, b"")])),
  758. )
  759. def test_simple_delta(self):
  760. b1 = Blob.from_string(b"a" * 101)
  761. b2 = Blob.from_string(b"a" * 100)
  762. delta = list(create_delta(b1.as_raw_chunks(), b2.as_raw_chunks()))
  763. self.assertEqual(
  764. [
  765. UnpackedObject(
  766. b1.type_num,
  767. sha=b1.sha().digest(),
  768. delta_base=None,
  769. decomp_chunks=b1.as_raw_chunks(),
  770. ),
  771. UnpackedObject(
  772. b2.type_num,
  773. sha=b2.sha().digest(),
  774. delta_base=b1.sha().digest(),
  775. decomp_chunks=delta,
  776. ),
  777. ],
  778. list(deltify_pack_objects([(b1, b""), (b2, b"")])),
  779. )
  780. class TestPackStreamReader(TestCase):
  781. def test_read_objects_emtpy(self):
  782. f = BytesIO()
  783. build_pack(f, [])
  784. reader = PackStreamReader(f.read)
  785. self.assertEqual(0, len(list(reader.read_objects())))
  786. def test_read_objects(self):
  787. f = BytesIO()
  788. entries = build_pack(
  789. f,
  790. [
  791. (Blob.type_num, b"blob"),
  792. (OFS_DELTA, (0, b"blob1")),
  793. ],
  794. )
  795. reader = PackStreamReader(f.read)
  796. objects = list(reader.read_objects(compute_crc32=True))
  797. self.assertEqual(2, len(objects))
  798. unpacked_blob, unpacked_delta = objects
  799. self.assertEqual(entries[0][0], unpacked_blob.offset)
  800. self.assertEqual(Blob.type_num, unpacked_blob.pack_type_num)
  801. self.assertEqual(Blob.type_num, unpacked_blob.obj_type_num)
  802. self.assertEqual(None, unpacked_blob.delta_base)
  803. self.assertEqual(b"blob", b"".join(unpacked_blob.decomp_chunks))
  804. self.assertEqual(entries[0][4], unpacked_blob.crc32)
  805. self.assertEqual(entries[1][0], unpacked_delta.offset)
  806. self.assertEqual(OFS_DELTA, unpacked_delta.pack_type_num)
  807. self.assertEqual(None, unpacked_delta.obj_type_num)
  808. self.assertEqual(
  809. unpacked_delta.offset - unpacked_blob.offset,
  810. unpacked_delta.delta_base,
  811. )
  812. delta = create_delta(b"blob", b"blob1")
  813. self.assertEqual(b"".join(delta), b"".join(unpacked_delta.decomp_chunks))
  814. self.assertEqual(entries[1][4], unpacked_delta.crc32)
  815. def test_read_objects_buffered(self):
  816. f = BytesIO()
  817. build_pack(
  818. f,
  819. [
  820. (Blob.type_num, b"blob"),
  821. (OFS_DELTA, (0, b"blob1")),
  822. ],
  823. )
  824. reader = PackStreamReader(f.read, zlib_bufsize=4)
  825. self.assertEqual(2, len(list(reader.read_objects())))
  826. def test_read_objects_empty(self):
  827. reader = PackStreamReader(BytesIO().read)
  828. self.assertRaises(AssertionError, list, reader.read_objects())
  829. class TestPackIterator(DeltaChainIterator):
  830. _compute_crc32 = True
  831. def __init__(self, *args, **kwargs) -> None:
  832. super().__init__(*args, **kwargs)
  833. self._unpacked_offsets: Set[int] = set()
  834. def _result(self, unpacked):
  835. """Return entries in the same format as build_pack."""
  836. return (
  837. unpacked.offset,
  838. unpacked.obj_type_num,
  839. b"".join(unpacked.obj_chunks),
  840. unpacked.sha(),
  841. unpacked.crc32,
  842. )
  843. def _resolve_object(self, offset, pack_type_num, base_chunks):
  844. assert offset not in self._unpacked_offsets, (
  845. "Attempted to re-inflate offset %i" % offset
  846. )
  847. self._unpacked_offsets.add(offset)
  848. return super()._resolve_object(offset, pack_type_num, base_chunks)
  849. class DeltaChainIteratorTests(TestCase):
  850. def setUp(self):
  851. super().setUp()
  852. self.store = MemoryObjectStore()
  853. self.fetched = set()
  854. def store_blobs(self, blobs_data):
  855. blobs = []
  856. for data in blobs_data:
  857. blob = make_object(Blob, data=data)
  858. blobs.append(blob)
  859. self.store.add_object(blob)
  860. return blobs
  861. def get_raw_no_repeat(self, bin_sha):
  862. """Wrapper around store.get_raw that doesn't allow repeat lookups."""
  863. hex_sha = sha_to_hex(bin_sha)
  864. self.assertNotIn(
  865. hex_sha, self.fetched, f"Attempted to re-fetch object {hex_sha}"
  866. )
  867. self.fetched.add(hex_sha)
  868. return self.store.get_raw(hex_sha)
  869. def make_pack_iter(self, f, thin=None):
  870. if thin is None:
  871. thin = bool(list(self.store))
  872. resolve_ext_ref = thin and self.get_raw_no_repeat or None
  873. data = PackData("test.pack", file=f)
  874. return TestPackIterator.for_pack_data(data, resolve_ext_ref=resolve_ext_ref)
  875. def make_pack_iter_subset(self, f, subset, thin=None):
  876. if thin is None:
  877. thin = bool(list(self.store))
  878. resolve_ext_ref = thin and self.get_raw_no_repeat or None
  879. data = PackData("test.pack", file=f)
  880. assert data
  881. index = MemoryPackIndex.for_pack(data)
  882. pack = Pack.from_objects(data, index)
  883. return TestPackIterator.for_pack_subset(
  884. pack, subset, resolve_ext_ref=resolve_ext_ref
  885. )
  886. def assertEntriesMatch(self, expected_indexes, entries, pack_iter):
  887. expected = [entries[i] for i in expected_indexes]
  888. self.assertEqual(expected, list(pack_iter._walk_all_chains()))
  889. def test_no_deltas(self):
  890. f = BytesIO()
  891. entries = build_pack(
  892. f,
  893. [
  894. (Commit.type_num, b"commit"),
  895. (Blob.type_num, b"blob"),
  896. (Tree.type_num, b"tree"),
  897. ],
  898. )
  899. self.assertEntriesMatch([0, 1, 2], entries, self.make_pack_iter(f))
  900. f.seek(0)
  901. self.assertEntriesMatch([], entries, self.make_pack_iter_subset(f, []))
  902. f.seek(0)
  903. self.assertEntriesMatch(
  904. [1, 0],
  905. entries,
  906. self.make_pack_iter_subset(f, [entries[0][3], entries[1][3]]),
  907. )
  908. f.seek(0)
  909. self.assertEntriesMatch(
  910. [1, 0],
  911. entries,
  912. self.make_pack_iter_subset(
  913. f, [sha_to_hex(entries[0][3]), sha_to_hex(entries[1][3])]
  914. ),
  915. )
  916. def test_ofs_deltas(self):
  917. f = BytesIO()
  918. entries = build_pack(
  919. f,
  920. [
  921. (Blob.type_num, b"blob"),
  922. (OFS_DELTA, (0, b"blob1")),
  923. (OFS_DELTA, (0, b"blob2")),
  924. ],
  925. )
  926. # Delta resolution changed to DFS
  927. self.assertEntriesMatch([0, 2, 1], entries, self.make_pack_iter(f))
  928. f.seek(0)
  929. self.assertEntriesMatch(
  930. [0, 2, 1],
  931. entries,
  932. self.make_pack_iter_subset(f, [entries[1][3], entries[2][3]]),
  933. )
  934. def test_ofs_deltas_chain(self):
  935. f = BytesIO()
  936. entries = build_pack(
  937. f,
  938. [
  939. (Blob.type_num, b"blob"),
  940. (OFS_DELTA, (0, b"blob1")),
  941. (OFS_DELTA, (1, b"blob2")),
  942. ],
  943. )
  944. self.assertEntriesMatch([0, 1, 2], entries, self.make_pack_iter(f))
  945. def test_ref_deltas(self):
  946. f = BytesIO()
  947. entries = build_pack(
  948. f,
  949. [
  950. (REF_DELTA, (1, b"blob1")),
  951. (Blob.type_num, (b"blob")),
  952. (REF_DELTA, (1, b"blob2")),
  953. ],
  954. )
  955. # Delta resolution changed to DFS
  956. self.assertEntriesMatch([1, 2, 0], entries, self.make_pack_iter(f))
  957. def test_ref_deltas_chain(self):
  958. f = BytesIO()
  959. entries = build_pack(
  960. f,
  961. [
  962. (REF_DELTA, (2, b"blob1")),
  963. (Blob.type_num, (b"blob")),
  964. (REF_DELTA, (1, b"blob2")),
  965. ],
  966. )
  967. self.assertEntriesMatch([1, 2, 0], entries, self.make_pack_iter(f))
  968. def test_ofs_and_ref_deltas(self):
  969. # Deltas pending on this offset are popped before deltas depending on
  970. # this ref.
  971. f = BytesIO()
  972. entries = build_pack(
  973. f,
  974. [
  975. (REF_DELTA, (1, b"blob1")),
  976. (Blob.type_num, (b"blob")),
  977. (OFS_DELTA, (1, b"blob2")),
  978. ],
  979. )
  980. # Delta resolution changed to DFS
  981. self.assertEntriesMatch([1, 0, 2], entries, self.make_pack_iter(f))
  982. def test_mixed_chain(self):
  983. f = BytesIO()
  984. entries = build_pack(
  985. f,
  986. [
  987. (Blob.type_num, b"blob"),
  988. (REF_DELTA, (2, b"blob2")),
  989. (OFS_DELTA, (0, b"blob1")),
  990. (OFS_DELTA, (1, b"blob3")),
  991. (OFS_DELTA, (0, b"bob")),
  992. ],
  993. )
  994. # Delta resolution changed to DFS
  995. self.assertEntriesMatch([0, 4, 2, 1, 3], entries, self.make_pack_iter(f))
  996. def test_long_chain(self):
  997. n = 100
  998. objects_spec = [(Blob.type_num, b"blob")]
  999. for i in range(n):
  1000. objects_spec.append((OFS_DELTA, (i, b"blob" + str(i).encode("ascii"))))
  1001. f = BytesIO()
  1002. entries = build_pack(f, objects_spec)
  1003. self.assertEntriesMatch(range(n + 1), entries, self.make_pack_iter(f))
  1004. def test_branchy_chain(self):
  1005. n = 100
  1006. objects_spec = [(Blob.type_num, b"blob")]
  1007. for i in range(n):
  1008. objects_spec.append((OFS_DELTA, (0, b"blob" + str(i).encode("ascii"))))
  1009. f = BytesIO()
  1010. entries = build_pack(f, objects_spec)
  1011. # Delta resolution changed to DFS
  1012. indices = [0, *list(range(100, 0, -1))]
  1013. self.assertEntriesMatch(indices, entries, self.make_pack_iter(f))
  1014. def test_ext_ref(self):
  1015. (blob,) = self.store_blobs([b"blob"])
  1016. f = BytesIO()
  1017. entries = build_pack(f, [(REF_DELTA, (blob.id, b"blob1"))], store=self.store)
  1018. pack_iter = self.make_pack_iter(f)
  1019. self.assertEntriesMatch([0], entries, pack_iter)
  1020. self.assertEqual([hex_to_sha(blob.id)], pack_iter.ext_refs())
  1021. def test_ext_ref_chain(self):
  1022. (blob,) = self.store_blobs([b"blob"])
  1023. f = BytesIO()
  1024. entries = build_pack(
  1025. f,
  1026. [
  1027. (REF_DELTA, (1, b"blob2")),
  1028. (REF_DELTA, (blob.id, b"blob1")),
  1029. ],
  1030. store=self.store,
  1031. )
  1032. pack_iter = self.make_pack_iter(f)
  1033. self.assertEntriesMatch([1, 0], entries, pack_iter)
  1034. self.assertEqual([hex_to_sha(blob.id)], pack_iter.ext_refs())
  1035. def test_ext_ref_chain_degenerate(self):
  1036. # Test a degenerate case where the sender is sending a REF_DELTA
  1037. # object that expands to an object already in the repository.
  1038. (blob,) = self.store_blobs([b"blob"])
  1039. (blob2,) = self.store_blobs([b"blob2"])
  1040. assert blob.id < blob2.id
  1041. f = BytesIO()
  1042. entries = build_pack(
  1043. f,
  1044. [
  1045. (REF_DELTA, (blob.id, b"blob2")),
  1046. (REF_DELTA, (0, b"blob3")),
  1047. ],
  1048. store=self.store,
  1049. )
  1050. pack_iter = self.make_pack_iter(f)
  1051. self.assertEntriesMatch([0, 1], entries, pack_iter)
  1052. self.assertEqual([hex_to_sha(blob.id)], pack_iter.ext_refs())
  1053. def test_ext_ref_multiple_times(self):
  1054. (blob,) = self.store_blobs([b"blob"])
  1055. f = BytesIO()
  1056. entries = build_pack(
  1057. f,
  1058. [
  1059. (REF_DELTA, (blob.id, b"blob1")),
  1060. (REF_DELTA, (blob.id, b"blob2")),
  1061. ],
  1062. store=self.store,
  1063. )
  1064. pack_iter = self.make_pack_iter(f)
  1065. self.assertEntriesMatch([0, 1], entries, pack_iter)
  1066. self.assertEqual([hex_to_sha(blob.id)], pack_iter.ext_refs())
  1067. def test_multiple_ext_refs(self):
  1068. b1, b2 = self.store_blobs([b"foo", b"bar"])
  1069. f = BytesIO()
  1070. entries = build_pack(
  1071. f,
  1072. [
  1073. (REF_DELTA, (b1.id, b"foo1")),
  1074. (REF_DELTA, (b2.id, b"bar2")),
  1075. ],
  1076. store=self.store,
  1077. )
  1078. pack_iter = self.make_pack_iter(f)
  1079. self.assertEntriesMatch([0, 1], entries, pack_iter)
  1080. self.assertEqual([hex_to_sha(b1.id), hex_to_sha(b2.id)], pack_iter.ext_refs())
  1081. def test_bad_ext_ref_non_thin_pack(self):
  1082. (blob,) = self.store_blobs([b"blob"])
  1083. f = BytesIO()
  1084. build_pack(f, [(REF_DELTA, (blob.id, b"blob1"))], store=self.store)
  1085. pack_iter = self.make_pack_iter(f, thin=False)
  1086. try:
  1087. list(pack_iter._walk_all_chains())
  1088. self.fail()
  1089. except UnresolvedDeltas as e:
  1090. self.assertEqual([blob.id], e.shas)
  1091. def test_bad_ext_ref_thin_pack(self):
  1092. b1, b2, b3 = self.store_blobs([b"foo", b"bar", b"baz"])
  1093. f = BytesIO()
  1094. build_pack(
  1095. f,
  1096. [
  1097. (REF_DELTA, (1, b"foo99")),
  1098. (REF_DELTA, (b1.id, b"foo1")),
  1099. (REF_DELTA, (b2.id, b"bar2")),
  1100. (REF_DELTA, (b3.id, b"baz3")),
  1101. ],
  1102. store=self.store,
  1103. )
  1104. del self.store[b2.id]
  1105. del self.store[b3.id]
  1106. pack_iter = self.make_pack_iter(f)
  1107. try:
  1108. list(pack_iter._walk_all_chains())
  1109. self.fail()
  1110. except UnresolvedDeltas as e:
  1111. self.assertEqual((sorted([b2.id, b3.id]),), (sorted(e.shas),))
  1112. class DeltaEncodeSizeTests(TestCase):
  1113. def test_basic(self):
  1114. self.assertEqual(b"\x00", _delta_encode_size(0))
  1115. self.assertEqual(b"\x01", _delta_encode_size(1))
  1116. self.assertEqual(b"\xfa\x01", _delta_encode_size(250))
  1117. self.assertEqual(b"\xe8\x07", _delta_encode_size(1000))
  1118. self.assertEqual(b"\xa0\x8d\x06", _delta_encode_size(100000))
  1119. class EncodeCopyOperationTests(TestCase):
  1120. def test_basic(self):
  1121. self.assertEqual(b"\x80", _encode_copy_operation(0, 0))
  1122. self.assertEqual(b"\x91\x01\x0a", _encode_copy_operation(1, 10))
  1123. self.assertEqual(b"\xb1\x64\xe8\x03", _encode_copy_operation(100, 1000))
  1124. self.assertEqual(b"\x93\xe8\x03\x01", _encode_copy_operation(1000, 1))