ffmap-backend/ffmap/node.py

90 lines
2.6 KiB
Python
Raw Normal View History

from collections import defaultdict
class NoneDict:
2014-07-07 23:27:21 +02:00
"""Act like None but return a NoneDict for every item request.
2014-07-07 23:27:21 +02:00
This is similar to the behaviour of collections.defaultdict in that
even previously inexistent keys can be accessed, but nothing is
stored permanently in this class.
"""
def __repr__(self):
return 'NoneDict()'
def __bool__(self):
return False
def __getitem__(self, k):
return NoneDict()
def __json__(self):
return None
def __float__(self):
return float('NaN')
def __iter__(self):
# empty generator
return
yield
2014-07-07 23:27:21 +02:00
def __setitem__(self, key, value):
raise RuntimeError("NoneDict is readonly")
2014-07-07 23:27:21 +02:00
class Node(defaultdict):
_id = None
def __init__(self, id_=None):
self._id = id_
super().__init__(NoneDict)
2014-07-07 23:27:21 +02:00
def __repr__(self):
return "Node(%s)" % self.id
2012-05-11 14:12:41 +02:00
2014-07-07 23:27:21 +02:00
@property
def id(self):
return self._id
2012-06-06 03:34:52 +02:00
2014-07-07 23:27:21 +02:00
def __hash__(self):
"""Generate hash from the node's id.
2012-05-11 14:12:41 +02:00
2014-07-07 23:27:21 +02:00
WARNING: Obviously this hash doesn't cover all of the node's
data, but we need nodes to be hashable in order to eliminate
duplicates in the NodeDB.
2014-07-07 23:27:21 +02:00
At least the id cannot change after initialization...
"""
return hash(self.id)
2012-05-11 14:12:41 +02:00
def deep_update(self, other):
"""Update the dictionary like dict.update() but recursively."""
def dmerge(a, b):
for k, v in b.items():
if isinstance(v, dict) and isinstance(a.get(k), dict):
dmerge(a[k], v)
else:
a[k] = v
dmerge(self, other)
2014-07-07 23:27:21 +02:00
@property
def vpn_neighbors(self):
try:
vpn_neighbors = []
for neighbor in self['neighbors']:
if neighbor['neighbor']['vpn']:
vpn_neighbors.append(neighbor)
return vpn_neighbors
except TypeError:
return []
def export(self):
"""Generate a serializable dict of the node.
2014-07-07 23:27:21 +02:00
In particular, this replaces any references to other nodes by
their id to prevent circular references.
"""
ret = dict(self)
if "neighbors" in self:
ret["neighbors"] = []
for neighbor in self["neighbors"]:
new_neighbor = {}
for key, val in neighbor.items():
if isinstance(val, Node):
new_neighbor[key] = val.id
else:
new_neighbor[key] = val
ret["neighbors"].append(new_neighbor)
return ret