test_refs.py 30 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816
  1. # test_refs.py -- tests for refs.py
  2. # Copyright (C) 2013 Jelmer Vernooij <jelmer@jelmer.uk>
  3. #
  4. # Dulwich is dual-licensed under the Apache License, Version 2.0 and the GNU
  5. # General Public License as public by the Free Software Foundation; version 2.0
  6. # or (at your option) any later version. You can redistribute it and/or
  7. # modify it under the terms of either of these two licenses.
  8. #
  9. # Unless required by applicable law or agreed to in writing, software
  10. # distributed under the License is distributed on an "AS IS" BASIS,
  11. # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  12. # See the License for the specific language governing permissions and
  13. # limitations under the License.
  14. #
  15. # You should have received a copy of the licenses; if not, see
  16. # <http://www.gnu.org/licenses/> for a copy of the GNU General Public License
  17. # and <http://www.apache.org/licenses/LICENSE-2.0> for a copy of the Apache
  18. # License, Version 2.0.
  19. #
  20. """Tests for dulwich.refs."""
  21. import os
  22. import sys
  23. import tempfile
  24. from io import BytesIO
  25. from typing import ClassVar, Dict
  26. from dulwich import errors
  27. from dulwich.file import GitFile
  28. from dulwich.objects import ZERO_SHA
  29. from dulwich.refs import (
  30. DictRefsContainer,
  31. InfoRefsContainer,
  32. SymrefLoop,
  33. _split_ref_line,
  34. check_ref_format,
  35. parse_symref_value,
  36. read_packed_refs,
  37. read_packed_refs_with_peeled,
  38. strip_peeled_refs,
  39. write_packed_refs,
  40. )
  41. from dulwich.repo import Repo
  42. from dulwich.tests.utils import open_repo, tear_down_repo
  43. from . import SkipTest, TestCase
  44. class CheckRefFormatTests(TestCase):
  45. """Tests for the check_ref_format function.
  46. These are the same tests as in the git test suite.
  47. """
  48. def test_valid(self):
  49. self.assertTrue(check_ref_format(b"heads/foo"))
  50. self.assertTrue(check_ref_format(b"foo/bar/baz"))
  51. self.assertTrue(check_ref_format(b"refs///heads/foo"))
  52. self.assertTrue(check_ref_format(b"foo./bar"))
  53. self.assertTrue(check_ref_format(b"heads/foo@bar"))
  54. self.assertTrue(check_ref_format(b"heads/fix.lock.error"))
  55. def test_invalid(self):
  56. self.assertFalse(check_ref_format(b"foo"))
  57. self.assertFalse(check_ref_format(b"heads/foo/"))
  58. self.assertFalse(check_ref_format(b"./foo"))
  59. self.assertFalse(check_ref_format(b".refs/foo"))
  60. self.assertFalse(check_ref_format(b"heads/foo..bar"))
  61. self.assertFalse(check_ref_format(b"heads/foo?bar"))
  62. self.assertFalse(check_ref_format(b"heads/foo.lock"))
  63. self.assertFalse(check_ref_format(b"heads/v@{ation"))
  64. self.assertFalse(check_ref_format(b"heads/foo\bar"))
  65. ONES = b"1" * 40
  66. TWOS = b"2" * 40
  67. THREES = b"3" * 40
  68. FOURS = b"4" * 40
  69. class PackedRefsFileTests(TestCase):
  70. def test_split_ref_line_errors(self):
  71. self.assertRaises(errors.PackedRefsException, _split_ref_line, b"singlefield")
  72. self.assertRaises(errors.PackedRefsException, _split_ref_line, b"badsha name")
  73. self.assertRaises(
  74. errors.PackedRefsException,
  75. _split_ref_line,
  76. ONES + b" bad/../refname",
  77. )
  78. def test_read_without_peeled(self):
  79. f = BytesIO(b"\n".join([b"# comment", ONES + b" ref/1", TWOS + b" ref/2"]))
  80. self.assertEqual(
  81. [(ONES, b"ref/1"), (TWOS, b"ref/2")], list(read_packed_refs(f))
  82. )
  83. def test_read_without_peeled_errors(self):
  84. f = BytesIO(b"\n".join([ONES + b" ref/1", b"^" + TWOS]))
  85. self.assertRaises(errors.PackedRefsException, list, read_packed_refs(f))
  86. def test_read_with_peeled(self):
  87. f = BytesIO(
  88. b"\n".join(
  89. [
  90. ONES + b" ref/1",
  91. TWOS + b" ref/2",
  92. b"^" + THREES,
  93. FOURS + b" ref/4",
  94. ]
  95. )
  96. )
  97. self.assertEqual(
  98. [
  99. (ONES, b"ref/1", None),
  100. (TWOS, b"ref/2", THREES),
  101. (FOURS, b"ref/4", None),
  102. ],
  103. list(read_packed_refs_with_peeled(f)),
  104. )
  105. def test_read_with_peeled_errors(self):
  106. f = BytesIO(b"\n".join([b"^" + TWOS, ONES + b" ref/1"]))
  107. self.assertRaises(errors.PackedRefsException, list, read_packed_refs(f))
  108. f = BytesIO(b"\n".join([ONES + b" ref/1", b"^" + TWOS, b"^" + THREES]))
  109. self.assertRaises(errors.PackedRefsException, list, read_packed_refs(f))
  110. def test_write_with_peeled(self):
  111. f = BytesIO()
  112. write_packed_refs(f, {b"ref/1": ONES, b"ref/2": TWOS}, {b"ref/1": THREES})
  113. self.assertEqual(
  114. b"\n".join(
  115. [
  116. b"# pack-refs with: peeled",
  117. ONES + b" ref/1",
  118. b"^" + THREES,
  119. TWOS + b" ref/2",
  120. ]
  121. )
  122. + b"\n",
  123. f.getvalue(),
  124. )
  125. def test_write_without_peeled(self):
  126. f = BytesIO()
  127. write_packed_refs(f, {b"ref/1": ONES, b"ref/2": TWOS})
  128. self.assertEqual(
  129. b"\n".join([ONES + b" ref/1", TWOS + b" ref/2"]) + b"\n",
  130. f.getvalue(),
  131. )
  132. # Dict of refs that we expect all RefsContainerTests subclasses to define.
  133. _TEST_REFS = {
  134. b"HEAD": b"42d06bd4b77fed026b154d16493e5deab78f02ec",
  135. b"refs/heads/40-char-ref-aaaaaaaaaaaaaaaaaa": b"42d06bd4b77fed026b154d16493e5deab78f02ec",
  136. b"refs/heads/master": b"42d06bd4b77fed026b154d16493e5deab78f02ec",
  137. b"refs/heads/packed": b"42d06bd4b77fed026b154d16493e5deab78f02ec",
  138. b"refs/tags/refs-0.1": b"df6800012397fb85c56e7418dd4eb9405dee075c",
  139. b"refs/tags/refs-0.2": b"3ec9c43c84ff242e3ef4a9fc5bc111fd780a76a8",
  140. b"refs/heads/loop": b"ref: refs/heads/loop",
  141. }
  142. class RefsContainerTests:
  143. def test_keys(self):
  144. actual_keys = set(self._refs.keys())
  145. self.assertEqual(set(self._refs.allkeys()), actual_keys)
  146. self.assertEqual(set(_TEST_REFS.keys()), actual_keys)
  147. actual_keys = self._refs.keys(b"refs/heads")
  148. actual_keys.discard(b"loop")
  149. self.assertEqual(
  150. [b"40-char-ref-aaaaaaaaaaaaaaaaaa", b"master", b"packed"],
  151. sorted(actual_keys),
  152. )
  153. self.assertEqual(
  154. [b"refs-0.1", b"refs-0.2"], sorted(self._refs.keys(b"refs/tags"))
  155. )
  156. def test_iter(self):
  157. actual_keys = set(self._refs.keys())
  158. self.assertEqual(set(self._refs), actual_keys)
  159. self.assertEqual(set(_TEST_REFS.keys()), actual_keys)
  160. def test_as_dict(self):
  161. # refs/heads/loop does not show up even if it exists
  162. expected_refs = dict(_TEST_REFS)
  163. del expected_refs[b"refs/heads/loop"]
  164. self.assertEqual(expected_refs, self._refs.as_dict())
  165. def test_get_symrefs(self):
  166. self._refs.set_symbolic_ref(b"refs/heads/src", b"refs/heads/dst")
  167. symrefs = self._refs.get_symrefs()
  168. if b"HEAD" in symrefs:
  169. symrefs.pop(b"HEAD")
  170. self.assertEqual(
  171. {
  172. b"refs/heads/src": b"refs/heads/dst",
  173. b"refs/heads/loop": b"refs/heads/loop",
  174. },
  175. symrefs,
  176. )
  177. def test_setitem(self):
  178. self._refs[b"refs/some/ref"] = b"42d06bd4b77fed026b154d16493e5deab78f02ec"
  179. self.assertEqual(
  180. b"42d06bd4b77fed026b154d16493e5deab78f02ec",
  181. self._refs[b"refs/some/ref"],
  182. )
  183. self.assertRaises(
  184. errors.RefFormatError,
  185. self._refs.__setitem__,
  186. b"notrefs/foo",
  187. b"42d06bd4b77fed026b154d16493e5deab78f02ec",
  188. )
  189. def test_set_if_equals(self):
  190. nines = b"9" * 40
  191. self.assertFalse(self._refs.set_if_equals(b"HEAD", b"c0ffee", nines))
  192. self.assertEqual(
  193. b"42d06bd4b77fed026b154d16493e5deab78f02ec", self._refs[b"HEAD"]
  194. )
  195. self.assertTrue(
  196. self._refs.set_if_equals(
  197. b"HEAD", b"42d06bd4b77fed026b154d16493e5deab78f02ec", nines
  198. )
  199. )
  200. self.assertEqual(nines, self._refs[b"HEAD"])
  201. # Setting the ref again is a no-op, but will return True.
  202. self.assertTrue(self._refs.set_if_equals(b"HEAD", nines, nines))
  203. self.assertEqual(nines, self._refs[b"HEAD"])
  204. self.assertTrue(self._refs.set_if_equals(b"refs/heads/master", None, nines))
  205. self.assertEqual(nines, self._refs[b"refs/heads/master"])
  206. self.assertTrue(
  207. self._refs.set_if_equals(b"refs/heads/nonexistent", ZERO_SHA, nines)
  208. )
  209. self.assertEqual(nines, self._refs[b"refs/heads/nonexistent"])
  210. def test_add_if_new(self):
  211. nines = b"9" * 40
  212. self.assertFalse(self._refs.add_if_new(b"refs/heads/master", nines))
  213. self.assertEqual(
  214. b"42d06bd4b77fed026b154d16493e5deab78f02ec",
  215. self._refs[b"refs/heads/master"],
  216. )
  217. self.assertTrue(self._refs.add_if_new(b"refs/some/ref", nines))
  218. self.assertEqual(nines, self._refs[b"refs/some/ref"])
  219. def test_set_symbolic_ref(self):
  220. self._refs.set_symbolic_ref(b"refs/heads/symbolic", b"refs/heads/master")
  221. self.assertEqual(
  222. b"ref: refs/heads/master",
  223. self._refs.read_loose_ref(b"refs/heads/symbolic"),
  224. )
  225. self.assertEqual(
  226. b"42d06bd4b77fed026b154d16493e5deab78f02ec",
  227. self._refs[b"refs/heads/symbolic"],
  228. )
  229. def test_set_symbolic_ref_overwrite(self):
  230. nines = b"9" * 40
  231. self.assertNotIn(b"refs/heads/symbolic", self._refs)
  232. self._refs[b"refs/heads/symbolic"] = nines
  233. self.assertEqual(nines, self._refs.read_loose_ref(b"refs/heads/symbolic"))
  234. self._refs.set_symbolic_ref(b"refs/heads/symbolic", b"refs/heads/master")
  235. self.assertEqual(
  236. b"ref: refs/heads/master",
  237. self._refs.read_loose_ref(b"refs/heads/symbolic"),
  238. )
  239. self.assertEqual(
  240. b"42d06bd4b77fed026b154d16493e5deab78f02ec",
  241. self._refs[b"refs/heads/symbolic"],
  242. )
  243. def test_check_refname(self):
  244. self._refs._check_refname(b"HEAD")
  245. self._refs._check_refname(b"refs/stash")
  246. self._refs._check_refname(b"refs/heads/foo")
  247. self.assertRaises(errors.RefFormatError, self._refs._check_refname, b"refs")
  248. self.assertRaises(
  249. errors.RefFormatError, self._refs._check_refname, b"notrefs/foo"
  250. )
  251. def test_contains(self):
  252. self.assertIn(b"refs/heads/master", self._refs)
  253. self.assertNotIn(b"refs/heads/bar", self._refs)
  254. def test_delitem(self):
  255. self.assertEqual(
  256. b"42d06bd4b77fed026b154d16493e5deab78f02ec",
  257. self._refs[b"refs/heads/master"],
  258. )
  259. del self._refs[b"refs/heads/master"]
  260. self.assertRaises(KeyError, lambda: self._refs[b"refs/heads/master"])
  261. def test_remove_if_equals(self):
  262. self.assertFalse(self._refs.remove_if_equals(b"HEAD", b"c0ffee"))
  263. self.assertEqual(
  264. b"42d06bd4b77fed026b154d16493e5deab78f02ec", self._refs[b"HEAD"]
  265. )
  266. self.assertTrue(
  267. self._refs.remove_if_equals(
  268. b"refs/tags/refs-0.2",
  269. b"3ec9c43c84ff242e3ef4a9fc5bc111fd780a76a8",
  270. )
  271. )
  272. self.assertTrue(self._refs.remove_if_equals(b"refs/tags/refs-0.2", ZERO_SHA))
  273. self.assertNotIn(b"refs/tags/refs-0.2", self._refs)
  274. def test_import_refs_name(self):
  275. self._refs[b"refs/remotes/origin/other"] = (
  276. b"48d01bd4b77fed026b154d16493e5deab78f02ec"
  277. )
  278. self._refs.import_refs(
  279. b"refs/remotes/origin",
  280. {b"master": b"42d06bd4b77fed026b154d16493e5deab78f02ec"},
  281. )
  282. self.assertEqual(
  283. b"42d06bd4b77fed026b154d16493e5deab78f02ec",
  284. self._refs[b"refs/remotes/origin/master"],
  285. )
  286. self.assertEqual(
  287. b"48d01bd4b77fed026b154d16493e5deab78f02ec",
  288. self._refs[b"refs/remotes/origin/other"],
  289. )
  290. def test_import_refs_name_prune(self):
  291. self._refs[b"refs/remotes/origin/other"] = (
  292. b"48d01bd4b77fed026b154d16493e5deab78f02ec"
  293. )
  294. self._refs.import_refs(
  295. b"refs/remotes/origin",
  296. {b"master": b"42d06bd4b77fed026b154d16493e5deab78f02ec"},
  297. prune=True,
  298. )
  299. self.assertEqual(
  300. b"42d06bd4b77fed026b154d16493e5deab78f02ec",
  301. self._refs[b"refs/remotes/origin/master"],
  302. )
  303. self.assertNotIn(b"refs/remotes/origin/other", self._refs)
  304. class DictRefsContainerTests(RefsContainerTests, TestCase):
  305. def setUp(self):
  306. TestCase.setUp(self)
  307. self._refs = DictRefsContainer(dict(_TEST_REFS))
  308. def test_invalid_refname(self):
  309. # FIXME: Move this test into RefsContainerTests, but requires
  310. # some way of injecting invalid refs.
  311. self._refs._refs[b"refs/stash"] = b"00" * 20
  312. expected_refs = dict(_TEST_REFS)
  313. del expected_refs[b"refs/heads/loop"]
  314. expected_refs[b"refs/stash"] = b"00" * 20
  315. self.assertEqual(expected_refs, self._refs.as_dict())
  316. class DiskRefsContainerTests(RefsContainerTests, TestCase):
  317. def setUp(self):
  318. TestCase.setUp(self)
  319. self._repo = open_repo("refs.git")
  320. self.addCleanup(tear_down_repo, self._repo)
  321. self._refs = self._repo.refs
  322. def test_get_packed_refs(self):
  323. self.assertEqual(
  324. {
  325. b"refs/heads/packed": b"42d06bd4b77fed026b154d16493e5deab78f02ec",
  326. b"refs/tags/refs-0.1": b"df6800012397fb85c56e7418dd4eb9405dee075c",
  327. },
  328. self._refs.get_packed_refs(),
  329. )
  330. def test_get_peeled_not_packed(self):
  331. # not packed
  332. self.assertEqual(None, self._refs.get_peeled(b"refs/tags/refs-0.2"))
  333. self.assertEqual(
  334. b"3ec9c43c84ff242e3ef4a9fc5bc111fd780a76a8",
  335. self._refs[b"refs/tags/refs-0.2"],
  336. )
  337. # packed, known not peelable
  338. self.assertEqual(
  339. self._refs[b"refs/heads/packed"],
  340. self._refs.get_peeled(b"refs/heads/packed"),
  341. )
  342. # packed, peeled
  343. self.assertEqual(
  344. b"42d06bd4b77fed026b154d16493e5deab78f02ec",
  345. self._refs.get_peeled(b"refs/tags/refs-0.1"),
  346. )
  347. def test_setitem(self):
  348. RefsContainerTests.test_setitem(self)
  349. path = os.path.join(self._refs.path, b"refs", b"some", b"ref")
  350. with open(path, "rb") as f:
  351. self.assertEqual(b"42d06bd4b77fed026b154d16493e5deab78f02ec", f.read()[:40])
  352. self.assertRaises(
  353. OSError,
  354. self._refs.__setitem__,
  355. b"refs/some/ref/sub",
  356. b"42d06bd4b77fed026b154d16493e5deab78f02ec",
  357. )
  358. def test_delete_refs_container(self):
  359. # We shouldn't delete the refs directory
  360. self._refs[b"refs/heads/blah"] = b"42d06bd4b77fed026b154d16493e5deab78f02ec"
  361. for ref in self._refs.allkeys():
  362. del self._refs[ref]
  363. self.assertTrue(os.path.exists(os.path.join(self._refs.path, b"refs")))
  364. def test_setitem_packed(self):
  365. with open(os.path.join(self._refs.path, b"packed-refs"), "w") as f:
  366. f.write("# pack-refs with: peeled fully-peeled sorted \n")
  367. f.write("42d06bd4b77fed026b154d16493e5deab78f02ec refs/heads/packed\n")
  368. # It's allowed to set a new ref on a packed ref, the new ref will be
  369. # placed outside on refs/
  370. self._refs[b"refs/heads/packed"] = b"3ec9c43c84ff242e3ef4a9fc5bc111fd780a76a8"
  371. packed_ref_path = os.path.join(self._refs.path, b"refs", b"heads", b"packed")
  372. with open(packed_ref_path, "rb") as f:
  373. self.assertEqual(b"3ec9c43c84ff242e3ef4a9fc5bc111fd780a76a8", f.read()[:40])
  374. self.assertRaises(
  375. OSError,
  376. self._refs.__setitem__,
  377. b"refs/heads/packed/sub",
  378. b"42d06bd4b77fed026b154d16493e5deab78f02ec",
  379. )
  380. # this shouldn't overwrite the packed refs
  381. self.assertEqual(
  382. {b"refs/heads/packed": b"42d06bd4b77fed026b154d16493e5deab78f02ec"},
  383. self._refs.get_packed_refs(),
  384. )
  385. def test_add_packed_refs(self):
  386. # first, create a non-packed ref
  387. self._refs[b"refs/heads/packed"] = b"3ec9c43c84ff242e3ef4a9fc5bc111fd780a76a8"
  388. packed_ref_path = os.path.join(self._refs.path, b"refs", b"heads", b"packed")
  389. self.assertTrue(os.path.exists(packed_ref_path))
  390. # now overwrite that with a packed ref
  391. packed_refs_file_path = os.path.join(self._refs.path, b"packed-refs")
  392. self._refs.add_packed_refs(
  393. {
  394. b"refs/heads/packed": b"42d06bd4b77fed026b154d16493e5deab78f02ec",
  395. }
  396. )
  397. # that should kill the file
  398. self.assertFalse(os.path.exists(packed_ref_path))
  399. # now delete the packed ref
  400. self._refs.add_packed_refs(
  401. {
  402. b"refs/heads/packed": None,
  403. }
  404. )
  405. # and it's gone!
  406. self.assertFalse(os.path.exists(packed_ref_path))
  407. self.assertRaises(
  408. KeyError,
  409. self._refs.__getitem__,
  410. b"refs/heads/packed",
  411. )
  412. # just in case, make sure we can't pack HEAD
  413. self.assertRaises(
  414. ValueError,
  415. self._refs.add_packed_refs,
  416. {b"HEAD": "02ac81614bcdbd585a37b4b0edf8cb8a"},
  417. )
  418. # delete all packed refs
  419. self._refs.add_packed_refs({ref: None for ref in self._refs.get_packed_refs()})
  420. self.assertEqual({}, self._refs.get_packed_refs())
  421. # remove the packed ref file, and check that adding nothing doesn't affect that
  422. os.remove(packed_refs_file_path)
  423. # adding nothing doesn't make it reappear
  424. self._refs.add_packed_refs({})
  425. self.assertFalse(os.path.exists(packed_refs_file_path))
  426. def test_setitem_symbolic(self):
  427. ones = b"1" * 40
  428. self._refs[b"HEAD"] = ones
  429. self.assertEqual(ones, self._refs[b"HEAD"])
  430. # ensure HEAD was not modified
  431. f = open(os.path.join(self._refs.path, b"HEAD"), "rb")
  432. v = next(iter(f)).rstrip(b"\n\r")
  433. f.close()
  434. self.assertEqual(b"ref: refs/heads/master", v)
  435. # ensure the symbolic link was written through
  436. f = open(os.path.join(self._refs.path, b"refs", b"heads", b"master"), "rb")
  437. self.assertEqual(ones, f.read()[:40])
  438. f.close()
  439. def test_set_if_equals(self):
  440. RefsContainerTests.test_set_if_equals(self)
  441. # ensure symref was followed
  442. self.assertEqual(b"9" * 40, self._refs[b"refs/heads/master"])
  443. # ensure lockfile was deleted
  444. self.assertFalse(
  445. os.path.exists(
  446. os.path.join(self._refs.path, b"refs", b"heads", b"master.lock")
  447. )
  448. )
  449. self.assertFalse(os.path.exists(os.path.join(self._refs.path, b"HEAD.lock")))
  450. def test_add_if_new_packed(self):
  451. # don't overwrite packed ref
  452. self.assertFalse(self._refs.add_if_new(b"refs/tags/refs-0.1", b"9" * 40))
  453. self.assertEqual(
  454. b"df6800012397fb85c56e7418dd4eb9405dee075c",
  455. self._refs[b"refs/tags/refs-0.1"],
  456. )
  457. def test_add_if_new_symbolic(self):
  458. # Use an empty repo instead of the default.
  459. repo_dir = os.path.join(tempfile.mkdtemp(), "test")
  460. os.makedirs(repo_dir)
  461. repo = Repo.init(repo_dir)
  462. self.addCleanup(tear_down_repo, repo)
  463. refs = repo.refs
  464. nines = b"9" * 40
  465. self.assertEqual(b"ref: refs/heads/master", refs.read_ref(b"HEAD"))
  466. self.assertNotIn(b"refs/heads/master", refs)
  467. self.assertTrue(refs.add_if_new(b"HEAD", nines))
  468. self.assertEqual(b"ref: refs/heads/master", refs.read_ref(b"HEAD"))
  469. self.assertEqual(nines, refs[b"HEAD"])
  470. self.assertEqual(nines, refs[b"refs/heads/master"])
  471. self.assertFalse(refs.add_if_new(b"HEAD", b"1" * 40))
  472. self.assertEqual(nines, refs[b"HEAD"])
  473. self.assertEqual(nines, refs[b"refs/heads/master"])
  474. def test_follow(self):
  475. self.assertEqual(
  476. (
  477. [b"HEAD", b"refs/heads/master"],
  478. b"42d06bd4b77fed026b154d16493e5deab78f02ec",
  479. ),
  480. self._refs.follow(b"HEAD"),
  481. )
  482. self.assertEqual(
  483. (
  484. [b"refs/heads/master"],
  485. b"42d06bd4b77fed026b154d16493e5deab78f02ec",
  486. ),
  487. self._refs.follow(b"refs/heads/master"),
  488. )
  489. self.assertRaises(SymrefLoop, self._refs.follow, b"refs/heads/loop")
  490. def test_set_overwrite_loop(self):
  491. self.assertRaises(SymrefLoop, self._refs.follow, b"refs/heads/loop")
  492. self._refs[b"refs/heads/loop"] = b"42d06bd4b77fed026b154d16493e5deab78f02ec"
  493. self.assertEqual(
  494. ([b"refs/heads/loop"], b"42d06bd4b77fed026b154d16493e5deab78f02ec"),
  495. self._refs.follow(b"refs/heads/loop"),
  496. )
  497. def test_delitem(self):
  498. RefsContainerTests.test_delitem(self)
  499. ref_file = os.path.join(self._refs.path, b"refs", b"heads", b"master")
  500. self.assertFalse(os.path.exists(ref_file))
  501. self.assertNotIn(b"refs/heads/master", self._refs.get_packed_refs())
  502. def test_delitem_symbolic(self):
  503. self.assertEqual(b"ref: refs/heads/master", self._refs.read_loose_ref(b"HEAD"))
  504. del self._refs[b"HEAD"]
  505. self.assertRaises(KeyError, lambda: self._refs[b"HEAD"])
  506. self.assertEqual(
  507. b"42d06bd4b77fed026b154d16493e5deab78f02ec",
  508. self._refs[b"refs/heads/master"],
  509. )
  510. self.assertFalse(os.path.exists(os.path.join(self._refs.path, b"HEAD")))
  511. def test_remove_if_equals_symref(self):
  512. # HEAD is a symref, so shouldn't equal its dereferenced value
  513. self.assertFalse(
  514. self._refs.remove_if_equals(
  515. b"HEAD", b"42d06bd4b77fed026b154d16493e5deab78f02ec"
  516. )
  517. )
  518. self.assertTrue(
  519. self._refs.remove_if_equals(
  520. b"refs/heads/master",
  521. b"42d06bd4b77fed026b154d16493e5deab78f02ec",
  522. )
  523. )
  524. self.assertRaises(KeyError, lambda: self._refs[b"refs/heads/master"])
  525. # HEAD is now a broken symref
  526. self.assertRaises(KeyError, lambda: self._refs[b"HEAD"])
  527. self.assertEqual(b"ref: refs/heads/master", self._refs.read_loose_ref(b"HEAD"))
  528. self.assertFalse(
  529. os.path.exists(
  530. os.path.join(self._refs.path, b"refs", b"heads", b"master.lock")
  531. )
  532. )
  533. self.assertFalse(os.path.exists(os.path.join(self._refs.path, b"HEAD.lock")))
  534. def test_remove_packed_without_peeled(self):
  535. refs_file = os.path.join(self._repo.path, "packed-refs")
  536. f = GitFile(refs_file)
  537. refs_data = f.read()
  538. f.close()
  539. f = GitFile(refs_file, "wb")
  540. f.write(
  541. b"\n".join(
  542. line
  543. for line in refs_data.split(b"\n")
  544. if not line or line[0] not in b"#^"
  545. )
  546. )
  547. f.close()
  548. self._repo = Repo(self._repo.path)
  549. refs = self._repo.refs
  550. self.assertTrue(
  551. refs.remove_if_equals(
  552. b"refs/heads/packed",
  553. b"42d06bd4b77fed026b154d16493e5deab78f02ec",
  554. )
  555. )
  556. def test_remove_if_equals_packed(self):
  557. # test removing ref that is only packed
  558. self.assertEqual(
  559. b"df6800012397fb85c56e7418dd4eb9405dee075c",
  560. self._refs[b"refs/tags/refs-0.1"],
  561. )
  562. self.assertTrue(
  563. self._refs.remove_if_equals(
  564. b"refs/tags/refs-0.1",
  565. b"df6800012397fb85c56e7418dd4eb9405dee075c",
  566. )
  567. )
  568. self.assertRaises(KeyError, lambda: self._refs[b"refs/tags/refs-0.1"])
  569. def test_remove_parent(self):
  570. self._refs[b"refs/heads/foo/bar"] = b"df6800012397fb85c56e7418dd4eb9405dee075c"
  571. del self._refs[b"refs/heads/foo/bar"]
  572. ref_file = os.path.join(
  573. self._refs.path,
  574. b"refs",
  575. b"heads",
  576. b"foo",
  577. b"bar",
  578. )
  579. self.assertFalse(os.path.exists(ref_file))
  580. ref_file = os.path.join(self._refs.path, b"refs", b"heads", b"foo")
  581. self.assertFalse(os.path.exists(ref_file))
  582. ref_file = os.path.join(self._refs.path, b"refs", b"heads")
  583. self.assertTrue(os.path.exists(ref_file))
  584. self._refs[b"refs/heads/foo"] = b"df6800012397fb85c56e7418dd4eb9405dee075c"
  585. def test_read_ref(self):
  586. self.assertEqual(b"ref: refs/heads/master", self._refs.read_ref(b"HEAD"))
  587. self.assertEqual(
  588. b"42d06bd4b77fed026b154d16493e5deab78f02ec",
  589. self._refs.read_ref(b"refs/heads/packed"),
  590. )
  591. self.assertEqual(None, self._refs.read_ref(b"nonexistent"))
  592. def test_read_loose_ref(self):
  593. self._refs[b"refs/heads/foo"] = b"df6800012397fb85c56e7418dd4eb9405dee075c"
  594. self.assertEqual(None, self._refs.read_ref(b"refs/heads/foo/bar"))
  595. def test_non_ascii(self):
  596. try:
  597. encoded_ref = os.fsencode("refs/tags/schön")
  598. except UnicodeEncodeError as exc:
  599. raise SkipTest(
  600. "filesystem encoding doesn't support special character"
  601. ) from exc
  602. p = os.path.join(os.fsencode(self._repo.path), encoded_ref)
  603. with open(p, "w") as f:
  604. f.write("00" * 20)
  605. expected_refs = dict(_TEST_REFS)
  606. expected_refs[encoded_ref] = b"00" * 20
  607. del expected_refs[b"refs/heads/loop"]
  608. self.assertEqual(expected_refs, self._repo.get_refs())
  609. def test_cyrillic(self):
  610. if sys.platform in ("darwin", "win32"):
  611. raise SkipTest("filesystem encoding doesn't support arbitrary bytes")
  612. # reported in https://github.com/dulwich/dulwich/issues/608
  613. name = b"\xcd\xee\xe2\xe0\xff\xe2\xe5\xf2\xea\xe01"
  614. encoded_ref = b"refs/heads/" + name
  615. with open(os.path.join(os.fsencode(self._repo.path), encoded_ref), "w") as f:
  616. f.write("00" * 20)
  617. expected_refs = set(_TEST_REFS.keys())
  618. expected_refs.add(encoded_ref)
  619. self.assertEqual(expected_refs, set(self._repo.refs.allkeys()))
  620. self.assertEqual(
  621. {r[len(b"refs/") :] for r in expected_refs if r.startswith(b"refs/")},
  622. set(self._repo.refs.subkeys(b"refs/")),
  623. )
  624. expected_refs.remove(b"refs/heads/loop")
  625. expected_refs.add(b"HEAD")
  626. self.assertEqual(expected_refs, set(self._repo.get_refs().keys()))
  627. _TEST_REFS_SERIALIZED = (
  628. b"42d06bd4b77fed026b154d16493e5deab78f02ec\t"
  629. b"refs/heads/40-char-ref-aaaaaaaaaaaaaaaaaa\n"
  630. b"42d06bd4b77fed026b154d16493e5deab78f02ec\trefs/heads/master\n"
  631. b"42d06bd4b77fed026b154d16493e5deab78f02ec\trefs/heads/packed\n"
  632. b"df6800012397fb85c56e7418dd4eb9405dee075c\trefs/tags/refs-0.1\n"
  633. b"3ec9c43c84ff242e3ef4a9fc5bc111fd780a76a8\trefs/tags/refs-0.2\n"
  634. )
  635. class InfoRefsContainerTests(TestCase):
  636. def test_invalid_refname(self):
  637. text = _TEST_REFS_SERIALIZED + b"00" * 20 + b"\trefs/stash\n"
  638. refs = InfoRefsContainer(BytesIO(text))
  639. expected_refs = dict(_TEST_REFS)
  640. del expected_refs[b"HEAD"]
  641. expected_refs[b"refs/stash"] = b"00" * 20
  642. del expected_refs[b"refs/heads/loop"]
  643. self.assertEqual(expected_refs, refs.as_dict())
  644. def test_keys(self):
  645. refs = InfoRefsContainer(BytesIO(_TEST_REFS_SERIALIZED))
  646. actual_keys = set(refs.keys())
  647. self.assertEqual(set(refs.allkeys()), actual_keys)
  648. expected_refs = dict(_TEST_REFS)
  649. del expected_refs[b"HEAD"]
  650. del expected_refs[b"refs/heads/loop"]
  651. self.assertEqual(set(expected_refs.keys()), actual_keys)
  652. actual_keys = refs.keys(b"refs/heads")
  653. actual_keys.discard(b"loop")
  654. self.assertEqual(
  655. [b"40-char-ref-aaaaaaaaaaaaaaaaaa", b"master", b"packed"],
  656. sorted(actual_keys),
  657. )
  658. self.assertEqual([b"refs-0.1", b"refs-0.2"], sorted(refs.keys(b"refs/tags")))
  659. def test_as_dict(self):
  660. refs = InfoRefsContainer(BytesIO(_TEST_REFS_SERIALIZED))
  661. # refs/heads/loop does not show up even if it exists
  662. expected_refs = dict(_TEST_REFS)
  663. del expected_refs[b"HEAD"]
  664. del expected_refs[b"refs/heads/loop"]
  665. self.assertEqual(expected_refs, refs.as_dict())
  666. def test_contains(self):
  667. refs = InfoRefsContainer(BytesIO(_TEST_REFS_SERIALIZED))
  668. self.assertIn(b"refs/heads/master", refs)
  669. self.assertNotIn(b"refs/heads/bar", refs)
  670. def test_get_peeled(self):
  671. refs = InfoRefsContainer(BytesIO(_TEST_REFS_SERIALIZED))
  672. # refs/heads/loop does not show up even if it exists
  673. self.assertEqual(
  674. _TEST_REFS[b"refs/heads/master"],
  675. refs.get_peeled(b"refs/heads/master"),
  676. )
  677. class ParseSymrefValueTests(TestCase):
  678. def test_valid(self):
  679. self.assertEqual(b"refs/heads/foo", parse_symref_value(b"ref: refs/heads/foo"))
  680. def test_invalid(self):
  681. self.assertRaises(ValueError, parse_symref_value, b"foobar")
  682. class StripPeeledRefsTests(TestCase):
  683. all_refs: ClassVar[Dict[bytes, bytes]] = {
  684. b"refs/heads/master": b"8843d7f92416211de9ebb963ff4ce28125932878",
  685. b"refs/heads/testing": b"186a005b134d8639a58b6731c7c1ea821a6eedba",
  686. b"refs/tags/1.0.0": b"a93db4b0360cc635a2b93675010bac8d101f73f0",
  687. b"refs/tags/1.0.0^{}": b"a93db4b0360cc635a2b93675010bac8d101f73f0",
  688. b"refs/tags/2.0.0": b"0749936d0956c661ac8f8d3483774509c165f89e",
  689. b"refs/tags/2.0.0^{}": b"0749936d0956c661ac8f8d3483774509c165f89e",
  690. }
  691. non_peeled_refs: ClassVar[Dict[bytes, bytes]] = {
  692. b"refs/heads/master": b"8843d7f92416211de9ebb963ff4ce28125932878",
  693. b"refs/heads/testing": b"186a005b134d8639a58b6731c7c1ea821a6eedba",
  694. b"refs/tags/1.0.0": b"a93db4b0360cc635a2b93675010bac8d101f73f0",
  695. b"refs/tags/2.0.0": b"0749936d0956c661ac8f8d3483774509c165f89e",
  696. }
  697. def test_strip_peeled_refs(self):
  698. # Simple check of two dicts
  699. self.assertEqual(strip_peeled_refs(self.all_refs), self.non_peeled_refs)