python-benedict/benedict/dicts/keyattr/keyattr_dict.py

63 lines
2.0 KiB
Python

from benedict.dicts.base import BaseDict
class KeyattrDict(BaseDict):
_keyattr_enabled = None
_keyattr_dynamic = None
def __init__(self, *args, **kwargs):
self._keyattr_enabled = kwargs.pop("keyattr_enabled", True)
self._keyattr_dynamic = kwargs.pop("keyattr_dynamic", False)
super().__init__(*args, **kwargs)
@property
def keyattr_enabled(self):
return self._keyattr_enabled
@keyattr_enabled.setter
def keyattr_enabled(self, value):
self._keyattr_enabled = value
@property
def keyattr_dynamic(self):
return self._keyattr_dynamic
@keyattr_dynamic.setter
def keyattr_dynamic(self, value):
self._keyattr_dynamic = value
def __getattr__(self, attr):
attr_message = f"{self.__class__.__name__!r} object has no attribute {attr!r}"
if not self._keyattr_enabled:
raise AttributeError(attr_message)
try:
return self.__getitem__(attr)
except KeyError:
if attr.startswith("_"):
raise AttributeError(attr_message) from None
if not self._keyattr_dynamic:
raise AttributeError(attr_message) from None
self.__setitem__(attr, {})
return self.__getitem__(attr)
def __setattr__(self, attr, value):
attr_message = f"{self.__class__.__name__!r} object has no attribute {attr!r}"
if attr in self:
# set existing key
if not self._keyattr_enabled:
raise AttributeError(attr_message)
self.__setitem__(attr, value)
elif hasattr(self.__class__, attr):
# set existing attr
super().__setattr__(attr, value)
else:
# set new key
if not self._keyattr_enabled:
raise AttributeError(attr_message)
self.__setitem__(attr, value)
def __setstate__(self, state):
super().__setstate__(state)
self._keyattr_enabled = state["_keyattr_enabled"]
self._keyattr_dynamic = state["_keyattr_dynamic"]