2023-08-24 12:40:00 +01:00
|
|
|
from typing import Any, Dict, Union
|
2023-08-16 16:45:52 +01:00
|
|
|
|
2023-08-23 14:41:30 +01:00
|
|
|
from primaite import getLogger
|
2023-08-16 16:45:52 +01:00
|
|
|
from primaite.simulator.core import Action, ActionManager, AllowAllValidator, SimComponent
|
2023-08-24 12:40:00 +01:00
|
|
|
from primaite.simulator.network.hardware.base import Link, NIC, Node, SwitchPort
|
2023-08-16 16:45:52 +01:00
|
|
|
|
2023-08-23 14:41:30 +01:00
|
|
|
_LOGGER = getLogger(__name__)
|
|
|
|
|
|
2023-08-16 16:45:52 +01:00
|
|
|
|
2023-08-24 12:41:46 +01:00
|
|
|
class Network(SimComponent):
|
2023-08-21 10:04:23 +01:00
|
|
|
"""Top level container object representing the physical network."""
|
2023-08-16 16:45:52 +01:00
|
|
|
|
|
|
|
|
nodes: Dict[str, Node] = {}
|
|
|
|
|
links: Dict[str, Link] = {}
|
|
|
|
|
|
|
|
|
|
def __init__(self, **kwargs):
|
2023-08-21 10:04:23 +01:00
|
|
|
"""Initialise the network."""
|
2023-08-16 16:45:52 +01:00
|
|
|
super().__init__(**kwargs)
|
|
|
|
|
|
2023-08-17 15:32:12 +01:00
|
|
|
def describe_state(self) -> Dict:
|
|
|
|
|
"""
|
|
|
|
|
Produce a dictionary describing the current state of this object.
|
|
|
|
|
|
|
|
|
|
Please see :py:meth:`primaite.simulator.core.SimComponent.describe_state` for a more detailed explanation.
|
|
|
|
|
|
|
|
|
|
:return: Current state of this object and child objects.
|
|
|
|
|
:rtype: Dict
|
|
|
|
|
"""
|
|
|
|
|
state = super().describe_state()
|
|
|
|
|
state.update(
|
|
|
|
|
{
|
|
|
|
|
"nodes": {uuid: node.describe_state() for uuid, node in self.nodes.items()},
|
|
|
|
|
"links": {uuid: link.describe_state() for uuid, link in self.links.items()},
|
|
|
|
|
}
|
|
|
|
|
)
|
|
|
|
|
return state
|
2023-08-23 14:41:30 +01:00
|
|
|
|
2023-08-28 22:34:20 +01:00
|
|
|
def _init_action_manager(self) -> ActionManager:
|
|
|
|
|
am = super()._init_action_manager()
|
|
|
|
|
|
|
|
|
|
am.add_action(
|
|
|
|
|
"node",
|
|
|
|
|
Action(
|
|
|
|
|
func=lambda request, context: self.nodes[request.pop(0)].apply_action(request, context),
|
|
|
|
|
validator=AllowAllValidator(),
|
|
|
|
|
),
|
|
|
|
|
)
|
|
|
|
|
return am
|
|
|
|
|
|
2023-08-23 14:41:30 +01:00
|
|
|
def add_node(self, node: Node) -> None:
|
|
|
|
|
"""
|
|
|
|
|
Add an existing node to the network.
|
|
|
|
|
|
|
|
|
|
:param node: Node instance that the network should keep track of.
|
|
|
|
|
:type node: Node
|
|
|
|
|
"""
|
|
|
|
|
if node in self:
|
2023-08-25 15:58:07 +01:00
|
|
|
_LOGGER.warning(f"Can't add node {node.uuid}. It is already in the network.")
|
|
|
|
|
return
|
2023-08-23 14:41:30 +01:00
|
|
|
self.nodes[node.uuid] = node
|
|
|
|
|
node.parent = self
|
2023-08-25 15:58:07 +01:00
|
|
|
_LOGGER.info(f"Added node {node.uuid} to Network {self.uuid}")
|
2023-08-23 14:41:30 +01:00
|
|
|
|
|
|
|
|
def remove_node(self, node: Node) -> None:
|
|
|
|
|
"""
|
|
|
|
|
Remove a node from the network.
|
|
|
|
|
|
|
|
|
|
:param node: Node instance that is currently part of the network that should be removed.
|
|
|
|
|
:type node: Node
|
|
|
|
|
"""
|
|
|
|
|
if node not in self:
|
2023-08-25 15:58:07 +01:00
|
|
|
_LOGGER.warning(f"Can't remove node {node.uuid}. It's not in the network.")
|
|
|
|
|
return
|
|
|
|
|
self.nodes.pop(node.uuid)
|
|
|
|
|
node.parent = None
|
|
|
|
|
_LOGGER.info(f"Removed node {node.uuid} from network {self.uuid}")
|
2023-08-23 14:41:30 +01:00
|
|
|
|
2023-08-24 13:03:16 +01:00
|
|
|
def connect(self, endpoint_a: Union[NIC, SwitchPort], endpoint_b: Union[NIC, SwitchPort], **kwargs) -> None:
|
2023-08-24 12:40:00 +01:00
|
|
|
"""Connect two nodes on the network by creating a link between an NIC/SwitchPort of each one.
|
2023-08-23 14:41:30 +01:00
|
|
|
|
2023-08-24 12:40:00 +01:00
|
|
|
:param endpoint_a: The endpoint to which to connect the link on the first node
|
|
|
|
|
:type endpoint_a: Union[NIC, SwitchPort]
|
|
|
|
|
:param endpoint_b: The endpoint to which to connct the link on the second node
|
|
|
|
|
:type endpoint_b: Union[NIC, SwitchPort]
|
|
|
|
|
:raises RuntimeError: _description_
|
|
|
|
|
"""
|
|
|
|
|
node_a = endpoint_a.parent
|
|
|
|
|
node_b = endpoint_b.parent
|
|
|
|
|
if node_a not in self:
|
2023-08-25 15:58:07 +01:00
|
|
|
self.add_node(node_a)
|
2023-08-24 12:40:00 +01:00
|
|
|
if node_b not in self:
|
2023-08-25 15:58:07 +01:00
|
|
|
self.add_node(node_b)
|
2023-08-24 12:40:00 +01:00
|
|
|
if node_a is node_b:
|
2023-08-25 15:58:07 +01:00
|
|
|
_LOGGER.warn(f"Cannot link endpoint {endpoint_a} to {endpoint_b} because they belong to the same node.")
|
|
|
|
|
return
|
2023-08-24 12:40:00 +01:00
|
|
|
|
|
|
|
|
link = Link(endpoint_a=endpoint_a, endpoint_b=endpoint_b, **kwargs)
|
|
|
|
|
self.links[link.uuid] = link
|
|
|
|
|
link.parent = self
|
2023-08-25 15:58:07 +01:00
|
|
|
_LOGGER.info(f"Added link {link.uuid} to connect {endpoint_a} and {endpoint_b}")
|
2023-08-24 12:40:00 +01:00
|
|
|
|
|
|
|
|
def remove_link(self, link: Link) -> None:
|
|
|
|
|
"""Disconnect a link from the network.
|
|
|
|
|
|
|
|
|
|
:param link: The link to be removed
|
|
|
|
|
:type link: Link
|
|
|
|
|
"""
|
|
|
|
|
link.endpoint_a.disconnect_link()
|
|
|
|
|
link.endpoint_b.disconnect_link()
|
2023-08-25 15:58:07 +01:00
|
|
|
self.links.pop(link.uuid)
|
|
|
|
|
link.parent = None
|
|
|
|
|
_LOGGER.info(f"Removed link {link.uuid} from network {self.uuid}.")
|
2023-08-23 14:41:30 +01:00
|
|
|
|
|
|
|
|
def __contains__(self, item: Any) -> bool:
|
|
|
|
|
if isinstance(item, Node):
|
|
|
|
|
return item.uuid in self.nodes
|
|
|
|
|
elif isinstance(item, Link):
|
|
|
|
|
return item.uuid in self.links
|
2023-08-25 15:58:07 +01:00
|
|
|
return False
|