Source code for astropy.config.paths

# Licensed under a 3-clause BSD style license - see LICENSE.rst
"""This module contains functions to determine where configuration and
data/cache files used by Astropy should be placed.
"""

from __future__ import annotations

import os
import shutil
import sys
from functools import wraps
from inspect import cleandoc
from pathlib import Path
from typing import TYPE_CHECKING

if TYPE_CHECKING:
    from collections.abc import Callable
    from types import TracebackType
    from typing import Literal, ParamSpec

    P = ParamSpec("P")

__all__ = [
    "get_cache_dir",
    "get_cache_dir_path",
    "get_config_dir",
    "get_config_dir_path",
    "set_temp_cache",
    "set_temp_config",
]


[docs] def get_config_dir_path(rootname: str = "astropy") -> Path: """ Determines the package configuration directory name and creates the directory if it doesn't exist. This directory is typically ``$HOME/.astropy/config``, but if the XDG_CONFIG_HOME environment variable is set and the ``$XDG_CONFIG_HOME/astropy`` directory exists, it will be that directory. If neither exists, the former will be created and symlinked to the latter. Parameters ---------- rootname : str Name of the root configuration directory. For example, if ``rootname = 'pkgname'``, the configuration directory would be ``<home>/.pkgname/`` rather than ``<home>/.astropy`` (depending on platform). Returns ------- configdir : Path The absolute path to the configuration directory. """ return set_temp_config._get_dir_path(rootname)
[docs] def get_config_dir(rootname: str = "astropy") -> str: return str(get_config_dir_path(rootname))
if get_config_dir_path.__doc__ is not None: # guard against PYTHONOPTIMIZE mode get_config_dir.__doc__ = cleandoc( get_config_dir_path.__doc__ + """ See Also -------- get_config_dir_path : same as this function, except that the return value is a pathlib.Path """ )
[docs] def get_cache_dir_path(rootname: str = "astropy") -> Path: """ Determines the Astropy cache directory name and creates the directory if it doesn't exist. This directory is typically ``$HOME/.astropy/cache``, but if the XDG_CACHE_HOME environment variable is set and the ``$XDG_CACHE_HOME/astropy`` directory exists, it will be that directory. If neither exists, the former will be created and symlinked to the latter. Parameters ---------- rootname : str Name of the root cache directory. For example, if ``rootname = 'pkgname'``, the cache directory will be ``<cache>/.pkgname/``. Returns ------- cachedir : Path The absolute path to the cache directory. """ return set_temp_cache._get_dir_path(rootname)
[docs] def get_cache_dir(rootname: str = "astropy") -> str: return str(get_cache_dir_path(rootname))
if get_cache_dir_path.__doc__ is not None: # guard against PYTHONOPTIMIZE mode get_cache_dir.__doc__ = cleandoc( get_cache_dir_path.__doc__ + """ See Also -------- get_cache_dir_path : same as this function, except that the return value is a pathlib.Path """ ) class _SetTempPath: _temp_path: Path | None = None # This base class serves as a deduplication layer for its only two intended # children (set_temp_cache and set_temp_config) _directory_type: Literal["cache", "config"] _directory_env_var: Literal["XDG_CACHE_HOME", "XDG_CONFIG_HOME"] def __init__( self, path: os.PathLike[str] | str | None = None, delete: bool = False ) -> None: if path is not None: path = Path(path).resolve() self._path = path self._delete = delete self._prev_path = self.__class__._temp_path def __enter__(self) -> str: self.__class__._temp_path = self._path try: return str(self.__class__._get_dir_path(rootname="astropy")) except Exception: self.__class__._temp_path = self._prev_path raise def __exit__( self, type: type[BaseException] | None, value: BaseException | None, tb: TracebackType | None, ) -> None: self.__class__._temp_path = self._prev_path if self._delete and self._path is not None: shutil.rmtree(self._path) def __call__(self, func: Callable[P, object]) -> Callable[P, None]: """Implements use as a decorator.""" @wraps(func) def wrapper(*args: P.args, **kwargs: P.kwargs) -> None: with self: func(*args, **kwargs) return wrapper @classmethod def _get_dir_path(cls, rootname: str) -> Path: if (xch := cls._temp_path) is not None: path = xch / rootname if not path.is_file(): path.mkdir(exist_ok=True) return path.resolve() if ( (dir_ := os.getenv(cls._directory_env_var)) is not None and (xch := Path(dir_)).exists() and not (xchpth := xch / rootname).is_symlink() ): if xchpth.exists(): return xchpth.resolve() # symlink will be set to this if the directory is created linkto = xchpth else: linkto = None return cls._find_or_create_root_dir(linkto, rootname) @classmethod def _find_or_create_root_dir( cls, linkto: Path | None, pkgname: str = "astropy", ) -> Path: innerdir = Path.home() / f".{pkgname}" maindir = innerdir / cls._directory_type if maindir.is_file(): raise OSError( f"Intended {pkgname} {cls._directory_type} directory {maindir} is actually a file." ) if not maindir.is_dir(): # first create .astropy dir if needed if innerdir.is_file(): raise OSError( f"Intended {pkgname} {cls._directory_type} directory {maindir} is actually a file." ) maindir.mkdir(parents=True, exist_ok=True) if ( not sys.platform.startswith("win") and linkto is not None and not linkto.exists() ): os.symlink(maindir, linkto) return maindir.resolve()
[docs] class set_temp_config(_SetTempPath): """ Context manager to set a temporary path for the Astropy config, primarily for use with testing. If the path set by this context manager does not already exist it will be created, if possible. This may also be used as a decorator on a function to set the config path just within that function. Parameters ---------- path : str, optional The directory (which must exist) in which to find the Astropy config files, or create them if they do not already exist. If None, this restores the config path to the user's default config path as returned by `get_config_dir` as though this context manager were not in effect (this is useful for testing). In this case the ``delete`` argument is always ignored. delete : bool, optional If True, cleans up the temporary directory after exiting the temp context (default: False). """ _directory_type = "config" _directory_env_var = "XDG_CONFIG_HOME" def __enter__(self) -> str: # Special case for the config case, where we need to reset all the # cached config objects. We do keep the cache, since some of it # may have been set programmatically rather than be stored in the # config file (e.g., iers.conf.auto_download=False for our tests). from .configuration import _cfgobjs self._cfgobjs_copy = _cfgobjs.copy() _cfgobjs.clear() return super().__enter__() def __exit__( self, type: type[BaseException] | None, value: BaseException | None, tb: TracebackType | None, ) -> None: from .configuration import _cfgobjs _cfgobjs.clear() _cfgobjs.update(self._cfgobjs_copy) del self._cfgobjs_copy super().__exit__(type, value, tb)
[docs] class set_temp_cache(_SetTempPath): """ Context manager to set a temporary path for the Astropy download cache, primarily for use with testing (though there may be other applications for setting a different cache directory, for example to switch to a cache dedicated to large files). If the path set by this context manager does not already exist it will be created, if possible. This may also be used as a decorator on a function to set the cache path just within that function. Parameters ---------- path : str The directory (which must exist) in which to find the Astropy cache files, or create them if they do not already exist. If None, this restores the cache path to the user's default cache path as returned by `get_cache_dir` as though this context manager were not in effect (this is useful for testing). In this case the ``delete`` argument is always ignored. delete : bool, optional If True, cleans up the temporary directory after exiting the temp context (default: False). """ _directory_type = "cache" _directory_env_var = "XDG_CACHE_HOME"