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.
"""

import os
import shutil
import sys
from collections.abc import Callable
from functools import partial, wraps
from inspect import cleandoc
from pathlib import Path
from types import TracebackType
from typing import Literal, ParamSpec
from warnings import warn

from astropy.utils.exceptions import AstropyUserWarning

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


P = ParamSpec("P")


[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 @staticmethod def _warn_env_var_is_ignored( reason: str, *, env_var_name: str, stacklevel: int = 4 ) -> None: env_var_value = os.getenv(env_var_name) warning_msg_template = ( f"{env_var_name} is set to '{env_var_value}', " "but {reason}. This environment variable will be ignored." ) warn( warning_msg_template.format(reason=reason), category=AstropyUserWarning, stacklevel=stacklevel, ) @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 (env_dir_str := os.getenv(cls._directory_env_var)) is None: return cls._find_or_create_root_dir(linkto=None, pkgname=rootname) warn = partial( cls._warn_env_var_is_ignored, env_var_name=cls._directory_env_var ) if not (env_dir_path := Path(env_dir_str)).is_dir(): warn("no such directory was found") return cls._find_or_create_root_dir(linkto=None, pkgname=rootname) path = env_dir_path / rootname if path.is_symlink(): return cls._find_or_create_root_dir(linkto=None, pkgname=rootname) elif path.is_dir(): return path.resolve() linkto: Path | None if path.is_file(): warn( f"this directory already contains a file under '{path.name}', " "where a directory was expected" ) # renounce linking early to avoid redundant warnings linkto = None else: # symlink will be set to this if the directory is created linkto = path return cls._find_or_create_root_dir(linkto=linkto, pkgname=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." ) warn = partial( cls._warn_env_var_is_ignored, env_var_name=cls._directory_env_var ) # At this point, linkto is either None or a path that doesn't exist yet # Other linkto conditions already filtered in _get_dir_path() above. if linkto is not None: if sys.platform.startswith("win"): warn("is not supported on Windows") linkto = None elif maindir.is_dir(): warn( f"the default location, {maindir}, already exists, and takes precedence" ) linkto = None # redundant, but shouldn't make a difference 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 linkto is not None: linkto.symlink_to(maindir) 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"