Ускоряем Apache.JMeter

Вячеслав Смирнов, Райффайзенбанк

Ускоряем
Apache.JMeter

Вячеслав Смирнов

Эксперт по тестированию,
Райффайзенбанк

Бываю на конференциях

Apache.JMeter

    Рассмотрим
  1. HTTP Request. Максимальная интенсивность
  2. HTTP Request. Скачивание и отправка
  3. PostProcessor
  4. PreProcessor
  5. Секретное оружие

Цели
Разрабатывать такие тесты,
чтобы система тормозила, а не Apache.JMeter.
Экономить на ресурсах нагрузочных агентов.
Разрабатывать тесты быстро и просто.

HTTP Request
Максимальная интенсивность

Samplers.HTTP.Request.X. Простой тест

Несколько GET-запросов на локальный сервер

Запуск и остановка NGinx в тесте, используя OS Process Sampler

NGinx c конфигурацией по умолчанию

Для скорости — два процесса вместо одного

			worker_processes  2; # default 1
			worker_connections  512;
			sendfile off;
			tcp_nodelay on; 
			tcp_nopush off; 
			keepalive_timeout  75s;
			keepalive_requests 100;
			keepalive_disable msie6;
			gzip off;
		

Настраиваются параметры HTTP-запроса

Параметризованы хост, порт, путь. Все настройки — по умолчанию.

Используется цикл для цепочки запросов

${RequestCount} последовательных запросов в каждой итерации

Количество потоков и итераций настраивается

Профиль — стабильная высокая нагрузка от ${Threads} потоков

HTTP Request. Серия экспериментов

Средняя интенсивность для одного потока

Средняя интенсивность для двух потоков

Средняя интенсивность для трёх потоков

На коротком сценарии лучше без Keep-Alive

На среднем сценарии — влияние Keep-Alive небольшое

На больших сценариях — Keep-Alive ускоряет работу

На больших сценариях — Keep-Alive ускоряет работу

Очевидно, что
Keep-Alive полезен, не нужно отключать его.
Лучший результат —
с большим сценарием (50+) и Keep-Alive.

Не делать benchmark на Apache.JMeter
из одного HTTP Request в сценарии
при настройках по умолчанию.

Сценарий на 10 запросов самый частый

Ускорим тест c 10 запросами в сценарии
на 3 потока/пользователя в катушке
c 4k req/s до 16k req/s
и больше.

Понадобится

Средняя интенсивность 4855/s,

должна быть стабильная нагрузка,
а видим всплески от 500/s до 16000/s

Есть ошибки 5.24%

Non HTTP response code: java.net.NoRouteToHostException/Non HTTP response message: Невозможно назначить запрошенный адрес (Address not available)

SJK: socketConnect (64,33%)

Apache.JMeter ждёт соединения

Скрипт JMeter: увеличить количество запросов (RequestCount) до 50

Интенсивность 16500/s (x 3,3) и без ошибок

Нужно удлиннять сценарий — запрещённый вариант

Скрипт JMeter: добавить HTTP Cache Manager

Может ускорит?

Скрипт JMeter: добавить HTTP Cache Manager

Интенсивность 4800/s и 4.9% ошибок (не повлиял)

SJK: socketConnect (62,73%)

Apache.JMeter ждёт соединения

Модификацией скрипта интенсивность не поднять.
Решили оставить 10 запросов в итерации.
А HTTP Cache Manager не ускорил
получение маленького ответа.

Настройки JMeter: jmeter.httpsampler=Java

Интенсивность 16300/s (x 3,3) и без ошибок

SJK: socketRead (34,34%)

Apache.JMeter читает заголовок ответа

Настройки JMeter: не сбрасывать состояние

httpclient.reset_state_on_thread_group_iteration (false)

			#---------------------------------------------------------------------------
			# SSL configuration
			#---------------------------------------------------------------------------
			httpclient.reset_state_on_thread_group_iteration=false
		

Настройки JMeter: не сбрасывать состояние

Интенсивность 19240/s (x 3,9) и без ошибок

Настройки JMeter

Если интенсивность скачет (500/s - 16000/s), есть ошибки (Address not available) и JMeter занят socketConnect (> 50%)

Можно перейти с HTTPClient4 на Java-клиент (x 3,3)

			jmeter.httpsampler=Java
		

Или кешировать соединение между итерациями (x 3,9)

			httpclient.reset_state_on_thread_group_iteration=false
		

Подключения к NGinx идут волнами

