BAEL-6538: Geospatial Support in ElasticSearch (#14554)
* Migrated to spring-boot and created the first working test. * JAVA-6538: Refactor Java GeoQueries to Java Client for Ealastic.
This commit is contained in:
parent
4609814e3c
commit
677f8ae2d0
|
@ -8,77 +8,56 @@
|
||||||
|
|
||||||
<parent>
|
<parent>
|
||||||
<groupId>com.baeldung</groupId>
|
<groupId>com.baeldung</groupId>
|
||||||
<artifactId>parent-spring-5</artifactId>
|
<artifactId>parent-boot-3</artifactId>
|
||||||
<version>0.0.1-SNAPSHOT</version>
|
<version>0.0.1-SNAPSHOT</version>
|
||||||
<relativePath>../../parent-spring-5</relativePath>
|
<relativePath>../../parent-boot-3</relativePath>
|
||||||
</parent>
|
</parent>
|
||||||
|
|
||||||
<dependencies>
|
<dependencies>
|
||||||
<dependency>
|
|
||||||
<groupId>org.springframework</groupId>
|
|
||||||
<artifactId>spring-web</artifactId>
|
|
||||||
<version>${spring.version}</version>
|
|
||||||
</dependency>
|
|
||||||
<dependency>
|
<dependency>
|
||||||
<groupId>org.springframework.data</groupId>
|
<groupId>org.springframework.data</groupId>
|
||||||
<artifactId>spring-data-elasticsearch</artifactId>
|
<artifactId>spring-data-elasticsearch</artifactId>
|
||||||
<version>${spring-data-elasticsearch.version}</version>
|
<version>${spring-data-elasticsearch.version}</version>
|
||||||
</dependency>
|
</dependency>
|
||||||
<dependency>
|
<!--These two dependencies are inherited from spring-data-elasticsearch. Overriding to the newer version
|
||||||
<groupId>org.elasticsearch</groupId>
|
will break tests. -->
|
||||||
<artifactId>elasticsearch</artifactId>
|
<!-- <dependency>-->
|
||||||
<version>${elasticsearch.version}</version>
|
<!-- <groupId>co.elastic.clients</groupId>-->
|
||||||
</dependency>
|
<!-- <artifactId>elasticsearch-java</artifactId>-->
|
||||||
<dependency>
|
<!-- <version>${elasticsearch.version}</version>-->
|
||||||
<groupId>com.alibaba</groupId>
|
<!-- </dependency>-->
|
||||||
<artifactId>fastjson</artifactId>
|
<!-- <dependency>-->
|
||||||
<version>${fastjson.version}</version>
|
<!-- <groupId>com.fasterxml.jackson.core</groupId>-->
|
||||||
</dependency>
|
<!-- <artifactId>jackson-databind</artifactId>-->
|
||||||
|
<!-- <version>${jackson.version}</version>-->
|
||||||
|
<!-- </dependency>-->
|
||||||
|
<!-- Here for backward compatibility-->
|
||||||
<dependency>
|
<dependency>
|
||||||
<groupId>org.elasticsearch.client</groupId>
|
<groupId>org.elasticsearch.client</groupId>
|
||||||
<artifactId>elasticsearch-rest-high-level-client</artifactId>
|
<artifactId>elasticsearch-rest-high-level-client</artifactId>
|
||||||
<version>7.17.11</version>
|
<version>7.17.11</version>
|
||||||
</dependency>
|
</dependency>
|
||||||
<dependency>
|
|
||||||
<groupId>com.fasterxml.jackson.core</groupId>
|
|
||||||
<artifactId>jackson-databind</artifactId>
|
|
||||||
<version>${jackson.version}</version>
|
|
||||||
</dependency>
|
|
||||||
<dependency>
|
<dependency>
|
||||||
<groupId>org.projectlombok</groupId>
|
<groupId>org.projectlombok</groupId>
|
||||||
<artifactId>lombok</artifactId>
|
<artifactId>lombok</artifactId>
|
||||||
<version>1.18.28</version>
|
<version>1.18.28</version>
|
||||||
</dependency>
|
</dependency>
|
||||||
<dependency>
|
<dependency>
|
||||||
<groupId>org.locationtech.spatial4j</groupId>
|
<groupId>org.springframework.boot</groupId>
|
||||||
<artifactId>spatial4j</artifactId>
|
<artifactId>spring-boot-autoconfigure</artifactId>
|
||||||
<version>${spatial4j.version}</version>
|
|
||||||
</dependency>
|
|
||||||
<dependency>
|
|
||||||
<groupId>org.locationtech.jts</groupId>
|
|
||||||
<artifactId>jts-core</artifactId>
|
|
||||||
<version>${jts.version}</version>
|
|
||||||
<exclusions>
|
|
||||||
<exclusion>
|
|
||||||
<groupId>xerces</groupId>
|
|
||||||
<artifactId>xercesImpl</artifactId>
|
|
||||||
</exclusion>
|
|
||||||
</exclusions>
|
|
||||||
</dependency>
|
|
||||||
<dependency>
|
|
||||||
<groupId>org.springframework</groupId>
|
|
||||||
<artifactId>spring-test</artifactId>
|
|
||||||
<version>${spring.version}</version>
|
|
||||||
<scope>test</scope>
|
|
||||||
</dependency>
|
</dependency>
|
||||||
</dependencies>
|
</dependencies>
|
||||||
|
<build>
|
||||||
|
<plugins>
|
||||||
|
<plugin>
|
||||||
|
<groupId>org.springframework.boot</groupId>
|
||||||
|
<artifactId>spring-boot-maven-plugin</artifactId>
|
||||||
|
</plugin>
|
||||||
|
</plugins>
|
||||||
|
</build>
|
||||||
<properties>
|
<properties>
|
||||||
<spring-data-elasticsearch.version>5.1.2</spring-data-elasticsearch.version>
|
<spring-data-elasticsearch.version>5.1.2</spring-data-elasticsearch.version>
|
||||||
<elasticsearch.version>8.9.0</elasticsearch.version>
|
<elasticsearch.version>8.9.0</elasticsearch.version>
|
||||||
<fastjson.version>2.0.37</fastjson.version>
|
|
||||||
<spatial4j.version>0.8</spatial4j.version>
|
|
||||||
<jts.version>1.18.2</jts.version>
|
|
||||||
<jackson.version>2.15.2</jackson.version>
|
<jackson.version>2.15.2</jackson.version>
|
||||||
</properties>
|
</properties>
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,12 @@
|
||||||
|
package com.baeldung;
|
||||||
|
|
||||||
|
import org.springframework.boot.SpringApplication;
|
||||||
|
import org.springframework.boot.autoconfigure.SpringBootApplication;
|
||||||
|
|
||||||
|
@SpringBootApplication
|
||||||
|
public class Application {
|
||||||
|
|
||||||
|
public static void main(String[] args) {
|
||||||
|
SpringApplication.run(Application.class, args);
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,16 @@
|
||||||
|
package com.baeldung.elasticsearch;
|
||||||
|
|
||||||
|
import lombok.AllArgsConstructor;
|
||||||
|
import lombok.Data;
|
||||||
|
import lombok.NoArgsConstructor;
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
@Data
|
||||||
|
@AllArgsConstructor
|
||||||
|
@NoArgsConstructor
|
||||||
|
public class Location {
|
||||||
|
|
||||||
|
private String name;
|
||||||
|
private List<Double> location;
|
||||||
|
}
|
|
@ -12,8 +12,6 @@ import java.util.Date;
|
||||||
public class Person {
|
public class Person {
|
||||||
|
|
||||||
private int age;
|
private int age;
|
||||||
|
|
||||||
private String fullName;
|
private String fullName;
|
||||||
|
|
||||||
private Date dateOfBirth;
|
private Date dateOfBirth;
|
||||||
}
|
}
|
||||||
|
|
|
@ -15,18 +15,18 @@ import co.elastic.clients.json.jackson.JacksonJsonpMapper;
|
||||||
import co.elastic.clients.transport.ElasticsearchTransport;
|
import co.elastic.clients.transport.ElasticsearchTransport;
|
||||||
import co.elastic.clients.transport.rest_client.RestClientTransport;
|
import co.elastic.clients.transport.rest_client.RestClientTransport;
|
||||||
import lombok.extern.slf4j.Slf4j;
|
import lombok.extern.slf4j.Slf4j;
|
||||||
|
|
||||||
import org.apache.http.HttpHost;
|
import org.apache.http.HttpHost;
|
||||||
import org.elasticsearch.client.RestClient;
|
import org.elasticsearch.client.RestClient;
|
||||||
import org.junit.Before;
|
import org.junit.jupiter.api.BeforeEach;
|
||||||
import org.junit.Test;
|
import org.junit.jupiter.api.Test;
|
||||||
import org.junit.jupiter.api.Disabled;
|
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.io.StringReader;
|
import java.io.StringReader;
|
||||||
import java.util.Date;
|
import java.util.Date;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
|
||||||
import static org.junit.Assert.assertEquals;
|
import static org.junit.jupiter.api.Assertions.assertEquals;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This Manual test requires: Elasticsearch instance running on localhost:9200.
|
* This Manual test requires: Elasticsearch instance running on localhost:9200.
|
||||||
|
@ -35,41 +35,35 @@ import static org.junit.Assert.assertEquals;
|
||||||
* docker run -d --name elastic-test -p 9200:9200 -e "discovery.type=single-node" -e "xpack.security.enabled=false" docker.elastic.co/elasticsearch/elasticsearch:8.9.0
|
* docker run -d --name elastic-test -p 9200:9200 -e "discovery.type=single-node" -e "xpack.security.enabled=false" docker.elastic.co/elasticsearch/elasticsearch:8.9.0
|
||||||
*/
|
*/
|
||||||
@Slf4j
|
@Slf4j
|
||||||
@Disabled("Manual test")
|
|
||||||
public class ElasticSearchManualTest {
|
public class ElasticSearchManualTest {
|
||||||
private ElasticsearchClient client = null;
|
private ElasticsearchClient client = null;
|
||||||
|
|
||||||
@Before
|
@BeforeEach
|
||||||
public void setUp() throws IOException {
|
public void setUp() throws IOException {
|
||||||
RestClient restClient = RestClient
|
RestClient restClient = RestClient.builder(HttpHost.create("http://localhost:9200"))
|
||||||
.builder(HttpHost.create("http://localhost:9200"))
|
.build();
|
||||||
.build();
|
|
||||||
ElasticsearchTransport transport = new RestClientTransport(restClient, new JacksonJsonpMapper());
|
ElasticsearchTransport transport = new RestClientTransport(restClient, new JacksonJsonpMapper());
|
||||||
client = new ElasticsearchClient(transport);
|
client = new ElasticsearchClient(transport);
|
||||||
Person person1 = new Person(10, "John Doe", new Date());
|
Person person1 = new Person(10, "John Doe", new Date());
|
||||||
Person person2 = new Person(25, "Janette Doe", new Date());
|
Person person2 = new Person(25, "Janette Doe", new Date());
|
||||||
Person person3 = new Person(8, "Mark Doe", new Date());
|
Person person3 = new Person(8, "Mark Doe", new Date());
|
||||||
client.index(builder -> builder
|
client.index(builder -> builder.index("person")
|
||||||
.index("person")
|
.id(person1.getFullName())
|
||||||
.id(person1.getFullName())
|
.document(person1));
|
||||||
.document(person1));
|
client.index(builder -> builder.index("person")
|
||||||
client.index(builder -> builder
|
.id(person2.getFullName())
|
||||||
.index("person")
|
.document(person2));
|
||||||
.id(person2.getFullName())
|
client.index(builder -> builder.index("person")
|
||||||
.document(person2));
|
.id(person3.getFullName())
|
||||||
client.index(builder -> builder
|
.document(person3));
|
||||||
.index("person")
|
|
||||||
.id(person3.getFullName())
|
|
||||||
.document(person3));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void givenJsonDocument_whenJavaObject_thenIndexDocument() throws Exception {
|
public void givenJsonDocument_whenJavaObject_thenIndexDocument() throws Exception {
|
||||||
Person person = new Person(20, "Mark Doe", new Date(1471466076564L));
|
Person person = new Person(20, "Mark Doe", new Date(1471466076564L));
|
||||||
IndexResponse response = client.index(i -> i
|
IndexResponse response = client.index(i -> i.index("person")
|
||||||
.index("person")
|
.id(person.getFullName())
|
||||||
.id(person.getFullName())
|
.document(person));
|
||||||
.document(person));
|
|
||||||
|
|
||||||
log.info("Indexed with version: {}", response.version());
|
log.info("Indexed with version: {}", response.version());
|
||||||
assertEquals(Result.Created, response.result());
|
assertEquals(Result.Created, response.result());
|
||||||
|
@ -81,10 +75,9 @@ public class ElasticSearchManualTest {
|
||||||
public void givenJsonString_whenJavaObject_thenIndexDocument() throws Exception {
|
public void givenJsonString_whenJavaObject_thenIndexDocument() throws Exception {
|
||||||
String jsonString = "{\"age\":10,\"dateOfBirth\":1471466076564,\"fullName\":\"John Doe\"}";
|
String jsonString = "{\"age\":10,\"dateOfBirth\":1471466076564,\"fullName\":\"John Doe\"}";
|
||||||
StringReader stringReader = new StringReader(jsonString);
|
StringReader stringReader = new StringReader(jsonString);
|
||||||
IndexResponse response = client.index(i -> i
|
IndexResponse response = client.index(i -> i.index("person")
|
||||||
.index("person")
|
.id("John Doe")
|
||||||
.id("John Doe")
|
.withJson(stringReader));
|
||||||
.withJson(stringReader));
|
|
||||||
log.info("Indexed with version: {}", response.version());
|
log.info("Indexed with version: {}", response.version());
|
||||||
assertEquals("person", response.index());
|
assertEquals("person", response.index());
|
||||||
assertEquals("John Doe", response.id());
|
assertEquals("John Doe", response.id());
|
||||||
|
@ -93,9 +86,8 @@ public class ElasticSearchManualTest {
|
||||||
@Test
|
@Test
|
||||||
public void givenDocumentId_whenJavaObject_thenDeleteDocument() throws Exception {
|
public void givenDocumentId_whenJavaObject_thenDeleteDocument() throws Exception {
|
||||||
String documentId = "Mark Doe";
|
String documentId = "Mark Doe";
|
||||||
DeleteResponse response = client.delete(i -> i
|
DeleteResponse response = client.delete(i -> i.index("person")
|
||||||
.index("person")
|
.id(documentId));
|
||||||
.id(documentId));
|
|
||||||
assertEquals(Result.Deleted, response.result());
|
assertEquals(Result.Deleted, response.result());
|
||||||
assertEquals("Mark Doe", response.id());
|
assertEquals("Mark Doe", response.id());
|
||||||
}
|
}
|
||||||
|
@ -103,24 +95,23 @@ public class ElasticSearchManualTest {
|
||||||
@Test
|
@Test
|
||||||
public void givenSearchRequest_whenMatch_thenReturnAllResults() throws Exception {
|
public void givenSearchRequest_whenMatch_thenReturnAllResults() throws Exception {
|
||||||
String searchText = "John";
|
String searchText = "John";
|
||||||
SearchResponse<Person> searchResponse = client.search(s -> s
|
SearchResponse<Person> searchResponse = client.search(s -> s.index("person")
|
||||||
.index("person")
|
.query(q -> q.match(t -> t.field("fullName")
|
||||||
.query(q -> q
|
.query(searchText))), Person.class);
|
||||||
.match(t -> t
|
|
||||||
.field("fullName")
|
|
||||||
.query(searchText))), Person.class);
|
|
||||||
|
|
||||||
List<Hit<Person>> hits = searchResponse.hits().hits();
|
List<Hit<Person>> hits = searchResponse.hits()
|
||||||
|
.hits();
|
||||||
assertEquals(1, hits.size());
|
assertEquals(1, hits.size());
|
||||||
assertEquals("John Doe", hits.get(0).source().getFullName());
|
assertEquals("John Doe", hits.get(0)
|
||||||
|
.source()
|
||||||
|
.getFullName());
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void givenGetRequest_whenMatch_thenReturnAllResults() throws IOException {
|
public void givenGetRequest_whenMatch_thenReturnAllResults() throws IOException {
|
||||||
String documentId = "John Doe";
|
String documentId = "John Doe";
|
||||||
GetResponse<Person> getResponse = client.get(s -> s
|
GetResponse<Person> getResponse = client.get(s -> s.index("person")
|
||||||
.index("person")
|
.id(documentId), Person.class);
|
||||||
.id(documentId), Person.class);
|
|
||||||
Person source = getResponse.source();
|
Person source = getResponse.source();
|
||||||
assertEquals("John Doe", source.getFullName());
|
assertEquals("John Doe", source.getFullName());
|
||||||
}
|
}
|
||||||
|
@ -128,47 +119,58 @@ public class ElasticSearchManualTest {
|
||||||
@Test
|
@Test
|
||||||
public void givenSearchRequest_whenMatchAndRange_thenReturnAllResults() throws Exception {
|
public void givenSearchRequest_whenMatchAndRange_thenReturnAllResults() throws Exception {
|
||||||
String searchText = "John";
|
String searchText = "John";
|
||||||
SearchResponse<Person> searchResponse = client.search(s -> s
|
SearchResponse<Person> searchResponse = client.search(s -> s.index("person")
|
||||||
.index("person")
|
.query(q -> q.match(t -> t.field("fullName")
|
||||||
.query(q -> q
|
.query(searchText)))
|
||||||
.match(t -> t
|
.query(q -> q.range(range -> range.field("age")
|
||||||
.field("fullName").query(searchText)))
|
.from("1")
|
||||||
.query(q -> q
|
.to("10"))), Person.class);
|
||||||
.range(range -> range
|
|
||||||
.field("age").from("1").to("10"))),
|
|
||||||
Person.class);
|
|
||||||
|
|
||||||
List<Hit<Person>> hits = searchResponse.hits().hits();
|
List<Hit<Person>> hits = searchResponse.hits()
|
||||||
|
.hits();
|
||||||
assertEquals(1, hits.size());
|
assertEquals(1, hits.size());
|
||||||
assertEquals("John Doe", hits.get(0).source().getFullName());
|
assertEquals("John Doe", hits.get(0)
|
||||||
|
.source()
|
||||||
|
.getFullName());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void givenMultipleQueries_thenReturnResults() throws Exception {
|
public void givenMultipleQueries_thenReturnResults() throws Exception {
|
||||||
Query ageQuery = RangeQuery.of(r -> r.field("age").from("5").to("15"))._toQuery();
|
Query ageQuery = RangeQuery.of(r -> r.field("age")
|
||||||
SearchResponse<Person> response1 = client.search(s -> s.query(q -> q.bool(b -> b
|
.from("5")
|
||||||
.must(ageQuery))), Person.class);
|
.to("15"))
|
||||||
response1.hits().hits().forEach(hit -> log.info("Response 1: {}", hit.source()));
|
._toQuery();
|
||||||
|
SearchResponse<Person> response1 = client.search(s -> s.query(q -> q.bool(b -> b.must(ageQuery))), Person.class);
|
||||||
|
response1.hits()
|
||||||
|
.hits()
|
||||||
|
.forEach(hit -> log.info("Response 1: {}", hit.source()));
|
||||||
|
|
||||||
Query fullNameQuery = MatchQuery.of(m -> m.field("fullName").query("John"))._toQuery();
|
Query fullNameQuery = MatchQuery.of(m -> m.field("fullName")
|
||||||
SearchResponse<Person> response2 = client.search(s -> s.query(q -> q.bool(b -> b
|
.query("John"))
|
||||||
.must(fullNameQuery))), Person.class);
|
._toQuery();
|
||||||
response2.hits().hits().forEach(hit -> log.info("Response 2: {}", hit.source()));
|
SearchResponse<Person> response2 = client.search(s -> s.query(q -> q.bool(b -> b.must(fullNameQuery))), Person.class);
|
||||||
Query doeContainsQuery = SimpleQueryStringQuery.of(q -> q.query("*Doe"))._toQuery();
|
response2.hits()
|
||||||
SearchResponse<Person> response3 = client.search(s -> s.query(q -> q.bool(b -> b
|
.hits()
|
||||||
.must(doeContainsQuery))), Person.class);
|
.forEach(hit -> log.info("Response 2: {}", hit.source()));
|
||||||
response3.hits().hits().forEach(hit -> log.info("Response 3: {}", hit.source()));
|
Query doeContainsQuery = SimpleQueryStringQuery.of(q -> q.query("*Doe"))
|
||||||
|
._toQuery();
|
||||||
|
SearchResponse<Person> response3 = client.search(s -> s.query(q -> q.bool(b -> b.must(doeContainsQuery))), Person.class);
|
||||||
|
response3.hits()
|
||||||
|
.hits()
|
||||||
|
.forEach(hit -> log.info("Response 3: {}", hit.source()));
|
||||||
|
|
||||||
Query simpleStringQuery = SimpleQueryStringQuery.of(q -> q.query("+John -Doe OR Janette"))._toQuery();
|
Query simpleStringQuery = SimpleQueryStringQuery.of(q -> q.query("+John -Doe OR Janette"))
|
||||||
SearchResponse<Person> response4 = client.search(s -> s.query(q -> q.bool(b -> b
|
._toQuery();
|
||||||
.must(simpleStringQuery))), Person.class);
|
SearchResponse<Person> response4 = client.search(s -> s.query(q -> q.bool(b -> b.must(simpleStringQuery))), Person.class);
|
||||||
response4.hits().hits().forEach(hit -> log.info("Response 4: {}", hit.source()));
|
response4.hits()
|
||||||
|
.hits()
|
||||||
|
.forEach(hit -> log.info("Response 4: {}", hit.source()));
|
||||||
|
|
||||||
SearchResponse<Person> response5 = client.search(s -> s.query(q -> q.bool(b -> b
|
SearchResponse<Person> response5 = client.search(s -> s.query(q -> q.bool(b -> b.must(ageQuery)
|
||||||
.must(ageQuery)
|
.must(fullNameQuery)
|
||||||
.must(fullNameQuery)
|
.must(simpleStringQuery))), Person.class);
|
||||||
.must(simpleStringQuery))), Person.class);
|
response5.hits()
|
||||||
response5.hits().hits().forEach(hit -> log.info("Response 5: {}", hit.source()));
|
.hits()
|
||||||
|
.forEach(hit -> log.info("Response 5: {}", hit.source()));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,199 +1,198 @@
|
||||||
package com.baeldung.elasticsearch;
|
package com.baeldung.elasticsearch;
|
||||||
|
|
||||||
import static org.junit.Assert.assertTrue;
|
import static org.junit.jupiter.api.Assertions.assertEquals;
|
||||||
|
import static org.junit.jupiter.api.Assertions.assertTrue;
|
||||||
|
|
||||||
|
import co.elastic.clients.elasticsearch.ElasticsearchClient;
|
||||||
|
import co.elastic.clients.elasticsearch._types.GeoShapeRelation;
|
||||||
|
import co.elastic.clients.elasticsearch.core.IndexResponse;
|
||||||
|
import co.elastic.clients.elasticsearch.core.SearchRequest;
|
||||||
|
import co.elastic.clients.elasticsearch.core.SearchResponse;
|
||||||
|
import co.elastic.clients.elasticsearch.core.search.Hit;
|
||||||
|
import co.elastic.clients.json.JsonData;
|
||||||
|
import co.elastic.clients.json.jackson.JacksonJsonpMapper;
|
||||||
|
import co.elastic.clients.transport.ElasticsearchTransport;
|
||||||
|
import co.elastic.clients.transport.rest_client.RestClientTransport;
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.util.ArrayList;
|
import java.io.StringReader;
|
||||||
import java.util.Arrays;
|
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.stream.Collectors;
|
|
||||||
|
|
||||||
import com.baeldung.spring.data.es.config.Config;
|
import lombok.extern.slf4j.Slf4j;
|
||||||
|
|
||||||
import org.elasticsearch.action.admin.indices.delete.DeleteIndexRequest;
|
import org.apache.http.HttpHost;
|
||||||
import org.elasticsearch.action.admin.indices.refresh.RefreshRequest;
|
import org.elasticsearch.client.RestClient;
|
||||||
import org.elasticsearch.action.index.IndexRequest;
|
import org.junit.jupiter.api.AfterEach;
|
||||||
import org.elasticsearch.action.index.IndexResponse;
|
import org.junit.jupiter.api.BeforeEach;
|
||||||
import org.elasticsearch.action.search.SearchRequest;
|
import org.junit.jupiter.api.Test;
|
||||||
import org.elasticsearch.action.search.SearchResponse;
|
|
||||||
import org.elasticsearch.client.RequestOptions;
|
|
||||||
import org.elasticsearch.client.RestHighLevelClient;
|
|
||||||
import org.elasticsearch.client.indices.CreateIndexRequest;
|
|
||||||
import org.elasticsearch.common.geo.GeoPoint;
|
|
||||||
import org.elasticsearch.common.geo.ShapeRelation;
|
|
||||||
import org.elasticsearch.common.unit.DistanceUnit;
|
|
||||||
import org.elasticsearch.index.query.GeoShapeQueryBuilder;
|
|
||||||
import org.elasticsearch.index.query.QueryBuilder;
|
|
||||||
import org.elasticsearch.index.query.QueryBuilders;
|
|
||||||
import org.elasticsearch.search.SearchHit;
|
|
||||||
import org.elasticsearch.search.builder.SearchSourceBuilder;
|
|
||||||
import org.elasticsearch.xcontent.XContentType;
|
|
||||||
import org.junit.After;
|
|
||||||
import org.junit.Before;
|
|
||||||
import org.junit.Test;
|
|
||||||
import org.junit.runner.RunWith;
|
|
||||||
import org.locationtech.jts.geom.Coordinate;
|
|
||||||
import org.springframework.beans.factory.annotation.Autowired;
|
|
||||||
import org.springframework.test.context.ContextConfiguration;
|
|
||||||
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This Manual test requires: Elasticsearch instance running on localhost:9200.
|
* This Manual test requires: Elasticsearch instance running on localhost:9200.
|
||||||
*
|
* <p>
|
||||||
* The following docker command can be used: docker run -d --name es762 -p
|
* The following docker command can be used: docker run -d --name elastic-test -p 9200:9200 -e
|
||||||
* 9200:9200 -e "discovery.type=single-node" elasticsearch:7.6.2
|
* "discovery.type=single-node" -e "xpack.security.enabled=false"
|
||||||
|
* docker.elastic.co/elasticsearch/elasticsearch:8.9.0
|
||||||
*/
|
*/
|
||||||
@RunWith(SpringJUnit4ClassRunner.class)
|
|
||||||
@ContextConfiguration(classes = Config.class)
|
@Slf4j
|
||||||
public class GeoQueriesManualTest {
|
class GeoQueriesManualTest {
|
||||||
|
|
||||||
private static final String WONDERS_OF_WORLD = "wonders-of-world";
|
private static final String WONDERS_OF_WORLD = "wonders-of-world";
|
||||||
|
|
||||||
@Autowired
|
private ElasticsearchClient client;
|
||||||
private RestHighLevelClient client;
|
|
||||||
|
|
||||||
@Before
|
@BeforeEach
|
||||||
public void setUp() throws Exception {
|
public void setUp() throws Exception {
|
||||||
String jsonObject = "{\"properties\":{\"name\":{\"type\":\"text\",\"index\":false},\"region\":{\"type\":\"geo_shape\"},\"location\":{\"type\":\"geo_point\"}}}";
|
RestClient restClient = RestClient.builder(HttpHost.create("http://localhost:9200"))
|
||||||
|
.build();
|
||||||
CreateIndexRequest req = new CreateIndexRequest(WONDERS_OF_WORLD);
|
ElasticsearchTransport transport = new RestClientTransport(restClient, new JacksonJsonpMapper());
|
||||||
req.mapping(jsonObject, XContentType.JSON);
|
client = new ElasticsearchClient(transport);
|
||||||
|
log.info("Creating index: {}", WONDERS_OF_WORLD);
|
||||||
client.indices()
|
client.indices()
|
||||||
.create(req, RequestOptions.DEFAULT);
|
.create(builder -> builder.index(WONDERS_OF_WORLD)
|
||||||
}
|
.mappings(typeMapping -> typeMapping.properties("region", region -> region.geoShape(gs -> gs))
|
||||||
|
.properties("location", location -> location.geoPoint(gp -> gp))));
|
||||||
// @Test
|
|
||||||
// public void givenGeoShapeData_whenExecutedGeoShapeQuery_thenResultNonEmpty() throws IOException {
|
|
||||||
// String jsonObject = "{\"name\":\"Agra\",\"region\":{\"type\":\"envelope\",\"coordinates\":[[75,30.2],[80.1, 25]]}}";
|
|
||||||
// IndexRequest indexRequest = new IndexRequest(WONDERS_OF_WORLD);
|
|
||||||
// indexRequest.source(jsonObject, XContentType.JSON);
|
|
||||||
// IndexResponse response = client.index(indexRequest, RequestOptions.DEFAULT);
|
|
||||||
//
|
|
||||||
// String tajMahalId = response.getId();
|
|
||||||
//
|
|
||||||
// RefreshRequest refreshRequest = new RefreshRequest(WONDERS_OF_WORLD);
|
|
||||||
// client.indices()
|
|
||||||
// .refresh(refreshRequest, RequestOptions.DEFAULT);
|
|
||||||
//
|
|
||||||
// Coordinate topLeft = new Coordinate(74, 31.2);
|
|
||||||
// Coordinate bottomRight = new Coordinate(81.1, 24);
|
|
||||||
//
|
|
||||||
// GeoShapeQueryBuilder qb = QueryBuilders.geoShapeQuery("region", new EnvelopeBuilder(topLeft, bottomRight).buildGeometry());
|
|
||||||
// qb.relation(ShapeRelation.INTERSECTS);
|
|
||||||
//
|
|
||||||
// SearchSourceBuilder source = new SearchSourceBuilder().query(qb);
|
|
||||||
// SearchRequest searchRequest = new SearchRequest(WONDERS_OF_WORLD);
|
|
||||||
// searchRequest.source(source);
|
|
||||||
//
|
|
||||||
// SearchResponse searchResponse = client.search(searchRequest, RequestOptions.DEFAULT);
|
|
||||||
//
|
|
||||||
// List<String> ids = Arrays.stream(searchResponse.getHits()
|
|
||||||
// .getHits())
|
|
||||||
// .map(SearchHit::getId)
|
|
||||||
// .collect(Collectors.toList());
|
|
||||||
//
|
|
||||||
// assertTrue(ids.contains(tajMahalId));
|
|
||||||
// }
|
|
||||||
|
|
||||||
@Test
|
|
||||||
public void givenGeoPointData_whenExecutedGeoBoundingBoxQuery_thenResultNonEmpty() throws Exception {
|
|
||||||
String jsonObject = "{\"name\":\"Pyramids of Giza\",\"location\":[31.131302,29.976480]}";
|
|
||||||
|
|
||||||
IndexRequest indexRequest = new IndexRequest(WONDERS_OF_WORLD);
|
|
||||||
indexRequest.source(jsonObject, XContentType.JSON);
|
|
||||||
IndexResponse response = client.index(indexRequest, RequestOptions.DEFAULT);
|
|
||||||
|
|
||||||
String pyramidsOfGizaId = response.getId();
|
|
||||||
|
|
||||||
RefreshRequest refreshRequest = new RefreshRequest(WONDERS_OF_WORLD);
|
|
||||||
client.indices()
|
|
||||||
.refresh(refreshRequest, RequestOptions.DEFAULT);
|
|
||||||
|
|
||||||
QueryBuilder qb = QueryBuilders.geoBoundingBoxQuery("location")
|
|
||||||
.setCorners(31, 30, 28, 32);
|
|
||||||
|
|
||||||
SearchSourceBuilder source = new SearchSourceBuilder().query(qb);
|
|
||||||
SearchRequest searchRequest = new SearchRequest(WONDERS_OF_WORLD);
|
|
||||||
searchRequest.source(source);
|
|
||||||
|
|
||||||
SearchResponse searchResponse = client.search(searchRequest, RequestOptions.DEFAULT);
|
|
||||||
|
|
||||||
List<String> ids = Arrays.stream(searchResponse.getHits()
|
|
||||||
.getHits())
|
|
||||||
.map(SearchHit::getId)
|
|
||||||
.collect(Collectors.toList());
|
|
||||||
assertTrue(ids.contains(pyramidsOfGizaId));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void givenGeoPointData_whenExecutedGeoDistanceQuery_thenResultNonEmpty() throws Exception {
|
void givenGeoShapeData_whenExecutedGeoShapeQuery_thenResultNonEmpty() throws IOException {
|
||||||
String jsonObject = "{\"name\":\"Lighthouse of alexandria\",\"location\":[31.131302,29.976480]}";
|
String jsonObject = """
|
||||||
|
{
|
||||||
|
"name":"Agra",
|
||||||
|
"region":{
|
||||||
|
"type":"envelope",
|
||||||
|
"coordinates":[[75,30.2],[80.1,25]]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
""";
|
||||||
|
IndexResponse response = client.index(idx -> idx.index(WONDERS_OF_WORLD)
|
||||||
|
.withJson(new StringReader(jsonObject)));
|
||||||
|
|
||||||
IndexRequest indexRequest = new IndexRequest(WONDERS_OF_WORLD);
|
String tajMahalId = response.id();
|
||||||
indexRequest.source(jsonObject, XContentType.JSON);
|
|
||||||
IndexResponse response = client.index(indexRequest, RequestOptions.DEFAULT);
|
|
||||||
|
|
||||||
String lighthouseOfAlexandriaId = response.getId();
|
|
||||||
|
|
||||||
RefreshRequest refreshRequest = new RefreshRequest(WONDERS_OF_WORLD);
|
|
||||||
client.indices()
|
client.indices()
|
||||||
.refresh(refreshRequest, RequestOptions.DEFAULT);
|
.refresh();
|
||||||
|
|
||||||
QueryBuilder qb = QueryBuilders.geoDistanceQuery("location")
|
StringReader jsonData = new StringReader("""
|
||||||
.point(29.976, 31.131)
|
{
|
||||||
.distance(10, DistanceUnit.MILES);
|
"type":"envelope",
|
||||||
|
"coordinates": [[74.0, 31.2], [81.1, 24.0 ] ]
|
||||||
|
}
|
||||||
|
""");
|
||||||
|
|
||||||
SearchSourceBuilder source = new SearchSourceBuilder().query(qb);
|
SearchRequest searchRequest = new SearchRequest.Builder().query(query -> query.bool(boolQuery -> boolQuery.filter(query1 -> query1.geoShape(geoShapeQuery -> geoShapeQuery.field("region")
|
||||||
SearchRequest searchRequest = new SearchRequest(WONDERS_OF_WORLD);
|
.shape(geoShapeFieldQuery -> geoShapeFieldQuery.relation(GeoShapeRelation.Within)
|
||||||
searchRequest.source(source);
|
.shape(JsonData.from(jsonData)))))))
|
||||||
|
.build();
|
||||||
SearchResponse searchResponse = client.search(searchRequest, RequestOptions.DEFAULT);
|
log.info("Search request: {}", searchRequest);
|
||||||
|
SearchResponse<Object> search = client.search(searchRequest, Object.class);
|
||||||
List<String> ids = Arrays.stream(searchResponse.getHits()
|
log.info("Search response: {}", search);
|
||||||
.getHits())
|
List<String> searchResults = search.hits()
|
||||||
.map(SearchHit::getId)
|
.hits()
|
||||||
.collect(Collectors.toList());
|
.stream()
|
||||||
assertTrue(ids.contains(lighthouseOfAlexandriaId));
|
.map(Hit::id)
|
||||||
|
.toList();
|
||||||
|
assertTrue(searchResults.contains(tajMahalId));
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void givenGeoPointData_whenExecutedGeoPolygonQuery_thenResultNonEmpty() throws Exception {
|
void givenGeoPointData_whenExecutedGeoBoundingBoxQuery_thenResultNonEmpty() throws Exception {
|
||||||
String jsonObject = "{\"name\":\"The Great Rann of Kutch\",\"location\":[69.859741,23.733732]}";
|
Location pyramidsOfGiza = new Location("Pyramids of Giza", List.of(31.1328, 29.9761));
|
||||||
|
IndexResponse response = client.index(builder -> builder.index(WONDERS_OF_WORLD)
|
||||||
|
.document(pyramidsOfGiza));
|
||||||
|
|
||||||
IndexRequest indexRequest = new IndexRequest(WONDERS_OF_WORLD);
|
String pyramidsOfGizaId = response.id();
|
||||||
indexRequest.source(jsonObject, XContentType.JSON);
|
|
||||||
IndexResponse response = client.index(indexRequest, RequestOptions.DEFAULT);
|
|
||||||
|
|
||||||
String greatRannOfKutchid = response.getId();
|
log.info("Indexed pyramid of Giza: {}", pyramidsOfGizaId);
|
||||||
|
|
||||||
RefreshRequest refreshRequest = new RefreshRequest(WONDERS_OF_WORLD);
|
|
||||||
client.indices()
|
client.indices()
|
||||||
.refresh(refreshRequest, RequestOptions.DEFAULT);
|
.refresh();
|
||||||
|
|
||||||
List<GeoPoint> allPoints = new ArrayList<GeoPoint>();
|
SearchRequest.Builder builder = new SearchRequest.Builder().index(WONDERS_OF_WORLD);
|
||||||
allPoints.add(new GeoPoint(22.733, 68.859));
|
builder.query(query -> query.geoBoundingBox(geoBoundingBoxQuery -> geoBoundingBoxQuery.field("location")
|
||||||
allPoints.add(new GeoPoint(24.733, 68.859));
|
.boundingBox(geoBounds -> geoBounds.tlbr(bl4 -> bl4.topLeft(geoLocation -> geoLocation.coords(List.of(30.0, 31.0)))
|
||||||
allPoints.add(new GeoPoint(23, 70.859));
|
.bottomRight(geoLocation -> geoLocation.coords(List.of(32.0, 28.0)))))));
|
||||||
QueryBuilder qb = QueryBuilders.geoPolygonQuery("location", allPoints);
|
SearchRequest build = builder.build();
|
||||||
|
log.info("Search request: {}", build);
|
||||||
SearchSourceBuilder source = new SearchSourceBuilder().query(qb);
|
SearchResponse<Location> searchResponse = client.search(build, Location.class);
|
||||||
SearchRequest searchRequest = new SearchRequest(WONDERS_OF_WORLD);
|
log.info("Search response: {}", searchResponse);
|
||||||
searchRequest.source(source);
|
List<Location> returnedLocations = searchResponse.hits()
|
||||||
|
.hits()
|
||||||
SearchResponse searchResponse = client.search(searchRequest, RequestOptions.DEFAULT);
|
.stream()
|
||||||
|
.map(Hit::source)
|
||||||
List<String> ids = Arrays.stream(searchResponse.getHits()
|
.toList();
|
||||||
.getHits())
|
assertEquals(pyramidsOfGiza, returnedLocations.get(0));
|
||||||
.map(SearchHit::getId)
|
|
||||||
.collect(Collectors.toList());
|
|
||||||
assertTrue(ids.contains(greatRannOfKutchid));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@After
|
@Test
|
||||||
|
void givenGeoPointData_whenExecutedGeoDistanceQuery_thenResultNonEmpty() throws Exception {
|
||||||
|
String jsonObject = """
|
||||||
|
{
|
||||||
|
"name":"Lighthouse of alexandria",
|
||||||
|
"location":{ "lat": 31.2139, "lon": 29.8856 }
|
||||||
|
}
|
||||||
|
""";
|
||||||
|
IndexResponse response = client.index(idx -> idx.index(WONDERS_OF_WORLD)
|
||||||
|
.withJson(new StringReader(jsonObject)));
|
||||||
|
|
||||||
|
String lightHouseOfAlexandriaId = response.id();
|
||||||
|
client.indices()
|
||||||
|
.refresh();
|
||||||
|
SearchRequest searchRequest = new SearchRequest.Builder().index(WONDERS_OF_WORLD)
|
||||||
|
.query(query -> query.geoDistance(geoDistanceQuery -> geoDistanceQuery.field("location")
|
||||||
|
.distance("10 miles")
|
||||||
|
.location(geoLocation -> geoLocation.latlon(latLonGeoLocation -> latLonGeoLocation.lon(29.88)
|
||||||
|
.lat(31.21)))))
|
||||||
|
.build();
|
||||||
|
log.info("Search request: {}", searchRequest);
|
||||||
|
SearchResponse<Object> search = client.search(searchRequest, Object.class);
|
||||||
|
log.info("Search response: {}", search);
|
||||||
|
List<String> ids = search.hits()
|
||||||
|
.hits()
|
||||||
|
.stream()
|
||||||
|
.map(Hit::id)
|
||||||
|
.toList();
|
||||||
|
assertTrue(ids.contains(lightHouseOfAlexandriaId));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void givenGeoPointData_whenExecutedGeoPolygonQuery_thenResultNonEmpty() throws Exception {
|
||||||
|
String jsonObject = """
|
||||||
|
{
|
||||||
|
"name":"The Great Rann polygonPoints Kutch",
|
||||||
|
"location":{"lon": 69.859741, "lat": 23.733732}
|
||||||
|
}
|
||||||
|
""";
|
||||||
|
IndexResponse response = client.index(idx -> idx.index(WONDERS_OF_WORLD)
|
||||||
|
.withJson(new StringReader(jsonObject)));
|
||||||
|
String greatRannOfKutchid = response.id();
|
||||||
|
client.indices()
|
||||||
|
.refresh();
|
||||||
|
log.info("Indexed greatRannOfKutchid: {}", greatRannOfKutchid);
|
||||||
|
|
||||||
|
JsonData jsonData = JsonData.fromJson("""
|
||||||
|
{
|
||||||
|
"type":"polygon",
|
||||||
|
"coordinates":[[[68.859,22.733],[68.859,24.733],[70.859,23]]]
|
||||||
|
}
|
||||||
|
""");
|
||||||
|
|
||||||
|
SearchRequest build = new SearchRequest.Builder().query(query -> query.bool(boolQuery -> boolQuery.filter(query1 -> query1.geoShape(geoShapeQuery -> geoShapeQuery.field("location")
|
||||||
|
.shape(geoShapeFieldQuery -> geoShapeFieldQuery.relation(GeoShapeRelation.Within)
|
||||||
|
.shape(jsonData))))))
|
||||||
|
.build();
|
||||||
|
log.info("Search request: {}", build);
|
||||||
|
SearchResponse<Object> search = client.search(build, Object.class);
|
||||||
|
log.info("Search response: {}", search);
|
||||||
|
List<String> searchResults = search.hits()
|
||||||
|
.hits()
|
||||||
|
.stream()
|
||||||
|
.map(Hit::id)
|
||||||
|
.toList();
|
||||||
|
assertTrue(searchResults.contains(greatRannOfKutchid));
|
||||||
|
}
|
||||||
|
|
||||||
|
@AfterEach
|
||||||
public void destroy() throws Exception {
|
public void destroy() throws Exception {
|
||||||
DeleteIndexRequest deleteIndex = new DeleteIndexRequest(WONDERS_OF_WORLD);
|
|
||||||
client.indices()
|
client.indices()
|
||||||
.delete(deleteIndex, RequestOptions.DEFAULT);
|
.delete(builder -> builder.index(WONDERS_OF_WORLD));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue