Service Call
The Service Call EIP allows to call remote services in a distributed system. The service to call is looked up in a service registry of some sorts such as Kubernetes, Consul, Etcd, Zookeeper, DNS. The EIP separates the configuration of the service registry from the calling of the service.
When calling a service you may just refer to the name of the service in the EIP as shown below:
from("direct:start")
.serviceCall("foo")
.to("mock:result");
<camelContext xmlns="http://camel.apache.org/schema/spring">
<route>
<from uri="direct:start"/>
<serviceCall name="foo"/>
<to uri="mock:result"/>
</route>
</camelContext>
Camel will then:
-
search for a service call configuration from the Camel context and registry
-
lookup a service with the name
from an external service registryfoo
-
filter the servers
-
select the server to use
-
build a Camel URI using the chosen server info
By default the Service Call EIP uses camel-http so assuming that the selected service instance runs on host
on port myhost.com
, the computed Camel URI will be:80
http:myhost.com:80
Service Name to Camel URI Examples
It is often needed to build more complex Camel URI which may include options or paths which is possible through different options:name: value
The service name supports a limited uri like syntax, here some examples
Name | Resolution |
---|---|
foo |
|
foo/path |
|
foo/path?foo=bar |
from("direct:start")
.serviceCall("foo/hello")
.to("mock:result");
If you want to have more control over the uri construction, you can use the uri directive:
Name | URI | Resolution |
---|---|---|
foo |
undertow:http://foo/hello |
undertow:http://host:port/hello |
foo |
undertow:http://foo.host:foo.port/hello |
undertow:http://host:port/hello |
from("direct:start")
.serviceCall("foo", "undertow:http://foo/hello")
.to("mock:result");
Advanced users can have full control over the uri construction through expressions:
from("direct:start")
.serviceCall()
.name("foo")
.expression()
.simple("undertow:http://${header.CamelServiceCallServiceHost}:${header.CamelServiceCallServicePort}/hello");
Options
The Service Call EIP supports 13 options which are listed below:
Name | Description | Default | Type |
---|---|---|---|
name |
Required Sets the name of the service to use |
String |
|
uri |
The uri of the endpoint to send to. The uri can be dynamic computed using the org.apache.camel.language.simple.SimpleLanguage expression. |
String |
|
component |
The component to use. |
http |
String |
pattern |
Sets the optional ExchangePattern used to invoke this endpoint |
ExchangePattern |
|
configurationRef |
Refers to a ServiceCall configuration to use |
String |
|
serviceDiscoveryRef |
Sets a reference to a custom ServiceDiscovery to use. |
String |
|
serviceFilterRef |
Sets a reference to a custom ServiceFilter to use. |
String |
|
serviceChooserRef |
Sets a reference to a custom ServiceChooser to use. |
String |
|
loadBalancerRef |
Sets a reference to a custom ServiceLoadBalancer to use. |
String |
|
expressionRef |
Set a reference to a custom Expression to use. |
String |
|
serviceDiscoveryConfiguration |
Required Configures the ServiceDiscovery using the given configuration. |
ServiceCallServiceDiscoveryConfiguration |
|
serviceFilterConfiguration |
Required Configures the ServiceFilter using the given configuration. |
ServiceCallServiceFilterConfiguration |
|
loadBalancerConfiguration |
Required Configures the LoadBalancer using the given configuration. |
ServiceCallServiceLoadBalancerConfiguration |
In addition to ref/binding configuration style you can leverage specific configuration DSL to customize specific options:
Static Service Discovery
This service discovery implementation does not query any external services to find out the list of services associated to a named service but keep them in memory. Each service should be provided in the following form:
[service@]host:port
The |
This implementation is provided by |
Available options:
Name | Java Type | Description |
---|---|---|
servers |
|
A comma separated list of servers in the form: [service@]host:port,[service@]host2:port,[service@]host3:port |
from("direct:start")
.serviceCall("foo")
.staticServiceDiscovery()
.servers("service1@host1:80,service1@host2:80")
.servers("service2@host1:8080,service2@host2:8080,service2@host3:8080")
.end()
.to("mock:result");
<camelContext xmlns="http://camel.apache.org/schema/spring">
<route>
<from uri="direct:start"/>
<serviceCall name="foo">
<staticServiceDiscovery>
<servers>service1@host1:80,service1@host2:80</servers>
<servers>service2@host1:8080,service2@host2:8080,service2@host3:8080</servers>
</staticServiceDiscovery>
</serviceCall
<to uri="mock:result"/>
</route>
</camelContext>
Consul Service Discovery
To leverage Consul for Service Discovery, maven users will need to add the following dependency to their pom.xml
<dependency>
<groupId>org.apache.camel</groupId>
<artifactId>camel-consul</artifactId>
<!-- use the same version as your Camel core version -->
<version>x.y.z</version>
</dependency>
Available options:
Name | Java Type | Description |
---|---|---|
url |
|
The Consul agent URL |
datacenter |
|
The data center |
aclToken |
|
Sets the ACL token to be used with Consul |
userName |
|
Sets the username to be used for basic authentication |
password |
|
Sets the password to be used for basic authentication |
connectTimeoutMillis |
|
Connect timeout for OkHttpClient |
readTimeoutMillis |
|
Read timeout for OkHttpClient |
writeTimeoutMillis |
|
Write timeout for OkHttpClient |
And example in Java
from("direct:start")
.serviceCall("foo")
.consulServiceDiscovery()
.url("http://consul-cluster:8500")
.datacenter("neverland")
.end()
.to("mock:result");
DNS Service Discovery
To leverage DNS for Service Discovery, maven users will need to add the following dependency to their pom.xml
<dependency>
<groupId>org.apache.camel</groupId>
<artifactId>camel-dns</artifactId>
<!-- use the same version as your Camel core version -->
<version>x.y.z</version>
</dependency>
Available options:
Name | Java Type | Description |
---|---|---|
proto |
|
The transport protocol of the desired service, default "_tcp" |
domain |
|
The user name to use for basic authentication |
Example in Java:
from("direct:start")
.serviceCall("foo")
.dnsServiceDiscovery("my.domain.com")
.to("mock:result");
And in XML:
<camelContext xmlns="http://camel.apache.org/schema/spring">
<route>
<from uri="direct:start"/>
<serviceCall name="foo">
<dnsServiceDiscovery domain="my.domain.com"/>
</serviceCall>
<to uri="mock:result"/>
</route>
</camelContext>
Etcd Service Discovery
To leverage Etcd for Service Discovery, maven users will need to add the following dependency to their pom.xml
<dependency>
<groupId>org.apache.camel</groupId>
<artifactId>camel-etcd</artifactId>
<!-- use the same version as your Camel core version -->
<version>x.y.z</version>
</dependency>
Available options:
Name | Java Type | Description |
---|---|---|
uris |
|
The URIs the client can connect to |
userName |
|
The user name to use for basic authentication |
password |
|
The password to use for basic authentication |
timeout |
|
To set the maximum time an action could take to complete |
servicePath |
|
The path to look for service discovery, default "/services" |
type |
|
To set the discovery type, valid values are "on-demand" and "watch" |
Example in Java
from("direct:start") .serviceCall("foo") .etcdServiceDiscovery() .uris("http://etcd1:4001,http://etcd2:4001") .servicePath("/camel/services") .end() .to("mock:result");
And in XML
<camelContext xmlns="http://camel.apache.org/schema/spring">
<route>
<from uri="direct:start"/>
<serviceCall name="foo">
<etcdServiceDiscovery uris="http://etcd1:4001,http://etcd2:4001" servicePath="/camel/services"/>
</serviceCall>
<to uri="mock:result"/>
</route>
</camelContext>
Kubernetes Service Discovery
To leverage Kubernetes for Service Discovery, maven users will need to add the following dependency to their pom.xml
<dependency>
<groupId>org.apache.camel</groupId>
<artifactId>camel-kubernetes</artifactId>
<!-- use the same version as your Camel core version -->
<version>x.y.z</version>
</dependency>
Available options:
Name | Java Type | Description |
---|---|---|
lookup |
|
How to perform service lookup. Possible values: client, dns, environment |
apiVersion |
|
Kubernetes API version when using client lookup |
caCertData |
|
Sets the Certificate Authority data when using client lookup |
caCertFile |
|
Sets the Certificate Authority data that are loaded from the file when using client lookup |
clientCertData |
|
Sets the Client Certificate data when using client lookup |
clientCertFile |
|
Sets the Client Certificate data that are loaded from the file when using client lookup |
clientKeyAlgo |
|
Sets the Client Keystore algorithm, such as RSA when using client lookup |
clientKeyData |
|
Sets the Client Keystore data when using client lookup |
clientKeyFile |
|
Sets the Client Keystore data that are loaded from the file when using client lookup |
clientKeyPassphrase |
|
Sets the Client Keystore passphrase when using client lookup |
dnsDomain |
|
Sets the DNS domain to use for dns lookup |
namespace |
|
The Kubernetes namespace to use. By default the namespace’s name is taken from the environment variable KUBERNETES_MASTER |
oauthToken |
|
Sets the OAUTH token for authentication (instead of username/password) when using client lookup |
username |
|
Sets the username for authentication when using client lookup |
password |
|
Sets the password for authentication when using client lookup |
trustCerts |
|
Sets whether to turn on trust certificate check when using client lookup |
Example in Java
from("direct:start")
.serviceCall("foo")
.kubernetesServiceDiscovery()
.lookup("dns")
.namespace("myNamespace")
.dnsDomain("my.domain.com")
.end()
.to("mock:result");
And in XML
<camelContext xmlns="http://camel.apache.org/schema/spring">
<route>
<from uri="direct:start"/>
<serviceCall name="foo">
<kubernetesServiceDiscovery lookup="dns" namespace="myNamespace" dnsDomain="my.domain.com"/>
</serviceCall>
<to uri="mock:result"/>
</route>
</camelContext>
Service Filter
Blacklist Service Filter
This service filter implementation removes the listed services from those found by the service discovery. Each service should be provided in the following form:
[service@]host:port
The services are removed if they fully match |
Available options:
Name | Java Type | Description |
---|---|---|
servers |
|
A comma separated list of servers to blacklist: [service@]host:port,[service@]host2:port,[service@]host3:port |
Example in Java
from("direct:start")
.serviceCall("foo")
.staticServiceDiscovery()
.servers("service1@host1:80,service1@host2:80")
.servers("service2@host1:8080,service2@host2:8080,service2@host3:8080")
.end()
.blacklistFilter()
.servers("service2@host2:8080")
.end()
.to("mock:result");
And in XML
<camelContext xmlns="http://camel.apache.org/schema/spring">
<route>
<from uri="direct:start"/>
<serviceCall name="foo">
<staticServiceDiscovery>
<servers>service1@host1:80,service1@host2:80</servers>
<servers>service2@host1:8080,service2@host2:8080,service2@host3:8080</servers>
</staticServiceDiscovery>
<blacklistServiceFilter>
<servers>service2@host2:8080</servers>
</blacklistServiceFilter>
</serviceCall
<to uri="mock:result"/>
</route>
</camelContext>
Custom Service Filter
Service Filters choose suitable candidates from the service definitions found in the service discovery.
As of Camel 3.10.0 they have access to the current exchange, which allows you to create service filters comparing service metadata with message content.
Assuming you have labeled one of the services in your service discovery to support a certain type of requests:
serviceDiscovery.addServer(new DefaultServiceDefinition("service", "127.0.0.1", 1003,
Collections.singletonMap("supports", "foo")));
The current exchange has a property which says that it needs a foo service:
exchange.setProperty("needs", "foo")
You can then use a ServiceFilter
to select the service instances which match the exchange:
from("direct:start")
.serviceCall()
.name("service")
.serviceFilter((exchange, services) -> services.stream()
.filter(serviceDefinition -> Optional.ofNullable(serviceDefinition.getMetadata()
.get("supports"))
.orElse("")
.equals(exchange.getProperty("needs", String.class)))
.collect(Collectors.toList()));
.end()
.to("mock:result");
Load Balancer
The Service Call EIP comes with its own loadbalancer which is instantiated by default if a custom loadbalancer is not configured. It glues Service Discovery, Service Filter, Service Chooser and Service Expression together to load balance requests among the available services.
If you need a more sophisticated load balancer you can use Ribbon by adding camel-ribbon to the mix, maven users will need to add the following dependency to their pom.xml
The RibbonServiceLoadBalancer has no concept of a current Exchange .
Service filters therefore receive a dummy exchange when used with Ribbon.
|
<dependency>
<groupId>org.apache.camel</groupId>
<artifactId>camel-ribbon</artifactId>
<!-- use the same version as your Camel core version -->
<version>x.y.z</version>
</dependency>
Available options:
Name | Java Type | Description |
---|---|---|
clientName |
|
The Ribbon client name |
properties |
|
Custom client config properties |
To leverage Ribbon, it is required to explicit enable it:
Java example
from("direct:start")
.serviceCall("foo")
.ribbonLoadBalancer()
.to("mock:result");
And in XML
<camelContext xmlns="http://camel.apache.org/schema/spring">
<route>
<from uri="direct:start"/>
<serviceCall name="foo">
<ribbonLoadBalancer/>
</serviceCall>
<to uri="mock:result"/>
</route>
</camelContext>
You can configure Ribbon key programmatically using RibbonConfiguration
:
RibbonConfiguration configuration = new RibbonConfiguration();
configuration.addProperty("listOfServers", "localhost:9090,localhost:9091");
from("direct:start")
.serviceCall("foo")
.loadBalancer(new RibbonServiceLoadBalancer(configuration))
.to("mock:result");
Or leveraging XML specific configuration:
<camelContext xmlns="http://camel.apache.org/schema/spring">
<route>
<from uri="direct:start"/>
<serviceCall name="foo">
<ribbonLoadBalancer>
<properties key="listOfServers" value="localhost:9090,localhost:9091"/>
</ribbonLoadBalancer>
</serviceCall>
<to uri="mock:result"/>
</route>
</camelContext>
Shared configurations
The Service Call EIP can be configured straight on the route definition or through shared configurations, here an example with two configurations registered in the Camel Context:
ServiceCallConfigurationDefinition globalConf = new ServiceCallConfigurationDefinition();
globalConf.setServiceDiscovery(
name -> Arrays.asList(
new DefaultServiceDefinition(name, "my.host1.com", 8080),
new DefaultServiceDefinition(name, "my.host2.com", 443))
);
globalConf.setServiceChooser(
list -> list.get(ThreadLocalRandom.current().nextInt(list.size()))
);
ServiceCallConfigurationDefinition httpsConf = new ServiceCallConfigurationDefinition();
httpsConf.setServiceFilter(
list -> list.stream().filter((exchange, s) -> s.getPort() == 443).collect(toList())
);
getContext().setServiceCallConfiguration(globalConf);
getContext().addServiceCallConfiguration("https", httpsConf);
Each Service Call definition and configuration will inherit from the globalConf
which can be seen as default configuration,
then you can reference the httpsConf
in your route as follow:
from("direct:start")
.serviceCall()
.name("foo")
.serviceCallConfiguration("https")
.end()
.to("mock:result");
This route will leverages the service discovery and service chooser from globalConf
and the service filter from httpsConf
but you can override any of them if needed straight on the route:
from("direct:start")
.serviceCall()
.name("foo")
.serviceCallConfiguration("https")
.serviceChooser(list -> list.get(0))
.end()
.to("mock:result");
Spring Boot support
In a Spring-Boot application you can externalize most of the configuration options:
# this can be configured stright tot he route and it has been included to show
# property placeholders support
service.name = foo
# this property is not mandatory and it has been included to show how to configure
# the service discovery implementation provided by camel-consul
camel.cloud.consul.service-discovery.url = http://localhost:8500
# Add a static list of servers for the service named foo
camel.cloud.service-discovery.services[foo] = host1.static:8080,host2.static:8080
@Component
public class MyRouteBuilder implements RouteBuilder {
@Override
public void configure() throws Exception {
from("direct:start")
.serviceCall("{{service.name}}");
}
}
Spring Cloud support
If you are using Camel in an application based on Spring Cloud, you can leverage Spring Cloud service discovery and load balancing capabilities by adding the Spring Cloud related dependencies (i.e. spring-cloud-consul, spring-cloud-kubernetes) as any Spring Boot/Cloud application in addition to Camel’s own camel-spring-cloud dependency.
<dependency>
<groupId>org.apache.camel</groupId>
<artifactId>camel-spring-cloud dependency</artifactId>
<!-- use the same version as your Camel core version -->
<version>x.y.z</version>
</dependency>