Skip to content

Commit

Permalink
Add capability to disable automatic startup of the Camel Quarkus runtime
Browse files Browse the repository at this point in the history
  • Loading branch information
jamesnetherton committed Oct 28, 2024
1 parent bad31c1 commit 444fa48
Show file tree
Hide file tree
Showing 10 changed files with 156 additions and 37 deletions.
12 changes: 6 additions & 6 deletions docs/modules/ROOT/pages/reference/extensions/core.adoc
Original file line number Diff line number Diff line change
Expand Up @@ -130,12 +130,6 @@ As such, the class `PropertiesCustomBeanWithSetterInjection` needs to be link:ht
| Configuration property | Type | Default


|icon:lock[title=Fixed at build time] [[quarkus.camel.bootstrap.enabled]]`link:#quarkus.camel.bootstrap.enabled[quarkus.camel.bootstrap.enabled]`

When set to true, the `CamelRuntime` will be started automatically.
| `boolean`
| `true`

|icon:lock[title=Fixed at build time] [[quarkus.camel.service.discovery.exclude-patterns]]`link:#quarkus.camel.service.discovery.exclude-patterns[quarkus.camel.service.discovery.exclude-patterns]`

A comma-separated list of Ant-path style patterns to match Camel service definition files in the classpath. The
Expand Down Expand Up @@ -481,6 +475,12 @@ throws a `RuntimeException` ignore - Suppresses any warnings and the application
Prints the `CamelMain` usage statement but allows the application startup to proceed as normal
| `fail`, `warn`, `ignore`
| `warn`

| [[quarkus.camel.bootstrap.enabled]]`link:#quarkus.camel.bootstrap.enabled[quarkus.camel.bootstrap.enabled]`

When set to true, the {@link CamelRuntime} will be started automatically.
| `boolean`
| `true`
|===

[.configuration-legend]
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,7 @@
import io.quarkus.deployment.builditem.ShutdownContextBuildItem;
import io.quarkus.runtime.ShutdownContext;
import org.apache.camel.quarkus.core.CamelBootstrapRecorder;
import org.apache.camel.quarkus.core.CamelRuntimeConfig;
import org.apache.camel.quarkus.core.deployment.spi.CamelBootstrapCompletedBuildItem;
import org.apache.camel.quarkus.core.deployment.spi.CamelRuntimeBuildItem;
import org.apache.camel.quarkus.core.deployment.util.CamelQuarkusVersion;
Expand All @@ -39,6 +40,7 @@ class CamelBootstrapProcessor {
* @param commandLineArguments a reference to the raw command line arguments as they were passed to the application.
* @param shutdown a reference to a {@link ShutdownContext} used tor register the Camel's related shutdown
* tasks.
* @param camelRuntimeConfig The {@link CamelRuntimeConfig} instance.
*/
@BuildStep
@Record(value = ExecutionTime.RUNTIME_INIT)
Expand All @@ -48,11 +50,12 @@ void boot(
CamelRuntimeBuildItem runtime,
RawCommandLineArgumentsBuildItem commandLineArguments,
ShutdownContextBuildItem shutdown,
BuildProducer<ServiceStartBuildItem> serviceStartBuildItems) {
BuildProducer<ServiceStartBuildItem> serviceStartBuildItems,
CamelRuntimeConfig camelRuntimeConfig) {

recorder.addShutdownTask(shutdown, runtime.runtime());
if (runtime.isAutoStartup()) {
recorder.start(runtime.runtime(), commandLineArguments, CamelQuarkusVersion.getVersion());
recorder.start(camelRuntimeConfig, runtime.runtime(), commandLineArguments, CamelQuarkusVersion.getVersion());
}
/* Make sure that Quarkus orders this method before starting to serve HTTP endpoints.
* Otherwise first requests might reach Camel context in a non-yet-started state. */
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -38,13 +38,16 @@ public void run() {
});
}

