config.py 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800
  1. # config.py - Reading and writing Git config files
  2. # Copyright (C) 2011-2013 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. """Reading and writing Git configuration files.
  21. Todo:
  22. * preserve formatting when updating configuration files
  23. * treat subsection names as case-insensitive for [branch.foo] style
  24. subsections
  25. """
  26. import os
  27. import sys
  28. from contextlib import suppress
  29. from typing import (
  30. Any,
  31. BinaryIO,
  32. Dict,
  33. Iterable,
  34. Iterator,
  35. KeysView,
  36. List,
  37. MutableMapping,
  38. Optional,
  39. Tuple,
  40. Union,
  41. overload,
  42. )
  43. from .file import GitFile
  44. SENTINEL = object()
  45. def lower_key(key):
  46. if isinstance(key, (bytes, str)):
  47. return key.lower()
  48. if isinstance(key, Iterable):
  49. return type(key)(map(lower_key, key)) # type: ignore
  50. return key
  51. class CaseInsensitiveOrderedMultiDict(MutableMapping):
  52. def __init__(self) -> None:
  53. self._real: List[Any] = []
  54. self._keyed: Dict[Any, Any] = {}
  55. @classmethod
  56. def make(cls, dict_in=None):
  57. if isinstance(dict_in, cls):
  58. return dict_in
  59. out = cls()
  60. if dict_in is None:
  61. return out
  62. if not isinstance(dict_in, MutableMapping):
  63. raise TypeError
  64. for key, value in dict_in.items():
  65. out[key] = value
  66. return out
  67. def __len__(self) -> int:
  68. return len(self._keyed)
  69. def keys(self) -> KeysView[Tuple[bytes, ...]]:
  70. return self._keyed.keys()
  71. def items(self):
  72. return iter(self._real)
  73. def __iter__(self):
  74. return self._keyed.__iter__()
  75. def values(self):
  76. return self._keyed.values()
  77. def __setitem__(self, key, value) -> None:
  78. self._real.append((key, value))
  79. self._keyed[lower_key(key)] = value
  80. def __delitem__(self, key) -> None:
  81. key = lower_key(key)
  82. del self._keyed[key]
  83. for i, (actual, unused_value) in reversed(list(enumerate(self._real))):
  84. if lower_key(actual) == key:
  85. del self._real[i]
  86. def __getitem__(self, item):
  87. return self._keyed[lower_key(item)]
  88. def get(self, key, default=SENTINEL):
  89. try:
  90. return self[key]
  91. except KeyError:
  92. pass
  93. if default is SENTINEL:
  94. return type(self)()
  95. return default
  96. def get_all(self, key):
  97. key = lower_key(key)
  98. for actual, value in self._real:
  99. if lower_key(actual) == key:
  100. yield value
  101. def setdefault(self, key, default=SENTINEL):
  102. try:
  103. return self[key]
  104. except KeyError:
  105. self[key] = self.get(key, default)
  106. return self[key]
  107. Name = bytes
  108. NameLike = Union[bytes, str]
  109. Section = Tuple[bytes, ...]
  110. SectionLike = Union[bytes, str, Tuple[Union[bytes, str], ...]]
  111. Value = bytes
  112. ValueLike = Union[bytes, str]
  113. class Config:
  114. """A Git configuration."""
  115. def get(self, section: SectionLike, name: NameLike) -> Value:
  116. """Retrieve the contents of a configuration setting.
  117. Args:
  118. section: Tuple with section name and optional subsection name
  119. name: Variable name
  120. Returns:
  121. Contents of the setting
  122. Raises:
  123. KeyError: if the value is not set
  124. """
  125. raise NotImplementedError(self.get)
  126. def get_multivar(self, section: SectionLike, name: NameLike) -> Iterator[Value]:
  127. """Retrieve the contents of a multivar configuration setting.
  128. Args:
  129. section: Tuple with section name and optional subsection namee
  130. name: Variable name
  131. Returns:
  132. Contents of the setting as iterable
  133. Raises:
  134. KeyError: if the value is not set
  135. """
  136. raise NotImplementedError(self.get_multivar)
  137. @overload
  138. def get_boolean(
  139. self, section: SectionLike, name: NameLike, default: bool
  140. ) -> bool: ...
  141. @overload
  142. def get_boolean(self, section: SectionLike, name: NameLike) -> Optional[bool]: ...
  143. def get_boolean(
  144. self, section: SectionLike, name: NameLike, default: Optional[bool] = None
  145. ) -> Optional[bool]:
  146. """Retrieve a configuration setting as boolean.
  147. Args:
  148. section: Tuple with section name and optional subsection name
  149. name: Name of the setting, including section and possible
  150. subsection.
  151. Returns:
  152. Contents of the setting
  153. """
  154. try:
  155. value = self.get(section, name)
  156. except KeyError:
  157. return default
  158. if value.lower() == b"true":
  159. return True
  160. elif value.lower() == b"false":
  161. return False
  162. raise ValueError(f"not a valid boolean string: {value!r}")
  163. def set(
  164. self, section: SectionLike, name: NameLike, value: Union[ValueLike, bool]
  165. ) -> None:
  166. """Set a configuration value.
  167. Args:
  168. section: Tuple with section name and optional subsection namee
  169. name: Name of the configuration value, including section
  170. and optional subsection
  171. value: value of the setting
  172. """
  173. raise NotImplementedError(self.set)
  174. def items(self, section: SectionLike) -> Iterator[Tuple[Name, Value]]:
  175. """Iterate over the configuration pairs for a specific section.
  176. Args:
  177. section: Tuple with section name and optional subsection namee
  178. Returns:
  179. Iterator over (name, value) pairs
  180. """
  181. raise NotImplementedError(self.items)
  182. def sections(self) -> Iterator[Section]:
  183. """Iterate over the sections.
  184. Returns: Iterator over section tuples
  185. """
  186. raise NotImplementedError(self.sections)
  187. def has_section(self, name: Section) -> bool:
  188. """Check if a specified section exists.
  189. Args:
  190. name: Name of section to check for
  191. Returns:
  192. boolean indicating whether the section exists
  193. """
  194. return name in self.sections()
  195. class ConfigDict(Config, MutableMapping[Section, MutableMapping[Name, Value]]):
  196. """Git configuration stored in a dictionary."""
  197. def __init__(
  198. self,
  199. values: Union[
  200. MutableMapping[Section, MutableMapping[Name, Value]], None
  201. ] = None,
  202. encoding: Union[str, None] = None,
  203. ) -> None:
  204. """Create a new ConfigDict."""
  205. if encoding is None:
  206. encoding = sys.getdefaultencoding()
  207. self.encoding = encoding
  208. self._values = CaseInsensitiveOrderedMultiDict.make(values)
  209. def __repr__(self) -> str:
  210. return f"{self.__class__.__name__}({self._values!r})"
  211. def __eq__(self, other: object) -> bool:
  212. return isinstance(other, self.__class__) and other._values == self._values
  213. def __getitem__(self, key: Section) -> MutableMapping[Name, Value]:
  214. return self._values.__getitem__(key)
  215. def __setitem__(self, key: Section, value: MutableMapping[Name, Value]) -> None:
  216. return self._values.__setitem__(key, value)
  217. def __delitem__(self, key: Section) -> None:
  218. return self._values.__delitem__(key)
  219. def __iter__(self) -> Iterator[Section]:
  220. return self._values.__iter__()
  221. def __len__(self) -> int:
  222. return self._values.__len__()
  223. @classmethod
  224. def _parse_setting(cls, name: str):
  225. parts = name.split(".")
  226. if len(parts) == 3:
  227. return (parts[0], parts[1], parts[2])
  228. else:
  229. return (parts[0], None, parts[1])
  230. def _check_section_and_name(
  231. self, section: SectionLike, name: NameLike
  232. ) -> Tuple[Section, Name]:
  233. if not isinstance(section, tuple):
  234. section = (section,)
  235. checked_section = tuple(
  236. [
  237. subsection.encode(self.encoding)
  238. if not isinstance(subsection, bytes)
  239. else subsection
  240. for subsection in section
  241. ]
  242. )
  243. if not isinstance(name, bytes):
  244. name = name.encode(self.encoding)
  245. return checked_section, name
  246. def get_multivar(self, section: SectionLike, name: NameLike) -> Iterator[Value]:
  247. section, name = self._check_section_and_name(section, name)
  248. if len(section) > 1:
  249. try:
  250. return self._values[section].get_all(name)
  251. except KeyError:
  252. pass
  253. return self._values[(section[0],)].get_all(name)
  254. def get( # type: ignore[override]
  255. self,
  256. section: SectionLike,
  257. name: NameLike,
  258. ) -> Value:
  259. section, name = self._check_section_and_name(section, name)
  260. if len(section) > 1:
  261. try:
  262. return self._values[section][name]
  263. except KeyError:
  264. pass
  265. return self._values[(section[0],)][name]
  266. def set(
  267. self,
  268. section: SectionLike,
  269. name: NameLike,
  270. value: Union[ValueLike, bool],
  271. ) -> None:
  272. section, name = self._check_section_and_name(section, name)
  273. if isinstance(value, bool):
  274. value = b"true" if value else b"false"
  275. if not isinstance(value, bytes):
  276. value = value.encode(self.encoding)
  277. self._values.setdefault(section)[name] = value
  278. def items( # type: ignore[override]
  279. self, section: Section
  280. ) -> Iterator[Tuple[Name, Value]]:
  281. return self._values.get(section).items()
  282. def sections(self) -> Iterator[Section]:
  283. return self._values.keys()
  284. def _format_string(value: bytes) -> bytes:
  285. if (
  286. value.startswith((b" ", b"\t"))
  287. or value.endswith((b" ", b"\t"))
  288. or b"#" in value
  289. ):
  290. return b'"' + _escape_value(value) + b'"'
  291. else:
  292. return _escape_value(value)
  293. _ESCAPE_TABLE = {
  294. ord(b"\\"): ord(b"\\"),
  295. ord(b'"'): ord(b'"'),
  296. ord(b"n"): ord(b"\n"),
  297. ord(b"t"): ord(b"\t"),
  298. ord(b"b"): ord(b"\b"),
  299. }
  300. _COMMENT_CHARS = [ord(b"#"), ord(b";")]
  301. _WHITESPACE_CHARS = [ord(b"\t"), ord(b" ")]
  302. def _parse_string(value: bytes) -> bytes:
  303. value = bytearray(value.strip())
  304. ret = bytearray()
  305. whitespace = bytearray()
  306. in_quotes = False
  307. i = 0
  308. while i < len(value):
  309. c = value[i]
  310. if c == ord(b"\\"):
  311. i += 1
  312. try:
  313. v = _ESCAPE_TABLE[value[i]]
  314. except IndexError as exc:
  315. raise ValueError(
  316. "escape character in %r at %d before end of string" % (value, i)
  317. ) from exc
  318. except KeyError as exc:
  319. raise ValueError(
  320. "escape character followed by unknown character "
  321. "%s at %d in %r" % (value[i], i, value)
  322. ) from exc
  323. if whitespace:
  324. ret.extend(whitespace)
  325. whitespace = bytearray()
  326. ret.append(v)
  327. elif c == ord(b'"'):
  328. in_quotes = not in_quotes
  329. elif c in _COMMENT_CHARS and not in_quotes:
  330. # the rest of the line is a comment
  331. break
  332. elif c in _WHITESPACE_CHARS:
  333. whitespace.append(c)
  334. else:
  335. if whitespace:
  336. ret.extend(whitespace)
  337. whitespace = bytearray()
  338. ret.append(c)
  339. i += 1
  340. if in_quotes:
  341. raise ValueError("missing end quote")
  342. return bytes(ret)
  343. def _escape_value(value: bytes) -> bytes:
  344. """Escape a value."""
  345. value = value.replace(b"\\", b"\\\\")
  346. value = value.replace(b"\r", b"\\r")
  347. value = value.replace(b"\n", b"\\n")
  348. value = value.replace(b"\t", b"\\t")
  349. value = value.replace(b'"', b'\\"')
  350. return value
  351. def _check_variable_name(name: bytes) -> bool:
  352. for i in range(len(name)):
  353. c = name[i : i + 1]
  354. if not c.isalnum() and c != b"-":
  355. return False
  356. return True
  357. def _check_section_name(name: bytes) -> bool:
  358. for i in range(len(name)):
  359. c = name[i : i + 1]
  360. if not c.isalnum() and c not in (b"-", b"."):
  361. return False
  362. return True
  363. def _strip_comments(line: bytes) -> bytes:
  364. comment_bytes = {ord(b"#"), ord(b";")}
  365. quote = ord(b'"')
  366. string_open = False
  367. # Normalize line to bytearray for simple 2/3 compatibility
  368. for i, character in enumerate(bytearray(line)):
  369. # Comment characters outside balanced quotes denote comment start
  370. if character == quote:
  371. string_open = not string_open
  372. elif not string_open and character in comment_bytes:
  373. return line[:i]
  374. return line
  375. def _parse_section_header_line(line: bytes) -> Tuple[Section, bytes]:
  376. # Parse section header ("[bla]")
  377. line = _strip_comments(line).rstrip()
  378. in_quotes = False
  379. escaped = False
  380. for i, c in enumerate(line):
  381. if escaped:
  382. escaped = False
  383. continue
  384. if c == ord(b'"'):
  385. in_quotes = not in_quotes
  386. if c == ord(b"\\"):
  387. escaped = True
  388. if c == ord(b"]") and not in_quotes:
  389. last = i
  390. break
  391. else:
  392. raise ValueError("expected trailing ]")
  393. pts = line[1:last].split(b" ", 1)
  394. line = line[last + 1 :]
  395. section: Section
  396. if len(pts) == 2:
  397. if pts[1][:1] != b'"' or pts[1][-1:] != b'"':
  398. raise ValueError(f"Invalid subsection {pts[1]!r}")
  399. else:
  400. pts[1] = pts[1][1:-1]
  401. if not _check_section_name(pts[0]):
  402. raise ValueError(f"invalid section name {pts[0]!r}")
  403. section = (pts[0], pts[1])
  404. else:
  405. if not _check_section_name(pts[0]):
  406. raise ValueError(f"invalid section name {pts[0]!r}")
  407. pts = pts[0].split(b".", 1)
  408. if len(pts) == 2:
  409. section = (pts[0], pts[1])
  410. else:
  411. section = (pts[0],)
  412. return section, line
  413. class ConfigFile(ConfigDict):
  414. """A Git configuration file, like .git/config or ~/.gitconfig."""
  415. def __init__(
  416. self,
  417. values: Union[
  418. MutableMapping[Section, MutableMapping[Name, Value]], None
  419. ] = None,
  420. encoding: Union[str, None] = None,
  421. ) -> None:
  422. super().__init__(values=values, encoding=encoding)
  423. self.path: Optional[str] = None
  424. @classmethod
  425. def from_file(cls, f: BinaryIO) -> "ConfigFile":
  426. """Read configuration from a file-like object."""
  427. ret = cls()
  428. section: Optional[Section] = None
  429. setting = None
  430. continuation = None
  431. for lineno, line in enumerate(f.readlines()):
  432. if lineno == 0 and line.startswith(b"\xef\xbb\xbf"):
  433. line = line[3:]
  434. line = line.lstrip()
  435. if setting is None:
  436. if len(line) > 0 and line[:1] == b"[":
  437. section, line = _parse_section_header_line(line)
  438. ret._values.setdefault(section)
  439. if _strip_comments(line).strip() == b"":
  440. continue
  441. if section is None:
  442. raise ValueError(f"setting {line!r} without section")
  443. try:
  444. setting, value = line.split(b"=", 1)
  445. except ValueError:
  446. setting = line
  447. value = b"true"
  448. setting = setting.strip()
  449. if not _check_variable_name(setting):
  450. raise ValueError(f"invalid variable name {setting!r}")
  451. if value.endswith(b"\\\n"):
  452. continuation = value[:-2]
  453. elif value.endswith(b"\\\r\n"):
  454. continuation = value[:-3]
  455. else:
  456. continuation = None
  457. value = _parse_string(value)
  458. ret._values[section][setting] = value
  459. setting = None
  460. else: # continuation line
  461. if line.endswith(b"\\\n"):
  462. continuation += line[:-2]
  463. elif line.endswith(b"\\\r\n"):
  464. continuation += line[:-3]
  465. else:
  466. continuation += line
  467. value = _parse_string(continuation)
  468. ret._values[section][setting] = value
  469. continuation = None
  470. setting = None
  471. return ret
  472. @classmethod
  473. def from_path(cls, path: str) -> "ConfigFile":
  474. """Read configuration from a file on disk."""
  475. with GitFile(path, "rb") as f:
  476. ret = cls.from_file(f)
  477. ret.path = path
  478. return ret
  479. def write_to_path(self, path: Optional[str] = None) -> None:
  480. """Write configuration to a file on disk."""
  481. if path is None:
  482. path = self.path
  483. with GitFile(path, "wb") as f:
  484. self.write_to_file(f)
  485. def write_to_file(self, f: BinaryIO) -> None:
  486. """Write configuration to a file-like object."""
  487. for section, values in self._values.items():
  488. try:
  489. section_name, subsection_name = section
  490. except ValueError:
  491. (section_name,) = section
  492. subsection_name = None
  493. if subsection_name is None:
  494. f.write(b"[" + section_name + b"]\n")
  495. else:
  496. f.write(b"[" + section_name + b' "' + subsection_name + b'"]\n')
  497. for key, value in values.items():
  498. value = _format_string(value)
  499. f.write(b"\t" + key + b" = " + value + b"\n")
  500. def get_xdg_config_home_path(*path_segments):
  501. xdg_config_home = os.environ.get(
  502. "XDG_CONFIG_HOME",
  503. os.path.expanduser("~/.config/"),
  504. )
  505. return os.path.join(xdg_config_home, *path_segments)
  506. def _find_git_in_win_path():
  507. for exe in ("git.exe", "git.cmd"):
  508. for path in os.environ.get("PATH", "").split(";"):
  509. if os.path.exists(os.path.join(path, exe)):
  510. # in windows native shells (powershell/cmd) exe path is
  511. # .../Git/bin/git.exe or .../Git/cmd/git.exe
  512. #
  513. # in git-bash exe path is .../Git/mingw64/bin/git.exe
  514. git_dir, _bin_dir = os.path.split(path)
  515. yield git_dir
  516. parent_dir, basename = os.path.split(git_dir)
  517. if basename == "mingw32" or basename == "mingw64":
  518. yield parent_dir
  519. break
  520. def _find_git_in_win_reg():
  521. import platform
  522. import winreg
  523. if platform.machine() == "AMD64":
  524. subkey = (
  525. "SOFTWARE\\Wow6432Node\\Microsoft\\Windows\\"
  526. "CurrentVersion\\Uninstall\\Git_is1"
  527. )
  528. else:
  529. subkey = "SOFTWARE\\Microsoft\\Windows\\CurrentVersion\\" "Uninstall\\Git_is1"
  530. for key in (winreg.HKEY_CURRENT_USER, winreg.HKEY_LOCAL_MACHINE): # type: ignore
  531. with suppress(OSError):
  532. with winreg.OpenKey(key, subkey) as k: # type: ignore
  533. val, typ = winreg.QueryValueEx(k, "InstallLocation") # type: ignore
  534. if typ == winreg.REG_SZ: # type: ignore
  535. yield val
  536. # There is no set standard for system config dirs on windows. We try the
  537. # following:
  538. # - %PROGRAMDATA%/Git/config - (deprecated) Windows config dir per CGit docs
  539. # - %PROGRAMFILES%/Git/etc/gitconfig - Git for Windows (msysgit) config dir
  540. # Used if CGit installation (Git/bin/git.exe) is found in PATH in the
  541. # system registry
  542. def get_win_system_paths():
  543. if "PROGRAMDATA" in os.environ:
  544. yield os.path.join(os.environ["PROGRAMDATA"], "Git", "config")
  545. for git_dir in _find_git_in_win_path():
  546. yield os.path.join(git_dir, "etc", "gitconfig")
  547. for git_dir in _find_git_in_win_reg():
  548. yield os.path.join(git_dir, "etc", "gitconfig")
  549. class StackedConfig(Config):
  550. """Configuration which reads from multiple config files.."""
  551. def __init__(
  552. self, backends: List[ConfigFile], writable: Optional[ConfigFile] = None
  553. ) -> None:
  554. self.backends = backends
  555. self.writable = writable
  556. def __repr__(self) -> str:
  557. return f"<{self.__class__.__name__} for {self.backends!r}>"
  558. @classmethod
  559. def default(cls) -> "StackedConfig":
  560. return cls(cls.default_backends())
  561. @classmethod
  562. def default_backends(cls) -> List[ConfigFile]:
  563. """Retrieve the default configuration.
  564. See git-config(1) for details on the files searched.
  565. """
  566. paths = []
  567. paths.append(os.path.expanduser("~/.gitconfig"))
  568. paths.append(get_xdg_config_home_path("git", "config"))
  569. if "GIT_CONFIG_NOSYSTEM" not in os.environ:
  570. paths.append("/etc/gitconfig")
  571. if sys.platform == "win32":
  572. paths.extend(get_win_system_paths())
  573. backends = []
  574. for path in paths:
  575. try:
  576. cf = ConfigFile.from_path(path)
  577. except FileNotFoundError:
  578. continue
  579. backends.append(cf)
  580. return backends
  581. def get(self, section: SectionLike, name: NameLike) -> Value:
  582. if not isinstance(section, tuple):
  583. section = (section,)
  584. for backend in self.backends:
  585. try:
  586. return backend.get(section, name)
  587. except KeyError:
  588. pass
  589. raise KeyError(name)
  590. def get_multivar(self, section: SectionLike, name: NameLike) -> Iterator[Value]:
  591. if not isinstance(section, tuple):
  592. section = (section,)
  593. for backend in self.backends:
  594. try:
  595. yield from backend.get_multivar(section, name)
  596. except KeyError:
  597. pass
  598. def set(
  599. self, section: SectionLike, name: NameLike, value: Union[ValueLike, bool]
  600. ) -> None:
  601. if self.writable is None:
  602. raise NotImplementedError(self.set)
  603. return self.writable.set(section, name, value)
  604. def sections(self) -> Iterator[Section]:
  605. seen = set()
  606. for backend in self.backends:
  607. for section in backend.sections():
  608. if section not in seen:
  609. seen.add(section)
  610. yield section
  611. def read_submodules(path: str) -> Iterator[Tuple[bytes, bytes, bytes]]:
  612. """Read a .gitmodules file."""
  613. cfg = ConfigFile.from_path(path)
  614. return parse_submodules(cfg)
  615. def parse_submodules(config: ConfigFile) -> Iterator[Tuple[bytes, bytes, bytes]]:
  616. """Parse a gitmodules GitConfig file, returning submodules.
  617. Args:
  618. config: A `ConfigFile`
  619. Returns:
  620. list of tuples (submodule path, url, name),
  621. where name is quoted part of the section's name.
  622. """
  623. for section in config.keys():
  624. section_kind, section_name = section
  625. if section_kind == b"submodule":
  626. try:
  627. sm_path = config.get(section, b"path")
  628. sm_url = config.get(section, b"url")
  629. yield (sm_path, sm_url, section_name)
  630. except KeyError:
  631. # If either path or url is missing, just ignore this
  632. # submodule entry and move on to the next one. This is
  633. # how git itself handles malformed .gitmodule entries.
  634. pass
  635. def iter_instead_of(config: Config, push: bool = False) -> Iterable[Tuple[str, str]]:
  636. """Iterate over insteadOf / pushInsteadOf values."""
  637. for section in config.sections():
  638. if section[0] != b"url":
  639. continue
  640. replacement = section[1]
  641. try:
  642. needles = list(config.get_multivar(section, "insteadOf"))
  643. except KeyError:
  644. needles = []
  645. if push:
  646. try:
  647. needles += list(config.get_multivar(section, "pushInsteadOf"))
  648. except KeyError:
  649. pass
  650. for needle in needles:
  651. assert isinstance(needle, bytes)
  652. yield needle.decode("utf-8"), replacement.decode("utf-8")
  653. def apply_instead_of(config: Config, orig_url: str, push: bool = False) -> str:
  654. """Apply insteadOf / pushInsteadOf to a URL."""
  655. longest_needle = ""
  656. updated_url = orig_url
  657. for needle, replacement in iter_instead_of(config, push):
  658. if not orig_url.startswith(needle):
  659. continue
  660. if len(longest_needle) < len(needle):
  661. longest_needle = needle
  662. updated_url = replacement + orig_url[len(needle) :]
  663. return updated_url