test_maintenance.py 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314
  1. # test_maintenance.py -- tests for maintenance functionality
  2. # Copyright (C) 2024 Jelmer Vernooij <jelmer@jelmer.uk>
  3. #
  4. # SPDX-License-Identifier: Apache-2.0 OR GPL-2.0-or-later
  5. # Dulwich is dual-licensed under the Apache License, Version 2.0 and the GNU
  6. # General Public License as published 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.maintenance."""
  22. import tempfile
  23. from dulwich import porcelain
  24. from dulwich.maintenance import (
  25. CommitGraphTask,
  26. GcTask,
  27. IncrementalRepackTask,
  28. LooseObjectsTask,
  29. PackRefsTask,
  30. PrefetchTask,
  31. get_enabled_tasks,
  32. run_maintenance,
  33. )
  34. from dulwich.objects import Blob
  35. from dulwich.repo import Repo
  36. from . import TestCase
  37. class MaintenanceTaskTestCase(TestCase):
  38. """Base class for maintenance task tests."""
  39. def setUp(self):
  40. super().setUp()
  41. self.test_dir = tempfile.mkdtemp()
  42. self.addCleanup(self._cleanup_test_dir)
  43. self.repo = Repo.init(self.test_dir)
  44. self.addCleanup(self.repo.close)
  45. def _cleanup_test_dir(self):
  46. import shutil
  47. shutil.rmtree(self.test_dir)
  48. def _create_commit(self):
  49. """Create a simple commit in the test repository."""
  50. blob = Blob.from_string(b"test content")
  51. self.repo.object_store.add_object(blob)
  52. return blob
  53. class GcTaskTest(MaintenanceTaskTestCase):
  54. """Tests for GcTask."""
  55. def test_default_enabled(self):
  56. """Test that GC task is enabled by default."""
  57. task = GcTask(self.repo)
  58. self.assertTrue(task.default_enabled())
  59. self.assertTrue(task.is_enabled())
  60. def test_run(self):
  61. """Test running GC task."""
  62. self._create_commit()
  63. task = GcTask(self.repo)
  64. result = task.run()
  65. self.assertTrue(result)
  66. class CommitGraphTaskTest(MaintenanceTaskTestCase):
  67. """Tests for CommitGraphTask."""
  68. def test_default_enabled(self):
  69. """Test that commit-graph task is enabled by default."""
  70. task = CommitGraphTask(self.repo)
  71. self.assertTrue(task.default_enabled())
  72. self.assertTrue(task.is_enabled())
  73. def test_run(self):
  74. """Test running commit-graph task."""
  75. self._create_commit()
  76. task = CommitGraphTask(self.repo)
  77. result = task.run()
  78. self.assertTrue(result)
  79. class LooseObjectsTaskTest(MaintenanceTaskTestCase):
  80. """Tests for LooseObjectsTask."""
  81. def test_default_enabled(self):
  82. """Test that loose-objects task is disabled by default."""
  83. task = LooseObjectsTask(self.repo)
  84. self.assertFalse(task.default_enabled())
  85. def test_run(self):
  86. """Test running loose-objects task."""
  87. self._create_commit()
  88. task = LooseObjectsTask(self.repo)
  89. result = task.run()
  90. self.assertTrue(result)
  91. class IncrementalRepackTaskTest(MaintenanceTaskTestCase):
  92. """Tests for IncrementalRepackTask."""
  93. def test_default_enabled(self):
  94. """Test that incremental-repack task is disabled by default."""
  95. task = IncrementalRepackTask(self.repo)
  96. self.assertFalse(task.default_enabled())
  97. def test_run_no_packs(self):
  98. """Test running incremental-repack with no packs."""
  99. task = IncrementalRepackTask(self.repo)
  100. result = task.run()
  101. self.assertTrue(result)
  102. def test_run_auto_few_packs(self):
  103. """Test that auto mode skips repacking when there are few packs."""
  104. self._create_commit()
  105. task = IncrementalRepackTask(self.repo, auto=True)
  106. result = task.run()
  107. self.assertTrue(result)
  108. class PackRefsTaskTest(MaintenanceTaskTestCase):
  109. """Tests for PackRefsTask."""
  110. def test_default_enabled(self):
  111. """Test that pack-refs task is disabled by default."""
  112. task = PackRefsTask(self.repo)
  113. self.assertFalse(task.default_enabled())
  114. def test_run(self):
  115. """Test running pack-refs task."""
  116. task = PackRefsTask(self.repo)
  117. result = task.run()
  118. self.assertTrue(result)
  119. class PrefetchTaskTest(MaintenanceTaskTestCase):
  120. """Tests for PrefetchTask."""
  121. def test_default_enabled(self):
  122. """Test that prefetch task is disabled by default."""
  123. task = PrefetchTask(self.repo)
  124. self.assertFalse(task.default_enabled())
  125. def test_run_no_remotes(self):
  126. """Test running prefetch with no remotes configured."""
  127. task = PrefetchTask(self.repo)
  128. result = task.run()
  129. self.assertTrue(result)
  130. class MaintenanceFunctionsTest(MaintenanceTaskTestCase):
  131. """Tests for maintenance module functions."""
  132. def test_get_enabled_tasks_default(self):
  133. """Test getting enabled tasks with defaults."""
  134. enabled = get_enabled_tasks(self.repo)
  135. # By default, only gc and commit-graph are enabled
  136. self.assertIn("gc", enabled)
  137. self.assertIn("commit-graph", enabled)
  138. self.assertNotIn("loose-objects", enabled)
  139. self.assertNotIn("incremental-repack", enabled)
  140. self.assertNotIn("pack-refs", enabled)
  141. self.assertNotIn("prefetch", enabled)
  142. def test_get_enabled_tasks_with_filter(self):
  143. """Test getting enabled tasks with a filter."""
  144. enabled = get_enabled_tasks(self.repo, ["gc", "pack-refs"])
  145. self.assertEqual(set(enabled), {"gc", "pack-refs"})
  146. def test_get_enabled_tasks_invalid(self):
  147. """Test that invalid task names are ignored."""
  148. enabled = get_enabled_tasks(self.repo, ["gc", "invalid-task"])
  149. self.assertEqual(enabled, ["gc"])
  150. def test_run_maintenance(self):
  151. """Test running maintenance tasks."""
  152. self._create_commit()
  153. result = run_maintenance(self.repo)
  154. self.assertIn("gc", result.tasks_run)
  155. self.assertIn("commit-graph", result.tasks_run)
  156. self.assertIn("gc", result.tasks_succeeded)
  157. self.assertIn("commit-graph", result.tasks_succeeded)
  158. self.assertEqual(len(result.tasks_failed), 0)
  159. def test_run_maintenance_specific_tasks(self):
  160. """Test running specific maintenance tasks."""
  161. result = run_maintenance(self.repo, tasks=["pack-refs"])
  162. self.assertEqual(result.tasks_run, ["pack-refs"])
  163. self.assertEqual(result.tasks_succeeded, ["pack-refs"])
  164. self.assertEqual(len(result.tasks_failed), 0)
  165. def test_run_maintenance_with_progress(self):
  166. """Test running maintenance with progress callback."""
  167. messages = []
  168. def progress(msg):
  169. messages.append(msg)
  170. self._create_commit()
  171. result = run_maintenance(self.repo, progress=progress)
  172. self.assertGreater(len(messages), 0)
  173. self.assertIn("gc", result.tasks_succeeded)
  174. class PorcelainMaintenanceTest(MaintenanceTaskTestCase):
  175. """Tests for porcelain.maintenance_run function."""
  176. def test_maintenance_run(self):
  177. """Test porcelain maintenance_run function."""
  178. self._create_commit()
  179. result = porcelain.maintenance_run(self.test_dir)
  180. self.assertIn("gc", result.tasks_succeeded)
  181. self.assertIn("commit-graph", result.tasks_succeeded)
  182. def test_maintenance_run_with_tasks(self):
  183. """Test porcelain maintenance_run with specific tasks."""
  184. result = porcelain.maintenance_run(self.test_dir, tasks=["pack-refs"])
  185. self.assertEqual(result.tasks_run, ["pack-refs"])
  186. self.assertEqual(result.tasks_succeeded, ["pack-refs"])
  187. class MaintenanceRegisterTest(MaintenanceTaskTestCase):
  188. """Tests for maintenance register/unregister."""
  189. def setUp(self):
  190. super().setUp()
  191. # Set up a temporary HOME for testing global config
  192. self.temp_home = tempfile.mkdtemp()
  193. self.addCleanup(self._cleanup_temp_home)
  194. self.overrideEnv("HOME", self.temp_home)
  195. def _cleanup_temp_home(self):
  196. import shutil
  197. shutil.rmtree(self.temp_home)
  198. def test_register_repository(self):
  199. """Test registering a repository for maintenance."""
  200. porcelain.maintenance_register(self.test_dir)
  201. # Verify repository was added to global config
  202. import os
  203. from dulwich.config import ConfigFile
  204. global_config_path = os.path.expanduser("~/.gitconfig")
  205. global_config = ConfigFile.from_path(global_config_path)
  206. repos = list(global_config.get_multivar((b"maintenance",), b"repo"))
  207. self.assertIn(self.test_dir.encode(), repos)
  208. # Verify strategy was set
  209. strategy = global_config.get((b"maintenance",), b"strategy")
  210. self.assertEqual(strategy, b"incremental")
  211. # Verify auto maintenance was disabled in repo
  212. repo_config = self.repo.get_config()
  213. auto = repo_config.get_boolean((b"maintenance",), b"auto")
  214. self.assertFalse(auto)
  215. def test_register_already_registered(self):
  216. """Test registering an already registered repository."""
  217. porcelain.maintenance_register(self.test_dir)
  218. # Should not error when registering again
  219. porcelain.maintenance_register(self.test_dir)
  220. def test_unregister_repository(self):
  221. """Test unregistering a repository."""
  222. # First register
  223. porcelain.maintenance_register(self.test_dir)
  224. # Then unregister
  225. porcelain.maintenance_unregister(self.test_dir)
  226. # Verify repository was removed from global config
  227. import os
  228. from dulwich.config import ConfigFile
  229. global_config_path = os.path.expanduser("~/.gitconfig")
  230. global_config = ConfigFile.from_path(global_config_path)
  231. try:
  232. repos = list(global_config.get_multivar((b"maintenance",), b"repo"))
  233. self.assertNotIn(self.test_dir.encode(), repos)
  234. except KeyError:
  235. # No repos registered, which is fine
  236. pass
  237. def test_unregister_not_registered(self):
  238. """Test unregistering a repository that is not registered."""
  239. with self.assertRaises(ValueError):
  240. porcelain.maintenance_unregister(self.test_dir)
  241. def test_unregister_not_registered_force(self):
  242. """Test unregistering with force flag."""
  243. # Should not error with force=True
  244. porcelain.maintenance_unregister(self.test_dir, force=True)