Source code for sunset.autosaver

import logging
import os
import pathlib
import tempfile

from types import TracebackType

from typing import Any, Callable, IO, Optional, Protocol, TypeVar, Union

from sunset.timer import PersistentTimer, TimerProtocol

_TimerT = TypeVar("_TimerT", bound=TimerProtocol)


class _SavableProtocol(Protocol):
    """
    This protocol lets AutoSaver know how to use a Settings instance without
    having to import the actual Settings class.
    """

    def load(self, file: IO[str]) -> None: ...

    def save(self, file: IO[str], blanklines: bool = False) -> None: ...

    def onUpdateCall(self, callback: Callable[[Any], Any]) -> None: ...


[docs] class AutoSaver: """ AutoSaver is a helper class that can take care of loading and saving settings automatically and safely. On instantiation, it automatically loads settings from the given file path, if that file exists, unless the `load_on_init` argument is set to False. When saving, it uses a two-steps mechanism to ensure the atomicity of the operation. That is to say, the operation either entirely succeeds or entirely fails; AutoSaver will never write an incomplete settings file. Saving automatically creates the parent directories of the target path if those don't exist yet. If the `save_on_update` argument is set to True, AutoSaver will save the settings whenever they are updated. If the `save_on_delete` argument is set to True, AutoSaver will save the settings when its own instance is about to get deleted. AutoSaver can also be used as a context manager. Using it as a context manager is the recommended usage. Args: settings: The Settings instance to load to and save from. path: The full path to the file to load the settings from and save them to. If this file does not exist yet, it will be created when saving for the first time. save_on_update: Whether to save the settings when they are updated in any way. Default: True. save_on_delete: Whether to save the settings when this AutoSaver instance is deleted. Default: True. load_on_init: Whether to load the settings when instantiating this AutoSaver, provided the settings path exists. Default: True. save_delay: How long to wait, in seconds, before actually saving the settings when `save_on_update` is True and an update occurs. Setting this to a few seconds will batch updates for that long before triggering a save. If set to 0, the save is triggered immediately. Default: 0. logger: A logger instance that will be used to log OS errors, if any, while loading or saving settings. If none is given, the default root logger will be used. Example: >>> from sunset import AutoSaver, Settings >>> class ExampleSettings(Settings): ... ... >>> settings = ExampleSettings() >>> with AutoSaver(settings, "~/.config/my_app.conf"): # doctest: +SKIP ... main_program_loop(settings) """ _DIR_MODE: int = 0o755 _FILE_MODE: int = 0o644 _ENCODING: str = "UTF-8" _path: pathlib.Path _settings: _SavableProtocol _dirty: bool _save_on_update: bool _save_on_delete: bool _save_delay: float _save_timer: TimerProtocol _logger: logging.Logger def __init__( self, settings: _SavableProtocol, path: Union[pathlib.Path, str], *, save_on_update: bool = True, save_on_delete: bool = True, load_on_init: bool = True, save_delay: float = 0.0, logger: Optional[logging.Logger] = None, ) -> None: if isinstance(path, str): path = pathlib.Path(path) path = path.expanduser() if logger is None: logger = logging.getLogger() self._dirty = False self._path = path self._save_on_update = save_on_update self._save_on_delete = save_on_delete self._save_delay = save_delay self._save_timer = PersistentTimer(self.saveIfNeeded) self._settings = settings self._settings.onUpdateCall(self._onSettingsUpdated) self._logger = logger if load_on_init: self.doLoad() def path(self) -> pathlib.Path: """ Returns the settings path for this AutoSaver. Returns: A path. It may or may not exist; that's not up to AutoSaver. """ return self._path
[docs] def doLoad(self) -> bool: """ Load the settings from this AutoSaver's settings file path, if it exists. Unsaved settings, if any, will be lost. Returns: True if loading succeeded, else False. If an unexpected error occurred, the error will be logged using the logger passed during initialization. Note that it's not considered an error if the settings file path does not exist yet. Note: Prefer letting AutoSaver load the settings during its initialization in order to avoid race conditions. """ try: if (path := self._path).exists(): self._logger.debug("Loading settings file '%s'...", path) with open(path, encoding=self._ENCODING) as f: self._settings.load(f) self._logger.debug("Loaded.") return True except OSError as e: self._logger.error( "Error while loading from '%s': %s", self._path, e ) return False
[docs] def doSave(self) -> bool: """ Unconditionally saves the settings attached to this AutoSaver instance. This method uses a two-steps mechanism to perform the save, in order to make the save atomic. The settings are first saved to a temporary file, and if successful, that temporary file then replaces the actual settings file. If the directory where the settings file is located does not exist, this method automatically creates it. Returns: True if saving succeeded, else False. If False, the error will be logged using the logger passed during initialization. """ self._save_timer.cancel() directory = self._path.parent directory.mkdir(parents=True, mode=self._DIR_MODE, exist_ok=True) try: with tempfile.NamedTemporaryFile( dir=directory, prefix=self._path.name, mode="xt", encoding=self._ENCODING, delete=False, ) as tmp: self._logger.debug("Saving settings file '%s'...", self._path) self._settings.save(tmp.file, blanklines=True) os.chmod(tmp.name, self._FILE_MODE) os.rename(tmp.name, self._path) self._logger.debug("Saved.") except OSError as e: self._logger.error("Error while saving to '%s': %s", self._path, e) return False self._dirty = False return True
[docs] def saveIfNeeded(self) -> bool: """ Performs a save if and only if there are pending, unsaved changes in the settings attached to this AutoSaver instance. Returns: True if a save was performed and succeeded, else False. """ save_needed = self._dirty self._save_timer.cancel() if save_needed: return self.doSave() return save_needed
def _onSettingsUpdated(self, _: Any) -> None: self._dirty = True if self._save_on_update: self._save_timer.start(self._save_delay) def setSaveTimerClass(self, timer_class: type[_TimerT]) -> _TimerT: """ Internal. """ timer = timer_class(self.saveIfNeeded) self._save_timer = timer return timer def __del__(self) -> None: if self._save_on_delete: self.saveIfNeeded() def __enter__(self) -> "AutoSaver": return self def __exit__( self, exc_type: Optional[type[BaseException]], exc_value: Optional[BaseException], traceback: Optional[TracebackType], ) -> None: self.saveIfNeeded()