objectspec.py 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219
  1. # objectspec.py -- Object specification
  2. # Copyright (C) 2014 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. """Object specification."""
  21. def to_bytes(text):
  22. if getattr(text, "encode", None) is not None:
  23. text = text.encode('ascii')
  24. return text
  25. def parse_object(repo, objectish):
  26. """Parse a string referring to an object.
  27. :param repo: A `Repo` object
  28. :param objectish: A string referring to an object
  29. :return: A git object
  30. :raise KeyError: If the object can not be found
  31. """
  32. objectish = to_bytes(objectish)
  33. return repo[objectish]
  34. def parse_tree(repo, treeish):
  35. """Parse a string referring to a tree.
  36. :param repo: A `Repo` object
  37. :param treeish: A string referring to a tree
  38. :return: A git object
  39. :raise KeyError: If the object can not be found
  40. """
  41. treeish = to_bytes(treeish)
  42. o = repo[treeish]
  43. if o.type_name == b"commit":
  44. return repo[o.tree]
  45. return o
  46. def parse_ref(container, refspec):
  47. """Parse a string referring to a reference.
  48. :param container: A RefsContainer object
  49. :param refspec: A string referring to a ref
  50. :return: A ref
  51. :raise KeyError: If the ref can not be found
  52. """
  53. refspec = to_bytes(refspec)
  54. possible_refs = [
  55. refspec,
  56. b"refs/" + refspec,
  57. b"refs/tags/" + refspec,
  58. b"refs/heads/" + refspec,
  59. b"refs/remotes/" + refspec,
  60. b"refs/remotes/" + refspec + b"/HEAD"
  61. ]
  62. for ref in possible_refs:
  63. if ref in container:
  64. return ref
  65. else:
  66. raise KeyError(refspec)
  67. def parse_reftuple(lh_container, rh_container, refspec):
  68. """Parse a reftuple spec.
  69. :param lh_container: A RefsContainer object
  70. :param hh_container: A RefsContainer object
  71. :param refspec: A string
  72. :return: A tuple with left and right ref
  73. :raise KeyError: If one of the refs can not be found
  74. """
  75. refspec = to_bytes(refspec)
  76. if refspec.startswith(b"+"):
  77. force = True
  78. refspec = refspec[1:]
  79. else:
  80. force = False
  81. if b":" in refspec:
  82. (lh, rh) = refspec.split(b":")
  83. else:
  84. lh = rh = refspec
  85. if lh == b"":
  86. lh = None
  87. else:
  88. lh = parse_ref(lh_container, lh)
  89. if rh == b"":
  90. rh = None
  91. else:
  92. try:
  93. rh = parse_ref(rh_container, rh)
  94. except KeyError:
  95. # TODO: check force?
  96. if b"/" not in rh:
  97. rh = b"refs/heads/" + rh
  98. return (lh, rh, force)
  99. def parse_reftuples(lh_container, rh_container, refspecs):
  100. """Parse a list of reftuple specs to a list of reftuples.
  101. :param lh_container: A RefsContainer object
  102. :param hh_container: A RefsContainer object
  103. :param refspecs: A list of refspecs or a string
  104. :return: A list of refs
  105. :raise KeyError: If one of the refs can not be found
  106. """
  107. if not isinstance(refspecs, list):
  108. refspecs = [refspecs]
  109. ret = []
  110. # TODO: Support * in refspecs
  111. for refspec in refspecs:
  112. ret.append(parse_reftuple(lh_container, rh_container, refspec))
  113. return ret
  114. def parse_refs(container, refspecs):
  115. """Parse a list of refspecs to a list of refs.
  116. :param container: A RefsContainer object
  117. :param refspecs: A list of refspecs or a string
  118. :return: A list of refs
  119. :raise KeyError: If one of the refs can not be found
  120. """
  121. # TODO: Support * in refspecs
  122. if not isinstance(refspecs, list):
  123. refspecs = [refspecs]
  124. ret = []
  125. for refspec in refspecs:
  126. ret.append(parse_ref(container, refspec))
  127. return ret
  128. def parse_commit_range(repo, committishs):
  129. """Parse a string referring to a range of commits.
  130. :param repo: A `Repo` object
  131. :param committishs: A string referring to a range of commits.
  132. :return: An iterator over `Commit` objects
  133. :raise KeyError: When the reference commits can not be found
  134. :raise ValueError: If the range can not be parsed
  135. """
  136. committishs = to_bytes(committishs)
  137. # TODO(jelmer): Support more than a single commit..
  138. return iter([parse_commit(repo, committishs)])
  139. class AmbiguousShortId(Exception):
  140. """The short id is ambiguous."""
  141. def __init__(self, prefix, options):
  142. self.prefix = prefix
  143. self.options = options
  144. def scan_for_short_id(object_store, prefix):
  145. """Scan an object store for a short id."""
  146. # TODO(jelmer): This could short-circuit looking for objects
  147. # starting with a certain prefix.
  148. ret = []
  149. for object_id in object_store:
  150. if object_id.startswith(prefix):
  151. ret.append(object_store[object_id])
  152. if not ret:
  153. raise KeyError(prefix)
  154. if len(ret) == 1:
  155. return ret[0]
  156. raise AmbiguousShortId(prefix, ret)
  157. def parse_commit(repo, committish):
  158. """Parse a string referring to a single commit.
  159. :param repo: A` Repo` object
  160. :param commitish: A string referring to a single commit.
  161. :return: A Commit object
  162. :raise KeyError: When the reference commits can not be found
  163. :raise ValueError: If the range can not be parsed
  164. """
  165. committish = to_bytes(committish)
  166. try:
  167. return repo[committish]
  168. except KeyError:
  169. pass
  170. try:
  171. return repo[parse_ref(repo, committish)]
  172. except KeyError:
  173. pass
  174. if len(committish) >= 4 and len(committish) < 40:
  175. try:
  176. int(committish, 16)
  177. except ValueError:
  178. pass
  179. else:
  180. try:
  181. return scan_for_short_id(repo.object_store, committish)
  182. except KeyError:
  183. pass
  184. raise KeyError(committish)
  185. # TODO: parse_path_in_tree(), which handles e.g. v1.0:Documentation