от 0/s до 1200/s (x 0.1 от интенсивности)

CPU: преимущественно системное время

%System в 3,5 раза выше, чем %User

netstat: tcp_time_wait (28 231)

Открывается слишком много соединений, лимит портов

Закрытые соединения с nginx (localhost:5555)

Одно ESTABLISHED и 27500 TIME_WAIT

			netstat | grep -oP \
			"^\w+\s+\d+\s+\d+\s+(\w+:\w+)\s+(localhost:5555)\s.*"
		
			tcp  0  0 localhost:*****  localhost:5555  ESTABLISHED
			tcp6 0  0 localhost:*****  localhost:5555  TIME_WAIT
			tcp6 0  0 localhost:*****  localhost:5555  TIME_WAIT
			...
		

Linux: net.ipv4.ip_local_port_range (32768 60999)

Больше диапазон — больше соединений

Изначально диапазон на 28231 портов:

			cat /proc/sys/net/ipv4/ip_local_port_range
			32768 60999
		

Linux: net.ipv4.ip_local_port_range (1025 60999)

Больше диапазон — больше соединений

Расширим c 28231 до 59974 (x 2,1):

			echo 1025 60999 > /proc/sys/net/ipv4/ip_local_port_range
		
Проблемы с очередью TIME_WAIT (@blog.kireev.pro)

Средняя интенсивность 7772/s (x 1,6)

нестабильная: 3500/s ... 19000/s

Теперь без ошибок

ошибки ушли

SJK: socketConnect (53,28%)

Apache.JMeter ждёт соединения

Ускорение в 1,6 раза — c 4800/s до 7700/s
и избавление от ошибок Address not available
при расширения диапазона портов
net.ipv4.ip_local_port_range (1025 60999)

netstat: tcp_time_wait (32 768, не 59 975)

новый лимит — симптомы полечены, проблема осталась

Linux: net.ipv4.tcp_max_tw_buckets (32768)

Расширим размер очереди TIME_WAIT

Значение в Linux можно посмотреть с помощью команды cat так:

			cat /proc/sys/net/ipv4/tcp_max_tw_buckets
			32768
		

Linux: net.ipv4.tcp_max_tw_buckets (65536)

Расширим размер очереди TIME_WAIT (временно)

Увеличим в два раза до значения 65536. Временную настройку можно выполнить так:

			echo 65536 > /proc/sys/net/ipv4/tcp_max_tw_buckets
			
		

Средняя интенсивность 7700/s (та же)

нестабильная: 3500/s ... 19000/s

netstat: tcp_time_wait (40 000 - 46 000)

лимита в 32768 больше нет — но не помогло

Увеличение размера очереди TIME_WAIT
net.ipv4.tcp_max_tw_buckets (65536)
не дало положительного эффекта для скорости,
но нет больше полки для netstat: tcp_time_wait

SJK: socketConnect (54,85%)

Apache.JMeter ждёт соединения

Linux: net.ipv4.tcp_tw_reuse (1)

Разрешим переиспользовать TIME_WAIT

По умолчанию для исходящих подключений нельзя использовать TIME_WAIT-соединения.

Разрешим:

			echo 1 > /proc/sys/net/ipv4/tcp_tw_reuse
		

Средняя интенсивность 17300/s (x 3,6)

более стабильная: 16500/s ... 19000/s

SJK: socketRead (38,31%)

Apache.JMeter читает заголовок ответа

CPU: больше не упирается в систему

%User в 2 раза выше, чем %System

netstat: tcp_time_wait (30 000, снижение)

подозрительно ровная полка

Рекомендации для Gatling (bit.ly/gatling-tuning)

Gatling documentation / 3.1 / General / Operations

Рекомендации для Gatling

-Djava.net.preferIPv4Stack=true
-Djava.net.preferIPv6Addresses=false
ulimit -n 65536
sudo sysctl -w net.ipv4.ip_local_port_range="1025 65535"
echo 300000 | sudo tee /proc/sys/fs/nr_open
echo 300000 | sudo tee /proc/sys/fs/file-max
net.ipv4.tcp_max_syn_backlog = 40000
net.core.somaxconn = 40000
net.core.wmem_default = 8388608
net.core.rmem_default = 8388608
net.ipv4.tcp_sack = 1
		

Рекомендации для Gatling

