# -*- coding: utf-8 -*-
"""Useful Utility classes."""
__all__ = [
"AttributeStore",
"TypedList",
"Options",
"get_option",
"set_option",
"copy_into",
]
import copy
from collections.abc import Iterable, MutableSequence
from typing import Any
from typing import Iterable as IterableType
from typing import List, Optional, Self, Union
from .tests import all_type, isiterable
from .typing import Args, Data, Kwargs
_options = {
"short_repr": False,
"short_data_repr": False,
"short_folder_rrepr": True,
"short_img_repr": True,
"no_figs": True,
"multiprocessing": False, # Change default to not use multiprocessing for now
"threading": False,
"warnings": False,
}
class AttributeStore(dict):
"""A dictionary=like class that provides attributes that work like indices.
Used to implement the mapping of column types to indices in the setas attriobutes.
"""
def __init__(self: Self, *args: Args, **kwargs: Kwargs) -> None:
"""Initialise from a dictionary."""
if len(args) == 1 and isinstance(args[0], dict):
self.update(args[0])
else:
super().__init__(*args, **kwargs)
def __setattr__(self: Self, name: str, value: Any) -> None:
"""Set an attribute (equivalent to setting an item)."""
self[name] = value
def __getattr__(self: Self, name: str) -> Any:
"""Get an attribute (equivalent to getting an item)."""
try:
return self[name]
except KeyError as err:
raise AttributeError from err
class TypedList(MutableSequence):
"""Subclass list to make setitem enforce strict typing of members of the list."""
def __init__(self: Self, *args: Args, **kwargs: Kwargs) -> None:
"""Construct the TypedList."""
self._store = []
if (not args) or not (isinstance(args[0], type) or (isinstance(args[0], tuple) and all_type(args[0], type))):
self._type = str # Default list type is a string
else:
args = list(args)
self._type = args.pop(0)
if not args:
self._store = list(*args, **kwargs)
elif len(args) == 1 and all_type(args[0], self._type):
self._store = list(*args, **kwargs)
else:
if len(args) > 1:
raise SyntaxError("List should be constructed with at most two arguments, a type and an iterable")
raise TypeError(f"List should be initialised with elements that are all of type {self._type}")
def __add__(self: Self, other: IterableType) -> "TypedList":
"""Add operator works like ordinary lists."""
if isiterable(other):
new = copy.deepcopy(self)
new.extend(other)
return new
return NotImplemented
def __iadd__(self: Self, other: IterableType) -> "TypedList":
"""Inplace-add works like a list."""
if isiterable(other):
self.extend(other)
return self
return NotImplemented
def __radd__(self: Self, other: IterableType) -> "TypedList":
"""Support add on the right like a list."""
if isinstance(other, list):
return other + self._store
return NotImplemented
def __eq__(self: Self, other: List) -> bool:
"""Equality test."""
return self._store == other
def __delitem__(self: Self, index: int) -> None:
"""Remove an item like in a list."""
del self._store[index]
def __getitem__(self: Self, index: int) -> Any:
"""Get an item like in a list."""
if isinstance(index, Iterable):
return [self._store[i] for i in index]
return self._store[index]
def __len__(self) -> int:
"""Implement the len function like a list."""
return len(self._store)
def __repr__(self) -> str:
"""Textual representation like a list."""
return repr(self._store)
def __setitem__(self: Self, name: Union[int, IterableType, slice], value: Any) -> None:
"""Sett an item and do some type checks."""
if isiterable(name) or isinstance(name, slice):
if not isiterable(value) or not all_type(value, self._type):
raise TypeError(
f"Elements of this list should be of type {self._type} and must set "
+ "the correct number of elements"
)
elif not isinstance(value, self._type):
raise TypeError(f"Elements of this list should be of type {self._type}")
self._store[name] = value
def extend(self: Self, values: IterableType) -> None: # pylint: disable=arguments-differ
"""Extend the list and do some type checking."""
if not isiterable(values) or not all_type(values, self._type):
raise TypeError(f"Elements of this list should be of type {self._type}")
self._store.extend(values)
def index(self: Self, value: Any, start: int = 0, stop: Optional[int] = None) -> int:
"""Index works like a list except we support Python 3 optional parameters everywhere."""
if stop is None:
stop = len(self._store)
return self._store[start:stop].index(value) + start
def insert(self: Self, index: int, value: Any) -> None: # pylint: disable=arguments-differ
"""Insert an element and do some type checking."""
if not isinstance(value, self._type):
raise TypeError(f"Elements of this list should be of type {self._type}")
self._store.insert(index, value)
def get_option(name: str) -> bool:
"""Return the option value."""
if name not in _options:
raise IndexError(f"{name} is not a valid package option")
return _options[name]
def set_option(name: str, value: bool) -> None:
"""Set a global package option.
Args:
name (str):
Option Name, one of:
- short_repr (bool):
Instead of using a rich representation, use a short description for DataFile and Imagefile.
- short_data_repr (bool):
Just use short representation for DataFiles
- short_img_repr (bool):
Just use a short representation for image file
- no_figs (bool):
Do not return figures from plotting functions, just plot them.
value (depends on name):
The value to set (see *name*)
"""
if name not in _options:
raise IndexError(f"{name} is not a valid package option")
if not isinstance(value, bool):
raise ValueError(f"{name} takes a boolean value not a {type(value)}")
_options[name] = value
class Options:
"""Dead simple class to allow access to package options."""
def __init__(self: Self):
"""Options class wraps the get/set_option calls into an object orientated interface."""
self._defaults = copy.copy(_options)
def __setattr__(self: Self, name: str, value: bool) -> None:
"""Set an option value."""
if name.startswith("_"):
super().__setattr__(name, value)
return
if name not in _options:
raise AttributeError(f"{name} is not a recognised option.")
if not isinstance(value, type(_options[name])):
raise ValueError(f"{name} takes a {type(_options[name])} not a {type(value)}")
set_option(name, value)
def __getattr__(self: Self, name: str) -> bool:
"""Lookup an option value."""
if name not in _options:
raise AttributeError(f"{name} is not a recognised option.")
return get_option(name)
def __delattr__(self: Self, name: str) -> None:
"""Clear and Option value back to defaults."""
if name not in _options:
raise AttributeError(f"{name} is not a recognised option.")
set_option(name, self._defaults[name])
def __dir__(self) -> List[str]:
"""Return a list of the aviailable options."""
return list(_options.keys())
def __repr__(self) -> str:
"""Make a standard text representation of Options class."""
s = "Stoner Package Options\n"
s += "~~~~~~~~~~~~~~~~~~~~~~\n"
for k in dir(self):
s += f"{k} : {get_option(k)}\n"
return s
[docs]
def copy_into(source: Data, dest: Data) -> Data:
"""Copy the data associated with source to dest.
Args:
source(DataFile): The DataFile object to be copied from
dest (DataFile): The DataFile objrct to be changed by receiving the copiued data.
Returns:
The modified *dest* DataFile.
Unlike copying or deepcopying a DataFile, this function preserves the class of the destination and just
overwrites the attributes that represent the data in the DataFile.
"""
dest.data = source.data.copy()
dest.setas = source.setas
dest.fig = getattr(source, "fig", None)
for attr in source._public_attrs:
if not hasattr(source, attr) or callable(getattr(source, attr)) or attr in ["data", "fig"]:
continue
try:
setattr(dest, attr, copy.deepcopy(getattr(source, attr)))
except (NotImplementedError, TypeError, ValueError): # Deepcopying failed, so just copy a reference instead
try:
setattr(dest, attr, getattr(source, attr))
except ValueError:
pass
dest._punlic_attrs = source._public_attrs
return dest