Учебное пособие по WebSocket «IM Series»: как использовать JMeter для стресс-тестирования
Учебное пособие по WebSocket «IM Series»: как использовать JMeter для стресс-тестирования

JMeter

JMeter в настоящее время является самым популярным инструментом тестирования производительности с открытым исходным кодом. Механизм на основе плагинов, предоставляемый самим JMeter, позволяет третьим сторонам реализовывать протоколы, которые не поддерживаются стандартным JMeter. Лучшей реализацией WebSocket является WebSocketSampler. С помощью этого плагина можно выполнить базовое тестирование производительности на основе протокола WebSocket.

Установить

Адрес загрузки: https://jmeter.apache.org/download_jmeter.cgi.

Windows может напрямую загрузить двоичный файл

Файлы для скачиваниякопировать/Перейти в локальный каталог。Я здесь прямокопироватьприезжатьDтарелка,D:\apache-jmeter-5.4.1

Установить пакет плагина Websocket

Адрес загрузки: https://bitbucket.org/pjtr/jmeter-websocket-samplers/downloads/

Файлы для скачиваниякопировать/двигатьсяприезжатьD:\apache-jmeter-5.4.1\lib\extв каталоге

Запустить Jmeter

Введите каталог исполняемого файлаD:\apache-jmeter-5.4.1\binНайти командный файлjmeter.bat

Первыйдобавить вгруппа потоковTinywan Thread Group,Нажмите еще раздобавить вsampler, вы можете увидеть информацию о веб-сокете

На приведенном выше снимке экрана показано, что установка плагина прошла успешно.

Напишите план тестирования

Здесь вы можете заполнить соответствующие параметры согласно предыдущему уроку.

Язык кода:javascript
копировать
var ws = new WebSocket("ws://127.0.0.1:8783");

План тестирования малого стека технологии с открытым исходным кодом.jmx

1. Подключитесь

Выберите протоколws/wss,входитьipпортпутьПараметры запроса(Если параметров нет, оставьте их пустыми.)

2.0 Присоединиться к групповому чату

JMeter

Язык кода:javascript
копировать
let $_content = {
  "event": "join",
  "mode": 2,
  "group_id": 100,
  "from_user_id": "10086",
  "from_username": «Аксу»,
  "to_user_id": "10000",
  "content": «Присоединиться к разговору»,
};
console.log(JSON.stringify($_content))

JSON.stringify() метод преобразует JavaScript Объект или значение, преобразованное в JSON нить.

Разберите приведенное выше в строку JSON.

Язык кода:javascript
копировать
{"event":"join","mode":2,"group_id":100,"from_user_id":"10086","from_username":«Аксу»,"to_user_id":"10000","content":"Присоединиться к беседе"}

3.0 Обсуждение

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

Язык кода:javascript
копировать
let content = {
      "event": "speak",
      "mode": 2,
      "group_id": 100,
      "from_user_id": "10086",
      "from_username": «Аксу»,
      "to_user_id": "10000",
      "content": "WebSocketУчебное пособие: Реализация и применение сохраняемости сообщений",
   }
console.log(JSON.stringify(content))

JSON.stringify() метод преобразует JavaScript Объект или значение, преобразованное в JSON нить.

Разберите приведенное выше в строку JSON.

Язык кода:javascript
копировать
{"event":"speak","mode":2,"group_id":100,"from_user_id":"10086","from_username":«Аксу»,"to_user_id":"10000","content":"WebSocketУчебное пособие: Реализация и применение сохраняемости сообщений"}

4. Сердцебиение

Создайте контроллер цикла

Добавить сэмплер WebSocket

Запросить контент

Язык кода:javascript
копировать
{"event":"ping","content":"ping heartbeat"}

5. Добавить дерево результатов просмотра

На рисунке выше показано дерево просмотра результатов выполнения. В правом окне вы можете увидеть результаты выборки, запросы и ответы. Запрос — это запрос, отправленный клиентом на сервер, а данные ответа — это возвращаемый результат. сервером после получения запроса. Вы можете выбрать различные методы просмотра результатов, включая json, html, xpath и т. д.

6. Добавьте агрегированные отчеты

Запустите JMeter

1.0 Подключение

Запросить данные

данные ответа

2.0 Обсуждение

Запросить данные

данные ответа

