Welcome to the Spring factories for Elasticsearch project.
Actually, since version 1.4.1, this project has been split in two parts:
- Elasticsearch Beyonder which find resources in project classpath to automatically create indices, types and templates.
- This project which is building Client beans using Spring framework.
From 5.0, this project provides 2 implementations of an elasticsearch Client:
- The REST client
- The Transport client (deprecated)
From 6.0, the REST client implementation has been replaced by a High Level REST client. It now also supports X-Pack for official security.
- For 6.x elasticsearch versions, you are reading the latest documentation.
- For 5.x elasticsearch versions, look at es-5.x branch.
- For 2.x elasticsearch versions, look at es-2.x branch.
- For 1.x elasticsearch versions, look at es-1.4 branch.
- For 0.x elasticsearch versions, look at 0.x branch.
spring-elasticsearch | elasticsearch | Spring | Release date |
---|---|---|---|
6.2-SNAPSHOT | 6.0 - 6.x | 5.0.7 | |
6.1 | 6.0 - 6.x | 5.0.7 | 2018-07-22 |
6.0 | 6.0 - 6.x | 5.0.3 | 2018-02-08 |
5.0 | 5.0 - 5.x | 4.3.10 | 2018-02-04 |
2.2.0 | 2.0 - 2.4 | 4.2.3 | 2017-03-09 |
2.1.0 | 2.0, 2.1 | 4.2.3 | 2015-11-25 |
2.0.0 | 2.0 | 4.1.4 | 2015-10-25 |
1.4.2 | < 2.0 | 4.1.4 | 2015-03-03 |
1.4.1 | 1.4 | 4.1.4 | 2015-02-28 |
1.4.0 | 1.4 | 4.1.4 | 2015-01-03 |
1.3.0 | 1.3 | 4.0.6 | 2014-09-01 |
1.0.0 | 1.0 | 3.2.2 | 2014-02-14 |
Thanks to Travis for the build status:
Import spring-elasticsearch in you project pom.xml
file:
<dependency>
<groupId>fr.pilato.spring</groupId>
<artifactId>spring-elasticsearch</artifactId>
<version>6.1</version>
</dependency>
If you want to set a specific version of the High Level Rest client, add it to your pom.xml
file:
<dependency>
<groupId>org.elasticsearch.client</groupId>
<artifactId>elasticsearch-rest-high-level-client</artifactId>
<version>6.3.1</version>
</dependency>
If you want to use a transport client (deprecated), you must add it to your pom.xml
file:
<dependency>
<groupId>org.elasticsearch.client</groupId>
<artifactId>transport</artifactId>
<version>6.3.1</version>
</dependency>
If you want to use a transport client secured with X-Pack (deprecated), you must add it to your pom.xml
file:
<dependency>
<groupId>org.elasticsearch.client</groupId>
<artifactId>x-pack-transport</artifactId>
<version>6.3.1</version>
</dependency>
Note that you'd probably to add also the elastic maven repository:
<repositories>
<repository>
<id>elastic-download-service</id>
<name>Elastic Download Service</name>
<url>https://artifacts.elastic.co/maven/</url>
<releases><enabled>true</enabled></releases>
<snapshots><enabled>false</enabled></snapshots>
</repository>
</repositories>
If you want to try out the most recent SNAPSHOT version deployed on Sonatype:
<dependency>
<groupId>fr.pilato.spring</groupId>
<artifactId>spring-elasticsearch</artifactId>
<version>6.1-SNAPSHOT</version>
</dependency>
Don't forget to add if needed the following repository in your pom.xml
:
<repositories>
<repository>
<id>oss-snapshots</id>
<name>Sonatype OSS Snapshots</name>
<url>https://oss.sonatype.org/content/repositories/snapshots/</url>
<releases><enabled>false</enabled></releases>
<snapshots><enabled>true</enabled></snapshots>
</repository>
</repositories>
If you depend on an elasticsearch SNAPSHOT version, you need to add the following repository to your pom.xml
:
<repositories>
<repository>
<id>elastic-snapshots</id>
<name>Elastic Snapshots</name>
<url>http://snapshots.elastic.co/maven/</url>
<releases><enabled>false</enabled></releases>
<snapshots><enabled>true</enabled></snapshots>
</repository>
</repositories>
We are using slf4j for logging but you have to provide the logging implementation you want to use and bind it.
For example for this project we are using for tests log4j2.
If you want to do so, add to your pom.xml
:
<dependency>
<groupId>org.apache.logging.log4j</groupId>
<artifactId>log4j-1.2-api</artifactId>
<version>2.11.0</version>
</dependency>
<dependency>
<groupId>org.apache.logging.log4j</groupId>
<artifactId>log4j-slf4j-impl</artifactId>
<version>2.11.0</version>
</dependency>
<dependency>
<groupId>org.apache.logging.log4j</groupId>
<artifactId>log4j-core</artifactId>
<version>2.11.0</version>
</dependency>
In your spring context file, just add namespaces like this:
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:elasticsearch="http://www.pilato.fr/schema/elasticsearch"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.0.xsd
http://www.pilato.fr/schema/elasticsearch http://www.pilato.fr/schema/elasticsearch/elasticsearch-6.0.xsd">
</beans>
From 5.0, you can now get a REST Client implementation.
In your spring context file, just define a client like this:
<elasticsearch:rest-client id="esClient" />
By default, you will get an Elasticsearch High Level Rest Client
connected to an Elasticsearch node already running at localhost:9200
.
You can set the nodes you want to connect to:
<elasticsearch:rest-client id="esClient" esNodes="localhost:9200,localhost:9201" />
You can use the rest client in your java classes.
import org.elasticsearch.client.RestHighLevelClient;
RestHighLevelClient client = ctx.getBean("esClient", RestHighLevelClient.class);
Better, you should use @Autowired
annotation.
// Inject your client...
@Autowired RestHighLevelClient client;
You need to define the xpack.security.user
property as follows:
<util:properties id="esProperties">
<prop key="xpack.security.user">elastic:changeme</prop>
</util:properties>
<elasticsearch:rest-client id="esClient" properties="esProperties" />
From 5.0, the Transport Client implementation is deprecated. It is now marked as optional
so
you need to explicitly add it to your project as described in Maven dependency chapter.
In your spring context file, just define a client like this:
<elasticsearch:client id="esClient" />
By default, you will get an Elasticsearch Transport Client
connected to an Elasticsearch node already running at localhost:9300
using elasticsearch
as cluster name.
You can set the nodes you want to connect to:
<elasticsearch:client id="esClient" esNodes="localhost:9300,localhost:9301" />
Important notes
Note that you should define the same clustername as the one you defined on your running nodes. Otherwise, your Transport Client won't connect to the node. See Elasticsearch properties.
Note also that you must define the transport client port (9300-9399) and not the REST port (9200-9299). Transport client does not use REST API.
Now, you can use the transport client in your java classes.
import org.elasticsearch.client.Client;
Client client = ctx.getBean("esClient", Client.class);
Better, you should use @Autowired
annotation.
// Inject your client...
@Autowired Client client;
You can define your transport client properties using a property file such as:
cluster.name=myclustername
And load it in Spring context:
<util:properties id="esproperties"
location="classpath:fr/pilato/spring/elasticsearch/xml/esclient-transport.properties"/>
Note that you can also define properties as follow:
<util:map id="esProperties">
<entry key="cluster.name" value="newclustername"/>
</util:map>
<elasticsearch:client id="esClient" esNodes="localhost:9300,localhost:9301" properties="esproperties"/>
Injecting properties in client is now easy:
<elasticsearch:client id="esClient" properties="esproperties" />
You can also add plugins to the transport client in case it needs it:
<elasticsearch:client id="esClient" plugins="org.elasticsearch.plugin.deletebyquery.DeleteByQueryPlugin" />
You need to define the xpack.security.user
property as follows:
<util:properties id="esProperties">
<prop key="xpack.security.user">elastic:changeme</prop>
</util:properties>
<elasticsearch:client id="esClient" properties="esProperties" />
Note that it needs that you imported to your project the x-pack-transport
jar.
Client bean initialization is by default synchronously. It can be initialized asynchronously with the attributes async
and taskExecutor
.
<task:executor pool-size="4" id="taskExecutor"/>
<elasticsearch:client id="esClient" async="true" taskExecutor="taskExecutor"/>
Asynchronous initialization does not block Spring startup but it continues on background on another thread.
Any methods call to these beans before elasticsearch is initialized will be blocked. taskExecutor
references a standard Spring's task executor.
The following examples are documented using the Rest Client implementation elasticsearch:rest-client
but you can
replace them with the Transport client by using elasticsearch:client
instead.
If you want to manage indexes and types at startup (creating missing indexes/types and applying mappings):
<elasticsearch:rest-client id="esClient"
mappings="twitter/tweet" />
This will create an Elasticsearch Low Level Rest Client
that will check when starting that index twitter
exists and tweet
type is defined.
If you need to manage more than one type or index, just use a comma separated list:
<elasticsearch:rest-client id="esClient"
mappings="twitter/tweet,twitter/user,facebook/user" />
If you add in your classpath a file named es/twitter/_settings.json
, it will be automatically applied to define
settings for your twitter
index.
For example, create the following file src/main/resources/es/twitter/_settings.json
in your project:
{
"index" : {
"number_of_shards" : 3,
"number_of_replicas" : 2
}
}
Also, if you define a file named es/twitter/tweet.json
, it will be automatically applied as the mapping for
the tweet
type in the twitter
index.
For example, create the following file src/main/resources/es/twitter/tweet.json
in your project:
{
"tweet" : {
"properties" : {
"message" : {"type" : "text", "store" : "yes"}
}
}
}
By default, the factory will find every mapping file located under es
directory.
So, if you have a mapping file named es/twitter/tweet.json
in your classpath, it will be automatically used by
the factory without defining anything:
<elasticsearch:rest-client id="esClient" />
You can disable this automatic lookup by setting the autoscan
property to false
:
<elasticsearch:rest-client id="esClient" autoscan="false" mappings="twitter/tweet" />
When creating an index, it could be useful to add an alias on it.
For example, if you planned to have indexes per year for twitter feeds (twitter2012, twitter2013, twitter2014) and you want
to define an alias named twitter, you can use the aliases
property:
<elasticsearch:rest-client id="esClient"
aliases="twitter:twitter2012,twitter:twitter2013,twitter:twitter2014" />
Sometimes it's useful to define a template mapping that will automatically be applied to new indices created.
For example, if you planned to have indexes per year for twitter feeds (twitter2012, twitter2013, twitter2014) and you want
to define a template named twitter_template
, you can use the templates
property:
<!--
We add also a facebook_template template just for showing how to
define more than one template...
-->
<elasticsearch:rest-client id="esClient"
templates="twitter_template,facebook_template" />
To configure your template you have to define a file named es/_template/twitter_template.json
in your project:
{
"template" : "twitter*",
"settings" : {
"number_of_shards" : 1
},
"mappings" : {
"tweet" : {
"properties" : {
"message" : {
"type" : "text",
"store" : "yes"
}
}
}
}
}
By default, the factory look in es
classpath folder to find if there is index settings or mappings definitions.
If you need to change it, you can use the classpathRoot
property:
<elasticsearch:rest-client id="esClient" classpathRoot="myownfolder" />
So, if a myownfolder/twitter/_settings.xml
file exists in your classpath, it will be used by the factory.
If you need to merge mapping for an existing type
, set mergeMapping
property to true
.
<elasticsearch:rest-client id="esClient" mergeMapping="true" />
If merging fails, the factory will not start (BeanCreationException will be raised).
If you need to merge settings for an existing index
, add a file named es/twitter/_update_settings.json
in your
classpath. The factory will detect it and will try to merge settings unless you explicitly set mergeSettings
to false
.
<elasticsearch:rest-client id="esClient" mergeSettings="false" />
If merging fails, the factory will not start.
For test purpose or for continuous integration, you could force the factory to clean the previous indices
when starting the client.
It will remove all your datas for every index which has been defined. Just set forceMapping
property to true
.
<elasticsearch:rest-client id="esClient" forceMapping="true" />
For test purpose or for continuous integration, you could force the factory to clean the previous template
when starting the client.
Just set forceTemplate
property to true
.
<elasticsearch:rest-client id="esClient" forceTemplate="true" />
Let's say you want to use Spring Java Annotations, here is a typical application you can build.
pom.xml
:
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<groupId>fr.pilato.tests</groupId>
<artifactId>spring-elasticsearch-test</artifactId>
<version>1.0-SNAPSHOT</version>
<dependencies>
<dependency>
<groupId>fr.pilato.spring</groupId>
<artifactId>spring-elasticsearch</artifactId>
<version>6.1</version>
</dependency>
</dependencies>
</project>
App.java
:
package fr.pilato.tests;
import fr.pilato.spring.elasticsearch.ElasticsearchRestClientFactoryBean;
import org.elasticsearch.client.RestHighLevelClient;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.annotation.AnnotationConfigApplicationContext;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.stereotype.Component;
import java.io.IOException;
@Component
public class RestApp {
@Configuration
public class AppConfig {
@Bean
public RestHighLevelClient esClient() throws Exception {
ElasticsearchRestClientFactoryBean factory = new ElasticsearchRestClientFactoryBean();
factory.setEsNodes(new String[]{"127.0.0.1:9200"});
// Begin: If you are running with x-pack
Properties props = new Properties();
props.setProperty("xpack.security.user", "elastic:changeme");
factory.setProperties(props);
// End: If you are running with x-pack
factory.afterPropertiesSet();
return factory.getObject();
}
}
@Autowired
private RestClient client;
public static void main(String[] args) throws IOException {
AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext();
context.scan("fr.pilato.tests");
context.refresh();
RestApp p = context.getBean(RestApp.class);
p.run();
context.close();
}
private void run() throws IOException {
// Run a High Level request
client.info();
// You still have access to the Low Level client
client.getLowLevel().performRequest("GET", "/");
}
}
Note that you can use the old fashion method to define your beans instead of using <elasticsearch:...>
namespace:
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:util="http://www.springframework.org/schema/util"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.0.xsd
http://www.springframework.org/schema/util http://www.springframework.org/schema/util/spring-util-3.0.xsd">
<util:map id="esproperties">
<entry key="cluster.name" value="newclustername"/>
</util:map>
<!-- The Rest Client -->
<bean id="esRestClient" class="fr.pilato.spring.elasticsearch.ElasticsearchRestClientFactoryBean" >
<property name="esNodes">
<list>
<value>localhost:9200</value>
<value>localhost:9201</value>
</list>
</property>
<property name="autoscan" value="false" />
<property name="mappings">
<list>
<value>twitter/tweet</value>
</list>
</property>
<property name="classpathRoot" value="myownfolder" />
<property name="forceMapping" value="true" />
<property name="mergeSettings" value="true" />
<property name="templates">
<list>
<value>twitter_template</value>
</list>
</property>
<property name="forceTemplate" value="true" />
<property name="aliases">
<list>
<value>twitter:twitter2012</value>
<value>twitter:twitter2013</value>
<value>twitter:twitter2014</value>
</list>
</property>
</bean>
<!-- The deprecated Transport Client -->
<bean id="esTransportClient" class="fr.pilato.spring.elasticsearch.ElasticsearchTransportClientFactoryBean" >
<property name="esNodes">
<list>
<value>localhost:9300</value>
<value>localhost:9301</value>
</list>
</property>
<property name="properties" ref="esproperties" />
<property name="autoscan" value="false" />
<property name="mappings">
<list>
<value>twitter/tweet</value>
</list>
</property>
<property name="classpathRoot" value="myownfolder" />
<property name="forceMapping" value="true" />
<property name="mergeSettings" value="true" />
<property name="templates">
<list>
<value>twitter_template</value>
</list>
</property>
<property name="forceTemplate" value="true" />
<property name="aliases">
<list>
<value>twitter:twitter2012</value>
<value>twitter:twitter2013</value>
<value>twitter:twitter2014</value>
</list>
</property>
</bean>
</beans>
Special thanks to
- Nicolas Huray for his contribution about templates
- Nicolas Labrot for his contribution about async
If you want to run tests (integration tests) from your IDE, you need to start first an elasticsearch instance.
If you are not using x-pack, then just run the tests from your IDE. Tests are expecting a node running at localhost:9200
.
If you are using x-pack, tests are expecting a user named elastic
with changeme
as the password.
You can set this user by running bin/x-pack/setup-passwords interactive
.
To run the tests using Maven (on the CLI), just run:
mvn clean install
Note that when the tests are launched with maven, they are not running with x-pack yet. To run tests against x-pack, you need to start elasticsearch with x-pack manually and run the tests with:
mvn clean install -Px-pack
To release the project you need to run the release plugin with the release
profile as you need to sign the artifacts:
mvn release:prepare
git push --tags
git push
mvn release:perform -Prelease
If you need to skip the tests, run:
mvn release:perform -Prelease -Darguments="-DskipTests"
To announce the release, run:
cd target/checkout
# Run the following command if you want to check the announcement email
mvn changes:announcement-generate
cat target/announcement/announcement.vm
# Announce the release (change your smtp username and password)
mvn changes:announcement-mail -Dchanges.username='YourSmtpUserName' -Dchanges.password='YourSmtpUserPassword'
This software is licensed under the Apache 2 license, quoted below.
Copyright 2011-2018 David Pilato
Licensed under the Apache License, Version 2.0 (the "License"); you may not
use this file except in compliance with the License. You may obtain a copy of
the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
License for the specific language governing permissions and limitations under
the License.