public void start(RuntimeValue<CamelRuntime> runtime, Supplier<String[]> arguments, String camelQuarkusVersion) {
try {
Logger logger = Logger.getLogger(CamelBootstrapRecorder.class);
logger.infof("Apache Camel Quarkus %s is starting", camelQuarkusVersion);
runtime.getValue().start(arguments.get());
} catch (Exception e) {
throw new RuntimeException(e);
public void start(CamelRuntimeConfig camelRuntimeConfig, RuntimeValue<CamelRuntime> runtime, Supplier<String[]> arguments,
String camelQuarkusVersion) {
if (camelRuntimeConfig.bootstrap.enabled) {
try {
Logger logger = Logger.getLogger(CamelBootstrapRecorder.class);
logger.infof("Apache Camel Quarkus %s is starting", camelQuarkusVersion);
runtime.getValue().start(arguments.get());
} catch (Exception e) {
throw new RuntimeException(e);
}
}
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -32,14 +32,6 @@ public enum FailureRemedy {
fail, warn, ignore
}

/**
* Build time configuration options for `CamelRuntime` bootstrap.
*
* @asciidoclet
*/
@ConfigItem
public BootstrapConfig bootstrap;

/**
* Build time configuration options for Camel services.
*
Expand Down Expand Up @@ -121,18 +113,6 @@ public enum FailureRemedy {
@ConfigItem
public TypeConverterConfig typeConverter;

@ConfigGroup
public static class BootstrapConfig {

/**
* When set to true, the `CamelRuntime` will be started automatically.
*
* @asciidoclet
*/
@ConfigItem(defaultValue = "true")
public boolean enabled;
}

@ConfigGroup
public static class RoutesDiscoveryConfig {

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -22,7 +22,7 @@
* Represent a runnable Camel instance.
*/
public interface CamelRuntime extends HasCamelContext {
void start(String[] args);
void start(String... args);

void stop();

Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,28 @@
package org.apache.camel.quarkus.core;

import io.quarkus.runtime.annotations.ConfigGroup;
import io.quarkus.runtime.annotations.ConfigItem;
import io.quarkus.runtime.annotations.ConfigPhase;
import io.quarkus.runtime.annotations.ConfigRoot;

@ConfigRoot(name = "camel", phase = ConfigPhase.RUN_TIME)
public class CamelRuntimeConfig {
/**
* Runtime configuration options for {@link CamelRuntime} bootstrap.
*
* @asciidoclet
*/
@ConfigItem
public BootstrapConfig bootstrap;

@ConfigGroup
public static class BootstrapConfig {
/**
* When set to true, the {@link CamelRuntime} will be started automatically.
*
* @asciidoclet
*/
@ConfigItem(defaultValue = "true")
public boolean enabled;
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -68,7 +68,10 @@ public void start(String[] args) {

@Override
public void stop() {
main.stop();
if (!main.isStopped()) {
main.stop();
}

/* Wait till the Camel shutdown is finished in camel-main thread started in start(String[]) above */
final Thread worker = this.mainThread;
if (worker != null) {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -41,6 +41,7 @@
import org.apache.camel.component.log.LogComponent;
import org.apache.camel.impl.debugger.DebuggerJmxConnectorService;
import org.apache.camel.model.ModelCamelContext;
import org.apache.camel.quarkus.core.CamelRuntime;
import org.apache.camel.quarkus.core.FastFactoryFinderResolver;
import org.apache.camel.quarkus.it.support.typeconverter.MyPair;
import org.apache.camel.reactive.vertx.VertXReactiveExecutor;
Expand All @@ -63,6 +64,9 @@ public class CoreMainResource {
@Inject
CamelMain main;

@Inject
CamelRuntime camelRuntime;

@Path("/property/{name}")
@GET
@Produces(MediaType.TEXT_PLAIN)
Expand Down Expand Up @@ -310,4 +314,17 @@ public void contextReload() {
contextReloadStrategy.onReload(this);
}
}

@Path("/runtime/status")
@GET
@Produces(MediaType.TEXT_PLAIN)
public String runtimeStatus() {
return main.getStatus().name();
}

@Path("/runtime/start")
@POST
public void runtimeStart() {
camelRuntime.start();
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,23 @@
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You under the Apache License, Version 2.0
* (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.apache.camel.quarkus.main;

import io.quarkus.test.junit.QuarkusIntegrationTest;

@QuarkusIntegrationTest
class BootstrapDisabledIT extends BootstrapDisabledTest {
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,62 @@
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You under the Apache License, Version 2.0
* (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.apache.camel.quarkus.main;

import java.time.Duration;
import java.util.Map;

import io.quarkus.test.junit.QuarkusTest;
import io.quarkus.test.junit.QuarkusTestProfile;
import io.quarkus.test.junit.TestProfile;
import io.restassured.RestAssured;
import org.apache.camel.ServiceStatus;
import org.junit.jupiter.api.Test;

import static org.awaitility.Awaitility.await;
import static org.hamcrest.Matchers.is;

@QuarkusTest
@TestProfile(BootstrapDisabledTest.BootstrapDisabledTestProfile.class)
class BootstrapDisabledTest {
@Test
void initializeManualRuntimeBootstrap() {
// Initial runtime state should be stopped
RestAssured.get("/test/runtime/status")
.then()
.statusCode(200)
.body(is(ServiceStatus.Stopped.name()));

// Ensure the runtime can be started manually
RestAssured.post("/test/runtime/start")
.then()
.statusCode(204);

await().atMost(Duration.ofSeconds(10)).pollDelay(Duration.ofMillis(100)).untilAsserted(() -> {
RestAssured.get("/test/runtime/status")
.then()
.statusCode(200)
.body(is(ServiceStatus.Started.name()));
});
}

public static final class BootstrapDisabledTestProfile implements QuarkusTestProfile {
@Override
public Map<String, String> getConfigOverrides() {
return Map.of("quarkus.camel.bootstrap.enabled", "false");
}
}
}

0 comments on commit 444fa48

Please sign in to comment.