#2967 - Enhance AirSpace simulation with dynamic environment and bandwidth/channel management
This commit introduces several key enhancements to the AirSpace class, improving the realism and configurability of the wireless network. Major additions include the AirSpaceEnvironmentType and ChannelWidth enums, dynamic adjustment of interface speeds based on environmental settings, and comprehensive bandwidth management features. Additionally, the software now supports configuration of channel widths via the config file, incorporates accurate SNR and capacity calculations, and enforces bandwidth limits more effectively across wireless interfaces. Updated tests ensure that the new functionalities integrate seamlessly with existing systems.
This commit is contained in:
35
CHANGELOG.md
35
CHANGELOG.md
@@ -2,9 +2,42 @@
|
||||
|
||||
All notable changes to this project will be documented in this file.
|
||||
|
||||
The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/),
|
||||
The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.1.0/),
|
||||
and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).
|
||||
|
||||
## [Unreleased]
|
||||
### Added
|
||||
|
||||
- **AirSpaceEnvironmentType Enum Class**: Introduced in `airspace.py` to define different environmental settings affecting wireless network behavior.
|
||||
- **ChannelWidth Enum Class**: Added in `airspace.py` to specify channel width options for wireless network interfaces.
|
||||
- **Channel Width Attribute**: Incorporated into the `WirelessNetworkInterface` class to allow dynamic setting based on `AirSpaceFrequency` and `AirSpaceEnvironmentType`.
|
||||
- **SNR and Capacity Calculation Functions**: Functions `estimate_snr` and `calculate_total_channel_capacity` added to `airspace.py` for computing signal-to-noise ratio and capacity based on frequency and channel width.
|
||||
- **Dynamic Speed Setting**: WirelessInterface speed attribute now dynamically adjusts based on the operational environment, frequency, and channel width.
|
||||
- **airspace_key Attribute**: Added to `WirelessNetworkInterface` as a tuple of frequency and channel width, serving as a key for bandwidth/channel management.
|
||||
- **airspace_environment_type Attribute**: Determines the environmental type for the airspace, influencing data rate calculations and capacity sharing.
|
||||
- **show_bandwidth_load Function**: Displays current bandwidth load for each frequency and channel width in the airspace.
|
||||
- **Configuration Schema Update**: The `simulation.network` config file now includes settings for the `airspace_environment_type`.
|
||||
- **Bandwidth Tracking**: Tracks data transmission across each frequency/channel width pairing.
|
||||
- **Configuration Support for Wireless Routers**: `channel_width` can now be configured in the config file under `wireless_access_point`.
|
||||
- **New Tests**: Added to validate the respect of bandwidth capacities and the correct parsing of airspace configurations from YAML files.
|
||||
|
||||
### Changed
|
||||
|
||||
- **NetworkInterface Speed Type**: The `speed` attribute of `NetworkInterface` has been changed from `int` to `float`.
|
||||
- **Transmission Feasibility Check**: Updated `_can_transmit` function in `Link` to account for current load and total bandwidth capacity, ensuring transmissions do not exceed limits.
|
||||
- **Frame Size Details**: Frame `size` attribute now includes both core size and payload size in bytes.
|
||||
- **WirelessRouter Configuration Function**: `configure_wireless_access_point` function now accepts `channel_width` as a parameter.
|
||||
- **Interface Grouping**: `WirelessNetworkInterfaces` are now grouped by both `AirSpaceFrequency` and `ChannelWidth`.
|
||||
- **Interface Frequency/Channel Width Adjustment**: Changing an interface's settings now involves removal from the airspace, recalculation of its data rate, and re-addition under new settings.
|
||||
- **Transmission Blocking**: Enhanced `AirSpace` logic to block transmissions that would exceed the available capacity.
|
||||
|
||||
### Fixed
|
||||
|
||||
- **Transmission Permission Logic**: Corrected the logic in `can_transmit_frame` to accurately prevent overloads by checking if the transmission of a frame stays within allowable bandwidth limits after considering current load.
|
||||
|
||||
|
||||
[//]: # (This file needs tidying up between 2.0.0 and this line as it hasn't been segmented into 3.0.0 and 3.1.0 and isn't compliant with https://keepachangelog.com/en/1.1.0/)
|
||||
|
||||
## 3.0.0b9
|
||||
- Removed deprecated `PrimaiteSession` class.
|
||||
- Added ability to set log levels via configuration.
|
||||
|
||||
@@ -7,7 +7,7 @@
|
||||
==============
|
||||
In this section the network layout is defined. This part of the config follows a hierarchical structure. Almost every component defines a ``ref`` field which acts as a human-readable unique identifier, used by other parts of the config, such as agents.
|
||||
|
||||
At the top level of the network are ``nodes`` and ``links``.
|
||||
At the top level of the network are ``nodes``, ``links`` and ``airspace``.
|
||||
|
||||
e.g.
|
||||
|
||||
@@ -19,6 +19,9 @@ e.g.
|
||||
...
|
||||
links:
|
||||
...
|
||||
airspace:
|
||||
...
|
||||
|
||||
|
||||
``nodes``
|
||||
---------
|
||||
@@ -101,3 +104,11 @@ This accepts an integer value e.g. if port 1 is to be connected, the configurati
|
||||
``bandwidth``
|
||||
|
||||
This is an integer value specifying the allowed bandwidth across the connection. Units are in Mbps.
|
||||
|
||||
``airspace``
|
||||
------------
|
||||
|
||||
This is where the airspace settings for wireless networks arte set.
|
||||
|
||||
``airspace_environment_type``
|
||||
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
|
||||
|
||||
@@ -27,6 +27,7 @@ Contents
|
||||
simulation_components/network/nodes/firewall
|
||||
simulation_components/network/switch
|
||||
simulation_components/network/network
|
||||
simulation_components/network/airspace
|
||||
simulation_components/system/internal_frame_processing
|
||||
simulation_components/system/sys_log
|
||||
simulation_components/system/pcap
|
||||
|
||||
100
docs/source/simulation_components/network/airspace.rst
Normal file
100
docs/source/simulation_components/network/airspace.rst
Normal file
@@ -0,0 +1,100 @@
|
||||
.. only:: comment
|
||||
|
||||
© Crown-owned copyright 2024, Defence Science and Technology Laboratory UK
|
||||
|
||||
.. _airspace:
|
||||
|
||||
AirSpace
|
||||
========
|
||||
|
||||
|
||||
1. Introduction
|
||||
---------------
|
||||
|
||||
The AirSpace class is the central component for wireless networks in PrimAITE and is designed to model and manage the behavior and interactions of wireless network interfaces within a simulated wireless network environment. This documentation provides a detailed overview of the AirSpace class, its components, and how they interact to create a realistic simulation of wireless network dynamics.
|
||||
|
||||
2. Overview of the AirSpace System
|
||||
----------------------------------
|
||||
|
||||
The AirSpace is a virtual representation of a physical wireless environment, managing multiple wireless network interfaces that simulate devices connected to the wireless network. These interfaces communicate over radio frequencies, with their interactions influenced by various factors modeled within the AirSpace.
|
||||
|
||||
2.1 Key Components
|
||||
^^^^^^^^^^^^^^^^^^
|
||||
|
||||
- **Wireless Network Interfaces**: Representations of network interfaces connected physical devices like routers, computers, or IoT devices that can send and receive data wirelessly.
|
||||
- **Environmental Settings**: Different types of environments (e.g., urban, rural) that affect signal propagation and interference.
|
||||
- **Channel Management**: Handles channels and their widths (e.g., 20 MHz, 40 MHz) to determine data transmission over different frequencies.
|
||||
- **Bandwidth Management**: Tracks data transmission over channels to prevent overloading and simulate real-world network congestion.
|
||||
|
||||
3. AirSpace Environment Types
|
||||
-----------------------------
|
||||
|
||||
The AirspaceEnvironmentType is a critical component that simulates different physical environments:
|
||||
|
||||
- Urban, Suburban, Rural, etc.
|
||||
- Each type simulates different levels of electromagnetic interference and signal propagation characteristics.
|
||||
- Changing the AirspaceEnvironmentType impacts data rates by affecting the signal-to-noise ratio (SNR).
|
||||
|
||||
4. Simulation of Environment Changes
|
||||
------------------------------------
|
||||
|
||||
When an AirspaceEnvironmentType is set or changed, the AirSpace:
|
||||
|
||||
1. Recalculates the maximum data transmission capacities for all managed frequencies and channel widths.
|
||||
2. Updates all wireless interfaces to reflect new capacities.
|
||||
|
||||
5. Managing Wireless Network Interfaces
|
||||
---------------------------------------
|
||||
|
||||
- Interfaces can be dynamically added or removed.
|
||||
- Configurations can be changed in real-time.
|
||||
- The AirSpace handles data transmissions, ensuring data sent by an interface is received by all other interfaces on the same frequency and channel.
|
||||
|
||||
6. Signal-to-Noise Ratio (SNR) Calculation
|
||||
------------------------------------------
|
||||
|
||||
SNR is crucial in determining the quality of a wireless communication channel:
|
||||
|
||||
.. math::
|
||||
|
||||
SNR = \frac{\text{Signal Power}}{\text{Noise Power}}
|
||||
|
||||
- Impacted by environment type, frequency, and channel width
|
||||
- Higher SNR indicates a clearer signal, leading to higher data transmission rates
|
||||
|
||||
7. Total Channel Capacity Calculation
|
||||
-------------------------------------
|
||||
|
||||
Channel capacity is calculated using the Shannon-Hartley theorem:
|
||||
|
||||
.. math::
|
||||
|
||||
C = B \cdot \log_2(1 + SNR)
|
||||
|
||||
Where:
|
||||
|
||||
- C: channel capacity in bits per second (bps)
|
||||
- B: bandwidth of the channel in hertz (Hz)
|
||||
- SNR: signal-to-noise ratio
|
||||
|
||||
Implementation in AirSpace:
|
||||
|
||||
1. Convert channel width from MHz to Hz.
|
||||
2. Recalculate SNR based on new environment or interface settings.
|
||||
3. Apply Shannon-Hartley theorem to determine new maximum channel capacity in Mbps.
|
||||
|
||||
8. Shared Maximum Capacity Across Devices
|
||||
-----------------------------------------
|
||||
|
||||
While individual devices have theoretical maximum data rates, the actual achievable rate is often less due to:
|
||||
|
||||
- Shared wireless medium among all devices on the same frequency and channel width
|
||||
- Interference and congestion from multiple devices transmitting simultaneously
|
||||
|
||||
9. AirSpace Inspection
|
||||
----------------------
|
||||
|
||||
The AirSpace class provides methods for visualizing network behavior:
|
||||
|
||||
- ``show_wireless_interfaces()``: Displays current state of all interfaces
|
||||
- ``show_bandwidth_load()``: Shows channel loads and bandwidth utilization
|
||||
@@ -37,7 +37,7 @@ additional steps to configure wireless settings:
|
||||
.. code-block:: python
|
||||
|
||||
from primaite.simulator.network.hardware.nodes.network.wireless_router import WirelessRouter
|
||||
from primaite.simulator.network.airspace import AirSpaceFrequency
|
||||
from primaite.simulator.network.airspace import AirSpaceFrequency, ChannelWidth
|
||||
|
||||
# Instantiate the WirelessRouter
|
||||
wireless_router = WirelessRouter(hostname="MyWirelessRouter")
|
||||
@@ -49,7 +49,8 @@ additional steps to configure wireless settings:
|
||||
wireless_router.configure_wireless_access_point(
|
||||
port=1, ip_address="192.168.2.1",
|
||||
subnet_mask="255.255.255.0",
|
||||
frequency=AirSpaceFrequency.WIFI_2_4
|
||||
frequency=AirSpaceFrequency.WIFI_2_4,
|
||||
channel_width=ChannelWidth.ChannelWidth.WIDTH_40_MHZ
|
||||
)
|
||||
|
||||
|
||||
@@ -71,7 +72,7 @@ ICMP traffic, ensuring basic network connectivity and ping functionality.
|
||||
|
||||
.. code-block:: python
|
||||
|
||||
from primaite.simulator.network.airspace import AIR_SPACE, AirSpaceFrequency
|
||||
from primaite.simulator.network.airspace import AirSpaceFrequency, ChannelWidth
|
||||
from primaite.simulator.network.container import Network
|
||||
from primaite.simulator.network.hardware.nodes.host.computer import Computer
|
||||
from primaite.simulator.network.hardware.nodes.network.router import ACLAction
|
||||
@@ -130,13 +131,15 @@ ICMP traffic, ensuring basic network connectivity and ping functionality.
|
||||
port=1,
|
||||
ip_address="192.168.1.1",
|
||||
subnet_mask="255.255.255.0",
|
||||
frequency=AirSpaceFrequency.WIFI_2_4
|
||||
frequency=AirSpaceFrequency.WIFI_2_4,
|
||||
channel_width=ChannelWidth.ChannelWidth.WIDTH_40_MHZ
|
||||
)
|
||||
router_2.configure_wireless_access_point(
|
||||
port=1,
|
||||
ip_address="192.168.1.2",
|
||||
subnet_mask="255.255.255.0",
|
||||
frequency=AirSpaceFrequency.WIFI_2_4
|
||||
frequency=AirSpaceFrequency.WIFI_2_4,
|
||||
channel_width=ChannelWidth.ChannelWidth.WIDTH_40_MHZ
|
||||
)
|
||||
|
||||
# Configure routes for inter-router communication
|
||||
|
||||
@@ -15,6 +15,7 @@ from primaite.game.agent.scripted_agents.probabilistic_agent import Probabilisti
|
||||
from primaite.game.agent.scripted_agents.random_agent import PeriodicAgent
|
||||
from primaite.game.agent.scripted_agents.tap001 import TAP001
|
||||
from primaite.game.science import graph_has_cycle, topological_sort
|
||||
from primaite.simulator.network.airspace import AirspaceEnvironmentType
|
||||
from primaite.simulator.network.hardware.base import NodeOperatingState
|
||||
from primaite.simulator.network.hardware.nodes.host.computer import Computer
|
||||
from primaite.simulator.network.hardware.nodes.host.host_node import NIC
|
||||
@@ -233,6 +234,11 @@ class PrimaiteGame:
|
||||
|
||||
simulation_config = cfg.get("simulation", {})
|
||||
network_config = simulation_config.get("network", {})
|
||||
airspace_cfg = network_config.get("airspace", {})
|
||||
airspace_environment_type_str = airspace_cfg.get("airspace_environment_type", "urban")
|
||||
|
||||
airspace_environment_type: AirspaceEnvironmentType = AirspaceEnvironmentType(airspace_environment_type_str)
|
||||
net.airspace.airspace_environment_type = airspace_environment_type
|
||||
|
||||
nodes_cfg = network_config.get("nodes", [])
|
||||
links_cfg = network_config.get("links", [])
|
||||
|
||||
@@ -3,9 +3,11 @@ from __future__ import annotations
|
||||
|
||||
from abc import ABC, abstractmethod
|
||||
from enum import Enum
|
||||
from typing import Any, Dict, List, Optional
|
||||
from typing import Any, Dict, List, Tuple
|
||||
|
||||
from prettytable import PrettyTable
|
||||
import numpy as np
|
||||
from prettytable import MARKDOWN, PrettyTable
|
||||
from pydantic import BaseModel, computed_field, Field, model_validator
|
||||
|
||||
from primaite import getLogger
|
||||
from primaite.simulator.network.hardware.base import Layer3Interface, NetworkInterface, WiredNetworkInterface
|
||||
@@ -15,90 +17,29 @@ from primaite.simulator.system.core.packet_capture import PacketCapture
|
||||
|
||||
_LOGGER = getLogger(__name__)
|
||||
|
||||
__all__ = ["AirSpaceFrequency", "WirelessNetworkInterface", "IPWirelessNetworkInterface"]
|
||||
|
||||
def format_hertz(hertz: float, format_terahertz: bool = False, decimals: int = 3) -> str:
|
||||
"""
|
||||
Convert a frequency in Hertz to a formatted string using the most appropriate unit.
|
||||
|
||||
class AirSpace:
|
||||
"""Represents a wireless airspace, managing wireless network interfaces and handling wireless transmission."""
|
||||
Optionally includes formatting for Terahertz.
|
||||
|
||||
def __init__(self):
|
||||
self._wireless_interfaces: Dict[str, WirelessNetworkInterface] = {}
|
||||
self._wireless_interfaces_by_frequency: Dict[AirSpaceFrequency, List[WirelessNetworkInterface]] = {}
|
||||
|
||||
def show(self, frequency: Optional[AirSpaceFrequency] = None):
|
||||
"""
|
||||
Displays a summary of wireless interfaces in the airspace, optionally filtered by a specific frequency.
|
||||
|
||||
:param frequency: The frequency band to filter devices by. If None, devices for all frequencies are shown.
|
||||
"""
|
||||
table = PrettyTable()
|
||||
table.field_names = ["Connected Node", "MAC Address", "IP Address", "Subnet Mask", "Frequency", "Status"]
|
||||
|
||||
# If a specific frequency is provided, filter by it; otherwise, use all frequencies.
|
||||
frequencies_to_show = [frequency] if frequency else self._wireless_interfaces_by_frequency.keys()
|
||||
|
||||
for freq in frequencies_to_show:
|
||||
interfaces = self._wireless_interfaces_by_frequency.get(freq, [])
|
||||
for interface in interfaces:
|
||||
status = "Enabled" if interface.enabled else "Disabled"
|
||||
table.add_row(
|
||||
[
|
||||
interface._connected_node.hostname, # noqa
|
||||
interface.mac_address,
|
||||
interface.ip_address if hasattr(interface, "ip_address") else None,
|
||||
interface.subnet_mask if hasattr(interface, "subnet_mask") else None,
|
||||
str(freq),
|
||||
status,
|
||||
]
|
||||
)
|
||||
|
||||
print(table)
|
||||
|
||||
def add_wireless_interface(self, wireless_interface: WirelessNetworkInterface):
|
||||
"""
|
||||
Adds a wireless network interface to the airspace if it's not already present.
|
||||
|
||||
:param wireless_interface: The wireless network interface to be added.
|
||||
"""
|
||||
if wireless_interface.mac_address not in self._wireless_interfaces:
|
||||
self._wireless_interfaces[wireless_interface.mac_address] = wireless_interface
|
||||
if wireless_interface.frequency not in self._wireless_interfaces_by_frequency:
|
||||
self._wireless_interfaces_by_frequency[wireless_interface.frequency] = []
|
||||
self._wireless_interfaces_by_frequency[wireless_interface.frequency].append(wireless_interface)
|
||||
|
||||
def remove_wireless_interface(self, wireless_interface: WirelessNetworkInterface):
|
||||
"""
|
||||
Removes a wireless network interface from the airspace if it's present.
|
||||
|
||||
:param wireless_interface: The wireless network interface to be removed.
|
||||
"""
|
||||
if wireless_interface.mac_address in self._wireless_interfaces:
|
||||
self._wireless_interfaces.pop(wireless_interface.mac_address)
|
||||
self._wireless_interfaces_by_frequency[wireless_interface.frequency].remove(wireless_interface)
|
||||
|
||||
def clear(self):
|
||||
"""
|
||||
Clears all wireless network interfaces and their frequency associations from the airspace.
|
||||
|
||||
After calling this method, the airspace will contain no wireless network interfaces, and transmissions cannot
|
||||
occur until new interfaces are added again.
|
||||
"""
|
||||
self._wireless_interfaces.clear()
|
||||
self._wireless_interfaces_by_frequency.clear()
|
||||
|
||||
def transmit(self, frame: Frame, sender_network_interface: WirelessNetworkInterface):
|
||||
"""
|
||||
Transmits a frame to all enabled wireless network interfaces on a specific frequency within the airspace.
|
||||
|
||||
This ensures that a wireless interface does not receive its own transmission.
|
||||
|
||||
:param frame: The frame to be transmitted.
|
||||
:param sender_network_interface: The wireless network interface sending the frame. This interface will be
|
||||
excluded from the list of receivers to prevent it from receiving its own transmission.
|
||||
"""
|
||||
for wireless_interface in self._wireless_interfaces_by_frequency.get(sender_network_interface.frequency, []):
|
||||
if wireless_interface != sender_network_interface and wireless_interface.enabled:
|
||||
wireless_interface.receive_frame(frame)
|
||||
:param hertz: Frequency in Hertz.
|
||||
:param format_terahertz: Whether to format frequency in Terahertz, default is False.
|
||||
:param decimals: Number of decimal places to round to, default is 3.
|
||||
:returns: Formatted string with the frequency in the most suitable unit.
|
||||
"""
|
||||
format_str = f"{{:.{decimals}f}}"
|
||||
if format_terahertz and hertz >= 1e12: # Terahertz
|
||||
return format_str.format(hertz / 1e12) + " THz"
|
||||
elif hertz >= 1e9: # Gigahertz
|
||||
return format_str.format(hertz / 1e9) + " GHz"
|
||||
elif hertz >= 1e6: # Megahertz
|
||||
return format_str.format(hertz / 1e6) + " MHz"
|
||||
elif hertz >= 1e3: # Kilohertz
|
||||
return format_str.format(hertz / 1e3) + " kHz"
|
||||
else: # Hertz
|
||||
return format_str.format(hertz) + " Hz"
|
||||
|
||||
|
||||
class AirSpaceFrequency(Enum):
|
||||
@@ -110,12 +51,478 @@ class AirSpaceFrequency(Enum):
|
||||
"""WiFi 5 GHz. Known for its higher data transmission speeds and reduced interference from other devices."""
|
||||
|
||||
def __str__(self) -> str:
|
||||
hertz_str = format_hertz(hertz=self.value)
|
||||
if self == AirSpaceFrequency.WIFI_2_4:
|
||||
return "WiFi 2.4 GHz"
|
||||
elif self == AirSpaceFrequency.WIFI_5:
|
||||
return "WiFi 5 GHz"
|
||||
else:
|
||||
return "Unknown Frequency"
|
||||
return f"WiFi {hertz_str}"
|
||||
if self == AirSpaceFrequency.WIFI_5:
|
||||
return f"WiFi {hertz_str}"
|
||||
return "Unknown Frequency"
|
||||
|
||||
|
||||
class ChannelWidth(Enum):
|
||||
"""
|
||||
Enumeration representing the available channel widths in MHz for wireless communications.
|
||||
|
||||
This enum facilitates standardising and validating channel width configurations.
|
||||
|
||||
Attributes:
|
||||
WIDTH_20_MHZ (int): Represents a channel width of 20 MHz, commonly used for basic
|
||||
Wi-Fi connectivity with standard range and interference resistance.
|
||||
WIDTH_40_MHZ (int): Represents a channel width of 40 MHz, offering higher data
|
||||
throughput at the expense of potentially increased interference.
|
||||
WIDTH_80_MHZ (int): Represents a channel width of 80 MHz, typically used in modern
|
||||
Wi-Fi setups for high data rate applications but with higher susceptibility to interference.
|
||||
WIDTH_160_MHZ (int): Represents a channel width of 160 MHz, used for ultra-high-speed
|
||||
network applications, providing maximum data throughput with significant
|
||||
requirements on the spectral environment to minimize interference.
|
||||
"""
|
||||
|
||||
WIDTH_20_MHZ = 20
|
||||
"""
|
||||
Represents a channel width of 20 MHz, commonly used for basic Wi-Fi connectivity with standard range and
|
||||
interference resistance
|
||||
"""
|
||||
|
||||
WIDTH_40_MHZ = 40
|
||||
"""
|
||||
Represents a channel width of 40 MHz, offering higher data throughput at the expense of potentially increased
|
||||
interference.
|
||||
"""
|
||||
|
||||
WIDTH_80_MHZ = 80
|
||||
"""
|
||||
Represents a channel width of 80 MHz, typically used in modern Wi-Fi setups for high data rate applications but
|
||||
with higher susceptibility to interference.
|
||||
"""
|
||||
|
||||
WIDTH_160_MHZ = 160
|
||||
"""
|
||||
Represents a channel width of 160 MHz, used for ultra-high-speed network applications, providing maximum data
|
||||
throughput with significant requirements on the spectral environment to minimize interference.
|
||||
"""
|
||||
|
||||
def __str__(self) -> str:
|
||||
"""
|
||||
Returns a string representation of the channel width.
|
||||
|
||||
:return: String in the format of "<value> MHz" indicating the channel width.
|
||||
"""
|
||||
return f"{self.value} MHz"
|
||||
|
||||
|
||||
AirSpaceKeyType = Tuple[AirSpaceFrequency, ChannelWidth]
|
||||
|
||||
|
||||
class AirspaceEnvironmentType(Enum):
|
||||
"""Enum representing different types of airspace environments which affect wireless communication signals."""
|
||||
|
||||
RURAL = "rural"
|
||||
"""
|
||||
A rural environment offers clear channel conditions due to low population density and minimal electronic device
|
||||
presence.
|
||||
"""
|
||||
|
||||
OUTDOOR = "outdoor"
|
||||
"""
|
||||
Outdoor environments like parks or fields have minimal electronic interference.
|
||||
"""
|
||||
|
||||
SUBURBAN = "suburban"
|
||||
"""
|
||||
Suburban environments strike a balance with fewer electronic interferences than urban but more than rural.
|
||||
"""
|
||||
|
||||
OFFICE = "office"
|
||||
"""
|
||||
Office environments have moderate interference from numerous electronic devices and overlapping networks.
|
||||
"""
|
||||
|
||||
URBAN = "urban"
|
||||
"""
|
||||
Urban environments are characterized by tall buildings and a high density of electronic devices, leading to
|
||||
significant interference.
|
||||
"""
|
||||
|
||||
INDUSTRIAL = "industrial"
|
||||
"""
|
||||
Industrial areas face high interference from heavy machinery and numerous electronic devices.
|
||||
"""
|
||||
|
||||
TRANSPORT = "transport"
|
||||
"""
|
||||
Environments such as subways and buses where metal structures and high mobility create complex interference
|
||||
patterns.
|
||||
"""
|
||||
|
||||
DENSE_URBAN = "dense_urban"
|
||||
"""
|
||||
Dense urban areas like city centers have the highest level of signal interference due to the very high density of
|
||||
buildings and devices.
|
||||
"""
|
||||
|
||||
JAMMING_ZONE = "jamming_zone"
|
||||
"""
|
||||
A jamming zone environment where signals are actively interfered with, typically through the use of signal jammers
|
||||
or scrambling devices. This represents the environment with the highest level of interference.
|
||||
"""
|
||||
|
||||
BLOCKED = "blocked"
|
||||
"""
|
||||
A jamming zone environment with total levels of interference. Airspace is completely blocked.
|
||||
"""
|
||||
|
||||
@property
|
||||
def snr_impact(self) -> int:
|
||||
"""
|
||||
Returns the SNR impact associated with the environment.
|
||||
|
||||
:return: SNR impact in dB.
|
||||
"""
|
||||
impacts = {
|
||||
AirspaceEnvironmentType.RURAL: 0,
|
||||
AirspaceEnvironmentType.OUTDOOR: 1,
|
||||
AirspaceEnvironmentType.SUBURBAN: -5,
|
||||
AirspaceEnvironmentType.OFFICE: -7,
|
||||
AirspaceEnvironmentType.URBAN: -10,
|
||||
AirspaceEnvironmentType.INDUSTRIAL: -15,
|
||||
AirspaceEnvironmentType.TRANSPORT: -12,
|
||||
AirspaceEnvironmentType.DENSE_URBAN: -20,
|
||||
AirspaceEnvironmentType.JAMMING_ZONE: -40,
|
||||
AirspaceEnvironmentType.BLOCKED: -100,
|
||||
}
|
||||
return impacts[self]
|
||||
|
||||
def __str__(self) -> str:
|
||||
return f"{self.value.title()} Environment (SNR Impact: {self.snr_impact})"
|
||||
|
||||
|
||||
def estimate_snr(
|
||||
frequency: AirSpaceFrequency, environment_type: AirspaceEnvironmentType, channel_width: ChannelWidth
|
||||
) -> float:
|
||||
"""
|
||||
Estimate the Signal-to-Noise Ratio (SNR) based on the communication frequency, environment, and channel width.
|
||||
|
||||
This function considers both the base SNR value dependent on the frequency and the impact of environmental
|
||||
factors and channel width on the SNR.
|
||||
|
||||
The SNR is adjusted by reducing it for wider channels, reflecting the increased noise floor from a broader
|
||||
frequency range.
|
||||
|
||||
:param frequency: The operating frequency as defined by AirSpaceFrequency enum, influencing the base SNR. Higher
|
||||
frequencies like 5 GHz generally start with a higher base SNR due to less noise.
|
||||
:param environment_type: The type of environment from AirspaceEnvironmentType enum, which adjusts the SNR based on
|
||||
expected environmental noise and interference levels.
|
||||
:param channel_width: The channel width from ChannelWidth enum, where wider channels (80 MHz and 160 MHz) decrease
|
||||
the SNR slightly due to an increased noise floor.
|
||||
:return: Estimated SNR in dB, calculated as the base SNR modified by environmental and channel width impacts.
|
||||
"""
|
||||
base_snr = 40 if frequency == AirSpaceFrequency.WIFI_5 else 30
|
||||
snr_impact = environment_type.snr_impact
|
||||
|
||||
# Adjust SNR impact based on channel width
|
||||
if channel_width == ChannelWidth.WIDTH_80_MHZ or channel_width == ChannelWidth.WIDTH_160_MHZ:
|
||||
snr_impact -= 3 # Assume wider channels have slightly lower SNR due to increased noise floor
|
||||
|
||||
return base_snr + snr_impact
|
||||
|
||||
|
||||
def calculate_total_channel_capacity(
|
||||
channel_width: ChannelWidth, frequency: AirSpaceFrequency, environment_type: AirspaceEnvironmentType
|
||||
) -> float:
|
||||
"""
|
||||
Calculate the total theoretical data rate for the channel using the Shannon-Hartley theorem.
|
||||
|
||||
This function determines the channel's capacity by considering the bandwidth (derived from channel width),
|
||||
and the signal-to-noise ratio (SNR) adjusted by frequency and environmental conditions.
|
||||
|
||||
The Shannon-Hartley theorem states that channel capacity C (in bits per second) can be calculated as:
|
||||
``C = B * log2(1 + SNR)`` where B is the bandwidth in Hertz and SNR is the signal-to-noise ratio.
|
||||
|
||||
:param channel_width: The width of the channel as defined by ChannelWidth enum, converted to Hz for calculation.
|
||||
:param frequency: The operating frequency as defined by AirSpaceFrequency enum, influencing the base SNR and part
|
||||
of the SNR estimation.
|
||||
:param environment_type: The type of environment as defined by AirspaceEnvironmentType enum, used in SNR estimation.
|
||||
:return: Theoretical total data rate in Mbps for the entire channel.
|
||||
"""
|
||||
bandwidth_hz = channel_width.value * 1_000_000 # Convert MHz to Hz
|
||||
snr_db = estimate_snr(frequency, environment_type, channel_width)
|
||||
snr_linear = 10 ** (snr_db / 10)
|
||||
|
||||
total_capacity_bps = bandwidth_hz * np.log2(1 + snr_linear)
|
||||
total_capacity_mbps = total_capacity_bps / 1_000_000
|
||||
|
||||
return total_capacity_mbps
|
||||
|
||||
|
||||
def calculate_individual_device_rate(
|
||||
channel_width: ChannelWidth,
|
||||
frequency: AirSpaceFrequency,
|
||||
environment_type: AirspaceEnvironmentType,
|
||||
device_count: int,
|
||||
) -> float:
|
||||
"""
|
||||
Calculate the theoretical data rate available to each individual device on the channel.
|
||||
|
||||
This function first calculates the total channel capacity and then divides this capacity by the number
|
||||
of active devices to estimate each device's share of the bandwidth. This reflects the practical limitation
|
||||
that multiple devices must share the same channel resources.
|
||||
|
||||
:param channel_width: The channel width as defined by ChannelWidth enum, used in total capacity calculation.
|
||||
:param frequency: The operating frequency as defined by AirSpaceFrequency enum, used in total capacity calculation.
|
||||
:param environment_type: The environment type as defined by AirspaceEnvironmentType enum, impacting SNR and
|
||||
capacity.
|
||||
:param device_count: The number of devices sharing the channel. If zero, returns zero to avoid division by zero.
|
||||
:return: Theoretical data rate in Mbps available per device, based on shared channel capacity.
|
||||
"""
|
||||
total_capacity_mbps = calculate_total_channel_capacity(channel_width, frequency, environment_type)
|
||||
if device_count == 0:
|
||||
return 0 # Avoid division by zero
|
||||
individual_device_rate_mbps = total_capacity_mbps / device_count
|
||||
|
||||
return individual_device_rate_mbps
|
||||
|
||||
|
||||
class AirSpace(BaseModel):
|
||||
"""
|
||||
Represents a wireless airspace, managing wireless network interfaces and handling wireless transmission.
|
||||
|
||||
This class provides functionalities to manage a collection of wireless network interfaces, each associated with
|
||||
specific frequencies and channel widths. It includes methods to calculate and manage bandwidth loads, add and
|
||||
remove wireless interfaces, and handle data transmission across these interfaces.
|
||||
"""
|
||||
|
||||
airspace_environment_type_: AirspaceEnvironmentType = AirspaceEnvironmentType.URBAN
|
||||
wireless_interfaces: Dict[str, WirelessNetworkInterface] = Field(default_factory=lambda: {})
|
||||
wireless_interfaces_by_frequency_channel_width: Dict[AirSpaceKeyType, List[WirelessNetworkInterface]] = Field(
|
||||
default_factory=lambda: {}
|
||||
)
|
||||
bandwidth_load: Dict[AirSpaceKeyType, float] = Field(default_factory=lambda: {})
|
||||
frequency_channel_width_max_capacity_mbps: Dict[AirSpaceKeyType, float] = Field(default_factory=lambda: {})
|
||||
|
||||
def model_post_init(self, __context: Any) -> None:
|
||||
"""
|
||||
Initialize the airspace metadata after instantiation.
|
||||
|
||||
This method is called to set up initial configurations like the maximum capacity of each channel width and
|
||||
frequency based on the current environment setting.
|
||||
|
||||
:param __context: Contextual data or settings, typically used for further initializations beyond
|
||||
the basic constructor.
|
||||
"""
|
||||
self._set_frequency_channel_width_max_capacity_mbps()
|
||||
|
||||
def _set_frequency_channel_width_max_capacity_mbps(self):
|
||||
"""
|
||||
Private method to compute and set the maximum channel capacity in Mbps for each frequency and channel width.
|
||||
|
||||
Based on the airspace environment type, this method calculates the maximum possible data transmission
|
||||
capacity for each combination of frequency and channel width available and stores these values.
|
||||
These capacities are critical for managing and limiting bandwidth load during operations.
|
||||
"""
|
||||
print(
|
||||
f"Rebuilding the frequency channel width maximum capacity dictionary based on "
|
||||
f"airspace environment type {self.airspace_environment_type_}"
|
||||
)
|
||||
for frequency in AirSpaceFrequency:
|
||||
for channel_width in ChannelWidth:
|
||||
max_capacity = calculate_total_channel_capacity(
|
||||
frequency=frequency, channel_width=channel_width, environment_type=self.airspace_environment_type
|
||||
)
|
||||
self.frequency_channel_width_max_capacity_mbps[frequency, channel_width] = max_capacity
|
||||
|
||||
@computed_field
|
||||
@property
|
||||
def airspace_environment_type(self) -> AirspaceEnvironmentType:
|
||||
"""
|
||||
Gets the current environment type of the airspace.
|
||||
|
||||
:return: The AirspaceEnvironmentType representing the current environment type.
|
||||
"""
|
||||
return self.airspace_environment_type_
|
||||
|
||||
@airspace_environment_type.setter
|
||||
def airspace_environment_type(self, value: AirspaceEnvironmentType) -> None:
|
||||
"""
|
||||
Sets a new environment type for the airspace and updates related configurations.
|
||||
|
||||
Changing the environment type triggers a re-calculation of the maximum channel capacities and
|
||||
adjustments to the current setup of wireless interfaces to ensure they are aligned with the
|
||||
new environment settings.
|
||||
|
||||
:param value: The new environment type as an AirspaceEnvironmentType.
|
||||
"""
|
||||
if value != self.airspace_environment_type_:
|
||||
print(f"Setting airspace_environment_type to {value}")
|
||||
self.airspace_environment_type_ = value
|
||||
self._set_frequency_channel_width_max_capacity_mbps()
|
||||
wireless_interface_keys = list(self.wireless_interfaces.keys())
|
||||
for wireless_interface_key in wireless_interface_keys:
|
||||
wireless_interface = self.wireless_interfaces[wireless_interface_key]
|
||||
self.remove_wireless_interface(wireless_interface)
|
||||
self.add_wireless_interface(wireless_interface)
|
||||
|
||||
def show_bandwidth_load(self, markdown: bool = False):
|
||||
"""
|
||||
Prints a table of the current bandwidth load for each frequency and channel width combination on the airspace.
|
||||
|
||||
This method prints a tabulated view showing the utilisation of available bandwidth capacities for all configured
|
||||
frequency and channel width pairings. The table includes the current capacity usage as a percentage of the
|
||||
maximum capacity, alongside the absolute maximum capacity values in Mbps.
|
||||
|
||||
:param markdown: Flag indicating if output should be in markdown format.
|
||||
"""
|
||||
headers = ["Frequency", "Channel Width", "Current Capacity (%)", "Maximum Capacity (Mbit)"]
|
||||
table = PrettyTable(headers)
|
||||
if markdown:
|
||||
table.set_style(MARKDOWN)
|
||||
table.align = "l"
|
||||
table.title = "Airspace Frequency Channel Loads"
|
||||
for key, load in self.bandwidth_load.items():
|
||||
frequency, channel_width = key
|
||||
maximum_capacity = self.frequency_channel_width_max_capacity_mbps[key]
|
||||
load_percent = load / maximum_capacity
|
||||
if load_percent > 1.0:
|
||||
load_percent = 1.0
|
||||
table.add_row(
|
||||
[format_hertz(frequency.value), str(channel_width), f"{load_percent:.0%}", f"{maximum_capacity:.3f}"]
|
||||
)
|
||||
print(table)
|
||||
|
||||
def show_wireless_interfaces(self, markdown: bool = False):
|
||||
"""
|
||||
Prints a table of wireless interfaces in the airspace.
|
||||
|
||||
:param markdown: Flag indicating if output should be in markdown format.
|
||||
"""
|
||||
headers = [
|
||||
"Connected Node",
|
||||
"MAC Address",
|
||||
"IP Address",
|
||||
"Subnet Mask",
|
||||
"Frequency",
|
||||
"Channel Width",
|
||||
"Speed (Mbps)",
|
||||
"Status",
|
||||
]
|
||||
table = PrettyTable(headers)
|
||||
if markdown:
|
||||
table.set_style(MARKDOWN)
|
||||
table.align = "l"
|
||||
table.title = f"Devices on Air Space - {self.airspace_environment_type}"
|
||||
|
||||
for interface in self.wireless_interfaces.values():
|
||||
status = "Enabled" if interface.enabled else "Disabled"
|
||||
table.add_row(
|
||||
[
|
||||
interface._connected_node.hostname, # noqa
|
||||
interface.mac_address,
|
||||
interface.ip_address if hasattr(interface, "ip_address") else None,
|
||||
interface.subnet_mask if hasattr(interface, "subnet_mask") else None,
|
||||
format_hertz(interface.frequency.value),
|
||||
str(interface.channel_width),
|
||||
f"{interface.speed:.3f}",
|
||||
status,
|
||||
]
|
||||
)
|
||||
print(table.get_string(sortby="Frequency"))
|
||||
|
||||
def show(self, markdown: bool = False):
|
||||
"""
|
||||
Prints a summary of the current state of the airspace, including both wireless interfaces and bandwidth loads.
|
||||
|
||||
This method is a convenient wrapper that calls two separate methods to display detailed tables: one for
|
||||
wireless interfaces and another for bandwidth load across all frequencies and channel widths managed within the
|
||||
airspace. It provides a holistic view of the operational status and performance metrics of the airspace.
|
||||
|
||||
:param markdown: Flag indicating if output should be in markdown format.
|
||||
"""
|
||||
self.show_wireless_interfaces(markdown)
|
||||
self.show_bandwidth_load(markdown)
|
||||
|
||||
def add_wireless_interface(self, wireless_interface: WirelessNetworkInterface):
|
||||
"""
|
||||
Adds a wireless network interface to the airspace if it's not already present.
|
||||
|
||||
:param wireless_interface: The wireless network interface to be added.
|
||||
"""
|
||||
if wireless_interface.mac_address not in self.wireless_interfaces:
|
||||
self.wireless_interfaces[wireless_interface.mac_address] = wireless_interface
|
||||
if wireless_interface.airspace_key not in self.wireless_interfaces_by_frequency_channel_width:
|
||||
self.wireless_interfaces_by_frequency_channel_width[wireless_interface.airspace_key] = []
|
||||
self.wireless_interfaces_by_frequency_channel_width[wireless_interface.airspace_key].append(
|
||||
wireless_interface
|
||||
)
|
||||
speed = calculate_total_channel_capacity(
|
||||
wireless_interface.channel_width, wireless_interface.frequency, self.airspace_environment_type
|
||||
)
|
||||
wireless_interface.set_speed(speed)
|
||||
|
||||
def remove_wireless_interface(self, wireless_interface: WirelessNetworkInterface):
|
||||
"""
|
||||
Removes a wireless network interface from the airspace if it's present.
|
||||
|
||||
:param wireless_interface: The wireless network interface to be removed.
|
||||
"""
|
||||
if wireless_interface.mac_address in self.wireless_interfaces:
|
||||
self.wireless_interfaces.pop(wireless_interface.mac_address)
|
||||
self.wireless_interfaces_by_frequency_channel_width[wireless_interface.airspace_key].remove(
|
||||
wireless_interface
|
||||
)
|
||||
|
||||
def clear(self):
|
||||
"""
|
||||
Clears all wireless network interfaces and their frequency associations from the airspace.
|
||||
|
||||
After calling this method, the airspace will contain no wireless network interfaces, and transmissions cannot
|
||||
occur until new interfaces are added again.
|
||||
"""
|
||||
self.wireless_interfaces.clear()
|
||||
self.wireless_interfaces_by_frequency_channel_width.clear()
|
||||
|
||||
def reset_bandwidth_load(self):
|
||||
"""
|
||||
Resets the bandwidth load tracking for all frequencies in the airspace.
|
||||
|
||||
This method clears the current load metrics for all operating frequencies, effectively setting the load to zero.
|
||||
"""
|
||||
self.bandwidth_load = {}
|
||||
|
||||
def can_transmit_frame(self, frame: Frame, sender_network_interface: WirelessNetworkInterface) -> bool:
|
||||
"""
|
||||
Determines if a frame can be transmitted by the sender network interface based on the current bandwidth load.
|
||||
|
||||
This method checks if adding the size of the frame to the current bandwidth load of the frequency used by the
|
||||
sender network interface would exceed the maximum allowed bandwidth for that frequency. It returns True if the
|
||||
frame can be transmitted without exceeding the limit, and False otherwise.
|
||||
|
||||
:param frame: The frame to be transmitted, used to check its size against the frequency's bandwidth limit.
|
||||
:param sender_network_interface: The network interface attempting to transmit the frame, used to determine the
|
||||
relevant frequency and its current bandwidth load.
|
||||
:return: True if the frame can be transmitted within the bandwidth limit, False if it would exceed the limit.
|
||||
"""
|
||||
if sender_network_interface.airspace_key not in self.bandwidth_load:
|
||||
self.bandwidth_load[sender_network_interface.airspace_key] = 0.0
|
||||
return (
|
||||
self.bandwidth_load[sender_network_interface.airspace_key] + frame.size_Mbits
|
||||
<= self.frequency_channel_width_max_capacity_mbps[sender_network_interface.airspace_key]
|
||||
)
|
||||
|
||||
def transmit(self, frame: Frame, sender_network_interface: WirelessNetworkInterface):
|
||||
"""
|
||||
Transmits a frame to all enabled wireless network interfaces on a specific frequency within the airspace.
|
||||
|
||||
This ensures that a wireless interface does not receive its own transmission.
|
||||
|
||||
:param frame: The frame to be transmitted.
|
||||
:param sender_network_interface: The wireless network interface sending the frame. This interface will be
|
||||
excluded from the list of receivers to prevent it from receiving its own transmission.
|
||||
"""
|
||||
self.bandwidth_load[sender_network_interface.airspace_key] += frame.size_Mbits
|
||||
for wireless_interface in self.wireless_interfaces_by_frequency_channel_width.get(
|
||||
sender_network_interface.airspace_key, []
|
||||
):
|
||||
if wireless_interface != sender_network_interface and wireless_interface.enabled:
|
||||
wireless_interface.receive_frame(frame)
|
||||
|
||||
|
||||
class WirelessNetworkInterface(NetworkInterface, ABC):
|
||||
@@ -139,7 +546,135 @@ class WirelessNetworkInterface(NetworkInterface, ABC):
|
||||
"""
|
||||
|
||||
airspace: AirSpace
|
||||
frequency: AirSpaceFrequency = AirSpaceFrequency.WIFI_2_4
|
||||
frequency_: AirSpaceFrequency = AirSpaceFrequency.WIFI_2_4
|
||||
channel_width_: ChannelWidth = ChannelWidth.WIDTH_40_MHZ
|
||||
|
||||
@model_validator(mode="after") # noqa
|
||||
def validate_channel_width_for_2_4_ghz(self) -> "WirelessNetworkInterface":
|
||||
"""
|
||||
Validate the wireless interface's channel width settings after model changes.
|
||||
|
||||
This method serves as a model validator to ensure that the channel width settings for the 2.4 GHz frequency
|
||||
comply with accepted standards (either 20 MHz or 40 MHz). It's triggered after model instantiation.
|
||||
|
||||
Ensures that the channel width is appropriate for the current frequency setting, particularly checking
|
||||
and adjusting the settings for the 2.4 GHz frequency band to not exceed 40 MHz. This is crucial for
|
||||
avoiding interference and ensuring optimal performance in densely populated wireless environments.
|
||||
"""
|
||||
self._check_wifi_24_channel_width()
|
||||
return self
|
||||
|
||||
def model_post_init(self, __context: Any) -> None:
|
||||
"""Initialise the model after its creation, setting the speed based on the calculated channel capacity."""
|
||||
speed = calculate_total_channel_capacity(
|
||||
channel_width=self.channel_width,
|
||||
frequency=self.frequency,
|
||||
environment_type=self.airspace.airspace_environment_type,
|
||||
)
|
||||
self.set_speed(speed)
|
||||
|
||||
def _check_wifi_24_channel_width(self) -> None:
|
||||
"""
|
||||
Ensures that the channel width for 2.4 GHz frequency does not exceed 40 MHz.
|
||||
|
||||
This method checks the current frequency and channel width settings and adjusts the channel width
|
||||
to 40 MHz if the frequency is set to 2.4 GHz and the channel width exceeds 40 MHz. This is done to
|
||||
comply with typical Wi-Fi standards for 2.4 GHz frequencies, which commonly support up to 40 MHz.
|
||||
|
||||
Logs a SysLog warning if the channel width had to be adjusted, logging this change either to the connected
|
||||
node's system log or the global logger, depending on whether the interface is connected to a node.
|
||||
"""
|
||||
if self.frequency_ == AirSpaceFrequency.WIFI_2_4 and self.channel_width_.value > 40:
|
||||
self.channel_width_ = ChannelWidth.WIDTH_40_MHZ
|
||||
msg = (
|
||||
f"Channel width must be either 20 Mhz or 40 Mhz when using {AirSpaceFrequency.WIFI_2_4}. "
|
||||
f"Overriding value to use {ChannelWidth.WIDTH_40_MHZ}."
|
||||
)
|
||||
if self._connected_node:
|
||||
self._connected_node.sys_log.warning(f"Wireless Interface {self.port_num}: {msg}")
|
||||
else:
|
||||
_LOGGER.warning(msg)
|
||||
|
||||
@computed_field
|
||||
@property
|
||||
def frequency(self) -> AirSpaceFrequency:
|
||||
"""
|
||||
Get the current operating frequency of the wireless interface.
|
||||
|
||||
:return: The current frequency as an AirSpaceFrequency enum value.
|
||||
"""
|
||||
return self.frequency_
|
||||
|
||||
@frequency.setter
|
||||
def frequency(self, value: AirSpaceFrequency) -> None:
|
||||
"""
|
||||
Set the operating frequency of the wireless interface and update the network configuration.
|
||||
|
||||
This setter updates the frequency of the wireless interface if the new value differs from the current setting.
|
||||
It handles the update by first removing the interface from the current airspace management to avoid conflicts,
|
||||
setting the new frequency, ensuring the channel width remains compliant, and then re-adding the interface
|
||||
to the airspace with the new settings.
|
||||
|
||||
:param value: The new frequency to set, as an AirSpaceFrequency enum value.
|
||||
"""
|
||||
if value != self.frequency_:
|
||||
self.airspace.remove_wireless_interface(self)
|
||||
self.frequency_ = value
|
||||
self._check_wifi_24_channel_width()
|
||||
self.airspace.add_wireless_interface(self)
|
||||
|
||||
@computed_field
|
||||
@property
|
||||
def channel_width(self) -> ChannelWidth:
|
||||
"""
|
||||
Get the current channel width setting of the wireless interface.
|
||||
|
||||
:return: The current channel width as a ChannelWidth enum value.
|
||||
"""
|
||||
return self.channel_width_
|
||||
|
||||
@channel_width.setter
|
||||
def channel_width(self, value: ChannelWidth) -> None:
|
||||
"""
|
||||
Set the channel width of the wireless interface and manage configuration compliance.
|
||||
|
||||
Updates the channel width of the wireless interface. If the new channel width is different from the existing
|
||||
one, it first removes the interface from the airspace to prevent configuration conflicts, sets the new channel
|
||||
width, checks and adjusts it if necessary (especially for 2.4 GHz frequency to comply with typical standards),
|
||||
and then re-registers the interface in the airspace with updated settings.
|
||||
|
||||
:param value: The new channel width to set, as a ChannelWidth enum value.
|
||||
"""
|
||||
if value != self.channel_width_:
|
||||
self.airspace.remove_wireless_interface(self)
|
||||
self.channel_width_ = value
|
||||
self._check_wifi_24_channel_width()
|
||||
self.airspace.add_wireless_interface(self)
|
||||
|
||||
@property
|
||||
def airspace_key(self) -> tuple:
|
||||
"""
|
||||
The airspace bandwidth/channel identifier for the wireless interface based on its frequency and channel width.
|
||||
|
||||
:return: A tuple containing the frequency and channel width, serving as a bandwidth/channel key.
|
||||
"""
|
||||
return self.frequency_, self.channel_width_
|
||||
|
||||
def set_speed(self, speed: float):
|
||||
"""
|
||||
Sets the network interface speed to the specified value and logs this action.
|
||||
|
||||
This method updates the speed attribute of the network interface to the given value, reflecting
|
||||
the theoretical maximum data rate that the interface can support based on the current settings.
|
||||
It logs the new speed to the system log of the connected node if available.
|
||||
|
||||
:param speed: The speed in Mbps to be set for the network interface.
|
||||
"""
|
||||
self.speed = speed
|
||||
if self._connected_node:
|
||||
self._connected_node.sys_log.info(
|
||||
f"Wireless Interface {self.port_num}: Setting theoretical maximum data rate to {speed:.3f} Mbps."
|
||||
)
|
||||
|
||||
def enable(self):
|
||||
"""Attempt to enable the network interface."""
|
||||
@@ -188,8 +723,12 @@ class WirelessNetworkInterface(NetworkInterface, ABC):
|
||||
if self.enabled:
|
||||
frame.set_sent_timestamp()
|
||||
self.pcap.capture_outbound(frame)
|
||||
self.airspace.transmit(frame, self)
|
||||
return True
|
||||
if self.airspace.can_transmit_frame(frame, self):
|
||||
self.airspace.transmit(frame, self)
|
||||
return True
|
||||
else:
|
||||
# Cannot send Frame as the frequency bandwidth is at capacity
|
||||
return False
|
||||
# Cannot send Frame as the network interface is not enabled
|
||||
return False
|
||||
|
||||
|
||||
@@ -96,6 +96,8 @@ class Network(SimComponent):
|
||||
"""Apply pre-timestep logic."""
|
||||
super().pre_timestep(timestep)
|
||||
|
||||
self.airspace.reset_bandwidth_load()
|
||||
|
||||
for node in self.nodes.values():
|
||||
node.pre_timestep(timestep)
|
||||
|
||||
|
||||
@@ -87,7 +87,7 @@ class NetworkInterface(SimComponent, ABC):
|
||||
mac_address: str = Field(default_factory=generate_mac_address)
|
||||
"The MAC address of the interface."
|
||||
|
||||
speed: int = 100
|
||||
speed: float = 100.0
|
||||
"The speed of the interface in Mbps. Default is 100 Mbps."
|
||||
|
||||
mtu: int = 1500
|
||||
@@ -679,11 +679,20 @@ class Link(SimComponent):
|
||||
return self.endpoint_a.enabled and self.endpoint_b.enabled
|
||||
|
||||
def _can_transmit(self, frame: Frame) -> bool:
|
||||
"""
|
||||
Determines whether a frame can be transmitted considering the current Link load and the Link's bandwidth.
|
||||
|
||||
This method assesses if the transmission of a given frame is possible without exceeding the Link's total
|
||||
bandwidth capacity. It checks if the current load of the Link plus the size of the frame (expressed in Mbps)
|
||||
would remain within the defined bandwidth limits. The transmission is only feasible if the Link is active
|
||||
('up') and the total load including the new frame does not surpass the bandwidth limit.
|
||||
|
||||
:param frame: The frame intended for transmission, which contains its size in Mbps.
|
||||
:return: True if the frame can be transmitted without exceeding the bandwidth limit, False otherwise.
|
||||
"""
|
||||
if self.is_up:
|
||||
frame_size_Mbits = frame.size_Mbits # noqa - Leaving it as Mbits as this is how they're expressed
|
||||
# return self.current_load + frame_size_Mbits <= self.bandwidth
|
||||
# TODO: re add this check once packet size limiting and MTU checks are implemented
|
||||
return True
|
||||
return self.current_load + frame.size_Mbits <= self.bandwidth
|
||||
return False
|
||||
|
||||
def transmit_frame(self, sender_nic: WiredNetworkInterface, frame: Frame) -> bool:
|
||||
|
||||
@@ -1,10 +1,10 @@
|
||||
# © Crown-owned copyright 2024, Defence Science and Technology Laboratory UK
|
||||
from ipaddress import IPv4Address
|
||||
from typing import Any, Dict, Union
|
||||
from typing import Any, Dict, Optional, Union
|
||||
|
||||
from pydantic import validate_call
|
||||
|
||||
from primaite.simulator.network.airspace import AirSpace, AirSpaceFrequency, IPWirelessNetworkInterface
|
||||
from primaite.simulator.network.airspace import AirSpace, AirSpaceFrequency, ChannelWidth, IPWirelessNetworkInterface
|
||||
from primaite.simulator.network.hardware.node_operating_state import NodeOperatingState
|
||||
from primaite.simulator.network.hardware.nodes.network.router import ACLAction, Router, RouterInterface
|
||||
from primaite.simulator.network.transmission.data_link_layer import Frame
|
||||
@@ -153,7 +153,8 @@ class WirelessRouter(Router):
|
||||
self,
|
||||
ip_address: IPV4Address,
|
||||
subnet_mask: IPV4Address,
|
||||
frequency: AirSpaceFrequency = AirSpaceFrequency.WIFI_2_4,
|
||||
frequency: Optional[AirSpaceFrequency] = AirSpaceFrequency.WIFI_2_4,
|
||||
channel_width: Optional[ChannelWidth] = ChannelWidth.WIDTH_40_MHZ,
|
||||
):
|
||||
"""
|
||||
Configures a wireless access point (WAP).
|
||||
@@ -170,13 +171,23 @@ class WirelessRouter(Router):
|
||||
enum. This determines the frequency band (e.g., 2.4 GHz or 5 GHz) the access point will use for wireless
|
||||
communication. Default is AirSpaceFrequency.WIFI_2_4.
|
||||
"""
|
||||
if not frequency:
|
||||
frequency = AirSpaceFrequency.WIFI_2_4
|
||||
if not channel_width:
|
||||
channel_width = ChannelWidth.WIDTH_40_MHZ
|
||||
self.sys_log.info("Configuring wireless access point")
|
||||
|
||||
self.wireless_access_point.disable() # Temporarily disable the WAP for reconfiguration
|
||||
|
||||
network_interface = self.network_interface[1]
|
||||
|
||||
network_interface.ip_address = ip_address
|
||||
network_interface.subnet_mask = subnet_mask
|
||||
self.sys_log.info(f"Configured WAP {network_interface}")
|
||||
|
||||
self.wireless_access_point.frequency = frequency # Set operating frequency
|
||||
self.wireless_access_point.channel_width = channel_width
|
||||
self.wireless_access_point.enable() # Re-enable the WAP with new settings
|
||||
self.sys_log.info(f"Configured WAP {network_interface}")
|
||||
|
||||
@property
|
||||
def router_interface(self) -> RouterInterface:
|
||||
@@ -258,7 +269,12 @@ class WirelessRouter(Router):
|
||||
ip_address = cfg["wireless_access_point"]["ip_address"]
|
||||
subnet_mask = cfg["wireless_access_point"]["subnet_mask"]
|
||||
frequency = AirSpaceFrequency[cfg["wireless_access_point"]["frequency"]]
|
||||
router.configure_wireless_access_point(ip_address=ip_address, subnet_mask=subnet_mask, frequency=frequency)
|
||||
channel_width = cfg["wireless_access_point"].get("channel_width")
|
||||
if channel_width:
|
||||
channel_width = ChannelWidth(channel_width)
|
||||
router.configure_wireless_access_point(
|
||||
ip_address=ip_address, subnet_mask=subnet_mask, frequency=frequency, channel_width=channel_width
|
||||
)
|
||||
|
||||
if "acl" in cfg:
|
||||
for r_num, r_cfg in cfg["acl"].items():
|
||||
|
||||
@@ -133,10 +133,11 @@ class Frame(BaseModel):
|
||||
def size(self) -> float: # noqa - Keep it as MBits as this is how they're expressed
|
||||
"""The size of the Frame in Bytes."""
|
||||
# get the payload size if it is a data packet
|
||||
payload_size = 0.0
|
||||
if isinstance(self.payload, DataPacket):
|
||||
return self.payload.get_packet_size()
|
||||
payload_size = self.payload.get_packet_size()
|
||||
|
||||
return float(len(self.model_dump_json().encode("utf-8")))
|
||||
return float(len(self.model_dump_json().encode("utf-8"))) + payload_size
|
||||
|
||||
@property
|
||||
def size_Mbits(self) -> float: # noqa - Keep it as MBits as this is how they're expressed
|
||||
|
||||
@@ -9,6 +9,8 @@ game:
|
||||
|
||||
simulation:
|
||||
network:
|
||||
airspace:
|
||||
airspace_environment_type: blocked
|
||||
nodes:
|
||||
- type: computer
|
||||
hostname: pc_a
|
||||
|
||||
@@ -0,0 +1,81 @@
|
||||
game:
|
||||
max_episode_length: 256
|
||||
ports:
|
||||
- ARP
|
||||
protocols:
|
||||
- ICMP
|
||||
- TCP
|
||||
- UDP
|
||||
|
||||
simulation:
|
||||
network:
|
||||
airspace:
|
||||
airspace_environment_type: blocked
|
||||
nodes:
|
||||
- type: computer
|
||||
hostname: pc_a
|
||||
ip_address: 192.168.0.2
|
||||
subnet_mask: 255.255.255.0
|
||||
default_gateway: 192.168.0.1
|
||||
start_up_duration: 0
|
||||
|
||||
- type: computer
|
||||
hostname: pc_b
|
||||
ip_address: 192.168.2.2
|
||||
subnet_mask: 255.255.255.0
|
||||
default_gateway: 192.168.2.1
|
||||
start_up_duration: 0
|
||||
|
||||
- type: wireless_router
|
||||
hostname: router_1
|
||||
start_up_duration: 0
|
||||
|
||||
router_interface:
|
||||
ip_address: 192.168.0.1
|
||||
subnet_mask: 255.255.255.0
|
||||
|
||||
wireless_access_point:
|
||||
ip_address: 192.168.1.1
|
||||
subnet_mask: 255.255.255.0
|
||||
frequency: WIFI_5
|
||||
channel_width: 80
|
||||
acl:
|
||||
1:
|
||||
action: PERMIT
|
||||
routes:
|
||||
- address: 192.168.2.0 # PC B subnet
|
||||
subnet_mask: 255.255.255.0
|
||||
next_hop_ip_address: 192.168.1.2
|
||||
metric: 0
|
||||
|
||||
- type: wireless_router
|
||||
hostname: router_2
|
||||
start_up_duration: 0
|
||||
|
||||
router_interface:
|
||||
ip_address: 192.168.2.1
|
||||
subnet_mask: 255.255.255.0
|
||||
|
||||
wireless_access_point:
|
||||
ip_address: 192.168.1.2
|
||||
subnet_mask: 255.255.255.0
|
||||
frequency: WIFI_5
|
||||
channel_width: 80
|
||||
acl:
|
||||
1:
|
||||
action: PERMIT
|
||||
routes:
|
||||
- address: 192.168.0.0 # PC A subnet
|
||||
subnet_mask: 255.255.255.0
|
||||
next_hop_ip_address: 192.168.1.1
|
||||
metric: 0
|
||||
links:
|
||||
- endpoint_a_hostname: pc_a
|
||||
endpoint_a_port: 1
|
||||
endpoint_b_hostname: router_1
|
||||
endpoint_b_port: 2
|
||||
|
||||
- endpoint_a_hostname: pc_b
|
||||
endpoint_a_port: 1
|
||||
endpoint_b_hostname: router_2
|
||||
endpoint_b_port: 2
|
||||
@@ -0,0 +1,81 @@
|
||||
game:
|
||||
max_episode_length: 256
|
||||
ports:
|
||||
- ARP
|
||||
protocols:
|
||||
- ICMP
|
||||
- TCP
|
||||
- UDP
|
||||
|
||||
simulation:
|
||||
network:
|
||||
airspace:
|
||||
airspace_environment_type: urban
|
||||
nodes:
|
||||
- type: computer
|
||||
hostname: pc_a
|
||||
ip_address: 192.168.0.2
|
||||
subnet_mask: 255.255.255.0
|
||||
default_gateway: 192.168.0.1
|
||||
start_up_duration: 0
|
||||
|
||||
- type: computer
|
||||
hostname: pc_b
|
||||
ip_address: 192.168.2.2
|
||||
subnet_mask: 255.255.255.0
|
||||
default_gateway: 192.168.2.1
|
||||
start_up_duration: 0
|
||||
|
||||
- type: wireless_router
|
||||
hostname: router_1
|
||||
start_up_duration: 0
|
||||
|
||||
router_interface:
|
||||
ip_address: 192.168.0.1
|
||||
subnet_mask: 255.255.255.0
|
||||
|
||||
wireless_access_point:
|
||||
ip_address: 192.168.1.1
|
||||
subnet_mask: 255.255.255.0
|
||||
frequency: WIFI_5
|
||||
channel_width: 80
|
||||
acl:
|
||||
1:
|
||||
action: PERMIT
|
||||
routes:
|
||||
- address: 192.168.2.0 # PC B subnet
|
||||
subnet_mask: 255.255.255.0
|
||||
next_hop_ip_address: 192.168.1.2
|
||||
metric: 0
|
||||
|
||||
- type: wireless_router
|
||||
hostname: router_2
|
||||
start_up_duration: 0
|
||||
|
||||
router_interface:
|
||||
ip_address: 192.168.2.1
|
||||
subnet_mask: 255.255.255.0
|
||||
|
||||
wireless_access_point:
|
||||
ip_address: 192.168.1.2
|
||||
subnet_mask: 255.255.255.0
|
||||
frequency: WIFI_5
|
||||
channel_width: 80
|
||||
acl:
|
||||
1:
|
||||
action: PERMIT
|
||||
routes:
|
||||
- address: 192.168.0.0 # PC A subnet
|
||||
subnet_mask: 255.255.255.0
|
||||
next_hop_ip_address: 192.168.1.1
|
||||
metric: 0
|
||||
links:
|
||||
- endpoint_a_hostname: pc_a
|
||||
endpoint_a_port: 1
|
||||
endpoint_b_hostname: router_1
|
||||
endpoint_b_port: 2
|
||||
|
||||
- endpoint_a_hostname: pc_b
|
||||
endpoint_a_port: 1
|
||||
endpoint_b_hostname: router_2
|
||||
endpoint_b_port: 2
|
||||
@@ -0,0 +1,106 @@
|
||||
# © Crown-owned copyright 2024, Defence Science and Technology Laboratory UK
|
||||
import yaml
|
||||
|
||||
from primaite.game.game import PrimaiteGame
|
||||
from primaite.simulator.network.airspace import (
|
||||
AirspaceEnvironmentType,
|
||||
AirSpaceFrequency,
|
||||
calculate_total_channel_capacity,
|
||||
ChannelWidth,
|
||||
)
|
||||
from primaite.simulator.network.hardware.nodes.network.wireless_router import WirelessRouter
|
||||
from tests import TEST_ASSETS_ROOT
|
||||
|
||||
|
||||
def test_wireless_wan_wifi_5_80_channel_width_urban():
|
||||
config_path = TEST_ASSETS_ROOT / "configs" / "wireless_wan_wifi_5_80_channel_width_urban.yaml"
|
||||
|
||||
with open(config_path, "r") as f:
|
||||
config_dict = yaml.safe_load(f)
|
||||
network = PrimaiteGame.from_config(cfg=config_dict).simulation.network
|
||||
|
||||
airspace = network.airspace
|
||||
|
||||
assert airspace.airspace_environment_type == AirspaceEnvironmentType.URBAN
|
||||
|
||||
router_1: WirelessRouter = network.get_node_by_hostname("router_1")
|
||||
router_2: WirelessRouter = network.get_node_by_hostname("router_2")
|
||||
|
||||
expected_speed = calculate_total_channel_capacity(
|
||||
channel_width=ChannelWidth.WIDTH_80_MHZ,
|
||||
frequency=AirSpaceFrequency.WIFI_5,
|
||||
environment_type=AirspaceEnvironmentType.URBAN,
|
||||
)
|
||||
|
||||
assert router_1.wireless_access_point.speed == expected_speed
|
||||
assert router_2.wireless_access_point.speed == expected_speed
|
||||
|
||||
pc_a = network.get_node_by_hostname("pc_a")
|
||||
pc_b = network.get_node_by_hostname("pc_b")
|
||||
|
||||
assert pc_a.ping(pc_a.default_gateway), "PC A should ping its default gateway successfully."
|
||||
assert pc_b.ping(pc_b.default_gateway), "PC B should ping its default gateway successfully."
|
||||
|
||||
assert pc_a.ping(pc_b.network_interface[1].ip_address), "PC A should ping PC B across routers successfully."
|
||||
assert pc_b.ping(pc_a.network_interface[1].ip_address), "PC B should ping PC A across routers successfully."
|
||||
|
||||
|
||||
def test_wireless_wan_wifi_5_80_channel_width_blocked():
|
||||
config_path = TEST_ASSETS_ROOT / "configs" / "wireless_wan_wifi_5_80_channel_width_blocked.yaml"
|
||||
|
||||
with open(config_path, "r") as f:
|
||||
config_dict = yaml.safe_load(f)
|
||||
network = PrimaiteGame.from_config(cfg=config_dict).simulation.network
|
||||
|
||||
airspace = network.airspace
|
||||
|
||||
assert airspace.airspace_environment_type == AirspaceEnvironmentType.BLOCKED
|
||||
|
||||
router_1: WirelessRouter = network.get_node_by_hostname("router_1")
|
||||
router_2: WirelessRouter = network.get_node_by_hostname("router_2")
|
||||
|
||||
expected_speed = calculate_total_channel_capacity(
|
||||
channel_width=ChannelWidth.WIDTH_80_MHZ,
|
||||
frequency=AirSpaceFrequency.WIFI_5,
|
||||
environment_type=AirspaceEnvironmentType.BLOCKED,
|
||||
)
|
||||
|
||||
assert router_1.wireless_access_point.speed == expected_speed
|
||||
assert router_2.wireless_access_point.speed == expected_speed
|
||||
|
||||
pc_a = network.get_node_by_hostname("pc_a")
|
||||
pc_b = network.get_node_by_hostname("pc_b")
|
||||
|
||||
assert pc_a.ping(pc_a.default_gateway), "PC A should ping its default gateway successfully."
|
||||
assert pc_b.ping(pc_b.default_gateway), "PC B should ping its default gateway successfully."
|
||||
|
||||
assert not pc_a.ping(pc_b.network_interface[1].ip_address), "PC A should ping PC B across routers unsuccessfully."
|
||||
assert not pc_b.ping(pc_a.network_interface[1].ip_address), "PC B should ping PC A across routers unsuccessfully."
|
||||
|
||||
|
||||
def test_wireless_wan_blocking_and_unblocking_airspace():
|
||||
config_path = TEST_ASSETS_ROOT / "configs" / "wireless_wan_wifi_5_80_channel_width_urban.yaml"
|
||||
|
||||
with open(config_path, "r") as f:
|
||||
config_dict = yaml.safe_load(f)
|
||||
network = PrimaiteGame.from_config(cfg=config_dict).simulation.network
|
||||
|
||||
airspace = network.airspace
|
||||
|
||||
assert airspace.airspace_environment_type == AirspaceEnvironmentType.URBAN
|
||||
|
||||
pc_a = network.get_node_by_hostname("pc_a")
|
||||
pc_b = network.get_node_by_hostname("pc_b")
|
||||
|
||||
assert pc_a.ping(pc_b.network_interface[1].ip_address), "PC A should ping PC B across routers successfully."
|
||||
assert pc_b.ping(pc_a.network_interface[1].ip_address), "PC B should ping PC A across routers successfully."
|
||||
|
||||
airspace.airspace_environment_type = AirspaceEnvironmentType.BLOCKED
|
||||
|
||||
assert not pc_a.ping(pc_b.network_interface[1].ip_address), "PC A should ping PC B across routers unsuccessfully."
|
||||
assert not pc_b.ping(pc_a.network_interface[1].ip_address), "PC B should ping PC A across routers unsuccessfully."
|
||||
|
||||
airspace.airspace_environment_type = AirspaceEnvironmentType.URBAN
|
||||
|
||||
assert pc_a.ping(pc_b.network_interface[1].ip_address), "PC A should ping PC B across routers successfully."
|
||||
assert pc_b.ping(pc_a.network_interface[1].ip_address), "PC B should ping PC A across routers successfully."
|
||||
@@ -0,0 +1,138 @@
|
||||
# © Crown-owned copyright 2024, Defence Science and Technology Laboratory UK
|
||||
from primaite.simulator.file_system.file_type import FileType
|
||||
from primaite.simulator.network.hardware.nodes.network.router import ACLAction
|
||||
from primaite.simulator.system.services.ftp.ftp_client import FTPClient
|
||||
from primaite.simulator.system.services.ftp.ftp_server import FTPServer
|
||||
from tests.integration_tests.network.test_wireless_router import wireless_wan_network
|
||||
from tests.integration_tests.system.test_ftp_client_server import ftp_client_and_ftp_server
|
||||
|
||||
|
||||
def test_wireless_link_loading(wireless_wan_network):
|
||||
client, server, router_1, router_2 = wireless_wan_network
|
||||
|
||||
# Configure Router 1 ACLs
|
||||
router_1.acl.add_rule(action=ACLAction.PERMIT, position=1)
|
||||
|
||||
# Configure Router 2 ACLs
|
||||
router_2.acl.add_rule(action=ACLAction.PERMIT, position=1)
|
||||
|
||||
airspace = router_1.airspace
|
||||
|
||||
client.software_manager.install(FTPClient)
|
||||
ftp_client: FTPClient = client.software_manager.software.get("FTPClient")
|
||||
ftp_client.start()
|
||||
|
||||
server.software_manager.install(FTPServer)
|
||||
ftp_server: FTPServer = server.software_manager.software.get("FTPServer")
|
||||
ftp_server.start()
|
||||
|
||||
client.file_system.create_file(file_name="mixtape", size=10 * 10**6, file_type=FileType.MP3, folder_name="music")
|
||||
|
||||
assert ftp_client.send_file(
|
||||
src_file_name="mixtape.mp3",
|
||||
src_folder_name="music",
|
||||
dest_ip_address=server.network_interface[1].ip_address,
|
||||
dest_file_name="mixtape.mp3",
|
||||
dest_folder_name="music",
|
||||
)
|
||||
|
||||
# Reset the physical links between the host nodes and the routers
|
||||
client.network_interface[1]._connected_link.pre_timestep(1)
|
||||
server.network_interface[1]._connected_link.pre_timestep(1)
|
||||
|
||||
assert ftp_client.send_file(
|
||||
src_file_name="mixtape.mp3",
|
||||
src_folder_name="music",
|
||||
dest_ip_address=server.network_interface[1].ip_address,
|
||||
dest_file_name="mixtape1.mp3",
|
||||
dest_folder_name="music",
|
||||
)
|
||||
|
||||
# Reset the physical links between the host nodes and the routers
|
||||
client.network_interface[1]._connected_link.pre_timestep(1)
|
||||
server.network_interface[1]._connected_link.pre_timestep(1)
|
||||
|
||||
assert ftp_client.send_file(
|
||||
src_file_name="mixtape.mp3",
|
||||
src_folder_name="music",
|
||||
dest_ip_address=server.network_interface[1].ip_address,
|
||||
dest_file_name="mixtape2.mp3",
|
||||
dest_folder_name="music",
|
||||
)
|
||||
|
||||
# Reset the physical links between the host nodes and the routers
|
||||
client.network_interface[1]._connected_link.pre_timestep(1)
|
||||
server.network_interface[1]._connected_link.pre_timestep(1)
|
||||
|
||||
assert not ftp_client.send_file(
|
||||
src_file_name="mixtape.mp3",
|
||||
src_folder_name="music",
|
||||
dest_ip_address=server.network_interface[1].ip_address,
|
||||
dest_file_name="mixtape3.mp3",
|
||||
dest_folder_name="music",
|
||||
)
|
||||
|
||||
# Reset the physical links between the host nodes and the routers
|
||||
client.network_interface[1]._connected_link.pre_timestep(1)
|
||||
server.network_interface[1]._connected_link.pre_timestep(1)
|
||||
|
||||
airspace.reset_bandwidth_load()
|
||||
|
||||
assert ftp_client.send_file(
|
||||
src_file_name="mixtape.mp3",
|
||||
src_folder_name="music",
|
||||
dest_ip_address=server.network_interface[1].ip_address,
|
||||
dest_file_name="mixtape3.mp3",
|
||||
dest_folder_name="music",
|
||||
)
|
||||
|
||||
|
||||
def test_wired_link_loading(ftp_client_and_ftp_server):
|
||||
ftp_client, computer, ftp_server, server = ftp_client_and_ftp_server
|
||||
|
||||
link = computer.network_interface[1]._connected_link # noqa
|
||||
|
||||
assert link.is_up
|
||||
|
||||
link.pre_timestep(1)
|
||||
|
||||
computer.file_system.create_file(
|
||||
file_name="mixtape", size=10 * 10**6, file_type=FileType.MP3, folder_name="music"
|
||||
)
|
||||
link_load = link.current_load
|
||||
assert link_load == 0.0
|
||||
|
||||
assert ftp_client.send_file(
|
||||
src_file_name="mixtape.mp3",
|
||||
src_folder_name="music",
|
||||
dest_ip_address=server.network_interface[1].ip_address,
|
||||
dest_file_name="mixtape.mp3",
|
||||
dest_folder_name="music",
|
||||
)
|
||||
|
||||
new_link_load = link.current_load
|
||||
assert new_link_load > link_load
|
||||
|
||||
assert not ftp_client.send_file(
|
||||
src_file_name="mixtape.mp3",
|
||||
src_folder_name="music",
|
||||
dest_ip_address=server.network_interface[1].ip_address,
|
||||
dest_file_name="mixtape1.mp3",
|
||||
dest_folder_name="music",
|
||||
)
|
||||
|
||||
link.pre_timestep(2)
|
||||
|
||||
link_load = link.current_load
|
||||
assert link_load == 0.0
|
||||
|
||||
assert ftp_client.send_file(
|
||||
src_file_name="mixtape.mp3",
|
||||
src_folder_name="music",
|
||||
dest_ip_address=server.network_interface[1].ip_address,
|
||||
dest_file_name="mixtape1.mp3",
|
||||
dest_folder_name="music",
|
||||
)
|
||||
|
||||
new_link_load = link.current_load
|
||||
assert new_link_load > link_load
|
||||
Reference in New Issue
Block a user