Введение в Iperf.
1 Общая информация
1.1 Цели
- Освоить основные приёмы работы с генератором трафика
iperf3
.
1.2 Задачи
- Подготовить лабораторный стенд.
- В процессе интерактивного эксперимента изучить основные опции и шаблоны использования генератора трафика
iperf3
. - Построить графики по данным интерактивного эксперимента.
- В процессе воспроизводимого эксперимента получить представление о создании скриптов эксперимента.
- Построить графики по данным воспроизводимого эксперимента.
1.3 Основные термины
- Полоса пропускания (Bandwidth) — это физическое свойство среды передачи, которое зависит от таких факторов, как конструкция и длина провода или волокна.
- Для сетевых инженеров пропускная способность — это максимальная скорость передачи данных канала, величина, измеряемая в битах в секунду (бит/с)
- На производительность могут влиять несколько факторов, таких как задержка, потеря пакетов, джиттер и другие.
- Пропускная способность (throughput) — это скорость в битах в секунду, с которой процесс-отправитель может доставлять биты процессу-получателю (при связи между двумя конечными устройствами).
- Поскольку другие сеансы будут совместно использовать полосу пропускания по сетевому пути и поскольку эти другие сеансы будут повторяться, доступная пропускная способность может колебаться со временем
- Иногда термины пропускная способность (throughput) и полоса пропускания (bandwidth) используются взаимозаменяемо.
1.4 Iperf3
- iPerf3 — это инструмент для измерения пропускной способности сети в режиме реального времени.
- Сайт: https://iperf.fr/.
- Репозиторий: https://github.com/esnet/iperf.
- Лицензия: 3-пунктовая лицензия BSD.
- iPerf3 может работать с протоколами TCP, UDP и SCTP:
- TCP и SCTP:
- измерение пропускной способности;
- размер MSS/MTU;
- поддержка размера окна TCP.
- UDP:
- клиент может создавать потоки UDP с заданной пропускной способностью;
- измерение потери пакетов;
- измерение джиттера задержки;
- поддержка групповой рассылки (multicast).
- TCP и SCTP:
1.5 Графики по iperf3
- Для построения графиков по iperf3 будем использовать пакет
iperf3_plotter
. - Репозиторий: https://github.com/ekfoury/iperf3_plotter.
- Скрипты:
preprocessor.sh
: преобразует файл JSON iPerf3 в файл значений, разделённых запятыми (CSV). Использует AWK для форматирования полей.plot_iperf.sh
: принимает JSON-файл iPerf3, вызывает препроцессор и вызывает gnuplot для создания выходных PDF-файлов.
1.6 Видео: Введение в Iperf. Введение
2 Подготовка стенда
2.1 Лабораторная топология
- В топологии используется сеть 10.0.0.0/8, назначенная Mininet по умолчанию.
- Топология представляет собой коммутатор с двумя подключёнными к нему хостами.
- Для запуска топологии используем параметры командной строки:
sudo mn --topo=single,2 -x
- Посмотрите топологию:
mininet> net mininet> links mininet> dump
2.2 Рабочие файлы
- Подключимся к виртуальной машине:
ssh -Y mininet@192.168.x.y
- Для рабочих файлов будем использовать каталог
~/work/lab_iperf3
:mkdir -p ~/work/lab_iperf3
2.3 Установка необходимого программного обеспечения
2.3.1 Конфигурация интерфейсов виртуальной машины
- Добавьте к виртуальной машине интерфейс NAT.
- Загрузите виртуальную машину и войдите в учётную запись
mininet
(парольmininet
). - Добавьте в файл
/etc/netplan/01-netcfg.yaml
второй интерфейс. - В результате файл примет следующий вид:
network: version: 2 renderer: networkd ethernets: eth0: dhcp4: yes eth1: dhcp4: yes
- Перезагрузите виртуальную машину.
2.3.2 Установка iperf3
- По умолчанию стоит генератор трафика iperf2 (команда
iperf
). - Установим iperf3.
- Загрузите виртуальную машину и войдите в учётную запись
mininet
:ssh -Y mininet@192.168.x.y
- Обновим список пакетов:
sudo apt-get update
- Установите iperf3:
sudo apt-get install iperf3
2.3.3 Установка скриптов для визуализации результатов
- Для функционирования скриптов необходимы следующие пакеты:
- gnuplot;
- jq;
- awk.
2.3.4 Установка iperf3_plotter
на виртуальную машину
Зайдём на виртуальную машину:
ssh -Y mininet@192.168.x.y
Обновим репозитории программного обеспечения на виртуальной машине:
sudo apt-get update
Установим необходимое программное обеспечение на виртуальную машину:
sudo apt-get install git jq gnuplot-nox mc
Скачаем
iperf3_plotter
. Для этого перейдём во временную рабочую папку и скачаем репозиторий:cd /tmp git clone https://github.com/ekfoury/iperf3_plotter.git
Установим
iperf3_plotter
:cd /tmp/iperf3_plotter sudo cp plot_* /usr/bin sudo cp *.sh /usr/bin
Скрипты написаны плохо, не работают с путями, имеющими в названии пробелы.
2.3.5 Установка iperf3_plotter
на локальный компьютер
- Установим
iperf3_plotter
(на своём компьютере). Для этого перейдём во временную рабочую папку и скачаем репозиторий:git clone https://github.com/ekfoury/iperf3_plotter.git
- Заменим жёстко закодированные пути в файлах:
cd iperf3_plotter sed -i -e "s:/usr/bin:~/.local/bin:g" plot_iperf.sh
- Скопируем нужные файлы в каталог с локальными исполняемыми файлами:
cp plot_* ~/.local/bin cp *.sh ~/.local/bin
- Скрипты написаны плохо, не работают с путями, имеющими в названии пробелы.
2.4 Видео: Введение в Iperf. Подготовка стенда
3 Интерактивный эксперимент
3.1 Параметры команды iperf3
Пользователь взаимодействует с iPerf3 с помощью команды
iperf3
.Основной синтаксис
iperf3
, используемый как на клиенте, так и на сервере, выглядит следующим образом:iperf3 [-s|-c] [ options ]
-s
: запуск сервера;-c
: запуск клиента.
3.2 Запуск клиента и сервера
Запустим
mininet
:sudo mn --topo=single,2 -x
В терминале
h2
запустите сервер:iperf3 -s
- Теперь сервер прослушивает порт 5201 в ожидании входящих подключений.
В терминале
h1
запустите клиента:iperf3 -c 10.0.0.2
- Параметр
-c
указывает, что хостh1
настроен как клиент. - Параметр
10.0.0.2
— это IP-адрес сервера (хостh2
).
- Параметр
После завершения теста отображается сводный отчет как на клиенте, так и на сервере, содержащий следующие данные:
ID
: идентификационный номер соединения.- Интервал (Interval): временной интервал для периодических отчетов о пропускной способности. По умолчанию временной интервал равен 1 секунде.
- Передача (Transfer): сколько данных было передано за каждый интервал времени.
- Пропускная способность (Bitrate): измеренная пропускная способность в каждом временном интервале.
- Retr: количество повторно переданных TCP-сегментов за каждый временной интервал. Это поле увеличивается, когда TCP-сегменты теряются в сети из-за перегрузки или повреждения.
- Cwnd: указывает размер окна перегрузки в каждом временном интервале. TCP использует эту переменную для ограничения объема данных, которые клиент TCP может отправить до получения подтверждения отправленных данных.
Чтобы остановить сервер, нажмите
Ctrl+c
в терминале хостаh2
.Пользователь также может видеть результаты пропускной способности на стороне сервера.
Суммарные данные на сервере аналогичны данным на стороне клиента и должны интерпретироваться таким же образом.
3.3 Запуск клиента и сервера в интерфейсе mininet
- Тоже самое можно выполнить и интерфейсе mininet
- Запустите сервер на хосте
h2
:mininet> h2 iperf3 -s &
- Запустите клиента на хосте
h1
:mininet> h1 iperf3 -c h2
- Остановите серверный процесс:
mininet> h2 killall iperf3
3.4 Установка периода времени передачи
- В терминале
h2
запустите сервер:iperf3 -s
- В терминале
h1
запустите клиента с параметром-t
, за которым следует количество секунд:iperf3 -c 10.0.0.2 -t 5
- Чтобы остановить сервер, нажмите
Ctrl+c
в терминале хостаh2
.
3.5 Установка временного интервала
- Настроим клиент для выполнения теста пропускной способности с 2-секундным интервалом времени отсчёта как на клиенте, так и на сервере.
- Опция
-i
позволяет установить интервал между отсчётами в секундах. - В этом случае значение должно быть установлено на 2 секунды как на клиенте, так и на сервере.
- Параметр
-i
может быть указан по-разному на клиенте и сервере. - В терминале
h2
запустите сервер:iperf3 -s -i 2
- В терминале
h1
запустите клиента с параметром-i
, за которым следует количество секунд:iperf3 -c 10.0.0.2 -i 2
- Чтобы остановить сервер, нажмите
Ctrl+c
в терминале хостаh2
.
3.6 Изменение количества байтов для передачи
Зададим на клиенте отправку определённого объёма данных путём установки количества байтов для передачи.
По умолчанию iPerf3 выполняет измерение пропускной способности в течение 10 секунд.
Однако при задании количества данных клиент будет продолжать отправлять пакеты до тех пор, пока не будут отправлены все байты, указанные пользователем.
В терминале
h2
запустите сервер:iperf3 -s
В терминале
h1
запустите клиента, задав объём данных:iperf3 -c 10.0.0.2 -n 16G
- Параметр
-n
указывает объем данных для передачи: 16 Гбайт.
- Параметр
Чтобы остановить сервер, нажмите
Ctrl+c
в терминале хостаh2
.
3.7 Определение протокола транспортного уровня
- По умолчанию измерения пропускной способности проводятся по протоколу TCP, который является протоколом конфигурации по умолчанию.
- Чтобы изменить протокол на UDP, нужно задать опцию
-u
на стороне клиента. - Для протокола SCTP используется опция
--sctp
. - iPerf3 автоматически определяет протокол транспортного уровня на стороне сервера.
- В терминале
h2
запустите сервер:iperf3 -s
- В терминале
h1
запустите клиента, задав протокол UDP:iperf3 -c 10.0.0.2 -u
- Чтобы остановить сервер, нажмите
Ctrl+c
в терминале хостаh2
. - После завершения теста он покажет следующие сводные данные:
- ID, интервал, передача, битрейт: то же, что и у TCP.
- Джиттер (Jitter): разница в задержке пакетов.
- Lost/Total: указывает количество потерянных дейтаграмм по сравнению с общим количеством отправленных на сервер (и процентное соотношение).
3.8 Изменение номера порта
- Если нужно измерить пропускную способность определённого порта, используется параметр
-p
для настройки клиента и сервера для отправки/получения пакетов или датаграмм через указанный порт. - В терминале
h2
запустите сервер, используя параметр-p
, чтобы указать порт прослушивания:iperf3 -s -p 3250
- В терминале
h1
запустите клиента, указав порт:iperf3 -c 10.0.0.2 -p 3250
- Чтобы остановить сервер, нажмите
Ctrl+c
в терминале хостаh2
.
3.9 Работа с одним клиентом
- По умолчанию сервер iPerf3 продолжает прослушивать входящие соединения.
- Чтобы сервер мог обрабатывать одного клиента, а затем останавливаться, к серверу добавляется параметр
-1
. - В терминале
h2
запустите сервер, используя параметр-1
, чтобы принять только одного клиента:iperf3 -s -1
- В терминале
h1
запустите клиента, указав порт:iperf3 -c 10.0.0.2
- После завершения этого теста сервер немедленно останавливается.
3.10 Экспорт результатов в файл JSON
- JSON (JavaScript Object Notation, нотация объектов JavaScript) — это облегченный формат обмена данными.
- iPerf3 позволяет экспортировать результаты теста в файл JSON, что позволяет другим приложениям легко анализировать файл и интерпретировать результаты.
- В терминале
h2
запустите сервер:iperf3 -s
- В терминале
h1
запустите клиента, указав параметр-J
для отображения вывода в формате JSON:iperf3 -c 10.0.0.2 -J
- Параметр
-J
выводит текст JSON на экран через стандартный вывод (stdout
) после завершения теста. - Чтобы экспортировать вывод в файл нужно перенаправить стандартный вывод в файл:
iperf3 -c 10.0.0.2 -J > /home/mininet/work/lab_iperf3/iperf_results.json chown mininet:mininet /home/mininet/work/lab_iperf3/iperf_results.json
- После создания файла JSON команда
ls
используется для проверки того, что файл создан. - Команда
cat
может использоваться для отображения содержимого файла. - Чтобы остановить сервер, нажмите
Ctrl+c
в терминале хостаh2
.
3.11 График результатов iPerf3
- Визуализируем результаты эксперимента.
- Скопируйте файл
iperf3_results.json
с виртуальной машины к себе, в рабочий каталог. - Обращайте внимание, чтобы путь к файлу не содержал пробелов.
- Чтобы сгенерировать выходные данные для файла JSON iPerf3, выполните следующую команду:
plot_iperf.sh iperf3_results.json
- Этот сценарий создаёт графики для следующих полей:
- окно перегрузки (
cwnd.pdf
), - повторная передача (
retransmits.pdf
), - время приема-передачи (
RTT.pdf
), - отклонение времени приема-передачи (
RTT_Var.pdf
), - пропускная способность (
throughput.pdf
), - максимальная единица передачи (
MTU.pdf
), - количество переданных байтов (
bytes.pdf
).
- окно перегрузки (
- Сценарий построения также создаёт файл CSV (
1.dat
), который может использоваться другими приложениями. - Эти файлы хранятся в каталоге результатов, созданном в том же каталоге, где был выполнен скрипт.
3.12 Видео: Введение в Iperf. Интерактивный эксперимент
4 Воспроизводимый эксперимент
4.1 API Mininet
- API Mininet построен на трех основных уровнях:
- Низкоуровневый API. Низкоуровневый API состоит из базовых узлов и классов ссылок (таких как
Host
,Switch
,Link
и их подклассы), которые на самом деле могут быть созданы по отдельности и использоваться для создания сети, но это немного громоздко. - API среднего уровня. API среднего уровня добавляет объект Mininet, который служит контейнером для узлов и ссылок. Он предоставляет ряд методов (
addHost()
,addSwitch()
,addLink()
) для добавления узлов и ссылок в сеть, а также настройки сети, запуска и завершения работы (start()
,stop()
). - Высокоуровневый API. Высокоуровневый API добавляет абстракцию шаблона топологии (класс
Topo
), который предоставляет возможность создавать повторно используемые параметризованные шаблоны топологии. Эти шаблоны можно передать команде mn (через параметр –custom) и использовать из командной строки.
- Низкоуровневый API. Низкоуровневый API состоит из базовых узлов и классов ссылок (таких как
- Когда напрямую управляют узлами и коммутаторами, используется низкоуровневый API.
- Когда запускают или останавливают сеть, используют API среднего уровня (в частности, класс
Mininet
). - Полные сети могут быть созданы с использованием любого из уровней API, но обычно выбирают либо API среднего уровня (например,
Mininet.add*()
), либо API высокого уровня (Topo. add*()
) для создания сетей.
4.2 Примеры создания сетей с использованием уровней API Mininet
- Низкоуровневый API. Узлы и ссылки:
h1 = Host( 'h1' ) h2 = Host( 'h2' ) s1 = OVSSwitch( 's1', inNamespace=False ) c0 = Controller( 'c0', inNamespace=False ) Link( h1, s1 ) Link( h2, s1 ) h1.setIP( '10.1/8' ) h2.setIP( '10.2/8' ) c0.start() s1.start( [ c0 ] ) print( h1.cmd( 'ping -c1', h2.IP() ) ) s1.stop() c0.stop()
- API среднего уровня. Сетевой объект:
net = Mininet() h1 = net.addHost( 'h1' ) h2 = net.addHost( 'h2' ) s1 = net.addSwitch( 's1' ) c0 = net.addController( 'c0' ) net.addLink( h1, s1 ) net.addLink( h2, s1 ) net.start() print( h1.cmd( 'ping -c1', h2.IP() ) ) CLI( net ) net.stop()
- API высокого уровня. Шаблоны топологии:
class SingleSwitchTopo( Topo ): "Single Switch Topology" def build( self, count=1 ): hosts = [ self.addHost( 'h%d' % i ) for i in range( 1, count + 1 ) ] s1 = self.addSwitch( 's1' ) for h in hosts: self.addLink( h, s1 ) net = Mininet( topo=SingleSwitchTopo( 3 ) ) net.start() CLI( net ) net.stop()
- API среднего уровня на самом деле является самым простым и лаконичным для этого примера, поскольку не требует создания класса топологии.
- API низкого и среднего уровня являются гибкими и мощными, но могут быть менее удобными для повторного использования по сравнению с высокоуровневым API
Topo
и шаблонами топологии.
4.3 Простейшая топология
На основе скрипта
mininet/examples/emptynet.py
создадим простейшую топологиюlab_iperf3_topo
:#!/usr/bin/env python """ Simple topology """ from mininet.net import Mininet from mininet.node import Controller from mininet.cli import CLI from mininet.log import setLogLevel, info def emptyNet(): "Create an empty network and add nodes to it." net = Mininet( controller=Controller, waitConnected=True ) info( '*** Adding controller\n' ) net.addController( 'c0' ) info( '*** Adding hosts\n' ) h1 = net.addHost( 'h1', ip='10.0.0.1' ) h2 = net.addHost( 'h2', ip='10.0.0.2' ) info( '*** Adding switch\n' ) s1 = net.addSwitch( 's1' ) info( '*** Creating links\n' ) net.addLink( h1, s1 ) net.addLink( h2, s1 ) info( '*** Starting network\n') net.start() info( '*** Running CLI\n' ) CLI( net ) info( '*** Stopping network' ) net.stop() if __name__ == '__main__': setLogLevel( 'info' ) emptyNet()
Основные элементы:
addSwitch()
: добавляет коммутатор в топологию и возвращает имя коммутатора;ddHost()
: добавляет хост в топологию и возвращает имя хоста;addLink()
: добавляет двунаправленную ссылку в топологию (и возвращает ключ ссылки, но это не важно). Ссылки в Mininet являются двунаправленными, если не указано иное.Mininet
: основной класс для создания и управления сетью;start()
: запускает вашу сеть;pingAll()
: проверяет подключение, пытаясь заставить все узлы пинговать друг друга;stop()
: останавливает вашу сеть;net.hosts
: все хосты в сети;dumpNodeConnections()
: сбрасывает подключения к/от набора узлов;setLogLevel( 'info' | 'debug' | 'output' )
: установить уровень вывода Mininet по умолчанию; рекомендуетсяinfo
.
Создадим в лабораторной работе подкаталог для данного эксперимента:
mkdir -p ~/work/lab_iperf3/iperf3_topo
Скопируем скрипт на виртуальную машину в каталог лабораторной и запустим:
sudo python lab_iperf3_topo.py
После запуска можно посмотреть элементы топологии:
mininet> net mininet> links mininet> dump
4.4 Методы настройки хоста
- Хосты Mininet предоставляют ряд удобных методов настройки сети:
IP()
: возвращает IP-адрес хоста или определенного интерфейса;MAC()
: возвращает MAC-адрес хоста или определенного интерфейса;setARP()
: добавить статическую запись ARP в кэш ARP хоста;setIP()
: установить IP-адрес для хоста или определенного интерфейса;setMAC()
: установить MAC-адрес для хоста или определенного интерфейса.
- Пример использования в скрипте:
print( "Host", h1.name, "has IP address", h1.IP(), "and MAC address", h1.MAC() )
- Задание: Попробуйте изменить приведённый выше код, чтобы он печатал IP-адрес и MAC-адрес каждого хоста.
4.5 Настройка параметров производительности
Mininet предоставляет функции ограничения производительности и изоляции с помощью классов
CPULimitedHost
иTCLink
:from mininet.node import CPULimitedHost from mininet.link import TCLink ... net = Mininet( controller=Controller, host = CPULimitedHost, link = TCLink )
Основные функции:
self.addHost(name, cpu=f)
: позволяет указать долю общих ресурсов процессора системы, которая будет выделена виртуальному хосту;self.addLink(node1, node2, bw=10, delay='5ms', max_queue_size=1000, loss=10, use_htb=True)
: добавляет двунаправленный канал с характеристиками пропускной способности, задержки и потерь:- максимальный размер очереди 1000 пакетов;
- использование ограничителя скорости Hierarchical Token Bucket;
- эмулятор задержки/потери
netem
; - параметр
bw
выражается числом в Мбит; - задержка выражается в виде строки с заданными единицами измерения (например,
5ms
,100us
,1s
); - потери выражаются в процентах (от 0 до 100);
max_queue_size
выражается в пакетах.
Задание: Измените приведённый выше код, чтобы каналы были 100 Мбит/с с задержкой 75 мс. Рассчитать производительность связи между
h1
иh2
.
4.6 Полный эксперимент
- Сгенерируем трафик и сохраним результат в файл json.
- Зададим скрипт
lab_iperf3.py
для полного эксперимента:#!/usr/bin/env python """ Simple experiment. Output: iperf_result.json """ from mininet.net import Mininet from mininet.node import Controller from mininet.cli import CLI from mininet.log import setLogLevel, info from mininet.link import TCLink import time def emptyNet(): "Create an empty network and add nodes to it." net = Mininet( controller=Controller, waitConnected=True, link = TCLink ) info( '*** Adding controller\n' ) net.addController( 'c0' ) info( '*** Adding hosts\n' ) h1 = net.addHost( 'h1', ip='10.0.0.1' ) h2 = net.addHost( 'h2', ip='10.0.0.2' ) info( '*** Adding switch\n' ) s1 = net.addSwitch( 's1' ) info( '*** Creating links\n' ) net.addLink( h1, s1, cls=TCLink, bw=100, delay='75ms' ) net.addLink( h2, s1, cls=TCLink, bw=100, delay='75ms' ) info( '*** Starting network\n') net.start() info( '*** Traffic generation\n') h2.cmdPrint( 'iperf3 -s -D -1' ) time.sleep(10) # Wait 10 seconds for servers to start h1.cmdPrint( 'iperf3 -c', h2.IP(), '-J > iperf_result.json' ) info( '*** Stopping network' ) net.stop() if __name__ == '__main__': setLogLevel( 'info' ) emptyNet()
- Создадим в лабораторной работе подкаталог для данного эксперимента:
mkdir -p ~/work/lab_iperf3/iperf3
- Запустите скрипт на виртуальной машине:
sudo python lab_iperf3.py
- Постройм графики из получившегося файла json:
plot_iperf.sh iperf_result.json
- Создадим
Makefile
для проведения всего эксперимента:all: iperf_result.json plot iperf_result.json: sudo python lab_iperf3.py plot: iperf_result.json plot_iperf.sh iperf_result.json clean: -rm -f *.json *.csv -rm -rf results