1
|
1 |
|
from .. import exceptions |
2
|
1 |
|
from ..log import logger |
3
|
|
|
|
4
|
1 |
|
class AttributesHolder(object): |
5
|
|
|
"""Stores attributes and provides read-only access to them.""" |
6
|
1 |
|
__slots__ = ('_attributes') |
7
|
1 |
|
_possible_attributes = None |
8
|
1 |
|
def __init__(self, *args, **attributes): |
9
|
1 |
|
attributes.update(dict(zip(self._possible_attributes, args))) |
10
|
1 |
|
self._check_attributes(attributes) |
11
|
1 |
|
self._parse_attributes(attributes) |
12
|
|
|
|
13
|
1 |
|
def _check_attributes(self, attributes, extra=None): |
14
|
|
|
"""Check if attributes given to the constructor can be used to |
15
|
|
|
instanciate a valid node.""" |
16
|
1 |
|
extra = extra or () |
17
|
1 |
|
unknown_keys = set(attributes) - set(self._possible_attributes) - set(extra) |
18
|
1 |
|
if unknown_keys: |
19
|
1 |
|
logger.warning('%s got unknown attributes: %s' % |
20
|
|
|
(self.__class__.__name__, unknown_keys)) |
21
|
|
|
|
22
|
1 |
|
def __repr__(self): |
23
|
|
|
return '<%s %r>' % (self.__class__.__name__, |
24
|
|
|
{x:y for (x,y) in self._attributes.items()}) |
25
|
|
|
|
26
|
1 |
|
def _parse_attributes(self, attributes): |
27
|
1 |
|
self._attributes = attributes |
28
|
|
|
|
29
|
1 |
|
def __eq__(self, other): |
30
|
|
|
"""Tests equality with another AttributesHolder instance.""" |
31
|
1 |
|
if isinstance(other, AttributesHolder): |
32
|
1 |
|
return self._attributes == other._attributes |
33
|
|
|
else: |
34
|
|
|
return False |
35
|
1 |
|
def __ne__(self, other): |
36
|
1 |
|
return not (self == other) |
37
|
|
|
|
38
|
1 |
|
def __hash__(self): |
39
|
|
|
return hash(frozenset(self._attributes.items())) |
40
|
|
|
|
41
|
1 |
|
def get(self, name, strict=True): |
42
|
|
|
"""Get an attribute of the holder (read-only access).""" |
43
|
1 |
|
if not isinstance(name, str) or name.startswith('_'): |
44
|
|
|
raise AttributeError(self.__class__.__name__, name) |
45
|
1 |
|
elif strict and name not in self._possible_attributes: |
46
|
1 |
|
raise AttributeError('%s is not a valid attribute of %r.' % |
47
|
|
|
(name, self)) |
48
|
1 |
|
elif name in self._attributes: |
49
|
1 |
|
return self._attributes[name] |
50
|
|
|
else: |
51
|
1 |
|
raise exceptions.AttributeNotProvided(name) |
52
|
1 |
|
__getattr__ = __getitem__ = get |
53
|
|
|
|
54
|
1 |
|
def __setattr__(self, name, value): |
55
|
1 |
|
if name.startswith('_'): |
56
|
1 |
|
super(AttributesHolder, self).__setattr__(name, value) |
57
|
|
|
else: |
58
|
1 |
|
raise TypeError('%s\'s attributes are not settable.' % |
59
|
|
|
self.__class__.__name__) |
60
|
1 |
|
def __delattr__(self, name): |
61
|
1 |
|
if name.startswith('_'): |
62
|
|
|
super(AttributesHolder, self).__delattr__(name, value) |
63
|
|
|
else: |
64
|
1 |
|
raise TypeError('%s\'s attributes are not settable.' % |
65
|
|
|
self.__class__.__name__) |
66
|
|
|
|
67
|
1 |
|
def has(self, name): |
68
|
|
|
"""Check existence of an attribute.""" |
69
|
1 |
|
return name in self._attributes |
70
|
1 |
|
__hasattr__ = __contains__ = has |
71
|
|
|
|
72
|
|
|
|