/quarkus-startstop

Generates, starts, tests, stops small Quarkus applications and measures time and memory

Primary LanguageJavaApache License 2.0Apache-2.0

Quarkus start-stop

Generates, starts, tests, stops small Quarkus applications and measures time and memory

Prerequisites

The TS expects you run Apache Maven 3.6.3+, Java 11+ and have ps program available on your Linux/Mac and wmic (by default present) on your Windows system. Native image build requires GraalVM with Native image toolchain installed. Refer to Building Native Image Guide for details.

Branches

Branches are created for important Quarkus versions, usually related to RHBQ releases.

When creating new branch please consult New branch adjustments document.

Usage

Run with a community version without native images:

mvn clean verify -Ptestsuite-community-no-native -Dquarkus.version=1.4.2.Final

Run with a community version, including native images:

mvn clean verify -Ptestsuite-community -Dquarkus.version=1.4.2.Final

All tests, including Quarkus product builds and native images:

mvn clean verify -Ptestsuite \
 -Dquarkus.version=1.3.0.Final-redhat-00019 \
 -Dquarkus.platform.version=1.3.0.Final-redhat-00010 \
 -Dtests.maven.repo.local=/home/karm/QUARKUS/quarkus-1.3.0.ER11/maven-repository \
 -Dmaven.repo.local=/home/karm/QUARKUS/quarkus-1.3.0.ER11/maven-repository

One can fine-tune excluded test cases or tests with excludeTags, e.g. -DexcludeTags=startstop.

Linux/Mac:

mvn clean verify -Ptestsuite

Windows:

mvn clean verify -Ptestsuite-no-native

Native compilation is not yet supported on Windows. You may also want to disable native tests with -Ptestsuite-no-native if you need just a quick check on the JVM mode.

StartStopTest

The goal is to build and start applications with some real source code that actually exercises some rudimentary business logic of selected extensions.

start-stop.iterations property - adjustment of the number of the start-stop cycles

  • append for example -Dstart-stop.iterations=25 to the mvn command

start-stop.cold-start property - use cold start mode to drop OS page cache entries, dentries and inodes.

  • append for example -Dstart-stop.cold-start to the mvn command

Collect results:

cat  testsuite/target/archived-logs/io.quarkus.ts.startstop.StartStopTest/measurements.csv

e.g. on Windows:

λ type testsuite\target\archived-logs\io.quarkus.ts.startstop.StartStopTest\measurements.csv
App,Mode,buildTimeMs,timeToFirstOKRequestMs,startedInMs,stoppedInMs,RSSKb,FDs
FULL_MICROPROFILE,JVM,9391,2162,1480,54,3820,78
JAKARTA_REST_MINIMAL,JVM,6594,1645,949,34,3824,78

and on Linux:

$ cat ./testsuite/target/archived-logs/io.quarkus.ts.startstop.StartStopTest/measurements.csv
App,Mode,buildTimeMs,timeToFirstOKRequestMs,startedInMs,stoppedInMs,RSSKb,FDs
FULL_MICROPROFILE,JVM,9117,1439,1160,18,179932,307
FULL_MICROPROFILE,NATIVE,142680,22,17,1,51592,129
JAKARTA_REST_MINIMAL,JVM,5934,1020,745,22,141996,162
JAKARTA_REST_MINIMAL,NATIVE,93943,10,7,3,29768,74

ArtifactGeneratorTest

The goal of this test is to test Quarkus maven artifact generator, i.e. to to use it to generate an empty (Hello World) skeleton and build it in Quarkus dev mode. The objective is to make sure all required extensions are correctly found.

Next, the project is run in dev mode, time to the first O.K. request is measured. Then, an existing .java file is changed and a new one is added, which causes hot reload. After that, a request to the modified file is issued, and the time it took to get the expected results is measured. Also, separate request to the added file is issued to verify that the new resource has been loaded. Response time is not measured in this case.

The whole run is executed as a warm-up to download the Internet and then again to measure the times. The properties for thresholds are stored in app-generated-skeleton/threshold.properties.

Build and run logs are archived and checked for errors, see:

**/io.quarkus.ts.startstop.ArtifactGeneratorTest/manyExtensions/dev-run.log
**/io.quarkus.ts.startstop.ArtifactGeneratorTest/manyExtensions/artifact-build.log
**/io.quarkus.ts.startstop.ArtifactGeneratorTest/manyExtensions/warmup-artifact-build.log
**/io.quarkus.ts.startstop.ArtifactGeneratorTest/manyExtensions/warmup-dev-run.log

Measurements example, e.g. Windows and OpenJDK 11 J9:

λ type testsuite\target\archived-logs\io.quarkus.ts.startstop.ArtifactGeneratorTest\measurements.csv
App,Mode,buildTimeMs,timeToFirstOKRequestMs,timeToReloadMs,startedInMs,stoppedInMs,RSSKb,FDs
GENERATED_SKELETON,GENERATOR,3766,37064,8859,18249,1172,4240,81

e.g. it took 3.766s to generate the skeleton project, it took 37.064s to build and start the Dev mode and it took 8.859s to do the live reload and get the expected response to a request.

Linux and OpenJDK 11 HotSpot:

App,Mode,buildTimeMs,timeToFirstOKRequestMs,timeToReloadMs,startedInMs,stoppedInMs,RSSKb,FDs
GENERATED_SKELETON,GENERATOR,2644,13871,3091,5597,1154,565340,198

See ArtifactGeneratorTest#manyExtensions for the list of used extensions.

SpecialCharsTest

The goal of this test is to check the build of Quarkus applications on paths with special characters - japanese characters, spaces, diacritics, etc.

Code Quarkus test

Add -Dcode.quarkus.url= to test against a selected Code Quarkus site.

See README.md for details about CodeQuarkusExtensions.java.

NativeDebugTest

The goal of this test is to check the presence of the Quarkus debug symbols file and source cache directory after executing with -Dquarkus.native.debug.enabled=true flag. For more information please refer to:
https://quarkus.io/guides/building-native-image#debugging-native-executable

Parameters for NATIVE mode

Properties starting with quarkus.native get appended to the command for the native image build. This allows customization of the native image build procedure as described in https://quarkus.io/guides/building-native-image#configuration-reference guide.

Example command:

mvn clean verify -Ptestsuite -Dtest=SpecialCharsTest#diacriticsNative \
    -Dquarkus.version=1.6.0.Final -Dquarkus.platform.version=1.6.0.Final \
    -Dquarkus.native.container-build=true -Dquarkus.native.container-runtime=docker\
    -Dquarkus.native.builder-image=quay.io/quarkus/ubi-quarkus-mandrel-builder-image:22.3-java17

Values

  • App - the test app used
  • Mode - one in DEV, NATIVE, JVM
  • buildTimeMs - how long it tool the build process to terminate
  • timeToFirstOKRequestMs - how long it took since the process was started till the first request got a valid response
  • timeToReloadMs - how long it took to get a valid response after dev mode reload
  • startedInMs - "started in" value reported in Quarkus log
  • stoppedInMs - "stopped in" value reported in Quarkus log
  • RSSkB - memory used in kB; not comparable between Linux and Windows, see below
  • FDs - file descriptors held by the process; Windows is lower due to static linking of JVM libs etc.

Logs and Whitelist

Both build logs and runtime logs are checked for error messages. Expected error messages can be whitelisted in Whitelist.java.

To examine logs yourself see ./testsuite/target/archived-logs/ , e.g.

./testsuite/target/archived-logs/io.quarkus.ts.startstop.StartStopTest/jakartaRESTMinimalNative/native-build.log
./testsuite/target/archived-logs/io.quarkus.ts.startstop.StartStopTest/jakartaRESTMinimalNative/native-run.log
./testsuite/target/archived-logs/io.quarkus.ts.startstop.StartStopTest/fullMicroProfileJVM/jvm-run.log
./testsuite/target/archived-logs/io.quarkus.ts.startstop.StartStopTest/fullMicroProfileJVM/jvm-build.log
./testsuite/target/archived-logs/io.quarkus.ts.startstop.StartStopTest/jakartaRESTMinimalJVM/jvm-run.log
./testsuite/target/archived-logs/io.quarkus.ts.startstop.StartStopTest/jakartaRESTMinimalJVM/jvm-build.log
./testsuite/target/archived-logs/io.quarkus.ts.startstop.StartStopTest/fullMicroProfileNative/native-build.log
./testsuite/target/archived-logs/io.quarkus.ts.startstop.StartStopTest/fullMicroProfileNative/native-run.log

What I Did Report

The test suite records Maven commands it used, the directories where those commands were executed etc. in a neat markdown file for each test run. e.g.

# io.quarkus.ts.startstop.StartStopTest, jakartaRESTMinimalJVM
/home/karm/workspaceRH/quarkus-startstop/app-jakarta-rest-minimal

mvn clean compile quarkus:build -Dquarkus.package.output-name=quarkus -Dmaven.repo.local=/home/karm/QUARKUS/quarkus-1.3.2.CR1/maven-repository

---
/home/karm/workspaceRH/quarkus-startstop/app-jakarta-rest-minimal

java -jar target/quarkus-runner.jar

---
Measurements:

|App|Mode|buildTimeMs|timeToFirstOKRequestMs|startedInMs|stoppedInMs|RSSKb|FDs|
| --- | --- | --- | --- | --- | --- | --- | --- |
|JAKARTA_REST_MINIMAL|JVM|4787|906|643|18|140504|162|

