we have developped application based on Karaf and Apache Camel. While our application is entirely based on bundles ( OSGI ) we are also loading the Camel context ( and its' Route Contexts ) on startup, whcih would mean that we have defined some static routes.
My question is. Is there a way to dynamically LOAD routes while the application is running without the need to reread the Camel Context as this will reset/restart the already existing routes. The same would apply to already created routes, for example if we want to edit a route whcih already exist.
The whole idea is that we are planning to put the routes in a database and as such, the database will be edited by a GUI.
So what's the best approach to do this? I do not beleive that it is really the best approach to reload OSGI bundle/bundles in case of a route being added, editted etc...
More or less during operation of the application, Different endpoints + their related routes will be added, edited, removed.
Please advise.
Thanks,
Tiho
I think a good approach is to group your routes into small contexts with just a few (or maybe even single) routes per context. Then you reload that small context without causing interruption in other routes.
However, as you don't believe in that approach, you can easily add and remove routes with methods on the CamelContext. Create a route builder that constructs routes from your database and use addRoutes and removeRoute.
See also this cookbook example how to load/edit routes form xml format at runtime
http://camel.apache.org/loading-routes-from-xml-files.html
Adding/Removing routes dynamically does not restart/reset camelContext.
Please find the sample.
DynamicAddRouteProcessor.java
public class DynamicAddRouteProcessor implements Processor {
@Override
public void process(Exchange paramExchange) throws Exception {
final String routeId = "DYNANMIC.ROUTE.1";
Route route = paramExchange.getContext().getRoute(routeId);
if (null == route) {
System.out.println("No route exist, creating one with name ");
paramExchange.getContext().addRoutes(new RouteBuilder() {
public void configure() throws Exception {
from("direct:DYNANMIC.ROUTE.1").routeId(routeId).to("direct:myloggerRoute");
}
});
} else {
System.out.println("Route already exist, no action"+ route.getId());
}
}
}
DynamicRemoveRouteProcessor.java
public class DynamicRemoveRouteProcessor implements Processor {
@Override
public void process(Exchange paramExchange) throws Exception {
final String routeId = "DYNANMIC.ROUTE.1";
Route route = paramExchange.getContext().getRoute(routeId);
if (null != route) {
System.out.println("Route already exist, deleting it!!!" + route.getId());
paramExchange.getContext().stopRoute(routeId);
paramExchange.getContext().removeRoute(routeId);
} else {
System.out.println("No sucn route exist, no action done "
+ routeId);
}
}
}
blueprint.xml
<camelContext xmlns="http://camel.apache.org/schema/blueprint">
<route id="timerToLog">
<from uri="timer:foo?period=5000" />
<setBody>
<method ref="helloBean" method="hello" />
</setBody>
<log message="The message contains ${body}" />
<to uri="mock:result" />
</route>
<route id="routeAddition">
<from uri="timer:foo?period=10000" />
<process ref="dynamicAddRouteProcessor" />
<log message="Added new route to context....DONE " />
<delay ><simple>5000</simple></delay>
<process ref="dynamicRemoveRouteProcessor" />
<to uri="mock:result" />
</route>
<route id="myloggerRoute">
<from uri="direct:myloggerRoute" />
<log message="Route add/removal completed - ${body}" />
<to uri="mock:result" />
</route>
</camelContext>