Совокупные отчеты, чтобы увидеть производительность

  • Выборки: общее количество выборок, равное общему количеству потоков * количеству циклов.
  • Среднее: Среднее время обработки запроса (миллисекунды), которое является одним из основных показателей стресс-теста.
  • Медиана: медианное время обработки запроса (мс). Половина времени обработки по количеству выборок превышает это значение, а половина времени обработки ниже этого значения.
  • 90%Line,95%Line,99%Line: какой процент времени обработки в выборке находится ниже этого значения.,Является одним из основных показателей стресс-теста.
  • Мин. Время запроса, которое занимает наименьшее количество времени.
  • Макс: Запрос, который занимает больше всего времени.
  • Error%: процент ошибок.
  • Пропускная способность: Пропускная способность, количество запросов, которые сервер обрабатывает в секунду.
  • КБ/сек: количество байтов, запрашиваемых сервером в секунду.

другой

Скрипт стресс-теста JMeter

Скачать адрес:https://github.com/Tinywan/webman-admin/blob/main/db/План тестирования малого стека технологии с открытым исходным кодом.jmx

План тестирования малого стека технологии с открытым исходным кодом.jmx

Язык кода:javascript
копировать
<?xml version="1.0" encoding="UTF-8"?>
<jmeterTestPlan version="1.2" properties="5.0" jmeter="5.4.1">
  <hashTree>
    <TestPlan guiclass="TestPlanGui" testclass="TestPlan" testname="Стек технологий с открытым исходным кодом План тестирования" enabled="true">
      <stringProp name="TestPlan.comments"></stringProp>
      <boolProp name="TestPlan.functional_mode">false</boolProp>
      <boolProp name="TestPlan.serialize_threadgroups">true</boolProp>
      <elementProp name="TestPlan.user_defined_variables" elementType="Arguments" guiclass="ArgumentsPanel" testclass="Arguments" testname="User Defined Variables" enabled="true">
        <collectionProp name="Arguments.arguments">
          <elementProp name="username" elementType="Argument">
            <stringProp name="Argument.name">username</stringProp>
            <stringProp name="Argument.value">Tinywan</stringProp>
            <stringProp name="Argument.metadata">=</stringProp>
          </elementProp>
          <elementProp name="password" elementType="Argument">
            <stringProp name="Argument.name">password</stringProp>
            <stringProp name="Argument.value">xxxx</stringProp>
            <stringProp name="Argument.metadata">=</stringProp>
          </elementProp>
        </collectionProp>
      </elementProp>
      <stringProp name="TestPlan.user_define_classpath"></stringProp>
    </TestPlan>
    <hashTree>
      <ThreadGroup guiclass="ThreadGroupGui" testclass="ThreadGroup" testname="Im Websocket Испытание давлением" enabled="true">
        <stringProp name="ThreadGroup.on_sample_error">continue</stringProp>
        <elementProp name="ThreadGroup.main_controller" elementType="LoopController" guiclass="LoopControlPanel" testclass="LoopController" testname="Контроллер цикла" enabled="true">
          <boolProp name="LoopController.continue_forever">false</boolProp>
          <stringProp name="LoopController.loops">500</stringProp>
        </elementProp>
        <stringProp name="ThreadGroup.num_threads">1</stringProp>
        <stringProp name="ThreadGroup.ramp_time">1</stringProp>
        <boolProp name="ThreadGroup.scheduler">false</boolProp>
        <stringProp name="ThreadGroup.duration"></stringProp>
        <stringProp name="ThreadGroup.delay"></stringProp>
        <boolProp name="ThreadGroup.same_user_on_next_iteration">true</boolProp>
      </ThreadGroup>
      <hashTree>
        <CriticalSectionController guiclass="CriticalSectionControllerGui" testclass="CriticalSectionController" testname="контроллер критической секции" enabled="false">
          <stringProp name="CriticalSectionController.lockName">global_lock</stringProp>
        </CriticalSectionController>
        <hashTree/>
        <eu.luminis.jmeter.wssampler.OpenWebSocketSampler guiclass="eu.luminis.jmeter.wssampler.OpenWebSocketSamplerGui" testclass="eu.luminis.jmeter.wssampler.OpenWebSocketSampler" testname="1.0 Подключение" enabled="true">
          <boolProp name="TLS">false</boolProp>
          <stringProp name="server">127.0.0.1</stringProp>
          <stringProp name="port">8783</stringProp>
          <stringProp name="path">/?sign=ca7a2df4c9850239ded1974f5abe8fc7&amp;ws_timestamp=1636079196</stringProp>
          <stringProp name="connectTimeout">20000</stringProp>
          <stringProp name="readTimeout">6000</stringProp>
        </eu.luminis.jmeter.wssampler.OpenWebSocketSampler>
        <hashTree/>
        <eu.luminis.jmeter.wssampler.RequestResponseWebSocketSampler guiclass="eu.luminis.jmeter.wssampler.RequestResponseWebSocketSamplerGui" testclass="eu.luminis.jmeter.wssampler.RequestResponseWebSocketSampler" testname="2.0 Присоединиться к групповому чату" enabled="true">
          <boolProp name="createNewConnection">false</boolProp>
          <boolProp name="TLS">false</boolProp>
          <stringProp name="server"></stringProp>
          <stringProp name="port">80</stringProp>
          <stringProp name="path"></stringProp>
          <stringProp name="connectTimeout">20000</stringProp>
          <boolProp name="binaryPayload">false</boolProp>
          <stringProp name="requestData">{&quot;event&quot;:&quot;join&quot;,&quot;mode&quot;:2,&quot;group_id&quot;:100,&quot;from_user_id&quot;:&quot;10086&quot;,&quot;from_username&quot;:&quot;Аксу&quot;,&quot;to_user_id&quot;:&quot;10000&quot;,&quot;content&quot;:&quot;Присоединиться к разговору&quot;}</stringProp>
          <stringProp name="readTimeout">6000</stringProp>
          <boolProp name="loadDataFromFile">false</boolProp>
          <stringProp name="dataFile"></stringProp>
        </eu.luminis.jmeter.wssampler.RequestResponseWebSocketSampler>
        <hashTree/>
        <eu.luminis.jmeter.wssampler.RequestResponseWebSocketSampler guiclass="eu.luminis.jmeter.wssampler.RequestResponseWebSocketSamplerGui" testclass="eu.luminis.jmeter.wssampler.RequestResponseWebSocketSampler" testname="3.0 Обсуждение" enabled="true">
          <boolProp name="createNewConnection">false</boolProp>
          <boolProp name="TLS">false</boolProp>
          <stringProp name="server"></stringProp>
          <stringProp name="port">80</stringProp>
          <stringProp name="path"></stringProp>
          <stringProp name="connectTimeout">20000</stringProp>
          <boolProp name="binaryPayload">false</boolProp>
          <stringProp name="requestData">{&quot;event&quot;:&quot;speak&quot;,&quot;mode&quot;:2,&quot;group_id&quot;:100,&quot;from_user_id&quot;:&quot;10086&quot;,&quot;from_username&quot;:&quot;Аксу&quot;,&quot;to_user_id&quot;:&quot;10000&quot;,&quot;content&quot;:&quot;WebSocketУчебное пособие:Реализация и применение сохранения сообщений&quot;}</stringProp>
          <stringProp name="readTimeout">6000</stringProp>
          <boolProp name="loadDataFromFile">false</boolProp>
          <stringProp name="dataFile"></stringProp>
        </eu.luminis.jmeter.wssampler.RequestResponseWebSocketSampler>
        <hashTree/>
        <ResultCollector guiclass="ViewResultsFullVisualizer" testclass="ResultCollector" testname="Просмотреть дерево результатов" enabled="true">
          <boolProp name="ResultCollector.error_logging">false</boolProp>
          <objProp>
            <name>saveConfig</name>
            <value class="SampleSaveConfiguration">
              <time>true</time>
              <latency>true</latency>
              <timestamp>true</timestamp>
              <success>true</success>
              <label>true</label>
              <code>true</code>
              <message>true</message>
              <threadName>true</threadName>
              <dataType>true</dataType>
              <encoding>false</encoding>
              <assertions>true</assertions>
              <subresults>true</subresults>
              <responseData>false</responseData>
              <samplerData>false</samplerData>
              <xml>false</xml>
              <fieldNames>true</fieldNames>
              <responseHeaders>false</responseHeaders>
              <requestHeaders>false</requestHeaders>
              <responseDataOnError>false</responseDataOnError>
              <saveAssertionResultsFailureMessage>true</saveAssertionResultsFailureMessage>
              <assertionsResultsToSave>0</assertionsResultsToSave>
              <bytes>true</bytes>
              <sentBytes>true</sentBytes>
              <url>true</url>
              <threadCounts>true</threadCounts>
              <idleTime>true</idleTime>
              <connectTime>true</connectTime>
            </value>
          </objProp>
          <stringProp name="filename"></stringProp>
        </ResultCollector>
        <hashTree/>
        <ResultCollector guiclass="SummaryReport" testclass="ResultCollector" testname="Сводный отчет" enabled="true">
          <boolProp name="ResultCollector.error_logging">false</boolProp>
          <objProp>
            <name>saveConfig</name>
            <value class="SampleSaveConfiguration">
              <time>true</time>
              <latency>true</latency>
              <timestamp>true</timestamp>
              <success>true</success>
              <label>true</label>
              <code>true</code>
              <message>true</message>
              <threadName>true</threadName>
              <dataType>true</dataType>
              <encoding>false</encoding>
              <assertions>true</assertions>
              <subresults>true</subresults>
              <responseData>false</responseData>
              <samplerData>false</samplerData>
              <xml>false</xml>
              <fieldNames>true</fieldNames>
              <responseHeaders>false</responseHeaders>
              <requestHeaders>false</requestHeaders>
              <responseDataOnError>false</responseDataOnError>
              <saveAssertionResultsFailureMessage>true</saveAssertionResultsFailureMessage>
              <assertionsResultsToSave>0</assertionsResultsToSave>
              <bytes>true</bytes>
              <sentBytes>true</sentBytes>
              <url>true</url>
              <threadCounts>true</threadCounts>
              <idleTime>true</idleTime>
              <connectTime>true</connectTime>
            </value>
          </objProp>
          <stringProp name="filename"></stringProp>
        </ResultCollector>
        <hashTree/>
        <ResultCollector guiclass="StatVisualizer" testclass="ResultCollector" testname="Агрегационный отчет" enabled="true">
          <boolProp name="ResultCollector.error_logging">false</boolProp>
          <objProp>
            <name>saveConfig</name>
            <value class="SampleSaveConfiguration">
              <time>true</time>
              <latency>true</latency>
              <timestamp>true</timestamp>
              <success>true</success>
              <label>true</label>
              <code>true</code>
              <message>true</message>
              <threadName>true</threadName>
              <dataType>true</dataType>
              <encoding>false</encoding>
              <assertions>true</assertions>
              <subresults>true</subresults>
              <responseData>false</responseData>
              <samplerData>false</samplerData>
              <xml>false</xml>
              <fieldNames>true</fieldNames>
              <responseHeaders>false</responseHeaders>
              <requestHeaders>false</requestHeaders>
              <responseDataOnError>false</responseDataOnError>
              <saveAssertionResultsFailureMessage>true</saveAssertionResultsFailureMessage>
              <assertionsResultsToSave>0</assertionsResultsToSave>
              <bytes>true</bytes>
              <sentBytes>true</sentBytes>
              <url>true</url>
              <threadCounts>true</threadCounts>
              <idleTime>true</idleTime>
              <connectTime>true</connectTime>
            </value>
          </objProp>
          <stringProp name="filename"></stringProp>
        </ResultCollector>
        <hashTree/>
        <LoopController guiclass="LoopControlPanel" testclass="LoopController" testname="Контроллер цикла" enabled="true">
          <boolProp name="LoopController.continue_forever">true</boolProp>
          <stringProp name="LoopController.loops">500</stringProp>
        </LoopController>
        <hashTree>
          <ConstantTimer guiclass="ConstantTimerGui" testclass="ConstantTimer" testname="Фиксированный таймер" enabled="true">
            <stringProp name="ConstantTimer.delay">10000</stringProp>
          </ConstantTimer>
          <hashTree/>
          <eu.luminis.jmeter.wssampler.RequestResponseWebSocketSampler guiclass="eu.luminis.jmeter.wssampler.RequestResponseWebSocketSamplerGui" testclass="eu.luminis.jmeter.wssampler.RequestResponseWebSocketSampler" testname="ws-сердцебиение Обнаружение" enabled="true">
            <boolProp name="createNewConnection">false</boolProp>
            <boolProp name="TLS">false</boolProp>
            <stringProp name="server"></stringProp>
            <stringProp name="port">80</stringProp>
            <stringProp name="path"></stringProp>
            <stringProp name="connectTimeout">20000</stringProp>
            <boolProp name="binaryPayload">false</boolProp>
            <stringProp name="requestData">{&quot;event&quot;:&quot;ping&quot;,&quot;content&quot;:&quot;ping heartbeat&quot;}</stringProp>
            <stringProp name="readTimeout">6000</stringProp>
            <boolProp name="loadDataFromFile">false</boolProp>
            <stringProp name="dataFile"></stringProp>
          </eu.luminis.jmeter.wssampler.RequestResponseWebSocketSampler>
          <hashTree/>
        </hashTree>
      </hashTree>
    </hashTree>
  </hashTree>
</jmeterTestPlan>
boy illustration
Неразрушающее увеличение изображений одним щелчком мыши, чтобы сделать их более четкими артефактами искусственного интеллекта, включая руководства по установке и использованию.
boy illustration
Копикодер: этот инструмент отлично работает с Cursor, Bolt и V0! Предоставьте более качественные подсказки для разработки интерфейса (создание навигационного веб-сайта с использованием искусственного интеллекта).
boy illustration
Новый бесплатный RooCline превосходит Cline v3.1? ! Быстрее, умнее и лучше вилка Cline! (Независимое программирование AI, порог 0)
boy illustration
Разработав более 10 проектов с помощью Cursor, я собрал 10 примеров и 60 подсказок.
boy illustration
Я потратил 72 часа на изучение курсорных агентов, и вот неоспоримые факты, которыми я должен поделиться!
boy illustration
Идеальная интеграция Cursor и DeepSeek API
boy illustration
DeepSeek V3 снижает затраты на обучение больших моделей
boy illustration
Артефакт, увеличивающий количество очков: на основе улучшения характеристик препятствия малым целям Yolov8 (SEAM, MultiSEAM).
boy illustration
DeepSeek V3 раскручивался уже три дня. Сегодня я попробовал самопровозглашенную модель «ChatGPT».
boy illustration
Open Devin — инженер-программист искусственного интеллекта с открытым исходным кодом, который меньше программирует и больше создает.
boy illustration
Эксклюзивное оригинальное улучшение YOLOv8: собственная разработка SPPF | SPPF сочетается с воспринимаемой большой сверткой ядра UniRepLK, а свертка с большим ядром + без расширения улучшает восприимчивое поле
boy illustration
Популярное и подробное объяснение DeepSeek-V3: от его появления до преимуществ и сравнения с GPT-4o.
boy illustration
9 основных словесных инструкций по доработке академических работ с помощью ChatGPT, эффективных и практичных, которые стоит собрать
boy illustration
Вызовите deepseek в vscode для реализации программирования с помощью искусственного интеллекта.
boy illustration
Познакомьтесь с принципами сверточных нейронных сетей (CNN) в одной статье (суперподробно)
boy illustration
50,3 тыс. звезд! Immich: автономное решение для резервного копирования фотографий и видео, которое экономит деньги и избавляет от беспокойства.
boy illustration
Cloud Native|Практика: установка Dashbaord для K8s, графика неплохая
boy illustration
Краткий обзор статьи — использование синтетических данных при обучении больших моделей и оптимизации производительности
boy illustration
MiniPerplx: новая поисковая система искусственного интеллекта с открытым исходным кодом, спонсируемая xAI и Vercel.
boy illustration
Конструкция сервиса Synology Drive сочетает проникновение в интрасеть и синхронизацию папок заметок Obsidian в облаке.
boy illustration
Центр конфигурации————Накос
boy illustration
Начинаем с нуля при разработке в облаке Copilot: начать разработку с минимальным использованием кода стало проще
boy illustration
[Серия Docker] Docker создает мультиплатформенные образы: практика архитектуры Arm64
boy illustration
Обновление новых возможностей coze | Я использовал coze для создания апплета помощника по исправлению домашних заданий по математике
boy illustration
Советы по развертыванию Nginx: практическое создание статических веб-сайтов на облачных серверах
boy illustration
Feiniu fnos использует Docker для развертывания личного блокнота Notepad
boy illustration
Сверточная нейронная сеть VGG реализует классификацию изображений Cifar10 — практический опыт Pytorch
boy illustration
Начало работы с EdgeonePages — новым недорогим решением для хостинга веб-сайтов
boy illustration
[Зона легкого облачного игрового сервера] Управление игровыми архивами
boy illustration
Развертывание SpringCloud-проекта на базе Docker и Docker-Compose