The file format is somewhat loose and it differs a bit test to test, it is meant for humans to take a quick look at what the TS did. If you need to machine process the data, we suggest:

  • capturing the TS stdout log where all commands are logged in a machine friendly format
  • reading archived measurements.csv files

One can also look into ./testsuite/target/archived-logs/aggregated-report.md for an overall concatenation of all reports from all test runs. Subsequent test suite executions without mvn clean keep appending to this aggregate file on purpose.

Thresholds

The test suite works with threshold.properties for each test app. E.g. app-jakarta-rest-minimal/threshold.properties:

linux.jvm.time.to.first.ok.request.threshold.ms=1500
linux.jvm.RSS.threshold.kB=170000
linux.native.time.to.first.ok.request.threshold.ms=35
linux.native.RSS.threshold.kB=75000
windows.jvm.time.to.first.ok.request.threshold.ms=2000
windows.jvm.RSS.threshold.kB=4000

The measured values are simply compared to be less or equal to the set threshold. One can overwrite the threshold properties by using env variables or system properties (in this order). All letter are capitalized and dot is replaced with underscore, e.g.

APP_JAKARTA_REST_MINIMAL_LINUX_JVM_TIME_TO_FIRST_OK_REQUEST_THRESHOLD_MS=500 mvn clean verify -Ptestsuite

Results in:

[INFO] Results:
[INFO] 
[ERROR] Failures: 
[ERROR]   StartStopTest.jakartaRESTMinimalJVM:137->testRuntime:121 Application JAKARTA_REST_MINIMAL 
          in JVM mode took 957 ms to get the first OK request, which is over 500 ms threshold. 
          ==> expected: <true> but was: <false>
[INFO] 

Debugging

To debug particular test in IDEA, open Maven tab, select module and lifecycle phase e.x. (module: testsuite, phase: verify), right click to create a new configuration. Parameters Tab -> Command line, specify full Maven command you are using to run the test and add -DforkCount=0 Example: verify -Ptestsuite -Dtest=CodeQuarkusTest#supportedExtensionsSubsetC -DforkCount=0 -f pom.xml Set the breakpoint where needed and click on green bug symbol on top bar of IDEA (make sure a new configuration is selected).

Troubleshooting

To help to troubleshoot the issues, some performance insights from the application are needed. One of the best way to gather performance insights is to generate CPU and allocation FlameGraphs using Async Profiler.

Very good starting point is https://github.com/quarkusio/quarkus/blob/main/TROUBLESHOOTING.md document. Please follow the instructions from Installing Async Profiler section of the guide.

On TS side perform modification similar to this change:

diff --git a/testsuite/src/it/java/io/quarkus/ts/startstop/utils/MvnCmds.java b/testsuite/src/it/java/io/quarkus/ts/startstop/utils/MvnCmds.java
index 0c02537..6278605 100755
--- a/testsuite/src/it/java/io/quarkus/ts/startstop/utils/MvnCmds.java
+++ b/testsuite/src/it/java/io/quarkus/ts/startstop/utils/MvnCmds.java
@@ -33,7 +33,9 @@ import static io.quarkus.ts.startstop.utils.Commands.getQuarkusVersion;
 public enum MvnCmds {
     JVM(new String[][]{
             new String[]{"mvn", "clean", "compile", "quarkus:build", "-Dquarkus.package.output-name=quarkus"},
-            new String[]{"java", "-jar", "target/quarkus-runner.jar"}
+            new String[]{"java",
+                    "-agentpath:/var/lib/jenkins/async-profiler-1.8.1-linux-x64/build/libasyncProfiler.so=start,event=cpu,file=/tmp/startup-cpu-profile.svg,interval=1000000,width=1600,simple",
+                    "-jar", "target/quarkus-runner.jar"}
     }),
     DEV(new String[][]{
             new String[]{"mvn", "clean", "quarkus:dev", "-Dmaven.repo.local=" + getLocalMavenRepoDir()}

Similar change is needed for alloc instead of cpu insights.

In case of StartStopTest, you should consider reducing the number of iterations performed for start command in StartStopTest.java#L121

Windows notes

Note on Native image

Not yet supported on Windows.

Stopping Quarkus

Works well, see details: See README.md

Parsing Quarkus logs

Works well, see caveats in Logs.java, e.g. control characters such as: stopped in �[38;5;188m0.024�[39ms�[39m�[38;5;203m�[39m�[38;5;227m

Memory consumption, RSS

We use "Working Set Size", see win32-process, to measure the memory used. It is not calculated the same way as ps does it on Linux and one cannot compare it directly to the Linux RSS.

Note on Dev Mode

Not yet supported in the TS.