net.ipv4.tcp_window_scaling = 1
net.ipv4.tcp_fin_timeout = 15
net.ipv4.tcp_keepalive_intvl = 30
net.ipv4.tcp_tw_reuse = 1
net.ipv4.tcp_moderate_rcvbuf = 1
net.core.rmem_max = 134217728
net.core.wmem_max = 134217728
net.ipv4.tcp_mem  = 134217728 134217728 134217728
net.ipv4.tcp_rmem = 4096 277750 134217728
net.ipv4.tcp_wmem = 4096 277750 134217728
net.core.netdev_max_backlog = 300000
		

И для Yandex.Tank (bit.ly/tank-tuning)

ulimit -n 30000

net.ipv4.tcp_max_tw_buckets = 65536
net.ipv4.tcp_tw_recycle = 1
net.ipv4.tcp_tw_reuse = 0
net.ipv4.tcp_max_syn_backlog = 131072
net.ipv4.tcp_syn_retries = 3
net.ipv4.tcp_synack_retries = 3
net.ipv4.tcp_retries1 = 3
net.ipv4.tcp_retries2 = 8
net.ipv4.tcp_rmem = 16384 174760 349520
net.ipv4.tcp_wmem = 16384 131072 262144
net.ipv4.tcp_mem = 262144 524288 1048576
net.ipv4.tcp_max_orphans = 65536
net.ipv4.tcp_fin_timeout = 10
net.ipv4.tcp_low_latency = 1
net.ipv4.tcp_syncookies = 0
net.netfilter.nf_conntrack_max = 1048576	
		

Ускорение в 3,6 раза — c 4800/s до 17300/s
при разрешении переиспользовать TIME_WAIT
net.ipv4.tcp_tw_reuse (1)

Ускорители Apache.JMeter

Опция Значение Ускорение
httpclient.reset_state_on_thread_group_iteration false x3.9
jmeter.httpsampler Java x3.3
Linux: net.ipv4.tcp_tw_reuse 1 x3.6
Linux: net.ipv4.tcp_max_tw_buckets 65536
Linux: net.ipv4.ip_local_port_range 1025 60999 x1.6
NGinx: worker_processes 2 (+1)
Настройки Gatling bit.ly/gatling-tuning и Yandex.Tank bit.ly/tank-tuning

Если попробовать смешать?

Apache.JMeter читает, пишет, собирает статистику

HTTP Request
Скачивание файлов

Последовательное скачивание файла, используя Thread Group на один поток и 200 итераций, имея 4 ГБайт Heap Size

Создадим файл на 1 ГБайт и скачаем его 200 раз

			dd  if=/dev/urandom of=/tmp/data/1g.img  \
			    bs=1 count=0 seek=1G
		

200 ГБайт скачались достаточно быстро

Параметр Значение
Среднее время 1460 ms
Длительность 5 минут
Необходимая память 4096 МБайт

Трафик памяти огромный 1,81 ГБайт/сек

Память используется постоянно. 1 минута на GC

SJK: socketRead (94,72%)

Apache.JMeter читает тело ответа

Больше 1 ГБайт не скачать для Heap 4 ГБайт

Ошибка OutOfMemoryError уже для файла в 1100 МБайт

HTTP Request с настройками по умолчанию,
расходует очень много памяти
4-х кратный буфер от размера файла
генерируя объекты по 1,8 ГБайт/с,
но качает 200 ГБайт за 5 минут

☑️ Save response as MD5 Hash

экономит память при скачивании больших файлов

200 ГБайт с MD5 скачивались дольше

Параметр Значение
Среднее время 5475 ms
Длительность 18 минут 17 сек
Необходимая память 0 МБайт (минимум)

Трафик памяти при включенном MD5 нулевой

Память не использовалась вообще, CPU тоже меньше

SJK: MessageDigest (66.90%)

JMeter считает MD5 в два раза дольше, чем читает

☑️ Save response as MD5 Hash
экономит память до 0,
экономит процессорное время,
не имеет ограничений на размер файла,
но замедляет скачивание до 3-x раз

Как скачивать быстро
с минимальными расходами памяти
из Apache.JMeter?
Может быть curl/wget и bash помогут?

wget есть в Linux

wget-download.sh url

			wget -S \
			    --progress=dot:mega \ 
			    --output-document=/dev/null \ 
			    $1 
		

wget есть и в Microsoft Windows (MinGW)

wget-download.bat url

			wget -S ^
			    --progress=dot:mega ^ 
			    --output-document=- ^ 
			    %1 ^
			    1>NUL
		

Можно ограничить максимальную скорость

