test_pack.py 42 KB

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