Add basic network representation and comparison.
Very PoC quality code, but will do for now.old_stateparser
parent
9edd3dea02
commit
dcf8b0184e
@ -0,0 +1,85 @@
|
||||
from birdvisu.ospffile import load
|
||||
import re
|
||||
|
||||
class Topology:
|
||||
"""Basically a graph"""
|
||||
|
||||
def __init__(self):
|
||||
self.routers = [] # Tuples like in ospffile for router section
|
||||
self.networks = [] # Ditto, but the key is the address, not the ad-hoc string from bird
|
||||
self.network_addrs = {} # Keeps mapping from network to address
|
||||
self.links = [] # Tuples (router, network_addr, cost)
|
||||
# FIXME: we are ugly, so everything are strings and I don't care as of now.
|
||||
|
||||
def add_area(self, dirtuple):
|
||||
# FIXME: Do not use BIRD's strings, create an abstraction above them.
|
||||
_area, topo = dirtuple # Ignoring the area
|
||||
for obj in topo:
|
||||
directive, details = obj
|
||||
if directive.startswith('router'):
|
||||
self.routers.append(obj)
|
||||
elif directive.startswith('network'):
|
||||
net_addr = None
|
||||
for d, _ in details:
|
||||
if d.startswith('address'):
|
||||
net_addr = d
|
||||
break
|
||||
assert net_addr is not None
|
||||
fixed_network = (net_addr, details + [directive])
|
||||
self.networks.append(fixed_network)
|
||||
self.network_addrs[directive] = net_addr
|
||||
|
||||
# Fix the topology: find other networks (stubnets, external, ...) and links:
|
||||
for r, rd in self.routers:
|
||||
for n, nd in rd:
|
||||
if n.startswith(('network', 'stubnet', 'external')):
|
||||
net_id = re.match(r'((network|stubnet|external) [^ ]+)', n).group(1)
|
||||
if n.startswith(('stubnet', 'external')):
|
||||
self.networks.append((net_id, []))
|
||||
# Add dummy mapping
|
||||
self.network_addrs[net_id] = net_id
|
||||
metric = int(re.search(r'metric ([0-9]+)', n).group(1))
|
||||
net_addr = self.network_addrs[net_id]
|
||||
self.links.append((r, net_addr, metric))
|
||||
|
||||
@classmethod
|
||||
def from_ospffile(cls, f):
|
||||
# FIXME: We should create own classes from the OSPF file as soon as
|
||||
# possible to avoid changing whole codebase when BIRD's format changes.
|
||||
result = cls()
|
||||
parsed = load(f)
|
||||
for directive, details in parsed:
|
||||
if directive.startswith('area'):
|
||||
result.add_area((directive, details))
|
||||
return result
|
||||
|
||||
|
||||
class TopologyDifference:
|
||||
def __init__(self, actual, reference):
|
||||
self.actual = actual
|
||||
self.reference = reference
|
||||
|
||||
def compare(self):
|
||||
# FIXME: This also relies on BIRD's current format.
|
||||
just_ident = lambda x: x[0]
|
||||
act_routers = set(map(just_ident, self.actual.routers))
|
||||
ref_routers = set(map(just_ident, self.reference.routers))
|
||||
act_networks = set(map(just_ident, self.actual.networks))
|
||||
ref_networks = set(map(just_ident, self.reference.networks))
|
||||
act_links = set(self.actual.links)
|
||||
ref_links = set(self.reference.links)
|
||||
|
||||
# *_missing: in reference, not in actual network.
|
||||
# *_extra: in actual network, not in reference.
|
||||
# *_discrepancies: different settings (currently only metrics)
|
||||
self.routers_missing = ref_routers - act_routers
|
||||
self.routers_extra = act_routers - ref_routers
|
||||
self.networks_missing = ref_networks - act_networks
|
||||
self.networks_extra = act_networks - ref_networks
|
||||
|
||||
# FIXME: be more clever. Discrepancy is not missing and extra…
|
||||
self.links_missing = ref_links - act_links
|
||||
self.links_extra = act_links - ref_links
|
||||
self.links_discrepancies = act_links ^ ref_links
|
||||
|
||||
|
Loading…
Reference in New Issue