wget-download.bat url

			wget -S ^
			    --limit-rate=20m ^ 
			    --progress=dot:mega ^ 
			    --output-document=- ^ 
			    %1 ^
			    1>NUL
		

Что превосходит аналог (cps)

по удобству использования

			# Define characters per second > 0 to emulate slow connections
			httpclient.socket.http.cps=0
			httpclient.socket.https.cps=0
		

Прикинуться браузером с поддержкой gzip

wget-download.gzip.bat url

			wget -S --progress=dot:mega ^ 
			    --header "User-Agent: Mozilla/5.0 (Windows NT 6.1; WOW64; rv:44.0) Gecko/20100101 Firefox/44.0" ^ 
			    --header "Accept: text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8" ^ 
			    --header "Accept-Encoding: gzip, deflate" ^ 
			    --output-document=- %1 1>NUL
		

Передать первые 1000 байт в PostProcessor

wget-download.gzip.head.bat url

			wget -S --progress=dot:mega ^ 
			    --header "User-Agent: Mozilla/5.0 (Windows NT 6.1; WOW64; rv:44.0) Gecko/20100101 Firefox/44.0" ^ 
			    --header "Accept: text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8" ^ 
			    --header "Accept-Encoding: gzip, deflate" ^ 
			    --output-document=- %1 | ^
			    head -c 1000 | gzip -d -c -
		

JSR-223 PostProcessor встраивает wget

Программное заполнение параметров SampleResult

JSR-223 PostProcessor встраивает wget

Сохраняется Response Body

JSR-223 PostProcessor встраивает wget

Сохраняется Response Headers

200 ГБайт с wget скачивались очень быстро

Параметр Значение
Среднее время 727 ms
Длительность 2 минуты 27 сек
Необходимая память 0 МБайт (минимум)

Скачать 200 ГБайт

Способ Длительность
HTTP Request 5 минут
☑️ Save response as MD5 Hash 18 минут 17 сек
OS Process Sampler и wget 2 минуты 27 сек

OS Process Sampler + JSR-223 PostProcessor
позволяют вызывать консольные инструменты,
которые хорошо работают с большими файлами.
Так wget даёт ускорение в 2 раза,
и в 7 раз по сравнению с MD5.

Попробуем ограничить читаемый размер до 100 байт

httpsampler.max_bytes_to_store_per_request=100

			# Max size of bytes stored in memory per SampleResult
			# Ensure you don't exceed max capacity of a Java Array and remember
			# that the higher it is, the higher JMeter will consume heap
			# Defaults to 0, which means no truncation
			#httpsampler.max_bytes_to_store_per_request=0
		

Быстрее с max_bytes_to_store_per_request=100

Параметр Значение
Среднее время 475 ms
Длительность 1 минута 36 сек
Необходимая память 180 МБайт (немного)

SJK: socketRead (99.35%)

JMeter читает, ни на что не отвлекается

Скачать 200 ГБайт

Способ Длительность
HTTP Request 5 минут
☑️ Save response as MD5 Hash 18 минут 17 сек
OS Process Sampler и wget 2 минуты 27 сек
max_bytes_to_store_per_request=100 1 минута 36 сек

HTTP Request
Отправка файлов

HTTP Request. Отправка файлов
Загрузка файлов быстрая ⏱️ и экономная
Пользуйтесь вкладкой Files Upload.

Комбинации вариантов передачи параметров

Вариант передачи Один С С С С
Parameters
Body Data
Files Upload (All)
Files Upload (File Path)

Комбинации вариантов передачи параметров

Parameters и Files Upload (parameters) дружат

Комбинации вариантов передачи параметров

Files Upload (body data) и Body Data сами по себе

HTTP POST один параметр

multipart/form-data

HTTP POST один параметр

multipart/form-data

HTTP POST несколько параметров

multipart/form-data

HTTP POST (raw body)

Загрузка тела запроса из файла

HTTP POST (raw body)

Нужно заполнить только File Path

Всё просто прекрасно,
но загружаемые файлы должны быть готовы.
Если нужна параметризация содержимого, то
чем заменить вкладку Body Data?

HTTP Request. Отправка файлов
Как готовить файлы на 10 ГБайт?
Во время теста! С минимальными затратами!

OS Process Sampler: tar, gzip, sed, ...

Подготовить архив, распаковать, заменить в нём строки

HTTP Request. Отправка файлов
Большие файлы лучше читать с диска.
Большие тела запросов лучше читать с диска.
Готовить большие файлы удобно с gzip, sed, ...
Пользуйтесь вкладкой Files Upload,
где также можно загружать и тело запроса.

PostProcessor-ы
Как отличается их скорость?

Тестовая страница: apache.jmeter.org

Найдём заголовок страницы

Boundary Extractor

для htmlContent

Regular Expresion Extractor

для htmlContent

JSR-223 PostProcessor

JSR-223 PostProcessor

aналогичный работе Boundary Extractor

BeanShell PostProcessor

BeanShell PostProcessor

Аналогичный работе Boundary Extractor

CSS Selector Extractor

для htmlContent

XPath Extractor

XPath (HTML) для htmlContent

Boundary (58k) самый быстрый

RegExp (50k) почти такой же быстрый

Общее влияние PostPorcessor-ов небольшое

Так CSS Selector выполняется дольше CSV Logger лишь в 2 раза

PostProcessor для HTML
Точно не использовать XPath для HTML.
Лидеры RegExp и Boundary Extractor.
Но и CSS Selector неплох — лишь x2 от логгера.

PreProcessor-ы
Как быстро подготовить XML?

Как быстро подготовить XML?

groovy.xml.MarkupBuilder x5 от SimpleTemplateEngine

		

PreProcessor-ы
В Groovy groovy.xml.MarkupBuilder быстрее
SimpleTemplateEngine, быстрее сериализации.
И экономит много времени.

Секретное оружие
Как быстро разрабатывать скрипты на JMeter?
Как быстро скачивать расширения и библиотеки?
Как запустить профилирование теста?
Как не терять изменения в скрипте?

Maven, jmeter-maven-plugin и IntelliJ IDEA

Ускорение работы в 10 раз, не меньше

IntelliJ IDEA

Подстветка синтаксиса, горячие клавиши, git

jmeter-maven-plugin

Плагины хранятся в git (не бинарники)

jmeter-maven-plugin

Библиотеки хранятся в git (не бинарники)

jmeter-maven-plugin

Зависимости скачиваются сами

Любые параметры модифицируются

Создадим файл на 1 ГБайт и скачаем его 200 раз

			dd  if=/dev/urandom of=/tmp/data/1g.img \
			    bs=1 count=0 seek=1G
			mvn jmeter:jmeter \
			    -P Samplers.HTTP.Request.File \ 
			    -DThreads=1 -DLoopCount=200 -DRequestCount=1 \ 
			    -DRequestPath=/tmp/data/1g.img
		

Профили комбинируются между собой

Выполним тестирование с большим размером Heap

			dd  if=/dev/urandom of=/tmp/data/1g.img \
			    bs=1 count=0 seek=1G
			mvn jmeter:jmeter \
			    -P Samplers.HTTP.Request.File,_.jvm.heap.big \ 
			    -DThreads=1 -DLoopCount=200 -DRequestCount=1 \ 
			    -DRequestPath=/tmp/data/1g.img
		

Профилирование SJK в одну строку

И через 10 минут готов отчёт

			# Запуск sjk-профайлера
			mvn exec:exec@sjk
			
			# Запуск профилируемого теста
			mvn jmeter:jmeter \
			    -P Samplers.HTTP.Request.File, \ 
			    -DThreads=1 -DLoopCount=200 -DRequestCount=1 \ 
			    -DRequestPath=/tmp/data/1g.img
		

Профилирование Java Fligth Recorder в одну строку

И через 10 минут готов отчёт

			# Запуск профилируемого теста
			mvn jmeter:jmeter \
			    -P Samplers.HTTP.Request.File,_.jvm.profiler.JFR \ 
			    -DThreads=1 -DLoopCount=200 -DRequestCount=1 \ 
			    -DRequestPath=/tmp/data/1g.img
		

Профилирование JVisual VM в одну строку

И через 10 минут готов отчёт

			# Запуск профилируемого теста
			mvn jmeter:jmeter \
			    -P Samplers.HTTP.Request.File,_.jvm.profiler.jvisualvm \ 
			    -DThreads=1 -DLoopCount=200 -DRequestCount=1 \ 
			    -DRequestPath=/tmp/data/1g.img
		

Проект: bit.ly/jmeter-bench

github.com/polarnik/Apache.JMeter.Benchmark.NG

		

Как ускорить Apache.JMeter

Скрипт.

Настройки Apache.JMeter.

Настройки операционной системы.

Maven, jmeter-maven-plugin, IntelliJ IDEA.

Вячеслав Смирнов

github.com/polarnik

Telegram: @qa_load

Спасибо!