blob: b05d69735e341c5f7418ae1b720db10b80db47f4 [file] [log] [blame]
# -*- Mode:python; c-file-style:"gnu"; indent-tabs-mode:nil -*- */
#
# Copyright (C) 2015-2019, The University of Memphis
#
# This file is part of Mini-NDN.
# See AUTHORS.md for a complete list of Mini-NDN authors and contributors.
#
# Mini-NDN is free software: you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
#
# Mini-NDN is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with Mini-NDN, e.g., in COPYING.md file.
# If not, see <http://www.gnu.org/licenses/>.
# IMPORTANT! This feature is in highly experimental phase and may go several changes
# in future
from mininet.log import info
from ndn.apps.nfdc import Nfdc as nfdc
from collections import defaultdict
from ndn.apps.calculate_routing import CalculateRoutes
from mininet.log import warn
class GlobalRoutingHelper():
"""
This module is a helper class which helps to create face and register routes
to NFD from a given node to all of its neighbors.
:param NetObject netObject: Mininet net object
:param FaceType faceType: UDP, Ethernet etc.
:param Routing routingType: (optional) Routing algorithm, link-state or hr etc
"""
def __init__(self, netObject, faceType=nfdc.PROTOCOL_UDP, routingType="link-state"):
self.net = netObject
self.faceType = faceType
self.routingType = routingType
self.routes = []
self.namePrefixes = {host_name.name: [] for host_name in self.net.hosts}
self.routeObject = CalculateRoutes(self.net, self.routingType)
def globalRoutingHelperHandler(self):
for host in self.net.hosts:
neighborIPs = self.getNeighbor(host)
self.createFaces(host, neighborIPs)
self.routeAdd(host, neighborIPs)
info('Processed all the routes to NFD\n')
def addOrigin(self, nodes, prefix):
"""
Add prefix/s as origin on node/s
:param Prefix prefix: Prefix that is originated by node/s (as producer) for this prefix
:param Nodes nodes: List of nodes from net object
"""
for node in nodes:
self.namePrefixes[node.name] = prefix
def calculateNPossibleRoutes(self, nFaces=0):
"""
By default, calculates all possible routes i.e. routes via all the faces of a node.
pass nFaces if want to compute routes via n number of faces. e.g. 2. For larger topology
the computation might take huge amount of time.
:param int nFaces: (optional) number of faces to consider while computing routes. Default
i.e. nFaces = 0 will compute all possible routes
"""
self.routes = self.routeObject.getRoutes(nFaces)
if self.routes:
self.globalRoutingHelperHandler()
else:
warn("Route computation failed\n")
def calculateRoutes(self):
# Calculate shortest path for every node
calculateNPossibleRoutes(self, nFaces=1)
def createFaces(self, node, neighborIPs):
for ip in neighborIPs.values():
nfdc.createFace(node, ip, self.faceType)
def routeAdd(self, node, neighborIPs):
"""
Add route from a node to its neighbors for each prefix/s advertised by destination node
:param Node node: source node (Mininet net.host)
:param IP neighborIPs: IP addresses of neighbors
"""
neighbors = self.routes[node.name]
for route in neighbors:
destination = route[0]
cost = int(route[1])
nextHop = route[2]
defaultPrefix = "/ndn/{}-site/{}".format(destination, destination)
prefixes = [defaultPrefix] + self.namePrefixes[destination]
for prefix in prefixes:
# Register routes to all the available destination name prefix/s
nfdc.registerRoute(node, prefix, neighborIPs[nextHop], \
nfdc.PROTOCOL_UDP, cost=cost)
@staticmethod
def getNeighbor(node):
# Nodes to IP mapping
neighborIPs = defaultdict()
for intf in node.intfList():
link = intf.link
if link:
node1, node2 = link.intf1.node, link.intf2.node
if node1 == node:
other = node2
ip = other.IP(str(link.intf2))
else:
other = node1
ip = other.IP(str(link.intf1))
# Used later to create faces
neighborIPs[other.name] = ip
return neighborIPs