test_repository.py 7.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209
  1. # test_repo.py -- Git repo compatibility tests
  2. # Copyright (C) 2010 Google, Inc.
  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. """Compatibility tests for dulwich repositories."""
  21. from io import BytesIO
  22. from itertools import chain
  23. import os
  24. import tempfile
  25. from dulwich.objects import (
  26. hex_to_sha,
  27. )
  28. from dulwich.repo import (
  29. check_ref_format,
  30. Repo,
  31. )
  32. from dulwich.tests.compat.utils import (
  33. rmtree_ro,
  34. run_git_or_fail,
  35. CompatTestCase,
  36. )
  37. class ObjectStoreTestCase(CompatTestCase):
  38. """Tests for git repository compatibility."""
  39. def setUp(self):
  40. super(ObjectStoreTestCase, self).setUp()
  41. self._repo = self.import_repo('server_new.export')
  42. def _run_git(self, args):
  43. return run_git_or_fail(args, cwd=self._repo.path)
  44. def _parse_refs(self, output):
  45. refs = {}
  46. for line in BytesIO(output):
  47. fields = line.rstrip(b'\n').split(b' ')
  48. self.assertEqual(3, len(fields))
  49. refname, type_name, sha = fields
  50. check_ref_format(refname[5:])
  51. hex_to_sha(sha)
  52. refs[refname] = (type_name, sha)
  53. return refs
  54. def _parse_objects(self, output):
  55. return set(s.rstrip(b'\n').split(b' ')[0] for s in BytesIO(output))
  56. def test_bare(self):
  57. self.assertTrue(self._repo.bare)
  58. self.assertFalse(os.path.exists(os.path.join(self._repo.path, '.git')))
  59. def test_head(self):
  60. output = self._run_git(['rev-parse', 'HEAD'])
  61. head_sha = output.rstrip(b'\n')
  62. hex_to_sha(head_sha)
  63. self.assertEqual(head_sha, self._repo.refs[b'HEAD'])
  64. def test_refs(self):
  65. output = self._run_git(
  66. ['for-each-ref', '--format=%(refname) %(objecttype) %(objectname)'])
  67. expected_refs = self._parse_refs(output)
  68. actual_refs = {}
  69. for refname, sha in self._repo.refs.as_dict().items():
  70. if refname == b'HEAD':
  71. continue # handled in test_head
  72. obj = self._repo[sha]
  73. self.assertEqual(sha, obj.id)
  74. actual_refs[refname] = (obj.type_name, obj.id)
  75. self.assertEqual(expected_refs, actual_refs)
  76. # TODO(dborowitz): peeled ref tests
  77. def _get_loose_shas(self):
  78. output = self._run_git(['rev-list', '--all', '--objects', '--unpacked'])
  79. return self._parse_objects(output)
  80. def _get_all_shas(self):
  81. output = self._run_git(['rev-list', '--all', '--objects'])
  82. return self._parse_objects(output)
  83. def assertShasMatch(self, expected_shas, actual_shas_iter):
  84. actual_shas = set()
  85. for sha in actual_shas_iter:
  86. obj = self._repo[sha]
  87. self.assertEqual(sha, obj.id)
  88. actual_shas.add(sha)
  89. self.assertEqual(expected_shas, actual_shas)
  90. def test_loose_objects(self):
  91. # TODO(dborowitz): This is currently not very useful since fast-imported
  92. # repos only contained packed objects.
  93. expected_shas = self._get_loose_shas()
  94. self.assertShasMatch(expected_shas,
  95. self._repo.object_store._iter_loose_objects())
  96. def test_packed_objects(self):
  97. expected_shas = self._get_all_shas() - self._get_loose_shas()
  98. self.assertShasMatch(expected_shas,
  99. chain(*self._repo.object_store.packs))
  100. def test_all_objects(self):
  101. expected_shas = self._get_all_shas()
  102. self.assertShasMatch(expected_shas, iter(self._repo.object_store))
  103. class WorkingTreeTestCase(ObjectStoreTestCase):
  104. """Test for compatibility with git-worktree."""
  105. min_git_version = (2, 5, 0)
  106. def create_new_worktree(self, repo_dir, branch):
  107. """Create a new worktree using git-worktree.
  108. :param repo_dir: The directory of the main working tree.
  109. :param branch: The branch or commit to checkout in the new worktree.
  110. :returns: The path to the new working tree.
  111. """
  112. temp_dir = tempfile.mkdtemp()
  113. run_git_or_fail(['worktree', 'add', temp_dir, branch],
  114. cwd=repo_dir)
  115. self.addCleanup(rmtree_ro, temp_dir)
  116. return temp_dir
  117. def setUp(self):
  118. super(WorkingTreeTestCase, self).setUp()
  119. self._worktree_path = self.create_new_worktree(self._repo.path, 'branch')
  120. self._worktree_repo = Repo(self._worktree_path)
  121. self.addCleanup(self._worktree_repo.close)
  122. self._mainworktree_repo = self._repo
  123. self._number_of_working_tree = 2
  124. self._repo = self._worktree_repo
  125. def test_refs(self):
  126. super(WorkingTreeTestCase, self).test_refs()
  127. self.assertEqual(self._mainworktree_repo.refs.allkeys(),
  128. self._repo.refs.allkeys())
  129. def test_head_equality(self):
  130. self.assertNotEqual(self._repo.refs[b'HEAD'],
  131. self._mainworktree_repo.refs[b'HEAD'])
  132. def test_bare(self):
  133. self.assertFalse(self._repo.bare)
  134. self.assertTrue(os.path.isfile(os.path.join(self._repo.path, '.git')))
  135. def _parse_worktree_list(self, output):
  136. worktrees = []
  137. for line in BytesIO(output):
  138. fields = line.rstrip(b'\n').split()
  139. worktrees.append(tuple(f.decode() for f in fields))
  140. return worktrees
  141. def test_git_worktree_list(self):
  142. output = run_git_or_fail(['worktree', 'list'], cwd=self._repo.path)
  143. worktrees = self._parse_worktree_list(output)
  144. self.assertEqual(len(worktrees), self._number_of_working_tree)
  145. self.assertEqual(worktrees[0][1], '(bare)')
  146. self.assertEqual(worktrees[0][0], self._mainworktree_repo.path)
  147. output = run_git_or_fail(['worktree', 'list'], cwd=self._mainworktree_repo.path)
  148. worktrees = self._parse_worktree_list(output)
  149. self.assertEqual(len(worktrees), self._number_of_working_tree)
  150. self.assertEqual(worktrees[0][1], '(bare)')
  151. self.assertEqual(worktrees[0][0], self._mainworktree_repo.path)
  152. class InitNewWorkingDirectoryTestCase(WorkingTreeTestCase):
  153. """Test compatibility of Repo.init_new_working_directory."""
  154. min_git_version = (2, 5, 0)
  155. def setUp(self):
  156. super(InitNewWorkingDirectoryTestCase, self).setUp()
  157. self._other_worktree = self._repo
  158. worktree_repo_path = tempfile.mkdtemp()
  159. self.addCleanup(rmtree_ro, worktree_repo_path)
  160. self._repo = Repo._init_new_working_directory(
  161. worktree_repo_path, self._mainworktree_repo)
  162. self.addCleanup(self._repo.close)
  163. self._number_of_working_tree = 3
  164. def test_head_equality(self):
  165. self.assertEqual(self._repo.refs[b'HEAD'],
  166. self._mainworktree_repo.refs[b'HEAD'])
  167. def test_bare(self):
  168. self.assertFalse(self._repo.bare)
  169. self.assertTrue(os.path.isfile(os.path.join(self._repo.path, '.git')))