Camel Manual 2.4.0

User Manual: Pdf

Open the PDF directly: View PDF PDF.
Page Count: 673

DownloadCamel-manual-2.4.0
Open PDF In BrowserView PDF
Apache Camel

USER GUIDE
Version 2.4.0

Copyright 2007-2010, Apache Software Foundation

Table of Contents

Table of Contents......................................................................... ii
Chapter 1
Introduction ...................................................................................1
Chapter 2
Quickstart .......................................................................................1
Chapter 3
Getting Started..............................................................................7
Chapter 4
Architecture ................................................................................ 18
Chapter 5
Enterprise Integration Patterns.............................................. 30
Chapter 6
Cook Book ................................................................................... 35
Chapter 7
Tutorials....................................................................................... 90
Chapter 8
Language Appendix.................................................................. 195
Chapter 9
Pattern Appendix..................................................................... 245
Chapter 10
Component Appendix ............................................................. 361
Index ................................................................................................0

ii

AP A C H E C A M E L

CHAPTER

1

°°°°

Introduction

Apache Camel is a powerful open source integration framework based on known Enterprise
Integration Patterns with powerful Bean Integration.
Camel lets you create the Enterprise Integration Patterns to implement routing and mediation
rules in either a Java based Domain Specific Language (or Fluent API), via Spring based Xml
Configuration files or via the Scala DSL. This means you get smart completion of routing rules
in your IDE whether in your Java, Scala or XML editor.
Apache Camel uses URIs so that it can easily work directly with any kind of Transport or
messaging model such as HTTP, ActiveMQ, JMS, JBI, SCA, MINA or CXF Bus API together with
working with pluggable Data Format options. Apache Camel is a small library which has minimal
dependencies for easy embedding in any Java application. Apache Camel lets you work with the
same API regardless which kind of Transport used, so learn the API once and you will be able
to interact with all the Components that is provided out-of-the-box.
Apache Camel has powerful Bean Binding and integrated seamless with popular frameworks
such as Spring and Guice.
Apache Camel has extensive Testing support allowing you to easily unit test your routes.
Apache Camel can be used as a routing and mediation engine for the following projects:
• Apache ServiceMix which is the most popular and powerful distributed open source
ESB and JBI container
• Apache ActiveMQ which is the most popular and powerful open source message
broker
• Apache CXF which is a smart web services suite (JAX-WS)
• Apache MINA a networking framework
So don't get the hump, try Camel today!

CH APT ER 1 - IN T R O DUCT IO N

1

CHAPTER

2

°°°°

Quickstart

To start using Apache Camel quickly, you can read through some simple examples in this
chapter. For readers who would like a more thorough introduction, please skip ahead to
Chapter 3.

WALK THROUGH AN EXAMPLE CODE
This mini-guide takes you through the source code of a simple example.
Camel can be configured either by using Spring or directly in Java - which this example does.
We start with creating a CamelContext - which is a container for Components, Routes etc:
CamelContext context = new DefaultCamelContext();

There is more than one way of adding a Component to the CamelContext. You can add
components implicitly - when we set up the routing - as we do here for the FileComponent:
context.addRoutes(new RouteBuilder() {
public void configure() {
from("test-jms:queue:test.queue").to("file://test");
// set up a listener on the file component
from("file://test").process(new Processor() {
public void process(Exchange e) {
System.out.println("Received exchange: " + e.getIn());
}
});
}
});

or explicitly - as we do here when we add the JMS Component:
ConnectionFactory connectionFactory = new
ActiveMQConnectionFactory("vm://localhost?broker.persistent=false");
// Note we can explicity name the component
context.addComponent("test-jms",
JmsComponent.jmsComponentAutoAcknowledge(connectionFactory));

1

CH A PTE R 2 - Q U I CKS T A R T

The above works with any JMS provider. If we know we are using ActiveMQ we can use an
even simpler form using the activeMQComponent() method while specifying the brokerURL
used to connect to ActiveMQ
camelContext.addComponent("activemq",
activeMQComponent("vm://localhost?broker.persistent=false"));

In normal use, an external system would be firing messages or events directly into Camel
through one if its Components but we are going to use the ProducerTemplate which is a really
easy way for testing your configuration:
ProducerTemplate template = context.createProducerTemplate();

Next you must start the camel context. If you are using Spring to configure the camel context
this is automatically done for you; though if you are using a pure Java approach then you just
need to call the start() method
camelContext.start();

This will start all of the configured routing rules.
So after starting the CamelContext, we can fire some objects into camel:
for (int i = 0; i < 10; i++) {
template.sendBody("test-jms:queue:test.queue", "Test Message: " + i);
}

WHAT HAPPENS?
From the ProducerTemplate - we send objects (in this case text) into the CamelContext to the
Component test-jms:queue:test.queue. These text objects will be converted automatically into
JMS Messages and posted to a JMS Queue named test.queue. When we set up the Route, we
configured the FileComponent to listen of the test.queue.
The File FileComponent will take messages off the Queue, and save them to a directory
named test. Every message will be saved in a file that corresponds to its destination and message
id.
Finally, we configured our own listener in the Route - to take notifications from the
FileComponent and print them out as text.
That's it!
If you have the time then use 5 more minutes to Walk through another example that
demonstrates the Spring DSL (XML based) routing.

CH APT ER 2 - Q UICK ST AR T

2

Camel 1.4.0 change
In Camel 1.4.0, CamelTemplate has been marked as @deprecated.
ProducerTemplate should be used instead and its created from the CamelContext
itself.
ProducerTemplate template = context.createProducerTemplate();

WALK THROUGH ANOTHER EXAMPLE
Introduction
We continue the walk from Walk through an Example. This time we take a closer look at the
routing and explains a few pointers so you wont walk into a bear trap, but can enjoy a walk
after hours to the local pub for a large beer
First we take a moment to look at the Enterprise Integration Patterns that is the base
pattern catalog for integrations. In particular we focus on the Pipes and Filters EIP pattern, that
is a central pattern. This is used for: route through a sequence of processing steps, each
performing a specific function - much like the Java Servlet Filters.
Pipes and filters
In this sample we want to process a message in a sequence of steps where each steps can
perform their specific function. In our example we have a JMS queue for receiving new orders.
When an order is received we need to process it in several steps:
▪ validate
▪ register
▪ send confirm email
This can be created in a route like this:









Where as the bean ref is a reference for a spring bean id, so we define our beans using
regular Spring XML as:

3

CH A PTE R 2 - Q U I CKS T A R T

Pipeline is default
In the route above we specify pipeline but it can be omitted as its default, so
you can write the route as:



uri="jms:queue:order"/>
ref="validateOrder"/>
ref="registerOrder"/>
ref="sendConfirmEmail"/>

This is commonly used not to state the pipeline.
An example where the pipeline needs to be used, is when using a multicast and "one" of
the endpoints to send to (as a logical group) is a pipeline of other endpoints. For example.












The above sends the order (from jms:queue:order) to two locations at the same time,
our log component, and to the "pipeline" of beans which goes one to the other. If you
consider the opposite, sans the 










you would see that multicast would not "flow" the message from one bean to the next, but
rather send the order to all 4 endpoints (1x log, 3x bean) in parallel, which is not (for this

CH APT ER 2 - Q UICK ST AR T

4

example) what we want. We need the message to flow to the validateOrder, then to the
registerOrder, then the sendConfirmEmail so adding the pipeline, provides this facility.



Our validator bean is a plain POJO that has no dependencies to Camel what so ever. So you
can implement this POJO as you like. Camel uses rather intelligent Bean Binding to invoke your
POJO with the payload of the received message. In this example we will not dig into this how
this happens. You should return to this topic later when you got some hands on experience
with Camel how it can easily bind routing using your existing POJO beans.
So what happens in the route above. Well when an order is received from the JMS queue
the message is routed like Pipes and Filters:
1. payload from the JMS is sent as input to the validateOrder bean
2. the output from validateOrder bean is sent as input to the registerOrder bean
3. the output from registerOrder bean is sent as input to the sendConfirmEmail bean
Using Camel Components
In the route lets imagine that the registration of the order has to be done by sending data to a
TCP socket that could be a big mainframe. As Camel has many Components we will use the
camel-mina component that supports TCP connectivity. So we change the route to:







What we now have in the route is a to type that can be used as a direct replacement for the
bean type. The steps is now:
1. payload from the JMS is sent as input to the validateOrder bean
2. the output from validateOrder bean is sent as text to the mainframe using TCP
3. the output from mainframe is sent back as input to the sendConfirmEmai bean
What to notice here is that the to is not the end of the route (the world
) in this
example it's used in the middle of the Pipes and Filters. In fact we can change the bean types to
to as well:

5

CH A PTE R 2 - Q U I CKS T A R T








As the to is a generic type we must state in the uri scheme which component it is. So we must
write bean: for the Bean component that we are using.
Conclusion
This example was provided to demonstrate the Spring DSL (XML based) as opposed to the
pure Java DSL from the first example. And as well to point about that the to doesn't have to be
the last node in a route graph.
This example is also based on the in-only message exchange pattern. What you must
understand as well is the in-out message exchange pattern, where the caller expects a
response. We will look into this in another example.
See also
▪ Examples
▪ Tutorials
▪ User Guide

CH APT ER 2 - Q UICK ST AR T

6

CHAPTER

3

°°°°

Getting Started with Apache
Camel

THE ENTERPRISE INTEGRATION PATTERNS (EIP) BOOK
The purpose of a "patterns" book is not to advocate new techniques that the authors have
invented, but rather to document existing best practices within a particular field. By doing this,
the authors of a patterns book hope to spread knowledge of best practices and promote a
vocabulary for discussing architectural designs.
One of the most famous patterns books is Design Patterns: Elements of Reusable Object-oriented
Software by Erich Gamma, Richard Helm, Ralph Johnson and John Vlissides. Some people refer
to this as the "gang of four" book, partly to distinguish this book from other books that use
"Design Patterns" in their titles and, perhaps, partly because they cannot remember the names
of all four authors.
Since the publication of Design Patterns, many other patterns books, of varying quality, have been
written. One famous patterns book is called Enterprise Integration Patterns: Designing, Building, and
Deploying Messaging Solutions by Gregor Hohpe and Bobby Woolfe. It is common for people to
refer to this book as EIP, which is an acronym of its title. As the subtitle of EIP suggests, the
book focusses on design patterns for asynchronous messaging systems. The book discusses 65
patterns. Each pattern is given a textual name and most are also given a graphical symbol. The
graphical symbols are intended to be used in architectural diagrams.

THE CAMEL PROJECT
Camel (http://activemq.apache.org/camel/) is an open-source, Java-based project that is a part of
the Apache ActiveMQ project. Camel provides a class library that, according to its
documentation, can be used to implement 31 design patterns in the EIP book. I am not sure
why the Camel documentation discusses only 31 of the 65 EIP design patterns. Perhaps this is
due to incomplete documentation. Or perhaps it means that the Camel project, which is less
than 1 year old at the time of writing, is not yet as feature rich as the EIP book.
Because Camel implements many of the design patterns in the EIP book, it would be a good
idea for people who work with Camel to read the EIP book.

7

CH A PTE R 3 - GE T T I NG S T A R T E D W I T H A PACH E CAM EL

ONLINE DOCUMENTATION FOR CAMEL
The Camel project was started in early 2007. At the time of writing, the Camel project is too
young for there to be published books available on how to use Camel. Instead, the only source
of documentation seems to the documentation page on the Apache Camel website.
Here in early 2010 the Camel project have been successful for the last couple of years and
thus there are now books available about Camel. We have gathered a list of books here.
Problems with Camel's online documentation
Currently, the online documentation for the Apache Camel project suffers from two problems.
First, the documentation is incomplete. Second, there is no clearly specified reading order to
the documentation. For example, there is no table of contents. Instead, documentation is
fragmented over a collection of 100+ web pages, and hypertext links haphazardly tie these web
pages to each other. This documentation might suffice as reference material for people already
familiar with Camel but it does not qualify as a tutorial for beginners.
The problems with the documentation are unlikely to be due to, say, its author(s) lacking
writing ability. Rather, it is more likely that the problems are due to the author(s) lack of time. I
expect Camel's documentation will improve over time. I am writing this overview of Camel to
partially counter some of the problems that currently afflict the Camel documentation. In
particular, this document aims to serve as a (so far, incomplete) "beginner's guide to Camel". As
such, this document tries to complement, rather than compete with, the online Camel
documentation.
A useful tip for navigating the online documentation
There is one useful hint I can provide for reading the online Camel documentation. Each
documentation page has a logo at the top, and immediately underneath this is a think reddish
bar that contains some hypertext links. The Hypertext links on left side of this reddish bar
indicate your position in documentation. For example, If you are on the "Languages"
documentation page then the left-hand side of the reddish bar contains the following links.
Apache Camel > Documentation > Architecture > Languages

As you might expect, clicking on "Apache Camel" takes you back to the home page of the
Apache Camel project, and clicking on "Documentation" takes you to the main documentation
page. You can interpret the "Architeture" and "Languages" buttons as indicating you are in the
"Languages" section of the "Architecture" chapter. Doing this gives you at least some sense of
where you are within the documentation. If you are patient then you can spend a few hours
clicking on all the hypertext links you can find in the documentation pages, bookmark each page
with a hierarchical name (for example, you might bookmark the above page with the name
"Camel – Arch – Languages") and then you can use your bookmarks to serve as a primitive
table of contents for the online Camel documentation.

C H A P T E R 3 - G ET T IN G ST AR T ED W IT H APACH E CAM EL

8

Camel in Action book
If you are on the lookup for a comprehensive and organized documentation, then
we can recommend the Camel in Action book, which is the Camel bible.

ONLINE JAVADOC DOCUMENTATION
The Apache Camel website provides Javadoc documentation. It is important to note that the
Javadoc documentation is spread over several independent Javadoc hierarchies rather than being
all contained in a single Javadoc hierarchy. In particular, there is one Javadoc hierarchy for the
core APIs of Camel, and a separate Javadoc hierarchy for each communications technology
supported by Camel. For example, if you will be using Camel with ActiveMQ and FTP then you
need to look at the Javadoc hierarchies for the core API and Spring API.

CONCEPTS AND TERMINOLOGY FUNDAMENTAL TO CAMEL
I said in Section 3.1 ("Problems with Camel's online documentation") that the online Camel
documentation does not provide a tutorial for beginners. Because of this, in this section I try to
explain some of the concepts and terminology that are fundamental to Camel. This section is
not a complete Camel tutorial, but it is a first step in that direction.
Endpoint
The term endpoint is often used when talking about inter-process communication. For example,
in client-server communication, the client is one endpoint and the server is the other endpoint.
Depending on the context, an endpoint might refer to an address, such as a host:port pair for
TCP-based communication, or it might refer to a software entity that is contactable at that
address. For example, if somebody uses "www.example.com:80" as an example of an endpoint,
they might be referring to the actual port at that host name (that is, an address), or they might
be referring to the web server (that is, software contactable at that address). Often, the
distinction between the address and software contactable at that address is not an important
one.
Some middleware technologies make it possible for several software entities to be contactable
at the same physical address. For example, CORBA is an object-oriented, remote-procedurecall (RPC) middleware standard. If a CORBA server process contains several objects then a
client can communicate with any of these objects at the same physical address (host:port), but a
client communicates with a particular object via that object's logical address (called an IOR in
CORBA terminology), which consists of the physical address (host:port) plus an id that uniquely
identifies the object within its server process. (An IOR contains some additional information
that is not relevant to this present discussion.) When talking about CORBA, some people may
use the term "endpoint" to refer to a CORBA server's physical address, while other people may

9

CH A PTE R 3 - GE T T I NG S T A R T E D W I T H A PACH E CAM EL

use the term to refer to the logical address of a single CORBA object, and other people still
might use the term to refer to any of the following:
• The physical address (host:port) of the CORBA server process
• The logical address (host:port plus id) of a CORBA object.
• The CORBA server process (a relatively heavyweight software entity)
• A CORBA object (a lightweight software entity)
Because of this, you can see that the term endpoint is ambiguous in at least two ways.
First, it is ambiguous because it might refer to an address or to a software entity
contactable at that address. Second, it is ambiguous in the granularity of what it refers
to: a heavyweight versus lightweight software entity, or physical address versus logical
address. It is useful to understand that different people use the term endpoint in
slightly different (and hence ambiguous) ways because Camel's usage of this term
might be different to whatever meaning you had previously associated with the term.
Camel provides out-of-the-box support for endpoints implemented with many
different communication technologies. Here are some examples of the Camelsupported endpoint technologies.
• A JMS queue.
• A web service.
• A file. A file may sound like an unlikely type of endpoint, until you realize that in some
systems one application might write information to a file and, later, another
application might read that file.
• An FTP server.
• An email address. A client can send a message to an email address, and a server can
read an incoming message from a mail server.
• A POJO (plain old Java object).
In a Camel-based application, you create (Camel wrappers around) some endpoints
and connect these endpoints with routes, which I will discuss later in Section 4.8
("Routes, RouteBuilders and Java DSL"). Camel defines a Java interface called
Endpoint. Each Camel-supported endpoint has a class that implements this
Endpoint interface. As I discussed in Section 3.3 ("Online Javadoc documentation"),
Camel provides a separate Javadoc hierarchy for each communications technology
supported by Camel. Because of this, you will find documentation on, say, the
JmsEndpoint class in the JMS Javadoc hierarchy, while documentation for, say, the
FtpEndpoint class is in the FTP Javadoc hierarchy.
CamelContext
A CamelContext object represents the Camel runtime system. You typically have one
CamelContext object in an application. A typical application executes the following steps.
1. Create a CamelContext object.
2. Add endpoints – and possibly Components, which are discussed in Section 4.5
("Components") – to the CamelContext object.
3. Add routes to the CamelContext object to connect the endpoints.

C H A P T E R 3 - G ET T IN G ST AR T ED W IT H APACH E CAM EL

10

4. Invoke the start() operation on the CamelContext object. This starts Camelinternal threads that are used to process the sending, receiving and processing of
messages in the endpoints.
5. Eventually invoke the stop() operation on the CamelContext object. Doing this
gracefully stops all the endpoints and Camel-internal threads.
Note that the CamelContext.start() operation does not block indefinitely.
Rather, it starts threads internal to each Component and Endpoint and then
start() returns. Conversely, CamelContext.stop() waits for all the threads
internal to each Endpoint and Component to terminate and then stop()
returns.
If you neglect to call CamelContext.start() in your application then messages
will not be processed because internal threads will not have been created.
If you neglect to call CamelContext.stop() before terminating your application
then the application may terminate in an inconsistent state. If you neglect to call
CamelContext.stop() in a JUnit test then the test may fail due to messages not
having had a chance to be fully processed.
CamelTemplate
Camel used to have a class called CamelClient, but this was renamed to be
CamelTemplate to be similar to a naming convention used in some other open-source
projects, such as the TransactionTemplate and JmsTemplate classes in Spring.
The CamelTemplate class is a thin wrapper around the CamelContext class. It has
methods that send a Message or Exchange – both discussed in Section 4.6 ("Message and
Exchange")) – to an Endpoint – discussed in Section 4.1 ("Endpoint"). This provides a way to
enter messages into source endpoints, so that the messages will move along routes – discussed
in Section 4.8 ("Routes, RouteBuilders and Java DSL") – to destination endpoints.
The Meaning of URL, URI, URN and IRI
Some Camel methods take a parameter that is a URI string. Many people know that a URI is
"something like a URL" but do not properly understand the relationship between URI and URL,
or indeed its relationship with other acronyms such as IRI and URN.
Most people are familiar with URLs (uniform resource locators), such as "http://...", "ftp://...",
"mailto:...". Put simply, a URL specifies the location of a resource.
A URI (uniform resource identifier) is a URL or a URN. So, to fully understand what URI means,
you need to first understand what is a URN.
URN is an acronym for uniform resource name. There are may "unique identifier" schemes in the
world, for example, ISBNs (globally unique for books), social security numbers (unique within a
country), customer numbers (unique within a company's customers database) and telephone
numbers. Each "unique identifier" scheme has its own notation. A URN is a wrapper for
different "unique identifier" schemes. The syntax of a URN is "urn::". A URN uniquely identifies a resource, such as a book, person or piece of equipment.
By itself, a URN does not specify the location of the resource. Instead, it is assumed that a
registry provides a mapping from a resource's URN to its location. The URN specification does
not state what form a registry takes, but it might be a database, a server application, a wall chart
or anything else that is convenient. Some hypothetical examples of URNs are
"urn:employee:08765245", "urn:customer:uk:3458:hul8" and
"urn:foo:0000-0000-9E59-0000-5E-2". The  ("employee", "customer" and "foo"
in these examples) part of a URN implicitly defines how to parse and interpret the  that follows it. An arbitrary URN is meaningless unless: (1) you know the semantics
implied by the , and (2) you have access to the registry appropriate for the
. A registry does not have to be public or globally accessible. For example,
"urn:employee:08765245" might be meaningful only within a specific company.
To date, URNs are not (yet) as popular as URLs. For this reason, URI is widely misused as a
synonym for URL.
IRI is an acronym for internationalized resource identifier. An IRI is simply an internationalized
version of a URI. In particular, a URI can contain letters and digits in the US-ASCII character
set, while a IRI can contain those same letters and digits, and also European accented characters,
Greek letters, Chinese ideograms and so on.
Components
Component is confusing terminology; EndpointFactory would have been more appropriate because
a Component is a factory for creating Endpoint instances. For example, if a Camel-based
application uses several JMS queues then the application will create one instance of the
JmsComponent class (which implements the Component interface), and then the application
invokes the createEndpoint() operation on this JmsComponent object several times.
Each invocation of JmsComponent.createEndpoint() creates an instance of the
JmsEndpoint class (which implements the Endpoint interface). Actually, application-level
code does not invoke Component.createEndpoint() directly. Instead, application-level
code normally invokes CamelContext.getEndpoint(); internally, the CamelContext
object finds the desired Component object (as I will discuss shortly) and then invokes
createEndpoint() on it.
Consider the following code.
myCamelContext.getEndpoint("pop3://john.smith@mailserv.example.com?password=myPassword");

The parameter to getEndpoint() is a URI. The URI prefix (that is, the part before ":")
specifies the name of a component. Internally, the CamelContext object maintains a mapping
from names of components to Component objects. For the URI given in the above example,
the CamelContext object would probably map the pop3 prefix to an instance of the
MailComponent class. Then the CamelContext object invokes
createEndpoint("pop3://john.smith@mailserv.example.com?password=myPassword"

C H A P T E R 3 - G ET T IN G ST AR T ED W IT H APACH E CAM EL

12

on that MailComponent object. The createEndpoint() operation splits the URI into its
component parts and uses these parts to create and configure an Endpoint object.
In the previous paragraph, I mentioned that a CamelContext object maintains a mapping
from component names to Component objects. This raises the question of how this map is
populated with named Component objects. There are two ways of populating the map. The
first way is for application-level code to invoke CamelContext.addComponent(String
componentName, Component component). The example below shows a single
MailComponent object being registered in the map under 3 different names.
Component mailComponent = new org.apache.camel.component.mail.MailComponent();
myCamelContext.addComponent("pop3", mailComponent);
myCamelContext.addComponent("imap", mailComponent);
myCamelContext.addComponent("smtp", mailComponent);

The second (and preferred) way to populate the map of named Component objects in the
CamelContext object is to let the CamelContext object perform lazy initialization. This
approach relies on developers following a convention when they write a class that implements
the Component interface. I illustrate the convention by an example. Let's assume you write a
class called com.example.myproject.FooComponent and you want Camel to
automatically recognize this by the name "foo". To do this, you have to write a properties file
called "META-INF/services/org/apache/camel/component/foo" (without a ".properties" file
extension) that has a single entry in it called class, the value of which is the fully-scoped name
of your class. This is shown below.
Listing 1. META-INF/services/org/apache/camel/component/foo
class=com.example.myproject.FooComponent

If you want Camel to also recognize the class by the name "bar" then you write another
properties file in the same directory called "bar" that has the same contents. Once you have
written the properties file(s), you create a jar file that contains the
com.example.myproject.FooComponent class and the properties file(s), and you add
this jar file to your CLASSPATH. Then, when application-level code invokes
createEndpoint("foo:...") on a CamelContext object, Camel will find the "foo""
properties file on the CLASSPATH, get the value of the class property from that properties
file, and use reflection APIs to create an instance of the specified class.
As I said in Section 4.1 ("Endpoint"), Camel provides out-of-the-box support for numerous
communication technologies. The out-of-the-box support consists of classes that implement the
Component interface plus properties files that enable a CamelContext object to populate
its map of named Component objects.
Earlier in this section I gave the following example of calling
CamelContext.getEndpoint().

13

CH A PTE R 3 - GE T T I NG S T A R T E D W I T H A PACH E CAM EL

myCamelContext.getEndpoint("pop3://john.smith@mailserv.example.com?password=myPassword");

When I originally gave that example, I said that the parameter to getEndpoint() was a URI.
I said that because the online Camel documentation and the Camel source code both claim the
parameter is a URI. In reality, the parameter is restricted to being a URL. This is because when
Camel extracts the component name from the parameter, it looks for the first ":", which is a
simplistic algorithm. To understand why, recall from Section 4.4 ("The Meaning of URL, URI,
URN and IRI") that a URI can be a URL or a URN. Now consider the following calls to
getEndpoint.
myCamelContext.getEndpoint("pop3:...");
myCamelContext.getEndpoint("jms:...");
myCamelContext.getEndpoint("urn:foo:...");
myCamelContext.getEndpoint("urn:bar:...");

Camel identifies the components in the above example as "pop3", "jms", "urn" and "urn". It
would be more useful if the latter components were identified as "urn:foo" and "urn:bar" or,
alternatively, as "foo" and "bar" (that is, by skipping over the "urn:" prefix). So, in practice you
must identify an endpoint with a URL (a string of the form ":...") rather than with a
URN (a string of the form "urn::..."). This lack of proper support for URNs means the
you should consider the parameter to getEndpoint() as being a URL rather than (as
claimed) a URI.
Message and Exchange
The Message interface provides an abstraction for a single message, such as a request, reply
or exception message.
There are concrete classes that implement the Message interface for each Camel-supported
communications technology. For example, the JmsMessage class provides a JMS-specific
implementation of the Message interface. The public API of the Message interface provides
get- and set-style methods to access the message id, body and individual header fields of a
messge.
The Exchange interface provides an abstraction for an exchange of messages, that is, a
request message and its corresponding reply or exception message. In Camel terminology, the
request, reply and exception messages are called in, out and fault messages.
There are concrete classes that implement the Exchange interface for each Camel-supported
communications technology. For example, the JmsExchange class provides a JMS-specific
implementation of the Exchange interface. The public API of the Exchange interface is quite
limited. This is intentional, and it is expected that each class that implements this interface will
provide its own technology-specific operations.
Application-level programmers rarely access the Exchange interface (or classes that
implement it) directly. However, many classes in Camel are generic types that are instantiated

C H A P T E R 3 - G ET T IN G ST AR T ED W IT H APACH E CAM EL

14

on (a class that implements) Exchange. Because of this, the Exchange interface appears a
lot in the generic signatures of classes and methods.
Processor
The Processor interface represents a class that processes a message. The signature of this
interface is shown below.
Listing 2. Processor
package org.apache.camel;
public interface Processor {
void process(Exchange exchange) throws Exception;
}

Notice that the parameter to the process() method is an Exchange rather than a
Message. This provides flexibility. For example, an implementation of this method initially
might call exchange.getIn() to get the input message and process it. If an error occurs
during processing then the method can call exchange.setException().
An application-level developer might implement the Processor interface with a class that
executes some business logic. However, there are many classes in the Camel library that
implement the Processor interface in a way that provides support for a design pattern in the
EIP book. For example, ChoiceProcessor implements the message router pattern, that is, it
uses a cascading if-then-else statement to route a message from an input queue to one of
several output queues. Another example is the FilterProcessor class which discards
messages that do not satisfy a stated predicate (that is, condition).
Routes, RouteBuilders and Java DSL
A route is the step-by-step movement of a Message from an input queue, through arbitrary
types of decision making (such as filters and routers) to a destination queue (if any). Camel
provides two ways for an application developer to specify routes. One way is to specify route
information in an XML file. A discussion of that approach is outside the scope of this document.
The other way is through what Camel calls a Java DSL (domain-specific language).

Introduction to Java DSL
For many people, the term "domain-specific language" implies a compiler or interpreter that can
process an input file containing keywords and syntax specific to a particular domain. This is not
the approach taken by Camel. Camel documentation consistently uses the term "Java DSL"
instead of "DSL", but this does not entirely avoid potential confusion. The Camel "Java DSL" is a
class library that can be used in a way that looks almost like a DSL, except that it has a bit of
Java syntactic baggage. You can see this in the example below. Comments afterwards explain
some of the constructs used in the example.

15

CH A PTE R 3 - GE T T I NG S T A R T E D W I T H A PACH E CAM EL

Listing 3. Example of Camel's "Java DSL"
RouteBuilder builder = new RouteBuilder() {
public void configure() {
from("queue:a").filter(header("foo").isEqualTo("bar")).to("queue:b");
from("queue:c").choice()
.when(header("foo").isEqualTo("bar")).to("queue:d")
.when(header("foo").isEqualTo("cheese")).to("queue:e")
.otherwise().to("queue:f");
}
};
CamelContext myCamelContext = new DefaultCamelContext();
myCamelContext.addRoutes(builder);

The first line in the above example creates an object which is an instance of an anonymous
subclass of RouteBuilder with the specified configure() method.
The CamelContext.addRoutes(RouterBuilder builder) method invokes
builder.setContext(this) – so the RouteBuilder object knows which
CamelContext object it is associated with – and then invokes builder.configure().
The body of configure() invokes methods such as from(), filter(), choice(),
when(), isEqualTo(), otherwise() and to().
The RouteBuilder.from(String uri) method invokes getEndpoint(uri) on the
CamelContext associated with the RouteBuilder object to get the specified Endpoint
and then puts a FromBuilder "wrapper" around this Endpoint. The
FromBuilder.filter(Predicate predicate) method creates a
FilterProcessor object for the Predicate (that is, condition) object built from the
header("foo").isEqualTo("bar") expression. In this way, these operations
incrementally build up a Route object (with a RouteBuilder wrapper around it) and add it
to the CamelContext object associated with the RouteBuilder.

Critique of Java DSL
The online Camel documentation compares Java DSL favourably against the alternative of
configuring routes and endpoints in a XML-based Spring configuration file. In particular, Java
DSL is less verbose than its XML counterpart. In addition, many integrated development
environments (IDEs) provide an auto-completion feature in their editors. This auto-completion
feature works with Java DSL, thereby making it easier for developers to write Java DSL.
However, there is another option that the Camel documentation neglects to consider: that of
writing a parser that can process DSL stored in, say, an external file. Currently, Camel does not
provide such a DSL parser, and I do not know if it is on the "to do" list of the Camel
maintainers. I think that a DSL parser would offer a significant benefit over the current Java
DSL. In particular, the DSL would have a syntactic definition that could be expressed in a
relatively short BNF form. The effort required by a Camel user to learn how to use DSL by
reading this BNF would almost certainly be significantly less than the effort currently required
to study the API of the RouterBuilder classes.
C H A P T E R 3 - G ET T IN G ST AR T ED W IT H APACH E CAM EL

16

Want to know more about Camel
The Camel in Action book, contains a free chapter 1, which is highly recommended to read to
get more familiar with Camel. After all three authors have worked on this chapter, it has been
reviewed extensively at four different occasions by a professional review panel, and had two
passes of copy editing. What we are saying is that it is 100+ hours of work which nobody every
would voluntarily would have used to write on these wiki pages about Camel.

17

CH A PTE R 3 - GE T T I NG S T A R T E D W I T H A PACH E CAM EL

CHAPTER

4

°°°°

Architecture

Camel uses a Java based Routing Domain Specific Language (DSL) or an Xml Configuration to
configure routing and mediation rules which are added to a CamelContext to implement the
various Enterprise Integration Patterns.
At a high level Camel consists of a CamelContext which contains a collection of Component
instances. A Component is essentially a factory of Endpoint instances. You can explicitly
configure Component instances in Java code or an IoC container like Spring or Guice, or they
can be auto-discovered using URIs.
An Endpoint acts rather like a URI or URL in a web application or a Destination in a JMS
system; you can communicate with an endpoint; either sending messages to it or consuming
messages from it. You can then create a Producer or Consumer on an Endpoint to exchange
messages with it.
The DSL makes heavy use of pluggable Languages to create an Expression or Predicate to
make a truly powerful DSL which is extensible to the most suitable language depending on your
needs. The following languages are supported
• Bean Language for using Java for expressions
• Constant
• the unified EL from JSP and JSF
• Header
• JXPath
• Mvel
• OGNL
• Property
• Scala DSL
• Scripting Languages such as
◦ BeanShell
◦ JavaScript
◦ Groovy
◦ Python
◦ PHP
◦ Ruby
• Simple
◦ File Language
• SQL
• XPath
CH APT ER 4 - AR CH IT ECT UR E

18

• XQuery
Most of these languages is also supported used as Annotation Based Expression Language.
For a full details of the individual languages see the Language Appendix

URIS
Camel makes extensive use of URIs to allow you to refer to endpoints which are lazily created
by a Component if you refer to them within Routes
Current Supported URIs
Component / ArtifactId / URI

Description

ActiveMQ / activemq-camel
activemq:[topic:]destinationName

ActiveMQ Journal / activemq-core
activemq.journal:directory-on-filesystem

For JMS Messaging with Apache
ActiveMQ

Uses ActiveMQ's fast disk
journaling implementation to store
message bodies in a rolling log file

AMQP / camel-amqp
amqp:[topic:]destinationName

Atom / camel-atom
atom:uri

Bean / camel-core
bean:beanName[?method=someMethod]

Bean Validation / camel-bean-validator
bean-validator:something

19

CH A PTE R 4 - AR C H I T E C T U R E

For Messaging with AMQP
protocol

Working with Apache Abdera for
atom integration, such as
consuming an atom feed.
Uses the Bean Binding to bind
message exchanges to beans in the
Registry. Is also used for exposing
and invoking POJO (Plain Old Java
Objects).
Validates the payload of a message
using the Java Validation API (JSR
303 and JAXP Validation) and its
reference implementation
Hibernate Validator

Browse / camel-core
browse:someName

Cache / camel-cache
cache://cachename[?options]

Class / camel-core
class:className[?method=someMethod]

Cometd / camel-cometd
cometd://host:port/channelname

Provides a simple
BrowsableEndpoint which can be
useful for testing, visualisation tools
or debugging. The exchanges sent
to the endpoint are all available to
be browsed.
The cache component facilitates
creation of caching endpoints and
processors using EHCache as the
cache implementation.
Uses the Bean Binding to bind
message exchanges to beans in the
Registry. Is also used for exposing
and invoking POJO (Plain Old Java
Objects).
Used to deliver messages using the
jetty cometd implementation of the
bayeux protocol

Crypto (Digital Signatures) / camel-crypto
crypto:sign:name[?options]
crypto:verify:name[?options]

Used to sign and verify exchanges
using the Signature Service of the
Java Cryptographic Extension.

CXF / camel-cxf
cxf:address[?serviceClass=...]

Working with Apache CXF for
web services integration

CXFRS / camel-cxf
cxfrs:address[?resourcesClasses=...]

DataSet / camel-core
dataset:name

Working with Apache CXF for
REST services integration
For load & soak testing the DataSet
provides a way to create huge
numbers of messages for sending
to Components or asserting that
they are consumed correctly

CH APT ER 4 - AR CH IT ECT UR E

20

Direct / camel-core
direct:name

EJB / camel-ejb
ejb:ejbName[?method=someMethod]

Synchronous call to another
endpoint
Uses the Bean Binding to bind
message exchanges to EJBs. It
works like the Bean component
but just for accessing EJBs.
Supports EJB 3.0 onwards.

Esper / camel-esper in camel-extra
esper:name

Working with the Esper Library for
Event Stream Processing

Event / camel-spring
event://default
spring-event://default

Working with Spring
ApplicationEvents

Exec / camel-exec
exec://executable[?options]

File / camel-core
file://nameOfFileOrDirectory

Flatpack / camel-flatpack
flatpack:[fixed|delim]:configFile

For executing system commands

Sending messages to a file or
polling a file or directory. Camel
1.x use this link File.
Processing fixed width or delimited
files or messages using the FlatPack
library

Freemarker / camel-freemarker
freemarker:someTemplateResource

FTP / camel-ftp
ftp://host[:port]/fileName

21

CH A PTE R 4 - AR C H I T E C T U R E

Generates a response using a
Freemarker template

Sending and receiving files over
FTP. Camel 1.x use this link
FTP.

FTPS / camel-ftp
ftps://host[:port]/fileName

GAuth / camel-gae
gauth://name[?options]

GHttp / camel-gae
ghttp://hostname[:port][/path][?options]
ghttp:///path[?options]

GLogin / camel-gae
glogin://hostname[:port][?options]

GTask / camel-gae
gtask://queue-name

GMail / camel-gae
gmail://user@gmail.com[?options]
gmail://user@googlemail.com[?options]

Sending and receiving files over FTP
Secure (TLS and SSL).
Used by web applications to
implement an OAuth consumer.
See also Camel Components for
Google App Engine.
Provides connectivity to the URL
fetch service of Google App Engine
but can also be used to receive
messages from servlets. See also
Camel Components for Google
App Engine.
Used by Camel applications outside
Google App Engine (GAE) for
programmatic login to GAE
applications. See also Camel
Components for Google App
Engine.
Supports asynchronous message
processing on Google App Engine
by using the task queueing service
as message queue. See also Camel
Components for Google App
Engine.
Supports sending of emails via the
mail service of Google App Engine.
See also Camel Components for
Google App Engine.

HDFS / camel-hdfs in camel-hdfs
hdfs://path[?options]

For reading/writing from/to an
HDFS filesystem

Hibernate / camel-hibernate in camel-extra
hibernate://entityName

For using a database as a queue via
the Hibernate library

CH APT ER 4 - AR CH IT ECT UR E

22

HL7 / camel-hl7
mina:tcp://hostname[:port]

For working with the HL7 MLLP
protocol and the HL7 model using
the HAPI library

HTTP / camel-http
http://hostname[:port]

iBATIS / camel-ibatis
ibatis://sqlOperationName

For calling out to external HTTP
servers

Performs a query, poll, insert,
update or delete in a relational
database using Apache iBATIS

IMap / camel-mail
imap://hostname[:port]

Receiving email using IMap

IRC / camel-irc
irc:host[:port]/#room

For IRC communication

JavaSpace / camel-javaspace
javaspace:jini://host?spaceName=mySpace?...

Sending and receiving messages
through JavaSpace

JBI / servicemix-camel
jbi:serviceName

JCR / camel-jcr
jcr://user:password@repository/path/to/node

For JBI integration such as working
with Apache ServiceMix

Storing a message in a JCR
(JSR-170) compliant repository like
Apache Jackrabbit

JDBC / camel-jdbc
jdbc:dataSourceName?options

23

CH A PTE R 4 - AR C H I T E C T U R E

For performing JDBC queries and
operations

Jetty / camel-jetty
jetty:url

For exposing services over HTTP

JMS / camel-jms
jms:[topic:]destinationName

JPA / camel-jpa
jpa://entityName

JT/400 / camel-jt400
jt400://user:pwd@system/

LDAP / camel-ldap
ldap:host[:port]?base=...[&scope=]

Log / camel-core
log:loggingCategory[?level=ERROR]

Lucene / camel-lucene
lucene:searcherName:insert[?analyzer=]
lucene:searcherName:query[?analyzer=]

Working with JMS providers

For using a database as a queue via
the JPA specification for working
with OpenJPA, Hibernate or
TopLink
For integrating with data queues on
an AS/400 (aka System i, IBM i, i5,
...) system
Performing searches on LDAP
servers ( must be one of
object|onelevel|subtree)
Uses Jakarta Commons Logging to
log the message exchange to some
underlying logging system like log4j
Uses Apache Lucene to perform
Java-based indexing and full text
based searches using advanced
analysis/tokenization capabilities

Mail / camel-mail
mail://user-info@host:port

Sending and receiving email

MINA / camel-mina
[tcp|udp|vm]:host[:port]

Working with Apache MINA

CH APT ER 4 - AR CH IT ECT UR E

24

Mock / camel-core
mock:name

For testing routes and mediation
rules using mocks

MSV / camel-msv
msv:someLocalOrRemoteResource

Validates the payload of a message
using the MSV Library

Nagios / camel-nagios
nagios://host[:port]?options

Netty / camel-netty
netty:tcp//host[:port]?options
netty:udp//host[:port]?options

NMR / servicemix-nmr
nmr://serviceName

Sending passive checks to Nagios
using JSendNSCA

Working with TCP and UDP
protocols using Java NIO based
capabilities offered by the JBoss
Netty community project
Integration with the Normalized
Message Router BUS in ServiceMix
4.x

POP / camel-mail
pop3://user-info@host:port

Printer / camel-printer
lpr://host:port/path/to/printer[?options]

Properties / camel-core
properties://key[?options]

25

CH A PTE R 4 - AR C H I T E C T U R E

Receiving email using POP3 and
JavaMail
The printer component facilitates
creation of printer endpoints to
local, remote and wireless printers.
The endpoints provide the ability
to print camel directed payloads
when utilized on camel routes.
The properties component
facilitates using property
placeholders directly in endpoint
uri definitions.

Quartz / camel-quartz
quartz://groupName/timerName

Provides a scheduled delivery of
messages using the Quartz
scheduler

Quickfix / camel-quickfix
quickfix-server:config file
quickfix-client:config-file

Implementation of the QuickFix for
Java engine which allow to send/
receive FIX messages

Ref / camel-core
ref:name

Restlet / camel-restlet
restlet:restletUrl[?options]

Component for lookup of existing
endpoints bound in the Registry.

Component for consuming and
producing Restful resources using
Restlet

RMI / camel-rmi
rmi://host[:port]

Working with RMI

RNC / camel-jing
rnc:/relativeOrAbsoluteUri

Validates the payload of a message
using RelaxNG Compact Syntax

RNG / camel-jing
rng:/relativeOrAbsoluteUri

RSS / camel-rss
rss:uri

Validates the payload of a message
using RelaxNG

Working with ROME for RSS
integration, such as consuming an
RSS feed.

Scalate / scalate-camel
scalate:templateName

Uses the given Scalate template to
transform the message

CH APT ER 4 - AR CH IT ECT UR E

26

SEDA / camel-core
seda:name

SERVLET / camel-servlet
servlet:uri

SFTP / camel-ftp
sftp://host[:port]/fileName

Asynchronous call to another
endpoint in the same Camel
Context
For exposing services over HTTP
through the servlet which is
deployed into the Web container.
Sending and receiving files over
SFTP (FTP over SSH). Camel 1.x
use this link FTP.

Smooks / camel-smooks in camel-extra
unmarshal(edi)

For working with EDI parsing using
the Smooks library

SMTP / camel-mail
smtp://user-info@host[:port]

SMPP / camel-smpp
smpp://user-info@host[:port]?options

SNMP / camel-snmp
snmp://host[:port]?options

Sending email using SMTP and
JavaMail

To send and receive SMS using
Short Messaging Service Center
using the JSMPP library
Polling OID values and receiving
traps using SNMP via SNMP4J
library

SpringIntegration / camel-spring-integration
spring-integration:defaultChannelName

The bridge component of Camel
and Spring Integration

SQL / camel-sql
sql:select * from table where id=#

27

CH A PTE R 4 - AR C H I T E C T U R E

Performing SQL queries using
JDBC

Stream / camel-stream
stream:[in|out|err|file]

Read or write to an input/output/
error/file stream rather like unix
pipes

StringTemplate / camel-stringtemplate
string-template:someTemplateResource

Generates a response using a String
Template

TCP / camel-mina
mina:tcp://host:port

Test / camel-spring
test:expectedMessagesEndpointUri

Working with TCP protocols using
Apache MINA
Creates a Mock endpoint which
expects to receive all the message
bodies that could be polled from
the given underlying endpoint

Timer / camel-core
timer://name

A timer endpoint

UDP / camel-mina
mina:udp://host:port

Validation / camel-spring
validation:someLocalOrRemoteResource

Working with UDP protocols using
Apache MINA

Validates the payload of a message
using XML Schema and JAXP
Validation

Velocity / camel-velocity
velocity:someTemplateResource

Generates a response using an
Apache Velocity template

VM / camel-core
vm:name

Asynchronous call to another
endpoint in the same JVM

CH APT ER 4 - AR CH IT ECT UR E

28

XMPP / camel-xmpp
xmpp://host:port/room

Working with XMPP and Jabber

XQuery / camel-saxon
xquery:someXQueryResource

Generates a response using an
XQuery template

XSLT / camel-spring
xslt:someTemplateResource

Generates a response using an
XSLT template

For a full details of the individual components see the Component Appendix

29

CH A PTE R 4 - AR C H I T E C T U R E

CHAPTER

5

°°°°

Enterprise Integration Patterns

Camel supports most of the Enterprise Integration Patterns from the excellent book of the
same name by Gregor Hohpe and Bobby Woolf. Its a highly recommended book, particularly
for users of Camel.

PATTERN INDEX
There now follows a list of the Enterprise Integration Patterns from the book along with
examples of the various patterns using Apache Camel
Messaging Systems
Message
Channel

How does one application communicate with another using
messaging?

Message

How can two applications connected by a message channel
exchange a piece of information?

Pipes and
Filters

How can we perform complex processing on a message while
maintaining independence and flexibility?

Message
Router

How can you decouple individual processing steps so that
messages can be passed to different filters depending on a set of
conditions?

Message
Translator

How can systems using different data formats communicate with
each other using messaging?

Message
Endpoint

How does an application connect to a messaging channel to send
and receive messages?

C H A P T E R 5 - EN T ER PR ISE IN T EGR AT IO N PAT T ER N S

30

Messaging Channels
Point to
Point
Channel

How can the caller be sure that exactly one receiver will receive
the document or perform the call?

Publish
Subscribe
Channel

How can the sender broadcast an event to all interested
receivers?

Dead
Letter
Channel

What will the messaging system do with a message it cannot
deliver?

Guaranteed
Delivery

How can the sender make sure that a message will be delivered,
even if the messaging system fails?

Message
Bus

What is an architecture that enables separate applications to
work together, but in a de-coupled fashion such that applications
can be easily added or removed without affecting the others?

Message Construction
Event Message

How can messaging be used to transmit events from one
application to another?

Request Reply

When an application sends a message, how can it get a
response from the receiver?

Correlation
Identifier

How does a requestor that has received a reply know which
request this is the reply for?

Return
Address

How does a replier know where to send the reply?

Message Routing

31

Content
Based
Router

How do we handle a situation where the implementation of a
single logical function (e.g., inventory check) is spread across
multiple physical systems?

Message
Filter

How can a component avoid receiving uninteresting messages?

Dynamic
Router

How can you avoid the dependency of the router on all
possible destinations while maintaining its efficiency?

CH A PTE R 5 - E N T E R P R I SE I NT E G R A T I ON P AT T ER N S

Recipient
List

How do we route a message to a list of (static or dynamically)
specified recipients?

Splitter

How can we process a message if it contains multiple
elements, each of which may have to be processed in a
different way?

Aggregator

How do we combine the results of individual, but related
messages so that they can be processed as a whole?

Resequencer

How can we get a stream of related but out-of-sequence
messages back into the correct order?

Composed
Message
Processor

How can you maintain the overall message flow when
processing a message consisting of multiple elements, each of
which may require different processing?

ScatterGather

How do you maintain the overall message flow when a
message needs to be sent to multiple recipients, each of which
may send a reply?

Routing Slip

How do we route a message consecutively through a series of
processing steps when the sequence of steps is not known at
design-time and may vary for each message?

Throttler

How can I throttle messages to ensure that a specific endpoint
does not get overloaded, or we don't exceed an agreed SLA
with some external service?

Sampling

How can I sample one message out of many in a given period
to avoid downstream route does not get overloaded?

Delayer

How can I delay the sending of a message?

Load
Balancer

How can I balance load across a number of endpoints?

Multicast

How can I route a message to a number of endpoints at the
same time?

Loop

How can I repeat processing a message in a loop?

Message Transformation
Content
Enricher

How do we communicate with another system if the message
originator does not have all the required data items available?

C H A P T E R 5 - EN T ER PR ISE IN T EGR AT IO N PAT T ER N S

32

Content
Filter

How do you simplify dealing with a large message, when you are
interested only in a few data items?

Claim
Check

How can we reduce the data volume of message sent across the
system without sacrificing information content?

Normalizer

How do you process messages that are semantically equivalent,
but arrive in a different format?

Sort

How can I sort the body of a message?

Validate

How can I validate a message?

Messaging Endpoints

33

Messaging
Mapper

How do you move data between domain objects and the
messaging infrastructure while keeping the two independent of
each other?

Event Driven
Consumer

How can an application automatically consume messages as they
become available?

Polling
Consumer

How can an application consume a message when the
application is ready?

Competing
Consumers

How can a messaging client process multiple messages
concurrently?

Message
Dispatcher

How can multiple consumers on a single channel coordinate
their message processing?

Selective
Consumer

How can a message consumer select which messages it wishes
to receive?

Durable
Subscriber

How can a subscriber avoid missing messages while it's not
listening for them?

Idempotent
Consumer

How can a message receiver deal with duplicate messages?

Transactional
Client

How can a client control its transactions with the messaging
system?

Messaging
Gateway

How do you encapsulate access to the messaging system from
the rest of the application?

CH A PTE R 5 - E N T E R P R I SE I NT E G R A T I ON P AT T ER N S

How can an application design a service to be invoked both via
various messaging technologies and via non-messaging
techniques?

Service
Activator

System Management
Detour

How can you route a message through intermediate steps to
perform validation, testing or debugging functions?

Wire
Tap

How do you inspect messages that travel on a point-to-point
channel?

Log

How can I log processing a message?

For a full breakdown of each pattern see the Book Pattern Appendix

C H A P T E R 5 - EN T ER PR ISE IN T EGR AT IO N PAT T ER N S

34

CookBook

This document describes various recipes for working with Camel
• Bean Integration describes how to work with beans and Camel in a loosely coupled
way so that your beans do not have to depend on any Camel APIs
◦ Annotation Based Expression Language binds expressions to method
parameters
◦ Bean Binding defines which methods are invoked and how the Message is
converted into the parameters of the method when it is invoked
◦ Bean Injection for injecting Camel related resources into your POJOs
◦ Parameter Binding Annotations for extracting various headers, properties
or payloads from a Message
◦ POJO Consuming for consuming and possibly routing messages from Camel
◦ POJO Producing for producing camel messages from your POJOs
◦ RecipientList Annotation for creating a Recipient List from a POJO method
◦ Using Exchange Pattern Annotations describes how pattern annotations can
be used to change the behaviour of method invocations
• Hiding Middleware describes how to avoid your business logic being coupled to any
particular middleware APIs allowing you to easily switch from in JVM SEDA to JMS,
ActiveMQ, Hibernate, JPA, JDBC, iBATIS or JavaSpace etc.
• Visualisation describes how to visualise your Enterprise Integration Patterns to help
you understand your routing rules
• Business Activity Monitoring (BAM) for monitoring business processes across systems
• Extract Transform Load (ETL) to load data into systems or databases
• Testing for testing distributed and asynchronous systems using a messaging approach
◦ Camel Test for creating test cases using a single Java class for all your
configuration and routing
◦ Spring Testing uses Spring Test together with either XML or Java Config to
dependency inject your test classes
◦ Guice uses Guice to dependency inject your test classes
• Templating is a great way to create service stubs to be able to test your system
without some back end system.
• Database for working with databases
• Parallel Processing and Ordering on how using parallel processing and SEDA or JMS
based load balancing can be achieved.
• Asynchronous Processing in Camel Routes.
• Implementing Virtual Topics on other JMS providers shows how to get the effect of
Virtual Topics and avoid issues with JMS durable topics
• Camel Transport for CXF describes how to put the Camel context into the CXF
transport layer.

35

COO K BO O K

• Fine Grained Control Over a Channel describes how to deliver a sequence of
messages over a single channel and then stopping any more messages being sent over
that channel. Typically used for sending data over a socket and then closing the
socket.
• EventNotifier to log details about all sent Exchanges shows how to let Camels
EventNotifier log all sent to endpoint events and how long time it took.

BEAN INTEGRATION
Camel supports the integration of beans and POJOs in a number of ways
Bean Binding
Whenever Camel invokes a bean method, either via the Bean component, Spring Remoting or
POJO Consuming then the Bean Binding mechanism is used to figure out what method to use
(if it is not explicit) and how to bind the Message to the parameters possibly using the
Parameter Binding Annotations
Annotations
If a bean is defined in Spring XML or scanned using the Spring 2.5 component scanning
mechanism and a  is used or a CamelBeanPostProcessor then we process a
number of Camel annotations to do various things such as injecting resources or producing,
consuming or routing messages.
• POJO Consuming to consume and possibly route messages from Camel
• POJO Producing to make it easy to produce camel messages from your POJOs
• RecipientList Annotation for creating a Recipient List from a POJO method
• RoutingSlip Annotation for creating a Routing Slip for a POJO method
• Bean Injection to inject Camel related resources into your POJOs
• Using Exchange Pattern Annotations describes how the pattern annotations can be
used to change the behaviour of method invocations with Spring Remoting or POJO
Producing
Spring Remoting
We support a Spring Remoting provider which uses Camel as the underlying transport
mechanism. The nice thing about this approach is we can use any of the Camel transport
Components to communicate between beans. It also means we can use Content Based Router
and the other Enterprise Integration Patterns in between the beans; in particular we can use
Message Translator to be able to convert what the on-the-wire messages look like in addition
to adding various headers and so forth.

CO O K B O O K

36

Bean Component
The Bean component supports the creation of a proxy via ProxyHelper to a Java interface;
which the implementation just sends a message containing a BeanInvocation to some Camel
endpoint.
Then there is a server side implementation which consumes a message and uses the Bean
Binding to bind the message to invoke a method passing in its parameters.
Annotation Based Expression Language
You can also use any of the Languages supported in Camel to bind expressions to method
parameters when using Bean Integration. For example you can use any of these annotations:
Annotation

Description

@Bean

Inject a Bean expression

@BeanShell

Inject a BeanShell expression

@Constant

Inject a Constant expression

@EL

Inject an EL expression

@Groovy

Inject a Groovy expression

@Header

Inject a Header expression

@JavaScript

Inject a JavaScript expression

@MVEL

Inject a Mvel expression

@OGNL

Inject an OGNL expression

@PHP

Inject a PHP expression

@Python

Inject a Python expression

@Ruby

Inject a Ruby expression

@Simple

Inject an Simple expression

@XPath

Inject an XPath expression

@XQuery

Inject an XQuery expression

Example:
public class Foo {
@MessageDriven(uri = "activemq:my.queue")
public void doSomething(@XPath("/foo/bar/text()") String correlationID, @Body

37

COO K BO O K

String body) {
// process the inbound message here
}
}

Advanced example using @Bean
And an example of using the the @Bean binding annotation, where you can use a Pojo where
you can do whatever java code you like:
public class Foo {
@MessageDriven(uri = "activemq:my.queue")
public void doSomething(@Bean("myCorrelationIdGenerator") String correlationID,
@Body String body) {
// process the inbound message here
}
}

And then we can have a spring bean with the id myCorrelationIdGenerator where we
can compute the id.
public class MyIdGenerator {
private UserManager userManager;
public String generate(@Header(name = "user") String user, @Body String payload)
throws Exception {
User user = userManager.lookupUser(user);
String userId = user.getPrimaryId();
String id = userId + generateHashCodeForPayload(payload);
return id;
}
}

The Pojo MyIdGenerator has one public method that accepts two parameters. However we
have also annotated this one with the @Header and @Body annotation to help Camel know
what to bind here from the Message from the Exchange being processed.
Of course this could be simplified a lot if you for instance just have a simple id generator.
But we wanted to demonstrate that you can use the Bean Binding annotations anywhere.
public class MySimpleIdGenerator {
public static int generate()
// generate a unique id
return 123;

{

CO O K B O O K

38

}
}

And finally we just need to remember to have our bean registered in the Spring Registry:


Example using Groovy
In this example we have an Exchange that has a User object stored in the in header. This User
object has methods to get some user information. We want to use Groovy to inject an
expression that extracts and concats the fullname of the user into the fullName parameter.
public void doSomething(@Groovy("$request.header['user'].firstName
$request.header['user'].familyName) String fullName, @Body String body) {
// process the inbound message here
}

Groovy supports GStrings that is like a template where we can insert $ placeholders that will
be evaluated by Groovy.
Bean Binding
The Bean Binding in Camel defines both which methods are invoked and also how the Message
is converted into the parameters of the method when it is invoked.

Choosing the method to invoke
The binding of a Camel Message to a bean method call can occur in different ways, order if
importance:
• if the message contains the header CamelBeanMethodName
(org.apache.camel.MethodName in Camel 1.x) then that method is invoked,
converting the body to whatever the argument is to the method
• the method name can be specified explicitly in the DSL or when using POJO
Consuming
• Camel 2.0: if the bean has a method that is marked with @Handler annotation
then that method is selected
• if the bean can be converted to a Processor using the Type Converter mechanism
then this is used to process the message. This mechanism is used by the ActiveMQ
component to allow any JMS MessageListener to be invoked directly by Camel

39

COO K BO O K

without having to write any integration glue code. You can use the same mechanism
to integrate Camel into any other messaging/remoting frameworks.
• if the body of the message can be converted to a BeanInvocation (the default payload
used by the ProxyHelper) - then that its used to invoke the method and pass the
arguments
• otherwise the type of the method body is used to try find a method which matches;
an error is thrown if a single method cannot be chosen unambiguously.
• you can also use Exchange as the parameter itself, but then the return type must be
void.
In case where Camel will not be able to choose a method to invoke an
AmbiguousMethodCallException is thrown.
By default the return value is set on the outbound message body.

Parameter binding
When a method have been chosen to be invoked Camel will bind to the parameters of the
method.
The following Camel specific types is automatic binded:
▪ org.apache.camel.Exchange
▪ org.apache.camel.Message
▪ Camel 2.0: org.apache.camel.CamelContext
▪ org.apache.camel.TypeConverter
▪ Camel 2.0: org.apache.camel.spi.Registry
▪ java.lang.Exception
So if you declare any of the given type above they will be provided by Camel. A note on the
Exception is that it will bind to the caught exception of the Exchange. So its often usable if
you use a Pojo to handle a given using using eg an onException route.
What is most interresting is that Camel will also try to bind the body of the Exchange to the
first parameter of the method signature (albeit not of any of the types above). So if we for
instance declare e parameter as: String body then Camel will bind the IN body to this type.
Camel will also automatic type convert to the given type declared.
Okay lets show some examples.
Below is just a simple method with a body binding. Camel will bind the IN body to the body
parameter and convert it to a String type.
public String doSomething(String body)

And in this sample we got one of the automatic binded type as well, for instance the
Registry that we can use to lookup beans.

CO O K B O O K

40

public String doSomething(String body, Registry registry)

And we can also use Exchange as well:
public String doSomething(String body, Exchange exchange)

You can have multiple types as well
public String doSomething(String body, Exchange exchange, TypeConverter converter)

And imagine you use a Pojo to handle a given custom exception InvalidOrderException
then we can bind that as well:
Notice we can bind to it even if we use a sub type of java.lang.Exception as Camel still
knows its an exception and thus can bind the caused exception (if any exists).
public String badOrder(String body, InvalidOrderException invalid)

So what about headers and other stuff? Well now it gets a bit tricky so we can use annotations
to help us. See next section for details.

Binding Annotations
You can use the Parameter Binding Annotations to customize how parameter values are
created from the Message

Examples
For example a Bean such as:
public class Bar {
public String doSomething(String body) {
// process the in body and return whatever you want
return "Bye World";
}

Or the Exchange example. Notice that the return type must be void when there is only a
single parameter:
public class Bar {
public void doSomething(Exchange exchange) {

41

COO K BO O K

// process the exchange
exchange.getIn().setBody("Bye World");
}

@Handler
Available as of Camel 2.0
You can mark a method in your bean with the @Handler annotation to indicate that this
method should be used for Bean Binding.
This has the advantage as you do not have to specify the method name in the Camel route. And
thus you do not run into problems when you rename the method name using an IDE that don't
find all references.
public class Bar {
@Handler
public String doSomething(String body) {
// process the in body and return whatever you want
return "Bye World";
}

POJO consuming
For example you could use POJO Consuming to write a bean like this
public class Foo {
@Consume(uri = "activemq:my.queue")
public void doSomething(String body) {
// process the inbound message here
}
}

Here Camel with subscribe to an ActiveMQ queue, then convert the message payload to a
String (so dealing with TextMessage, ObjectMessage and BytesMessage in JMS), then process
this method.
Bean Injection
We support the injection of various resources using @EndpointInject. This can be used to
inject

CO O K B O O K

42

@Consume requires camel-spring
Using the @Consume annotations requires camel-spring that uses the
org.apache.camel.spring.CamelBeanPostProcessor to perform the
setup for this consumer and the needed bean bindings.

@MessageDriven is @deprecated
The @MessageDriven has been replaced with @Consume in Camel 1.5.0 or newer.
Its now marked as @deprecated and will be removed in Camel 2.0.
• Endpoint instances which can be used for testing when used with Mock endpoints; see
the Spring Testing for an example.
• ProducerTemplate instances for POJO Producing
• client side proxies for POJO Producing which is a simple approach to Spring
Remoting
Parameter Binding Annotations
Annotations can be used to define an Expression or to extract various headers, properties or
payloads from a Message when invoking a bean method (see Bean Integration for more detail of
how to invoke bean methods) together with being useful to help disambiguate which method to
invoke.
If no annotations are used then Camel assumes that a single parameter is the body of the
message. Camel will then use the Type Converter mechanism to convert from the expression
value to the actual type of the parameter.
The core annotations are as follows

43

Annotation

Meaning

@Body

To bind to an inbound message body

@ExchangeException

To bind to an Exception set on the exchange
(Camel 2.0)

@Header

To bind to an inbound message header

@Headers

To bind to the Map of the inbound message
headers

@OutHeaders

To bind to the Map of the outbound message
headers

COO K BO O K

Parameter

String name
of the header

camel-core
The annotations below are all part of camel-core and thus does not require
camel-spring or Spring. These annotations can be used with the Bean
component or when invoking beans in the DSL

@Property

To bind to a named property on the exchange

@Properties

To bind to the property map on the exchange

@Handler

Camel 2.0: Not part as a type parameter but
stated in this table anyway to spread the good word
that we have this annotation in Camel now. See
more at Bean Binding.

String name
of the
property

The follow annotations @Headers, @OutHeaders and @Properties binds to the backing
java.util.Map so you can alter the content of these maps directly, for instance using the
put method to add a new entry. See the OrderService class at Exception Clause for such an
example.
Example
In this example below we have a @Consume consumer (like message driven) that consumes
JMS messages from the activemq queue. We use the @Header and @Body parameter binding
annotations to bind from the JMSMessage to the method parameters.
public class Foo {
@Consume(uri = "activemq:my.queue")
public void doSomething(@Header(name = "JMSCorrelationID") String correlationID,
@Body String body) {
// process the inbound message here
}
}

In the above Camel will extract the value of Message.getJMSCorrelationID(), then using the
Type Converter to adapt the value to the type of the parameter if required - it will inject the
parameter value for the correlationID parameter. Then the payload of the message will be
converted to a String and injected into the body parameter.
You don't need to use the @Consume annotation; as you could use the Camel DSL to
route to the beans method

CO O K B O O K

44

Using the DSL to invoke the bean method
Here is another example which does not use POJO Consuming annotations but instead uses
the DSL to route messages to the bean method
public class Foo {
public void doSomething(@Header(name = "JMSCorrelationID") String correlationID,
@Body String body) {
// process the inbound message here
}
}

The routing DSL then looks like this
from("activemq:someQueue").
to("bean:myBean");

Here myBean would be looked up in the Registry (such as JNDI or the Spring
ApplicationContext), then the body of the message would be used to try figure out what
method to call.
If you want to be explicit you can use
from("activemq:someQueue").
to("bean:myBean?methodName=doSomething");

And here we have a nifty example for you to show some great power in Camel. You can mix
and match the annotations with the normal parameters, so we can have this example with
annotations and the Exchange also:
public void doSomething(@Header(name = "user") String user, @Body String body,
Exchange exchange) {
exchange.getIn().setBody(body + "MyBean");
}

Annotation Based Expression Language
You can also use any of the Languages supported in Camel to bind expressions to method
parameters when using Bean Integration. For example you can use any of these annotations:

45

Annotation

Description

@Bean

Inject a Bean expression

@BeanShell

Inject a BeanShell expression

@Constant

Inject a Constant expression

COO K BO O K

@EL

Inject an EL expression

@Groovy

Inject a Groovy expression

@Header

Inject a Header expression

@JavaScript

Inject a JavaScript expression

@MVEL

Inject a Mvel expression

@OGNL

Inject an OGNL expression

@PHP

Inject a PHP expression

@Python

Inject a Python expression

@Ruby

Inject a Ruby expression

@Simple

Inject an Simple expression

@XPath

Inject an XPath expression

@XQuery

Inject an XQuery expression

Example:
public class Foo {
@MessageDriven(uri = "activemq:my.queue")
public void doSomething(@XPath("/foo/bar/text()") String correlationID, @Body
String body) {
// process the inbound message here
}
}

Advanced example using @Bean
And an example of using the the @Bean binding annotation, where you can use a Pojo where
you can do whatever java code you like:
public class Foo {
@MessageDriven(uri = "activemq:my.queue")
public void doSomething(@Bean("myCorrelationIdGenerator") String correlationID,
@Body String body) {
// process the inbound message here
}
}

CO O K B O O K

46

And then we can have a spring bean with the id myCorrelationIdGenerator where we
can compute the id.
public class MyIdGenerator {
private UserManager userManager;
public String generate(@Header(name = "user") String user, @Body String payload)
throws Exception {
User user = userManager.lookupUser(user);
String userId = user.getPrimaryId();
String id = userId + generateHashCodeForPayload(payload);
return id;
}
}

The Pojo MyIdGenerator has one public method that accepts two parameters. However we
have also annotated this one with the @Header and @Body annotation to help Camel know
what to bind here from the Message from the Exchange being processed.
Of course this could be simplified a lot if you for instance just have a simple id generator.
But we wanted to demonstrate that you can use the Bean Binding annotations anywhere.
public class MySimpleIdGenerator {
public static int generate()
// generate a unique id
return 123;

{

}
}

And finally we just need to remember to have our bean registered in the Spring Registry:


Example using Groovy
In this example we have an Exchange that has a User object stored in the in header. This User
object has methods to get some user information. We want to use Groovy to inject an
expression that extracts and concats the fullname of the user into the fullName parameter.
public void doSomething(@Groovy("$request.header['user'].firstName
$request.header['user'].familyName) String fullName, @Body String body) {
// process the inbound message here
}

47

COO K BO O K

Groovy supports GStrings that is like a template where we can insert $ placeholders that will
be evaluated by Groovy.

@MessageDriven or @Consume
To consume a message you use either the @MessageDriven annotation or from 1.5.0 the
@Consume annotation to mark a particular method of a bean as being a consumer method.
The uri of the annotation defines the Camel Endpoint to consume from.
e.g. lets invoke the onCheese() method with the String body of the inbound JMS message
from ActiveMQ on the cheese queue; this will use the Type Converter to convert the JMS
ObjectMessage or BytesMessage to a String - or just use a TextMessage from JMS
public class Foo {
@Consume(uri="activemq:cheese")
public void onCheese(String name) {
...
}
}

The Bean Binding is then used to convert the inbound Message to the parameter list used to
invoke the method .
What this does is basically create a route that looks kinda like this
from(uri).bean(theBean, "methodName");

Using context option to apply only a certain CamelContext
Available as of Camel 2.0
See the warning above.
You can use the context option to specify which CamelContext the consumer should
only apply for. For example:
@Consume(uri="activemq:cheese", context="camel-1")
public void onCheese(String name) {

The consumer above will only be created for the CamelContext that have the context id =
camel-1. You set this id in the XML tag:


CO O K B O O K

48

@MessageDriven is @deprecated
@MessageDriven is deprecated in Camel 1.x. You should use @Consume instead.
Its removed in Camel 2.0.

When using more than one CamelContext
When you use more than 1 CamelContext you might end up with each of them
creating a POJO Consuming.
In Camel 2.0 there is a new option on @Consume that allows you to specify
which CamelContext id/name you want it to apply for.

Using an explicit route
If you want to invoke a bean method from many different endpoints or within different complex
routes in different circumstances you can just use the normal routing DSL or the Spring XML
configuration file.
For example
from(uri).beanRef("myBean", "methodName");

which will then look up in the Registry and find the bean and invoke the given bean name. (You
can omit the method name and have Camel figure out the right method based on the method
annotations and body type).

Use the Bean endpoint
You can always use the bean endpoint
from(uri).to("bean:myBean?method=methodName");

Which approach to use?
Using the @MessageDriven/@Consume annotations are simpler when you are creating a
simple route with a single well defined input URI.
However if you require more complex routes or the same bean method needs to be
invoked from many places then please use the routing DSL as shown above.
There are two different ways to send messages to any Camel Endpoint from a POJO

49

COO K BO O K

@EndpointInject
To allow sending of messages from POJOs you can use @EndpointInject() annotation. This will
inject either a ProducerTemplate or CamelTemplate so that the bean can send message
exchanges.
e.g. lets send a message to the foo.bar queue in ActiveMQ at some point

public class Foo {
@EndpointInject(uri="activemq:foo.bar")
ProducerTemplate producer;
public void doSomething() {
if (whatever) {
producer.sendBody("world!");
}
}
}

The downside of this is that your code is now dependent on a Camel API, the
ProducerTemplate. The next section describes how to remove this
Hiding the Camel APIs from your code using @Produce
We recommend Hiding Middleware APIs from your application code so the next option might
be more suitable.
You can add the @Produce annotation to an injection point (a field or property setter) using a
ProducerTemplate or using some interface you use in your business logic. e.g.
public interface MyListener {
String sayHello(String name);
}
public class MyBean {
@Produce(uri = "activemq:foo")
protected MyListener producer;
public void doSomething() {
// lets send a message
String response = producer.sayHello("James");
}
}

Here Camel will automatically inject a smart client side proxy at the @Produce annotation - an
instance of the MyListener instance. When we invoke methods on this interface the method call
is turned into an object and using the Camel Spring Remoting mechanism it is sent to the
endpoint - in this case the ActiveMQ endpoint to queue foo; then the caller blocks for a
response.

CO O K B O O K

50

If you want to make asynchronous message sends then use an @InOnly annotation on the
injection point.

@RECIPIENTLIST ANNOTATION
As of 1.5.0 we now support the use of @RecipientList on a bean method to easily create a
dynamic Recipient List using a Java method.
Simple Example using @Consume
package com.acme.foo;
public class RouterBean {
@Consume(uri = "activemq:foo")
@RecipientList
public String[] route(String body) {
return new String[]{"activemq:bar", "activemq:whatnot"};
}
}

For example if the above bean is configured in Spring when using a 
element as follows






then a route will be created consuming from the foo queue on the ActiveMQ component
which when a message is received the message will be forwarded to the endpoints defined by
the result of this method call - namely the bar and whatnot queues.

51

COO K BO O K

How it works
The return value of the @RecipientList method is converted to either a java.util.Collection /
java.util.Iterator or array of objects where each element is converted to an Endpoint or a String,
or if you are only going to route to a single endpoint then just return either an Endpoint object
or an object that can be converted to a String. So the following methods are all valid
@RecipientList
public String[] route(String body) { ... }
@RecipientList
public List route(String body) { ... }
@RecipientList
public Endpoint route(String body) { ... }
@RecipientList
public Endpoint[] route(String body) { ... }
@RecipientList
public Collection route(String body) { ... }
@RecipientList
public URI route(String body) { ... }
@RecipientList
public URI[] route(String body) { ... }

Then for each endpoint or URI the message is forwarded a separate copy to that endpoint.
You can then use whatever Java code you wish to figure out what endpoints to route to; for
example you can use the Bean Binding annotations to inject parts of the message body or
headers or use Expression values on the message.
More Complex Example Using DSL
In this example we will use more complex Bean Binding, plus we will use a separate route to
invoke the Recipient List
public class RouterBean2 {
@RecipientList
public String route(@Header("customerID") String custID String body) {
if (custID == null) return null;
return "activemq:Customers.Orders." + custID;
}
}
public class MyRouteBuilder extends RouteBuilder {
protected void configure() {
from("activemq:Orders.Incoming").recipientList(bean("myRouterBean", "route"));

CO O K B O O K

52

}
}

Notice how we are injecting some headers or expressions and using them to determine the
recipients using Recipient List EIP.
See the Bean Integration for more details.

USING EXCHANGE PATTERN ANNOTATIONS
When working with POJO Producing or Spring Remoting you invoke methods which typically
by default are InOut for Request Reply. That is there is an In message and an Out for the result.
Typically invoking this operation will be synchronous, the caller will block until the server
returns a result.
Camel has flexible Exchange Pattern support - so you can also support the Event Message
pattern to use InOnly for asynchronous or one way operations. These are often called 'fire and
forget' like sending a JMS message but not waiting for any response.
From 1.5 onwards Camel supports annotations for specifying the message exchange pattern
on regular Java methods, classes or interfaces.
Specifying InOnly methods
Typically the default InOut is what most folks want but you can customize to use InOnly using
an annotation.
public interface Foo {
Object someInOutMethod(String input);
String anotherInOutMethod(Cheese input);
@InOnly
void someInOnlyMethod(Document input);
}

The above code shows three methods on an interface; the first two use the default InOut
mechanism but the someInOnlyMethod uses the InOnly annotation to specify it as being a
oneway method call.
Class level annotations
You can also use class level annotations to default all methods in an interface to some pattern
such as
@InOnly
public interface Foo {

53

COO K BO O K

void someInOnlyMethod(Document input);
void anotherInOnlyMethod(String input);
}

Annotations will also be detected on base classes or interfaces. So for example if you created a
client side proxy for
public class MyFoo implements Foo {
...
}

Then the methods inherited from Foo would be InOnly.
Overloading a class level annotation
You can overload a class level annotation on specific methods. A common use case for this is if
you have a class or interface with many InOnly methods but you want to just annote one or
two methods as InOut
@InOnly
public interface Foo {
void someInOnlyMethod(Document input);
void anotherInOnlyMethod(String input);
@InOut
String someInOutMethod(String input);
}

In the above Foo interface the someInOutMethod will be InOut
Using your own annotations
You might want to create your own annotations to represent a group of different bits of
metadata; such as combining synchrony, concurrency and transaction behaviour.
So you could annotate your annotation with the @Pattern annotation to default the
exchange pattern you wish to use.
For example lets say we want to create our own annotation called @MyAsyncService
@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.TYPE, ElementType.METHOD})
// lets add the message exchange pattern to it
@Pattern(ExchangePattern.InOnly)
// lets add some other annotations - maybe transaction behaviour?

CO O K B O O K

54

public @interface MyAsyncService {
}

Now we can use this annotation and Camel will figure out the correct exchange pattern...
public interface Foo {
void someInOnlyMethod(Document input);
void anotherInOnlyMethod(String input);
@MyAsyncService
String someInOutMethod(String input);
}

When writing software these days, its important to try and decouple as much middleware code
from your business logic as possible.
This provides a number of benefits...
• you can choose the right middleware solution for your deployment and switch at any
time
• you don't have to spend a large amount of time learning the specifics of any particular
technology, whether its JMS or JavaSpace or Hibernate or JPA or iBATIS whatever
For example if you want to implement some kind of message passing, remoting, reliable load
balancing or asynchronous processing in your application we recommend you use Camel
annotations to bind your services and business logic to Camel Components which means you
can then easily switch between things like
• in JVM messaging with SEDA
• using JMS via ActiveMQ or other JMS providers for reliable load balancing, grid or
publish and subscribe
• for low volume, but easier administration since you're probably already using a
database you could use
◦ Hibernate or JPA to use an entity bean / table as a queue
◦ iBATIS to work with SQL
◦ JDBC for raw SQL access
• use JavaSpace
How to decouple from middleware APIs
The best approach when using remoting is to use Spring Remoting which can then use any
messaging or remoting technology under the covers. When using Camel's implementation you
can then use any of the Camel Components along with any of the Enterprise Integration
Patterns.

55

COO K BO O K

Another approach is to bind Java beans to Camel endpoints via the Bean Integration. For
example using POJO Consuming and POJO Producing you can avoid using any Camel APIs to
decouple your code both from middleware APIs and Camel APIs!

VISUALISATION
Camel supports the visualisation of your Enterprise Integration Patterns using the GraphViz
DOT files which can either be rendered directly via a suitable GraphViz tool or turned into
HTML, PNG or SVG files via the Camel Maven Plugin.
Here is a typical example of the kind of thing we can generate

If you click on the actual generated htmlyou will see that you can navigate from an EIP node
to its pattern page, along with getting hover-over tool tips ec.
How to generate
See Camel Dot Maven Goal or the other maven goals Camel Maven Plugin
For OS X users
If you are using OS X then you can open the DOT file using graphviz which will then
automatically re-render if it changes, so you end up with a real time graphical representation of
the topic and queue hierarchies!

CO O K B O O K

56

Also if you want to edit the layout a little before adding it to a wiki to distribute to your
team, open the DOT file with OmniGraffle then just edit away

BUSINESS ACTIVITY MONITORING
The Camel BAM module provides a Business Activity Monitoring (BAM) framework for
testing business processes across multiple message exchanges on different Endpoint instances.
For example if you have a simple system which you submit Purchase Orders into system A
and then receive Invoices from system B, you might want to test that for a specific Purchase
Order you receive a matching Invoice from system B within a specific time period.
How Camel BAM Works
What Camel BAM does is use a Correlation Identifier on an input message to determine which
Process Instance a message belongs to. The process instance is an entity bean which can maintain
state for each Activity (where an activity typically maps to a single endpoint, such as the receipt
of Purchase orders, or the receipt of Invoices).
You can then add rules which are fired when a message is received on any activity such as to
set time expectations, or to perform real time reconciliation of values across activities etc.
Simple Example
The following example shows how to perform some time based rules on a simple business
process of 2 activities A and B (which maps to the Purchase Order and Invoice example above).
If you want to experiment with this scenario you could edit the Test Case which defines the
activities and rules, then tests that they work.
return new ProcessBuilder(jpaTemplate, transactionTemplate) {
public void configure() throws Exception {
// lets define some activities, correlating on an XPath on the message bodies
ActivityBuilder a = activity("seda:a").name("a")
.correlate(xpath("/hello/@id"));
ActivityBuilder b = activity("seda:b").name("b")
.correlate(xpath("/hello/@id"));
// now lets add some rules
b.starts().after(a.completes())
.expectWithin(seconds(1))
.errorIfOver(seconds(errorTimeout)).to("mock:overdue");
}
};

57

COO K BO O K

As you can see in the above example, we define two activities first, then we define rules on
when we expect the activities on an individual process instance to complete by along with the
time at which we should assume there is an error. The ProcessBuilder is-a RouteBuilder and
can be added to any CamelContext
Complete Example
For a complete example please see the BAM Example which is part of the standard Camel
Examples
Use Cases
In the world of finance a common requirement is tracking financial trades. Often a trader will
submit a Front Office Trade which then flows through the Middle Office and Back Office
through various systems to settle the trade so that money is exchanged. You may wish to add
tests that front and back office trades match up within a time period; if they don't match or a
back office trade does not arrive within a required amount of time, you might want to fire off
an alarm.

EXTRACT TRANSFORM LOAD (ETL)
The ETL (Extract, Transform, Load) is a mechanism for loading data into systems or databases
using some kind of Data Format from a variety of sources; often files then using Pipes and
Filters, Message Translator and possible other Enterprise Integration Patterns.
So you could query data from various Camel Components such as File, HTTP or JPA,
perform multiple patterns such as Splitter or Message Translator then send the messages to
some other Component.
To show how this all fits together, try the ETL Example

MOCK COMPONENT
Testing of distributed and asynchronous processing is notoriously difficult. The Mock, Test and
DataSet endpoints work great with the Camel Testing Framework to simplify your unit and
integration testing using Enterprise Integration Patterns and Camel's large range of Components
together with the powerful Bean Integration.
The Mock component provides a powerful declarative testing mechanism, which is similar to
jMock in that it allows declarative expectations to be created on any Mock endpoint before a
test begins. Then the test is run, which typically fires messages to one or more endpoints, and
finally the expectations can be asserted in a test case to ensure the system worked as expected.
This allows you to test various things like:
• The correct number of messages are received on each endpoint,
• The correct payloads are received, in the right order,
CO O K B O O K

58

• Messages arrive on an endpoint in order, using some Expression to create an order
testing function,
• Messages arrive match some kind of Predicate such as that specific headers have
certain values, or that parts of the messages match some predicate, such as by
evaluating an XPath or XQuery Expression.
Note that there is also the Test endpoint which is a Mock endpoint, but which uses a second
endpoint to provide the list of expected message bodies and automatically sets up the Mock
endpoint assertions. In other words, it's a Mock endpoint that automatically sets up its
assertions from some sample messages in a File or database, for example.
URI format
mock:someName[?options]

Where someName can be any string that uniquely identifies the endpoint.
You can append query options to the URI in the following format,
?option=value&option=value&...
Options
Option

Default

Description

reportGroup

null

A size to use a throughput logger for reporting

Simple Example
Here's a simple example of Mock endpoint in use. First, the endpoint is resolved on the
context. Then we set an expectation, and then, after the test has run, we assert that our
expectations have been met.
MockEndpoint resultEndpoint = context.resolveEndpoint("mock:foo", MockEndpoint.class);
resultEndpoint.expectedMessageCount(2);
// send some messages
...
// now lets assert that the mock:foo endpoint received 2 messages
resultEndpoint.assertIsSatisfied();

You typically always call the assertIsSatisfied() method to test that the expectations were met
after running a test.
Camel will by default wait 20 seconds when the assertIsSatisfied() is invoked. This
can be configured by setting the setResultWaitTime(millis) method.
59

COO K BO O K

Setting expectations
You can see from the javadoc of MockEndpoint the various helper methods you can use to set
expectations. The main methods are as follows:
Method

Description

expectedMessageCount(int)

To define the expected message count on the
endpoint.

expectedMinimumMessageCount(int)

To define the minimum number of expected
messages on the endpoint.

expectedBodiesReceived(...)

To define the expected bodies that should be
received (in order).

expectedHeaderReceived(...)

To define the expected header that should be
received

expectsAscending(Expression)

To add an expectation that messages are received in
order, using the given Expression to compare
messages.

expectsDescending(Expression)

To add an expectation that messages are received in
order, using the given Expression to compare
messages.

expectsNoDuplicates(Expression)

To add an expectation that no duplicate messages
are received; using an Expression to calculate a
unique identifier for each message. This could be
something like the JMSMessageID if using JMS, or
some unique reference number within the message.

Here's another example:
resultEndpoint.expectedBodiesReceived("firstMessageBody", "secondMessageBody",
"thirdMessageBody");

Adding expectations to specific messages
In addition, you can use the message(int messageIndex) method to add assertions about a
specific message that is received.
For example, to add expectations of the headers or body of the first message (using zerobased indexing like java.util.List), you can use the following code:
resultEndpoint.message(0).header("foo").isEqualTo("bar");

CO O K B O O K

60

There are some examples of the Mock endpoint in use in the camel-core processor tests.
A Spring Example
First, here's the spring.xml file




/person/city = 'London'






As you can see, it defines a simple routing rule which consumes messages from the local src/
test/data directory. The noop flag just means not to delete or move the file after its been
processed.
Also note we instantiate a bean called myBean, here is the source of the MyAssertions
bean.
public class MyAssertions implements InitializingBean {
@EndpointInject(uri = "mock:matched")
private MockEndpoint matched;
@EndpointInject(uri = "mock:notMatched")
private MockEndpoint notMatched;
public void afterPropertiesSet() throws Exception {
// lets add some expectations
matched.expectedMessageCount(1);
notMatched.expectedMessageCount(0);
}
public void assertEndpointsValid() throws Exception {
// now lets perform some assertions that the test worked as we expect
Assert.assertNotNull("Should have a matched endpoint", matched);
Assert.assertNotNull("Should have a notMatched endpoint", notMatched);
MockEndpoint.assertIsSatisfied(matched, notMatched);
}
}

The bean is injected with a bunch of Mock endpoints using the @EndpointInject annotation, it
then sets a bunch of expectations on startup (using Spring's InitializingBean interface
and afterPropertiesSet() method) before the CamelContext starts up.

61

COO K BO O K

Then in our test case (which could be JUnit or TesNG) we lookup myBean in Spring (or
have it injected into our test) and then invoke the assertEndpointsValid() method on
it to verify that the mock endpoints have their assertions met. You could then inspect the
message exchanges that were delivered to any of the endpoints using the
getReceivedExchanges() method on the Mock endpoint and perform further assertions or
debug logging.
Here is the actual JUnit test case we use.
See Also
•
•
•
•
•

Configuring Camel
Component
Endpoint
Getting Started
Spring Testing

TESTING
Testing is a crucial activity in any piece of software development or integration. Typically Camel
Riders use various different technologies wired together in a variety of patterns with different
expression languages together with different forms of Bean Integration and Dependency
Injection so its very easy for things to go wrong!
. Testing is the crucial weapon to ensure
that things work as you would expect.
Camel is a Java library so you can easily wire up tests in whatever unit testing framework
you use (JUnit 3.x, 4.x or TestNG). However the Camel project has tried to make the testing
of Camel as easy and powerful as possible so we have introduced the following features.
Testing mechanisms
The following mechanisms are supported
Name

Description

Camel
Test

is a library letting you easily create Camel test cases using a single Java class for all
your configuration and routing without using Spring or Guice for Dependency
Injection which does not require an in depth knowledge of Spring+SpringTest or
Guice

Spring
Testing

uses Spring Test together with either XML or Java Config to dependency inject
your test classes

Guice

uses Guice to dependency inject your test classes

In all approaches the test classes look pretty much the same in that they all reuse the Camel
binding and injection annotations.

CO O K B O O K

62

Camel Test Example
Here is the Camel Test example.
public class FilterTest extends CamelTestSupport {
@EndpointInject(uri = "mock:result")
protected MockEndpoint resultEndpoint;
@Produce(uri = "direct:start")
protected ProducerTemplate template;
public void testSendMatchingMessage() throws Exception {
String expectedBody = "";
resultEndpoint.expectedBodiesReceived(expectedBody);
template.sendBodyAndHeader(expectedBody, "foo", "bar");
resultEndpoint.assertIsSatisfied();
}
public void testSendNotMatchingMessage() throws Exception {
resultEndpoint.expectedMessageCount(0);
template.sendBodyAndHeader("", "foo", "notMatchedHeaderValue");
resultEndpoint.assertIsSatisfied();
}
@Override
protected RouteBuilder createRouteBuilder() {
return new RouteBuilder() {
public void configure() {
from("direct:start").filter(header("foo").isEqualTo("bar")).to("mock:result");
}
};
}
}

Notice how it derives from the Camel helper class CamelTestSupport but has no Spring or
Guice dependency injection configuration but instead overrides the createRouteBuilder()
method.

Spring Test with XML Config Example
Here is the Spring Testing example using XML Config.

63

COO K BO O K

@ContextConfiguration
public class FilterTest extends AbstractJUnit38SpringContextTests {
@EndpointInject(uri = "mock:result")
protected MockEndpoint resultEndpoint;
@Produce(uri = "direct:start")
protected ProducerTemplate template;
@DirtiesContext
public void testSendMatchingMessage() throws Exception {
String expectedBody = "";
resultEndpoint.expectedBodiesReceived(expectedBody);
template.sendBodyAndHeader(expectedBody, "foo", "bar");
resultEndpoint.assertIsSatisfied();
}
@DirtiesContext
public void testSendNotMatchingMessage() throws Exception {
resultEndpoint.expectedMessageCount(0);
template.sendBodyAndHeader("", "foo", "notMatchedHeaderValue");
resultEndpoint.assertIsSatisfied();
}
}

Notice that we use @DirtiesContext on the test methods to force Spring Testing to
automatically reload the CamelContext after each test method - this ensures that the tests
don't clash with each other (e.g. one test method sending to an endpoint that is then reused in
another test method).
Also notice the use of @ContextConfiguration to indicate that by default we should
look for the FilterTest-context.xml on the classpath to configure the test case which looks like
this





CO O K B O O K

64


$foo = 'bar'






Spring Test with Java Config Example
Here is the Spring Testing example using Java Config. For more information see Spring Java
Config.
@ContextConfiguration(
locations =
"org.apache.camel.spring.javaconfig.patterns.FilterTest$ContextConfig",
loader = JavaConfigContextLoader.class)
public class FilterTest extends AbstractJUnit4SpringContextTests {
@EndpointInject(uri = "mock:result")
protected MockEndpoint resultEndpoint;
@Produce(uri = "direct:start")
protected ProducerTemplate template;
@DirtiesContext
@Test
public void testSendMatchingMessage() throws Exception {
String expectedBody = "";
resultEndpoint.expectedBodiesReceived(expectedBody);
template.sendBodyAndHeader(expectedBody, "foo", "bar");
resultEndpoint.assertIsSatisfied();
}
@DirtiesContext
@Test
public void testSendNotMatchingMessage() throws Exception {
resultEndpoint.expectedMessageCount(0);
template.sendBodyAndHeader("", "foo", "notMatchedHeaderValue");
resultEndpoint.assertIsSatisfied();
}
@Configuration
public static class ContextConfig extends SingleRouteCamelConfiguration {

65

COO K BO O K

@Bean
public RouteBuilder route() {
return new RouteBuilder() {
public void configure() {
from("direct:start").filter(header("foo").isEqualTo("bar")).to("mock:result");
}
};
}
}
}

This is similar to the XML Config example above except that there is no XML file and instead
the nested ContextConfig class does all of the configuration; so your entire test case is
contained in a single Java class. We currently have to reference by class name this class in the
@ContextConfiguration which is a bit ugly. Please vote for SJC-238 to address this and
make Spring Test work more cleanly with Spring JavaConfig.
Its totally optional but for the ContextConfig implementation we derive from
SingleRouteCamelConfiguration which is a helper Spring Java Config class which will
configure the CamelContext for us and then register the RouteBuilder we create.
Testing endpoints
Camel provides a number of endpoints which can make testing easier.
Name

Description

DataSet

For load & soak testing this endpoint provides a way to create huge numbers of
messages for sending to Components and asserting that they are consumed
correctly

Mock

For testing routes and mediation rules using mocks and allowing assertions to be
added to an endpoint

Test

Creates a Mock endpoint which expects to receive all the message bodies that
could be polled from the given underlying endpoint

The main endpoint is the Mock endpoint which allows expectations to be added to different
endpoints; you can then run your tests and assert that your expectations are met at the end.
Stubbing out physical transport technologies
If you wish to test out a route but want to avoid actually using a real physical transport (for
example to unit test a transformation route rather than performing a full integration test) then
the following endpoints can be useful.
Name

Description

CO O K B O O K

66

Direct

Direct invocation of the consumer from the producer so that single threaded
(non-SEDA) in VM invocation is performed which can be useful to mock out
physical transports

SEDA

Delivers messages asynchonously to consumers via a
java.util.concurrent.BlockingQueue which is good for testing asynchronous
transports

Testing existing routes
Camel provides some features to aid during testing of existing routes where you cannot or will
not use Mock etc. For example you may have a production ready route which you want to test
with some 3rd party API which sends messages into this route.
Name

Description

NotifyBuilder

Allows you to be notified when a certain condition has occurred. For
example when the route has completed 5 messages. You can build complex
expressions to match your criteria when to be notified.

AdviceWith

Allows you to advice or enhance an existing route using a RouteBuilder
style. For example you can add interceptors to intercept sending outgoing
messages to assert those messages are as expected.

CAMEL TEST
As a simple alternative to using Spring Testing or Guice the camel-test module was
introduced into the Camel 2.0 trunk so you can perform powerful Testing of your Enterprise
Integration Patterns easily.
Adding to your pom.xml
To get started using Camel Test you will need to add an entry to your pom.xml

org.apache.camel
camel-test
${camel-version}
test


You might also want to add commons-logging and log4j to ensure nice logging messages (and
maybe adding a log4j.properties file into your src/test/resources directory).

67

COO K BO O K


commons-logging
commons-logging
test


log4j
log4j
test


Writing your test
You firstly need to derive from the class CamelTestSupport and typically you will need to
override the createRouteBuilder() method to create routes to be tested.
Here is an example.
public class FilterTest extends CamelTestSupport {
@EndpointInject(uri = "mock:result")
protected MockEndpoint resultEndpoint;
@Produce(uri = "direct:start")
protected ProducerTemplate template;
public void testSendMatchingMessage() throws Exception {
String expectedBody = "";
resultEndpoint.expectedBodiesReceived(expectedBody);
template.sendBodyAndHeader(expectedBody, "foo", "bar");
resultEndpoint.assertIsSatisfied();
}
public void testSendNotMatchingMessage() throws Exception {
resultEndpoint.expectedMessageCount(0);
template.sendBodyAndHeader("", "foo", "notMatchedHeaderValue");
resultEndpoint.assertIsSatisfied();
}
@Override
protected RouteBuilder createRouteBuilder() {
return new RouteBuilder() {
public void configure() {
from("direct:start").filter(header("foo").isEqualTo("bar")).to("mock:result");
}

CO O K B O O K

68

};
}
}

Notice how you can use the various Camel binding and injection annotations to inject individual
Endpoint objects - particularly the Mock endpoints which are very useful for Testing. Also you
can inject producer objects such as ProducerTemplate or some application code interface for
sending messages or invoking services.
JNDI
Camel uses a Registry to allow you to configure Component or Endpoint instances or Beans
used in your routes. If you are not using Spring or [OSGi] then JNDI is used as the default
registry implementation.
So you will also need to create a jndi.properties file in your src/test/resources
directory so that there is a default registry available to initialise the CamelContext.
Here is an example jndi.properties file
java.naming.factory.initial = org.apache.camel.util.jndi.CamelInitialContextFactory

See Also
• Testing
• Mock

SPRING TESTING
Testing is a crucial part of any development or integration work. The Spring Framework offers
a number of features that makes it easy to test while using Spring for Inversion of Control
which works with JUnit 3.x, JUnit 4.x or TestNG.
We can reuse Spring for IoC and the Camel Mock and Test endpoints to create
sophisticated integration tests that are easy to run and debug inside your IDE.
For example here is a simple unit test
import
import
import
import
import

org.apache.camel.CamelContext;
org.apache.camel.component.mock.MockEndpoint;
org.springframework.beans.factory.annotation.Autowired;
org.springframework.test.context.ContextConfiguration;
org.springframework.test.context.junit38.AbstractJUnit38SpringContextTests;

@ContextConfiguration
public class MyCamelTest extends AbstractJUnit38SpringContextTests {

69

COO K BO O K

@Autowired
protected CamelContext camelContext;
public void testMocksAreValid() throws Exception {
MockEndpoint.assertIsSatisfied(camelContext);
}
}

This test will load a Spring XML configuration file called MyCamelTest-context.xml from
the classpath in the same package structure as the MyCamelTest class and initialize it along with
any Camel routes we define inside it, then inject the CamelContext instance into our test case.
For instance, like this maven folder layout:
src/main/java/com/mycompany/MyCamelTest.class
src/main/resources/com/mycompany/MyCamelTest-context.xml

You can overload the method createApplicationContext to provide the Spring
ApplicationContext that isn't following the above default. For instance:
protected AbstractXmlApplicationContext createApplicationContext() {
return new ClassPathXmlApplicationContext("/config/MySpringConfig.xml");
}

Then the test method will then run which invokes the
MockEndpoint.assertIsSatisfied(camelContext) method which asserts that all of the Mock and
Test endpoints have their expectations met.
xml}
Spring Test with Java Config Example
You can completely avoid using an XML configuration file by using Spring Java Config.
Here is an example using Java Config.
@ContextConfiguration(
locations =
"org.apache.camel.spring.javaconfig.patterns.FilterTest$ContextConfig",
loader = JavaConfigContextLoader.class)
public class FilterTest extends AbstractJUnit4SpringContextTests {
@EndpointInject(uri = "mock:result")
protected MockEndpoint resultEndpoint;
@Produce(uri = "direct:start")
protected ProducerTemplate template;

CO O K B O O K

70

@DirtiesContext
@Test
public void testSendMatchingMessage() throws Exception {
String expectedBody = "";
resultEndpoint.expectedBodiesReceived(expectedBody);
template.sendBodyAndHeader(expectedBody, "foo", "bar");
resultEndpoint.assertIsSatisfied();
}
@DirtiesContext
@Test
public void testSendNotMatchingMessage() throws Exception {
resultEndpoint.expectedMessageCount(0);
template.sendBodyAndHeader("", "foo", "notMatchedHeaderValue");
resultEndpoint.assertIsSatisfied();
}
@Configuration
public static class ContextConfig extends SingleRouteCamelConfiguration {
@Bean
public RouteBuilder route() {
return new RouteBuilder() {
public void configure() {
from("direct:start").filter(header("foo").isEqualTo("bar")).to("mock:result");
}
};
}
}
}

This is similar to the XML Config example above except that there is no XML file and instead
the nested ContextConfig class does all of the configuration; so your entire test case is
contained in a single Java class. We currently have to reference by class name this class in the
@ContextConfiguration which is a bit ugly. Please vote for SJC-238 to address this and
make Spring Test work more cleanly with Spring JavaConfig.
Adding more Mock expectations
If you wish to programmatically add any new assertions to your test you can easily do so with
the following. Notice how we use @EndpointInject to inject a Camel endpoint into our code
then the Mock API to add an expectation on a specific message.
@ContextConfiguration
public class MyCamelTest extends AbstractJUnit38SpringContextTests {

71

COO K BO O K

@Autowired
protected CamelContext camelContext;
@EndpointInject(uri = "mock:foo")
protected MockEndpoint foo;
public void testMocksAreValid() throws Exception {
// lets add more expectations
foo.message(0).header("bar").isEqualTo("ABC");
MockEndpoint.assertIsSatisfied(camelContext);
}
}

Further processing the received messages
Sometimes once a Mock endpoint has received some messages you want to then process them
further to add further assertions that your test case worked as you expect.
So you can then process the received message exchanges if you like...
@ContextConfiguration
public class MyCamelTest extends AbstractJUnit38SpringContextTests {
@Autowired
protected CamelContext camelContext;
@EndpointInject(uri = "mock:foo")
protected MockEndpoint foo;
public void testMocksAreValid() throws Exception {
// lets add more expectations...
MockEndpoint.assertIsSatisfied(camelContext);
// now lets do some further assertions
List list = foo.getReceivedExchanges();
for (Exchange exchange : list) {
Message in = exchange.getIn();
...
}
}
}

Sending and receiving messages
It might be that the Enterprise Integration Patterns you have defined in either Spring XML or
using the Java DSL do all of the sending and receiving and you might just work with the Mock

CO O K B O O K

72

endpoints as described above. However sometimes in a test case its useful to explicitly send or
receive messages directly.
To send or receive messages you should use the Bean Integration mechanism. For example
to send messages inject a ProducerTemplate using the @EndpointInject annotation then call the
various send methods on this object to send a message to an endpoint. To consume messages
use the @MessageDriven annotation on a method to have the method invoked when a message
is received.
public class Foo {
@EndpointInject(uri="activemq:foo.bar")
ProducerTemplate producer;
public void doSomething() {
// lets send a message!
producer.sendBody("world!");
}
// lets consume messages from the 'cheese' queue
@MessageDriven(uri="activemq:cheese")
public void onCheese(String name) {
...
}
}

See Also
•
•
•
•

a real example test case using Mock and Spring along with its Spring XML
Bean Integration
Mock endpoint
Test endpoint

CAMEL GUICE
As of 1.5 we now have support for Google Guice as a dependency injection framework. To use
it just be dependent on camel-guice.jar which also depends on the following jars.
Dependency Injecting Camel with Guice
The GuiceCamelContext is designed to work nicely inside Guice. You then need to bind it
using some Guice Module.
The camel-guice library comes with a number of reusable Guice Modules you can use if you
wish - or you can bind the GuiceCamelContext yourself in your own module.
• CamelModule is the base module which binds the GuiceCamelContext but leaves it
up you to bind the RouteBuilder instances

73

COO K BO O K

• CamelModuleWithRouteTypes extends CamelModule so that in the constructor of
the module you specify the RouteBuilder classes or instances to use
• CamelModuleWithMatchingRoutes extends CamelModule so that all bound
RouteBuilder instances will be injected into the CamelContext or you can supply an
optional Matcher to find RouteBuilder instances matching some kind of predicate.
So you can specify the exact RouteBuilder instances you want
Injector injector = Guice.createInjector(new
CamelModuleWithRouteTypes(MyRouteBuilder.class, AnotherRouteBuilder.class));
// if required you can lookup the CamelContext
CamelContext camelContext = injector.getInstance(CamelContext.class);

Or inject them all
Injector injector = Guice.createInjector(new CamelModuleWithRouteTypes());
// if required you can lookup the CamelContext
CamelContext camelContext = injector.getInstance(CamelContext.class);

You can then use Guice in the usual way to inject the route instances or any other dependent
objects.
Bootstrapping with JNDI
A common pattern used in J2EE is to bootstrap your application or root objects by looking
them up in JNDI. This has long been the approach when working with JMS for example looking up the JMS ConnectionFactory in JNDI for example.
You can follow a similar pattern with Guice using the GuiceyFruit JNDI Provider which lets
you bootstrap Guice from a jndi.properties file which can include the Guice Modules to
create along with environment specific properties you can inject into your modules and objects.
If the jndi.properties is conflict with other component, you can specify the jndi
properties file name in the Guice Main with option -j or -jndiProperties with the properties file
location to let Guice Main to load right jndi properties file.
Configuring Component, Endpoint or RouteBuilder instances
You can use Guice to dependency inject whatever objects you need to create, be it an
Endpoint, Component, RouteBuilder or arbitrary bean used within a route.
The easiest way to do this is to create your own Guice Module class which extends one of
the above module classes and add a provider method for each object you wish to create. A
provider method is annotated with @Provides as follows
public class MyModule extends CamelModuleWithMatchingRoutes {

CO O K B O O K

74

@Provides
@JndiBind("jms")
JmsComponent jms(@Named("activemq.brokerURL") String brokerUrl) {
return JmsComponent.jmsComponent(new ActiveMQConnectionFactory(brokerUrl));
}
}

You can optionally annotate the method with @JndiBind to bind the object to JNDI at some
name if the object is a component, endpoint or bean you wish to refer to by name in your
routes.
You can inject any environment specific properties (such as URLs, machine names,
usernames/passwords and so forth) from the jndi.properties file easily using the @Named
annotation as shown above. This allows most of your configuration to be in Java code which is
typesafe and easily refactorable - then leaving some properties to be environment specific (the
jndi.properties file) which you can then change based on development, testing, production etc.
Creating multiple RouteBuilder instances per type
It is sometimes useful to create multiple instances of a particular RouteBuilder with different
configurations.
To do this just create multiple provider methods for each configuration; or create a single
provider method that returns a collection of RouteBuilder instances.
For example
import org.apache.camel.guice.CamelModuleWithMatchingRoutes;
import com.google.common.collect.Lists;
public class MyModule extends CamelModuleWithMatchingRoutes {
@Provides
@JndiBind("foo")
Collection foo(@Named("fooUrl") String fooUrl) {
return Lists.newArrayList(new MyRouteBuilder(fooUrl), new
MyRouteBuilder("activemq:CheeseQueue"));
}
}

See Also
• there are a number of Examples you can look at to see Guice and Camel being used
such as Guice JMS Example
• Guice Maven Plugin for running your Guice based routes via Maven

75

COO K BO O K

TEMPLATING
When you are testing distributed systems its a very common requirement to have to stub out
certain external systems with some stub so that you can test other parts of the system until a
specific system is available or written etc.
A great way to do this is using some kind of Template system to generate responses to
requests generating a dynamic message using a mostly-static body.
There are a number of templating components you could use
• Freemarker
• Scalate
• StringTemplate
• Velocity
• XQuery
• XSLT
Example
Here's a simple example showing how we can respond to InOut requests on the My.Queue
queue on ActiveMQ with a template generated response. The reply would be sent back to the
JMSReplyTo Destination.
from("activemq:My.Queue").
to("velocity:com/acme/MyResponse.vm");

If you want to use InOnly and consume the message and send it to another destination you
could use
from("activemq:My.Queue").
to("velocity:com/acme/MyResponse.vm").
to("activemq:Another.Queue");

See Also
• Mock for details of mock endpoint testing (as opposed to template based stubs).

DATABASE
Camel can work with databases in a number of different ways. This document tries to outline
the most common approaches.
Database endpoints
Camel provides a number of different endpoints for working with databases
CO O K B O O K

76

• JPA for working with hibernate, openjpa or toplink. When consuming from the
endpoints entity beans are read (and deleted/updated to mark as processed) then
when producing to the endpoints they are written to the database (via insert/update).
• iBATIS similar to the above but using Apache iBATIS
• JDBC similar though using explicit SQL
Database pattern implementations
Various patterns can work with databases as follows
• Idempotent Consumer
• Aggregator
• BAM for business activity monitoring

PARALLEL PROCESSING AND ORDERING
It is a common requirement to want to use parallel processing of messages for throughput and
load balancing, while at the same time process certain kinds of messages in order.
How to achieve parallel processing
You can send messages to a number of Camel Components to achieve parallel processing and
load balancing such as
• SEDA for in-JVM load balancing across a thread pool
• ActiveMQ or JMS for distributed load balancing and parallel processing
• JPA for using the database as a poor mans message broker
When processing messages concurrently, you should consider ordering and concurrency issues.
These are described below

Concurrency issues
Note that there is no concurrency or locking issue when using ActiveMQ, JMS or SEDA by
design; they are designed for highly concurrent use. However there are possible concurrency
issues in the Processor of the messages i.e. what the processor does with the message?
For example if a processor of a message transfers money from one account to another
account; you probably want to use a database with pessimistic locking to ensure that operation
takes place atomically.

77

COO K BO O K

Ordering issues
As soon as you send multiple messages to different threads or processes you will end up with
an unknown ordering across the entire message stream as each thread is going to process
messages concurrently.
For many use cases the order of messages is not too important. However for some
applications this can be crucial. e.g. if a customer submits a purchase order version 1, then
amends it and sends version 2; you don't want to process the first version last (so that you
loose the update). Your Processor might be clever enough to ignore old messages. If not you
need to preserve order.
Recommendations
This topic is large and diverse with lots of different requirements; but from a high level here are
our recommendations on parallel processing, ordering and concurrency
• for distributed locking, use a database by default, they are very good at it
• to preserve ordering across a JMS queue consider using Exclusive Consumers in the
ActiveMQ component
• even better are Message Groups which allows you to preserve ordering across
messages while still offering parallelisation via the JMSXGrouopID header to
determine what can be parallelized
• if you receive messages out of order you could use the Resequencer to put them
back together again
A good rule of thumb to help reduce ordering problems is to make sure each single can be
processed as an atomic unit in parallel (either without concurrency issues or using say, database
locking); or if it can't, use a Message Group to relate the messages together which need to be
processed in order by a single thread.
Using Message Groups with Camel
To use a Message Group with Camel you just need to add a header to the output JMS message
based on some kind of Correlation Identifier to correlate messages which should be processed
in order by a single thread - so that things which don't correlate together can be processed
concurrently.
For example the following code shows how to create a message group using an XPath
expression taking an invoice's product code as the Correlation Identifier
from("activemq:a").setHeader("JMSXGroupID", xpath("/invoice/
productCode")).to("activemq:b");

You can of course use the Xml Configuration if you prefer

CO O K B O O K

78

ASYNCHRONOUS PROCESSING
Overview
Camel supports a more complex asynchronous processing model. The asynchronous
processors implement the AsyncProcessor interface which is derived from the more
synchronous Processor interface. There are advantages and disadvantages when using
asynchronous processing when compared to using the standard synchronous processing model.
Advantages:
• Processing routes that are composed fully of asynchronous processors do not use up
threads waiting for processors to complete on blocking calls. This can increase the
scalability of your system by reducing the number of threads needed to process the
same workload.
• Processing routes can be broken up into SEDA processing stages where different
thread pools can process the different stages. This means that your routes can be
processed concurrently.
Disadvantages:
• Implementing asynchronous processors is more complex than implementing the
synchronous versions.
When to Use
We recommend that processors and components be implemented the more simple
synchronous APIs unless you identify a performance of scalability requirement that dictates
otherwise. A Processor whose process() method blocks for a long time would be good
candidates for being converted into an asynchronous processor.
Interface Details
public interface AsyncProcessor extends Processor {
boolean process(Exchange exchange, AsyncCallback callback);
}

The AsyncProcessor defines a single process() method which is very similar to it's
synchronous Processor.process() brethren. Here are the differences:
• A non-null AsyncCallback MUST be supplied which will be notified when the
exchange processing is completed.
• It MUST not throw any exceptions that occurred while processing the exchange.
Any such exceptions must be stored on the exchange's Exception property.
• It MUST know if it will complete the processing synchronously or asynchronously.
The method will return true if it does complete synchronously, otherwise it returns
false.

79

COO K BO O K

Supported versions
The information on this page applies for the Camel 1.x and Camel 2.4 onwards. In
Camel 1.x the asynchronous processing is only implemented for JBI where as in
Camel 2.4 onwards we have implemented it in many other areas.
• When the processor has completed processing the exchange, it must call the
callback.done(boolean sync) method. The sync parameter MUST match
the value returned by the process() method.
Implementing Processors that Use the AsyncProcessor API
All processors, even synchronous processors that do not implement the AsyncProcessor
interface, can be coerced to implement the AsyncProcessor interface. This is usually done when
you are implementing a Camel component consumer that supports asynchronous completion of
the exchanges that it is pushing through the Camel routes. Consumers are provided a
Processor object when created. All Processor object can be coerced to a AsyncProcessor using
the following API:
Processor processor = ...
AsyncProcessor asyncProcessor = AsyncProcessorTypeConverter.convert(processor);

For a route to be fully asynchronous and reap the benefits to lower Thread usage, it must start
with the consumer implementation making use of the asynchronous processing API. If it called
the synchronous process() method instead, the consumer's thread would be forced to be
blocked and in use for the duration that it takes to process the exchange.
It is important to take note that just because you call the asynchronous API, it does not
mean that the processing will take place asynchronously. It only allows the possibility that it can
be done without tying up the caller's thread. If the processing happens asynchronously is
dependent on the configuration of the Camel route.
Normally, the the process call is passed in an inline inner AsyncCallback class instance which
can reference the exchange object that was declared final. This allows it to finish up any post
processing that is needed when the called processor is done processing the exchange. See
below for an example.
final Exchange exchange = ...
AsyncProcessor asyncProcessor = ...
asyncProcessor.process(exchange, new AsyncCallback() {
public void done(boolean sync) {
if (exchange.isFailed()) {
... // do failure processing.. perhaps rollback etc.
} else {

CO O K B O O K

80

... // processing completed successfully, finish up
// perhaps commit etc.
}
}
});

Asynchronous Route Sequence Scenarios
Now that we have understood the interface contract of the AsyncProcessor, and have seen
how to make use of it when calling processors, lets looks a what the thread model/sequence
scenarios will look like for some sample routes.
The Jetty component's consumers support async processing by using continuations. Suffice
to say it can take a http request and pass it to a camel route for async processing. If the
processing is indeed async, it uses Jetty continuation so that the http request is 'parked' and the
thread is released. Once the camel route finishes processing the request, the jetty component
uses the AsyncCallback to tell Jetty to 'un-park' the request. Jetty un-parks the request, the http
response returned using the result of the exchange processing.
Notice that the jetty continuations feature is only used "If the processing is indeed async".
This is why AsyncProcessor.process() implementations MUST accurately report if request is
completed synchronously or not.
The jhc component's producer allows you to make HTTP requests and implement the
AsyncProcessor interface. A route that uses both the jetty asynchronous consumer and the jhc
asynchronous producer will be a fully asynchronous route and has some nice attributes that can
be seen if we take a look at a sequence diagram of the processing route. For the route:
from("jetty:http://localhost:8080/service").to("jhc:http://localhost/service-impl");

The sequence diagram would look something like this:

81

COO K BO O K

The diagram simplifies things by making it looks like processors implement the
AsyncCallback interface when in reality the AsyncCallback interfaces are inline inner classes, but
it illustrates the processing flow and shows how 2 separate threads are used to complete the
processing of the original http request. The first thread is synchronous up until processing hits
the jhc producer which issues the http request. It then reports that the exchange processing
will complete async since it will use a NIO to complete getting the response back. Once the jhc
component has received a full response it uses AsyncCallback.done() method to notify
the caller. These callback notifications continue up until it reaches the original jetty consumer
which then un-parks the http request and completes it by providing the response.
Mixing Synchronous and Asynchronous Processors
It is totally possible and reasonable to mix the use of synchronous and asynchronous
processors/components. The pipeline processor is the backbone of a Camel processing route. It
glues all the processing steps together. It is implemented as an AsyncProcessor and supports
interleaving synchronous and asynchronous processors as the processing steps in the pipeline.
Lets say we have 2 custom processors, MyValidator and MyTransformation, both of which
are synchronous processors. Lets say we want to load file from the data/in directory validate
them with the MyValidator() processor, Transform them into JPA java objects using
MyTransformation and then insert them into the database using the JPA component. Lets say
that the transformation process takes quite a bit of time and we want to allocate 20 threads to
do parallel transformations of the input files. The solution is to make use of the thread
processor. The thread is AsyncProcessor that forces subsequent processing in asynchronous
thread from a thread pool.
The route might look like:

CO O K B O O K

82

from("file:data/in").process(new MyValidator()).thread(20).process(new
MyTransformation()).to("jpa:PurchaseOrder");

The sequence diagram would look something like this:

You would actually have multiple threads executing the 2nd part of the thread sequence.
Staying synchronous in an AsyncProcessor
Generally speaking you get better throughput processing when you process things
synchronously. This is due to the fact that starting up an asynchronous thread and doing a
context switch to it adds a little bit of of overhead. So it is generally encouraged that
AsyncProcessors do as much work as they can synchronously. When they get to a step that
would block for a long time, at that point they should return from the process call and let the
caller know that it will be completing the call asynchronously.

IMPLEMENTING VIRTUAL TOPICS ON OTHER JMS
PROVIDERS
ActiveMQ supports Virtual Topics since durable topic subscriptions kinda suck (see this page
for more detail) mostly since they don't support Competing Consumers.
Most folks want Queue semantics when consuming messages; so that you can support
Competing Consumers for load balancing along with things like Message Groups and Exclusive
Consumers to preserve ordering or partition the queue across consumers.

83

COO K BO O K

However if you are using another JMS provider you can implement Virtual Topics by
switching to ActiveMQ
or you can use the following Camel pattern.
First here's the ActiveMQ approach.
• send to activemq:topic:VirtualTopic.Orders
• for consumer A consume from activemq:Consumer.A.VirtualTopic.Orders
When using another message broker use the following pattern
• send to jms:Orders
• add this route with a to() for each logical durable topic subscriber
from("jms:Orders").to("jms:Consumer.A", "jms:Consumer.B", ...);

• for consumer A consume from jms:Consumer.A

WHAT'S THE CAMEL TRANSPORT FOR CXF
In CXF you offer or consume a webservice by defining it´s address. The first part of the
address specifies the protocol to use. For example address="http://localhost:90000" in an
endpoint configuration means your service will be offered using the http protocol on port 9000
of localhost. When you integrate Camel Tranport into CXF you get a new transport "camel".
So you can specify address="camel://direct:MyEndpointName" to bind the CXF service address
to a camel direct endpoint.
Technically speaking Camel transport for CXF is a component which implements the CXF
transport API with the Camel core library. This allows you to use camel´s routing engine and
integration patterns support smoothly together with your CXF services.

INTEGRATE CAMEL INTO CXF TRANSPORT LAYER
To include the Camel Tranport into your CXF bus you use the CamelTransportFactory. You
can do this in Java as well as in Spring.
Setting up the Camel Transport in Spring
You can use the following snippet in your applicationcontext if you want to configure anything
special. If you only want to activate the camel transport you do not have to do anything in your
application context. As soon as you include the camel-cxf jar in your app cxf will scan the jar
and load a CamelTransportFactory for you.





CO O K B O O K

84





http://cxf.apache.org/transports/camel




Integrating the Camel Transport in a programmatic way
Camel transport provides a setContext method that you could use to set the Camel context
into the transport factory. If you want this factory take effect, you need to register the factory
into the CXF bus. Here is a full example for you.
import
import
import
import
...

org.apache.cxf.Bus;
org.apache.cxf.BusFactory;
org.apache.cxf.transport.ConduitInitiatorManager;
org.apache.cxf.transport.DestinationFactoryManager;

BusFactory bf = BusFactory.newInstance();
Bus bus = bf.createBus();
CamelTransportFactory camelTransportFactory = new CamelTransportFactory();
camelTransportFactory.setCamelContext(context)
// register the conduit initiator
ConduitInitiatorManager cim = bus.getExtension(ConduitInitiatorManager.class);
cim.registerConduitInitiator(CamelTransportFactory.TRANSPORT_ID,
camelTransportFactory);
// register the destination factory
DestinationFactoryManager dfm = bus.getExtension(DestinationFactoryManager.class);
dfm.registerDestinationFactory(CamelTransportFactory.TRANSPORT_ID,
camelTransportFactory);
// set or bus as the default bus for cxf
BusFactory.setDefaultBus(bus);

CONFIGURE THE DESTINATION AND CONDUIT
Namespace
The elements used to configure an Camel transport endpoint are defined in the namespace
http://cxf.apache.org/transports/camel. It is commonly referred to using the
prefix camel. In order to use the Camel transport configuration elements you will need to add
85

COO K BO O K

the lines shown below to the beans element of your endpoint's configuration file. In addition,
you will need to add the configuration elements' namespace to the xsi:schemaLocation
attribute.
Listing 4. Adding the Configuration Namespace


The destination element
You configure an Camel transport server endpoint using the camel:destination element
and its children. The camel:destination element takes a single attribute, name, the
specifies the WSDL port element that corresponds to the endpoint. The value for the name
attribute takes the form portQName.camel-destination. The example below shows the
camel:destination element that would be used to add configuration for an endpoint that
was specified by the WSDL fragment 







...

The camel:destination element has a number of child elements that specify configuration
information. They are described below.
Element

Description

camelspring:camelContext

You can specify the camel context in the camel destination

CO O K B O O K

86

camel:camelContextRef

The camel context id which you want inject into the camel
destination

The conduit element
You configure an Camel transport client using the camel:conduit element and its children.
The camel:conduit element takes a single attribute, name, that specifies the WSDL port
element that corresponds to the endpoint. The value for the name attribute takes the form
portQName.camel-conduit. For example, the code below shows the camel:conduit
element that would be used to add configuration for an endpoint that was specified by the
WSDL fragment 






conduit_context



...

...

The camel:conduit element has a number of child elements that specify configuration
information. They are described below.

87

Element

Description

camelspring:camelContext

You can specify the camel context in the camel conduit

camel:camelContextRef

The camel context id which you want inject into the
camel conduit

COO K BO O K

EXAMPLE USING CAMEL AS A LOAD BALANCER FOR CXF
This example show how to use the camel load balance feature in CXF, and you need load the
configuration file in CXF and publish the endpoints on the address "camel://direct:EndpointA"
and "camel://direct:EndpointB"













dest_context



COMPLETE HOWTO AND EXAMPLE FOR ATTACHING
CAMEL TO CXF
Better JMS Transport for CXF Webservice using Apache Camel

CO O K B O O K

88

Introduction

When sending an Exchange to an Endpoint you can either use a Route or a ProducerTemplate.
This works fine in many scenarios. However you may need to guarantee that an exchange is
delivered to the same endpoint that you delivered a previous exchange on. For example in the
case of delivering a batch of exchanges to a MINA socket you may need to ensure that they are
all delivered through the same socket connection. Furthermore once the batch of exchanges
have been delivered the protocol requirements may be such that you are responsible for
closing the socket.

Using a Producer

To achieve fine grained control over sending exchanges you will need to program directly to a
Producer. Your code will look similar to:
CamelContext camelContext = ...
// Obtain an endpoint and create the producer we will be using.
Endpoint endpoint = camelContext.getEndpoint("someuri:etc");
Producer producer = endpoint.createProducer();
producer.start();
try {
// For each message to send...
Object requestMessage = ...
Exchange exchangeToSend = producer.createExchange();
exchangeToSend().setBody(requestMessage);
producer.process(exchangeToSend);
...
} finally {
// Tidy the producer up.
producer.stop();
}

In the case of using Apache MINA the producer.stop() invocation will cause the socket to be
closed.

89

INTR O D U C T I ON

Tutorials

There now follows the documentation on camel tutorials
• OAuth Tutorial
This tutorial demonstrates how to implement OAuth for a web application with
Camel's gauth component. The sample application of this tutorial is also online at
http://gauthcloud.appspot.com/
• Tutorial for Camel on Google App Engine
This tutorial demonstrates the usage of the Camel Components for Google App
Engine. The sample application of this tutorial is also online at
http://camelcloud.appspot.com/
• Tutorial on Spring Remoting with JMS
This tutorial is focused on different techniques with Camel for Client-Server
communication.
• Report Incident - This tutorial introduces Camel steadily and is based on a real life
integration problem
This is a very long tutorial beginning from the start; its for entry level to Camel. Its
based on a real life integration, showing how Camel can be introduced in an existing
solution. We do this in baby steps. The tutorial is currently work in progress, so
check it out from time to time. The tutorial explains some of the inner building blocks
Camel uses under the covers. This is good knowledge to have when you start using
Camel on a higher abstract level where it can do wonders in a few lines of routing
DSL.
• Using Camel with ServiceMix a tutorial on using Camel inside Apache ServiceMix.
• Better JMS Transport for CXF Webservice using Apache Camel Describes how to
use the Camel Transport for CXF to attach a CXF Webservice to a JMS Queue
• Tutorial how to use good old Axis 1.4 with Camel
This tutorial shows that Camel does work with the good old frameworks such as
AXIS that is/was widely used for WebService.
• Tutorial on using Camel in a Web Application
This tutorial gives an overview of how to use Camel inside Tomcat, Jetty or any other
servlet engine
• Tutorial on Camel 1.4 for Integration
Another real-life scenario. The company sells widgets, with a somewhat unique
business process (their customers periodically report what they've purchased in order
to get billed). However every customer uses a different data format and protocol.
This tutorial goes through the process of integrating (and testing!) several customers
and their electronic reporting of the widgets they've bought, along with the company's
response.

T UT O R IALS

90

• Tutorial how to build a Service Oriented Architecture using Camel with OSGI Updated 20/11/2009
The tutorial has been designed in two parts. The first part introduces basic concept to
create a simple SOA solution using Camel and OSGI and deploy it in a OSGI Server
like Apache Felix Karaf and Spring DM Server while the second extends the
ReportIncident tutorial part 4 to show How we can separate the different layers
(domain, service, ...) of an application and deploy them in separate bundles. The Web
Application has also be modified in order to communicate to the OSGI bundles.
• Examples
While not actual tutorials you might find working through the source of the various
Examples useful

TUTORIAL ON SPRING REMOTING WITH JMS

PREFACE
This tutorial aims to guide the reader through the stages of creating a project which uses Camel
to facilitate the routing of messages from a JMS queue to a Spring service. The route works in a
synchronous fashion returning a response to the client.
• Tutorial on Spring Remoting with JMS
• Preface
• Prerequisites
• Distribution
• About
• Create the Camel Project
• Update the POM with Dependencies
• Writing the Server
• Create the Spring Service
• Define the Camel Routes
• Configure Spring
• AOP Enabled Server
• Run the Server
• Writing The Clients
• Client Using The ProducerTemplate
• Client Using Spring Remoting
• Client Using Message Endpoint EIP Pattern
• Run the Clients
• Using the Camel Maven Plugin
• Using Camel JMX
• See Also

91

TU TO RI A L S

Thanks
This tutorial was kindly donated to Apache Camel by Martin Gilday.

PREREQUISITES
This tutorial uses Maven to setup the Camel project and for dependencies for artifacts.

DISTRIBUTION
This sample is distributed with the Camel distribution as examples/camel-examplespring-jms.

ABOUT
This tutorial is a simple example that demonstrates more the fact how well Camel is seamless
integrated with Spring to leverage the best of both worlds. This sample is client server solution
using JMS messaging as the transport. The sample has two flavors of servers and also for clients
demonstrating different techniques for easy communication.
The Server is a JMS message broker that routes incoming messages to a business service that
does computations on the received message and returns a response.
The EIP patterns used in this sample are:
Pattern

Description

Message
Channel

We need a channel so the Clients can communicate with the server.

Message

The information is exchanged using the Camel Message interface.

Message
Translator

This is where Camel shines as the message exchange between the Server and
the Clients are text based strings with numbers. However our business service
uses int for numbers. So Camel can do the message translation automatically.

Message
Endpoint

It should be easy to send messages to the Server from the the clients. This is
archived with Camels powerful Endpoint pattern that even can be more
powerful combined with Spring remoting. The tutorial have clients using each
kind of technique for this.

Point to
Point
Channel

We using JMS queues so there are only one receive of the message exchange

T UT O R IALS

92

Event
Driven
Consumer

Yes the JMS broker is of course event driven and only reacts when the client
sends a message to the server.

We use the following Camel components:
Component

Description

ActiveMQ

We use Apache ActiveMQ as the JMS broker on the Server side

Bean

We use the bean binding to easily route the messages to our business
service. This is a very powerful component in Camel.

File

In the AOP enabled Server we store audit trails as files.

JMS

Used for the JMS messaging

CREATE THE CAMEL PROJECT
mvn archetype:create -DgroupId=org.example -DartifactId=CamelWithJmsAndSpring

Update the POM with Dependencies
First we need to have dependencies for the core Camel jars, its spring, jms components and
finally ActiveMQ as the message broker.


org.apache.camel
camel-core


org.apache.camel
camel-jms


org.apache.camel
camel-spring


org.apache.activemq
activemq-camel


As we use spring xml configuration for the ActiveMQ JMS broker we need this dependency:

93

TU TO RI A L S

For the purposes of the tutorial a single Maven project will be used for both the
client and server. Ideally you would break your application down into the
appropriate components.



org.apache.xbean
xbean-spring


And dependencies for the AOP enable server example. These dependencies are of course only
needed if you need full blown AOP stuff using AspejctJ with bytecode instrumentation.


org.springframework
spring-aop
${spring-version}


org.aspectj
aspectjrt
1.6.2


org.aspectj
aspectjweaver
1.6.2


cglib
cglib-nodep
2.1_3


WRITING THE SERVER
Create the Spring Service
For this example the Spring service (= our business service) on the server will be a simple
multiplier which trebles in the received value.

T UT O R IALS

94

public interface Multiplier {
/**
* Multiplies the given number by a pre-defined constant.
*
* @param originalNumber The number to be multiplied
* @return The result of the multiplication
*/
int multiply(int originalNumber);
}

And the implementation of this service is:
@Service(value = "multiplier")
public class Treble implements Multiplier {
public int multiply(final int originalNumber) {
return originalNumber * 3;
}
}

Notice that this class has been annotated with the @Service spring annotation. This ensures
that this class is registered as a bean in the registry with the given name multiplier.
Define the Camel Routes
public class ServerRoutes extends RouteBuilder {
@Override
public void configure() throws Exception {
// route from the numbers queue to our business that is a spring bean
registered with the id=multiplier
// Camel will introspect the multiplier bean and find the best candidate of
the method to invoke.
// You can add annotations etc to help Camel find the method to invoke.
// As our multiplier bean only have one method its easy for Camel to find the
method to use.
from("jms:queue:numbers").to("multiplier");
// Camel has several ways to configure the same routing, we have defined some
of them here below
// as above but with the bean: prefix
//from("jms:queue:numbers").to("bean:multiplier");
// beanRef is using explicity bean bindings to lookup the multiplier bean and
invoke the multiply method

95

TU TO RI A L S

//from("jms:queue:numbers").beanRef("multiplier", "multiply");
// the same as above but expressed as a URI configuration
//from("activemq:queue:numbers").to("bean:multiplier?methodName=multiply");
}
}

This defines a Camel route from the JMS queue named numbers to the Spring bean named
multiplier. Camel will create a consumer to the JMS queue which forwards all received
messages onto the the Spring bean, using the method named multiply.
Configure Spring
The Spring config file is placed under META-INF/spring as this is the default location used
by the Camel Maven Plugin, which we will later use to run our server.
First we need to do the standard scheme declarations in the top. In the camel-server.xml we
are using spring beans as the default bean: namespace and springs context:. For configuring
ActiveMQ we use broker: and for Camel we of course have camel:. Notice that we don't
use version numbers for the camel-spring schema. At runtime the schema is resolved in the
Camel bundle. If we use a specific version number such as 1.4 then its IDE friendly as it would
be able to import it and provide smart completion etc. See Xml Reference for further details.


We use Spring annotations for doing IoC dependencies and its component-scan features comes
to the rescue as it scans for spring annotations in the given package name:



Camel will of course not be less than Spring in this regard so it supports a similar feature for
scanning of Routes. This is configured as shown below.

T UT O R IALS

96

Notice that we also have enabled the JMXAgent so we will be able to introspect the Camel
Server with a JMX Console.


org.apache.camel.example.server





The ActiveMQ JMS broker is also configured in this xml file. We set it up to listen on TCP port
61610.







As this examples uses JMS then Camel needs a JMS component that is connected with the
ActiveMQ broker. This is configured as shown below:





Notice: The JMS component is configured in standard Spring beans, but the gem is that the
bean id can be referenced from Camel routes - meaning we can do routing using the JMS
Component by just using jms: prefix in the route URI. What happens is that Camel will find in
the Spring Registry for a bean with the id="jms". Since the bean id can have arbitrary name you
could have named it id="jmsbroker" and then referenced to it in the routing as
from="jmsbroker:queue:numbers).to("multiplier");
We use the vm protocol to connect to the ActiveMQ server as its embedded in this
application.

97

componentscan

Defines the package to be scanned for Spring stereotype annotations, in this
case, to load the "multiplier" bean

camelcontext

Defines the package to be scanned for Camel routes. Will find the
ServerRoutes class and create the routes contained within it

jms bean

Creates the Camel JMS component

TU TO RI A L S

AOP Enabled Server
The example has an enhanced Server example that uses fullblown AspejctJ AOP for doing a
audit tracking of invocations of the business service.
We leverage Spring AOP support in the {{camel-server-aop.xml} configuration file. First we
must declare the correct XML schema's to use:


Then we include all the existing configuration from the normal server example:

















Then we enable the AspejctJ AOP auto proxy feature of Spring that will scan for classes
annotated with the @Aspect annotation:

T UT O R IALS

98




Then we define our Audit tracker bean that does the actual audit logging. It's also the class that
is annotated with the @Aspect so Spring will pick this up, as the aspect.






And the gem is that we inject the AuditTracker aspect bean with a Camel endpoint that defines
where the audit should be stored. Noticed how easy it is to setup as we have just defined an
endpoint URI that is file based, meaning that we stored the audit tracks as files. We can change
this tore to any Camel components as we wish. To store it on a JMS queue simply change the
URI to jms:queue:audit.


org.apache.camel.example.server







And the full blown Aspejct for the audit tracker java code:
/**
* For audit tracking of all incoming invocations of our business (Multiplier)
*/
@Aspect
public class AuditTracker {
// endpoint we use for backup store of audit tracks
private Endpoint store;
@Required
public void setStore(Endpoint store) {
this.store = store;
}

99

TU TO RI A L S

@Before("execution(int org.apache.camel.example.server.Multiplier.multiply(int))
&& args(originalNumber)")
public void audit(int originalNumber) throws Exception {
String msg = "Someone called us with this number " + originalNumber;
System.out.println(msg);
// now send the message to the backup store using the Camel Message Endpoint
pattern
Exchange exchange = store.createExchange();
exchange.getIn().setBody(msg);
store.createProducer().process(exchange);
}
}

Run the Server
The Server is started using the org.apache.camel.spring.Main class that can start
camel-spring application out-of-the-box. The Server can be started in several flavors:
▪ as a standard java main application - just start the
org.apache.camel.spring.Main class
▪ using maven jave:exec
▪ using camel:run
In this sample as there are two servers (with and without AOP) we have prepared some
profiles in maven to start the Server of your choice.
The server is started with:
mvn compile exec:java -PCamelServer
Or for the AOP enabled Server example:
mvn compile exec:java -PCamelServerAOP

WRITING THE CLIENTS
This sample has three clients demonstrating different Camel techniques for communication
▪ CamelClient using the ProducerTemplate for Spring template style coding
▪ CamelRemoting using Spring Remoting
▪ CamelEndpoint using the Message Endpoint EIP pattern using a neutral Camel API
Client Using The ProducerTemplate
We will initially create a client by directly using ProducerTemplate. We will later create a
client which uses Spring remoting to hide the fact that messaging is being used.

T UT O R IALS

100












The client will not use the Camel Maven Plugin so the Spring XML has been placed in src/main/
resources to not conflict with the server configs.
camelContext

The Camel context is defined but does not contain any routes

template

The ProducerTemplate is used to place messages onto the JMS queue

jms bean

This initialises the Camel JMS component, allowing us to place messages
onto the queue

And the CamelClient source code:
public static void main(final String[] args) throws Exception {
System.out.println("Notice this client requires that the CamelServer is already
running!");
ApplicationContext context = new
ClassPathXmlApplicationContext("camel-client.xml");
// get the camel template for Spring template style sending of messages (=
producer)
ProducerTemplate camelTemplate = (ProducerTemplate)
context.getBean("camelTemplate");
System.out.println("Invoking the multiply with 22");
// as opposed to the CamelClientRemoting example we need to define the service URI
in this java code
int response = (Integer)camelTemplate.sendBody("jms:queue:numbers",
ExchangePattern.InOut, 22);
System.out.println("... the result is: " + response);

101

TU TO RI A L S

System.exit(0);
}

The ProducerTemplate is retrieved from a Spring ApplicationContext and used to
manually place a message on the "numbers" JMS queue. The exchange pattern
(ExchangePattern.InOut) states that the call should be synchronous, and that we will
receive a response.
Before running the client be sure that both the ActiveMQ broker and the CamelServer
are running.
Client Using Spring Remoting
Spring Remoting "eases the development of remote-enabled services". It does this by allowing
you to invoke remote services through your regular Java interface, masking that a remote
service is being called.



The snippet above only illustrates the different and how Camel easily can setup and use Spring
Remoting in one line configurations.
The proxy will create a proxy service bean for you to use to make the remote invocations.
The serviceInterface property details which Java interface is to be implemented by the
proxy. serviceUrl defines where messages sent to this proxy bean will be directed. Here we
define the JMS endpoint with the "numbers" queue we used when working with Camel template
directly. The value of the id property is the name that will be the given to the bean when it is
exposed through the Spring ApplicationContext. We will use this name to retrieve the
service in our client. I have named the bean multiplierProxy simply to highlight that it is not the
same multiplier bean as is being used by CamelServer. They are in completely independent
contexts and have no knowledge of each other. As you are trying to mask the fact that
remoting is being used in a real application you would generally not include proxy in the name.
And the Java client source code:
public static void main(final String[] args) {
System.out.println("Notice this client requires that the CamelServer is already
running!");
ApplicationContext context = new
ClassPathXmlApplicationContext("camel-client-remoting.xml");
// just get the proxy to the service and we as the client can use the "proxy" as

T UT O R IALS

102

it was
// a local object we are invoking. Camel will under the covers do the remote
communication
// to the remote ActiveMQ server and fetch the response.
Multiplier multiplier = (Multiplier)context.getBean("multiplierProxy");
System.out.println("Invoking the multiply with 33");
int response = multiplier.multiply(33);
System.out.println("... the result is: " + response);
System.exit(0);
}

Again, the client is similar to the original client, but with some important differences.
1. The Spring context is created with the new camel-client-remoting.xml
2. We retrieve the proxy bean instead of a ProducerTemplate. In a non-trivial
example you would have the bean injected as in the standard Spring manner.
3. The multiply method is then called directly. In the client we are now working to an
interface. There is no mention of Camel or JMS inside our Java code.
Client Using Message Endpoint EIP Pattern
This client uses the Message Endpoint EIP pattern to hide the complexity to communicate to
the Server. The Client uses the same simple API to get hold of the endpoint, create an
exchange that holds the message, set the payload and create a producer that does the send and
receive. All done using the same neutral Camel API for all the components in Camel. So if the
communication was socket TCP based you just get hold of a different endpoint and all the java
code stays the same. That is really powerful.
Okay enough talk, show me the code!
public static void main(final String[] args) throws Exception {
System.out.println("Notice this client requires that the CamelServer is already
running!");
ApplicationContext context = new
ClassPathXmlApplicationContext("camel-client.xml");
CamelContext camel = (CamelContext) context.getBean("camel");
// get the endpoint from the camel context
Endpoint endpoint = camel.getEndpoint("jms:queue:numbers");
// create the exchange used for the communication
// we use the in out pattern for a synchronized exchange where we expect a response
Exchange exchange = endpoint.createExchange(ExchangePattern.InOut);
// set the input on the in body
// must you correct type to match the expected type of an Integer object
exchange.getIn().setBody(11);

103

TU TO RI A L S

// to send the exchange we need an producer to do it for us
Producer producer = endpoint.createProducer();
// start the producer so it can operate
producer.start();
// let the producer process the exchange where it does all the work in this
oneline of code
System.out.println("Invoking the multiply with 11");
producer.process(exchange);
// get the response from the out body and cast it to an integer
int response = exchange.getOut().getBody(Integer.class);
System.out.println("... the result is: " + response);
// stop and exit the client
producer.stop();
System.exit(0);
}

Switching to a different component is just a matter of using the correct endpoint. So if we had
defined a TCP endpoint as: "mina:tcp://localhost:61610" then its just a matter of
getting hold of this endpoint instead of the JMS and all the rest of the java code is exactly the
same.
Run the Clients
The Clients is started using their main class respectively.
▪ as a standard java main application - just start their main class
▪ using maven jave:exec
In this sample we start the clients using maven:
mvn compile exec:java -PCamelClient
mvn compile exec:java -PCamelClientRemoting
mvn compile exec:java -PCamelClientEndpoint
Also see the Maven pom.xml file how the profiles for the clients is defined.

USING THE CAMEL MAVEN PLUGIN
The Camel Maven Plugin allows you to run your Camel routes directly from Maven. This
negates the need to create a host application, as we did with Camel server, simply to start up
the container. This can be very useful during development to get Camel routes running quickly.
Listing 7. pom.xml




T UT O R IALS

104

org.apache.camel
camel-maven-plugin




All that is required is a new plugin definition in your Maven POM. As we have already placed
our Camel config in the default location (camel-server.xml has been placed in META-INF/
spring/) we do not need to tell the plugin where the route definitions are located. Simply run
mvn camel:run.

USING CAMEL JMX
Camel has extensive support for JMX and allows us to inspect the Camel Server at runtime. As
we have enabled the JMXAgent in our tutorial we can fire up the jconsole and connect to the
following service URI: service:jmx:rmi:///jndi/rmi://localhost:1099/
jmxrmi/camel. Notice that Camel will log at INFO level the JMX Connector URI:
...
DefaultInstrumentationAgent
INFO JMX connector thread started on
service:jmx:rmi:///jndi/rmi://claus-acer:1099/jmxrmi/camel
...

In the screenshot below we can see the route and its performance metrics:

SEE ALSO
• Spring Remoting with JMS Example on Amin Abbaspour's Weblog

105

TU TO RI A L S

TUTORIAL - CAMEL-EXAMPLE-REPORTINCIDENT
INTRODUCTION
Creating this tutorial was inspired by a real life use-case I discussed over the phone with a
colleague. He was working at a client whom uses a heavy-weight integration platform from a
very large vendor. He was in talks with developer shops to implement a new integration on this
platform. His trouble was the shop tripled the price when they realized the platform of choice.
So I was wondering how we could do this integration with Camel. Can it be done, without
tripling the cost
.
This tutorial is written during the development of the integration. I have decided to start off
with a sample that isn't Camel's but standard Java and then plugin Camel as we goes. Just as
when people needed to learn Spring you could consume it piece by piece, the same goes with
Camel.
The target reader is person whom hasn't experience or just started using Camel.

MOTIVATION FOR THIS TUTORIAL
I wrote this tutorial motivated as Camel lacked an example application that was based on the
web application deployment model. The entire world hasn't moved to pure OSGi deployments
yet.

THE USE-CASE
The goal is to allow staff to report incidents into a central administration. For that they use
client software where they report the incident and submit it to the central administration. As
this is an integration in a transition phase the administration should get these incidents by email
whereas they are manually added to the database. The client software should gather the
incident and submit the information to the integration platform that in term will transform the
report into an email and send it to the central administrator for manual processing.
The figure below illustrates this process. The end users reports the incidents using the client
applications. The incident is sent to the central integration platform as webservice. The
integration platform will process the incident and send an OK acknowledgment back to the
client. Then the integration will transform the message to an email and send it to the
administration mail server. The users in the administration will receive the emails and take it
from there.

T UT O R IALS

106

In EIP patterns
We distill the use case as EIP patterns:

PARTS
This tutorial is divided into sections and parts:
Section A: Existing Solution, how to slowly use Camel
Part 1 - This first part explain how to setup the project and get a webservice exposed using
Apache CXF. In fact we don't touch Camel yet.
Part 2 - Now we are ready to introduce Camel piece by piece (without using Spring or any
XML configuration file) and create the full feature integration. This part will introduce different
Camel's concepts and How we can build our solution using them like :
▪ CamelContext
▪ Endpoint, Exchange & Producer
▪ Components : Log, File
Part 3 - Continued from part 2 where we implement that last part of the solution with the
event driven consumer and how to send the email through the Mail component.
Section B: The Camel Solution
Part 4 - We now turn into the path of Camel where it excels - the routing.
Part 5 - Is about how embed Camel with Spring and using CXF endpoints directly in Camel

LINKS
▪ Introduction
▪ Part 1
▪ Part 2

107

TU TO RI A L S

Using Axis 2
See this blog entry by Sagara demonstrating how to use Apache Axis 2 instead of
Apache CXF as the web service framework.
▪ Part 3
▪ Part 4
▪ Part 5

PART 1
PREREQUISITES
This tutorial uses the following frameworks:
• Maven 2.0.9
• Apache Camel 1.4.0
• Apache CXF 2.1.1
• Spring 2.5.5
Note: The sample project can be downloaded, see the resources section.

INITIAL PROJECT SETUP
We want the integration to be a standard .war application that can be deployed in any web
container such as Tomcat, Jetty or even heavy weight application servers such as WebLogic or
WebSphere. There fore we start off with the standard Maven webapp project that is created
with the following long archetype command:
mvn archetype:create -DgroupId=org.apache.camel
-DartifactId=camel-example-reportincident -DarchetypeArtifactId=maven-archetype-webapp

Notice that the groupId etc. doens't have to be org.apache.camel it can be
com.mycompany.whatever. But I have used these package names as the example is an official
part of the Camel distribution.
Then we have the basic maven folder layout. We start out with the webservice part where
we want to use Apache CXF for the webservice stuff. So we add this to the pom.xml

2.1.1


T UT O R IALS

108


org.apache.cxf
cxf-rt-core
${cxf-version}


org.apache.cxf
cxf-rt-frontend-jaxws
${cxf-version}


org.apache.cxf
cxf-rt-transports-http
${cxf-version}


DEVELOPING THE WEBSERVICE
As we want to develop webservice with the contract first approach we create our .wsdl file. As
this is a example we have simplified the model of the incident to only include 8 fields. In real life
the model would be a bit more complex, but not to much.
We put the wsdl file in the folder src/main/webapp/WEB-INF/wsdl and name the
file report_incident.wsdl.

























































T UT O R IALS

110








CXF wsdl2java
Then we integration the CXF wsdl2java generator in the pom.xml so we have CXF generate
the needed POJO classes for our webservice contract.
However at first we must configure maven to live in the modern world of Java 1.5 so we must
add this to the pom.xml


org.apache.maven.plugins
maven-compiler-plugin

1.5
1.5



And then we can add the CXF wsdl2java code generator that will hook into the compile goal so
its automatic run all the time:


org.apache.cxf
cxf-codegen-plugin
${cxf-version}


generate-sources
generate-sources

${basedir}/target/
generated/src/main/java


${basedir}/src/main/webapp/WEB-INF/wsdl/report_incident.wsdl



111

TU TO RI A L S



wsdl2java





You are now setup and should be able to compile the project. So running the mvn compile
should run the CXF wsdl2java and generate the source code in the folder &{basedir}/
target/generated/src/main/java that we specified in the pom.xml above. Since its in
the target/generated/src/main/java maven will pick it up and include it in the build
process.
Configuration of the web.xml
Next up is to configure the web.xml to be ready to use CXF so we can expose the webservice.
As Spring is the center of the universe, or at least is a very important framework in today's Java
land we start with the listener that kick-starts Spring. This is the usual piece of code:


org.springframework.web.context.ContextLoaderListener


And then we have the CXF part where we define the CXF servlet and its URI mappings to
which we have chosen that all our webservices should be in the path /webservices/


CXFServlet
org.apache.cxf.transport.servlet.CXFServlet
1



CXFServlet
/webservices/*


Then the last piece of the puzzle is to configure CXF, this is done in a spring XML that we link
to fron the web.xml by the standard Spring contextConfigLocation property in the
web.xml

T UT O R IALS

112



contextConfigLocation
classpath:cxf-config.xml


We have named our CXF configuration file cxf-config.xml and its located in the root of
the classpath. In Maven land that is we can have the cxf-config.xml file in the src/
main/resources folder. We could also have the file located in the WEB-INF folder for
instance /WEB-INF/cxf-config.xml.
Getting rid of the old jsp world
The maven archetype that created the basic folder structure also created a sample .jsp file
index.jsp. This file src/main/webapp/index.jsp should be deleted.
Configuration of CXF
The cxf-config.xml is as follows:










The configuration is standard CXF and is documented at the Apache CXF website.
The 3 import elements is needed by CXF and they must be in the file.
113

TU TO RI A L S

Noticed that we have a spring bean reportIncidentEndpoint that is the implementation
of the webservice endpoint we let CXF expose.
Its linked from the jaxws element with the implementator attribute as we use the # mark to
identify its a reference to a spring bean. We could have stated the classname directly as
implementor="org.apache.camel.example.reportincident.ReportIncidentEndpoint
but then we lose the ability to let the ReportIncidentEndpoint be configured by spring.
The address attribute defines the relative part of the URL of the exposed webservice.
wsdlLocation is an optional parameter but for persons like me that likes contract-first we
want to expose our own .wsdl contracts and not the auto generated by the frameworks, so
with this attribute we can link to the real .wsdl file. The last stuff is needed by CXF as you could
have several services so it needs to know which this one is. Configuring these is quite easy as all
the information is in the wsdl already.
Implementing the ReportIncidentEndpoint
Phew after all these meta files its time for some java code so we should code the implementor
of the webservice. So we fire up mvn compile to let CXF generate the POJO classes for our
webservice and we are ready to fire up a Java editor.
You can use mvn idea:idea or mvn eclipse:eclipse to create project files for
these editors so you can load the project. However IDEA has been smarter lately and can load
a pom.xml directly.
As we want to quickly see our webservice we implement just a quick and dirty as it can get.
At first beware that since its jaxws and Java 1.5 we get annotations for the money, but they
reside on the interface so we can remove them from our implementations so its a nice plain
POJO again:
package org.apache.camel.example.reportincident;
/**
* The webservice we have implemented.
*/
public class ReportIncidentEndpointImpl implements ReportIncidentEndpoint {
public OutputReportIncident reportIncident(InputReportIncident parameters) {
System.out.println("Hello ReportIncidentEndpointImpl is called from " +
parameters.getGivenName());
OutputReportIncident out = new OutputReportIncident();
out.setCode("OK");
return out;
}
}

We just output the person that invokes this webservice and returns a OK response. This class
should be in the maven source root folder src/main/java under the package name

T UT O R IALS

114

org.apache.camel.example.reportincident. Beware that the maven archetype
tool didn't create the src/main/java folder, so you should create it manually.
To test if we are home free we run mvn clean compile.
Running our webservice
Now that the code compiles we would like to run it in a web container, so we add jetty to our
pom.xml so we can run mvn jetty:run:

...
6.1.1



...


org.mortbay.jetty
maven-jetty-plugin
${jetty-version}


Notice: We use Jetty v6.1.1 as never versions has troubles on my laptop. Feel free to try a
newer version on your system, but v6.1.1 works flawless.
So to see if everything is in order we fire up jetty with mvn jetty:run and if everything
is okay you should be able to access http://localhost:8080.
Jetty is smart that it will list the correct URI on the page to our web application, so just click on
the link. This is smart as you don't have to remember the exact web context URI for your
application - just fire up the default page and Jetty will help you.
So where is the damn webservice then? Well as we did configure the web.xml to instruct
the CXF servlet to accept the pattern /webservices/* we should hit this URL to get the
attention of CXF: http://localhost:8080/camel-example-reportincident/
webservices.

115

TU TO RI A L S

Hitting the webservice
Now we have the webservice running in a standard .war application in a standard web container
such as Jetty we would like to invoke the webservice and see if we get our code executed.
Unfortunately this isn't the easiest task in the world - its not so easy as a REST URL, so we
need tools for this. So we fire up our trusty webservice tool SoapUI and let it be the one to fire
the webservice request and see the response.
Using SoapUI we sent a request to our webservice and we got the expected OK response
and the console outputs the System.out so we are ready to code.

Remote Debugging
Okay a little sidestep but wouldn't it be cool to be able to debug your code when its fired up
under Jetty? As Jetty is started from maven, we need to instruct maven to use debug mode.
Se we set the MAVEN_OPTS environment to start in debug mode and listen on port 5005.
MAVEN_OPTS=-Xmx512m -XX:MaxPermSize=128m -Xdebug
-Xrunjdwp:transport=dt_socket,server=y,suspend=n,address=5005

Then you need to restart Jetty so its stopped with ctrl + c. Remember to start a new shell to
pickup the new environment settings. And start jetty again.
Then we can from our IDE attach a remote debugger and debug as we want.
First we configure IDEA to attach to a remote debugger on port 5005:

T UT O R IALS

116

Then we set a breakpoint in our code ReportIncidentEndpoint and hit the SoapUI
once again and we are breaked at the breakpoint where we can inspect the parameters:

117

TU TO RI A L S

Adding a unit test
Oh so much hard work just to hit a webservice, why can't we just use an unit test to invoke
our webservice? Yes of course we can do this, and that's the next step.
First we create the folder structure src/test/java and src/test/resources. We
then create the unit test in the src/test/java folder.
package org.apache.camel.example.reportincident;
import junit.framework.TestCase;
/**
* Plain JUnit test of our webservice.
*/
public class ReportIncidentEndpointTest extends TestCase {
}

T UT O R IALS

118

Here we have a plain old JUnit class. As we want to test webservices we need to start and
expose our webservice in the unit test before we can test it. And JAXWS has pretty decent
methods to help us here, the code is simple as:
import javax.xml.ws.Endpoint;
...
private static String ADDRESS = "http://localhost:9090/unittest";
protected void startServer() throws Exception {
// We need to start a server that exposes or webservice during the unit testing
// We use jaxws to do this pretty simple
ReportIncidentEndpointImpl server = new ReportIncidentEndpointImpl();
Endpoint.publish(ADDRESS, server);
}

The Endpoint class is the javax.xml.ws.Endpoint that under the covers looks for a
provider and in our case its CXF - so its CXF that does the heavy lifting of exposing out
webservice on the given URL address. Since our class ReportIncidentEndpointImpl implements
the interface ReportIncidentEndpoint that is decorated with all the jaxws annotations it
got all the information it need to expose the webservice. Below is the CXF wsdl2java generated
interface:

/*
*
*/
package org.apache.camel.example.reportincident;
import
import
import
import
import
import
import

javax.jws.WebMethod;
javax.jws.WebParam;
javax.jws.WebResult;
javax.jws.WebService;
javax.jws.soap.SOAPBinding;
javax.jws.soap.SOAPBinding.ParameterStyle;
javax.xml.bind.annotation.XmlSeeAlso;

/**
* This class was generated by Apache CXF 2.1.1
* Wed Jul 16 12:40:31 CEST 2008
* Generated source version: 2.1.1
*
*/
/*
*
*/

@WebService(targetNamespace = "http://reportincident.example.camel.apache.org", name =

119

TU TO RI A L S

"ReportIncidentEndpoint")
@XmlSeeAlso({ObjectFactory.class})
@SOAPBinding(parameterStyle = SOAPBinding.ParameterStyle.BARE)
public interface ReportIncidentEndpoint {
/*
*
*/
@SOAPBinding(parameterStyle = SOAPBinding.ParameterStyle.BARE)
@WebResult(name = "outputReportIncident", targetNamespace =
"http://reportincident.example.camel.apache.org", partName = "parameters")
@WebMethod(operationName = "ReportIncident", action =
"http://reportincident.example.camel.apache.org/ReportIncident")
public OutputReportIncident reportIncident(
@WebParam(partName = "parameters", name = "inputReportIncident",
targetNamespace = "http://reportincident.example.camel.apache.org")
InputReportIncident parameters
);
}

Next up is to create a webservice client so we can invoke our webservice. For this we actually
use the CXF framework directly as its a bit more easier to create a client using this framework
than using the JAXWS style. We could have done the same for the server part, and you should
do this if you need more power and access more advanced features.
import org.apache.cxf.jaxws.JaxWsProxyFactoryBean;
...
protected ReportIncidentEndpoint createCXFClient() {
// we use CXF to create a client for us as its easier than JAXWS and works
JaxWsProxyFactoryBean factory = new JaxWsProxyFactoryBean();
factory.setServiceClass(ReportIncidentEndpoint.class);
factory.setAddress(ADDRESS);
return (ReportIncidentEndpoint) factory.create();
}

So now we are ready for creating a unit test. We have the server and the client. So we just
create a plain simple unit test method as the usual junit style:
public void testRendportIncident() throws Exception {
startServer();
ReportIncidentEndpoint client = createCXFClient();
InputReportIncident input = new InputReportIncident();
input.setIncidentId("123");
input.setIncidentDate("2008-07-16");
input.setGivenName("Claus");

T UT O R IALS

120

input.setFamilyName("Ibsen");
input.setSummary("bla bla");
input.setDetails("more bla bla");
input.setEmail("davsclaus@apache.org");
input.setPhone("+45 2962 7576");
OutputReportIncident out = client.reportIncident(input);
assertEquals("Response code is wrong", "OK", out.getCode());
}

Now we are nearly there. But if you run the unit test with mvn test then it will fail. Why!!!
Well its because that CXF needs is missing some dependencies during unit testing. In fact it
needs the web container, so we need to add this to our pom.xml.


org.apache.cxf
cxf-rt-transports-http-jetty
${cxf-version}
test


Well what is that, CXF also uses Jetty for unit test - well its just shows how agile, embedable
and popular Jetty is.
So lets run our junit test with, and it reports:
mvn test
Tests run: 1, Failures: 0, Errors: 0, Skipped: 0
[INFO] BUILD SUCCESSFUL

Yep thats it for now. We have a basic project setup.

END OF PART 1
Thanks for being patient and reading all this more or less standard Maven, Spring, JAXWS and
Apache CXF stuff. Its stuff that is well covered on the net, but I wanted a full fledged tutorial on
a maven project setup that is web service ready with Apache CXF. We will use this as a base
for the next part where we demonstrate how Camel can be digested slowly and piece by piece
just as it was back in the times when was introduced and was learning the Spring framework
that we take for granted today.

RESOURCES
• Apache CXF user guide

121

TU TO RI A L S

Name

tutorial_reportincident_partone.zi...

Size

Creator

Creation
Date

14
kB

Claus
Ibsen

Jul 17,
2008 23:34

Comment

LINKS
▪
▪
▪
▪
▪
▪

Introduction
Part 1
Part 2
Part 3
Part 4
Part 5

PART 2
ADDING CAMEL
In this part we will introduce Camel so we start by adding Camel to our pom.xml:

...
1.4.0



org.apache.camel
camel-core
${camel-version}


That's it, only one dependency for now.
Now we turn towards our webservice endpoint implementation where we want to let Camel
have a go at the input we receive. As Camel is very non invasive its basically a .jar file then we
can just grap Camel but creating a new instance of DefaultCamelContext that is the
hearth of Camel its context.
CamelContext camel = new DefaultCamelContext();

In fact we create a constructor in our webservice and add this code:

T UT O R IALS

122

Synchronize IDE
If you continue from part 1, remember to update your editor project settings since
we have introduce new .jar files. For instance IDEA has a feature to synchronize
with Maven projects.

private CamelContext camel;
public ReportIncidentEndpointImpl() throws Exception {
// create the camel context that is the "heart" of Camel
camel = new DefaultCamelContext();
// add the log component
camel.addComponent("log", new LogComponent());
// start Camel
camel.start();
}

LOGGING THE "HELLO WORLD"
Here at first we want Camel to log the givenName and familyName parameters we
receive, so we add the LogComponent with the key log. And we must start Camel before
its ready to act.
Then we change the code in the method that is invoked by Apache CXF when a webservice
request arrives. We get the name and let Camel have a go at it in the new method we create
sendToCamel:
public OutputReportIncident reportIncident(InputReportIncident parameters) {
String name = parameters.getGivenName() + " " + parameters.getFamilyName();
// let Camel do something with the name
sendToCamelLog(name);
OutputReportIncident out = new OutputReportIncident();
out.setCode("OK");
return out;
}

Next is the Camel code. At first it looks like there are many code lines to do a simple task of
logging the name - yes it is. But later you will in fact realize this is one of Camels true power. Its
concise API. Hint: The same code can be used for any component in Camel.

123

TU TO RI A L S

Component Documentation
The Log and File components is documented as well, just click on the links. Just
return to this documentation later when you must use these components for real.

private void sendToCamelLog(String name) {
try {
// get the log component
Component component = camel.getComponent("log");
//
//
//
//

create an endpoint and configure it.
Notice the URI parameters this is a common pratice in Camel to configure
endpoints based on URI.
com.mycompany.part2 = the log category used. Will log at INFO level as

default
Endpoint endpoint = component.createEndpoint("log:com.mycompany.part2");
// create an Exchange that we want to send to the endpoint
Exchange exchange = endpoint.createExchange();
// set the in message payload (=body) with the name parameter
exchange.getIn().setBody(name);
// now we want to send the exchange to this endpoint and we then need a
producer
// for this, so we create and start the producer.
Producer producer = endpoint.createProducer();
producer.start();
// process the exchange will send the exchange to the log component, that
will process
// the exchange and yes log the payload
producer.process(exchange);
// stop the producer, we want to be nice and cleanup
producer.stop();

} catch (Exception e) {
// we ignore any exceptions and just rethrow as runtime
throw new RuntimeException(e);
}
}

Okay there are code comments in the code block above that should explain what is happening.
We run the code by invoking our unit test with maven mvn test, and we should get this log
line:

T UT O R IALS

124

INFO: Exchange[BodyType:String, Body:Claus Ibsen]

WRITE TO FILE - EASY WITH THE SAME CODE STYLE
Okay that isn't to impressive, Camel can log
Well I promised that the above code style can
be used for any component, so let's store the payload in a file. We do this by adding the file
component to the Camel context
// add the file component
camel.addComponent("file", new FileComponent());

And then we let camel write the payload to the file after we have logged, by creating a new
method sendToCamelFile. We want to store the payload in filename with the incident id so
we need this parameter also:
// let Camel do something with the name
sendToCamelLog(name);
sendToCamelFile(parameters.getIncidentId(), name);

And then the code that is 99% identical. We have change the URI configuration when we create
the endpoint as we pass in configuration parameters to the file component.
And then we need to set the output filename and this is done by adding a special header to the
exchange. That's the only difference:
private void sendToCamelFile(String incidentId, String name) {
try {
// get the file component
Component component = camel.getComponent("file");
// create an endpoint and configure it.
// Notice the URI parameters this is a common pratice in Camel to configure
// endpoints based on URI.
// file://target instructs the base folder to output the files. We put in
the target folder
// then its actumatically cleaned by mvn clean
Endpoint endpoint = component.createEndpoint("file://target");
// create an Exchange that we want to send to the endpoint
Exchange exchange = endpoint.createExchange();
// set the in message payload (=body) with the name parameter
exchange.getIn().setBody(name);
// now a special header is set to instruct the file component what the
output filename
// should be

125

TU TO RI A L S

exchange.getIn().setHeader(FileComponent.HEADER_FILE_NAME, "incident-" +
incidentId + ".txt");
// now we want to send the exchange to this endpoint and we then need a
producer
// for this, so we create and start the producer.
Producer producer = endpoint.createProducer();
producer.start();
// process the exchange will send the exchange to the file component, that
will process
// the exchange and yes write the payload to the given filename
producer.process(exchange);
// stop the producer, we want to be nice and cleanup
producer.stop();
} catch (Exception e) {
// we ignore any exceptions and just rethrow as runtime
throw new RuntimeException(e);
}
}

After running our unit test again with mvn test we have a output file in the target folder:
D:\demo\part-two>type target\incident-123.txt
Claus Ibsen

FULLY JAVA BASED CONFIGURATION OF ENDPOINTS
In the file example above the configuration was URI based. What if you want 100% java setter
based style, well this is of course also possible. We just need to cast to the component specific
endpoint and then we have all the setters available:
// create the file endpoint, we cast to FileEndpoint because then we can do
// 100% java settter based configuration instead of the URI sting based
// must pass in an empty string, or part of the URI configuration if
wanted
FileEndpoint endpoint = (FileEndpoint)component.createEndpoint("");
endpoint.setFile(new File("target/subfolder"));
endpoint.setAutoCreate(true);

That's it. Now we have used the setters to configure the FileEndpoint that it should store
the file in the folder target/subfolder. Of course Camel now stores the file in the subfolder.
D:\demo\part-two>type target\subfolder\incident-123.txt
Claus Ibsen

T UT O R IALS

126

LESSONS LEARNED
Okay I wanted to demonstrate how you can be in 100% control of the configuration and usage
of Camel based on plain Java code with no hidden magic or special XML or other configuration
files. Just add the camel-core.jar and you are ready to go.
You must have noticed that the code for sending a message to a given endpoint is the same
for both the log and file, in fact any Camel endpoint. You as the client shouldn't bother with
component specific code such as file stuff for file components, jms stuff for JMS messaging etc.
This is what the Message Endpoint EIP pattern is all about and Camel solves this very very nice a key pattern in Camel.

REDUCING CODE LINES
Now that you have been introduced to Camel and one of its masterpiece patterns solved
elegantly with the Message Endpoint its time to give productive and show a solution in fewer
code lines, in fact we can get it down to 5, 4, 3, 2 .. yes only 1 line of code.
The key is the ProducerTemplate that is a Spring'ish xxxTemplate based producer.
Meaning that it has methods to send messages to any Camel endpoints. First of all we need to
get hold of such a template and this is done from the CamelContext
private ProducerTemplate template;
public ReportIncidentEndpointImpl() throws Exception {
...
// get the ProducerTemplate thst is a Spring'ish xxxTemplate based producer
for very
// easy sending exchanges to Camel.
template = camel.createProducerTemplate();
// start Camel
camel.start();
}

Now we can use template for sending payloads to any endpoint in Camel. So all the logging
gabble can be reduced to:
template.sendBody("log:com.mycompany.part2.easy", name);

And the same goes for the file, but we must also send the header to instruct what the output
filename should be:
String filename = "easy-incident-" + incidentId + ".txt";
template.sendBodyAndHeader("file://target/subfolder", name,
FileComponent.HEADER_FILE_NAME, filename);

127

TU TO RI A L S

REDUCING EVEN MORE CODE LINES
Well we got the Camel code down to 1-2 lines for sending the message to the component that
does all the heavy work of wring the message to a file etc. But we still got 5 lines to initialize
Camel.
camel = new DefaultCamelContext();
camel.addComponent("log", new LogComponent());
camel.addComponent("file", new FileComponent());
template = camel.createProducerTemplate();
camel.start();

This can also be reduced. All the standard components in Camel is auto discovered on-the-fly
so we can remove these code lines and we are down to 3 lines.
Okay back to the 3 code lines:
camel = new DefaultCamelContext();
template = camel.createProducerTemplate();
camel.start();

Later will we see how we can reduce this to ... in fact 0 java code lines. But the 3 lines will do
for now.

MESSAGE TRANSLATION
Okay lets head back to the over goal of the integration. Looking at the EIP diagrams at the
introduction page we need to be able to translate the incoming webservice to an email. Doing
so we need to create the email body. When doing the message translation we could put up our
sleeves and do it manually in pure java with a StringBuilder such as:
private String createMailBody(InputReportIncident parameters) {
StringBuilder sb = new StringBuilder();
sb.append("Incident ").append(parameters.getIncidentId());
sb.append(" has been reported on the ").append(parameters.getIncidentDate());
sb.append(" by ").append(parameters.getGivenName());
sb.append(" ").append(parameters.getFamilyName());
// and the rest of the mail body with more appends to the string builder
return sb.toString();
}

But as always it is a hardcoded template for the mail body and the code gets kinda ugly if the
mail message has to be a bit more advanced. But of course it just works out-of-the-box with
just classes already in the JDK.

T UT O R IALS

128

Component auto discovery
When an endpoint is requested with a scheme that Camel hasn't seen before it will
try to look for it in the classpath. It will do so by looking for special Camel
component marker files that reside in the folder META-INF/services/org/
apache/camel/component. If there are files in this folder it will read them as
the filename is the scheme part of the URL. For instance the log component is
defined in this file META-INF/services/org/apache/component/log
and its content is:
class=org.apache.camel.component.log.LogComponent

The class property defines the component implementation.
Tip: End-users can create their 3rd party components using the same technique and have
them been auto discovered on-the-fly.

Lets use a template language instead such as Apache Velocity. As Camel have a component
for Velocity integration we will use this component. Looking at the Component List overview
we can see that camel-velocity component uses the artifactId camel-velocity so therefore
we need to add this to the pom.xml

org.apache.camel
camel-velocity
${camel-version}


And now we have a Spring conflict as Apache CXF is dependent on Spring 2.0.8 and camelvelocity is dependent on Spring 2.5.5. To remedy this we could wrestle with the pom.xml
with excludes settings in the dependencies or just bring in another dependency camelspring:

org.apache.camel
camel-spring
${camel-version}


In fact camel-spring is such a vital part of Camel that you will end up using it in nearly all
situations - we will look into how well Camel is seamless integration with Spring in part 3. For
now its just another dependency.

129

TU TO RI A L S

We create the mail body with the Velocity template and create the file src/main/
resources/MailBody.vm. The content in the MailBody.vm file is:
Incident $body.incidentId has been reported on the $body.incidentDate by
$body.givenName $body.familyName.
The person can be contact by:
- email: $body.email
- phone: $body.phone
Summary: $body.summary
Details:
$body.details
This is an auto generated email. You can not reply.

Letting Camel creating the mail body and storing it as a file is as easy as the following 3 code
lines:
private void generateEmailBodyAndStoreAsFile(InputReportIncident parameters) {
// generate the mail body using velocity template
// notice that we just pass in our POJO (= InputReportIncident) that we
// got from Apache CXF to Velocity.
Object response = template.sendBody("velocity:MailBody.vm", parameters);
// Note: the response is a String and can be cast to String if needed
// store the mail in a file
String filename = "mail-incident-" + parameters.getIncidentId() + ".txt";
template.sendBodyAndHeader("file://target/subfolder", response,
FileComponent.HEADER_FILE_NAME, filename);
}

What is impressive is that we can just pass in our POJO object we got from Apache CXF to
Velocity and it will be able to generate the mail body with this object in its context. Thus we
don't need to prepare anything before we let Velocity loose and generate our mail body.
Notice that the template method returns a object with out response. This object contains
the mail body as a String object. We can cast to String if needed.
If we run our unit test with mvn test we can in fact see that Camel has produced the file
and we can type its content:
D:\demo\part-two>type target\subfolder\mail-incident-123.txt
Incident 123 has been reported on the 2008-07-16 by Claus Ibsen.
The person can be contact by:
- email: davsclaus@apache.org
- phone: +45 2962 7576
Summary: bla bla

T UT O R IALS

130

Details:
more bla bla
This is an auto generated email. You can not reply.

FIRST PART OF THE SOLUTION
What we have seen here is actually what it takes to build the first part of the integration flow.
Receiving a request from a webservice, transform it to a mail body and store it to a file, and
return an OK response to the webservice. All possible within 10 lines of code. So lets wrap it
up here is what it takes:
/**
* The webservice we have implemented.
*/
public class ReportIncidentEndpointImpl implements ReportIncidentEndpoint {
private CamelContext camel;
private ProducerTemplate template;
public ReportIncidentEndpointImpl() throws Exception {
// create the camel context that is the "heart" of Camel
camel = new DefaultCamelContext();
// get the ProducerTemplate thst is a Spring'ish xxxTemplate based producer
for very
// easy sending exchanges to Camel.
template = camel.createProducerTemplate();
// start Camel
camel.start();
}
public OutputReportIncident reportIncident(InputReportIncident parameters) {
// transform the request into a mail body
Object mailBody = template.sendBody("velocity:MailBody.vm", parameters);
// store the mail body in a file
String filename = "mail-incident-" + parameters.getIncidentId() + ".txt";
template.sendBodyAndHeader("file://target/subfolder", mailBody,
FileComponent.HEADER_FILE_NAME, filename);
// return an OK reply
OutputReportIncident out = new OutputReportIncident();
out.setCode("OK");
return out;
}

131

TU TO RI A L S

}

Okay I missed by one, its in fact only 9 lines of java code and 2 fields.

END OF PART 2
I know this is a bit different introduction to Camel to how you can start using it in your
projects just as a plain java .jar framework that isn't invasive at all. I took you through the
coding parts that requires 6 - 10 lines to send a message to an endpoint, buts it's important to
show the Message Endpoint EIP pattern in action and how its implemented in Camel. Yes of
course Camel also has to one liners that you can use, and will use in your projects for sending
messages to endpoints. This part has been about good old plain java, nothing fancy with Spring,
XML files, auto discovery, OGSi or other new technologies. I wanted to demonstrate the basic
building blocks in Camel and how its setup in pure god old fashioned Java. There are plenty of
eye catcher examples with one liners that does more than you can imagine - we will come
there in the later parts.
Okay part 3 is about building the last pieces of the solution and now it gets interesting since
we have to wrestle with the event driven consumer.
Brew a cup of coffee, tug the kids and kiss the wife, for now we will have us some fun with the
Camel. See you in part 3.

RESOURCES
• Name
part-two.zip

Size

Creator

Creation Date

17 kB

Claus Ibsen

Jul 19, 2008 00:52

Comment

LINKS
▪
▪
▪
▪
▪
▪

Introduction
Part 1
Part 2
Part 3
Part 4
Part 5

T UT O R IALS

132

PART 3
RECAP
Lets just recap on the solution we have now:
public class ReportIncidentEndpointImpl implements ReportIncidentEndpoint {
private CamelContext camel;
private ProducerTemplate template;
public ReportIncidentEndpointImpl() throws Exception {
// create the camel context that is the "heart" of Camel
camel = new DefaultCamelContext();
// get the ProducerTemplate thst is a Spring'ish xxxTemplate based producer
for very
// easy sending exchanges to Camel.
template = camel.createProducerTemplate();
// start Camel
camel.start();
}
/**
* This is the last solution displayed that is the most simple
*/
public OutputReportIncident reportIncident(InputReportIncident parameters) {
// transform the request into a mail body
Object mailBody = template.sendBody("velocity:MailBody.vm", parameters);
// store the mail body in a file
String filename = "mail-incident-" + parameters.getIncidentId() + ".txt";
template.sendBodyAndHeader("file://target/subfolder", mailBody,
FileComponent.HEADER_FILE_NAME, filename);
// return an OK reply
OutputReportIncident out = new OutputReportIncident();
out.setCode("OK");
return out;
}
}

This completes the first part of the solution: receiving the message using webservice, transform
it to a mail body and store it as a text file.
What is missing is the last part that polls the text files and send them as emails. Here is where
some fun starts, as this requires usage of the Event Driven Consumer EIP pattern to react when
new files arrives. So lets see how we can do this in Camel. There is a saying: Many roads lead to
Rome, and that is also true for Camel - there are many ways to do it in Camel.

133

TU TO RI A L S

ADDING THE EVENT DRIVEN CONSUMER
We want to add the consumer to our integration that listen for new files, we do this by
creating a private method where the consumer code lives. We must register our consumer in
Camel before its started so we need to add, and there fore we call the method
addMailSenderConsumer in the constructor below:
public ReportIncidentEndpointImpl() throws Exception {
// create the camel context that is the "heart" of Camel
camel = new DefaultCamelContext();
// get the ProducerTemplate thst is a Spring'ish xxxTemplate based producer
for very
// easy sending exchanges to Camel.
template = camel.createProducerTemplate();
// add the event driven consumer that will listen for mail files and process
them
addMailSendConsumer();
// start Camel
camel.start();
}

The consumer needs to be consuming from an endpoint so we grab the endpoint from Camel
we want to consume. It's file://target/subfolder. Don't be fooled this endpoint
doesn't have to 100% identical to the producer, i.e. the endpoint we used in the previous part
to create and store the files. We could change the URL to include some options, and to make it
more clear that it's possible we setup a delay value to 10 seconds, and the first poll starts after
2 seconds. This is done by adding
?consumer.delay=10000&consumer.initialDelay=2000 to the URL.
When we have the endpoint we can create the consumer (just as in part 1 where we created a
producer}. Creating the consumer requires a Processor where we implement the java code
what should happen when a message arrives. To get the mail body as a String object we can use
the getBody method where we can provide the type we want in return.
Sending the email is still left to be implemented, we will do this later. And finally we must
remember to start the consumer otherwise its not active and won't listen for new files.
private void addMailSendConsumer() throws Exception {
// Grab the endpoint where we should consume. Option - the first poll starts
after 2 seconds
Endpoint endpint = camel.getEndpoint("file://target/
subfolder?consumer.initialDelay=2000");
// create the event
// the Processor is
// (think it as the
Consumer consumer =

driven consumer
the code what should happen when there is an event
onMessage method)
endpint.createConsumer(new Processor() {

T UT O R IALS

134

URL Configuration
The URL configuration in Camel endpoints is just like regular URL we know from
the Internet. You use ? and & to set the options.

Camel Type Converter
Why don't we just cast it as we always do in Java? Well the biggest advantage when
you provide the type as a parameter you tell Camel what type you want and Camel
can automatically convert it for you, using its flexible Type Converter mechanism.
This is a great advantage, and you should try to use this instead of regular type
casting.

public void process(Exchange exchange) throws Exception {
// get the mail body as a String
String mailBody = exchange.getIn().getBody(String.class);
// okay now we are read to send it as an email
System.out.println("Sending email..." + mailBody);
}
});
// star the consumer, it will listen for files
consumer.start();
}

Before we test it we need to be aware that our unit test is only catering for the first part of the
solution, receiving the message with webservice, transforming it using Velocity and then storing
it as a file - it doesn't test the Event Driven Consumer we just added. As we are eager to see it
in action, we just do a common trick adding some sleep in our unit test, that gives our Event
Driven Consumer time to react and print to System.out. We will later refine the test:
public void testRendportIncident() throws Exception {
...
OutputReportIncident out = client.reportIncident(input);
assertEquals("Response code is wrong", "OK", out.getCode());
// give the event driven consumer time to react
Thread.sleep(10 * 1000);
}

We run the test with mvn clean test and have eyes fixed on the console output.
During all the output in the console, we see that our consumer has been triggered, as we want.

135

TU TO RI A L S

2008-07-19 12:09:24,140 [mponent@1f12c4e] DEBUG FileProcessStrategySupport - Locking
the file: target\subfolder\mail-incident-123.txt ...
Sending email...Incident 123 has been reported on the 2008-07-16 by Claus Ibsen.
The person can be contact by:
- email: davsclaus@apache.org
- phone: +45 2962 7576
Summary: bla bla
Details:
more bla bla
This is an auto generated email. You can not reply.
2008-07-19 12:09:24,156 [mponent@1f12c4e] DEBUG FileConsumer - Done processing file:
target\subfolder\mail-incident-123.txt. Status is: OK

SENDING THE EMAIL
Sending the email requires access to a SMTP mail server, but the implementation code is very
simple:
private void sendEmail(String body) {
// send the email to your mail server
String url =
"smtp://someone@localhost?password=secret&to=incident@mycompany.com";
template.sendBodyAndHeader(url, body, "subject", "New incident reported");
}

And just invoke the method from our consumer:
// okay now we are read to send it as an email
System.out.println("Sending email...");
sendEmail(mailBody);
System.out.println("Email sent");

UNIT TESTING MAIL
For unit testing the consumer part we will use a mock mail framework, so we add this to our
pom.xml:


org.jvnet.mock-javamail
mock-javamail

T UT O R IALS

136

1.7
test


Then we prepare our integration to run with or without the consumer enabled. We do this to
separate the route into the two parts:
▪ receive the webservice, transform and save mail file and return OK as repose
▪ the consumer that listen for mail files and send them as emails
So we change the constructor code a bit:
public ReportIncidentEndpointImpl() throws Exception {
init(true);
}
public ReportIncidentEndpointImpl(boolean enableConsumer) throws Exception {
init(enableConsumer);
}
private void init(boolean enableConsumer) throws Exception {
// create the camel context that is the "heart" of Camel
camel = new DefaultCamelContext();
// get the ProducerTemplate thst is a Spring'ish xxxTemplate based producer
for very
// easy sending exchanges to Camel.
template = camel.createProducerTemplate();
// add the event driven consumer that will listen for mail files and process
them
if (enableConsumer) {
addMailSendConsumer();
}
// start Camel
camel.start();
}

Then remember to change the ReportIncidentEndpointTest to pass in false in the
ReportIncidentEndpointImpl constructor.
And as always run mvn clean test to be sure that the latest code changes works.

ADDING NEW UNIT TEST
We are now ready to add a new unit test that tests the consumer part so we create a new test
class that has the following code structure:

137

TU TO RI A L S

/**
* Plain JUnit test of our consumer.
*/
public class ReportIncidentConsumerTest extends TestCase {
private ReportIncidentEndpointImpl endpoint;
public void testConsumer() throws Exception {
// we run this unit test with the consumer, hence the true parameter
endpoint = new ReportIncidentEndpointImpl(true);
}
}

As we want to test the consumer that it can listen for files, read the file content and send it as
an email to our mailbox we will test it by asserting that we receive 1 mail in our mailbox and
that the mail is the one we expect. To do so we need to grab the mailbox with the mockmail
API. This is done as simple as:
public void testConsumer() throws Exception {
// we run this unit test with the consumer, hence the true parameter
endpoint = new ReportIncidentEndpointImpl(true);
// get the mailbox
Mailbox box = Mailbox.get("incident@mycompany.com");
assertEquals("Should not have mails", 0, box.size());

How do we trigger the consumer? Well by creating a file in the folder it listen for. So we could
use plain java.io.File API to create the file, but wait isn't there an smarter solution? ... yes Camel
of course. Camel can do amazing stuff in one liner codes with its ProducerTemplate, so we
need to get a hold of this baby. We expose this template in our ReportIncidentEndpointImpl
but adding this getter:
protected ProducerTemplate getTemplate() {
return template;
}

Then we can use the template to create the file in one code line:
// drop a file in the folder that the consumer listen
// here is a trick to reuse Camel! so we get the producer template and just
// fire a message that will create the file for us
endpoint.getTemplate().sendBodyAndHeader("file://target/
subfolder?append=false", "Hello World",
FileComponent.HEADER_FILE_NAME, "mail-incident-test.txt");

Then we just need to wait a little for the consumer to kick in and do its work and then we
should assert that we got the new mail. Easy as just:

T UT O R IALS

138

// let the consumer have time to run
Thread.sleep(3 * 1000);
// get the mock mailbox and check if we got mail ;)
assertEquals("Should have got 1 mail", 1, box.size());
assertEquals("Subject wrong", "New incident reported",
box.get(0).getSubject());
assertEquals("Mail body wrong", "Hello World", box.get(0).getContent());
}

The final class for the unit test is:
/**
* Plain JUnit test of our consumer.
*/
public class ReportIncidentConsumerTest extends TestCase {
private ReportIncidentEndpointImpl endpoint;
public void testConsumer() throws Exception {
// we run this unit test with the consumer, hence the true parameter
endpoint = new ReportIncidentEndpointImpl(true);
// get the mailbox
Mailbox box = Mailbox.get("incident@mycompany.com");
assertEquals("Should not have mails", 0, box.size());
// drop a file in the folder that the consumer listen
// here is a trick to reuse Camel! so we get the producer template and just
// fire a message that will create the file for us
endpoint.getTemplate().sendBodyAndHeader("file://target/
subfolder?append=false", "Hello World",
FileComponent.HEADER_FILE_NAME, "mail-incident-test.txt");
// let the consumer have time to run
Thread.sleep(3 * 1000);
// get the mock mailbox and check if we got mail ;)
assertEquals("Should have got 1 mail", 1, box.size());
assertEquals("Subject wrong", "New incident reported",
box.get(0).getSubject());
assertEquals("Mail body wrong", "Hello World", box.get(0).getContent());
}
}

END OF PART 3
Okay we have reached the end of part 3. For now we have only scratched the surface of what
Camel is and what it can do. We have introduced Camel into our integration piece by piece and

139

TU TO RI A L S

slowly added more and more along the way. And the most important is: you as the
developer never lost control. We hit a sweet spot in the webservice implementation
where we could write our java code. Adding Camel to the mix is just to use it as a regular java
code, nothing magic. We were in control of the flow, we decided when it was time to translate
the input to a mail body, we decided when the content should be written to a file. This is very
important to not lose control, that the bigger and heavier frameworks tend to do. No names
mentioned, but boy do developers from time to time dislike these elephants. And Camel is no
elephant.
I suggest you download the samples from part 1 to 3 and try them out. It is great basic
knowledge to have in mind when we look at some of the features where Camel really excel the routing domain language.
From part 1 to 3 we touched concepts such as::
▪ Endpoint
▪ URI configuration
▪ Consumer
▪ Producer
▪ Event Driven Consumer
▪ Component
▪ CamelContext
▪ ProducerTemplate
▪ Processor
▪ Type Converter

RESOURCES
• Name
part-three.zip

Size

Creator

Creation Date

18 kB

Claus Ibsen

Jul 20, 2008 03:34

Comment

LINKS
▪
▪
▪
▪
▪
▪

Introduction
Part 1
Part 2
Part 3
Part 4
Part 5

T UT O R IALS

140

PART 4
INTRODUCTION
This section is about regular Camel. The examples presented here in this section is much more
in common of all the examples we have in the Camel documentation.

ROUTING
Camel is particular strong as a light-weight and agile routing and mediation framework. In
this part we will introduce the routing concept and how we can introduce this into our
solution.
Looking back at the figure from the Introduction page we want to implement this routing.
Camel has support for expressing this routing logic using Java as a DSL (Domain Specific
Language). In fact Camel also has DSL for XML and Scala. In this part we use the Java DSL as its
the most powerful and all developers know Java. Later we will introduce the XML version that
is very well integrated with Spring.
Before we jump into it, we want to state that this tutorial is about Developers not
loosing control. In my humble experience one of the key fears of developers is that they are
forced into a tool/framework where they loose control and/or power, and the possible is now
impossible. So in this part we stay clear with this vision and our starting point is as follows:
▪ We have generated the webservice source code using the CXF wsdl2java generator
and we have our ReportIncidentEndpointImpl.java file where we as a Developer feels
home and have the power.
So the starting point is:
/**
* The webservice we have implemented.
*/
public class ReportIncidentEndpointImpl implements ReportIncidentEndpoint {
/**
* This is the last solution displayed that is the most simple
*/
public OutputReportIncident reportIncident(InputReportIncident parameters) {
// WE ARE HERE !!!
return null;
}
}

Yes we have a simple plain Java class where we have the implementation of the webservice. The
cursor is blinking at the WE ARE HERE block and this is where we feel home. More or less any
Java Developers have implemented webservices using a stack such as: Apache AXIS, Apache
CXF or some other quite popular framework. They all allow the developer to be in control and

141

TU TO RI A L S

If you have been reading the previous 3 parts then, this quote applies:
you must unlearn what you have learned
Master Yoda, Star Wars IV
So we start all over again!

implement the code logic as plain Java code. Camel of course doesn't enforce this to be any
different. Okay the boss told us to implement the solution from the figure in the Introduction
page and we are now ready to code.
RouteBuilder
RouteBuilder is the hearth in Camel of the Java DSL routing. This class does all the heavy
lifting of supporting EIP verbs for end-users to express the routing. It does take a little while to
get settled and used to, but when you have worked with it for a while you will enjoy its power
and realize it is in fact a little language inside Java itself. Camel is the only integration
framework we are aware of that has Java DSL, all the others are usually only XML based.
As an end-user you usually use the RouteBuilder as of follows:
▪ create your own Route class that extends RouteBuilder
▪ implement your routing DSL in the configure method
So we create a new class ReportIncidentRoutes and implement the first part of the routing:
import org.apache.camel.builder.RouteBuilder;
public class ReportIncidentRoutes extends RouteBuilder {
public void configure() throws Exception {
// direct:start is a internal queue to kick-start the routing in our example
// we use this as the starting point where you can send messages to
direct:start
from("direct:start")
// to is the destination we send the message to our velocity endpoint
// where we transform the mail body
.to("velocity:MailBody.vm");
}
}

What to notice here is the configure method. Here is where all the action is. Here we have
the Java DSL langauge, that is expressed using the fluent builder syntax that is also known
from Hibernate when you build the dynamic queries etc. What you do is that you can stack
methods separating with the dot.

T UT O R IALS

142

In the example above we have a very common routing, that can be distilled from pseudo
verbs to actual code with:
▪ from A to B
▪ From Endpoint A To Endpoint B
▪ from("endpointA").to("endpointB")
▪ from("direct:start").to("velocity:MailBody.vm");
from("direct:start") is the consumer that is kick-starting our routing flow. It will wait for
messages to arrive on the direct queue and then dispatch the message.
to("velocity:MailBody.vm") is the producer that will receive a message and let Velocity
generate the mail body response.
So what we have implemented so far with our ReportIncidentRoutes RouteBuilder is this
part of the picture:

Adding the RouteBuilder
Now we have our RouteBuilder we need to add/connect it to our CamelContext that is the
hearth of Camel. So turning back to our webservice implementation class
ReportIncidentEndpointImpl we add this constructor to the code, to create the CamelContext
and add the routes from our route builder and finally to start it.
private CamelContext context;
public ReportIncidentEndpointImpl() throws Exception {
// create the context
context = new DefaultCamelContext();
// append the routes to the context
context.addRoutes(new ReportIncidentRoutes());
// at the end start the camel context
context.start();
}

Okay how do you use the routes then? Well its just as before we use a ProducerTemplate to
send messages to Endpoints, so we just send to the direct:start endpoint and it will take it
from there.
So we implement the logic in our webservice operation:

143

TU TO RI A L S

/**
* This is the last solution displayed that is the most simple
*/
public OutputReportIncident reportIncident(InputReportIncident parameters) {
Object mailBody = context.createProducerTemplate().sendBody("direct:start",
parameters);
System.out.println("Body:" + mailBody);
// return an OK reply
OutputReportIncident out = new OutputReportIncident();
out.setCode("OK");
return out;
}

Notice that we get the producer template using the createProducerTemplate method on
the CamelContext. Then we send the input parameters to the direct:start endpoint and it
will route it to the velocity endpoint that will generate the mail body. Since we use direct as
the consumer endpoint (=from) and its a synchronous exchange we will get the response
back from the route. And the response is of course the output from the velocity endpoint.
We have now completed this part of the picture:

UNIT TESTING
Now is the time we would like to unit test what we got now. So we call for camel and its great
test kit. For this to work we need to add it to the pom.xml

org.apache.camel
camel-core
1.4.0
test
test-jar


After adding it to the pom.xml you should refresh your Java Editor so it pickups the new jar.
Then we are ready to create out unit test class.
We create this unit test skeleton, where we extend this class ContextTestSupport

T UT O R IALS

144

About creating ProducerTemplate
In the example above we create a new ProducerTemplate when the
reportIncident method is invoked. However in reality you should only create
the template once and re-use it. See this FAQ entry.

package org.apache.camel.example.reportincident;
import org.apache.camel.ContextTestSupport;
import org.apache.camel.builder.RouteBuilder;
/**
* Unit test of our routes
*/
public class ReportIncidentRoutesTest extends ContextTestSupport {
}

ContextTestSupport is a supporting unit test class for much easier unit testing with
Apache Camel. The class is extending JUnit TestCase itself so you get all its glory. What we
need to do now is to somehow tell this unit test class that it should use our route builder as
this is the one we gonna test. So we do this by implementing the createRouteBuilder
method.
@Override
protected RouteBuilder createRouteBuilder() throws Exception {
return new ReportIncidentRoutes();
}

That is easy just return an instance of our route builder and this unit test will use our routes.
We then code our unit test method that sends a message to the route and assert that its
transformed to the mail body using the Velocity template.
public void testTransformMailBody() throws Exception {
// create a dummy input with some input data
InputReportIncident parameters = createInput();
// send the message (using the sendBody method that takes a parameters as the
input body)
// to "direct:start" that kick-starts the route
// the response is returned as the out object, and its also the body of the
response
Object out = context.createProducerTemplate().sendBody("direct:start",
parameters);
// convert the response to a string using camel converters. However we could

145

TU TO RI A L S

It is quite common in Camel itself to unit test using routes defined as an anonymous
inner class, such as illustrated below:
protected RouteBuilder createRouteBuilder() throws Exception {
return new RouteBuilder() {
public void configure() throws Exception {
// TODO: Add your routes here, such as:
from("jms:queue:inbox").to("file://target/out");
}
};
}

The same technique is of course also possible for end-users of Camel to create parts of your
routes and test them separately in many test classes.
However in this tutorial we test the real route that is to be used for production, so we just
return an instance of the real one.

also have casted it to
// a string directly but using the type converters ensure that Camel can
convert it if it wasn't a string
// in the first place. The type converters in Camel is really powerful and you
will later learn to
// appreciate them and wonder why its not build in Java out-of-the-box
String body = context.getTypeConverter().convertTo(String.class, out);
// do some simple assertions of the mail body
assertTrue(body.startsWith("Incident 123 has been reported on the 2008-07-16
by Claus Ibsen."));
}
/**
* Creates a dummy request to be used for input
*/
protected InputReportIncident createInput() {
InputReportIncident input = new InputReportIncident();
input.setIncidentId("123");
input.setIncidentDate("2008-07-16");
input.setGivenName("Claus");
input.setFamilyName("Ibsen");
input.setSummary("bla bla");
input.setDetails("more bla bla");
input.setEmail("davsclaus@apache.org");
input.setPhone("+45 2962 7576");
return input;
}

T UT O R IALS

146

ADDING THE FILE BACKUP
The next piece of puzzle that is missing is to store the mail body as a backup file. So we turn
back to our route and the EIP patterns. We use the Pipes and Filters pattern here to chain the
routing as:
public void configure() throws Exception {
from("direct:start")
.to("velocity:MailBody.vm")
// using pipes-and-filters we send the output from the previous to the next
.to("file://target/subfolder");
}

Notice that we just add a 2nd .to on the newline. Camel will default use the Pipes and Filters
pattern here when there are multi endpoints chained liked this. We could have used the
pipeline verb to let out stand out that its the Pipes and Filters pattern such as:
from("direct:start")
// using pipes-and-filters we send the output from the previous to the next
.pipeline("velocity:MailBody.vm", "file://target/subfolder");

But most people are using the multi .to style instead.
We re-run out unit test and verifies that it still passes:
Running org.apache.camel.example.reportincident.ReportIncidentRoutesTest
Tests run: 1, Failures: 0, Errors: 0, Skipped: 0, Time elapsed: 1.157 sec

But hey we have added the file producer endpoint and thus a file should also be created as the
backup file. If we look in the target/subfolder we can see that something happened.
On my humble laptop it created this folder: target\subfolder\ID-claus-acer. So the file
producer create a sub folder named ID-claus-acer what is this? Well Camel auto
generates an unique filename based on the unique message id if not given instructions to use a
fixed filename. In fact it creates another sub folder and name the file as: target\subfolder\IDclaus-acer\3750-1219148558921\1-0 where 1-0 is the file with the mail body. What we want is
to use our own filename instead of this auto generated filename. This is archived by adding a
header to the message with the filename to use. So we need to add this to our route and
compute the filename based on the message content.
Setting the filename
For starters we show the simple solution and build from there. We start by setting a constant
filename, just to verify that we are on the right path, to instruct the file producer what filename
to use. The file producer uses a special header FileComponent.HEADER_FILE_NAME to
set the filename.

147

TU TO RI A L S

What we do is to send the header when we "kick-start" the routing as the header will be
propagated from the direct queue to the file producer. What we need to do is to use the
ProducerTemplate.sendBodyAndHeader method that takes both a body and a
header. So we change out webservice code to include the filename also:
public OutputReportIncident reportIncident(InputReportIncident parameters) {
// create the producer template to use for sending messages
ProducerTemplate producer = context.createProducerTemplate();
// send the body and the filename defined with the special header key
Object mailBody = producer.sendBodyAndHeader("direct:start", parameters,
FileComponent.HEADER_FILE_NAME, "incident.txt");
System.out.println("Body:" + mailBody);
// return an OK reply
OutputReportIncident out = new OutputReportIncident();
out.setCode("OK");
return out;
}

However we could also have used the route builder itself to configure the constant filename as
shown below:
public void configure() throws Exception {
from("direct:start")
.to("velocity:MailBody.vm")
// set the filename to a constant before the file producer receives the
message
.setHeader(FileComponent.HEADER_FILE_NAME, constant("incident.txt"))
.to("file://target/subfolder");
}

But Camel can be smarter and we want to dynamic set the filename based on some of the input
parameters, how can we do this?
Well the obvious solution is to compute and set the filename from the webservice
implementation, but then the webservice implementation has such logic and we want this
decoupled, so we could create our own POJO bean that has a method to compute the
filename. We could then instruct the routing to invoke this method to get the computed
filename. This is a string feature in Camel, its Bean binding. So lets show how this can be done:

Using Bean Language to compute the filename
First we create our plain java class that computes the filename, and it has 100% no
dependencies to Camel what so ever.
/**
* Plain java class to be used for filename generation based on the reported incident

T UT O R IALS

148

*/
public class FilenameGenerator {
public String generateFilename(InputReportIncident input) {
// compute the filename
return "incident-" + input.getIncidentId() + ".txt";
}
}

The class is very simple and we could easily create unit tests for it to verify that it works as
expected. So what we want now is to let Camel invoke this class and its generateFilename with
the input parameters and use the output as the filename. Pheeeww is this really possible out-ofthe-box in Camel? Yes it is. So lets get on with the show. We have the code that computes the
filename, we just need to call it from our route using the Bean Language:
public void configure() throws Exception {
from("direct:start")
// set the filename using the bean language and call the FilenameGenerator
class.
// the 2nd null parameter is optional methodname, to be used to avoid
ambiguity.
// if not provided Camel will try to figure out the best method to invoke,
as we
// only have one method this is very simple
.setHeader(FileComponent.HEADER_FILE_NAME,
BeanLanguage.bean(FilenameGenerator.class, null))
.to("velocity:MailBody.vm")
.to("file://target/subfolder");
}

Notice that we use the bean language where we supply the class with our bean to invoke.
Camel will instantiate an instance of the class and invoke the suited method. For completeness
and ease of code readability we add the method name as the 2nd parameter
.setHeader(FileComponent.HEADER_FILE_NAME,
BeanLanguage.bean(FilenameGenerator.class, "generateFilename"))

Then other developers can understand what the parameter is, instead of null.
Now we have a nice solution, but as a sidetrack I want to demonstrate the Camel has other
languages out-of-the-box, and that scripting language is a first class citizen in Camel where it etc.
can be used in content based routing. However we want it to be used for the filename
generation.

149

TU TO RI A L S

Using a script language to set the filename
We could do as in the previous parts where we send the computed filename as a message
header when we "kick-start" the route. But we want to learn new stuff so we look for a
different solution using some of Camels many Languages. As OGNL is a favorite language of
mine (used by WebWork) so we pick this baby for a Camel ride. For starters we must add it
to our pom.xml:

org.apache.camel
camel-ognl
${camel-version}


And remember to refresh your editor so you got the new .jars.
We want to construct the filename based on this syntax: mail-incident-#ID#.txt
where #ID# is the incident id from the input parameters. As OGNL is a language that can
invoke methods on bean we can invoke the getIncidentId() on the message body and
then concat it with the fixed pre and postfix strings.
In OGNL glory this is done as:
"'mail-incident-' + request.body.incidentId + '.txt'"

where request.body.incidentId computes to:

▪ request is the IN message. See the OGNL for other predefined objects
available
▪ body is the body of the in message
▪ incidentId will invoke the getIncidentId() method on the body.
The rest is just more or less regular plain code where we can concat
strings.
Now we got the expression to dynamic compute the filename on the fly we need to set it on
our route so we turn back to our route, where we can add the OGNL expression:
public void configure() throws Exception {
from("direct:start")
// we need to set the filename and uses OGNL for this
.setHeader(FileComponent.HEADER_FILE_NAME,
OgnlExpression.ognl("'mail-incident-' + request.body.incidentId + '.txt'"))

T UT O R IALS

150

// using pipes-and-filters we send the output from the previous
to the next
.pipeline("velocity:MailBody.vm", "file://target/subfolder");
}

And since we are on Java 1.5 we can use the static import of ognl so we have:
import static org.apache.camel.language.ognl.OgnlExpression.ognl;
...
.setHeader(FileComponent.HEADER_FILE_NAME, ognl("'mail-incident-' +
request.body.incidentId + '.txt'"))

Notice the import static also applies for all the other languages, such as the Bean Language
we used previously.

Whatever worked for you we have now implemented the backup of the data files:

SENDING THE EMAIL
What we need to do before the solution is completed is to actually send the email with the mail
body we generated and stored as a file. In the previous part we did this with a File consumer,
that we manually added to the CamelContext. We can do this quite easily with the routing.
import org.apache.camel.builder.RouteBuilder;
public class ReportIncidentRoutes extends RouteBuilder {
public void configure() throws Exception {
// first part from the webservice -> file backup
from("direct:start")
.setHeader(FileComponent.HEADER_FILE_NAME, bean(FilenameGenerator.class,
"generateFilename"))
.to("velocity:MailBody.vm")
.to("file://target/subfolder");

151

TU TO RI A L S

// second part from the file backup -> send email
from("file://target/subfolder")
// set the subject of the email
.setHeader("subject", constant("new incident reported"))
// send the email
.to("smtp://someone@localhost?password=secret&to=incident@mycompany.com");
}
}

The last 3 lines of code does all this. It adds a file consumer from("file://target/
subfolder"), sets the mail subject, and finally send it as an email.
The DSL is really powerful where you can express your routing integration logic.
So we completed the last piece in the picture puzzle with just 3 lines of code.
We have now completed the integration:

CONCLUSION
We have just briefly touched the routing in Camel and shown how to implement them using
the fluent builder syntax in Java. There is much more to the routing in Camel than shown
here, but we are learning step by step. We continue in part 5. See you there.

RESOURCES
• Name
part-four.zip

Size

Creator

Creation Date

11 kB

Claus Ibsen

Aug 25, 2008 07:24

Comment

LINKS
▪
▪
▪
▪

Introduction
Part 1
Part 2
Part 3

T UT O R IALS

152

▪ Part 4
▪ Part 5

BETTER JMS TRANSPORT FOR CXF WEBSERVICE USING
APACHE CAMEL
Configuring JMS in Apache CXF before Version 2.1.3 is possible but not really easy or nice. This
article shows how to use Apache Camel to provide a better JMS Transport for CXF.
Update: Since CXF 2.1.3 there is a new way of configuring JMS (Using the
JMSConfigFeature). It makes JMS config for CXF as easy as with Camel. Using Camel for JMS is
still a good idea if you want to use the rich feature of Camel for routing and other Integration
Scenarios that CXF does not support.
You can find the original announcement for this Tutorial and some additional info on
Christian Schneider´s Blog
So how to connect Apache Camel and CXF
The best way to connect Camel and CXF is using the Camel transport for CXF. This is a camel
module that registers with cxf as a new transport. It is quite easy to configure.





http://cxf.apache.org/transports/camel




This bean registers with CXF and provides a new transport prefix camel:// that can be used in
CXF address configurations. The bean references a bean cxf which will be already present in
your config. The other refrenceis a camel context. We will later define this bean to provide the
routing config.
How is JMS configured in Camel
In camel you need two things to configure JMS. A ConnectionFactory and a JMSComponent. As
ConnectionFactory you can simply set up the normal Factory your JMS provider offers or bind
a JNDI ConnectionFactory. In this example we use the ConnectionFactory provided by
ActiveMQ.

153

TU TO RI A L S





Then we set up the JMSComponent. It offers a new transport prefix to camel that we simply
call jms. If we need several JMSComponents we can differentiate them by their name.





You can find more details about the JMSComponent at the Camel Wiki. For example you find
the complete configuration options and a JNDI sample there.
Setting up the CXF client
We will configure a simple CXF webservice client. It will use stub code generated from a wsdl.
The webservice client will be configured to use JMS directly. You can also use a direct: Endpoint
and do the routing to JMS in the Camel Context.



We explicitly configure serviceName and endpointName so they are not read from the wsdl.
The names we use are arbitrary and have no further function but we set them to look nice. The
serviceclass points to the service interface that was generated from the wsdl. Now the
important thing is address. Here we tell cxf to use the camel transport, use the JmsComponent
who registered the prefix "jms" and use the queue "CustomerService".
Setting up the CamelContext
As we do not need additional routing an empty CamelContext bean will suffice.



T UT O R IALS

154

Running the Example
• Download the example project here
• Follow the readme.txt
Conclusion
As you have seen in this example you can use Camel to connect services to JMS easily while
being able to also use the rich integration features of Apache Camel.

TUTORIAL USING AXIS 1.4 WITH APACHE CAMEL
•
•
•
•
•
•
•
•
•
•
•
•
•
•
•
•
•
•
•
•
•

Tutorial using Axis 1.4 with Apache Camel
Prerequisites
Distribution
Introduction
Setting up the project to run Axis
Maven 2
wsdl
Configuring Axis
Running the Example
Integrating Spring
Using Spring
Integrating Camel
CamelContext
Store a file backup
Running the example
Unit Testing
Smarter Unit Testing with Spring
Unit Test calling WebService
Annotations
The End
See Also

Prerequisites
This tutorial uses Maven 2 to setup the Camel project and for dependencies for artifacts.
Distribution
This sample is distributed with the Camel 1.5 distribution as examples/camel-exampleaxis.

155

TU TO RI A L S

Introduction
Apache Axis is/was widely used as a webservice framework. So in line with some of the other
tutorials to demonstrate how Camel is not an invasive framework but is flexible and integrates
well with existing solution.
We have an existing solution that exposes a webservice using Axis 1.4 deployed as web
applications. This is a common solution. We use contract first so we have Axis generated
source code from an existing wsdl file. Then we show how we introduce Spring and Camel to
integrate with Axis.
This tutorial uses the following frameworks:
• Maven 2.0.9
• Apache Camel 1.5.0
• Apache Axis 1.4
• Spring 2.5.5
Setting up the project to run Axis
This first part is about getting the project up to speed with Axis. We are not touching Camel or
Spring at this time.

Maven 2
Axis dependencies is available for maven 2 so we configure our pom.xml as:

org.apache.axis
axis
1.4


org.apache.axis
axis-jaxrpc
1.4


org.apache.axis
axis-saaj
1.4


axis
axis-wsdl4j
1.5.1


T UT O R IALS

156


commons-discovery
commons-discovery
0.4


log4j
log4j
1.2.14


Then we need to configure maven to use Java 1.5 and the Axis maven plugin that generates the
source code based on the wsdl file:


org.apache.maven.plugins
maven-compiler-plugin

1.5
1.5



org.codehaus.mojo
axistools-maven-plugin

src/main/resources/
com.mycompany.myschema
false
true
false




wsdl2java





wsdl
We use the same .wsdl file as the Tutorial-Example-ReportIncident and copy it to src/main/
webapp/WEB-INF/wsdl

157

TU TO RI A L S







































T UT O R IALS

158



























Configuring Axis
Okay we are now setup for the contract first development and can generate the source file. For
now we are still only using standard Axis and not Spring nor Camel. We still need to setup Axis
as a web application so we configure the web.xml in src/main/webapp/WEB-INF/
web.xml as:

axis
org.apache.axis.transport.http.AxisServlet


axis

159

TU TO RI A L S

/services/*


The web.xml just registers Axis servlet that is handling the incoming web requests to its servlet
mapping. We still need to configure Axis itself and this is done using its special configuration file
server-config.wsdd. We nearly get this file for free if we let Axis generate the source
code so we run the maven goal:
mvn axistools:wsdl2java

The tool will generate the source code based on the wsdl and save the files to the following
folder:
.\target\generated-sources\axistools\wsdl2java\org\apache\camel\example\reportincident
deploy.wsdd
InputReportIncident.java
OutputReportIncident.java
ReportIncidentBindingImpl.java
ReportIncidentBindingStub.java
ReportIncidentService_PortType.java
ReportIncidentService_Service.java
ReportIncidentService_ServiceLocator.java
undeploy.wsdd

This is standard Axis and so far no Camel or Spring has been touched. To implement our
webservice we will add our code, so we create a new class
AxisReportIncidentService that implements the port type interface where we can
implement our code logic what happens when the webservice is invoked.
package org.apache.camel.example.axis;
import org.apache.camel.example.reportincident.InputReportIncident;
import org.apache.camel.example.reportincident.OutputReportIncident;
import org.apache.camel.example.reportincident.ReportIncidentService_PortType;
import java.rmi.RemoteException;
/**
* Axis webservice
*/
public class AxisReportIncidentService implements ReportIncidentService_PortType {
public OutputReportIncident reportIncident(InputReportIncident parameters) throws
RemoteException {
System.out.println("Hello AxisReportIncidentService is called from " +
parameters.getGivenName());
OutputReportIncident out = new OutputReportIncident();

T UT O R IALS

160

out.setCode("OK");
return out;
}
}

Now we need to configure Axis itself and this is done using its server-config.wsdd file.
We nearly get this for for free from the auto generated code, we copy the stuff from
deploy.wsdd and made a few modifications:



































The globalConfiguration and transport is not in the deploy.wsdd file so you gotta write
that yourself. The service is a 100% copy from deploy.wsdd. Axis has more configuration to it
than shown here, but then you should check the Axis documentation.
What we need to do now is important, as we need to modify the above configuration to use
our webservice class than the default one, so we change the classname parameter to our class
AxisReportIncidentService:


Running the Example
Now we are ready to run our example for the first time, so we use Jetty as the quick web
container using its maven command:
mvn jetty:run

Then we can hit the web browser and enter this URL: http://localhost:8080/
camel-example-axis/services and you should see the famous Axis start page with the
text And now... Some Services.

T UT O R IALS

162

Clicking on the .wsdl link shows the wsdl file, but what. It's an auto generated one and not
our original .wsdl file. So we need to fix this ASAP and this is done by configuring Axis in the
server-config.wsdd file:

/WEB-INF/wsdl/report_incident.wsdl
...

We do this by adding the wsdlFile tag in the service element where we can point to the real
.wsdl file.
Integrating Spring
First we need to add its dependencies to the pom.xml.

org.springframework
spring-web
2.5.5


Spring is integrated just as it would like to, we add its listener to the web.xml and a context
parameter to be able to configure precisely what spring xml files to use:

contextConfigLocation

classpath:axis-example-context.xml



org.springframework.web.context.ContextLoaderListener


Next is to add a plain spring XML file named axis-example-context.xml in the src/main/
resources folder.




163

TU TO RI A L S

The spring XML file is currently empty. We hit jetty again with mvn jetty:run just to make
sure Spring was setup correctly.

Using Spring
We would like to be able to get hold of the Spring ApplicationContext from our webservice so
we can get access to the glory spring, but how do we do this? And our webservice class
AxisReportIncidentService is created and managed by Axis we want to let Spring do this. So we
have two problems.
We solve these problems by creating a delegate class that Axis creates, and this delegate
class gets hold on Spring and then gets our real webservice as a spring bean and invoke the
service.
First we create a new class that is 100% independent from Axis and just a plain POJO. This is
our real service.
package org.apache.camel.example.axis;
import org.apache.camel.example.reportincident.InputReportIncident;
import org.apache.camel.example.reportincident.OutputReportIncident;
/**
* Our real service that is not tied to Axis
*/
public class ReportIncidentService {
public OutputReportIncident reportIncident(InputReportIncident parameters) {
System.out.println("Hello ReportIncidentService is called from " +
parameters.getGivenName());
OutputReportIncident out = new OutputReportIncident();
out.setCode("OK");
return out;
}
}

So now we need to get from AxisReportIncidentService to this one ReportIncidentService using
Spring. Well first of all we add our real service to spring XML configuration file so Spring can
handle its lifecycle:



T UT O R IALS

164




And then we need to modify AxisReportIncidentService to use Spring to lookup the spring bean
id="incidentservice" and delegate the call. We do this by extending the spring class
org.springframework.remoting.jaxrpc.ServletEndpointSupport so the
refactored code is:
package org.apache.camel.example.axis;
import
import
import
import

org.apache.camel.example.reportincident.InputReportIncident;
org.apache.camel.example.reportincident.OutputReportIncident;
org.apache.camel.example.reportincident.ReportIncidentService_PortType;
org.springframework.remoting.jaxrpc.ServletEndpointSupport;

import java.rmi.RemoteException;
/**
* Axis webservice
*/
public class AxisReportIncidentService extends ServletEndpointSupport implements
ReportIncidentService_PortType {
public OutputReportIncident reportIncident(InputReportIncident parameters) throws
RemoteException {
// get hold of the spring bean from the application context
ReportIncidentService service = (ReportIncidentService)
getApplicationContext().getBean("incidentservice");
// delegate to the real service
return service.reportIncident(parameters);
}
}

To see if everything is okay we run mvn jetty:run.
In the code above we get hold of our service at each request by looking up in the application
context. However Spring also supports an init method where we can do this once. So we
change the code to:
public class AxisReportIncidentService extends ServletEndpointSupport implements
ReportIncidentService_PortType {
private ReportIncidentService service;
@Override
protected void onInit() throws ServiceException {

165

TU TO RI A L S

// get hold of the spring bean from the application context
service = (ReportIncidentService)
getApplicationContext().getBean("incidentservice");
}
public OutputReportIncident reportIncident(InputReportIncident parameters) throws
RemoteException {
// delegate to the real service
return service.reportIncident(parameters);
}
}

So now we have integrated Axis with Spring and we are ready for Camel.
Integrating Camel
Again the first step is to add the dependencies to the maven pom.xml file:

org.apache.camel
camel-core
1.5.0


org.apache.camel
camel-spring
1.5.0


Now that we have integrated with Spring then we easily integrate with Camel as Camel works
well with Spring.
We choose to integrate Camel in the Spring XML file so we add the camel namespace and the
schema location:
xmlns:camel="http://activemq.apache.org/camel/schema/spring"
http://activemq.apache.org/camel/schema/spring http://activemq.apache.org/camel/schema/
spring/camel-spring.xsd"

CamelContext
CamelContext is the heart of Camel its where all the routes, endpoints, components, etc. is
registered. So we setup a CamelContext and the spring XML files looks like:

T UT O R IALS

166

Camel does not require Spring
Camel does not require Spring, we could easily have used Camel without Spring,
but most users prefer to use Spring also.









Store a file backup
We want to store the web service request as a file before we return a response. To do this we
want to send the file content as a message to an endpoint that produces the file. So we need to
do two steps:
▪ configure the file backup endpoint
▪ send the message to the endpoint
The endpoint is configured in spring XML so we just add it as:





In the CamelContext we have defined our endpoint with the id backup and configured it use
the URL notation that we know from the internet. Its a file scheme that accepts a context
and some options. The contest is target and its the folder to store the file. The option is just
as the internet with ? and & for subsequent options. We configure it to not append, meaning
than any existing file will be overwritten. See the File component for options and how to use
the camel file endpoint.

167

TU TO RI A L S

Next up is to be able to send a message to this endpoint. The easiest way is to use a
ProducerTemplate. A ProducerTemplate is inspired by Spring template pattern with for
instance JmsTemplate or JdbcTemplate in mind. The template that all the grunt work and
exposes a simple interface to the end-user where he/she can set the payload to send. Then the
template will do proper resource handling and all related issues in that regard. But how do we
get hold of such a template? Well the CamelContext is able to provide one. This is done by
configuring the template on the camel context in the spring XML as:







Then we can expose a ProducerTemplate property on our service with a setter in the Java
code as:
public class ReportIncidentService {
private ProducerTemplate template;
public void setTemplate(ProducerTemplate template) {
this.template = template;
}

And then let Spring handle the dependency inject as below:





Now we are ready to use the producer template in our service to send the payload to the
endpoint. The template has many sendXXX methods for this purpose. But before we send
the payload to the file endpoint we must also specify what filename to store the file as. This is
done by sending meta data with the payload. In Camel metadata is sent as headers. Headers is
just a plain Map. So if we needed to send several metadata then we
could construct an ordinary HashMap and put the values in there. But as we just need to send
one header with the filename Camel has a convenient send method sendBodyAndHeader so
we choose this one.
public OutputReportIncident reportIncident(InputReportIncident parameters) {
System.out.println("Hello ReportIncidentService is called from " +

T UT O R IALS

168

parameters.getGivenName());
String data = parameters.getDetails();
// store the data as a file
String filename = parameters.getIncidentId() + ".txt";
// send the data to the endpoint and the header contains what filename it
should be stored as
template.sendBodyAndHeader("backup", data, "org.apache.camel.file.name",
filename);
OutputReportIncident out = new OutputReportIncident();
out.setCode("OK");
return out;
}

The template in the code above uses 4 parameters:
▪ the endpoint name, in this case the id referring to the endpoint defined in Spring XML
in the camelContext element.
▪ the payload, can be any kind of object
▪ the key for the header, in this case a Camel keyword to set the filename
▪ and the value for the header
Running the example
We start our integration with maven using mvn jetty:run. Then we open a browser and
hit http://localhost:8080. Jetty is so smart that it display a frontpage with links to the
deployed application so just hit the link and you get our application. Now we hit append
/services to the URL to access the Axis frontpage. The URL should be
http://localhost:8080/camel-example-axis/services.
You can then test it using a web service test tools such as SoapUI.
Hitting the service will output to the console
2008-09-06 15:01:41.718::INFO: Started SelectChannelConnector @ 0.0.0.0:8080
[INFO] Started Jetty Server
Hello ReportIncidentService is called from Ibsen

And there should be a file in the target subfolder.
dir target /b
123.txt

169

TU TO RI A L S

Unit Testing
We would like to be able to unit test our ReportIncidentService class. So we add junit to
the maven dependency:

junit
junit
3.8.2
test


And then we create a plain junit testcase for our service class.
package org.apache.camel.example.axis;
import junit.framework.TestCase;
import org.apache.camel.example.reportincident.InputReportIncident;
import org.apache.camel.example.reportincident.OutputReportIncident;
/**
* Unit test of service
*/
public class ReportIncidentServiceTest extends TestCase {
public void testIncident() {
ReportIncidentService service = new ReportIncidentService();
InputReportIncident input = createDummyIncident();
OutputReportIncident output = service.reportIncident(input);
assertEquals("OK", output.getCode());
}
protected InputReportIncident createDummyIncident() {
InputReportIncident input = new InputReportIncident();
input.setEmail("davsclaus@apache.org");
input.setIncidentId("12345678");
input.setIncidentDate("2008-07-13");
input.setPhone("+45 2962 7576");
input.setSummary("Failed operation");
input.setDetails("The wrong foot was operated.");
input.setFamilyName("Ibsen");
input.setGivenName("Claus");
return input;
}
}

Then we can run the test with maven using: mvn test. But we will get a failure:

T UT O R IALS

170

Running org.apache.camel.example.axis.ReportIncidentServiceTest
Hello ReportIncidentService is called from Claus
Tests run: 1, Failures: 0, Errors: 1, Skipped: 0, Time elapsed: 0.235 sec <<< FAILURE!
Results :
Tests in error:
testIncident(org.apache.camel.example.axis.ReportIncidentServiceTest)
Tests run: 1, Failures: 0, Errors: 1, Skipped: 0

What is the problem? Well our service uses a CamelProducer (the template) to send a message
to the file endpoint so the message will be stored in a file. What we need is to get hold of such
a producer and inject it on our service, by calling the setter.
Since Camel is very light weight and embedable we are able to create a CamelContext and
add the endpoint in our unit test code directly. We do this to show how this is possible:
private CamelContext context;
@Override
protected void setUp() throws Exception {
super.setUp();
// CamelContext is just created like this
context = new DefaultCamelContext();
// then we can create our endpoint and set the options
FileEndpoint endpoint = new FileEndpoint();
// the endpoint must have the camel context set also
endpoint.setCamelContext(context);
// our output folder
endpoint.setFile(new File("target"));
// and the option not to append
endpoint.setAppend(false);
// then we add the endpoint just in java code just as the spring XML, we
register it with the "backup" id.
context.addSingletonEndpoint("backup", endpoint);
// finally we need to start the context so Camel is ready to rock
context.start();
}
@Override
protected void tearDown() throws Exception {
super.tearDown();
// and we are nice boys so we stop it to allow resources to clean up
context.stop();
}

So now we are ready to set the ProducerTemplate on our service, and we get a hold of that
baby from the CamelContext as:

171

TU TO RI A L S

public void testIncident() {
ReportIncidentService service = new ReportIncidentService();
// get a producer template from the camel context
ProducerTemplate template = context.createProducerTemplate();
// inject it on our service using the setter
service.setTemplate(template);
InputReportIncident input = createDummyIncident();
OutputReportIncident output = service.reportIncident(input);
assertEquals("OK", output.getCode());
}

And this time when we run the unit test its a success:
Results :
Tests run: 1, Failures: 0, Errors: 0, Skipped: 0

We would like to test that the file exists so we add these two lines to our test method:
// should generate a file also
File file = new File("target/" + input.getIncidentId() + ".txt");
assertTrue("File should exists", file.exists());

Smarter Unit Testing with Spring
The unit test above requires us to assemble the Camel pieces manually in java code. What if we
would like our unit test to use our spring configuration file axis-example-context.xml
where we already have setup the endpoint. And of course we would like to test using this
configuration file as this is the real file we will use. Well hey presto the xml file is a spring
ApplicationContext file and spring is able to load it, so we go the spring path for unit testing.
First we add the spring-test jar to our maven dependency:

org.springframework
spring-test
test


And then we refactor our unit test to be a standard spring unit class. What we need to do is to
extend AbstractJUnit38SpringContextTests instead of TestCase in our unit test.
Since Spring 2.5 embraces annotations we will use one as well to instruct what our xml
configuration file is located:

T UT O R IALS

172

@ContextConfiguration(locations = "classpath:axis-example-context.xml")
public class ReportIncidentServiceTest extends AbstractJUnit38SpringContextTests {

What we must remember to add is the classpath: prefix as our xml file is located in src/
main/resources. If we omit the prefix then Spring will by default try to locate the xml file
in the current package and that is org.apache.camel.example.axis. If the xml file is located
outside the classpath you can use file: prefix instead. So with these two modifications we can
get rid of all the setup and teardown code we had before and now we will test our real
configuration.
The last change is to get hold of the producer template and now we can just refer to the
bean id it has in the spring xml file:



So we get hold of it by just getting it from the spring ApplicationContext as all spring users is
used to do:
// get a producer template from the the spring context
ProducerTemplate template = (ProducerTemplate)
applicationContext.getBean("camelTemplate");
// inject it on our service using the setter
service.setTemplate(template);

Now our unit test is much better, and a real power of Camel is that is fits nicely with Spring
and you can use standard Spring'ish unit test to test your Camel applications as well.
Unit Test calling WebService
What if you would like to execute a unit test where you send a webservice request to the
AxisReportIncidentService how do we unit test this one? Well first of all the code is
merely just a delegate to our real service that we have just tested, but nevertheless its a good
question and we would like to know how. Well the answer is that we can exploit that fact that
Jetty is also a slim web container that can be embedded anywhere just as Camel can. So we add
this to our pom.xml:

org.mortbay.jetty
jetty
${jetty-version}
test


173

TU TO RI A L S

Then we can create a new class AxisReportIncidentServiceTest to unit test with Jetty.
The code to setup Jetty is shown below with code comments:
public class AxisReportIncidentServiceTest extends TestCase {
private Server server;
private void startJetty() throws Exception {
// create an embedded Jetty server
server = new Server();
// add a listener on port 8080 on localhost (127.0.0.1)
Connector connector = new SelectChannelConnector();
connector.setPort(8080);
connector.setHost("127.0.0.1");
server.addConnector(connector);
// add our web context path
WebAppContext wac = new WebAppContext();
wac.setContextPath("/unittest");
// set the location of the exploded webapp where WEB-INF is located
// this is a nice feature of Jetty where we can point to src/main/webapp
wac.setWar("./src/main/webapp");
server.setHandler(wac);
// then start Jetty
server.setStopAtShutdown(true);
server.start();
}
@Override
protected void setUp() throws Exception {
super.setUp();
startJetty();
}
@Override
protected void tearDown() throws Exception {
super.tearDown();
server.stop();
}
}

Now we just need to send the incident as a webservice request using Axis. So we add the
following code:
public void testReportIncidentWithAxis() throws Exception {
// the url to the axis webservice exposed by jetty
URL url = new URL("http://localhost:8080/unittest/services/
ReportIncidentPort");

T UT O R IALS

174

// Axis stuff to get the port where we can send the webservice request
ReportIncidentService_ServiceLocator locator = new
ReportIncidentService_ServiceLocator();
ReportIncidentService_PortType port = locator.getReportIncidentPort(url);
// create input to send
InputReportIncident input = createDummyIncident();
// send the webservice and get the response
OutputReportIncident output = port.reportIncident(input);
assertEquals("OK", output.getCode());
// should generate a file also
File file = new File("target/" + input.getIncidentId() + ".txt");
assertTrue("File should exists", file.exists());
}
protected InputReportIncident createDummyIncident() {
InputReportIncident input = new InputReportIncident();
input.setEmail("davsclaus@apache.org");
input.setIncidentId("12345678");
input.setIncidentDate("2008-07-13");
input.setPhone("+45 2962 7576");
input.setSummary("Failed operation");
input.setDetails("The wrong foot was operated.");
input.setFamilyName("Ibsen");
input.setGivenName("Claus");
return input;
}

And now we have an unittest that sends a webservice request using good old Axis.
Annotations
Both Camel and Spring has annotations that can be used to configure and wire trivial settings
more elegantly. Camel has the endpoint annotation @EndpointInjected that is just what
we need. With this annotation we can inject the endpoint into our service. The annotation
takes either a name or uri parameter. The name is the bean id in the Registry. The uri is the
URI configuration for the endpoint. Using this you can actually inject an endpoint that you have
not defined in the camel context. As we have defined our endpoint with the id backup we use
the name parameter.
@EndpointInject(name = "backup")
private ProducerTemplate template;

Camel is smart as @EndpointInjected supports different kinds of object types. We like
the ProducerTemplate so we just keep it as it is.
Since we use annotations on the field directly we do not need to set the property in the spring
xml file so we change our service bean:

175

TU TO RI A L S



Running the unit test with mvn test reveals that it works nicely.
And since we use the @EndpointInjected that refers to the endpoint with the id
backup directly we can loose the template tag in the xml, so its shorter:








And the final touch we can do is that since the endpoint is injected with concrete endpoint to
use we can remove the "backup" name parameter when we send the message. So we change
from:
// send the data to the endpoint and the header contains what filename it
should be stored as
template.sendBodyAndHeader("backup", data, "org.apache.camel.file.name",
filename);

To without the name:
// send the data to the endpoint and the header contains what filename it
should be stored as
template.sendBodyAndHeader(data, "org.apache.camel.file.name", filename);

Then we avoid to duplicate the name and if we rename the endpoint name then we don't forget
to change it in the code also.
The End
This tutorial hasn't really touched the one of the key concept of Camel as a powerful routing
and mediation framework. But we wanted to demonstrate its flexibility and that it integrates
well with even older frameworks such as Apache Axis 1.4.
Check out the other tutorials on Camel and the other examples.

T UT O R IALS

176

Note that the code shown here also applies to Camel 1.4 so actually you can get started
right away with the released version of Camel. As this time of writing Camel 1.5 is work in
progress.
See Also
▪ Tutorials
▪ Examples

TUTORIAL ON USING CAMEL IN A WEB APPLICATION
Camel has been designed to work great with the Spring framework; so if you are already a
Spring user you can think of Camel as just a framework for adding to your Spring XML files.
So you can follow the usual Spring approach to working with web applications; namely to
add the standard Spring hook to load a /WEB-INF/applicationContext.xml file. In that
file you can include your usual Camel XML configuration.
Step1: Edit your web.xml
To enable spring add a context loader listener to your /WEB-INF/web.xml file



org.springframework.web.context.ContextLoaderListener



This will cause Spring to boot up and look for the /WEB-INF/applicationContext.xml
file.
Step 2: Create a /WEB-INF/applicationContext.xml file
Now you just need to create your Spring XML file and add your camel routes or configuration.
For example

177

TU TO RI A L S











Then boot up your web application and you're good to go!
Hints and Tips
If you use Maven to build your application your directory tree will look like this...
src/main/webapp/WEB-INF
web.xml
applicationContext.xml

To enable more rapid development we hightly recommend the jetty:run maven plugin.
Please refer to the help for more information on using jetty:run - but briefly if you add the
following to your pom.xml



org.mortbay.jetty
maven-jetty-plugin


/

10





T UT O R IALS

178

Then you can run your web application as follows
mvn jetty:run

Then Jetty will also monitor your target/classes directory and your src/main/webapp directory
so that if you modify your spring XML, your web.xml or your java code the web application will
be restarted, re-creating your Camel routes.
If your unit tests take a while to run, you could miss them out when running your web
application via
mvn -Dtest=false jetty:run

TUTORIAL BUSINESS PARTNERS
BACKGROUND AND INTRODUCTION
Business Background
So there's a company, which we'll call Acme. Acme sells widgets, in a fairly unusual way. Their
customers are responsible for telling Acme what they purchased. The customer enters into
their own systems (ERP or whatever) which widgets they bought from Acme. Then at some
point, their systems emit a record of the sale which needs to go to Acme so Acme can bill them
for it. Obviously, everyone wants this to be as automated as possible, so there needs to be
integration between the customer's system and Acme.
Sadly, Acme's sales people are, technically speaking, doormats. They tell all their prospects,
"you can send us the data in whatever format, using whatever protocols, whatever. You just
can't change once it's up and running."
The result is pretty much what you'd expect. Taking a random sample of 3 customers:
• Customer 1: XML over FTP
• Customer 2: CSV over HTTP
• Customer 3: Excel via e-mail
Now on the Acme side, all this has to be converted to a canonical XML format and submitted
to the Acme accounting system via JMS. Then the Acme accounting system does its stuff and
sends an XML reply via JMS, with a summary of what it processed (e.g. 3 line items accepted,
line item #2 in error, total invoice $123.45). Finally, that data needs to be formatted into an email, and sent to a contact at the customer in question ("Dear Joyce, we received an invoice on
1/2/08. We accepted 3 line items totaling $123.45, though there was an error with line items
#2 [invalid quantity ordered]. Thank you for your business. Love, Acme.").
So it turns out Camel can handle all this:
• Listen for HTTP, e-mail, and FTP files
179

TU TO RI A L S

Under Construction
This tutorial is a work in progress.
•
•
•
•
•
•

Grab attachments from the e-mail messages
Convert XML, XLS, and CSV files to a canonical XML format
read and write JMS messages
route based on company ID
format e-mails using Velocity templates
send outgoing e-mail messages

Tutorial Background
This tutorial will cover all that, plus setting up tests along the way.
Before starting, you should be familiar with:
• Camel concepts including the CamelContext, Routes, Components and Endpoints,
and Enterprise Integration Patterns
• Configuring Camel with the XML or Java DSL
You'll learn:
• How to set up a Maven build for a Camel project
• How to transform XML, CSV, and Excel data into a standard XML format with Camel
◦ How to write POJOs (Plain Old Java Objects), Velocity templates, and XSLT
stylesheets that are invoked by Camel routes for message transformation
• How to configure simple and complex Routes in Camel, using either the XML or the
Java DSL format
• How to set up unit tests that load a Camel configuration and test Camel routes
• How to use Camel's Data Formats to automatically convert data between Java objects
and XML, CSV files, etc.
• How to send and receive e-mail from Camel
• How to send and receive JMS messages from Camel
• How to use Enterprise Integration Patterns including Message Router and Pipes and
Filters
◦ How to use various languages to express content-based routing rules in
Camel
• How to deal with Camel messages, headers, and attachments
You may choose to treat this as a hands-on tutorial, and work through building the code and
configuration files yourself. Each of the sections gives detailed descriptions of the steps that
need to be taken to get the components and routes working in Camel, and takes you through
tests to make sure they are working as expected.
But each section also links to working copies of the source and configuration files, so if you
don't want the hands-on approach, you can simply review and/or download the finished files.

T UT O R IALS

180

High-Level Diagram
Here's more or less what the integration process looks like.
First, the input from the customers to Acme:

And then, the output from Acme to the customers:

Tutorial Tasks
To get through this scenario, we're going to break it down into smaller pieces, implement and
test those, and then try to assemble the big scenario and test that.
Here's what we'll try to accomplish:
1. Create a Maven build for the project
2. Get sample files for the customer Excel, CSV, and XML input
3. Get a sample file for the canonical XML format that Acme's accounting system uses
4. Create an XSD for the canonical XML format
5. Create JAXB POJOs corresponding to the canonical XSD
6. Create an XSLT stylesheet to convert the Customer 1 (XML over FTP) messages to
the canonical format

181

TU TO RI A L S

7. Create a unit test to ensure that a simple Camel route invoking the XSLT stylesheet
works
8. Create a POJO that converts a List> to the above JAXB POJOs
◦ Note that Camel can automatically convert CSV input to a List of Lists of
Strings representing the rows and columns of the CSV, so we'll use this
POJO to handle Customer 2 (CSV over HTTP)
9. Create a unit test to ensure that a simple Camel route invoking the CSV processing
works
10. Create a POJO that converts a Customer 3 Excel file to the above JAXB POJOs
(using POI to read Excel)
11. Create a unit test to ensure that a simple Camel route invoking the Excel processing
works
12. Create a POJO that reads an input message, takes an attachment off the message, and
replaces the body of the message with the attachment
◦ This is assuming for Customer 3 (Excel over e-mail) that the e-mail contains
a single Excel file as an attachment, and the actual e-mail body is throwaway
13. Build a set of Camel routes to handle the entire input (Customer -> Acme) side of
the scenario.
14. Build unit tests for the Camel input.
15. TODO: Tasks for the output (Acme -> Customer) side of the scenario

LET'S GET STARTED!
Step 1: Initial Maven build
We'll use Maven for this project as there will eventually be quite a few dependencies and it's
nice to have Maven handle them for us. You should have a current version of Maven (e.g. 2.0.9)
installed.
You can start with a pretty empty project directory and a Maven POM file, or use a simple
JAR archetype to create one.
Here's a sample POM. We've added a dependency on camel-core, and set the compile
version to 1.5 (so we can use annotations):
Listing 8. pom.xml


4.0.0
org.apache.camel.tutorial
business-partners
1.0-SNAPSHOT
Camel Business Partners Tutorial



T UT O R IALS

182

camel-core
org.apache.camel
1.4.0





org.apache.maven.plugins
maven-compiler-plugin

1.5
1.5






Step 2: Get Sample Files
You can make up your own if you like, but here are the "off the shelf" ones. You can save
yourself some time by downloading these to src/test/resources in your Maven project.
• Customer 1 (XML): input-customer1.xml
• Customer 2 (CSV): input-customer2.csv
• Customer 3 (Excel): input-customer3.xls
• Canonical Acme XML Request: canonical-acme-request.xml
• Canonical Acme XML Response: TODO
If you look at these files, you'll see that the different input formats use different field names and/
or ordering, because of course the sales guys were totally OK with that. Sigh.
Step 3: XSD and JAXB Beans for the Canonical XML Format
Here's the sample of the canonical XML file:


2
9/12/2008

134
A widget
3
10.45
6/5/2008



183

TU TO RI A L S

218.82


If you're ambitions, you can write your own XSD (XML Schema) for files that look like this, and
save it to src/main/xsd.
Solution: If not, you can download mine, and save that to save it to src/main/xsd.

Generating JAXB Beans
Down the road we'll want to deal with the XML as Java POJOs. We'll take a moment now to
set up those XML binding POJOs. So we'll update the Maven POM to generate JAXB beans
from the XSD file.
We need a dependency:

camel-jaxb
org.apache.camel
1.4.0


And a plugin configured:

org.codehaus.mojo
jaxb2-maven-plugin



xjc





That should do it (it automatically looks for XML Schemas in src/main/xsd to generate
beans for). Run mvn install and it should emit the beans into target/generatedsources/jaxb. Your IDE should see them there, though you may need to update the
project to reflect the new settings in the Maven POM.
Step 4: Initial Work on Customer 1 Input (XML over FTP)
To get a start on Customer 1, we'll create an XSLT template to convert the Customer 1
sample file into the canonical XML format, write a small Camel route to test it, and build that

T UT O R IALS

184

into a unit test. If we get through this, we can be pretty sure that the XSLT template is valid and
can be run safely in Camel.

Create an XSLT template
Start with the Customer 1 sample input. You want to create an XSLT template to generate
XML like the canonical XML sample above – an invoice element with line-item elements
(one per item in the original XML document). If you're especially clever, you can populate the
current date and order total elements too.
Solution: My sample XSLT template isn't that smart, but it'll get you going if you don't
want to write one of your own.

Create a unit test
Here's where we get to some meaty Camel work. We need to:
• Set up a unit test
• That loads a Camel configuration
• That has a route invoking our XSLT
• Where the test sends a message to the route
• And ensures that some XML comes out the end of the route
The easiest way to do this is to set up a Spring context that defines the Camel stuff, and then
use a base unit test class from Spring that knows how to load a Spring context to run tests
against. So, the procedure is:
Set Up a Skeletal Camel/Spring Unit Test
1. Add dependencies on Camel-Spring, and the Spring test JAR (which will automatically
bring in JUnit 3.8.x) to your POM:

camel-spring
org.apache.camel
1.4.0


spring-test
org.springframework
2.5.5
test


2. Create a new unit test class in src/test/java/your-package-here, perhaps
called XMLInputTest.java

185

TU TO RI A L S

3. Make the test extend Spring's AbstractJUnit38SpringContextTests class, so it can load
a Spring context for the test
4. Create a Spring context configuration file in src/test/resources, perhaps
called XMLInputTest-context.xml
5. In the unit test class, use the class-level @ContextConfiguration annotation to
indicate that a Spring context should be loaded
◦ By default, this looks for a Context configuration file called
TestClassName-context.xml in a subdirectory corresponding to the
package of the test class. For instance, if your test class was
org.apache.camel.tutorial.XMLInputTest, it would look for
org/apache/camel/tutorial/XMLInputTest-context.xml
◦ To override this default, use the locations attribute on the
@ContextConfiguration annotation to provide specific context file
locations (starting each path with a / if you don't want it to be relative to
the package directory). My solution does this so I can put the context file
directly in src/test/resources instead of in a package directory
under there.
6. Add a CamelContext instance variable to the test class, with the @Autowired
annotation. That way Spring will automatically pull the CamelContext out of the
Spring context and inject it into our test class.
7. Add a ProducerTemplate instance variable and a setUp method that instantiates it
from the CamelContext. We'll use the ProducerTemplate later to send messages to
the route.
protected ProducerTemplate template;
protected void setUp() throws Exception {
super.setUp();
template = camelContext.createProducerTemplate();
}

8. Put in an empty test method just for the moment (so when we run this we can see
that "1 test succeeded")
9. Add the Spring  element (including the Camel Namespace) with an empty
 element to the Spring context, like this:



T UT O R IALS

186





Test it by running mvn install and make sure there are no build errors. So far it doesn't test
much; just that your project and test and source files are all organized correctly, and the one
empty test method completes successfully.
Solution: Your test class might look something like this:
• src/test/java/org/apache/camel/tutorial/XMLInputTest.java
• src/test/resources/XMLInputTest-context.xml (same as just above)
Flesh Out the Unit Test
So now we're going to write a Camel route that applies the XSLT to the sample Customer 1
input file, and makes sure that some XML output comes out:
1. Save the input-customer1.xml file to src/test/resources
2. Save your XSLT file (created in the previous step) to src/main/resources
3. Write a Camel Route, either right in the Spring XML, or using the Java DSL (in
another class under src/test/java somewhere). This route should use the Pipes
and Filters integration pattern to:
1. Start from the endpoint direct:start (which lets the test conveniently pass
messages into the route)
2. Call the endpoint xslt:YourXSLTFile.xsl (to transform the message with the
specified XSLT template)
3. Send the result to the endpoint mock:finish (which lets the test verify the
route output)
4. Add a test method to the unit test class that:
1. Get a reference to the Mock endpoint mock:finish using code like this:
MockEndpoint finish = MockEndpoint.resolve(camelContext,
"mock:finish");

2. Set the expectedMessageCount on that endpoint to 1
3. Get a reference to the Customer 1 input file, using code like this:
InputStream in =
XMLInputTest.class.getResourceAsStream("/input-partner1.xml");
assertNotNull(in);

4. Send that InputStream as a message to the direct:start endpoint,
using code like this:

187

TU TO RI A L S

template.sendBody("direct:start", in);

Note that we can send the sample file body in several formats (File,
InputStream, String, etc.) but in this case an InputStream is pretty
convenient.
5. Ensure that the message made it through the route to the final endpoint, by
testing all configured Mock endpoints like this:
MockEndpoint.assertIsSatisfied(camelContext);

6. If you like, inspect the final message body using some code like
finish.getExchanges().get(0).getIn().getBody().
▪ If you do this, you'll need to know what format that body is –
String, byte array, InputStream, etc.
5. Run your test with mvn install and make sure the build completes successfully.
Solution: Your finished test might look something like this:
• src/test/java/org/apache/camel/tutorial/XMLInputTest.java
• For XML Configuration:
◦ src/test/resources/XMLInputTest-context.xml
• Or, for Java DSL Configuration:
◦ src/test/resources/XMLInputTest-dsl-context.xml
◦ src/test/java/org/apache/camel/tutorial/routes/XMLInputTestRoute.java
Step 5: Initial Work on Customer 2 Input (CSV over HTTP)
To get a start on Customer 2, we'll create a POJO to convert the Customer 2 sample CSV data
into the JAXB POJOs representing the canonical XML format, write a small Camel route to test
it, and build that into a unit test. If we get through this, we can be pretty sure that the CSV
conversion and JAXB handling is valid and can be run safely in Camel.

Create a CSV-handling POJO
To begin with, CSV is a known data format in Camel. Camel can convert a CSV file to a List
(representing rows in the CSV) of Lists (representing cells in the row) of Strings (the data for
each cell). That means our POJO can just assume the data coming in is of type
List>, and we can declare a method with that as the argument.
Looking at the JAXB code in target/generated-sources/jaxb, it looks like an
Invoice object represents the whole document, with a nested list of LineItemType objects
for the line items. Therefore our POJO method will return an Invoice (a document in the
canonical XML format).
So to implement the CSV-to-JAXB POJO, we need to do something like this:

T UT O R IALS

188

Test Base Class
Once your test class is working, you might want to extract things like the
@Autowired CamelContext, the ProducerTemplate, and the setUp method to a
custom base class that you extend with your other tests.
1. Create a new class under src/main/java, perhaps called CSVConverterBean.
2. Add a method, with one argument of type List> and the return
type Invoice
◦ You may annotate the argument with @Body to specifically designate it as
the body of the incoming message
3. In the method, the logic should look roughly like this:
1. Create a new Invoice, using the method on the generated
ObjectFactory class
2. Loop through all the rows in the incoming CSV (the outer List)
3. Skip the first row, which contains headers (column names)
4. For the other rows:
1. Create a new LineItemType (using the ObjectFactory
again)
2. Pick out all the cell values (the Strings in the inner List) and put
them into the correct fields of the LineItemType
▪ Not all of the values will actually go into the line item in
this example
▪ You may hardcode the column ordering based on the
sample data file, or else try to read it dynamically from
the headers in the first line
▪ Note that you'll need to use a JAXB
DatatypeFactory to create the
XMLGregorianCalendar values that JAXB uses for
the date fields in the XML – which probably means
using a SimpleDateFormat to parse the date and
setting that date on a GregorianCalendar
3. Add the line item to the invoice
5. Populate the partner ID, date of receipt, and order total on the Invoice
6. Throw any exceptions out of the method, so Camel knows something went
wrong
7. Return the finished Invoice
Solution: Here's an example of what the CSVConverterBean might look like.

189

TU TO RI A L S

Create a unit test
Start with a simple test class and test Spring context like last time, perhaps based on the name
CSVInputTest:
Listing 9. CSVInputTest.java
/**
* A test class the ensure we can convert Partner 2 CSV input files to the
* canonical XML output format, using JAXB POJOs.
*/
@ContextConfiguration(locations = "/CSVInputTest-context.xml")
public class CSVInputTest extends AbstractJUnit38SpringContextTests {
@Autowired
protected CamelContext camelContext;
protected ProducerTemplate template;
protected void setUp() throws Exception {
super.setUp();
template = camelContext.createProducerTemplate();
}
public void testCSVConversion() {
// TODO
}
}

Listing 10. CSVInputTest-context.xml







Now the meaty part is to flesh out the test class and write the Camel routes.
1. Update the Maven POM to include CSV Data Format support:

camel-csv
org.apache.camel
1.4.0


T UT O R IALS

190

2. Write the routes (right in the Spring XML context, or using the Java DSL) for the
CSV conversion process, again using the Pipes and Filters pattern:
1. Start from the endpoint direct:CSVstart (which lets the test conveniently
pass messages into the route). We'll name this differently than the starting
point for the previous test, in case you use the Java DSL and put all your
routes in the same package (which would mean that each test would load
the DSL routes for several tests.)
2. This time, there's a little preparation to be done. Camel doesn't know that
the initial input is a CSV, so it won't be able to convert it to the expected
List> without a little hint. For that, we need an
unmarshal transformation in the route. The unmarshal method (in the
DSL) or element (in the XML) takes a child indicating the format to
unmarshal; in this case that should be csv.
3. Next invoke the POJO to transform the message with a
bean:CSVConverter endpoint
4. As before, send the result to the endpoint mock:finish (which lets the test
verify the route output)
5. Finally, we need a Spring  element in the Spring context XML file
(but outside the  element) to define the Spring bean
that our route invokes. This Spring bean should have a name attribute that
matches the name used in the bean endpoint (CSVConverter in the
example above), and a class attribute that points to the CSV-to-JAXB
POJO class you wrote above (such as,
org.apache.camel.tutorial.CSVConverterBean). When
Spring is in the picture, any bean endpoints look up Spring beans with the
specified name.
3. Write a test method in the test class, which should look very similar to the previous
test class:
1. Get the MockEndpoint for the final endpoint, and tell it to expect one
message
2. Load the Partner 2 sample CSV file from the ClassPath, and send it as the
body of a message to the starting endpoint
3. Verify that the final MockEndpoint is satisfied (that is, it received one
message) and examine the message body if you like
▪ Note that we didn't marshal the JAXB POJOs to XML in this test,
so the final message should contain an Invoice as the body. You
could write a simple line of code to get the Exchange (and
Message) from the MockEndpoint to confirm that.
4. Run this new test with mvn install and make sure it passes and the build completes
successfully.
Solution: Your finished test might look something like this:
• src/test/java/org/apache/camel/tutorial/CSVInputTest.java
• For XML Configuration:

191

TU TO RI A L S

◦ src/test/resources/CSVInputTest-context.xml
• Or, for Java DSL Configuration:
◦ src/test/resources/CSVInputTest-dsl-context.xml
◦ src/test/java/org/apache/camel/tutorial/routes/CSVInputTestRoute.java
Step 6: Initial Work on Customer 3 Input (Excel over e-mail)
To get a start on Customer 3, we'll create a POJO to convert the Customer 3 sample Excel
data into the JAXB POJOs representing the canonical XML format, write a small Camel route
to test it, and build that into a unit test. If we get through this, we can be pretty sure that the
Excel conversion and JAXB handling is valid and can be run safely in Camel.

Create an Excel-handling POJO
Camel does not have a data format handler for Excel by default. We have two options – create
an Excel DataFormat (so Camel can convert Excel spreadsheets to something like the CSV
List> automatically), or create a POJO that can translate Excel data
manually. For now, the second approach is easier (if we go the DataFormat route, we need
code to both read and write Excel files, whereas otherwise read-only will do).
So, we need a POJO with a method that takes something like an InputStream or
byte[] as an argument, and returns in Invoice as before. The process should look
something like this:
1. Update the Maven POM to include POI support:

poi
org.apache.poi
3.1-FINAL


2. Create a new class under src/main/java, perhaps called
ExcelConverterBean.
3. Add a method, with one argument of type InputStream and the return type
Invoice
◦ You may annotate the argument with @Body to specifically designate it as
the body of the incoming message
4. In the method, the logic should look roughly like this:
1. Create a new Invoice, using the method on the generated
ObjectFactory class
2. Create a new HSSFWorkbook from the InputStream, and get the first
sheet from it
3. Loop through all the rows in the sheet
4. Skip the first row, which contains headers (column names)

T UT O R IALS

192

5. For the other rows:
1. Create a new LineItemType (using the ObjectFactory
again)
2. Pick out all the cell values and put them into the correct fields of
the LineItemType (you'll need some data type conversion
logic)
▪ Not all of the values will actually go into the line item in
this example
▪ You may hardcode the column ordering based on the
sample data file, or else try to read it dynamically from
the headers in the first line
▪ Note that you'll need to use a JAXB
DatatypeFactory to create the
XMLGregorianCalendar values that JAXB uses for
the date fields in the XML – which probably means
setting the date from a date cell on a
GregorianCalendar
3. Add the line item to the invoice
6. Populate the partner ID, date of receipt, and order total on the Invoice
7. Throw any exceptions out of the method, so Camel knows something went
wrong
8. Return the finished Invoice
Solution: Here's an example of what the ExcelConverterBean might look like.

Create a unit test
The unit tests should be pretty familiar now. The test class and context for the Excel bean
should be quite similar to the CSV bean.
1. Create the basic test class and corresponding Spring Context XML configuration file
2. The XML config should look a lot like the CSV test, except:
◦ Remember to use a different start endpoint name if you're using the Java
DSL and not use separate packages per test
◦ You don't need the unmarshal step since the Excel POJO takes the raw
InputStream from the source endpoint
◦ You'll declare a  and endpoint for the Excel bean prepared above
instead of the CSV bean
3. The test class should look a lot like the CSV test, except use the right input file name
and start endpoint name.
Solution: Your finished test might look something like this:
• src/test/java/org/apache/camel/tutorial/ExcelInputTest.java
• For XML Configuration:
◦ src/test/resources/ExcelInputTest-context.xml

193

TU TO RI A L S

Logging
You may notice that your tests emit a lot less output all of a sudden. The
dependency on POI brought in Log4J and configured commons-logging to use it, so
now we need a log4j.properties file to configure log output. You can use the
attached one (snarfed from ActiveMQ) or write your own; either way save it to
src/main/resources to ensure you continue to see log output.
• Or, for Java DSL Configuration:
◦ src/test/resources/ExcelInputTest-dsl-context.xml
◦ src/test/java/org/apache/camel/tutorial/routes/ExcelInputTestRoute.java
Step 7: Put this all together into Camel routes for the Customer Input
With all the data type conversions working, the next step is to write the real routes that listen
for HTTP, FTP, or e-mail input, and write the final XML output to an ActiveMQ queue. Along
the way these routes will use the data conversions we've developed above.
So we'll create 3 routes to start with, as shown in the diagram back at the beginning:
1. Accept XML orders over FTP from Customer 1 (we'll assume the FTP server dumps
files in a local directory on the Camel machine)
2. Accept CSV orders over HTTP from Customer 2
3. Accept Excel orders via e-mail from Customer 3 (we'll assume the messages are sent
to an account we can access via IMAP)
...
Step 8: Create a unit test for the Customer Input Routes

T UT O R IALS

194

Languages Supported Appendix

To support flexible and powerful Enterprise Integration Patterns Camel supports various
Languages to create an Expression or Predicate within either the Routing Domain Specific
Language or the Xml Configuration. The following languages are supported

BEAN LANGUAGE
The purpose of the Bean Language is to be able to implement an Expression or Predicate using
a simple method on a bean.
So the idea is you specify a bean name which will then be resolved in the Registry such as
the Spring ApplicationContext then a method is invoked to evaluate the Expression or
Predicate.
If no method name is provided then one is attempted to be chosen using the rules for Bean
Binding; using the type of the message body and using any annotations on the bean methods.
The Bean Binding rules are used to bind the Message Exchange to the method parameters;
so you can annotate the bean to extract headers or other expressions such as XPath or
XQuery from the message.
Using Bean Expressions from the Java DSL
from("activemq:topic:OrdersTopic").
filter().method("myBean", "isGoldCustomer").
to("activemq:BigSpendersQueue");

Using Bean Expressions from XML








195

L A N G U A G E S S U P P OR T E D A P P E ND I X

Writing the expression bean
The bean in the above examples is just any old Java Bean with a method called
isGoldCustomer() that returns some object that is easily converted to a boolean value in this
case, as its used as a predicate.
So we could implement it like this...
public class MyBean {
public boolean isGoldCustomer(Exchange exchange) {
...
}
}

We can also use the Bean Integration annotations. For example you could do...
public boolean isGoldCustomer(String body) {...}

or
public boolean isGoldCustomer(@Header(name = "foo") Integer fooHeader) {...}

So you can bind parameters of the method to the Exchange, the Message or individual headers,
properties, the body or other expressions.
Non registry beans
As of Camel 1.5 the Bean Language also supports invoking beans that isn't registered in the
Registry. This is usable for quickly to invoke a bean from Java DSL where you don't need to
register the bean in the Registry such as the Spring ApplicationContext.
Camel can instantiate the bean and invoke the method if given a class or invoke an already
existing instance. This is illustrated from the example below:
NOTE This bean DSL is supported since Camel 2.0-M2
from("activemq:topic:OrdersTopic").
filter().expression(BeanLanguage(MyBean.class, "isGoldCustomer")).
to("activemq:BigSpendersQueue");

The 2nd parameter isGoldCustomer is an optional parameter to explicit set the method
name to invoke. If not provided Camel will try to invoke the best suited method. If case of
ambiguity Camel will thrown an Exception. In these situations the 2nd parameter can solve this
problem. Also the code is more readable if the method name is provided. The 1st parameter
can also be an existing instance of a Bean such as:

LAN GUAGES SUPPO R T ED APPEN DIX

196

private MyBean my;
from("activemq:topic:OrdersTopic").
filter().expression(BeanLanguage.bean(my, "isGoldCustomer")).
to("activemq:BigSpendersQueue");

In Camel 2.2 onwards you can avoid the BeanLanguage and have it just as:
private MyBean my;
from("activemq:topic:OrdersTopic").
filter().expression(bean(my, "isGoldCustomer")).
to("activemq:BigSpendersQueue");

Which also can be done in a bit shorter and nice way:
private MyBean my;
from("activemq:topic:OrdersTopic").
filter().method(my, "isGoldCustomer").
to("activemq:BigSpendersQueue");

Other examples
We have some test cases you can look at if it'll help
• MethodFilterTest is a JUnit test case showing the Java DSL use of the bean expression
being used in a filter
• aggregator.xml is a Spring XML test case for the Aggregator which uses a bean
method call to test for the completion of the aggregation.
Dependencies
The Bean language is part of camel-core.

CONSTANT EXPRESSION LANGUAGE
The Constant Expression Language is really just a way to specify constant strings as a type of
expression.
Available as of Camel 1.5
Example usage
The setHeader element of the Spring DSL can utilize a constant expression like:

197

L A N G U A G E S S U P P OR T E D A P P E ND I X




the value




in this case, the Message coming from the seda:a Endpoint will have 'theHeader' header set to
the constant value 'the value'.
And the same example using Java DSL:
from("seda:a").setHeader("theHeader", constant("the value")).to("mock:b");

Dependencies
The Constant language is part of camel-core.

EL
Camel supports the unified JSP and JSF Expression Language via the JUEL to allow an Expression
or Predicate to be used in the DSL or Xml Configuration.
For example you could use EL inside a Message Filter in XML



${in.headers.foo == 'bar'}




You could also use slightly different syntax, e.g. if the header name is not a valid identifier:



${in.headers['My Header'] == 'bar'}




You could use EL to create an Predicate in a Message Filter or as an Expression for a Recipient
List

LAN GUAGES SUPPO R T ED APPEN DIX

198

Variables
Variable

Type

Description

exchange

Exchange

the Exchange object

in

Message

the exchange.in message

out

Message

the exchange.out message

Samples
You can use EL dot notation to invoke operations. If you for instance have a body that contains
a POJO that has a getFamiliyName method then you can construct the syntax as follows:
"$in.body.familyName"

Dependencies
To use EL in your camel routes you need to add the a dependency on camel-juel which
implements the EL language.
If you use maven you could just add the following to your pom.xml, substituting the version
number for the latest & greatest release (see the download page for the latest versions).

org.apache.camel
camel-juel
1.6.1


HEADER EXPRESSION LANGUAGE
The Header Expression Language allows you to extract values of named headers.
Available as of Camel 1.5
Example usage
The recipientList element of the Spring DSL can utilize a header expression like:





199

L A N G U A G E S S U P P OR T E D A P P E ND I X


myHeader
In this case, the list of recipients are contained in the header 'myHeader'. And the same example in Java DSL: from("direct:a").recipientList(header("myHeader")); And with a slightly different syntax where you use the builder to the fullest (i.e. avoid using parameters but using stacked operations, notice that header is not a parameter but a stacked method call) from("direct:a").recipientList().header("myHeader"); Dependencies The Header language is part of camel-core. JXPATH Camel supports JXPath to allow XPath expressions to be used on beans in an Expression or Predicate to be used in the DSL or Xml Configuration. For example you could use JXPath to create an Predicate in a Message Filter or as an Expression for a Recipient List. From 1.3 of Camel onwards you can use XPath expressions directly using smart completion in your IDE as follows from("queue:foo").filter(). jxpath("/in/body/foo"). to("queue:bar") Variables Variable Type Description this Exchange the Exchange object in Message the exchange.in message out Message the exchange.out message LAN GUAGES SUPPO R T ED APPEN DIX 200 Using XML configuration If you prefer to configure your routes in your Spring XML file then you can use JXPath expressions as follows in/body/name = 'James' Examples Here is a simple example using a JXPath expression as a predicate in a Message Filter from("direct:start"). filter().jxpath("in/body/name='James'"). to("mock:result"); JXPATH INJECTION You can use Bean Integration to invoke a method on a bean and use various languages such as JXPath to extract a value from the message and bind it to a method parameter. For example public class Foo { @MessageDriven(uri = "activemq:my.queue") public void doSomething(@JXPath("in/body/foo") String correlationID, @Body String body) { // process the inbound message here } } 201 L A N G U A G E S S U P P OR T E D A P P E ND I X Dependencies To use JXpath in your camel routes you need to add the a dependency on camel-jxpath which implements the JXpath language. If you use maven you could just add the following to your pom.xml, substituting the version number for the latest & greatest release (see the download page for the latest versions). org.apache.camel camel-jxpath 1.4.0 MVEL Avialable in Camel 2.0 Camel allows Mvel to be used as an Expression or Predicate the DSL or Xml Configuration. You could use Mvel to create an Predicate in a Message Filter or as an Expression for a Recipient List You can use Mvel dot notation to invoke operations. If you for instance have a body that contains a POJO that has a getFamiliyName method then you can construct the syntax as follows: "request.body.familyName" // or "getRequest().getBody().getFamilyName()" Variables Variable Type Description this Exchange the Exchange is the root object exchange Exchange the Exchange object exception Throwable the Exchange exception (if any) exchangeId String the exchange id fault Message the Fault message (if any) request Message the exchange.in message response Message the exchange.out message (if any) properties Map the exchange properties LAN GUAGES SUPPO R T ED APPEN DIX 202 property(name) Object the property by the given name property(name, type) Type the property by the given name as the given type Samples For example you could use Mvel inside a Message Filter in XML request.headers.foo == 'bar' And the sample using Java DSL: from("seda:foo").filter().mvel("request.headers.foo == 'bar'").to("seda:bar"); Dependencies To use Mvel in your camel routes you need to add the a dependency on camel-mvel which implements the Mvel language. If you use maven you could just add the following to your pom.xml, substituting the version number for the latest & greatest release (see the download page for the latest versions). org.apache.camel camel-mvel 2.0.0 OGNL Camel allows OGNL to be used as an Expression or Predicate the DSL or Xml Configuration. You could use OGNL to create an Predicate in a Message Filter or as an Expression for a Recipient List You can use OGNL dot notation to invoke operations. If you for instance have a body that contains a POJO that has a getFamiliyName method then you can construct the syntax as follows: 203 L A N G U A G E S S U P P OR T E D A P P E ND I X "request.body.familyName" // or "getRequest().getBody().getFamilyName()" Variables Variable Type Description this Exchange the Exchange is the root object exchange Exchange the Exchange object exception Throwable the Exchange exception (if any) exchangeId String the exchange id fault Message the Fault message (if any) request Message the exchange.in message response Message the exchange.out message (if any) properties Map the exchange properties property(name) Object the property by the given name property(name, type) Type the property by the given name as the given type Samples For example you could use OGNL inside a Message Filter in XML request.headers.foo = 'bar' And the sample using Java DSL: from("seda:foo").filter().ognl("request.headers.foo = 'bar'").to("seda:bar"); LAN GUAGES SUPPO R T ED APPEN DIX 204 Dependencies To use OGNL in your camel routes you need to add the a dependency on camel-ognl which implements the OGNL language. If you use maven you could just add the following to your pom.xml, substituting the version number for the latest & greatest release (see the download page for the latest versions). org.apache.camel camel-ognl 1.4.0 PROPERTY EXPRESSION LANGUAGE The Property Expression Language allows you to extract values of named exchange properties. Available as of Camel 2.0 Example usage The recipientList element of the Spring DSL can utilize a property expression like: myProperty In this case, the list of recipients are contained in the property 'myProperty'. And the same example in Java DSL: from("direct:a").recipientList(property("myProperty")); And with a slightly different syntax where you use the builder to the fullest (i.e. avoid using parameters but using stacked operations, notice that property is not a parameter but a stacked method call) from("direct:a").recipientList().property("myProperty"); Dependencies The Property language is part of camel-core. 205 L A N G U A G E S S U P P OR T E D A P P E ND I X SCRIPTING LANGUAGES Camel supports a number of scripting languages which can be used to create an Expression or Predicate via the standard JSR 223 which is a standard part of Java 6. The following scripting languages are integrated into the DSL: • BeanShell • JavaScript • Groovy • Python • PHP • Ruby However any JSR 223 scripting language can be used using the generic DSL methods. ScriptContext The JSR-223 scripting languages ScriptContext is pre configured with the following attributes all set at ENGINE_SCOPE: Attribute Type Value context org.apache.camel.CamelContext The Camel Context exchange org.apache.camel.Exchange The current Exchange request org.apache.camel.Message The IN message response org.apache.camel.Message The OUT message Attributes You can add your own attributes with the attribute(name, value) DSL method, such as: In the sample below we add an attribute user that is an object we already have instantiated as myUser. This object has a getFirstName() method that we want to set as header on the message. We use the groovy language to concat the first and last name into a single string that is returned. from("direct:in").setHeader("name").groovy("'$user.firstName $user.lastName'").attribute("user", myUser).to("seda:users"); Any scripting language Camel can run any JSR-223 scripting languages using the script DSL method such as: LAN GUAGES SUPPO R T ED APPEN DIX 206 from("direct:in").setHeader("firstName").script("jaskel", "user.firstName").attribute("user", myUser).to("seda:users"); This is a bit different using the Spring DSL where you use the expression element that doesn't support setting attributes (yet): user.firstName Dependencies To use scripting languages in your camel routes you need to add the a dependency on camelscript which integrates the JSR-223 scripting engine. If you use maven you could just add the following to your pom.xml, substituting the version number for the latest & greatest release (see the download page for the latest versions). org.apache.camel camel-script 1.4.0 SEE ALSO • Languages • DSL • Xml Configuration BEANSHELL Camel supports BeanShell among other Scripting Languages to allow an Expression or Predicate to be used in the DSL or Xml Configuration. To use a BeanShell expression use the following Java code ... beanShell("someBeanShellExpression") ... For example you could use the beanShell function to create an Predicate in a Message Filter or as an Expression for a Recipient List 207 L A N G U A G E S S U P P OR T E D A P P E ND I X ScriptContext The JSR-223 scripting languages ScriptContext is pre configured with the following attributes all set at ENGINE_SCOPE: Attribute Type Value context org.apache.camel.CamelContext The Camel Context exchange org.apache.camel.Exchange The current Exchange request org.apache.camel.Message The IN message response org.apache.camel.Message The OUT message Attributes You can add your own attributes with the attribute(name, value) DSL method, such as: In the sample below we add an attribute user that is an object we already have instantiated as myUser. This object has a getFirstName() method that we want to set as header on the message. We use the groovy language to concat the first and last name into a single string that is returned. from("direct:in").setHeader("name").groovy("'$user.firstName $user.lastName'").attribute("user", myUser).to("seda:users"); Any scripting language Camel can run any JSR-223 scripting languages using the script DSL method such as: from("direct:in").setHeader("firstName").script("jaskel", "user.firstName").attribute("user", myUser).to("seda:users"); This is a bit different using the Spring DSL where you use the expression element that doesn't support setting attributes (yet): user.firstName LAN GUAGES SUPPO R T ED APPEN DIX 208 Dependencies To use scripting languages in your camel routes you need to add the a dependency on camelscript which integrates the JSR-223 scripting engine. If you use maven you could just add the following to your pom.xml, substituting the version number for the latest & greatest release (see the download page for the latest versions). org.apache.camel camel-script 1.4.0 JAVASCRIPT Camel supports JavaScript/ECMAScript among other Scripting Languages to allow an Expression or Predicate to be used in the DSL or Xml Configuration. To use a JavaScript expression use the following Java code ... javaScript("someJavaScriptExpression") ... For example you could use the javaScript function to create an Predicate in a Message Filter or as an Expression for a Recipient List Example In the sample below we use JavaScript to create a Predicate use in the route path, to route exchanges from admin users to a special queue. from("direct:start") .choice() .when().javaScript("request.headers.get('user') == 'admin'").to("seda:adminQueue") .otherwise() .to("seda:regularQueue"); And a Spring DSL sample as well: request.headers.get('user') == 'admin' 209 L A N G U A G E S S U P P OR T E D A P P E ND I X ScriptContext The JSR-223 scripting languages ScriptContext is pre configured with the following attributes all set at ENGINE_SCOPE: Attribute Type Value context org.apache.camel.CamelContext The Camel Context exchange org.apache.camel.Exchange The current Exchange request org.apache.camel.Message The IN message response org.apache.camel.Message The OUT message Attributes You can add your own attributes with the attribute(name, value) DSL method, such as: In the sample below we add an attribute user that is an object we already have instantiated as myUser. This object has a getFirstName() method that we want to set as header on the message. We use the groovy language to concat the first and last name into a single string that is returned. from("direct:in").setHeader("name").groovy("'$user.firstName $user.lastName'").attribute("user", myUser).to("seda:users"); Any scripting language Camel can run any JSR-223 scripting languages using the script DSL method such as: from("direct:in").setHeader("firstName").script("jaskel", "user.firstName").attribute("user", myUser).to("seda:users"); This is a bit different using the Spring DSL where you use the expression element that doesn't support setting attributes (yet): LAN GUAGES SUPPO R T ED APPEN DIX 210 user.firstName Dependencies To use scripting languages in your camel routes you need to add the a dependency on camelscript which integrates the JSR-223 scripting engine. If you use maven you could just add the following to your pom.xml, substituting the version number for the latest & greatest release (see the download page for the latest versions). org.apache.camel camel-script 1.4.0 GROOVY Camel supports Groovy among other Scripting Languages to allow an Expression or Predicate to be used in the DSL or Xml Configuration. To use a Groovy expression use the following Java code ... groovy("someGroovyExpression") ... For example you could use the groovy function to create an Predicate in a Message Filter or as an Expression for a Recipient List Example // lets route if a line item is over $100 from("queue:foo").filter(groovy("request.lineItems.any { i -> i.value > 100 }")).to("queue:bar") And the Spring DSL: 211 L A N G U A G E S S U P P OR T E D A P P E ND I X request.lineItems.any { i -> i.value > 100 } ScriptContext The JSR-223 scripting languages ScriptContext is pre configured with the following attributes all set at ENGINE_SCOPE: Attribute Type Value context org.apache.camel.CamelContext The Camel Context exchange org.apache.camel.Exchange The current Exchange request org.apache.camel.Message The IN message response org.apache.camel.Message The OUT message Attributes You can add your own attributes with the attribute(name, value) DSL method, such as: In the sample below we add an attribute user that is an object we already have instantiated as myUser. This object has a getFirstName() method that we want to set as header on the message. We use the groovy language to concat the first and last name into a single string that is returned. from("direct:in").setHeader("name").groovy("'$user.firstName $user.lastName'").attribute("user", myUser).to("seda:users"); Any scripting language Camel can run any JSR-223 scripting languages using the script DSL method such as: from("direct:in").setHeader("firstName").script("jaskel", "user.firstName").attribute("user", myUser).to("seda:users"); This is a bit different using the Spring DSL where you use the expression element that doesn't support setting attributes (yet): LAN GUAGES SUPPO R T ED APPEN DIX 212 user.firstName Dependencies To use scripting languages in your camel routes you need to add the a dependency on camelscript which integrates the JSR-223 scripting engine. If you use maven you could just add the following to your pom.xml, substituting the version number for the latest & greatest release (see the download page for the latest versions). org.apache.camel camel-script 1.4.0 PYTHON Camel supports Python among other Scripting Languages to allow an Expression or Predicate to be used in the DSL or Xml Configuration. To use a Python expression use the following Java code ... python("somePythonExpression") ... For example you could use the python function to create an Predicate in a Message Filter or as an Expression for a Recipient List Example In the sample below we use Python to create a Predicate use in the route path, to route exchanges from admin users to a special queue. from("direct:start") .choice() .when().python("request.headers['user'] == 'admin'").to("seda:adminQueue") .otherwise() .to("seda:regularQueue"); And a Spring DSL sample as well: 213 L A N G U A G E S S U P P OR T E D A P P E ND I X request.headers['user'] == 'admin' ScriptContext The JSR-223 scripting languages ScriptContext is pre configured with the following attributes all set at ENGINE_SCOPE: Attribute Type Value context org.apache.camel.CamelContext The Camel Context exchange org.apache.camel.Exchange The current Exchange request org.apache.camel.Message The IN message response org.apache.camel.Message The OUT message Attributes You can add your own attributes with the attribute(name, value) DSL method, such as: In the sample below we add an attribute user that is an object we already have instantiated as myUser. This object has a getFirstName() method that we want to set as header on the message. We use the groovy language to concat the first and last name into a single string that is returned. from("direct:in").setHeader("name").groovy("'$user.firstName $user.lastName'").attribute("user", myUser).to("seda:users"); Any scripting language Camel can run any JSR-223 scripting languages using the script DSL method such as: LAN GUAGES SUPPO R T ED APPEN DIX 214 from("direct:in").setHeader("firstName").script("jaskel", "user.firstName").attribute("user", myUser).to("seda:users"); This is a bit different using the Spring DSL where you use the expression element that doesn't support setting attributes (yet): user.firstName Dependencies To use scripting languages in your camel routes you need to add the a dependency on camelscript which integrates the JSR-223 scripting engine. If you use maven you could just add the following to your pom.xml, substituting the version number for the latest & greatest release (see the download page for the latest versions). org.apache.camel camel-script 1.4.0 PHP Camel supports PHP among other Scripting Languages to allow an Expression or Predicate to be used in the DSL or Xml Configuration. To use a PHP expression use the following Java code ... php("somePHPExpression") ... For example you could use the php function to create an Predicate in a Message Filter or as an Expression for a Recipient List ScriptContext The JSR-223 scripting languages ScriptContext is pre configured with the following attributes all set at ENGINE_SCOPE: Attribute 215 Type L A N G U A G E S S U P P OR T E D A P P E ND I X Value context org.apache.camel.CamelContext The Camel Context exchange org.apache.camel.Exchange The current Exchange request org.apache.camel.Message The IN message response org.apache.camel.Message The OUT message Attributes You can add your own attributes with the attribute(name, value) DSL method, such as: In the sample below we add an attribute user that is an object we already have instantiated as myUser. This object has a getFirstName() method that we want to set as header on the message. We use the groovy language to concat the first and last name into a single string that is returned. from("direct:in").setHeader("name").groovy("'$user.firstName $user.lastName'").attribute("user", myUser).to("seda:users"); Any scripting language Camel can run any JSR-223 scripting languages using the script DSL method such as: from("direct:in").setHeader("firstName").script("jaskel", "user.firstName").attribute("user", myUser).to("seda:users"); This is a bit different using the Spring DSL where you use the expression element that doesn't support setting attributes (yet): user.firstName Dependencies To use scripting languages in your camel routes you need to add the a dependency on camelscript which integrates the JSR-223 scripting engine. If you use maven you could just add the following to your pom.xml, substituting the version number for the latest & greatest release (see the download page for the latest versions). LAN GUAGES SUPPO R T ED APPEN DIX 216 org.apache.camel camel-script 1.4.0 RUBY Camel supports Ruby among other Scripting Languages to allow an Expression or Predicate to be used in the DSL or Xml Configuration. To use a Ruby expression use the following Java code ... ruby("someRubyExpression") ... For example you could use the ruby function to create an Predicate in a Message Filter or as an Expression for a Recipient List Example In the sample below we use Ruby to create a Predicate use in the route path, to route exchanges from admin users to a special queue. from("direct:start") .choice() .when().ruby("$request.headers['user'] == 'admin'").to("seda:adminQueue") .otherwise() .to("seda:regularQueue"); And a Spring DSL sample as well: $request.headers['user'] == 'admin' 217 L A N G U A G E S S U P P OR T E D A P P E ND I X ScriptContext The JSR-223 scripting languages ScriptContext is pre configured with the following attributes all set at ENGINE_SCOPE: Attribute Type Value context org.apache.camel.CamelContext The Camel Context exchange org.apache.camel.Exchange The current Exchange request org.apache.camel.Message The IN message response org.apache.camel.Message The OUT message Attributes You can add your own attributes with the attribute(name, value) DSL method, such as: In the sample below we add an attribute user that is an object we already have instantiated as myUser. This object has a getFirstName() method that we want to set as header on the message. We use the groovy language to concat the first and last name into a single string that is returned. from("direct:in").setHeader("name").groovy("'$user.firstName $user.lastName'").attribute("user", myUser).to("seda:users"); Any scripting language Camel can run any JSR-223 scripting languages using the script DSL method such as: from("direct:in").setHeader("firstName").script("jaskel", "user.firstName").attribute("user", myUser).to("seda:users"); This is a bit different using the Spring DSL where you use the expression element that doesn't support setting attributes (yet): user.firstName LAN GUAGES SUPPO R T ED APPEN DIX 218 Dependencies To use scripting languages in your camel routes you need to add the a dependency on camelscript which integrates the JSR-223 scripting engine. If you use maven you could just add the following to your pom.xml, substituting the version number for the latest & greatest release (see the download page for the latest versions). org.apache.camel camel-script 1.4.0 SIMPLE EXPRESSION LANGUAGE The Simple Expression Language was a really simple language you can use, but has since grown more powerful. Its primarily intended for being a really small and simple language for evaluating Expression and Predicate without requiring any new dependencies or knowledge of XPath; so its ideal for testing in camel-core. Its ideal to cover 95% of the common use cases when you need a little bit of expression based script in your Camel routes. However for much more complex use cases you are generally recommended to choose a more expressive and powerful language such as: • JavaScript • EL • OGNL • Mvel • Groovy • one of the supported Scripting Languages The simple language uses ${body} placeholders for complex expressions where the expression contains constant literals. The ${ } placeholders can be omitted if the expression is only the token itself. To get the body of the in message: "body", or "in.body" or "${body}". A complex expression must use ${ } placeholders, such as: "Hello ${in.header.name} how are you?". You can have multiple tokens in the same expression: "Hello ${in.header.name} this is ${in.header.me} speaking". However you can not nest tokens (i.e. having another ${ } placeholder in an existing, is not allowed). Variables Variable 219 Type Description L A N G U A G E S S U P P OR T E D A P P E ND I X File language is now merged with Simple language From Camel 2.2 onwards, the File Language is now merged with Simple language which means you can use all the file syntax directly within the simple language. exchangeId String Camel 2.3: the exchange id id String the input message id body Object the input body in.body Object the input body body.OGNL Object Camel 2.3: the input body invoked using a Camel OGNL expression. in.body.OGNL Object Camel 2.3: the input body invoked using a Camel OGNL expression. bodyAs(type) Type Camel 2.3: Converts the body to the given type determined by its classname out.body Object the output body header.foo Object refer to the input foo header headers.foo Object refer to the input foo header in.header.foo Object refer to the input foo header in.headers.foo Object refer to the input foo header header.foo[bar] Object Camel 2.3: regard input foo header as a map and perform lookup on the map with bar as key in.header.foo[bar] Object Camel 2.3: regard input foo header as a map and perform lookup on the map with bar as key in.headers.foo[bar] Object Camel 2.3: regard input foo header as a map and perform lookup on the map with bar as key header.foo.OGNL Object Camel 2.3: refer to the input foo header and invoke its value using a Camel OGNL expression. in.header.foo.OGNL Object Camel 2.3: refer to the input foo header and invoke its value using a Camel OGNL expression. in.headers.foo.OGNL Object Camel 2.3: refer to the input foo header and invoke its value using a Camel OGNL expression. out.header.foo Object refer to the out header foo LAN GUAGES SUPPO R T ED APPEN DIX 220 out.headers.foo Object refer to the out header foo property.foo Object refer to the foo property on the exchange sys.foo String refer to the system property sysenv.foo String Camel 2.3: refer to the system environment exception Object Camel 2.4: Refer to the exception object on the exchange, is null if no exception set on exchange. Will fallback and grab caught exceptions (Exchange.EXCEPTION_CAUGHT) if the Exchange has any. exception.OGNL Object Camel 2.4: Refer to the exchange exception invoked using a Camel OGNLE expression object String Camel 2.0. Refer to the exception.message on the exchange, is null if no exception set on exchange. Will fallback and grab caught exceptions (Exchange.EXCEPTION_CAUGHT) if the Exchange has any. String Camel 1.5. Date formatting using the java.text.SimpleDataFormat patterns. Supported commands are: now for current timestamp, in.header.xxx or header.xxx to use the Date object in the IN header with the key xxx. out.header.xxx to use the Date object in the OUT header with the key xxx. Object Camel 1.5. Invoking a bean expression using the Bean language. Specifying a method name you must use dot as separator. In Camel 2.0 we also support the ?method=methodname syntax that is used by the Bean component. properties:locations:key String Camel 2.3: Lookup a property with the given key. The locations option is optional. See more at Using PropertyPlaceholder. threadName String Camel 2.3: Returns the name of the current thread. Can be used for logging purpose. exception.message date:command:pattern bean:bean expression OGNL support Available as of Camel 2.3 221 L A N G U A G E S S U P P OR T E D A P P E ND I X The Simple and Bean language now supports a Camel OGNL notation for invoking beans in a chain like fashion. Suppose the Message IN body contains a POJO which has a getAddress() method. Then you can use Camel OGNL notation to access the address object: simple("${body.address}") simple("${body.address.street}") simple("${body.address.zip}") Camel understands the shorthand names for getters, but you can invoke any method or use the real name such as: simple("${body.address}") simple("${body.getAddress.getStreet}") simple("${body.address.getZip}") simple("${body.doSomething}") You can also use the null safe operator (?.) to avoid NPE if for example the body does NOT have an address simple("${body?.address?.street}") Its also possible to index in Map or List types, so you can do: simple("${body[foo].name}") To assume the body is Map based and lookup the value with foo as key, and invoke the getName method on that value. Suppose there was no value with the key foo then you can use the elvis operator to avoid the NPE as shown: simple("${body[foo]?.name}") You can also access List types, for example to get lines from the address you can do: simple("${body.address.lines[0]}") simple("${body.address.lines[1]}") simple("${body.address.lines[2]}") There is a special last keyword which can be used to get the last value from a list. simple("${body.address.lines[last]}") LAN GUAGES SUPPO R T ED APPEN DIX 222 And to get the 2nd last you can subtract a number, so we can use last-1 to indicate this: simple("${body.address.lines[last-1]}") And the 3rd last is of course: simple("${body.address.lines[last-2]}") And yes you can combine this with the operator support as shown below: simple("${body.address.zip} > 1000") Operator support Available as of Camel 2.0 We added a basic set of operators supported in the simple language in Camel 2.0. The parser is limited to only support a single operator. To enable it the left value must be enclosed in ${ }. The syntax is: ${leftValue} OP rightValue Where the rightValue can be a String literal enclosed in ' ', null, a constant value or another expression enclosed in ${ }. Camel will automatically type convert the rightValue type to the leftValue type, so its able to eg. convert a string into a numeric so you can use > comparison for numeric values. The following operators is supported: 223 Operator Description == equals > greater than >= greater than or equals < less than <= less than or equals != not equals contains For testing if contains in a string based value not contains For testing if not contains in a string based value L A N G U A G E S S U P P OR T E D A P P E ND I X regex For matching against a given regular expression pattern defined as a String value not regex For not matching against a given regular expression pattern defined as a String value in For matching if in a set of values, each element must be separated by comma. not in For matching if not in a set of values, each element must be separated by comma. is For matching if the left hand side type is an instanceof the value. not is For matching if the left hand side type is not an instanceof the value. range For matching if the left hand side is within a range of values defined as numbers: from..to not range For matching if the left hand side is not within a range of values defined as numbers: from..to And the following operators can be used to group expressions: Operator Description and and is used to group two expressions or or is used to group two expressions Notice: Currently and or or can only be used once in a simple language expression. This might change in the future. The syntax for AND is: ${leftValue} OP rightValue and ${leftValue} OP rightValue And the syntax for OR is: ${leftValue} OP rightValue or ${leftValue} OP rightValue Some examples: simple("${in.header.foo} == 'foo'") // ' ' can be omitted simple("${in.header.foo} == foo") // here Camel will type convert '100' into the type of in.header.bar and if its an Integer '100' will also be converter to an Integer simple("${in.header.bar} == '100'") LAN GUAGES SUPPO R T ED APPEN DIX 224 simple("${in.header.bar} == 100") // 100 will be converter to the type of in.header.bar so we can do > comparison simple("${in.header.bar} > 100") // testing for null simple("${in.header.baz} == null") // testing for not null simple("${in.header.baz} != null") And a bit more advanced example where the right value is another expression simple("${in.header.date} == ${date:now:yyyyMMdd}") simple("${in.header.type} == ${bean:orderService?method=getOrderType}") And an example with contains, testing if the title contains the word Camel simple("${in.header.title} contains 'Camel'") And an example with regex, testing if the number header is a 4 digit value: simple("${in.header.number} regex '\d{4}'") And finally an example if the header equals any of the values in the list. Each element must be separated by comma, and no space around. This also works for numbers etc, as Camel will convert each element into the type of the left hand side. simple("${in.header.type} in 'gold,silver'") And for all the last 3 we also support the negate test using not: simple("${in.header.type} not in 'gold,silver'") And you can test for if the type is a certain instance, eg for instance a String simple("${in.header.type} is 'java.lang.String'") We have added a shorthand for all java.lang types so you can write it as: simple("${in.header.type} is String") 225 L A N G U A G E S S U P P OR T E D A P P E ND I X Ranges is also supported. The range interval requires numbers and both from and end is inclusive. For instance to test whether a value is between 100 and 199: simple("${in.header.number} range 100..199") Notice we use .. in the range without spaces. Its based on the same syntax as Groovy. Using and / or If you have two expressions you can combine them with the and or or operator. For instance: simple("${in.header.title} contains 'Camel' and ${in.header.type'} == 'gold'") And of course the or is also supported. The sample example would be: simple("${in.header.title} contains 'Camel' or ${in.header.type'} == 'gold'") Notice: Currently and or or can only be used once in a simple language expression. This might change in the future. So you cannot do: simple("${in.header.title} contains 'Camel' and ${in.header.type'} == 'gold' and ${in.header.number} range 100..200") Samples In the Spring XML sample below we filter based on a header value: in.header.foo The Simple language can be used for the predicate test above in the Message Filter pattern, where we test if the in message has a foo header (a header with the key foo exists). If the expression evaluates to true then the message is routed to the mock:foo endpoint, otherwise its lost in the deep blue sea The same example in Java DSL: . LAN GUAGES SUPPO R T ED APPEN DIX 226 Can be used in Spring XML As the Spring XML does not have all the power as the Java DSL with all its various builder methods, you had to resort to use some other languages for testing with simple operators. Now you can do this with the simple language. In the sample below we want to test if the header is a widget order: ${in.header.type} == 'widget' from("seda:orders") .filter().simple("in.header.foo").to("seda:fooOrders"); You can also use the simple language for simple text concatenations such as: from("direct:hello").transform().simple("Hello ${in.header.user} how are you?").to("mock:reply"); Notice that we must use ${ } placeholders in the expression now to let Camel be able to parse it correctly. And this sample uses the date command to output current date. from("direct:hello").transform().simple("The today is ${date:now:yyyyMMdd} and its a great day.").to("mock:reply"); And in the sample below we invoke the bean language to invoke a method on a bean to be included in the returned string: from("direct:order").transform().simple("OrderId: ${bean:orderIdGenerator}").to("mock:reply"); Where orderIdGenerator is the id of the bean registered in the Registry. If using Spring then its the Spring bean id. If we want to declare which method to invoke on the order id generator bean we must prepend .method name such as below where we invoke the generateId method. 227 L A N G U A G E S S U P P OR T E D A P P E ND I X from("direct:order").transform().simple("OrderId: ${bean:orderIdGenerator.generateId}").to("mock:reply"); And in Camel 2.0 we can use the ?method=methodname option that we are familiar with the Bean component itself: from("direct:order").transform().simple("OrderId: ${bean:orderIdGenerator?method=generateId}").to("mock:reply"); And from Camel 2.3 onwards you can also convert the body to a given type, for example to ensure its a String you can do: Hello ${bodyAs(String)} how are you? There is a few types which have a shorthand notation, hence why we can use String instead of java.lang.String. These are: byte[], String, Integer, Long. All other types must use their FQN name, e.g. org.w3c.dom.Document. Its also possible to lookup a value from a header Map in Camel 2.3 onwards: The gold value is ${header.type[gold]} In the code above we lookup the header with name type and regard it as a java.util.Map and we then lookup with the key gold and return the value. If the header is not convertible to Map an exception is thrown. If the header with name type does not exists null is returned. Dependencies The Bean language is part of camel-core. FILE EXPRESSION LANGUAGE Available as of Camel 1.5 The File Expression Language is an extension to the Simple language, adding file related capabilities. These capabilities is related to common use cases working with file path and names. The goal is to allow expression to be used with the File and FTP components for setting dynamic file patterns for both consumer and producer. LAN GUAGES SUPPO R T ED APPEN DIX 228 File language is now merged with Simple language From Camel 2.2 onwards, the file language is now merged with Simple language which means you can use all the file syntax directly within the simple language. Syntax This language is an extension to the Simple language so the Simple syntax applies also. So the table below only lists the additional. As opposed to Simple language File Language also supports Constant expressions so you can enter a fixed filename. All the file tokens uses the same expression name as the method on the java.io.File object, for instance file:absolute refers to the java.io.File.getAbsolute() method. Notice that not all expressions is supported by the current Exchange. For instance the FTP component supports some of the options, where as the File component support all of them. Expression Type File Consumer File Producer FTP Consumer FTP Producer Description file:name String yes no yes no refers to the file name (is relative to the starting directory, see note below) file:name.ext String yes no yes no Camel 2.3: refers to the file extension only file:name.noext String yes no yes no refers to the file name with no extension (is relative to the starting directory, see note below) file:onlyname String yes no yes no Camel 2.0: refers to the file name only with no leading paths. file:onlyname.noext String yes no yes no Camel 2.0: refers to the file name only with no extension and with no leading paths. file:ext String yes no yes no Camel 1.6.1/Camel 2.0: refers to the file extension only file:parent String yes no yes no refers to the file parent file:path String yes no yes no refers to the file path file:absolute Boolean yes no no no Camel 2.0: refers to whether the file is regarded as absolute or relative file:absolute.path String yes no no no refers to the absolute file path file:length Long yes no yes no refers to the file length returned as a Long type file:modified Date yes no yes no Camel 2.0: refers to the file last modified returned as a Date type yes for date formatting using the java.text.SimepleDataFormat patterns. Is an extension to the Simple language. Additional command is: file (consumers only) for the last modified timestamp of the file. Notice: all the commands from the Simple language can also be used. date:command:pattern 229 String yes yes L A N G U A G E S S U P P OR T E D A P P E ND I X yes File token example Relative paths We have a java.io.File handle for the file hello.txt in the following relative directory: .\filelanguage\test. And we configure out endpoint to use this starting directory .\filelanguage. The the file tokens will return as: Expression Returns file:name test\hello.txt file:name.ext txt file:name.noext test\hello file:onlyname hello.txt file:onlyname.noext hello file:ext txt file:parent filelanguage\test file:path filelanguage\test\hello.txt file:absolute false file:absolute.path \workspace\camel\camel-core\target\filelanguage\test\hello.txt Absolute paths We have a java.io.File handle for the file hello.txt in the following absolute directory: \workspace\camel\camel-core\target\filelanguage\test. And we configure out endpoint to use the absolute starting directory \workspace\camel\camel-core\target\filelanguage. The the file tokens will return as: Expression Returns file:name test\hello.txt file:name.ext txt file:name.noext test\hello file:onlyname hello.txt file:onlyname.noext hello file:ext txt LAN GUAGES SUPPO R T ED APPEN DIX 230 file:parent \workspace\camel\camel-core\target\filelanguage\test file:path \workspace\camel\camel-core\target\filelanguage\test\hello.txt file:absolute true file:absolute.path \workspace\camel\camel-core\target\filelanguage\test\hello.txt Samples You can enter a fixed Constant expression such as myfile.txt: fileName="myfile.txt" Lets assume we use the file consumer to read files and want to move the read files to backup folder with the current date as a sub folder. This can be archived using an expression like: fileName="backup/${date:now:yyyyMMdd}/${file:name.noext}.bak" relative folder names is also supported so suppose the backup folder should be a sibling folder then you can append .. as: fileName="../backup/${date:now:yyyyMMdd}/${file:name.noext}.bak" As this is an extension to the Simple language we have access to all the goodies from this language also, so in this use case we want to use the in.header.type as a parameter in the dynamic expression: fileName="../backup/${date:now:yyyyMMdd}/type-${in.header.type}/ backup-of-${file:name.noext}.bak" If you have a custom Date you want to use in the expression then Camel supports retrieving dates from the message header. fileName="orders/ order-${in.header.customerId}-${date:in.header.orderDate:yyyyMMdd}.xml" And finally we can also use a bean expression to invoke a POJO class that generates some String output (or convertible to String) to be used: fileName="uniquefile-${bean:myguidgenerator.generateid}.txt" 231 L A N G U A G E S S U P P OR T E D A P P E ND I X And of course all this can be combined in one expression where you can use the File Language, Simple and the Bean language in one combined expression. This is pretty powerful for those common file path patterns. Dependencies The File language is part of camel-core. SQL The SQL support is added by JoSQL and is primarily used for performing SQL queries on inmemory objects. If you prefer to perform actual database queries then check out the JPA component. Camel supports SQL to allow an Expression or Predicate to be used in the DSL or Xml Configuration. For example you could use SQL to create an Predicate in a Message Filter or as an Expression for a Recipient List. from("queue:foo").setBody().sql("select * from MyType").to("queue:bar") And the spring DSL: select * from MyType Variables Variable Type Description exchange Exchange the Exchange object in Message the exchange.in message out Message the exchange.out message the property key Object the Exchange properties the header key Object the exchange.in headers the variable key Object if any additional variables is added using setVariables method LAN GUAGES SUPPO R T ED APPEN DIX 232 Can be used in Spring XML In Camel 2.2 you can use the File Language directly from the Simple language which makes a Content Based Router more easy to do in Spring XML, where we can route based on file extensions as shown below: ${file:ext} == 'txt' ${file:ext} == 'xml' Dependencies To use SQL in your camel routes you need to add the a dependency on camel-josql which implements the SQL language. If you use maven you could just add the following to your pom.xml, substituting the version number for the latest & greatest release (see the download page for the latest versions). org.apache.camel camel-josql 1.4.0 XPATH Camel supports XPath to allow an Expression or Predicate to be used in the DSL or Xml Configuration. For example you could use XPath to create an Predicate in a Message Filter or as an Expression for a Recipient List. from("queue:foo"). filter().xpath("//foo")). to("queue:bar") 233 L A N G U A G E S S U P P OR T E D A P P E ND I X from("queue:foo"). choice().xpath("//foo")).to("queue:bar"). otherwise().to("queue:others"); Namespaces In 1.3 onwards you can easily use namespaces with XPath expressions using the Namespaces helper class. Namespaces ns = new Namespaces("c", "http://acme.com/cheese"); from("direct:start").filter(). xpath("/c:person[@name='James']", ns). to("mock:result"); Variables Variables in XPath is defined in different namespaces. The default namespace is http://camel.apache.org/schema/spring. Namespace URI Local part Type Description http://camel.apache.org/xml/in/ in Message the exchange.in message http://camel.apache.org/xml/out/ out Message the exchange.out message http://camel.apache.org/xml/variables/ environment-variables env Object OS environment variables http://camel.apache.org/xml/variables/systemproperties system Object Java System properties Object the exchange property http://camel.apache.org/xml/variables/ exchange-property Camel will resolve variables according to either: ▪ namespace given ▪ no namespace given LAN GUAGES SUPPO R T ED APPEN DIX 234 Namespace given If the namespace is given then Camel is instructed exactly what to return. However when resolving either in or out Camel will try to resolve a header with the given local part first, and return it. If the local part has the value body then the body is returned instead. No namespace given If there is no namespace given then Camel resolves only based on the local part. Camel will try to resolve a variable in the following steps: ▪ from variables that has been set using the variable(name, value) fluent builder ▪ from message.in.header if there is a header with the given key ▪ from exchange.properties if there is a property with the given key Functions Camel adds the following XPath functions that can be used to access the exchange: Function Argument Type Description in:body none Object Will return the in message body. in:header the header name Object Will return the in message header. out:body none Object Will return the out message body. out:header the header name Object Will return the out message header. Here's an example showing some of these functions in use. from("direct:start").choice() .when().xpath("in:header('foo') = 'bar'").to("mock:x") .when().xpath("in:body() = ''").to("mock:y") .otherwise().to("mock:z"); Using XML configuration If you prefer to configure your routes in your Spring XML file then you can use XPath expressions as follows /foo:person[@name='James'] Notice how we can reuse the namespace prefixes, foo in this case, in the XPath expression for easier namespace based XPath expressions! See also this discussion on the mailinglist about using your own namespaces with xpath Setting result type The XPath expression will return a result type using native XML objects such as org.w3c.dom.NodeList. But many times you want a result type to be a String. To do this you have to instruct the XPath which result type to use. In Java DSL: xpath("/foo:person/@id", String.class) In Spring DSL you use the resultType attribute to provide a fully qualified classname: /foo:person/@id In @XPath: Available as of Camel 2.1 @XPath(value = "concat('foo-',//order/name/)", resultType = String.class) String name) Where we use the xpath function concat to prefix the order name with foo-. In this case we have to specify that we want a String as result type so the concat function works. Examples Here is a simple example using an XPath expression as a predicate in a Message Filter LAN GUAGES SUPPO R T ED APPEN DIX 236 from("direct:start"). filter().xpath("/person[@name='James']"). to("mock:result"); If you have a standard set of namespaces you wish to work with and wish to share them across many different XPath expressions you can use the NamespaceBuilder as shown in this example // lets define the namespaces we'll need in our filters Namespaces ns = new Namespaces("c", "http://acme.com/cheese") .add("xsd", "http://www.w3.org/2001/XMLSchema"); // now lets create an xpath based Message Filter from("direct:start"). filter(ns.xpath("/c:person[@name='James']")). to("mock:result"); In this sample we have a choice construct. The first choice evaulates if the message has a header key type that has the value Camel. The 2nd choice evaluates if the message body has a name tag which values is Kong. If neither is true the message is routed in the otherwise block: from("direct:in").choice() // using $headerName is special notation in Camel to get the header key .when().xpath("$type = 'Camel'") .to("mock:camel") // here we test for the body name tag .when().xpath("//name = 'Kong'") .to("mock:donkey") .otherwise() .to("mock:other") .end(); And the spring XML equivalent of the route: $type = 'Camel' //name = 'Kong' 237 L A N G U A G E S S U P P OR T E D A P P E ND I X XPATH INJECTION You can use Bean Integration to invoke a method on a bean and use various languages such as XPath to extract a value from the message and bind it to a method parameter. The default XPath annotation has SOAP and XML namespaces available. If you want to use your own namespace URIs in an XPath expression you can use your own copy of the XPath annotation to create whatever namespace prefixes you want to use. import import import import java.lang.annotation.ElementType; java.lang.annotation.Retention; java.lang.annotation.RetentionPolicy; java.lang.annotation.Target; import org.w3c.dom.NodeList; import org.apache.camel.component.bean.XPathAnnotationExpressionFactory; import org.apache.camel.language.LanguageAnnotation; import org.apache.camel.language.NamespacePrefix; @Retention(RetentionPolicy.RUNTIME) @Target({ElementType.FIELD, ElementType.METHOD, ElementType.PARAMETER}) @LanguageAnnotation(language = "xpath", factory = XPathAnnotationExpressionFactory.class) public @interface MyXPath { String value(); // You can add the namespaces as the default value of the annotation NamespacePrefix[] namespaces() default { @NamespacePrefix(prefix = "n1", uri = "http://example.org/ns1"), @NamespacePrefix(prefix = "n2", uri = "http://example.org/ns2")}; Class resultType() default NodeList.class; } i.e. cut and paste upper code to your own project in a different package and/or annotation name then add whatever namespace prefix/uris you want in scope when you use your annotation on a method parameter. Then when you use your annotation on a method parameter all the namespaces you want will be available for use in your XPath expression. NOTE this feature is supported from Camel 1.6.1. For example LAN GUAGES SUPPO R T ED APPEN DIX 238 public class Foo { @MessageDriven(uri = "activemq:my.queue") public void doSomething(@Path("/ns1:foo/ns2:bar/text()") String correlationID, @Body String body) { // process the inbound message here } } Using XPathBuilder without an Exchange Available as of Camel 2.3 You can now use the org.apache.camel.builder.XPathBuilder without the need for an Exchange. This comes handy if you want to use it as a helper to do custom xpath evaluations. It requires that you pass in a CamelContext since a lot of the moving parts inside the XPathBuilder requires access to the Camel Type Converter and hence why CamelContext is needed. For example you can do something like this: boolean matches = XPathBuilder.xpath("/foo/bar/@xyz").matches(context, "")); This will match the given predicate. You can also evaluate for example as shown in the following three examples: String name = XPathBuilder.xpath("foo/bar").evaluate(context, "cheese", String.class); Integer number = XPathBuilder.xpath("foo/bar").evaluate(context, "123", Integer.class); Boolean bool = XPathBuilder.xpath("foo/bar").evaluate(context, "true", Boolean.class); Evaluating with a String result is a common requirement and thus you can do it a bit simpler: String name = XPathBuilder.xpath("foo/bar").evaluate(context, "cheese"); Using Saxon with XPathBuilder Available as of Camel 2.3 You need to add camel-saxon as dependency to your project. 239 L A N G U A G E S S U P P OR T E D A P P E ND I X Its now easier to use Saxon with the XPathBuilder which can be done in several ways as shown below. Where as the latter ones are the easiest ones. Using a factory // create a Saxon factory XPathFactory fac = new net.sf.saxon.xpath.XPathFactoryImpl(); // create a builder to evaluate the xpath using the saxon factory XPathBuilder builder = XPathBuilder.xpath("tokenize(/foo/bar, '_')[2]").factory(fac); // evaluate as a String result String result = builder.evaluate(context, "abc_def_ghi"); assertEquals("def", result); Using ObjectModel // create a builder to evaluate the xpath using saxon based on its object model uri XPathBuilder builder = XPathBuilder.xpath("tokenize(/foo/bar, '_')[2]").objectModel("http://saxon.sf.net/jaxp/xpath/om"); // evaluate as a String result String result = builder.evaluate(context, "abc_def_ghi"); assertEquals("def", result); The easy one // create a builder to evaluate the xpath using saxon XPathBuilder builder = XPathBuilder.xpath("tokenize(/foo/bar, '_')[2]").saxon(); // evaluate as a String result String result = builder.evaluate(context, "abc_def_ghi"); assertEquals("def", result); Setting a custom XPathFactory using System Property Available as of Camel 2.3 Camel now supports reading the JVM system property javax.xml.xpath.XPathFactory that can be used to set a custom XPathFactory to use. This unit test shows how this can be done to use Saxon instead: // set system property with the XPath factory to use which is Saxon System.setProperty(XPathFactory.DEFAULT_PROPERTY_NAME + ":" + "http://saxon.sf.net/ jaxp/xpath/om", "net.sf.saxon.xpath.XPathFactoryImpl"); LAN GUAGES SUPPO R T ED APPEN DIX 240 // create a builder to evaluate the xpath using saxon XPathBuilder builder = XPathBuilder.xpath("tokenize(/foo/bar, '_')[2]"); // evaluate as a String result String result = builder.evaluate(context, "abc_def_ghi"); assertEquals("def", result); Camel will log at INFO level if it uses a non default XPathFactory such as: XPathBuilder INFO Using system property javax.xml.xpath.XPathFactory:http://saxon.sf.net/jaxp/xpath/om with value: net.sf.saxon.xpath.XPathFactoryImpl when creating XPathFactory Dependencies The XPath language is part of camel-core. XQUERY Camel supports XQuery to allow an Expression or Predicate to be used in the DSL or Xml Configuration. For example you could use XQuery to create an Predicate in a Message Filter or as an Expression for a Recipient List. from("queue:foo").filter(). xquery("//foo")). to("queue:bar") You can also use functions inside your query, in which case you need an explicit type conversion (or you will get a org.w3c.dom.DOMException: HIERARCHY_REQUEST_ERR) by passing the Class as a second argument to the xquery() method. from("direct:start"). recipientList().xquery("concat('mock:foo.', /person/@city)", String.class); Variables The IN message body will be set as the contextItem. Besides this these Variables is also added as parameters: 241 Variable Type Description exchange Exchange The current Exchange L A N G U A G E S S U P P OR T E D A P P E ND I X Support version in.body Object The In message's body >= 1.6.1 out.body Object The OUT message's body (if any) >= 1.6.1 in.headers.* Object You can access the value of exchange.in.headers with key foo by using the variable which name is in.headers.foo >=1.6.1 out.headers.* Object You can access the value of exchange.out.headers with key foo by using the variable which name is out.headers.foo variable >=1.6.1 Object Any exchange.properties and exchange.in.headers (exchange.in.headers support was removed since camel 1.6.1) and any additional parameters set using setParameters(Map). These parameters is added with they own key name, for instance if there is an IN header with the key name foo then its added as foo. key name Using XML configuration If you prefer to configure your routes in your Spring XML file then you can use XPath expressions as follows /foo:person[@name='James'] Notice how we can reuse the namespace prefixes, foo in this case, in the XPath expression for easier namespace based XQuery expressions! When you use functions in your XQuery expression you need an explicit type conversion which is done in the xml configuration via the @type attribute: LAN GUAGES SUPPO R T ED APPEN DIX 242 concat('mock:foo.', /person/@city) Using XQuery as an endpoint Sometimes an XQuery expression can be quite large; it can essentally be used for Templating. So you may want to use an XQuery Endpoint so you can route using XQuery templates. The following example shows how to take a message of an ActiveMQ queue (MyQueue) and transform it using XQuery and send it to MQSeries. Examples Here is a simple example using an XQuery expression as a predicate in a Message Filter from("direct:start").filter().xquery("/person[@name='James']").to("mock:result"); This example uses XQuery with namespaces as a predicate in a Message Filter Namespaces ns = new Namespaces("c", "http://acme.com/cheese"); from("direct:start"). filter().xquery("/c:person[@name='James']", ns). to("mock:result"); Learning XQuery XQuery is a very powerful language for querying, searching, sorting and returning XML. For help learning XQuery try these tutorials • Mike Kay's XQuery Primer • the W3Schools XQuery Tutorial You might also find the XQuery function reference useful 243 L A N G U A G E S S U P P OR T E D A P P E ND I X Dependencies To use XQuery in your camel routes you need to add the a dependency on camel-saxon which implements the XQuery language. If you use maven you could just add the following to your pom.xml, substituting the version number for the latest & greatest release (see the download page for the latest versions). org.apache.camel camel-saxon 1.4.0 LAN GUAGES SUPPO R T ED APPEN DIX 244 CHAPTER 9 °°°° Pattern Appendix There now follows a breakdown of the various Enterprise Integration Patterns that Camel supports MESSAGING SYSTEMS Message Channel Camel supports the Message Channel from the EIP patterns. The Message Channel is an internal implementation detail of the Endpoint interface and all interactions with the Message Channel are via the Endpoint interfaces. For more details see • Message • Message Endpoint Using This Pattern If you would like to use this EIP Pattern then please read the Getting Started, you may also find the Architecture useful particularly the description of Endpoint and URIs. Then you could try out some of the Examples first before trying this pattern out. Message Camel supports the Message from the EIP patterns using the Message interface. 245 CH A PTE R 9 - PAT T E R N A P P E ND I X To support various message exchange patterns like one way Event Message and Request Reply messages Camel uses an Exchange interface which has a pattern property which can be set to InOnly for an Event Message which has a single inbound Message, or InOut for a Request Reply where there is an inbound and outbound message. Using This Pattern If you would like to use this EIP Pattern then please read the Getting Started, you may also find the Architecture useful particularly the description of Endpoint and URIs. Then you could try out some of the Examples first before trying this pattern out. Pipes and Filters Camel supports the Pipes and Filters from the EIP patterns in various ways. With Camel you can split your processing across multiple independent Endpoint instances which can then be chained together. Using Routing Logic You can create pipelines of logic using multiple Endpoint or Message Translator instances as follows from("direct:a").pipeline("direct:x", "direct:y", "direct:z", "mock:result"); Though pipeline is the default mode of operation when you specify multiple outputs in Camel. The opposite to pipeline is multicast; which fires the same message into each of its outputs. (See the example below). In Spring XML you can use the element as of 1.4.0 onwards CH APT ER 9 - PAT T ER N APPEN DIX 246 In the above the pipeline element is actually unnecessary, you could use this... Its just a bit more explicit. However if you wish to use to avoid a pipeline - to send the same message into multiple pipelines - then the element comes into its own. In the above example we are routing from a single Endpoint to a list of different endpoints specified using URIs. If you find the above a bit confusing, try reading about the Architecture or try the Examples Using This Pattern If you would like to use this EIP Pattern then please read the Getting Started, you may also find the Architecture useful particularly the description of Endpoint and URIs. Then you could try out some of the Examples first before trying this pattern out. Message Router The Message Router from the EIP patterns allows you to consume from an input destination, evaluate some predicate then choose the right output destination. 247 CH A PTE R 9 - PAT T E R N A P P E ND I X The following example shows how to route a request from an input queue:a endpoint to either queue:b, queue:c or queue:d depending on the evaluation of various Predicate expressions Using the Fluent Builders RouteBuilder builder = new RouteBuilder() { public void configure() { errorHandler(deadLetterChannel("mock:error")); from("seda:a").choice().when(header("foo").isEqualTo("bar")).to("seda:b") .when(header("foo").isEqualTo("cheese")).to("seda:c").otherwise().to("seda:d"); } }; Using the Spring XML Extensions $foo = 'bar' $foo = 'cheese' Choice without otherwise If you use a choice without adding an otherwise, any unmatched exchanges will be dropped by default. If you prefer to have an exception for an unmatched exchange, you can add a throwFault to the otherwise. CH APT ER 9 - PAT T ER N APPEN DIX 248 ....otherwise().throwFault("No matching when clause found on choice block"); Using This Pattern If you would like to use this EIP Pattern then please read the Getting Started, you may also find the Architecture useful particularly the description of Endpoint and URIs. Then you could try out some of the Examples first before trying this pattern out. Message Translator Camel supports the Message Translator from the EIP patterns by using an arbitrary Processor in the routing logic, by using a bean to perform the transformation, or by using transform() in the DSL. You can also use a Data Format to marshal and unmarshal messages in different encodings. Using the Fluent Builders You can transform a message using Camel's Bean Integration to call any method on a bean in your Registry such as your Spring XML configuration file as follows from("activemq:SomeQueue"). beanRef("myTransformerBean", "myMethodName"). to("mqseries:AnotherQueue"); Where the "myTransformerBean" would be defined in a Spring XML file or defined in JNDI etc. You can omit the method name parameter from beanRef() and the Bean Integration will try to deduce the method to invoke from the message exchange. or you can add your own explicit Processor to do the transformation from("direct:start").process(new Processor() { public void process(Exchange exchange) { Message in = exchange.getIn(); in.setBody(in.getBody(String.class) + " World!"); } }).to("mock:result"); or you can use the DSL to explicitly configure the transformation 249 CH A PTE R 9 - PAT T E R N A P P E ND I X from("direct:start").transform(body().append(" World!")).to("mock:result"); Use Spring XML You can also use Spring XML Extensions to do a transformation. Basically any Expression language can be substituted inside the transform element as shown below ${in.body} extra data! Or you can use the Bean Integration to invoke a bean You can also use Templating to consume a message from one destination, transform it with something like Velocity or XQuery and then send it on to another destination. For example using InOnly (one way messaging) from("activemq:My.Queue"). to("velocity:com/acme/MyResponse.vm"). to("activemq:Another.Queue"); If you want to use InOut (request-reply) semantics to process requests on the My.Queue queue on ActiveMQ with a template generated response, then sending responses back to the JMSReplyTo Destination you could use this. from("activemq:My.Queue"). to("velocity:com/acme/MyResponse.vm"); For further examples of this pattern in use you could look at one of the JUnit tests • TransformTest • TransformViaDSLTest • TransformProcessorTest • TransformWithExpressionTest (test resource) CH APT ER 9 - PAT T ER N APPEN DIX 250 Using This Pattern If you would like to use this EIP Pattern then please read the Getting Started, you may also find the Architecture useful particularly the description of Endpoint and URIs. Then you could try out some of the Examples first before trying this pattern out. Message Endpoint Camel supports the Message Endpoint from the EIP patterns using the Endpoint interface. When using the DSL to create Routes you typically refer to Message Endpoints by their URIs rather than directly using the Endpoint interface. Its then a responsibility of the CamelContext to create and activate the necessary Endpoint instances using the available Component implementations. For more details see • Message Using This Pattern If you would like to use this EIP Pattern then please read the Getting Started, you may also find the Architecture useful particularly the description of Endpoint and URIs. Then you could try out some of the Examples first before trying this pattern out. MESSAGING CHANNELS Point to Point Channel Camel supports the Point to Point Channel from the EIP patterns using the following components • SEDA for in-VM seda based messaging • JMS for working with JMS Queues for high performance, clustering and load balancing • JPA for using a database as a simple message queue • XMPP for point-to-point communication over XMPP (Jabber) • and others 251 CH A PTE R 9 - PAT T E R N A P P E ND I X Using This Pattern If you would like to use this EIP Pattern then please read the Getting Started, you may also find the Architecture useful particularly the description of Endpoint and URIs. Then you could try out some of the Examples first before trying this pattern out. Publish Subscribe Channel Camel supports the Publish Subscribe Channel from the EIP patterns using the following components • JMS for working with JMS Topics for high performance, clustering and load balancing • XMPP when using rooms for group communication Using Routing Logic Another option is to explicitly list the publish-subscribe relationship in your routing logic; this keeps the producer and consumer decoupled but lets you control the fine grained routing configuration using the DSL or Xml Configuration. Using the Fluent Builders RouteBuilder builder = new RouteBuilder() { public void configure() { errorHandler(deadLetterChannel("mock:error")); from("seda:a").multicast().to("seda:b", "seda:c", "seda:d"); CH APT ER 9 - PAT T ER N APPEN DIX 252 } }; Using the Spring XML Extensions Using This Pattern If you would like to use this EIP Pattern then please read the Getting Started, you may also find the Architecture useful particularly the description of Endpoint and URIs. Then you could try out some of the Examples first before trying this pattern out. DEAD LETTER CHANNEL Camel supports the Dead Letter Channel from the EIP patterns using the DeadLetterChannel processor which is an Error Handler. Redelivery It is common for a temporary outage or database deadlock to cause a message to fail to process; but the chances are if its tried a few more times with some time delay then it will 253 CH A PTE R 9 - PAT T E R N A P P E ND I X Difference between Dead Letter Channel and Default Error Handler The major difference is that Dead Letter Channel has a dead letter queue that whenever an Exchange could not be processed is moved to. It will always moved failed exchanges to this queue. Unlike the Default Error Handler that does not have a dead letter queue. So whenever an Exchange could not be processed the error is propagated back to the client. Notice: You can adjust this behavior of whether the client should be notified or not with the handled option. complete fine. So we typically wish to use some kind of redelivery policy to decide how many times to try redeliver a message and how long to wait before redelivery attempts. The RedeliveryPolicy defines how the message is to be redelivered. You can customize things like • how many times a message is attempted to be redelivered before it is considered a failure and sent to the dead letter channel • the initial redelivery timeout • whether or not exponential backoff is used (i.e. the time between retries increases using a backoff multiplier) • whether to use collision avoidance to add some randomness to the timings • delay pattern a new option in Camel 2.0, see below for details. Once all attempts at redelivering the message fails then the message is forwarded to the dead letter queue. About moving Exchange to dead letter queue and using handled Handled on Dead Letter Channel was introduced in Camel 2.0, this feature does not exist in Camel 1.x When all attempts of redelivery have failed the Exchange is moved to the dead letter queue (the dead letter endpoint). The exchange is then complete and from the client point of view it was processed. As such the Dead Letter Channel have handled the Exchange. For instance configuring the dead letter channel as: errorHandler(deadLetterChannel("jms:queue:dead").maximumRedeliveries(3).redeliverDealy(5000)); The Dead Letter Channel above will clear the caused exception when the Exchange is moved to the jms:queue:dead destination and the client will not notice the failure. By default handled is true. CH APT ER 9 - PAT T ER N APPEN DIX 254 How to let the client notice the error? If you want to move the message to the dead letter queue and also let the client notice the error, then you can configure the Dead Letter Channel to not handle the error. For example: errorHandler(deadLetterChannel("jms:queue:dead").maximumRedeliveries(3).redeliverDealy(5000).handled(f When all attempts of redelivery have failed the Exchange is moved to the dead letter queue (the dead letter endpoint). As the Dead Letter Channel is configured to not handle it, it will mark the Exchange as failed so the client will be notified of this error. About moving Exchange to dead letter queue and using the original message Available as of Camel 2.0 The option useOriginalMessage is used for routing the original input message instead of the current message that potentially is modified during routing. For instance if you have this route: from("jms:queue:order:input") .to("bean:validateOrder"); .to("bean:transformOrder") .to("bean:handleOrder"); The route listen for JMS messages and validates, transforms and handle it. During this the Exchange payload is transformed/modified. So in case something goes wrong and we want to move the message to another JMS destination, then we can configure our Dead Letter Channel with the useOriginalBody option. But when we move the Exchange to this destination we do not know in which state the message is in. Did the error happen in before the transformOrder or after? So to be sure we want to move the original input message we received from jms:queue:order:input. So we can do this by enabling the useOriginalMessage option as shown below: // will use original body errorHandler(deadLetterChannel("jms:queue:dead") .useOriginalMessage().mamimumRedeliveries(5).redeliverDelay(5000); Then the messages routed to the jms:queue:dead is the original input. If we want to manually retry we can move the JMS message from the failed to the input queue, with no problem as the message is the same as the original we received. OnRedelivery Available in Camel 1.6.0 onwards 255 CH A PTE R 9 - PAT T E R N A P P E ND I X Handled See also Exception Clause for more details on the handled policy as this feature was first introduced here and thus we have more docuemntation and samples there. When Dead Letter Channel is doing redeliver its possible to configure a Processor that is executed just before every redelivery attempt. This can be used for the situations where you need to alter the message before its redelivered. See below for sample. Redelivery default values In Camel 2.0 redelivery is disabled by default, as opposed to Camel 1.x in which Dead Letter Channel is configured with maximumRedeliveries=5. The default redeliver policy will use the following values: • maximumRedeliveries=0 (in Camel 1.x the default value is 5) • redeliverDelay=1000L (1 second, new as of Camel 2.0) ◦ use initialRedeliveryDelay for previous versions • maximumRedeliveryDelay = 60 * 1000L (60 seconds) • And the exponential backoff and collision avoidance is turned off. • The retriesExhaustedLogLevel are set to LoggingLevel.ERROR • The retryAttemptedLogLevel are set to LoggingLevel.DEBUG • Stack traces is logged for exhausted messages from Camel 2.2 onwards. • Handled exceptions is not logged from Camel 2.3 onwards The maximum redeliver delay ensures that a delay is never longer than the value, default 1 minute. This can happen if you turn on the exponential backoff. The maximum redeliveries is the number of re delivery attempts. By default Camel will try to process the exchange 1 + 5 times. 1 time for the normal attempt and then 5 attempts as redeliveries. Setting the maximumRedeliveries to a negative value such as -1 will then always redelivery (unlimited). Setting the maximumRedeliveries to 0 will disable any re delivery attempt. Camel will log delivery failures at the DEBUG logging level by default. You can change this by specifying retriesExhaustedLogLevel and/or retryAttemptedLogLevel. See ExceptionBuilderWithRetryLoggingLevelSetTest for an example. In Camel 2.0 you can turn logging of stack traces on/off. If turned off Camel will still log the redelivery attempt. Its just much less verbose. Redeliver Delay Pattern Available as of Camel 2.0 Delay pattern is used as a single option to set a range pattern for delays. If used then the CH APT ER 9 - PAT T ER N APPEN DIX 256 onException and onRedeliver In Camel 2.0 we also added support for per onException to set a onRedeliver. That means you can do special on redelivery for different exceptions, as opposed to onRedelivery set on Dead Letter Channel can be viewed as a global scope. following options does not apply: (delay, backOffMultiplier, useExponentialBackOff, useCollisionAvoidance, maximumRedeliveryDelay). The idea is to set groups of ranges using the following syntax: limit:delay;limit 2:delay 2;limit 3:delay 3;...;limit N:delay N Each group has two values separated with colon ▪ limit = upper limit ▪ delay = delay in millis And the groups is again separated with semi colon. The rule of thumb is that the next groups should have a higher limit than the previous group. Lets clarify this with an example: delayPattern=5:1000;10:5000;20:20000 That gives us 3 groups: ▪ 5:1000 ▪ 10:5000 ▪ 20:20000 Resulting in these delays for redelivery attempt: ▪ Attempt number 0..4 = 0 millis (as the first group start with 5) ▪ Attempt number 5..9 = 1000 millis (the first group) ▪ Attempt number 10..19 = 5000 millis (the second group) ▪ Attempt number 20.. = 20000 millis (the last group) You can start a group with limit 0 to eg have a starting delay: delayPattern=0:1000;5:5000 ▪ Attempt number 0..4 = 1000 millis (the first group) ▪ Attempt number 5.. = 5000 millis (the last group) There is no requirement that the next delay should be higher than the previous. You can use any delay value you like. For example with delayPattern=0:5000;3:1000 we start with 5 sec delay and then later reduce that to 1 second. Redelivery header When a message is redelivered the DeadLetterChannel will append a customizable header to the message to indicate how many times its been redelivered. In Camel 1.x: The header is org.apache.camel.redeliveryCount. 257 CH A PTE R 9 - PAT T E R N A P P E ND I X In Camel 2.0: The header is CamelRedeliveryCounter, which is also defined on the Exchange.REDELIVERY_COUNTER. And a boolean flag whether it is being redelivered or not (first attempt) In Camel 1.x: The header org.apache.camel.Redelivered contains a boolean if the message is redelivered or not. In Camel 2.0: The header CamelRedelivered contains a boolean if the message is redelivered or not, which is also defined on the Exchange.REDELIVERED. Which endpoint failed Available as of Camel 2.1 When Camel routes messages it will decorate the Exchange with a property that contains the last endpoint Camel send the Exchange to: String lastEndpointUri = exchange.getProperty(Exchange.TO_ENDPOINT, String.class); The Exchange.TO_ENDPOINT have the constant value CamelToEndpoint. This information is updated when Camel sends a message to any endpoint. So if it exists its the last endpoint which Camel send the Exchange to. When for example processing the Exchange at a given Endpoint and the message is to be moved into the dead letter queue, then Camel also decorates the Exchange with another property that contains that last endpoint: String failedEndpointUri = exchange.getProperty(Exchange.FAILURE_ENDPOINT, String.class); The Exchange.FAILURE_ENDPOINT have the constant value CamelFailureEndpoint. This allows for example you to fetch this information in your dead letter queue and use that for error reporting. This is useable if the Camel route is a bit dynamic such as the dynamic Recipient List so you know which endpoints failed. Notice: These information is kept on the Exchange even if the message was successfully processed by a given endpoint, and then later fails for example in a local Bean processing instead. So beware that this is a hint that helps pinpoint errors. from("activemq:queue:foo") .to("http://someserver/somepath") .beanRef("foo"); CH APT ER 9 - PAT T ER N APPEN DIX 258 Now suppose the route above and a failure happens in the foo bean. Then the Exchange.TO_ENDPOINT and Exchange.FAILURE_ENDPOINT will still contain the value of http://someserver/somepath. Samples The following example shows how to configure the Dead Letter Channel configuration using the DSL RouteBuilder builder = new RouteBuilder() { public void configure() { // using dead letter channel with a seda queue for errors errorHandler(deadLetterChannel("seda:errors")); // here is our route from("seda:a").to("seda:b"); } }; You can also configure the RedeliveryPolicy as this example shows RouteBuilder builder = new RouteBuilder() { public void configure() { // configures dead letter channel to use seda queue for errors and use at most 2 redelveries // and exponential backoff errorHandler(deadLetterChannel("seda:errors").maximumRedeliveries(2).useExponentialBackOff()); // here is our route from("seda:a").to("seda:b"); } }; How can I modify the Exchange before redelivery? In Camel 1.6.0 we added support directly in Dead Letter Channel to set a Processor that is executed before each redelivery attempt. When Dead Letter Channel is doing redeliver its possible to configure a Processor that is executed just before every redelivery attempt. This can be used for the situations where you need to alter the message before its redelivered. Here we configure the Dead Letter Channel to use our processor MyRedeliveryProcessor to be executed before each redelivery. // we configure our Dead Letter Channel to invoke // MyRedeliveryProcessor before a redelivery is 259 CH A PTE R 9 - PAT T E R N A P P E ND I X // attempted. This allows us to alter the message before errorHandler(deadLetterChannel("mock:error").maximumRedeliveries(5) .onRedelivery(new MyRedeliverPrcessor()) // setting delay to zero is just to make unit teting faster .redeliveryDelay(0L)); And this is the processor MyRedeliveryProcessor where we alter the message. // This is our processor that is executed before every redelivery attempt // here we can do what we want in the java code, such as altering the message public class MyRedeliverPrcessor implements Processor { public void process(Exchange exchange) throws Exception { // the message is being redelivered so we can alter it // we just append the redelivery counter to the body // you can of course do all kind of stuff instead String body = exchange.getIn().getBody(String.class); int count = exchange.getIn().getHeader("CamelRedeliveryCounter", Integer.class); exchange.getIn().setBody(body + count); } } Using This Pattern If you would like to use this EIP Pattern then please read the Getting Started, you may also find the Architecture useful particularly the description of Endpoint and URIs. Then you could try out some of the Examples first before trying this pattern out. ▪ Error Handler ▪ Exception Clause Guaranteed Delivery Camel supports the Guaranteed Delivery from the EIP patterns using the following components • File for using file systems as a persistent store of messages • JMS when using persistent delivery (the default) for working with JMS Queues and Topics for high performance, clustering and load balancing • JPA for using a database as a persistence layer CH APT ER 9 - PAT T ER N APPEN DIX 260 Using This Pattern If you would like to use this EIP Pattern then please read the Getting Started, you may also find the Architecture useful particularly the description of Endpoint and URIs. Then you could try out some of the Examples first before trying this pattern out. Message Bus Camel supports the Message Bus from the EIP patterns. You could view Camel as a Message Bus itself as it allows producers and consumers to be decoupled. Folks often assume that a Message Bus is a JMS though so you may wish to refer to the JMS component for traditional MOM support. Also worthy of node is the XMPP component for supporting messaging over XMPP (Jabber) Using This Pattern If you would like to use this EIP Pattern then please read the Getting Started, you may also find the Architecture useful particularly the description of Endpoint and URIs. Then you could try out some of the Examples first before trying this pattern out. 261 CH A PTE R 9 - PAT T E R N A P P E ND I X Message Construction EVENT MESSAGE Camel supports the Event Message from the EIP patterns by supporting the Exchange Pattern on a Message which can be set to InOnly to indicate a oneway event message. Camel Components then implement this pattern using the underlying transport or protocols. The default behaviour of many Components is InOnly such as for JMS or SEDA Explicitly specifying InOnly If you are using a component which defaults to InOut you can override the Exchange Pattern for an endpoint using the pattern property. foo:bar?exchangePattern=InOnly From 2.0 onwards on Camel you can specify the Exchange Pattern using the dsl. from("mq:someQueue"). inOnly(). bean(Foo.class); or you can invoke an endpoint with an explicit pattern from("mq:someQueue"). inOnly("mq:AnotherQueue"); CH APT ER 9 - PAT T ER N APPEN DIX 262 Using This Pattern If you would like to use this EIP Pattern then please read the Getting Started, you may also find the Architecture useful particularly the description of Endpoint and URIs. Then you could try out some of the Examples first before trying this pattern out. REQUEST REPLY Camel supports the Request Reply from the EIP patterns by supporting the Exchange Pattern on a Message which can be set to InOut to indicate a request/reply. Camel Components then implement this pattern using the underlying transport or protocols. For example when using JMS with InOut the component will by default perform these actions • create by default a temporary inbound queue • set the JMSReplyTo destination on the request message • set the JMSCorrelationID on the request message • send the request message • consume the response and associate the inbound message to the request using the JMSCorrelationID (as you may be performing many concurrent request/responses). Explicitly specifying InOut When consuming messages from JMS a Request-Reply is indicated by the presence of the JMSReplyTo header. You can explicitly force an endpoint to be in Request Reply mode by setting the exchange pattern on the URI. e.g. jms:MyQueue?exchangePattern=InOut NOTE From Camel 1.5.1 you can specify the exchange pattern in DSL rule or Spring configuration. // Send to an endpoint using InOut from("direct:testInOut").inOut("mock:result"); // Send to an endpoint using InOut 263 CH A PTE R 9 - PAT T E R N A P P E ND I X from("direct:testInOnly").inOnly("mock:result"); // Set the exchange pattern to InOut, then send it from direct:inOnly to mock:result endpoint from("direct:testSetToInOnlyThenTo").inOnly().to("mock:result"); from("direct:testSetToInOutThenTo").inOut().to("mock:result"); // Or we can pass the pattern as a parameter to the to() method from("direct:testToWithInOnlyParam").to(ExchangePattern.InOnly, "mock:result"); from("direct:testToWithInOutParam").to(ExchangePattern.InOut, "mock:result"); from("direct:testToWithRobustInOnlyParam").to(ExchangePattern.RobustInOnly, "mock:result"); // Set the exchange pattern to InOut, then send it on from("direct:testSetExchangePatternInOnly").setExchangePattern(ExchangePattern.InOnly).to("mock:result CH APT ER 9 - PAT T ER N APPEN DIX 264 Using This Pattern If you would like to use this EIP Pattern then please read the Getting Started, you may also find the Architecture useful particularly the description of Endpoint and URIs. Then you could try out some of the Examples first before trying this pattern out. Correlation Identifier Camel supports the Correlation Identifier from the EIP patterns by getting or setting a header on a Message. When working with the ActiveMQ or JMS components the correlation identifier header is called JMSCorrelationID. You can add your own correlation identifier to any message exchange to help correlate messages together to a single conversation (or business process). The use of a Correlation Identifier is key to working with the Camel Business Activity Monitoring Framework and can also be highly useful when testing with simulation or canned data such as with the Mock testing framework See Also • BAM 265 CH A PTE R 9 - PAT T E R N A P P E ND I X RETURN ADDRESS Camel supports the Return Address from the EIP patterns by using the JMSReplyTo header. For example when using JMS with InOut the component will by default return to the address given in JMSReplyTo. Using This Pattern If you would like to use this EIP Pattern then please read the Getting Started, you may also find the Architecture useful particularly the description of Endpoint and URIs. Then you could try out some of the Examples first before trying this pattern out. MESSAGE ROUTING Content Based Router The Content Based Router from the EIP patterns allows you to route messages to the correct destination based on the contents of the message exchanges. The following example shows how to route a request from an input seda:a endpoint to either seda:b, seda:c or seda:d depending on the evaluation of various Predicate expressions Using the Fluent Builders RouteBuilder builder = new RouteBuilder() { public void configure() { errorHandler(deadLetterChannel("mock:error")); CH APT ER 9 - PAT T ER N APPEN DIX 266 from("seda:a").choice().when(header("foo").isEqualTo("bar")).to("seda:b") .when(header("foo").isEqualTo("cheese")).to("seda:c").otherwise().to("seda:d"); } }; Using the Spring XML Extensions $foo = 'bar' $foo = 'cheese' For further examples of this pattern in use you could look at the junit test case Using This Pattern If you would like to use this EIP Pattern then please read the Getting Started, you may also find the Architecture useful particularly the description of Endpoint and URIs. Then you could try out some of the Examples first before trying this pattern out. Message Filter The Message Filter from the EIP patterns allows you to filter messages The following example shows how to create a Message Filter route consuming messages from an endpoint called queue:a which if the Predicate is true will be dispatched to queue:b 267 CH A PTE R 9 - PAT T E R N A P P E ND I X Using the Fluent Builders RouteBuilder builder = new RouteBuilder() { public void configure() { errorHandler(deadLetterChannel("mock:error")); from("seda:a").filter(header("foo").isEqualTo("bar")).to("seda:b"); } }; You can of course use many different Predicate languages such as XPath, XQuery, SQL or various Scripting Languages. Here is an XPath example from("direct:start"). filter().xpath("/person[@name='James']"). to("mock:result"); Using the Spring XML Extensions $foo = 'bar' For further examples of this pattern in use you could look at the junit test case Using stop Available as of Camel 2.0 Stop is a bit different than a message filter as it will filter out all messages. Stop is convenient to use in a Content Based Router when you for example need to stop further processing in one of the predicates. In the example below we do not want to route messages any further that has the word Bye in the message body. Notice how we prevent this in the when predicate by using the .stop(). from("direct:start") .choice() .when(body().contains("Hello")).to("mock:hello") .when(body().contains("Bye")).to("mock:bye").stop() CH APT ER 9 - PAT T ER N APPEN DIX 268 .otherwise().to("mock:other") .end() .to("mock:result"); Using This Pattern If you would like to use this EIP Pattern then please read the Getting Started, you may also find the Architecture useful particularly the description of Endpoint and URIs. Then you could try out some of the Examples first before trying this pattern out. Dynamic Router The Dynamic Router from the EIP patterns allows you to route messages while avoiding the dependency of the router on all possible destinations while maintaining its efficiency. The simplest way to implement this is to use the RecipientList Annotation on a Bean method to determine where to route the message. public class MyDynamicRouter { @Consume(uri = "activemq:foo") @RecipientList public List route(@XPath("/customer/id") String customerId, @Header("Location") String location, Document body) { // query a database to find the best match of the endpoint based on the input parameteres ... } } In the above we can use the Parameter Binding Annotations to bind different parts of the Message to method parameters or use an Expression such as using XPath or XQuery. The method can be invoked in a number of ways as described in the Bean Integration such as • POJO Producing 269 CH A PTE R 9 - PAT T E R N A P P E ND I X • Spring Remoting • Bean component Using This Pattern If you would like to use this EIP Pattern then please read the Getting Started, you may also find the Architecture useful particularly the description of Endpoint and URIs. Then you could try out some of the Examples first before trying this pattern out. Recipient List The Recipient List from the EIP patterns allows you to route messages to a number of dynamically specified recipients. The recipients will receive a copy of the same Exchange and Camel will execute them sequentially. Static Recipient List The following example shows how to route a request from an input queue:a endpoint to a static list of destinations Using Annotations You can use the RecipientList Annotation on a POJO to create a Dynamic Recipient List. For more details see the Bean Integration. Using the Fluent Builders RouteBuilder builder = new RouteBuilder() { public void configure() { errorHandler(deadLetterChannel("mock:error")); from("seda:a").multicast().to("seda:b", "seda:c", "seda:d"); } }; Using the Spring XML Extensions CH APT ER 9 - PAT T ER N APPEN DIX 270 Dynamic Recipient List Usually one of the main reasons for using the Recipient List pattern is that the list of recipients is dynamic and calculated at runtime. The following example demonstrates how to create a dynamic recipient list using an Expression (which in this case it extracts a named header value dynamically) to calculate the list of endpoints which are either of type Endpoint or are converted to a String and then resolved using the endpoint URIs. Using the Fluent Builders RouteBuilder builder = new RouteBuilder() { public void configure() { errorHandler(deadLetterChannel("mock:error")); from("seda:a").recipientList(header("foo")); } }; The above assumes that the header contains a list of endpoint URIs. The following takes a single string header and tokenizes it from("direct:a").recipientList( header("recipientListHeader").tokenize(",")); Iteratable value The dynamic list of recipients that are defined in the header must be iteratable such as: ▪ java.util.Collection ▪ java.util.Iterator ▪ arrays ▪ org.w3c.dom.NodeList ▪ Camel 1.6.0: a single String with values separated with comma ▪ any other type will be regarded as a single value 271 CH A PTE R 9 - PAT T E R N A P P E ND I X Using the Spring XML Extensions $foo For further examples of this pattern in use you could look at one of the junit test case Using delimiter in Spring XML Available as of Camel 1.6.0 In Spring DSL you can set the delimiter attribute for setting a delimiter to be used if the header value is a single String with multiple separated endpoints. By default Camel uses comma as delimiter, but this option lets you specify a customer delimiter to use instead.
myHeader
So if myHeader contains a String with the value "activemq:queue:foo, activemq:topic:hello , log:bar" then Camel will split the String using the delimiter given in the XML that was comma, resulting into 3 endpoints to send to. You can use spaces between the endpoints as Camel will trim the value when it lookup the endpoint to send to. Note: In Java DSL you use the tokenizer to archive the same. The route above in Java DSL: from("direct:a").recipientList(header("myHeader").tokenize(",")); In Camel 2.1 its a bit easier as you can pass in the delimiter as 2nd parameter: from("direct:a").recipientList(header("myHeader"), "#"); CH APT ER 9 - PAT T ER N APPEN DIX 272 Sending to multiple recipients in parallel Available as of Camel 2.2 The Recipient List now supports parallelProcessing that for example Splitter also supports. You can use it to use a thread pool to have concurrent tasks sending the Exchange to multiple recipients concurrently. from("direct:a").recipientList(header("myHeader")).parallelProcessing(); And in Spring XML its an attribute on the recipient list tag.
myHeader
Stop continuing in case one recipient failed Available as of Camel 2.2 The Recipient List now supports stopOnException that for example Splitter also supports. You can use it to stop sending to any further recipients in case any recipient failed. from("direct:a").recipientList(header("myHeader")).stopOnException(); And in Spring XML its an attribute on the recipient list tag.
myHeader
Note: You can combine parallelProcessing and stopOnException and have them both true. Ignore invalid endpoints Available as of Camel 2.3 The Recipient List now supports ignoreInvalidEndpoints which the Routing Slip also supports. You can use it to skip endpoints which is invalid. 273 CH A PTE R 9 - PAT T E R N A P P E ND I X from("direct:a").recipientList(header("myHeader")).ignoreInvalidEndpoints(); And in Spring XML its an attribute on the recipient list tag.
myHeader
Then lets say the myHeader contains the following two endpoints direct:foo,xxx:bar. The first endpoint is valid and works. However the 2nd is invalid and will just be ignored. Camel logs at INFO level about, so you can see why the endpoint was invalid. Using custom AggregationStrategy Available as of Camel 2.2 You can now use you own AggregationStrategy with the Recipient List. However its not that often you need that. What its good for is that in case you are using Request Reply messaging then the replies from the recipient can be aggregated. By default Camel uses UseLatestAggregationStrategy which just keeps that last received reply. What if you must remember all the bodies that all the recipients send back, then you can use your own custom aggregator that keeps those. Its the same principle as with the Aggregator EIP so check it out for details. from("direct:a") .recipientList(header("myHeader")).aggregationStrategy(new MyOwnAggregationStrategy()) .to("direct:b"); And in Spring XML its an attribute on the recipient list tag.
myHeader
CH APT ER 9 - PAT T ER N APPEN DIX 274 Using custom thread pool Available as of Camel 2.2 This is only needed when you use parallelProcessing. By default Camel uses a thread pool with 10 threads. Notice this is subject to change when we overhaul thread pool management and configuration later (hopefully in Camel 2.2). You configure this just as you would with the custom aggregation strategy. Using method call as recipient list You can use a Bean to provide the recipients, for example: from("activemq:queue:test").recipientList().method(MessageRouter.class, "routeTo"); And then MessageRouter: public class MessageRouter { public String routeTo() { String queueName = "activemq:queue:test2"; return queueName; } } When you use a Bean then do not also use the @RecipientList annotation as this will in fact add yet another recipient list, so you end up having two. Do not do like this. public class MessageRouter { @RecipientList public String routeTo() { String queueName = "activemq:queue:test2"; return queueName; } } Well you should only do like that above (using @RecipientList) if you route just route to a Bean which you then want to act as a recipient list. So the original route can be changed to: from("activemq:queue:test").bean(MessageRouter.class, "routeTo"); Which then would invoke the routeTo method and detect its annotated with @RecipientList and then act accordingly as if it was a recipient list EIP. 275 CH A PTE R 9 - PAT T E R N A P P E ND I X Using This Pattern If you would like to use this EIP Pattern then please read the Getting Started, you may also find the Architecture useful particularly the description of Endpoint and URIs. Then you could try out some of the Examples first before trying this pattern out. Splitter The Splitter from the EIP patterns allows you split a message into a number of pieces and process them individually As of Camel 2.0, you need to specify a Splitter as split(). In earlier versions of Camel, you need to use splitter(). Example The following example shows how to take a request from the queue:a endpoint the split it into pieces using an Expression, then forward each piece to queue:b Using the Fluent Builders RouteBuilder builder = new RouteBuilder() { public void configure() { errorHandler(deadLetterChannel("mock:error")); from("seda:a").split(body(String.class).tokenize("\n")).to("seda:b"); } }; The splitter can use any Expression language so you could use any of the Languages Supported such as XPath, XQuery, SQL or one of the Scripting Languages to perform the split. e.g. from("activemq:my.queue").split(xpath("//foo/ bar")).convertBodyTo(String.class).to("file://some/directory") Using the Spring XML Extensions CH APT ER 9 - PAT T ER N APPEN DIX 276 What does the splitter return? The Splitter will by default return the last splitted message. You can override this by suppling your own strategy as an AggregationStrategy. There is a sample on this page (Split aggregate request/reply sample). Notice its the same strategy as the Aggregator supports. This Splitter can be viewed as having a build in light weight Aggregator. /invoice/lineItems For further examples of this pattern in use you could look at one of the junit test case Using Tokenizer from Spring XML Extensions Avaiaible as of Camel 2.0 You can use the tokenizer expression in the Spring DSL to split bodies or headers using a token. This is a common use-case, so we provided a special tokenizer tag for this. In the sample below we split the body using a @ as separator. You can of course use comma or space or even a regex pattern, also set regex=true. Splitting the body in Spring XML is a bit harder as you need to use the Simple language to dictate this ${body} 277 CH A PTE R 9 - PAT T E R N A P P E ND I X Message Headers The following headers is set on each Exchange that are split: header type description org.apache.camel.splitCounter int Camel 1.x: A split counter that increases for each Exchange being split. The counter starts from 0. int Camel 1.x: The total number of Exchanges that was splitted. This header is not applied for stream based splitting. org.apache.camel.splitSize Exchange properties The following properties is set on each Exchange that are split: header type description org.apache.camel.splitCounter int Camel 1.6.2: A split counter that increases for each Exchange being split. The counter starts from 0. org.apache.camel.splitSize int Camel 1.6.2: The total number of Exchanges that was splitted. This header is not applied for stream based splitting. CamelSplitIndex int Camel 2.0: A split counter that increases for each Exchange being split. The counter starts from 0. CamelSplitSize int Camel 2.0: The total number of Exchanges that was splitted. This header is not applied for stream based splitting. CamelSplitComplete boolean Camel 2.4: Whether or not this Exchange is the last. Parallel execution of distinct 'parts' If you want to execute all parts in parallel you can use special notation of split() with two arguments, where the second one is a boolean flag if processing should be parallel. e.g. CH APT ER 9 - PAT T ER N APPEN DIX 278 XPathBuilder xPathBuilder = new XPathBuilder("//foo/bar"); from("activemq:my.queue").split(xPathBuilder, true).to("activemq:my.parts"); In Camel 2.0 the boolean option has been refactored into a builder method parallelProcessing so its easier to understand what the route does when we use a method instead of true|false. XPathBuilder xPathBuilder = new XPathBuilder("//foo/bar"); from("activemq:my.queue").split(xPathBuilder).parallelProcessing().to("activemq:my.parts"); Stream based Available as of Camel 1.5 You can split streams by enabling the streaming mode using the streaming builder method. from("direct:streaming").split(body().tokenize(",")).streaming().to("activemq:my.parts"); Specifying a custom aggregation strategy Available as of Camel 2.0 This is specified similar to the Aggregator. Specifying a custom ThreadPoolExecutor You can customize the underlying ThreadPoolExecutor used in the parallel splitter. In the Java DSL try something like this: Camel 1.x: XPathBuilder xPathBuilder = new XPathBuilder("//foo/bar"); ThreadPoolExecutor threadPoolExecutor = new ThreadPoolExecutor(8, 16, 0L, TimeUnit.MILLISECONDS, new LinkedBlockingQueue()); from("activemq:my.queue").split(xPathBuilder, true, threadPoolExecutor).to("activemq:my.parts"); In the Spring XML try this: Available as of Camel 1.6.0 Listing 11. Spring DSL 279 CH A PTE R 9 - PAT T E R N A P P E ND I X /invoice/lineItems Camel 2.x: XPathBuilder xPathBuilder = new XPathBuilder("//foo/bar"); ThreadPoolExecutor threadPoolExecutor = new ThreadPoolExecutor(8, 16, 0L, TimeUnit.MILLISECONDS, new LinkedBlockingQueue()); from("activemq:my.queue").split(xPathBuilder).parallelProcessing().executeService(threadPoolExecutor). Using a Pojo to do the splitting As the Splitter can use any Expression to do the actual splitting we leverage this fact and use a method expression to invoke a Bean to get the splitted parts. The Bean should return a value that is iterable such as: java.util.Collection, java.util.Iterator or an array. In the route we define the Expression as a method call to invoke our Bean that we have registered with the id mySplitterBean in the Registry. from("direct:body") // here we use a POJO bean mySplitterBean to do the split of the payload .split().method("mySplitterBean", "splitBody") .to("mock:result"); from("direct:message") // here we use a POJO bean mySplitterBean to do the split of the message // with a certain header value .split().method("mySplitterBean", "splitMessage") .to("mock:result"); CH APT ER 9 - PAT T ER N APPEN DIX 280 And the logic for our Bean is as simple as. Notice we use Camel Bean Binding to pass in the message body as a String object. public class MySplitterBean { /** * The split body method returns something that is iteratable such as a java.util.List. * * @param body the payload of the incoming message * @return a list containing each part splitted */ public List splitBody(String body) { // since this is based on an unit test you can of cause // use different logic for splitting as Camel have out // of the box support for splitting a String based on comma // but this is for show and tell, since this is java code // you have the full power how you like to split your messages List answer = new ArrayList(); String[] parts = body.split(","); for (String part : parts) { answer.add(part); } return answer; } /** * The split message method returns something that is iteratable such as a java.util.List. * * @param header the header of the incoming message with the name user * @param body the payload of the incoming message * @return a list containing each part splitted */ public List splitMessage(@Header(value = "user") String header, @Body String body) { // we can leverage the Parameter Binding Annotations // http://camel.apache.org/parameter-binding-annotations.html // to access the message header and body at same time, // then create the message that we want, splitter will // take care rest of them. // *NOTE* this feature requires Camel version >= 1.6.1 List answer = new ArrayList(); String[] parts = header.split(","); for (String part : parts) { DefaultMessage message = new DefaultMessage(); message.setHeader("user", part); message.setBody(body); answer.add(message); } return answer; } } 281 CH A PTE R 9 - PAT T E R N A P P E ND I X Split aggregate request/reply sample This sample shows how you can split an Exchange, process each splitted message, aggregate and return a combined response to the original caller using request/reply. The route below illustrates this and how the split supports a aggregationStrategy to hold the in progress processed messages: // this routes starts from the direct:start endpoint // the body is then splitted based on @ separator // the splitter in Camel supports InOut as well and for that we need // to be able to aggregate what response we need to send back, so we provide our // own strategy with the class MyOrderStrategy. from("direct:start") .split(body().tokenize("@"), new MyOrderStrategy()) // each splitted message is then send to this bean where we can process it .to("bean:MyOrderService?method=handleOrder") // this is important to end the splitter route as we do not want to do more routing // on each splitted message .end() // after we have splitted and handled each message we want to send a single combined // response back to the original caller, so we let this bean build it for us // this bean will receive the result of the aggregate strategy: MyOrderStrategy .to("bean:MyOrderService?method=buildCombinedResponse") And the OrderService bean is as follows: public static class MyOrderService { private static int counter; /** * We just handle the order by returning a id line for the order */ public String handleOrder(String line) { LOG.debug("HandleOrder: " + line); return "(id=" + ++counter + ",item=" + line + ")"; } /** * We use the same bean for building the combined response to send * back to the original caller */ public String buildCombinedResponse(String line) { LOG.debug("BuildCombinedResponse: " + line); return "Response[" + line + "]"; } } And our custom aggregationStrategy that is responsible for holding the in progress aggregated message that after the splitter is ended will be sent to the CH APT ER 9 - PAT T ER N APPEN DIX 282 buildCombinedResponse method for final processing before the combined response can be returned to the waiting caller. /** * This is our own order aggregation strategy where we can control * how each splitted message should be combined. As we do not want to * loos any message we copy from the new to the old to preserve the * order lines as long we process them */ public static class MyOrderStrategy implements AggregationStrategy { public Exchange aggregate(Exchange oldExchange, Exchange newExchange) { // put order together in old exchange by adding the order from new exchange if (oldExchange == null) { // the first time we aggregate we only have the new exchange, // so we just return it return newExchange; } String orders = oldExchange.getIn().getBody(String.class); String newLine = newExchange.getIn().getBody(String.class); LOG.debug("Aggregate old orders: " + orders); LOG.debug("Aggregate new order: " + newLine); // put orders together separating by semi colon orders = orders + ";" + newLine; // put combined order back on old to preserve it oldExchange.getIn().setBody(orders); // return old as this is the one that has all the orders gathered until now return oldExchange; } } So lets run the sample and see how it works. We send an Exchange to the direct:start endpoint containing a IN body with the String value: A@B@C. The flow is: HandleOrder: A HandleOrder: B Aggregate old orders: (id=1,item=A) Aggregate new order: (id=2,item=B) HandleOrder: C Aggregate old orders: (id=1,item=A);(id=2,item=B) Aggregate new order: (id=3,item=C) BuildCombinedResponse: (id=1,item=A);(id=2,item=B);(id=3,item=C) Response to caller: Response[(id=1,item=A);(id=2,item=B);(id=3,item=C)] 283 CH A PTE R 9 - PAT T E R N A P P E ND I X Stop processing in case of exception Available as of Camel 2.1 The Splitter will by default continue to process the entire Exchange even in case of one of the splitted message will thrown an exception during routing. For example if you have an Exchange with 1000 rows that you split and route each sub message. During processing of these sub messages an exception is thrown at the 17th. What Camel does by default is to process the remainder 983 messages. You have the chance to remedy or handle this in the AggregationStrategy. But sometimes you just want Camel to stop and let the exception be propagated back, and let the Camel error handler handle it. You can do this in Camel 2.1 by specifying that it should stop in case of an exception occurred. This is done by the stopOnException option as shown below: from("direct:start") .split(body().tokenize(",")).stopOnException() .process(new MyProcessor()) .to("mock:split"); And using XML DSL you specify it as follows: Using This Pattern If you would like to use this EIP Pattern then please read the Getting Started, you may also find the Architecture useful particularly the description of Endpoint and URIs. Then you could try out some of the Examples first before trying this pattern out. Aggregator This applies for Camel version 2.3 or newer. If you use an older version then use this Aggregator link instead. The Aggregator from the EIP patterns allows you to combine a number of messages together into a single message. CH APT ER 9 - PAT T ER N APPEN DIX 284 A correlation Expression is used to determine the messages which should be aggregated together. If you want to aggregate all messages into a single message, just use a constant expression. An AggregationStrategy is used to combine all the message exchanges for a single correlation key into a single message exchange. Aggregator options The aggregator supports the following options: Option Description correlationExpression Mandatory Expression which evaluates the correlation key to use for aggregation. The Exchange which has the same correlation key is aggregated together. If the correlation key could not be evaluated an Exception is thrown. You can disable this by using the ignoreBadCorrelationKeys option. aggregationStrategy Mandatory AggregationStrategy which is used to merge the incoming Exchange with the existing already merged exchanges. At first call the oldExchang parameter is null. On subsequent invocations the oldExchnage contains the merged exchanges and newExchange is of course the new incoming Exchange. strategyRef A reference to lookup the AggregationStrategy in the Registry. completionSize Number of messages aggregated before the aggregation is complete. This option can be set as either a fixed value or using an Expression which allows you to evaluate a size dynamically - will use Integer as result. If both are set Camel will fallback to use the fixed value if the Expression result was null or 0. completionTimeout Time in millis that an aggregated exchange should be inactive before its complete. Camel has a background task that runs once a minute to check for inactive aggregated exchanges. This option can be set as either a fixed value or using an Expression which allows you to evaluate a timeout dynamically - will use Long as result. If both are set Camel will fallback to use the fixed value if the Expression result was null or 0. You cannot use this option together with completionInterval, only one of the can be used. completionInterval A repeating period in millis by which the aggregator will complete all current aggregated exchanges. Camel has a background tasks which is trigger every period. You cannot use this option together with completionTimeout, only one of the can be used. completionPredicate A Predicate to indicate when an aggregated exchange is complete. false This option is if the exchanges is coming from a Batch Consumer. Then when enabled the Aggregator2 will use the batch size determined by the Batch Consumer in the message header CamelBatchSize. See more details at Batch Consumer. This can be used to aggregate all files consumed from a File endpoint in that given poll. eagerCheckCompletion false Whether or not to eager check for completion when a new incoming Exchange has been received. This option influences the behavior of the completionPredicate option as the Exchange being passed in changes accordingly. When false the Exchange passed in the Predicate is the aggregated Exchange which means any information you may store on the aggregated Exchange from the AggregationStrategy is avail for the Predicate. When true the Exchange passed in the Predicate is the incoming Exchange, which means you can access data from the incoming Exchange. groupExchanges false If enabled then Camel will group all aggregated Exchanges into a single combined org.apache.camel.impl.GroupedExchange holder class that holds all the aggregated Exchanges. And as a result only one Exchange is being sent out from the aggregator. Can be used to combine many incomming Exchanges into a single output Exchange without coding a custom AggregationStrategy yourself. ignoreInvalidCorrelationKeys false Whether or not to ignore correlation keys which could not be evaluated to a value. By default Camel will thrown an Exception, but you can enable this option and ignore the situation instead. completionFromBatchConsumer 285 Default closeCorrelationKeyOnCompletion Whether or not too late Exchange should be accepted or not. You can enable this to indicate that if a correlation key has already been completed, then any new exchanges with the same correlation key be denied. Camel will then throw a closedCorrelationKeyException exception. When using this option you pass in a integer which is a number for a LRUCache which keeps that last X number of closed correlation keys. You can pass in 0 or a negative value to indicate a unbounded cache. By passing in a number you are ensured that cache wont grown too big if you use a log of different correlation keys. aggregationRepository Allows you to plugin you own implementation of org.apache.camel.spi.AggregationRepository which keeps track of the current inflight aggregated exchanges. Camel uses by default a memory based implementation. aggregationRepositoryRef Reference to lookup a aggregationRepository in the Registry. CH A PTE R 9 - PAT T E R N A P P E ND I X parallelProcessing false When aggregated are completed they are being send out of the aggregator. This option indicates whether or not Camel should use a thread pool with multiple threads for concurrency. If not custom thread pool has been specified then Camel creates a default pool with 10 concurrent threads. executorService If using parallelProcessing you can specify a custom thread pool to be used. In fact also if you are not using parallelProcessing this custom thread pool is used to send out aggregated exchanges as well. executorServiceRef Reference to lookup a executorService in the Registry Exchange Properties The following properties is set on each Exchange that are aggregated: header type description CamelAggregatedSize int The total number of Exchanges aggregated into this combined Exchange. CamelAggregatedCompletedBy String Indicator how the aggregation was completed as a value of either: predicate, size, consumer, timeout or interval. About AggregationStrategy The AggregationStrategy is used for aggregate the old (lookup by its correlation id) and the new exchanges together into a single exchange. Possible implementations include performing some kind of combining or delta processing, such as adding line items together into an invoice or just using the newest exchange and removing old exchanges such as for state tracking or market data prices; where old values are of little use. Notice the aggregation strategy is a mandatory option and must be provided to the aggregator. About completion When aggregation Exchanges at some point you need to indicate that the aggregated exchanges is complete, so they can be send out of the aggregator. Camel allows you to indicate completion in various ways as follows: ▪ completionTimeout - Is an inactivity timeout in which is triggered if no new exchanges has been aggregated for that particular correlation key within the period. ▪ completionInterval - Once every X period all the current aggregated exchanges is completed. ▪ completionSize - Is a number indicating that after X aggregated exchanges its complete. ▪ completionPredicate - Runs a Predicate when a new exchange is aggregated to determine if we are complete or not ▪ completionFromBatchConsumer - Special option for Batch Consumer which allows you to complete when all the messages from the batch has been aggregated. | Notice that all the completion ways are per correlation key. And you can combine them in any way your like. Its basically the first which triggers that wins. So you can use a completion size together with a completion timeout. Only completionTimeout and completionInterval cannot be used at the same time. CH APT ER 9 - PAT T ER N APPEN DIX 286 Notice the completion is a mandatory option and must be provided to the aggregator. If not provided Camel will thrown an Exception on startup. Persistent AggregationRepository The aggregator provides a pluggable repository which you can implement your own org.apache.camel.spi.AggregationRepository. If you need persistent repository then you can use the Camel HawtDB component. Examples See some examples from the old Aggregator which is somewhat similar to this new aggregator. Using completionTimeout In this example we want to aggregate all incoming messages and after 3 seconds of inactivity we want the aggregation to complete. This is done using the completionTimeout option as shown: from("direct:start") // aggregate all exchanges correlated by the id header. // Aggregate them using the BodyInAggregatingStrategy strategy which // and after 3 seconds of inactivity them timeout and complete the aggregation // and send it to mock:aggregated .aggregate(header("id"), new BodyInAggregatingStrategy()).completionTimeout(3000) .to("mock:aggregated"); And the same example using Spring XML: header.id 287 CH A PTE R 9 - PAT T E R N A P P E ND I X Setting options in Spring XML Many of the options are configurable as attributes on the tag when using Spring XML. Using completionSize In this example we want to aggregate all incoming messages and when we have 3 messages aggregated (in the same correlation group) we want the aggregation to complete. This is done using the completionSize option as shown: from("direct:start") // aggregate all exchanges correlated by the id header. // Aggregate them using the BodyInAggregatingStrategy strategy which // and after 3 messages has been aggregated then complete the aggregation // and send it to mock:aggregated .aggregate(header("id"), new BodyInAggregatingStrategy()).completionSize(3) .to("mock:aggregated"); And the same example using Spring XML: header.id Using completionPredicate In this example we want to aggregate all incoming messages and use a Predicate to determine when we are complete. The Predicate can be evaluated using either the aggregated exchange (default) or the incoming exchange. We will so both situations as examples. We start with the default situation as shown: from("direct:start") // aggregate all exchanges correlated by the id header. CH APT ER 9 - PAT T ER N APPEN DIX 288 // Aggregate them using the BodyInAggregatingStrategy strategy which // and when the aggregated body contains A+B+C then complete the aggregation // and send it to mock:aggregated .aggregate(header("id"), new BodyInAggregatingStrategy()).completionPredicate(body().contains("A+B+C")) .to("mock:aggregated"); And the same example using Spring XML: header.id ${body} contains 'A+B+C' And the other situation where we use the eagerCheckCompletion option to tell Camel to use the incoming Exchange. Notice how we can just test in the completion predicate that the incoming message is the END message: from("direct:start") // aggregate all exchanges correlated by the id header. // Aggregate them using the BodyInAggregatingStrategy strategy // do eager checking which means the completion predicate will use the incoming exchange // which allows us to trigger completion when a certain exchange arrived which is the // END message .aggregate(header("id"), new BodyInAggregatingStrategy()) .eagerCheckCompletion().completionPredicate(body().isEqualTo("END")) .to("mock:aggregated"); And the same example using Spring XML: 289 CH A PTE R 9 - PAT T E R N A P P E ND I X header.id ${body} == 'END' Using dynamic completionTimeout In this example we want to aggregate all incoming messages and after a period of inactivity we want the aggregation to complete. The period should be computed at runtime based on the timeout header in the incoming messages. This is done using the completionTimeout option as shown: from("direct:start") // aggregate all exchanges correlated by the id header. // Aggregate them using the BodyInAggregatingStrategy strategy which // and the timeout header contains the timeout in millis of inactivity them timeout and complete the aggregation // and send it to mock:aggregated .aggregate(header("id"), new BodyInAggregatingStrategy()).completionTimeout(header("timeout")) .to("mock:aggregated"); And the same example using Spring XML: header.id
timeout
CH APT ER 9 - PAT T ER N APPEN DIX 290 Note: You can also add a fixed timeout value and Camel will fallback to use this value if the dynamic value was null or 0. Using dynamic completionSize In this example we want to aggregate all incoming messages based on a dynamic size per correlation key. The size is computed at runtime based on the mySize header in the incoming messages. This is done using the completionSize option as shown: from("direct:start") // aggregate all exchanges correlated by the id header. // Aggregate them using the BodyInAggregatingStrategy strategy which // and the header mySize determines the number of aggregated messages should trigger the completion // and send it to mock:aggregated .aggregate(header("id"), new BodyInAggregatingStrategy()).completionSize(header("mySize")) .to("mock:aggregated"); And the same example using Spring XML: header.id
mySize
Note: You can also add a fixed size value and Camel will fallback to use this value if the dynamic value was null or 0. 291 CH A PTE R 9 - PAT T E R N A P P E ND I X Using This Pattern If you would like to use this EIP Pattern then please read the Getting Started, you may also find the Architecture useful particularly the description of Endpoint and URIs. Then you could try out some of the Examples first before trying this pattern out. See also ▪ ▪ ▪ ▪ ▪ The Loan Broker Example which uses an aggregator Blog post by Torsten Mielke about using the aggregator correctly. The old Aggregator HawtDB for persistence support Aggregate Example for an example application Resequencer The Resequencer from the EIP patterns allows you to reorganise messages based on some comparator. By default in Camel we use an Expression to create the comparator; so that you can compare by a message header or the body or a piece of a message etc. Camel supports two resequencing algorithms: • Batch resequencing collects messages into a batch, sorts the messages and sends them to their output. • Stream resequencing re-orders (continuous) message streams based on the detection of gaps between messages. By default the Resequencer does not support duplicate messages and will only keep the last message, in case a message arrives with the same message expression. However in the batch mode you can enable it to allow duplicates. Batch Resequencing The following example shows how to use the batch-processing resequencer so that messages are sorted in order of the body() expression. That is messages are collected into a batch (either by a maximum number of messages per batch or using a timeout) then they are sorted in order and then sent out to their output. Using the Fluent Builders from("direct:start").resequence(body()).to("mock:result"); This is equvalent to CH APT ER 9 - PAT T ER N APPEN DIX 292 from("direct:start").resequence(body()).batch().to("mock:result"); The batch-processing resequencer can be further configured via the size() and timeout() methods. from("direct:start").resequence(body()).batch().size(300).timeout(4000L).to("mock:result") This sets the batch size to 300 and the batch timeout to 4000 ms (by default, the batch size is 100 and the timeout is 1000 ms). Alternatively, you can provide a configuration object. from("direct:start").resequence(body()).batch(new BatchResequencerConfig(300, 4000L)).to("mock:result") So the above example will reorder messages from endpoint direct:a in order of their bodies, to the endpoint mock:result. Typically you'd use a header rather than the body to order things; or maybe a part of the body. So you could replace this expression with resequencer(header("mySeqNo")) for example to reorder messages using a custom sequence number in the header mySeqNo. You can of course use many different Expression languages such as XPath, XQuery, SQL or various Scripting Languages. You can also use multiple expressions; so you could for example sort by priority first then some other custom header resequence(header("mySeqNo"), header("MyCustomerRating")) Using the Spring XML Extensions body 293 CH A PTE R 9 - PAT T E R N A P P E ND I X Allow Duplicates Available as of Camel 2.4 In the batch mode, you can now allow duplicates. In Java DSL there is a allowDuplicates() method and in Spring XML there is an allowDuplicates=true attribute on the you can use to enable it. Reverse Available as of Camel 2.4 In the batch mode, you can now reverse the expression ordering. By default the order is based on 0..9,A..Z, which would let messages with low numbers be ordered first, and thus also also outgoing first. In some cases you want to reverse order, which is now possible. In Java DSL there is a reverse() method and in Spring XML there is an reverse=true attribute on the you can use to enable it. Resequence JMS messages based on JMSPriority Available as of Camel 2.4 It's now much easier to use the Resequencer to resequence messages from JMS queues based on JMSPriority. For that to work you need to use the two new options allowDuplicates and reverse. from("jms:queue:foo") // sort by JMSPriority by allowing duplicates (message can have same JMSPriority) // and use reverse ordering so 9 is first output (most important), and 0 is last // use batch mode and fire every 3th second .resequence(header("JMSPriority")).batch().timeout(3000).allowDuplicates().reverse() .to("mock:result"); Notice this is only possible in the batch mode of the Resequencer. Stream Resequencing The next example shows how to use the stream-processing resequencer. Messages are reordered based on their sequence numbers given by a seqnum header using gap detection and timeouts on the level of individual messages. Using the Fluent Builders from("direct:start").resequence(header("seqnum")).stream().to("mock:result"); CH APT ER 9 - PAT T ER N APPEN DIX 294 The stream-processing resequencer can be further configured via the capacity() and timeout() methods. from("direct:start").resequence(header("seqnum")).stream().capacity(5000).timeout(4000L).to("mock:resu This sets the resequencer's capacity to 5000 and the timeout to 4000 ms (by default, the capacity is 100 and the timeout is 1000 ms). Alternatively, you can provide a configuration object. from("direct:start").resequence(header("seqnum")).stream(new StreamResequencerConfig(5000, 4000L)).to("mock:result") The stream-processing resequencer algorithm is based on the detection of gaps in a message stream rather than on a fixed batch size. Gap detection in combination with timeouts removes the constraint of having to know the number of messages of a sequence (i.e. the batch size) in advance. Messages must contain a unique sequence number for which a predecessor and a successor is known. For example a message with the sequence number 3 has a predecessor message with the sequence number 2 and a successor message with the sequence number 4. The message sequence 2,3,5 has a gap because the sucessor of 3 is missing. The resequencer therefore has to retain message 5 until message 4 arrives (or a timeout occurs). If the maximum time difference between messages (with successor/predecessor relationship with respect to the sequence number) in a message stream is known, then the resequencer's timeout parameter should be set to this value. In this case it is guaranteed that all messages of a stream are delivered in correct order to the next processor. The lower the timeout value is compared to the out-of-sequence time difference the higher is the probability for out-ofsequence messages delivered by this resequencer. Large timeout values should be supported by sufficiently high capacity values. The capacity parameter is used to prevent the resequencer from running out of memory. By default, the stream resequencer expects long sequence numbers but other sequence numbers types can be supported as well by providing a custom expression. public class MyFileNameExpression implements Expression { public String getFileName(Exchange exchange) { return exchange.getIn().getBody(String.class); } public Object evaluate(Exchange exchange) { // parser the file name with YYYYMMDD-DNNN pattern String fileName = getFileName(exchange); String[] files = fileName.split("-D"); Long answer = Long.parseLong(files[0]) * 1000 + Long.parseLong(files[1]); return answer; } 295 CH A PTE R 9 - PAT T E R N A P P E ND I X public T evaluate(Exchange exchange, Class type) { Object result = evaluate(exchange); return exchange.getContext().getTypeConverter().convertTo(type, result); } } from("direct:start").resequence(new MyFileNameExpression()).stream().timeout(500).to("mock:result"); or custom comparator via the comparator() method ExpressionResultComparator comparator = new MyComparator(); from("direct:start").resequence(header("seqnum")).stream().comparator(comparator).to("mock:result"); or via a StreamResequencerConfig object. ExpressionResultComparator comparator = new MyComparator(); StreamResequencerConfig config = new StreamResequencerConfig(100, 1000L, comparator); from("direct:start").resequence(header("seqnum")).stream(config).to("mock:result"); Using the Spring XML Extensions in.header.seqnum Further Examples For further examples of this pattern in use you could look at the batch-processing resequencer junit test case and the stream-processing resequencer junit test case CH APT ER 9 - PAT T ER N APPEN DIX 296 Using This Pattern If you would like to use this EIP Pattern then please read the Getting Started, you may also find the Architecture useful particularly the description of Endpoint and URIs. Then you could try out some of the Examples first before trying this pattern out. Composed Message Processor The Composed Message Processor from the EIP patterns allows you to process a composite message by splitting it up, routing the sub-messages to appropriate destinations and the reaggregating the responses back into a single message. Available in Camel 1.5. Example In this example we want to check that a multipart order can be filled. Each part of the order requires a check at a different inventory. // split up the order so individual OrderItems can be validated by the appropriate bean from("direct:start") .split().body() .choice() .when().method("orderItemHelper", "isWidget") .to("bean:widgetInventory") .otherwise() .to("bean:gadgetInventory") .end() .to("seda:aggregate"); // collect and re-assemble the validated OrderItems into an order again from("seda:aggregate") .aggregate(new MyOrderAggregationStrategy()).header("orderId").completionTimeout(1000L) .to("mock:result"); To do this we split up the order using a Splitter. The Splitter then sends individual OrderItems to a Content Based Router which checks the item type. Widget items get sent 297 CH A PTE R 9 - PAT T E R N A P P E ND I X for checking in the widgetInventory bean and gadgets get sent to the gadgetInventory bean. Once these OrderItems have been validated by the appropriate bean, they are sent on to the Aggregator which collects and re-assembles the validated OrderItems into an order again. When an order is sent it contains a header with the order id. We use this fact when we aggregate, as we configure this .header("orderId") on the aggregate DSL to instruct Camel to use the header with the key orderId as correlation expression. For full details, check the example source here: camel-core/src/test/java/org/apache/camel/processor/ComposedMessageProcessorTest.java Using This Pattern If you would like to use this EIP Pattern then please read the Getting Started, you may also find the Architecture useful particularly the description of Endpoint and URIs. Then you could try out some of the Examples first before trying this pattern out. Scatter-Gather The Scatter-Gather from the EIP patterns allows you to route messages to a number of dynamically specified recipients and re-aggregate the responses back into a single message. Available in Camel 1.5. Dynamic Scatter-Gather Example In this example we want to get the best quote for beer from several different vendors. We use a dynamic Recipient List to get the request for a quote to all vendors and an Aggregator to pick the best quote out of all the responses. The routes for this are defined as: CH APT ER 9 - PAT T ER N APPEN DIX 298
listOfVendors
quoteRequestId
So in the first route you see that the Recipient List is looking at the listOfVendors header for the list of recipients. So, we need to send a message like Map headers = new HashMap(); headers.put("listOfVendors", "bean:vendor1, bean:vendor2, bean:vendor3"); headers.put("quoteRequestId", "quoteRequest-1"); template.sendBodyAndHeaders("direct:start", "", headers); This message will be distributed to the following Endpoints: bean:vendor1, bean:vendor2, and bean:vendor3. These are all beans which look like public class MyVendor { private int beerPrice; @Produce(uri = "seda:quoteAggregator") private ProducerTemplate quoteAggregator; public MyVendor(int beerPrice) { this.beerPrice = beerPrice; } public void getQuote(@XPath("/quote_request/@item") String item, Exchange exchange) throws Exception { if ("beer".equals(item)) { exchange.getIn().setBody(beerPrice); quoteAggregator.send(exchange); } else { throw new Exception("No quote available for " + item); } } } and are loaded up in Spring like 299 CH A PTE R 9 - PAT T E R N A P P E ND I X 1 2 3 Each bean is loaded with a different price for beer. When the message is sent to each bean endpoint, it will arrive at the MyVendor.getQuote method. This method does a simple check whether this quote request is for beer and then sets the price of beer on the exchange for retrieval at a later step. The message is forwarded on to the next step using POJO Producing (see the @Produce annotation). At the next step we want to take the beer quotes from all vendors and find out which one was the best (i.e. the lowest!). To do this we use an Aggregator with a custom aggregation strategy. The Aggregator needs to be able to compare only the messages from this particular quote; this is easily done by specifying a correlationExpression equal to the value of the quoteRequestId header. As shown above in the message sending snippet, we set this header to quoteRequest-1. This correlation value should be unique or you may include responses that are not part of this quote. To pick the lowest quote out of the set, we use a custom aggregation strategy like public class LowestQuoteAggregationStrategy implements AggregationStrategy { public Exchange aggregate(Exchange oldExchange, Exchange newExchange) { // the first time we only have the new exchange if (oldExchange == null) { return newExchange; } if (oldExchange.getIn().getBody(int.class) < newExchange.getIn().getBody(int.class)) { return oldExchange; } else { return newExchange; } CH APT ER 9 - PAT T ER N APPEN DIX 300 } } Finally, we expect to get the lowest quote of $1 out of $1, $2, and $3. result.expectedBodiesReceived(1); // expect the lowest quote You can find the full example source here: camel-spring/src/test/java/org/apache/camel/spring/processor/scattergather/ camel-spring/src/test/resources/org/apache/camel/spring/processor/scattergather/scattergather.xml Static Scatter-Gather Example You can lock down which recipients are used in the Scatter-Gather by using a static Recipient List. It looks something like this from("direct:start").multicast().to("seda:vendor1", "seda:vendor2", "seda:vendor3"); from("seda:vendor1").to("bean:vendor1").to("seda:quoteAggregator"); from("seda:vendor2").to("bean:vendor2").to("seda:quoteAggregator"); from("seda:vendor3").to("bean:vendor3").to("seda:quoteAggregator"); from("seda:quoteAggregator") .aggregate(header("quoteRequestId"), new LowestQuoteAggregationStrategy()).to("mock:result") A full example of the static Scatter-Gather configuration can be found in the Loan Broker Example. Using This Pattern If you would like to use this EIP Pattern then please read the Getting Started, you may also find the Architecture useful particularly the description of Endpoint and URIs. Then you could try out some of the Examples first before trying this pattern out. Routing Slip The Routing Slip from the EIP patterns allows you to route a message consecutively through a series of processing steps where the sequence of steps is not known at design time and can vary for each message. 301 CH A PTE R 9 - PAT T E R N A P P E ND I X Example The following route will take any messages sent to the Apache ActiveMQ queue SomeQueue and pass them into the Routing Slip pattern. from("activemq:SomeQueue").routingSlip("headerName"); Messages will be checked for the existance of the "headerName" header. The value of this header should be a comma-delimited list of endpoint URIs you wish the message to be routed to. The Message will be routed in a pipeline fashion (i.e. one after the other). Note: In Camel 1.x the default header name routingSlipHeader has been @deprecated and is removed in Camel 2.0. We feel that the DSL needed to express, the header it uses to locate the destinations, directly in the DSL to not confuse readers. So the header name must be provided. Configuration options Here we set the header name and the URI delimiter to something different. Using the Fluent Builders from("direct:c").routingSlip("aRoutingSlipHeader", "#"); Using the Spring XML Extensions CH APT ER 9 - PAT T ER N APPEN DIX 302 Ignore invalid endpoints Available as of Camel 2.3 The Routing Slip now supports ignoreInvalidEndpoints which the Recipient List also supports. You can use it to skip endpoints which is invalid. from("direct:a").routingSlip("myHeader").ignoreInvalidEndpoints(); And in Spring XML its an attribute on the recipient list tag. Then lets say the myHeader contains the following two endpoints direct:foo,xxx:bar. The first endpoint is valid and works. However the 2nd is invalid and will just be ignored. Camel logs at INFO level about, so you can see why the endpoint was invalid. Expression supporting Available as of Camel 2.4 The Routing Slip now supports to take the expression parameter as the Recipient List does. You can tell the camel the expression that you want to use to get the routing slip. from("direct:a").routingSlip(header("myHeader")).ignoreInvalidEndpoints(); And in Spring XML its an attribute on the recipient list tag.
myHeader
Further Examples For further examples of this pattern in use you could look at the routing slip test cases. 303 CH A PTE R 9 - PAT T E R N A P P E ND I X Availability This pattern is available as of Camel version 1.3. Using This Pattern If you would like to use this EIP Pattern then please read the Getting Started, you may also find the Architecture useful particularly the description of Endpoint and URIs. Then you could try out some of the Examples first before trying this pattern out. Throttler The Throttler Pattern allows you to ensure that a specific endpoint does not get overloaded, or that we don't exceed an agreed SLA with some external service. Using the Fluent Builders from("seda:a").throttle(3).timePeriodMillis(10000).to("log:result", "mock:result"); So the above example will throttle messages all messages received on seda:a before being sent to mock:result ensuring that a maximum of 3 messages are sent in any 10 second window. Note that typically you would often use the default time period of a second. So to throttle requests at 100 requests per second between two endpoints it would look more like this... from("seda:a").throttle(100).to("seda:b"); For further examples of this pattern in use you could look at the junit test case Using the Spring XML Extensions Asynchronous delaying Available as of Camel 2.4 You can let the Throttler use non blocking asynchronous delaying, which means Camel will use a scheduler to schedule a task to be executed in the future. The task will then continue routing. This allows the caller thread to not block and be able to service other messages etc. CH APT ER 9 - PAT T ER N APPEN DIX 304 from("seda:a").throttle(100).asyncDelayed().to("seda:b"); Using This Pattern If you would like to use this EIP Pattern then please read the Getting Started, you may also find the Architecture useful particularly the description of Endpoint and URIs. Then you could try out some of the Examples first before trying this pattern out. SAMPLING THROTTLER Available as of Camel 2.1 A sampling throttler allowing you to extract a sample of exchanges from the traffic through a route. It is configured with a sampling period during which only a single exchange is allowed to pass through. All other exchanges will be stopped. Will by default use a sample period of 1 seconds. Samples You use this EIP with the sample DSL as show in these samples. Using the Fluent Builders These samples also show how you can use the different syntax to configure the sampling period: from("direct:sample").sample().to("mock:result"); from("direct:sample-configured").sample(1, TimeUnit.SECONDS).to("mock:result"); from("direct:sample-configured-via-dsl").sample().samplePeriod(1).timeUnits(TimeUnit.SECONDS).to("mock Using the Spring XML Extensions And the same example in Spring XML is: And since it uses a default of 1 second you can omit this configuration in case you also want to use 1 second 305 CH A PTE R 9 - PAT T E R N A P P E ND I X Using This Pattern If you would like to use this EIP Pattern then please read the Getting Started, you may also find the Architecture useful particularly the description of Endpoint and URIs. Then you could try out some of the Examples first before trying this pattern out. See Also ▪ Throttler ▪ Aggregator Delayer The Delayer Pattern allows you to delay the delivery of messages to some destination. Using the Fluent Builders from("seda:b").delay(1000).to("mock:result"); So the above example will delay all messages received on seda:b 1 second before sending them to mock:result. You can of course use many different Expression languages such as XPath, XQuery, SQL or various Scripting Languages. You can just delay things a fixed amount of time from the point at which the delayer receives the message. For example to delay things 2 seconds delayer(2000) The above assume that the delivery order is maintained and that the messages are delivered in delay order. If you want to reorder the messages based on delivery time, you can use the Resequencer with this pattern. For example from("activemq:someQueue").resequencer(header("MyDeliveryTime")).delay("MyRedeliveryTime").to("activem CH APT ER 9 - PAT T ER N APPEN DIX 306 The Delayer in Camel 1.x works a bit differently than Camel 2.0 onwards. In Camel 1.x the expression is used to calculate an absolute time in millis. So if you want to wait 3 sec from now and want to use the expression for that you have to set the absolute time as currentTimeInMillis() + 3000. In Camel 2.0 the expression is a value in millis to wait from the current time, so the expression should just be 3000. However in both Camel 1.x and 2.0 you can use a long value for a fixed value to indicate the delay in millis. See the Spring DSL samples for Delayer in Camel 1.x vs. Camel 2.0. Camel 2.0 - Spring DSL The sample below demonstrates the delay in Spring DSL:
MyDelay
1000
Camel 1.x - Spring DSL The delayer is using slightly different names in Camel 1.x: 3000 The empty tag is needed to fulfill the XSD validation as its an optional element and we use JAXB annotations to generated the XSD in Camel and some combinations is hard to auto generate with optional elements. 307 CH A PTE R 9 - PAT T E R N A P P E ND I X For further examples of this pattern in use you could look at the junit test case Asynchronous delaying Available as of Camel 2.4 You can let the Delayer use non blocking asynchronous delaying, which means Camel will use a scheduler to schedule a task to be executed in the future. The task will then continue routing. This allows the caller thread to not block and be able to service other messages etc. From Java DSL You use the asyncDelayed() to enable the async behavior. from("activemq:queue:foo").delay(1000).asyncDelayed().to("activemq:aDelayedQueue"); From Spring XML You use the asyncDelayed="true" attribute to enable the async behavior. 1000 Creating a custom delay You can use an expression to determine when to send a message using something like this from("activemq:foo"). delay().method("someBean", "computeDelay"). to("activemq:bar"); then the bean would look like this... public class SomeBean { public long computeDelay() { long delay = 0; // use java code to compute a delay value in millis return delay; CH APT ER 9 - PAT T ER N APPEN DIX 308 } } Using This Pattern If you would like to use this EIP Pattern then please read the Getting Started, you may also find the Architecture useful particularly the description of Endpoint and URIs. Then you could try out some of the Examples first before trying this pattern out. Load Balancer The Load Balancer Pattern allows you to delegate to one of a number of endpoints using a variety of different load balancing policies. Build in load balancing policies Camel has out of the box the following policies: Policy Description Round Robin The exchanges is selected in a round robin fashion. This is a well known and classic policy. This spreads the load even. Random A random endpoint is selected for each exchange Sticky Sticky load balancing using an Expression to calculate a correlation key to perform the sticky load balancing; rather like jsessionid in the web or JMSXGroupID in JMS. Topic Topic which sends to all destinations (rather like JMS Topics) Failover Camel 2.0: In case of failures the exchange is tried on the next endpoint. Round Robin Camel 1.x behavior The round robin load balancer can actually be used to failover with Camel 1.x. This is no longer possible in Camel 2.x as the underlying Error Handler foundation has been significantly overhauled in Camel 2.x. Frankly the round robin load balancer in Camel 1.x was not thought to be used in a failover scenario. Camel 2.x behavior The round robin load balancer is not meant to work with failover, for that you should use the dedicated failover load balancer. The round robin load balancer will only change to next endpoint per message. 309 CH A PTE R 9 - PAT T E R N A P P E ND I X The round robin load balancer is statefull as it keeps state which endpoint to use next time. Using the Fluent Builders from("direct:start").loadBalance(). roundRobin().to("mock:x", "mock:y", "mock:z"); Using the Spring configuration So the above example will load balance requests from direct:start to one of the available mock endpoint instances, in this case using a round robbin policy. For further examples of this pattern in use you could look at the junit test case Failover Available as of Camel 2.0 The failover load balancer is capable of trying the next processor in case an Exchange failed with an exception during processing. You can configure the failover with a list of specific exception to only failover. If you do not specify any exceptions it will failover over any exceptions. It uses the same strategy for matching exceptions as the Exception Clause does for the onException. It has the following options: Option Type Default Description CH APT ER 9 - PAT T ER N APPEN DIX 310 inheritErrorHandler maximumFailoverAttempts roundRobin boolean int boolean true Camel 2.3: Whether or not the Error Handler configured on the route should be used or not. You can disable it if you want the failover to trigger immediately and failover to the next endpoint. On the other hand if you have this option enabled, then Camel will first let the Error Handler try to process the message. The Error Handler may have been configured to redelivery and use delays between attempts. If you have enabled a number of redeliveries then Camel will try to redeliver to the same endpoint, and only failover to the next endpoint, when the Error Handler is exhausted. -1 Camel 2.3: A value to indicate after X failver attempts we should exhaust (give up). Use -1 to indicate newer give up and always try to failover. Use 0 to newer failover. And use e.g. 3 to failover at most 3 times before giving up. This option can be used whether or not round robin is enabled or not. false Camel 2.3: Whether or not the failover load balancer should operate in round robin mode or not. If not, then it will always start from the first endpoint when a new message is to be processed. In other words it restart from the top for every message. If round robin is enabled, then it keeps state and will continue with the next endpoint in a round robin fashion. When using round robin it will not stick to last known good endpoint, it will always pick the next endpoint to use. Camel 2.2 or older behavior The current implement of failover load balancer is a simple logic which always tries the first endpoint, and in case of an exception being thrown it tries the next in the list, and so forth. It has no state, and the next message will thus always start with the first endpoint. 311 CH A PTE R 9 - PAT T E R N A P P E ND I X Camel 2.3 onwards behavior The failover load balancer now supports round robin mode, which allows you to failover in a round robin fashion. See the roundRobin option. Here is a sample to failover only if a IOException related exception was thrown: from("direct:start") // here we will load balance if IOException was thrown // any other kind of exception will result in the Exchange as failed // to failover over any kind of exception we can just omit the exception // in the failOver DSL .loadBalance().failover(IOException.class) .to("direct:x", "direct:y", "direct:z"); You can specify multiple exceptions to failover as the option is varargs, for instance: // enable redelivery so failover can react errorHandler(defaultErrorHandler().maximumRedeliveries(5)); from("direct:foo"). loadBalance().failover(IOException.class, MyOtherException.class) .to("direct:a", "direct:b"); Using failover in Spring DSL Failover can also be used from Spring DSL and you configure it as: java.io.IOException com.mycompany.MyOtherException Using failover in round robin mode An example using Java DSL: from("direct:start") // Use failover load balancer in stateful round robin mode // which mean it will failover immediately in case of an exception CH APT ER 9 - PAT T ER N APPEN DIX 312 Redelivery must be enabled In Camel 2.2 or older the failover load balancer requires you have enabled Camel Error Handler to use redelivery. In Camel 2.3 onwards this is not required as such, as you can mix and match. See the inheritErrorHandler option. // as it does NOT inherit error handler. It will also keep retrying as // its configured to newer exhaust. .loadBalance().failover(-1, false, true). to("direct:bad", "direct:bad2", "direct:good", "direct:good2"); And the same example using Spring XML: Using This Pattern If you would like to use this EIP Pattern then please read the Getting Started, you may also find the Architecture useful particularly the description of Endpoint and URIs. Then you could try out some of the Examples first before trying this pattern out. Multicast The Multicast allows to route the same message to a number of endpoints and process them in a different way. The main difference between the Multicast and Splitter is that Splitter will split the message into several pieces but the Multicast will not modify the request message. Example The following example shows how to take a request from the direct:a endpoint , then multicast these request to direct:x, direct:y, direct:z. 313 CH A PTE R 9 - PAT T E R N A P P E ND I X Disabled inheritErrorHandler You can configure inheritErrorHandler=false if you want to failover to the next endpoint as fast as possible. By disabling the Error Handler you ensure it does not intervene which allows the failover load balancer to handle failover asap. By also enabling roundRobin mode, then it will keep retrying until it success. You can then configure the maximumFailoverAttempts option to a high value to let it eventually exhaust (give up) and fail. Using the Fluent Builders from("direct:a").multicast().to("direct:x", "direct:y", "direct:z"); Stop processing in case of exception Available as of Camel 2.1 The Multicast will by default continue to process the entire Exchange even in case one of the multicasted messages will thrown an exception during routing. For example if you want to multicast to 3 destinations and the 2nd destination fails by an exception. What Camel does by default is to process the remainder destinations. You have the chance to remedy or handle this in the AggregationStrategy. But sometimes you just want Camel to stop and let the exception be propagated back, and let the Camel error handler handle it. You can do this in Camel 2.1 by specifying that it should stop in case of an exception occurred. This is done by the stopOnException option as shown below: from("direct:start") .multicast() .stopOnException().to("direct:foo", "direct:bar", "direct:baz") .end() .to("mock:result"); from("direct:foo").to("mock:foo"); from("direct:bar").process(new MyProcessor()).to("mock:bar"); from("direct:baz").to("mock:baz"); And using XML DSL you specify it as follows: CH APT ER 9 - PAT T ER N APPEN DIX 314 Using This Pattern If you would like to use this EIP Pattern then please read the Getting Started, you may also find the Architecture useful particularly the description of Endpoint and URIs. Then you could try out some of the Examples first before trying this pattern out. LOOP The Loop allows to process the a message a number of times and possibly process them in a different way. Useful mostly for testing. For each iteration two properties are set on the Exchange that could be used by processors down the pipeline to process the Message in different ways. Property Description CamelIterationCount Camel 1.x: Total number of iterations to be run CamelIterationIndex Camel 1.x: Index of the current iteration (0 based) CamelLoopSize Camel 2.0: Total number of loops CamelLoopIndex Camel 2.0: Index of the current iteration (0 based) that could be used by processors down the pipeline to process the Message in different ways. 315 CH A PTE R 9 - PAT T E R N A P P E ND I X Examples The following example shows how to take a request from the direct:x endpoint , then send the message repetitively to mock:result. The number of times the message is sent is either passed as an argument to loop(), or determined at runtime by evaluating an expression. The expression must evaluate to an int, otherwise a RuntimeCamelException is thrown. Using the Fluent Builders Pass loop count as an argument from("direct:a").loop(8).to("mock:result"); Use expression to determine loop count from("direct:b").loop(header("loop")).to("mock:result"); Use expression to determine loop count from("direct:c").loop().xpath("/hello/@times").to("mock:result"); Using the Spring XML Extensions Pass loop count as an argument 8 Use expression to determine loop count
loop
For further examples of this pattern in use you could look at one of the junit test case CH APT ER 9 - PAT T ER N APPEN DIX 316 Using This Pattern If you would like to use this EIP Pattern then please read the Getting Started, you may also find the Architecture useful particularly the description of Endpoint and URIs. Then you could try out some of the Examples first before trying this pattern out. MESSAGE TRANSFORMATION Content Enricher Camel supports the Content Enricher from the EIP patterns using a Message Translator, an artibrary Processor in the routing logic or using the enrich DSL element to enrich the message. Content enrichment using a Message Translator or a Processor Using the Fluent Builders You can use Templating to consume a message from one destination, transform it with something like Velocity or XQuery and then send it on to another destination. For example using InOnly (one way messaging) from("activemq:My.Queue"). to("velocity:com/acme/MyResponse.vm"). to("activemq:Another.Queue"); If you want to use InOut (request-reply) semantics to process requests on the My.Queue queue on ActiveMQ with a template generated response, then sending responses back to the JMSReplyTo Destination you could use this. from("activemq:My.Queue"). to("velocity:com/acme/MyResponse.vm"); Here is a simple example using the DSL directly to transform the message body 317 CH A PTE R 9 - PAT T E R N A P P E ND I X from("direct:start").setBody(body().append(" World!")).to("mock:result"); In this example we add our own Processor using explicit Java code from("direct:start").process(new Processor() { public void process(Exchange exchange) { Message in = exchange.getIn(); in.setBody(in.getBody(String.class) + " World!"); } }).to("mock:result"); Finally we can use Bean Integration to use any Java method on any bean to act as the transformer from("activemq:My.Queue"). beanRef("myBeanName", "myMethodName"). to("activemq:Another.Queue"); For further examples of this pattern in use you could look at one of the JUnit tests • TransformTest • TransformViaDSLTest Using Spring XML Content enrichment using the enrich DSL element Camel comes with two flavors of content enricher in the DSL ▪ enrich ▪ pollEnrich enrich is using a Producer to obtain the additional data. It is usually used for Request Reply messaging, for instance to invoke an external web service. poolEnrich on the other hand is using a Polling Consumer to obtain the additional data. It is usually used for Event Message messaging, for instance to read a file or download a FTP file. This feature is available since Camel 2.0 Using the Fluent Builders CH APT ER 9 - PAT T ER N APPEN DIX 318 AggregationStrategy aggregationStrategy = ... from("direct:start") .enrich("direct:resource", aggregationStrategy) .to("direct:result"); from("direct:resource") ... The content enricher (enrich) retrieves additional data from a resource endpoint in order to enrich an incoming message (contained in the orginal exchange). An aggregation strategy is used to combine the original exchange and the resource exchange. The first parameter of the AggregationStrategy.aggregate(Exchange, Exchange) method corresponds to the the original exchange, the second parameter the resource exchange. The results from the resource endpoint are stored in the resource exchange's out-message. Here's an example template for implementing an aggregation strategy. public class ExampleAggregationStrategy implements AggregationStrategy { public Exchange aggregate(Exchange original, Exchange resource) { Object originalBody = original.getIn().getBody(); Object resourceResponse = resource.getOut().getBody(); Object mergeResult = ... // combine original body and resource response if (original.getPattern().isOutCapable()) { original.getOut().setBody(mergeResult); } else { original.getIn().setBody(mergeResult); } return original; } } Using this template the original exchange can be of any pattern. The resource exchange created by the enricher is always an in-out exchange. Using Spring XML The same example in the Spring DSL ... 319 CH A PTE R 9 - PAT T E R N A P P E ND I X Aggregation strategy is optional The aggregation strategy is optional. If you do not provide it Camel will by default just use the body obtained from the resource. from("direct:start") .enrich("direct:resource") .to("direct:result"); In the route above the message send to the direct:result endpoint will contain the output from the direct:resource as we do not use any custom aggregation. And in Spring DSL you just omit the strategyRef attribute: Content enrich using pollEnrich The pollEnrich works just as the enrich however as it uses a Polling Consumer we have 3 methods when polling ▪ receive ▪ receiveNoWait ▪ receive(timeout) By default Camel will use the receiveNoWait. If there is no data then the newExchange in the aggregation strategy is null. You can pass in a timeout value that determines which method to use ▪ timeout is -1 or negative then receive is selected ▪ timeout is 0 then receiveNoWait is selected ▪ otherwise receive(timeout) is selected The timeout values is in millis. The sample below reads a file based on a JMS message that contains a header with the filename. CH APT ER 9 - PAT T ER N APPEN DIX 320 Data from current Exchange not used pollEnrich does not access any data from the current Exchange which means when polling it cannot use any of the existing headers you may have set on the Exchange. For example you cannot set a filename in the Exchange.FILE_NAME header and use pollEnrich to consume only that file. For that you must set the filename in the endpoint URI. from("activemq:queue:order") .setHeader(Exchange.FILE_NAME, header("orderId")) .pollEnrich("file://order/data/additional") .to("bean:processOrder"); And if we want to wait at most 20 seconds for the file to be ready we can use a timeout: from("activemq:queue:order") .setHeader(Exchange.FILE_NAME, header("orderId")) .pollEnrich("file://order/data/additional", 20000) .to("bean:processOrder"); And yes pollEnrich also supports the aggregation strategy so we can pass it in as an argument too: .pollEnrich("file://order/data/additional", 20000, aggregationStrategy) Using This Pattern If you would like to use this EIP Pattern then please read the Getting Started, you may also find the Architecture useful particularly the description of Endpoint and URIs. Then you could try out some of the Examples first before trying this pattern out. Content Filter Camel supports the Content Filter from the EIP patterns using one of the following mechanisms in the routing logic to transform content from the inbound message. • Message Translator • invoking a Java bean • Processor object 321 CH A PTE R 9 - PAT T E R N A P P E ND I X A common way to filter messages is to use an Expression in the DSL like XQuery, SQL or one of the supported Scripting Languages. Using the Fluent Builders Here is a simple example using the DSL directly from("direct:start").setBody(body().append(" World!")).to("mock:result"); In this example we add our own Processor from("direct:start").process(new Processor() { public void process(Exchange exchange) { Message in = exchange.getIn(); in.setBody(in.getBody(String.class) + " World!"); } }).to("mock:result"); For further examples of this pattern in use you could look at one of the JUnit tests • TransformTest • TransformViaDSLTest Using Spring XML You can also use XPath to filter out part of the message you are interested in: //foo:bar CH APT ER 9 - PAT T ER N APPEN DIX 322 Using This Pattern If you would like to use this EIP Pattern then please read the Getting Started, you may also find the Architecture useful particularly the description of Endpoint and URIs. Then you could try out some of the Examples first before trying this pattern out. Claim Check The Claim Check from the EIP patterns allows you to replace message content with a claim check (a unique key), which can be used to retrieve the message content at a later time. The message content is stored temporarily in a persistent store like a database or file system. This pattern is very useful when message content is very large (thus it would be expensive to send around) and not all components require all information. It can also be useful in situations where you cannot trust the information with an outside party; in this case, you can use the Claim Check to hide the sensitive portions of data. Available in Camel 1.5. Example In this example we want to replace a message body with a claim check, and restore the body at a later step. from("direct:start").to("bean:checkLuggage", "mock:testCheckpoint", "bean:dataEnricher", "mock:result"); The example route is pretty simple - its just a Pipeline. In a real application you would have some other steps where the mock:testCheckpoint endpoint is in the example. The message is first sent to the checkLuggage bean which looks like public static final class CheckLuggageBean { public void checkLuggage(Exchange exchange, @Body String body, @XPath("/order/ @custId") String custId) { // store the message body into the data store, using the custId as the claim check 323 CH A PTE R 9 - PAT T E R N A P P E ND I X dataStore.put(custId, body); // add the claim check as a header exchange.getIn().setHeader("claimCheck", custId); // remove the body from the message exchange.getIn().setBody(null); } } This bean stores the message body into the data store, using the custId as the claim check. In this example, we're just using a HashMap to store the message body; in a real application you would use a database or file system, etc. Next the claim check is added as a message header for use later. Finally we remove the body from the message and pass it down the pipeline. The next step in the pipeline is the mock:testCheckpoint endpoint which is just used to check that the message body is removed, claim check added, etc. To add the message body back into the message, we use the dataEnricher bean which looks like public static final class DataEnricherBean { public void addDataBackIn(Exchange exchange, @Header("claimCheck") String claimCheck) { // query the data store using the claim check as the key and add the data // back into the message body exchange.getIn().setBody(dataStore.get(claimCheck)); // remove the message data from the data store dataStore.remove(claimCheck); // remove the claim check header exchange.getIn().removeHeader("claimCheck"); } } This bean queries the data store using the claim check as the key and then adds the data back into the message. The message body is then removed from the data store and finally the claim check is removed. Now the message is back to what we started with! For full details, check the example source here: camel-core/src/test/java/org/apache/camel/processor/ClaimCheckTest.java Using This Pattern If you would like to use this EIP Pattern then please read the Getting Started, you may also find the Architecture useful particularly the description of Endpoint and URIs. Then you could try out some of the Examples first before trying this pattern out. CH APT ER 9 - PAT T ER N APPEN DIX 324 Normalizer Camel supports the Normalizer from the EIP patterns by using a Message Router in front of a number of Message Translator instances. Example This example shows a Message Normalizer that converts two types of XML messages into a common format. Messages in this common format are then filtered. Using the Fluent Builders // we need to normalize two types of incoming messages from("direct:start") .choice() .when().xpath("/employee").to("bean:normalizer?method=employeeToPerson") .when().xpath("/customer").to("bean:normalizer?method=customerToPerson") .end() .to("mock:result"); In this case we're using a Java bean as the normalizer. The class looks like this public class MyNormalizer { public void employeeToPerson(Exchange exchange, @XPath("/employee/name/text()") String name) { exchange.getOut().setBody(createPerson(name)); } public void customerToPerson(Exchange exchange, @XPath("/customer/@name") String name) { exchange.getOut().setBody(createPerson(name)); } private String createPerson(String name) { return ""; } } Using the Spring XML Extensions 325 CH A PTE R 9 - PAT T E R N A P P E ND I X The same example in the Spring DSL /employee /customer See Also • Message Router • Content Based Router • Message Translator Using This Pattern If you would like to use this EIP Pattern then please read the Getting Started, you may also find the Architecture useful particularly the description of Endpoint and URIs. Then you could try out some of the Examples first before trying this pattern out. SORT Available as of Camel 2.0 Sort can be used to sort a message. Imagine you consume text files and before processing each file you want to be sure the content is sorted. Sort will by default sort the body using a default comparator that handles numeric values or uses the string representation. You can provide your own comparator, and even an expression to return the value to be sorted. Sort requires the value returned from the expression evaluation is convertible to java.util.List as this is required by the JDK sort operation. CH APT ER 9 - PAT T ER N APPEN DIX 326 Using from Java DSL In the route below it will read the file content and tokenize by line breaks so each line can be sorted. from("file://inbox").sort(body().tokenize("\n")).to("bean:MyServiceBean.processLine"); You can pass in your own comparator as a 2nd argument: from("file://inbox").sort(body().tokenize("\n"), new MyReverseComparator()).to("bean:MyServiceBean.processLine"); Using from Spring DSL In the route below it will read the file content and tokenize by line breaks so each line can be sorted. And to use our own comparator we can refer to it as a spring bean: Using This Pattern If you would like to use this EIP Pattern then please read the Getting Started, you may also find the Architecture useful particularly the description of Endpoint and URIs. Then you could try out some of the Examples first before trying this pattern out. 327 CH A PTE R 9 - PAT T E R N A P P E ND I X MESSAGING ENDPOINTS Messaging Mapper Camel supports the Messaging Mapper from the EIP patterns by using either Message Translator pattern or the Type Converter module. See also • • • • • Message Translator Type Converter CXF for JAX-WS support for binding business logic to messaging & web services Pojo Bean Using This Pattern If you would like to use this EIP Pattern then please read the Getting Started, you may also find the Architecture useful particularly the description of Endpoint and URIs. Then you could try out some of the Examples first before trying this pattern out. Event Driven Consumer Camel supports the Event Driven Consumer from the EIP patterns. The default consumer model is event based (i.e. asynchronous) as this means that the Camel container can then manage pooling, threading and concurrency for you in a declarative manner. The Event Driven Consumer is implemented by consumers implementing the Processor interface which is invoked by the Message Endpoint when a Message is available for processing. For more details see • Message • Message Endpoint CH APT ER 9 - PAT T ER N APPEN DIX 328 Using This Pattern If you would like to use this EIP Pattern then please read the Getting Started, you may also find the Architecture useful particularly the description of Endpoint and URIs. Then you could try out some of the Examples first before trying this pattern out. Polling Consumer Camel supports implementing the Polling Consumer from the EIP patterns using the PollingConsumer interface which can be created via the Endpoint.createPollingConsumer() method. So in your Java code you can do Endpoint endpoint = context.getEndpoint("activemq:my.queue"); PollingConsumer consumer = endpoint.createPollingConsumer(); Exchange exchange = consumer.receive(); Notice in Camel 2.0 we have introduced the ConsumerTemplate. There are 3 main polling methods on PollingConsumer Method name Description receive() Waits until a message is available and then returns it; potentially blocking forever receive(long) Attempts to receive a message exchange, waiting up to the given timeout and returning null if no message exchange could be received within the time available receiveNoWait() Attempts to receive a message exchange immediately without waiting and returning null if a message exchange is not available yet ConsumerTemplate Available as of Camel 2.0 The ConsumerTemplate is a template much like Spring's JmsTemplate or JdbcTemplate supporting the Polling Consumer EIP. With the template you can consume Exchanges from an Endpoint. 329 CH A PTE R 9 - PAT T E R N A P P E ND I X The template supports the 3 operations above, but also including convenient methods for returning the body, etc consumeBody. The example from above using ConsumerTemplate is: Exchange exchange = consumerTemplate.receive("activemq:my.queue"); Or to extract and get the body you can do: Object body = consumerTemplate.receiveBody("activemq:my.queue"); And you can provide the body type as a parameter and have it returned as the type: String body = consumerTemplate.receiveBody("activemq:my.queue", String.class); You get hold of a ConsumerTemplate from the CamelContext with the createConsumerTemplate operation: ConsumerTemplate consumer = context.createConsumerTemplate(); Using ConsumerTemplate with Spring DSL With the Spring DSL we can declare the consumer in the CamelContext with the consumerTemplate tag, just like the ProducerTemplate. The example below illustrates this: