20241209-QuarkusIntegrationTestingPlaywright

Playwright Integration Testing with Quarkus

Testing in Quarkus can sometimes feel overwhelming, with its mix of frameworks and concepts to navigate. In this article, we’ll simplify the process by walking you through the setup of a straightforward integration test for web pages using Playwright, a powerful tool for end-to-end testing.

# Playwright Integration Testing with Quarkus

When developing with Quarkus, you've likely come across the Getting Started with Testing guide . If you used the Quarkus CLI to bootstrap your project (quarkus create), it probably generated a simple GreetingResource and an associated integration test.
                @QuarkusIntegrationTest 
public class GreetingResourceIT extends GreetingResourceTest { 

    // Run the same tests

}

@QuarkusTest
public class GreetingResourceTest {

    @Test
    public void testHelloEndpoint() {
        
        given().when()
        	.get("/hello")
        	.then()
        	.statusCode(200)
        	.body(is("hello"));
    }

}
                
It's a good start but at the same time not ideal is
  • The test directly interacts with the actual GreetingResource implementation without mocking dependencies
  • It uses RestAssured , which is great for testing APIs but isn’t the best tool for asserting HTML content
  • GreetingResourceIT is tightly coupled to GreetingResourceTest, limiting flexibility

# Why Add Playwright?

If your application serves HTML content, Playwright is the go-to tool for parsing and asserting DOM elements. It offers robust features for end-to-end testing of web applications. Adding the Playwright dependency can elevate your test coverage and provide a more reliable way to validate HTML output.
                <dependency>
	<groupId>io.quarkiverse.playwright</groupId>
	<artifactId>quarkus-playwright</artifactId>
	<version>${quarkus-playwright.version}</version>
	<scope>test</scope>
</dependency>
                

# Using @QuarkusTest

Let’s consider a simple HomeResource and its counter part HomeResourceTest
                @Path("/")
@RequiredArgsConstructor
public class HomeResource {

    private final IMyService myService;

    @GET
    @Produces(MediaType.TEXT_HTML)
    public TemplateInstance home() {

        var content = myService.fetchSomeContent();

        return Templates.home().data("content", content);
    }

}

@QuarkusTest
@WithPlaywright
public class HomeResourceTest {

    @InjectMock
    IMyService myService;
    
    @InjectPlaywright
    BrowserContext context;

    @TestHTTPResource("/")
    URL home;

    @Test
    void home() {
        // given
        var page = context.newPage();
        when(myService.findNextUrl).thenReturn("next-url/go");

        // when
        var actual = page.navigate(home.toString());

        // then
        page.waitForLoadState();

        assertThat(actual.status()).isEqualTo(200);
        assertThat(page.locator(".my-content-link").getAttribute("href")).isEqualTo("next-url/go");
    }

}
                
The resource is exposed at the root level and depends on myService to display some content.
With @QuarkusTest, you can mock out dependencies like myService, allowing you to test the resource in isolation.
Key Points:
  • @QuarkusTest is similar to @SpringBootTest in the Spring ecosystem
  • Tests run within the same JVM process as the application , enabling bean mocking with @InjectMock
  • However, note that these tests don’t run natively—they execute within the JVM

# Using @QuarkusIntegrationTest

Integration tests in Quarkus are different:
  • They execute against the native executable and run outside the JVM process
  • This approach is ideal for validating your application in production mode.
Challenges:
  • Mocking out dependencies becomes more complex in this setup
  • You could use profiles ( quarkus.test.integration-test-profile ) to customize behavior, but this typically requires a more elaborate integration environment
                @WithPlaywright
@QuarkusIntegrationTest
public class HomeResourceIT {

	@InjectPlaywright
	BrowserContext context;

	@TestHTTPResource("/")
	URL home;

	@Test
	void home() {
		// given
		var page = context.newPage();

		// when
		var actual = page.navigate(home.toString());

		// then
		page.waitForLoadState();

		assertThat(actual.status()).isEqualTo(HttpStatus.SC_OK);
		assertThat(page.locator(".my-content-link").getAttribute("href")).isEqualTo("next-production-url/go");
	}

}
                

# Conclusion

  • Playwright : Consider using it for end-to-end testing, especially for applications serving HTML content
  • @QuarkusTest: Easily mock beans by annotating them with @InjectMock for efficient and isolated testing
  • @QuarkusIntegrationTest: Best suited for production-like scenarios but has limitations in mocking; profiles can help mitigate this
For more detailed examples, visit the GitHub repository of gdevxy.ch!
 
Leave a thumbs up Leave a thumbs up if you liked it

Remaining characters: 2000
Related blog posts
Quarkus and Maven multi-modules pitfalls /