Методика предназначена для проверки производительности и устойчивости системы Timetta при работе с целевым профилем нагрузки, характерным для облачной версии приложения. Тестирование направлено на:
| Параметр | Значение |
|---|---|
| Максимальное число зарегистрированных пользователей | 20 000 |
| Среда развертывания | Kubernetes-кластер с выделенными нодами (Node Selector / Node Affinity) |
| PostgreSQL | Отказоустойчивый кластер Postgres Pro на отдельной ВМ |
| Redis | Кластер на отдельной ВМ |
| RabbitMQ | Кластер на отдельной ВМ |
| Сервисы | Настроены с отказоустойчивостью и автоматическим failover |
В тестовом плане рекомендуется создать отдельные HTTP Request Sampler для каждого из ключевых запросов. Ниже приведены целевые эндпоинты и предполагаемое количество обращений в рамках тестового цикла:
| Endpoint | Метод | Примерное количество вызовов |
|---|---|---|
| /odata/TimeAllocations | POST | 348 |
| /odata/Projects(id)/UpdateResourcePlan | POST | 31 |
| /odata/GetClientProfile(clientType='web') | GET | 187 |
| /odata/GetEntityTypes | GET | 187 |
| /odata/GetNavigationItems | GET | 187 |
| /odata/Notifications? $apply=filter(read eq false)/aggregate(id with countdistinct as count) | GET | 222 |
| /odata/Notifications? $top=50& $orderby=created desc | GET | 198 |
| /odata/GetSession? $expand=configuration/metamodel/entities | GET | 196 |
| /odata/GetIndicators | GET | 193 |
| /odata/StartUndoRedo | POST | 44 |
| /odata/TimeSheetLines | POST | 108 |
| /odata/AbortUndoRedo | POST | 34 |
| /odata/ProjectTasks(id) | PATCH | 6 |
| /odata/ExtendedTimeSheets? $apply=filter((timeSheet/state/code eq 'Draft' and timeSheet/dueDate lt 01.12.2025))/aggregate($count as count) | GET | 55 |
| /odata/Projects(id)/UpdateResourcePlan | POST | 5 |
Примечание: числа вызовов ориентировочные, используются для моделирования нагрузки в JMeter Thread Group.
Thread Group (Группа потоков):
HTTP Request Defaults:
<base_url>HTTP Header Manager:
ApiToken <token>application/json (для POST/PATCH запросов)Timers:
Listeners:
Вот обновлённый пункт, с учётом твоих условий: 4 ноды Kubernetes и 1 реплика PostgreSQL, без HA для остальных компонентов, только для нагрузочного тестирования.
Для проведения нагрузочного тестирования Timetta выделяется минимальный набор ресурсов, достаточный для эмуляции целевого профиля нагрузки.
| Роль | Кол-во | vCPU (Intel Ice Lake, 100%) | RAM | SSD (IOPS) | Макс. bandwidth (чтение / запись) |
|---|---|---|---|---|---|
| Нода кластера K8s | 4 | 8 | 32 GB | 300 GB, ≥10k IOPS | 150 МБ/с |
| PostgreSQL Primary VM | 1 | 12 | 64 GB | 500 GB, ≥15k IOPS | 200 МБ/с |
| PostgreSQL Replica VM | 1 | 12 | 64 GB | 500 GB, ≥15k IOPS | 200 МБ/с |
| Redis VM | 1 | 4 | 8 GB | 50–100 GB SSD | 150 МБ/с |
| RabbitMQ VM | 1 | 4 | 8 GB | 200 GB, ≥10k IOPS | 150 МБ/с |
| Клиент JMeter (нагрузка) | 1 | 8 | 16 GB | 100 GB | 150 МБ/с |
<?xml version="1.0" encoding="UTF-8"?>
<jmeterTestPlan version="1.2" properties="5.0" jmeter="5.6">
<hashTree>
<TestPlan guiclass="TestPlanGui" testclass="TestPlan" testname="Timetta Load Test Plan" enabled="true">
<stringProp name="TestPlan.comments">Нагрузочный тест для Timetta с профилем нагрузки облачной версии</stringProp>
<boolProp name="TestPlan.functional_mode">false</boolProp>
<boolProp name="TestPlan.tearDown_on_shutdown">true</boolProp>
<elementProp name="TestPlan.user_defined_variables" elementType="Arguments">
<collectionProp name="Arguments.arguments">
<elementProp name="base_url" elementType="Argument">
<stringProp name="Argument.name">base_url</stringProp>
<stringProp name="Argument.value">https://your-timetta-domain.com</stringProp>
<stringProp name="Argument.metadata">=</stringProp>
</elementProp>
<elementProp name="api_token" elementType="Argument">
<stringProp name="Argument.name">api_token</stringProp>
<stringProp name="Argument.value">YOUR_API_TOKEN</stringProp>
<stringProp name="Argument.metadata">=</stringProp>
</elementProp>
</collectionProp>
</elementProp>
<stringProp name="TestPlan.user_define_classpath"></stringProp>
</TestPlan>
<hashTree>
<ThreadGroup guiclass="ThreadGroupGui" testclass="ThreadGroup" testname="Timetta Users Thread Group" enabled="true">
<stringProp name="ThreadGroup.on_sample_error">continue</stringProp>
<elementProp name="ThreadGroup.main_controller" elementType="LoopController" guiclass="LoopControlPanel" testclass="LoopController" testname="Loop Controller" enabled="true">
<boolProp name="LoopController.continue_forever">false</boolProp>
<stringProp name="LoopController.loops">1</stringProp>
</elementProp>
<stringProp name="ThreadGroup.num_threads">495</stringProp>
<stringProp name="ThreadGroup.ramp_time">300</stringProp>
<boolProp name="ThreadGroup.scheduler">false</boolProp>
</ThreadGroup>
<hashTree>
<ConfigTestElement guiclass="HttpDefaultsGui" testclass="ConfigTestElement" testname="HTTP Request Defaults" enabled="true">
<elementProp name="HTTPsampler.Arguments" elementType="Arguments">
<collectionProp name="Arguments.arguments"/>
</elementProp>
<stringProp name="HTTPSampler.domain">${base_url}</stringProp>
<stringProp name="HTTPSampler.port"></stringProp>
<stringProp name="HTTPSampler.protocol">https</stringProp>
<stringProp name="HTTPSampler.contentEncoding"></stringProp>
<stringProp name="HTTPSampler.path"></stringProp>
<stringProp name="HTTPSampler.concurrentPool">6</stringProp>
</ConfigTestElement>
<hashTree/>
<HeaderManager guiclass="HeaderPanel" testclass="HeaderManager" testname="HTTP Header Manager" enabled="true">
<collectionProp name="HeaderManager.headers">
<elementProp name="Authorization" elementType="Header">
<stringProp name="Header.name">Authorization</stringProp>
<stringProp name="Header.value">ApiToken ${api_token}</stringProp>
</elementProp>
<elementProp name="Content-Type" elementType="Header">
<stringProp name="Header.name">Content-Type</stringProp>
<stringProp name="Header.value">application/json</stringProp>
</elementProp>
</collectionProp>
</HeaderManager>
<hashTree/>
<!-- Пример HTTP Request Sampler -->
<HTTPSamplerProxy guiclass="HttpTestSampleGui" testclass="HTTPSamplerProxy" testname="/odata/TimeAllocations POST" enabled="true">
<elementProp name="HTTPsampler.Arguments" elementType="Arguments">
<collectionProp name="Arguments.arguments"/>
</elementProp>
<stringProp name="HTTPSampler.domain"></stringProp>
<stringProp name="HTTPSampler.port"></stringProp>
<stringProp name="HTTPSampler.protocol"></stringProp>
<stringProp name="HTTPSampler.path">/odata/TimeAllocations</stringProp>
<stringProp name="HTTPSampler.method">POST</stringProp>
<boolProp name="HTTPSampler.follow_redirects">true</boolProp>
<boolProp name="HTTPSampler.auto_redirects">false</boolProp>
<boolProp name="HTTPSampler.use_keepalive">true</boolProp>
<boolProp name="HTTPSampler.DO_MULTIPART_POST">false</boolProp>
<stringProp name="HTTPSampler.embedded_url_re"></stringProp>
</HTTPSamplerProxy>
<hashTree/>
<!-- Timer для имитации реального поведения пользователя -->
<ConstantTimer guiclass="ConstantTimerGui" testclass="ConstantTimer" testname="Think Time Timer" enabled="true">
<stringProp name="ConstantTimer.delay">500</stringProp>
</ConstantTimer>
<hashTree/>
<!-- Listener -->
<ResultCollector guiclass="ViewResultsFullVisualizer" testclass="ResultCollector" testname="View Results Tree" 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>true</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>
</ResultCollector>
<hashTree/>
</hashTree>
</hashTree>
</hashTree>
</jmeterTestPlan>
Перейти на русскую версию?