objectspec.py 7.4 KB

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