test_graph.py 5.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182
  1. # test_index.py -- Tests for merge
  2. # encoding: utf-8
  3. # Copyright (c) 2020 Kevin B. Hendricks, Stratford Ontario Canada
  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. """Tests for dulwich.graph."""
  21. from dulwich.graph import _find_lcas, can_fast_forward
  22. from dulwich.repo import MemoryRepo
  23. from dulwich.tests import TestCase
  24. from dulwich.tests.utils import make_commit
  25. class FindMergeBaseTests(TestCase):
  26. @staticmethod
  27. def run_test(dag, inputs):
  28. def lookup_parents(commit_id):
  29. return dag[commit_id]
  30. c1 = inputs[0]
  31. c2s = inputs[1:]
  32. return set(_find_lcas(lookup_parents, c1, c2s))
  33. def test_multiple_lca(self):
  34. # two lowest common ancestors
  35. graph = {
  36. "5": ["1", "2"],
  37. "4": ["3", "1"],
  38. "3": ["2"],
  39. "2": ["0"],
  40. "1": [],
  41. "0": [],
  42. }
  43. self.assertEqual(self.run_test(graph, ["4", "5"]), {"1", "2"})
  44. def test_no_common_ancestor(self):
  45. # no common ancestor
  46. graph = {
  47. "4": ["2"],
  48. "3": ["1"],
  49. "2": [],
  50. "1": ["0"],
  51. "0": [],
  52. }
  53. self.assertEqual(self.run_test(graph, ["4", "3"]), set())
  54. def test_ancestor(self):
  55. # ancestor
  56. graph = {
  57. "G": ["D", "F"],
  58. "F": ["E"],
  59. "D": ["C"],
  60. "C": ["B"],
  61. "E": ["B"],
  62. "B": ["A"],
  63. "A": [],
  64. }
  65. self.assertEqual(self.run_test(graph, ["D", "C"]), {"C"})
  66. def test_direct_parent(self):
  67. # parent
  68. graph = {
  69. "G": ["D", "F"],
  70. "F": ["E"],
  71. "D": ["C"],
  72. "C": ["B"],
  73. "E": ["B"],
  74. "B": ["A"],
  75. "A": [],
  76. }
  77. self.assertEqual(self.run_test(graph, ["G", "D"]), {"D"})
  78. def test_another_crossover(self):
  79. # Another cross over
  80. graph = {
  81. "G": ["D", "F"],
  82. "F": ["E", "C"],
  83. "D": ["C", "E"],
  84. "C": ["B"],
  85. "E": ["B"],
  86. "B": ["A"],
  87. "A": [],
  88. }
  89. self.assertEqual(self.run_test(graph, ["D", "F"]), {"E", "C"})
  90. def test_three_way_merge_lca(self):
  91. # three way merge commit straight from git docs
  92. graph = {
  93. "C": ["C1"],
  94. "C1": ["C2"],
  95. "C2": ["C3"],
  96. "C3": ["C4"],
  97. "C4": ["2"],
  98. "B": ["B1"],
  99. "B1": ["B2"],
  100. "B2": ["B3"],
  101. "B3": ["1"],
  102. "A": ["A1"],
  103. "A1": ["A2"],
  104. "A2": ["A3"],
  105. "A3": ["1"],
  106. "1": ["2"],
  107. "2": [],
  108. }
  109. # assumes a theoretical merge M exists that merges B and C first
  110. # which actually means find the first LCA from either of B OR C with A
  111. self.assertEqual(self.run_test(graph, ["A", "B", "C"]), {"1"})
  112. def test_octopus(self):
  113. # octopus algorithm test
  114. # test straight from git docs of A, B, and C
  115. # but this time use octopus to find lcas of A, B, and C simultaneously
  116. graph = {
  117. "C": ["C1"],
  118. "C1": ["C2"],
  119. "C2": ["C3"],
  120. "C3": ["C4"],
  121. "C4": ["2"],
  122. "B": ["B1"],
  123. "B1": ["B2"],
  124. "B2": ["B3"],
  125. "B3": ["1"],
  126. "A": ["A1"],
  127. "A1": ["A2"],
  128. "A2": ["A3"],
  129. "A3": ["1"],
  130. "1": ["2"],
  131. "2": [],
  132. }
  133. def lookup_parents(cid):
  134. return graph[cid]
  135. lcas = ["A"]
  136. others = ["B", "C"]
  137. for cmt in others:
  138. next_lcas = []
  139. for ca in lcas:
  140. res = _find_lcas(lookup_parents, cmt, [ca])
  141. next_lcas.extend(res)
  142. lcas = next_lcas[:]
  143. self.assertEqual(set(lcas), {"2"})
  144. class CanFastForwardTests(TestCase):
  145. def test_ff(self):
  146. r = MemoryRepo()
  147. base = make_commit()
  148. c1 = make_commit(parents=[base.id])
  149. c2 = make_commit(parents=[c1.id])
  150. r.object_store.add_objects([(base, None), (c1, None), (c2, None)])
  151. self.assertTrue(can_fast_forward(r, c1.id, c1.id))
  152. self.assertTrue(can_fast_forward(r, base.id, c1.id))
  153. self.assertTrue(can_fast_forward(r, c1.id, c2.id))
  154. self.assertFalse(can_fast_forward(r, c2.id, c1.id))
  155. def test_diverged(self):
  156. r = MemoryRepo()
  157. base = make_commit()
  158. c1 = make_commit(parents=[base.id])
  159. c2a = make_commit(parents=[c1.id], message=b"2a")
  160. c2b = make_commit(parents=[c1.id], message=b"2b")
  161. r.object_store.add_objects([(base, None), (c1, None), (c2a, None), (c2b, None)])
  162. self.assertTrue(can_fast_forward(r, c1.id, c2a.id))
  163. self.assertTrue(can_fast_forward(r, c1.id, c2b.id))
  164. self.assertFalse(can_fast_forward(r, c2a.id, c2b.id))
  165. self.assertFalse(can_fast_forward(r, c2b.id, c2a.id))