63 lines
2.0 KiB
Python
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"]
|