daemon: update CoreNode.newnetif to require parameters, CoreNode.newnetif now depends on being provided InterfaceData
This commit is contained in:
parent
3691c6029f
commit
3be15a1316
5 changed files with 71 additions and 130 deletions
|
@ -1,18 +1,22 @@
|
||||||
from dataclasses import dataclass, field
|
from dataclasses import dataclass, field
|
||||||
from typing import List, Optional, Union
|
from typing import TYPE_CHECKING, List, Optional, Union
|
||||||
|
|
||||||
import netaddr
|
import netaddr
|
||||||
|
|
||||||
from core import utils
|
from core import utils
|
||||||
from core.api.grpc.core_pb2 import LinkOptions
|
from core.api.grpc.core_pb2 import LinkOptions
|
||||||
from core.emulator.enumerations import LinkTypes
|
from core.emulator.enumerations import LinkTypes
|
||||||
from core.nodes.base import CoreNetworkBase, CoreNode
|
|
||||||
from core.nodes.interface import CoreInterface
|
from core.nodes.interface import CoreInterface
|
||||||
from core.nodes.physical import PhysicalNode
|
|
||||||
|
if TYPE_CHECKING:
|
||||||
|
from core.nodes.base import CoreNetworkBase, CoreNode
|
||||||
|
from core.nodes.physical import PhysicalNode
|
||||||
|
|
||||||
|
LinkConfigNode = Union[CoreNetworkBase, PhysicalNode]
|
||||||
|
|
||||||
|
|
||||||
def link_config(
|
def link_config(
|
||||||
node: Union[CoreNetworkBase, PhysicalNode],
|
node: "LinkConfigNode",
|
||||||
interface: CoreInterface,
|
interface: CoreInterface,
|
||||||
link_options: LinkOptions,
|
link_options: LinkOptions,
|
||||||
interface_two: CoreInterface = None,
|
interface_two: CoreInterface = None,
|
||||||
|
@ -160,7 +164,7 @@ class IpPrefixes:
|
||||||
if ip6_prefix:
|
if ip6_prefix:
|
||||||
self.ip6 = netaddr.IPNetwork(ip6_prefix)
|
self.ip6 = netaddr.IPNetwork(ip6_prefix)
|
||||||
|
|
||||||
def ip4_address(self, node: CoreNode) -> str:
|
def ip4_address(self, node: "CoreNode") -> str:
|
||||||
"""
|
"""
|
||||||
Convenience method to return the IP4 address for a node.
|
Convenience method to return the IP4 address for a node.
|
||||||
|
|
||||||
|
@ -171,7 +175,7 @@ class IpPrefixes:
|
||||||
raise ValueError("ip4 prefixes have not been set")
|
raise ValueError("ip4 prefixes have not been set")
|
||||||
return str(self.ip4[node.id])
|
return str(self.ip4[node.id])
|
||||||
|
|
||||||
def ip6_address(self, node: CoreNode) -> str:
|
def ip6_address(self, node: "CoreNode") -> str:
|
||||||
"""
|
"""
|
||||||
Convenience method to return the IP6 address for a node.
|
Convenience method to return the IP6 address for a node.
|
||||||
|
|
||||||
|
@ -183,7 +187,7 @@ class IpPrefixes:
|
||||||
return str(self.ip6[node.id])
|
return str(self.ip6[node.id])
|
||||||
|
|
||||||
def create_interface(
|
def create_interface(
|
||||||
self, node: CoreNode, name: str = None, mac: str = None
|
self, node: "CoreNode", name: str = None, mac: str = None
|
||||||
) -> InterfaceData:
|
) -> InterfaceData:
|
||||||
"""
|
"""
|
||||||
Creates interface data for linking nodes, using the nodes unique id for
|
Creates interface data for linking nodes, using the nodes unique id for
|
||||||
|
@ -225,24 +229,3 @@ class IpPrefixes:
|
||||||
ip6_mask=ip6_mask,
|
ip6_mask=ip6_mask,
|
||||||
mac=mac,
|
mac=mac,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
def create_interface(
|
|
||||||
node: CoreNode, network: CoreNetworkBase, interface_data: InterfaceData
|
|
||||||
):
|
|
||||||
"""
|
|
||||||
Create an interface for a node on a network using provided interface data.
|
|
||||||
|
|
||||||
:param node: node to create interface for
|
|
||||||
:param network: network to associate interface with
|
|
||||||
:param interface_data: interface data
|
|
||||||
:return: created interface
|
|
||||||
"""
|
|
||||||
node.newnetif(
|
|
||||||
network,
|
|
||||||
addrlist=interface_data.get_addresses(),
|
|
||||||
hwaddr=interface_data.mac,
|
|
||||||
ifindex=interface_data.id,
|
|
||||||
ifname=interface_data.name,
|
|
||||||
)
|
|
||||||
return node.netif(interface_data.id)
|
|
||||||
|
|
|
@ -19,13 +19,7 @@ from core.emane.emanemanager import EmaneManager
|
||||||
from core.emane.nodes import EmaneNet
|
from core.emane.nodes import EmaneNet
|
||||||
from core.emulator.data import ConfigData, EventData, ExceptionData, FileData, LinkData
|
from core.emulator.data import ConfigData, EventData, ExceptionData, FileData, LinkData
|
||||||
from core.emulator.distributed import DistributedController
|
from core.emulator.distributed import DistributedController
|
||||||
from core.emulator.emudata import (
|
from core.emulator.emudata import InterfaceData, LinkOptions, NodeOptions, link_config
|
||||||
InterfaceData,
|
|
||||||
LinkOptions,
|
|
||||||
NodeOptions,
|
|
||||||
create_interface,
|
|
||||||
link_config,
|
|
||||||
)
|
|
||||||
from core.emulator.enumerations import (
|
from core.emulator.enumerations import (
|
||||||
EventTypes,
|
EventTypes,
|
||||||
ExceptionLevels,
|
ExceptionLevels,
|
||||||
|
@ -360,11 +354,11 @@ class Session:
|
||||||
node_one.name,
|
node_one.name,
|
||||||
net_one.name,
|
net_one.name,
|
||||||
)
|
)
|
||||||
interface = create_interface(node_one, net_one, interface_one)
|
ifindex = node_one.newnetif(net_one, interface_one)
|
||||||
node_one_interface = interface
|
node_one_interface = node_one.netif(ifindex)
|
||||||
wireless_net = isinstance(net_one, (EmaneNet, WlanNode))
|
wireless_net = isinstance(net_one, (EmaneNet, WlanNode))
|
||||||
if not wireless_net:
|
if not wireless_net:
|
||||||
link_config(net_one, interface, link_options)
|
link_config(net_one, node_one_interface, link_options)
|
||||||
|
|
||||||
# network to node
|
# network to node
|
||||||
if node_two and net_one:
|
if node_two and net_one:
|
||||||
|
@ -373,11 +367,11 @@ class Session:
|
||||||
node_two.name,
|
node_two.name,
|
||||||
net_one.name,
|
net_one.name,
|
||||||
)
|
)
|
||||||
interface = create_interface(node_two, net_one, interface_two)
|
ifindex = node_two.newnetif(net_one, interface_two)
|
||||||
node_two_interface = interface
|
node_two_interface = node_two.netif(ifindex)
|
||||||
wireless_net = isinstance(net_one, (EmaneNet, WlanNode))
|
wireless_net = isinstance(net_one, (EmaneNet, WlanNode))
|
||||||
if not link_options.unidirectional and not wireless_net:
|
if not link_options.unidirectional and not wireless_net:
|
||||||
link_config(net_one, interface, link_options)
|
link_config(net_one, node_two_interface, link_options)
|
||||||
|
|
||||||
# network to network
|
# network to network
|
||||||
if net_one and net_two:
|
if net_one and net_two:
|
||||||
|
@ -1797,35 +1791,28 @@ class Session:
|
||||||
control_net = self.add_remove_control_net(net_index, remove, conf_required)
|
control_net = self.add_remove_control_net(net_index, remove, conf_required)
|
||||||
if not control_net:
|
if not control_net:
|
||||||
return
|
return
|
||||||
|
|
||||||
if not node:
|
if not node:
|
||||||
return
|
return
|
||||||
|
|
||||||
# ctrl# already exists
|
# ctrl# already exists
|
||||||
if node.netif(control_net.CTRLIF_IDX_BASE + net_index):
|
if node.netif(control_net.CTRLIF_IDX_BASE + net_index):
|
||||||
return
|
return
|
||||||
|
|
||||||
control_ip = node.id
|
|
||||||
|
|
||||||
try:
|
try:
|
||||||
address = control_net.prefix[control_ip]
|
ip4 = control_net.prefix[node.id]
|
||||||
prefix = control_net.prefix.prefixlen
|
ip4_mask = control_net.prefix.prefixlen
|
||||||
addrlist = [f"{address}/{prefix}"]
|
interface = InterfaceData(
|
||||||
|
id=control_net.CTRLIF_IDX_BASE + net_index,
|
||||||
|
name=f"ctrl{net_index}",
|
||||||
|
mac=utils.random_mac(),
|
||||||
|
ip4=ip4,
|
||||||
|
ip4_mask=ip4_mask,
|
||||||
|
)
|
||||||
|
ifindex = node.newnetif(control_net, interface)
|
||||||
|
node.netif(ifindex).control = True
|
||||||
except ValueError:
|
except ValueError:
|
||||||
msg = f"Control interface not added to node {node.id}. "
|
msg = f"Control interface not added to node {node.id}. "
|
||||||
msg += f"Invalid control network prefix ({control_net.prefix}). "
|
msg += f"Invalid control network prefix ({control_net.prefix}). "
|
||||||
msg += "A longer prefix length may be required for this many nodes."
|
msg += "A longer prefix length may be required for this many nodes."
|
||||||
logging.exception(msg)
|
logging.exception(msg)
|
||||||
return
|
|
||||||
|
|
||||||
interface1 = node.newnetif(
|
|
||||||
net=control_net,
|
|
||||||
ifindex=control_net.CTRLIF_IDX_BASE + net_index,
|
|
||||||
ifname=f"ctrl{net_index}",
|
|
||||||
hwaddr=utils.random_mac(),
|
|
||||||
addrlist=addrlist,
|
|
||||||
)
|
|
||||||
node.netif(interface1).control = True
|
|
||||||
|
|
||||||
def update_control_interface_hosts(
|
def update_control_interface_hosts(
|
||||||
self, net_index: int = 0, remove: bool = False
|
self, net_index: int = 0, remove: bool = False
|
||||||
|
|
|
@ -15,6 +15,7 @@ from core import utils
|
||||||
from core.configservice.dependencies import ConfigServiceDependencies
|
from core.configservice.dependencies import ConfigServiceDependencies
|
||||||
from core.constants import MOUNT_BIN, VNODED_BIN
|
from core.constants import MOUNT_BIN, VNODED_BIN
|
||||||
from core.emulator.data import LinkData, NodeData
|
from core.emulator.data import LinkData, NodeData
|
||||||
|
from core.emulator.emudata import InterfaceData
|
||||||
from core.emulator.enumerations import LinkTypes, MessageFlags, NodeTypes
|
from core.emulator.enumerations import LinkTypes, MessageFlags, NodeTypes
|
||||||
from core.errors import CoreCommandError, CoreError
|
from core.errors import CoreCommandError, CoreError
|
||||||
from core.nodes.client import VnodeClient
|
from core.nodes.client import VnodeClient
|
||||||
|
@ -845,53 +846,36 @@ class CoreNode(CoreNodeBase):
|
||||||
interface_name = self.ifname(ifindex)
|
interface_name = self.ifname(ifindex)
|
||||||
self.node_net_client.device_up(interface_name)
|
self.node_net_client.device_up(interface_name)
|
||||||
|
|
||||||
def newnetif(
|
def newnetif(self, net: "CoreNetworkBase", interface: InterfaceData) -> int:
|
||||||
self,
|
|
||||||
net: "CoreNetworkBase" = None,
|
|
||||||
addrlist: List[str] = None,
|
|
||||||
hwaddr: str = None,
|
|
||||||
ifindex: int = None,
|
|
||||||
ifname: str = None,
|
|
||||||
) -> int:
|
|
||||||
"""
|
"""
|
||||||
Create a new network interface.
|
Create a new network interface.
|
||||||
|
|
||||||
:param net: network to associate with
|
:param net: network to associate with
|
||||||
:param addrlist: addresses to add on the interface
|
:param interface: interface data for new interface
|
||||||
:param hwaddr: hardware address to set for interface
|
|
||||||
:param ifindex: index of interface to create
|
|
||||||
:param ifname: name for interface
|
|
||||||
:return: interface index
|
:return: interface index
|
||||||
"""
|
"""
|
||||||
if not addrlist:
|
addresses = interface.get_addresses()
|
||||||
addrlist = []
|
|
||||||
|
|
||||||
with self.lock:
|
with self.lock:
|
||||||
# TODO: emane specific code
|
# TODO: emane specific code
|
||||||
if net is not None and net.is_emane is True:
|
if net.is_emane is True:
|
||||||
ifindex = self.newtuntap(ifindex, ifname)
|
ifindex = self.newtuntap(interface.id, interface.name)
|
||||||
# TUN/TAP is not ready for addressing yet; the device may
|
# TUN/TAP is not ready for addressing yet; the device may
|
||||||
# take some time to appear, and installing it into a
|
# take some time to appear, and installing it into a
|
||||||
# namespace after it has been bound removes addressing;
|
# namespace after it has been bound removes addressing;
|
||||||
# save addresses with the interface now
|
# save addresses with the interface now
|
||||||
self.attachnet(ifindex, net)
|
self.attachnet(ifindex, net)
|
||||||
netif = self.netif(ifindex)
|
netif = self.netif(ifindex)
|
||||||
netif.sethwaddr(hwaddr)
|
netif.sethwaddr(interface.mac)
|
||||||
for address in utils.make_tuple(addrlist):
|
for address in addresses:
|
||||||
netif.addaddr(address)
|
netif.addaddr(address)
|
||||||
return ifindex
|
return ifindex
|
||||||
else:
|
else:
|
||||||
ifindex = self.newveth(ifindex, ifname)
|
ifindex = self.newveth(interface.id, interface.name)
|
||||||
|
|
||||||
if net is not None:
|
|
||||||
self.attachnet(ifindex, net)
|
self.attachnet(ifindex, net)
|
||||||
|
if interface.mac:
|
||||||
if hwaddr:
|
self.sethwaddr(ifindex, interface.mac)
|
||||||
self.sethwaddr(ifindex, hwaddr)
|
for address in addresses:
|
||||||
|
|
||||||
for address in utils.make_tuple(addrlist):
|
|
||||||
self.addaddr(ifindex, address)
|
self.addaddr(ifindex, address)
|
||||||
|
|
||||||
self.ifup(ifindex)
|
self.ifup(ifindex)
|
||||||
return ifindex
|
return ifindex
|
||||||
|
|
||||||
|
|
|
@ -10,10 +10,11 @@ from typing import IO, TYPE_CHECKING, List, Optional, Tuple
|
||||||
from core import utils
|
from core import utils
|
||||||
from core.constants import MOUNT_BIN, UMOUNT_BIN
|
from core.constants import MOUNT_BIN, UMOUNT_BIN
|
||||||
from core.emulator.distributed import DistributedServer
|
from core.emulator.distributed import DistributedServer
|
||||||
|
from core.emulator.emudata import InterfaceData
|
||||||
from core.emulator.enumerations import NodeTypes, TransportType
|
from core.emulator.enumerations import NodeTypes, TransportType
|
||||||
from core.errors import CoreCommandError, CoreError
|
from core.errors import CoreCommandError, CoreError
|
||||||
from core.nodes.base import CoreNetworkBase, CoreNodeBase
|
from core.nodes.base import CoreNetworkBase, CoreNodeBase
|
||||||
from core.nodes.interface import CoreInterface, Veth
|
from core.nodes.interface import CoreInterface
|
||||||
from core.nodes.network import CoreNetwork, GreTap
|
from core.nodes.network import CoreNetwork, GreTap
|
||||||
|
|
||||||
if TYPE_CHECKING:
|
if TYPE_CHECKING:
|
||||||
|
@ -168,37 +169,25 @@ class PhysicalNode(CoreNodeBase):
|
||||||
self.ifindex += 1
|
self.ifindex += 1
|
||||||
return ifindex
|
return ifindex
|
||||||
|
|
||||||
def newnetif(
|
def newnetif(self, net: CoreNetworkBase, interface: InterfaceData) -> int:
|
||||||
self,
|
|
||||||
net: Veth = None,
|
|
||||||
addrlist: List[str] = None,
|
|
||||||
hwaddr: str = None,
|
|
||||||
ifindex: int = None,
|
|
||||||
ifname: str = None,
|
|
||||||
) -> int:
|
|
||||||
logging.info("creating interface")
|
logging.info("creating interface")
|
||||||
if not addrlist:
|
addresses = interface.get_addresses()
|
||||||
addrlist = []
|
ifindex = interface.id
|
||||||
|
|
||||||
if self.up and net is None:
|
|
||||||
raise NotImplementedError
|
|
||||||
|
|
||||||
if ifindex is None:
|
if ifindex is None:
|
||||||
ifindex = self.newifindex()
|
ifindex = self.newifindex()
|
||||||
|
name = interface.name
|
||||||
if ifname is None:
|
if name is None:
|
||||||
ifname = f"gt{ifindex}"
|
name = f"gt{ifindex}"
|
||||||
|
|
||||||
if self.up:
|
if self.up:
|
||||||
# this is reached when this node is linked to a network node
|
# this is reached when this node is linked to a network node
|
||||||
# tunnel to net not built yet, so build it now and adopt it
|
# tunnel to net not built yet, so build it now and adopt it
|
||||||
_, remote_tap = self.session.distributed.create_gre_tunnel(net, self.server)
|
_, remote_tap = self.session.distributed.create_gre_tunnel(net, self.server)
|
||||||
self.adoptnetif(remote_tap, ifindex, hwaddr, addrlist)
|
self.adoptnetif(remote_tap, ifindex, interface.mac, addresses)
|
||||||
return ifindex
|
return ifindex
|
||||||
else:
|
else:
|
||||||
# this is reached when configuring services (self.up=False)
|
# this is reached when configuring services (self.up=False)
|
||||||
netif = GreTap(node=self, name=ifname, session=self.session, start=False)
|
netif = GreTap(node=self, name=name, session=self.session, start=False)
|
||||||
self.adoptnetif(netif, ifindex, hwaddr, addrlist)
|
self.adoptnetif(netif, ifindex, interface.mac, addresses)
|
||||||
return ifindex
|
return ifindex
|
||||||
|
|
||||||
def privatedir(self, path: str) -> None:
|
def privatedir(self, path: str) -> None:
|
||||||
|
@ -320,28 +309,19 @@ class Rj45Node(CoreNodeBase):
|
||||||
self.up = False
|
self.up = False
|
||||||
self.restorestate()
|
self.restorestate()
|
||||||
|
|
||||||
def newnetif(
|
def newnetif(self, net: CoreNetworkBase, interface: InterfaceData) -> int:
|
||||||
self,
|
|
||||||
net: CoreNetworkBase = None,
|
|
||||||
addrlist: List[str] = None,
|
|
||||||
hwaddr: str = None,
|
|
||||||
ifindex: int = None,
|
|
||||||
ifname: str = None,
|
|
||||||
) -> int:
|
|
||||||
"""
|
"""
|
||||||
This is called when linking with another node. Since this node
|
This is called when linking with another node. Since this node
|
||||||
represents an interface, we do not create another object here,
|
represents an interface, we do not create another object here,
|
||||||
but attach ourselves to the given network.
|
but attach ourselves to the given network.
|
||||||
|
|
||||||
:param net: new network instance
|
:param net: new network instance
|
||||||
:param addrlist: address list
|
:param interface: interface data for new interface
|
||||||
:param hwaddr: hardware address
|
|
||||||
:param ifindex: interface index
|
|
||||||
:param ifname: interface name
|
|
||||||
:return: interface index
|
:return: interface index
|
||||||
:raises ValueError: when an interface has already been created, one max
|
:raises ValueError: when an interface has already been created, one max
|
||||||
"""
|
"""
|
||||||
with self.lock:
|
with self.lock:
|
||||||
|
ifindex = interface.id
|
||||||
if ifindex is None:
|
if ifindex is None:
|
||||||
ifindex = 0
|
ifindex = 0
|
||||||
if self.interface.net is not None:
|
if self.interface.net is not None:
|
||||||
|
@ -350,8 +330,7 @@ class Rj45Node(CoreNodeBase):
|
||||||
self.ifindex = ifindex
|
self.ifindex = ifindex
|
||||||
if net is not None:
|
if net is not None:
|
||||||
self.interface.attachnet(net)
|
self.interface.attachnet(net)
|
||||||
if addrlist:
|
for addr in interface.get_addresses():
|
||||||
for addr in utils.make_tuple(addrlist):
|
|
||||||
self.addaddr(addr)
|
self.addaddr(addr)
|
||||||
return ifindex
|
return ifindex
|
||||||
|
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
import pytest
|
import pytest
|
||||||
|
|
||||||
from core.emulator.emudata import NodeOptions
|
from core.emulator.emudata import InterfaceData, NodeOptions
|
||||||
from core.emulator.session import Session
|
from core.emulator.session import Session
|
||||||
from core.errors import CoreError
|
from core.errors import CoreError
|
||||||
from core.nodes.base import CoreNode
|
from core.nodes.base import CoreNode
|
||||||
|
@ -52,7 +52,9 @@ class TestNodes:
|
||||||
def test_node_sethwaddr(self, session: Session):
|
def test_node_sethwaddr(self, session: Session):
|
||||||
# given
|
# given
|
||||||
node = session.add_node(CoreNode)
|
node = session.add_node(CoreNode)
|
||||||
index = node.newnetif()
|
switch = session.add_node(SwitchNode)
|
||||||
|
interface_data = InterfaceData()
|
||||||
|
index = node.newnetif(switch, interface_data)
|
||||||
interface = node.netif(index)
|
interface = node.netif(index)
|
||||||
mac = "aa:aa:aa:ff:ff:ff"
|
mac = "aa:aa:aa:ff:ff:ff"
|
||||||
|
|
||||||
|
@ -65,7 +67,9 @@ class TestNodes:
|
||||||
def test_node_sethwaddr_exception(self, session: Session):
|
def test_node_sethwaddr_exception(self, session: Session):
|
||||||
# given
|
# given
|
||||||
node = session.add_node(CoreNode)
|
node = session.add_node(CoreNode)
|
||||||
index = node.newnetif()
|
switch = session.add_node(SwitchNode)
|
||||||
|
interface_data = InterfaceData()
|
||||||
|
index = node.newnetif(switch, interface_data)
|
||||||
node.netif(index)
|
node.netif(index)
|
||||||
mac = "aa:aa:aa:ff:ff:fff"
|
mac = "aa:aa:aa:ff:ff:fff"
|
||||||
|
|
||||||
|
@ -76,7 +80,9 @@ class TestNodes:
|
||||||
def test_node_addaddr(self, session: Session):
|
def test_node_addaddr(self, session: Session):
|
||||||
# given
|
# given
|
||||||
node = session.add_node(CoreNode)
|
node = session.add_node(CoreNode)
|
||||||
index = node.newnetif()
|
switch = session.add_node(SwitchNode)
|
||||||
|
interface_data = InterfaceData()
|
||||||
|
index = node.newnetif(switch, interface_data)
|
||||||
interface = node.netif(index)
|
interface = node.netif(index)
|
||||||
addr = "192.168.0.1/24"
|
addr = "192.168.0.1/24"
|
||||||
|
|
||||||
|
@ -89,7 +95,9 @@ class TestNodes:
|
||||||
def test_node_addaddr_exception(self, session):
|
def test_node_addaddr_exception(self, session):
|
||||||
# given
|
# given
|
||||||
node = session.add_node(CoreNode)
|
node = session.add_node(CoreNode)
|
||||||
index = node.newnetif()
|
switch = session.add_node(SwitchNode)
|
||||||
|
interface_data = InterfaceData()
|
||||||
|
index = node.newnetif(switch, interface_data)
|
||||||
node.netif(index)
|
node.netif(index)
|
||||||
addr = "256.168.0.1/24"
|
addr = "256.168.0.1/24"
|
||||||
|
|
||||||
|
|
Loading…
Add table
Reference in a new issue