graph.py 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146
  1. #!/usr/bin/env python
  2. # vim:ts=4:sw=4:softtabstop=4:smarttab:expandtab
  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. """
  21. Implementation of merge-base following the approach of git
  22. """
  23. from typing import Deque
  24. from collections import deque
  25. def _find_lcas(lookup_parents, c1, c2s):
  26. cands = []
  27. cstates = {}
  28. # Flags to Record State
  29. _ANC_OF_1 = 1 # ancestor of commit 1
  30. _ANC_OF_2 = 2 # ancestor of commit 2
  31. _DNC = 4 # Do Not Consider
  32. _LCA = 8 # potential LCA
  33. def _has_candidates(wlst, cstates):
  34. for cmt in wlst:
  35. if cmt in cstates:
  36. if not (cstates[cmt] & _DNC):
  37. return True
  38. return False
  39. # initialize the working list
  40. wlst: Deque[int] = deque()
  41. cstates[c1] = _ANC_OF_1
  42. wlst.append(c1)
  43. for c2 in c2s:
  44. cstates[c2] = _ANC_OF_2
  45. wlst.append(c2)
  46. # loop until no other LCA candidates are viable in working list
  47. # adding any parents to the list in a breadth first manner
  48. while _has_candidates(wlst, cstates):
  49. cmt = wlst.popleft()
  50. flags = cstates[cmt]
  51. if flags == (_ANC_OF_1 | _ANC_OF_2):
  52. # potential common ancestor
  53. if not (flags & _LCA):
  54. flags = flags | _LCA
  55. cstates[cmt] = flags
  56. cands.append(cmt)
  57. # mark any parents of this node _DNC as all parents
  58. # would be one level further removed common ancestors
  59. flags = flags | _DNC
  60. parents = lookup_parents(cmt)
  61. if parents:
  62. for pcmt in parents:
  63. if pcmt in cstates:
  64. cstates[pcmt] = cstates[pcmt] | flags
  65. else:
  66. cstates[pcmt] = flags
  67. wlst.append(pcmt)
  68. # walk final candidates removing any superseded by _DNC by later lower LCAs
  69. results = []
  70. for cmt in cands:
  71. if not (cstates[cmt] & _DNC):
  72. results.append(cmt)
  73. return results
  74. def find_merge_base(repo, commit_ids):
  75. """Find lowest common ancestors of commit_ids[0] and *any* of commits_ids[1:]
  76. Args:
  77. repo: Repository object
  78. commit_ids: list of commit ids
  79. Returns:
  80. list of lowest common ancestor commit_ids
  81. """
  82. if not commit_ids:
  83. return []
  84. c1 = commit_ids[0]
  85. if not len(commit_ids) > 1:
  86. return [c1]
  87. c2s = commit_ids[1:]
  88. if c1 in c2s:
  89. return [c1]
  90. parents_provider = repo.parents_provider()
  91. return _find_lcas(parents_provider.get_parents, c1, c2s)
  92. def find_octopus_base(repo, commit_ids):
  93. """Find lowest common ancestors of *all* provided commit_ids
  94. Args:
  95. repo: Repository
  96. commit_ids: list of commit ids
  97. Returns:
  98. list of lowest common ancestor commit_ids
  99. """
  100. if not commit_ids:
  101. return []
  102. if len(commit_ids) <= 2:
  103. return find_merge_base(repo, commit_ids)
  104. parents_provider = repo.parents_provider()
  105. lcas = [commit_ids[0]]
  106. others = commit_ids[1:]
  107. for cmt in others:
  108. next_lcas = []
  109. for ca in lcas:
  110. res = _find_lcas(parents_provider.get_parents, cmt, [ca])
  111. next_lcas.extend(res)
  112. lcas = next_lcas[:]
  113. return lcas
  114. def can_fast_forward(repo, c1, c2):
  115. """Is it possible to fast-forward from c1 to c2?
  116. Args:
  117. repo: Repository to retrieve objects from
  118. c1: Commit id for first commit
  119. c2: Commit id for second commit
  120. """
  121. if c1 == c2:
  122. return True
  123. # Algorithm: Find the common ancestor
  124. parents_provider = repo.parents_provider()
  125. lcas = _find_lcas(parents_provider.get_parents, c1, [c2])
  126. return lcas == [c1]