이 콘텐츠는 선택한 언어로 제공되지 않습니다.
Chapter 5. Testing routes in Camel Quarkus
5.1. Testing Camel Quarkus Extensions 링크 복사링크가 클립보드에 복사되었습니다!
Testing offers a good way to ensure Camel routes behave as expected over time. If you haven’t already, read the Camel Quarkus user guide First Steps and the Quarkus documentation Testing your application section.
When it comes to testing a route in the context of Quarkus, the recommended approach is to write local integration tests. This has the advantage of covering both JVM and native mode.
In JVM mode, you can use the CamelTestSupport style of testing.
5.1.1. Running in JVM mode 링크 복사링크가 클립보드에 복사되었습니다!
In JVM mode, use the @QuarkusTest annotation to bootstrap Quarkus and start Camel routes before the @Test logic executes.
For example:
You can find a sample implementation in the Camel Quarkus source:
5.1.2. Running in native mode 링크 복사링크가 클립보드에 복사되었습니다!
Always test that your application works in native mode for all supported extensions.
You can reuse the test logic defined for JVM mode by inheriting the logic from the respective JVM mode class.
Add the @QuarkusIntegrationTest annotation to tell the Quarkus JUnit extension to compile the application under test to native image and start it before running the tests.
You can find a sample implementation in the Camel Quarkus source:
5.1.3. Differences between @QuarkusTest and @QuarkusIntegrationTest 링크 복사링크가 클립보드에 복사되었습니다!
A native executable does not need a JVM to run, and cannot run in a JVM, because it is native code, not bytecode.
There is no point in compiling tests to native code so they run using a traditional JVM.
This means that communication between tests and the application must go over the network (HTTP/REST, or any other protocol your application speaks), through watching filesystems (log files for example), or any other interprocess communication.
5.1.3.1. @QuarkusTest in JVM mode 링크 복사링크가 클립보드에 복사되었습니다!
In JVM mode, tests annotated with @QuarkusTest execute in the same JVM as the application under test.
This means you can use @Inject to add beans from the application into the test code.
You can also define new beans or even override the beans from the application using @jakarta.enterprise.inject.Alternative and @jakarta.annotation.Priority.
5.1.3.2. @QuarkusIntegrationTest in native mode 링크 복사링크가 클립보드에 복사되었습니다!
In native mode, tests annotated with @QuarkusIntegrationTest execute in a JVM hosted in a process separate from the running native application.
An important consequence of this, is that all communication between the tests and the native application, must take one or more of the following forms:
- Network calls. Typically, HTTP or any other network protocol your application supports.
-
Watching the filesystem for changes. (For example via Camel
fileendpoints.) - Any other kind of interprocess communication.
QuarkusIntegrationTest provides additional features that are not available through @QuarkusTest:
- In JVM mode, you can launch and test the runnable application JAR produced by the Quarkus build.
- In native mode, you can launch and test the native application produced by the Quarkus build.
- If you add a container image to the build, a container starts, and tests execute against it.
For more information about QuarkusIntegrationTest, see the Quarkus testing guide.
5.1.4. Testing with external services 링크 복사링크가 클립보드에 복사되었습니다!
5.1.4.1. Testcontainers 링크 복사링크가 클립보드에 복사되었습니다!
Sometimes your application needs to access some external resource, such as a messaging broker, a database, or other service.
If a container image is available for the service of interest, you can use Testcontainers to start and configure the services during testing.
5.1.4.1.1. Passing configuration data with QuarkusTestResourceLifecycleManager 링크 복사링크가 클립보드에 복사되었습니다!
For the application to work properly, it is often essential to pass the connection configuration data (host, port, user, password of the remote service) to the application before it starts.
In the Quarkus ecosystem, QuarkusTestResourceLifecycleManager serves this purpose.
You can start one or more Testcontainers in the start() method and return the connection configuration from the method in the form of a Map.
The entries of this map are then passed to the application in different ways depending on the mode:
-
Native mode: a command line (
-Dkey=value) - JVM Mode: a special MicroProfile configuration provider
Command line and MicroProfile settings have a higher precedence than the settings in the application.properties file.
Reference the defined test resource from the test classes with @QuarkusTestResource:
You can find a sample implementation in the Camel Quarkus source:
5.1.4.2. WireMock 링크 복사링크가 클립보드에 복사되었습니다!
Instead of having the tests connect to live endpoints, for example, if they are unavailable, unreliable, or expensive, you can stub HTTP interactions with third-party services & APIs.
You can use WireMock for mocking & recording HTTP interactions. It is used extensively throughout the Camel Quarkus test suite for various component extensions.
5.1.4.2.1. Setting up WireMock 링크 복사링크가 클립보드에 복사되었습니다!
Procedure
Set up the WireMock server.
NoteAlways configure the Camel component under test to pass any HTTP interactions through the WireMock proxy. You can achieve this by configuring a component property that determines the API endpoint URL.
Copy to Clipboard Copied! Toggle word wrap Toggle overflow -
Ensure your test class has the
@QuarkusTestResourceannotation with the appropriate test resource class specified as the value. The WireMock server will be started before all tests are executed and will be shut down when all tests are finished.
The WireMock server starts before all tests execute and shuts down when all tests finish.
You can find a sample implementation in the Camel Quarkus integration test source tree:
5.1.5. CamelTestSupport style of testing with CamelQuarkusTestSupport 링크 복사링크가 클립보드에 복사되었습니다!
Since Camel Quarkus 2.13.0, you can use CamelQuarkusTestSupport for testing. It is a replacement for CamelTestSupport, which does not work well with Quarkus.
CamelQuarkusTestSupport only works in JVM mode. If you need to test in native mode, then use one of the alternate test strategies described above.
5.1.5.1. Testing with CamelQuarkusTestSupport in JVM mode 링크 복사링크가 클립보드에 복사되었습니다!
Add the following dependency into your module (preferably in the test scope):
<dependency>
<groupId>org.apache.camel.quarkus</groupId>
<artifactId>camel-quarkus-junit5</artifactId>
<scope>test</scope>
</dependency>
<dependency>
<groupId>org.apache.camel.quarkus</groupId>
<artifactId>camel-quarkus-junit5</artifactId>
<scope>test</scope>
</dependency>
You can use CamelQuarkusTestSupport in your test like this:
@QuarkusTest
@TestProfile(SimpleTest.class) //necessary only if "newly created" context is required for the test (worse performance)
public class SimpleTest extends CamelQuarkusTestSupport {
...
}
@QuarkusTest
@TestProfile(SimpleTest.class) //necessary only if "newly created" context is required for the test (worse performance)
public class SimpleTest extends CamelQuarkusTestSupport {
...
}
5.1.5.2. Customizing the CamelContext for testing 링크 복사링크가 클립보드에 복사되었습니다!
You can customize the CamelContext for testing with configuration profiles, CDI beans, observers, mocks etc. You can also override the createCamelContext method and interact directly with the CamelContext.
When using createCamelContext you MUST NOT instantiate and return a new CamelContext. Instead, invoke super.createCamelContext() and modify the returned CamelContext as needed. Failing to follow this rule will result in an exception being thrown.
5.1.5.3. Configuring routes for testing 링크 복사링크가 클립보드에 복사되었습니다!
Any classes that extend RouteBuilder in your application will have their routes automatically added to the CamelContext. Similarly, any XML or YAML routes configured from camel.main.routes-include-pattern will also be loaded.
This may not always be desirable for your tests. You control which routes get loaded at test time with configuration properties:
-
quarkus.camel.routes-discovery.include-patterns -
quarkus.camel.routes-discovery.exclude-patterns, -
camel.main.routes-include-pattern -
camel.main.routes-exclude-pattern.
You can also define test specific routes per test class by overriding createRouteBuilder:
5.1.5.4. CamelContext test lifecycle 링크 복사링크가 클립보드에 복사되었습니다!
One of the main differences in CamelQuarkusTestSupport compared to CamelTestSupport is how the CamelContext lifecycle is managed.
On Camel Quarkus, a single CamelContext is created for you automatically by the runtime. By default, this CamelContext is shared among all tests and remains started for the duration of the entire test suite execution.
This can potentially have some unintended side effects for your tests. If you need to have the CamelContext restarted between tests, then you can create a custom test profile, which will force the application under test to be restarted.
For example, to define a test profile:
@QuarkusTest
class MyTestProfile implements QuarkusTestProfile {
...
}
@QuarkusTest
class MyTestProfile implements QuarkusTestProfile {
...
}
Then reference it on the test class with @TestProfile:
You cannot manually restart the CamelContext by invoking its stop() and start() methods. This will result in an exception.
5.1.5.5. Examples 링크 복사링크가 클립보드에 복사되었습니다!
5.1.5.5.1. Simple RouteBuilder and test class 링크 복사링크가 클립보드에 복사되었습니다!
Simple RouteBuilder:
Test sending a message payload to the direct:start endpoint:
5.1.5.5.2. Using AdviceWith 링크 복사링크가 클립보드에 복사되었습니다!
5.1.5.5.3. Explicitly enabling advice 링크 복사링크가 클립보드에 복사되었습니다!
When explicitly enabling advice you must invoke startRouteDefinitions when completing your AdviceWith setup.
Invoking startRouteDefinitions is only required if you have routes configured that are NOT being advised.
5.1.5.6. Limitations 링크 복사링크가 클립보드에 복사되었습니다!
5.1.5.6.1. Test lifecycle methods inherited from CamelTestSupport 링크 복사링크가 클립보드에 복사되었습니다!
CamelQuarkusTestSupport inherits some test lifecycle methods from CamelTestSupport. However, they should not be used and instead are replaced with equivalent methods in CamelQuarkusTestSupport.
| CamelTestSupport lifecycle methods | CamelQuarkusTestSupport equivalent |
|---|---|
|
|
|
|
|
|
|
|
|
|
|
|
5.1.5.6.2. Creating a custom Camel registry is not supported 링크 복사링크가 클립보드에 복사되었습니다!
The CamelQuarkusTestSupport implementation of createCamelRegistry will throw UnsupportedOperationException.
If you need to bind or unbind objects to the Camel registry, then you can do it by one of the following methods.
Produce named CDI beans
Copy to Clipboard Copied! Toggle word wrap Toggle overflow -
Override
createCamelContext(see example above) and invokecamelContext.getRegistry().bind("foo", fooBean) Use the
@BindToRegistryannotation@QuarkusTest class SimpleTest extends CamelQuarkusTestSupport { @BindToRegistry("myBean") MyBean myBean = new MyBean(); }@QuarkusTest class SimpleTest extends CamelQuarkusTestSupport { @BindToRegistry("myBean") MyBean myBean = new MyBean(); }Copy to Clipboard Copied! Toggle word wrap Toggle overflow NoteBeans bound to the Camel registry from individual test classes, will persist for the duration of the test suite execution. This could have unintended consequences, depending on your test expectations. You can use test profiles to restart the
CamelContextto avoid this.