Assert JDK flight recorder events using JfrUnit

Open source software engineers at Gunnar Morling Red Hat introduced JfrUnit, a new testing utility that can be used to detect performance regression from JUnit or Spock frameworks. Interpreting performance test results, such as response times, can be difficult because there may be regressions caused by other factors, such as other processes or networks, rather than the application itself. JfrUnit can be used to test the performance of an application by measuring memory allocation, IO, database queries, or other application-specific elements.

The JDK Flight Recorder (JFR) collects events from running applications that can be used to diagnose or analyze the application. These events can be almost anything, from memory allocation to garbage collection. You can use this tool directly from the command line, but it is often used in conjunction with JDK Mission Control, which provides a GUI and various plug-ins that can be used with JFR. JfrUnit can create assertions to validate JFR events from applications.

JfrUnit supports OpenJDK 16 and dependencies are available on Maven Central:

< dependency > <groupId>org.moditect.jfrunit</groupId> <artifactId> jfrUnit </artifactId> <version>1.0.0.Alpha1</version> < scope > test </ scope > </ dependency >Copy the code

Implementing JUnit tests starts by adding annotations to the ** @jFreventTest ** unit test class, unless the test automatically interacts with the JFR record using ** @Quarkustest ** annotations as the Quarkus test framework. The test uses ** @enableEvent ** to collect annotations for specific events, such as garbage collection events. After executing the program logic, **jfrEvents. AwaitEvents ()** The method waits for any JFR events from the JVM or application before using assertions to verify that the event has occurred:

@jfreventTest Public class GarbageCollectionTest {public JfrEvents JfrEvents = new JfrEvents (); @ @ EnableEvent test (" JDK. GarbageCollection ") public void testGarbageCollectionEvent () throws an exception {system. GC (); JfrEvents. Waiting for events (); AssertThat (jfrEvents). Contains (event (" jdk.garbagecollection ")); }}Copy the code

Alternatively, you can write the same test using the Spock framework:

Class GarbageCollectionSpec extension specification {JfrEvents JfrEvents = new JfrEvents () @enableEvent (' jdk.garbagecollection ') def 'contains garbage collection Jfr event' () {when: system. GC () then: jfrEvents [' jdk.garbagecollection ']}}Copy the code

In addition to verifying whether an event occurred, you can also verify details about the event, such as the duration of the event ** thread.sleep ()** method:

@测试 
@EnableEvent ( "jdk.ThreadSleep" ) 
public  void  testThreadSleepEvent () 抛出  异常 { 
    线程 。 睡眠 ( 42 ); 

    jfrEvents 。 等待事件 (); 

    assertThat ( jfrEvents ) 
. 包含 ( 事件 ( “jdk.ThreadSleep” ) 
. with ( "time" , Duration . of Millis ( 42 ))); 
} 
Copy the code

JfrUnit allows you to create more complex scenarios. Consider the following example, which collects memory allocation events and sums them before asserting that memory allocation is between a specific value:

@ @ EnableEvent test (" JDK. ObjectAllocationInNewTLAB ") @ EnableEvent (" JDK. ObjectAllocationOutsideTLAB ") public void TestAllocationEvent () throws an exception {string name = thread thread. Current thread (). Get the name (); // Create the application logic for the object jfrEvents. Waiting for events (); Long sum = jfrEvents. Filter (this :: isObjectAllocationEvent). Filter (Event - > Event. GetThread. GetJavaName (). () equals (threadName)).maptolong (this :: getAllocationSize). The sum (); Assertion (summation). isLessThan ( 43_000_000 ); Assertion (summation). isGreaterThan ( 42_000_000 ); } Private Boolean isObjectAllocationEvent (RecordedEvent re) {string name = re. Gets the event type (). Get the name (); Returns the name. Is equal to (" JDK. ObjectAllocationInNewTLAB ") | | name. Equal to (" JDK. ObjectAllocationOutsideTLAB "); } Private long getAllocationSize (RecordedEvent RecordedEvent) {return record events. Gets the event type (). Get names (.) is equal to (" JDK. ObjectAllocationInNewTLAB ")? Log events. GetLong ("tlabSize") : Logs events. getLong ( "allocationSize" ); }Copy the code

The wildcard “*” can also be used to enable multiple events, for example, ** @enableEvent (” jdk.objectalLocation \*”)** can be used to activate all ObjectAllocation events.

To reset collected events, the **jfrEvents. Reset ()** method can be used to ensure that only the **reset()** method is collected. For example, when running multiple iterations and asserting the results of each iteration:

for ( int i = 0 ; i < ITERATIONS ; I ++) {// Apply logic jfrEvents. Waiting for events (); // Assert jfrEvents. Reset (); }Copy the code

Frameworks such as Hibernate do not emit events themselves, but JMC proxies can be used to create events in these cases. With the JMC agent, SQL query events can be generated, which can then be used to assert the (number) of SQL queries entering the database. This led to a demonstration session in continuous performance regression testing using JfrUnit, which can be found in JfrUnit’s examples.