| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879 |
- from __future__ import annotations
- import logging
- import os
- from collections import OrderedDict
- from typing import TYPE_CHECKING
- if TYPE_CHECKING:
- from pathlib import Path
- LOGGER = logging.getLogger(__name__)
- class PyEnvCfg:
- def __init__(self, content: OrderedDict[str, str], path: Path) -> None:
- self.content = content
- self.path = path
- @classmethod
- def from_folder(cls, folder: Path) -> PyEnvCfg:
- return cls.from_file(folder / "pyvenv.cfg")
- @classmethod
- def from_file(cls, path: Path) -> PyEnvCfg:
- content = cls._read_values(path) if path.exists() else OrderedDict()
- return PyEnvCfg(content, path)
- @staticmethod
- def _read_values(path: Path) -> OrderedDict[str, str]:
- content = OrderedDict()
- for line in path.read_text(encoding="utf-8").splitlines():
- equals_at = line.index("=")
- key = line[:equals_at].strip()
- value = line[equals_at + 1 :].strip()
- if len(value) > 1 and value[0] in {"'", '"'} and value[0] == value[-1]:
- value = value[1:-1]
- content[key] = value
- return content
- def write(self) -> None:
- LOGGER.debug("write %s", self.path)
- text = ""
- for key, value in self.content.items():
- # Use abspath to normalize relative paths but preserve symlinks (match venv behavior)
- # See issue #2770 - realpath resolves symlinks which breaks prefix symlinks
- if key == "prompt" and value:
- normalized_value = f'"{value}"'
- else:
- normalized_value = os.path.abspath(value) if value and os.path.exists(value) else value
- line = f"{key} = {normalized_value}"
- LOGGER.debug("\t%s", line)
- text += line
- text += "\n"
- self.path.write_text(text, encoding="utf-8")
- def refresh(self) -> OrderedDict[str, str]:
- self.content = self._read_values(self.path)
- return self.content
- def __setitem__(self, key: str, value: str) -> None:
- self.content[key] = value
- def __getitem__(self, key: str) -> str:
- return self.content[key]
- def __contains__(self, item: str) -> bool:
- return item in self.content
- def update(self, other: dict[str, str]) -> PyEnvCfg:
- self.content.update(other)
- return self
- def __repr__(self) -> str:
- return f"{self.__class__.__name__}(path={self.path})"
- __all__ = [
- "PyEnvCfg",
- ]
|