Skip to content

Commit 2ad5326

Browse files
authored
[DE-568] geo s2 analyzer (#501)
* fixed HttpCommunication debug logs in case of db error * geo_s2 analyzer * test fixes
1 parent 3e1a9ab commit 2ad5326

File tree

7 files changed

+221
-9
lines changed

7 files changed

+221
-9
lines changed

core/src/main/java/com/arangodb/entity/arangosearch/AnalyzerType.java

+1
Original file line numberDiff line numberDiff line change
@@ -35,6 +35,7 @@ public enum AnalyzerType {
3535
aql,
3636
geojson,
3737
geopoint,
38+
geo_s2,
3839
segmentation,
3940
collation,
4041
classification,
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,68 @@
1+
/*
2+
* DISCLAIMER
3+
*
4+
* Copyright 2016 ArangoDB GmbH, Cologne, Germany
5+
*
6+
* Licensed under the Apache License, Version 2.0 (the "License");
7+
* you may not use this file except in compliance with the License.
8+
* You may obtain a copy of the License at
9+
*
10+
* http://www.apache.org/licenses/LICENSE-2.0
11+
*
12+
* Unless required by applicable law or agreed to in writing, software
13+
* distributed under the License is distributed on an "AS IS" BASIS,
14+
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
15+
* See the License for the specific language governing permissions and
16+
* limitations under the License.
17+
*
18+
* Copyright holder is ArangoDB GmbH, Cologne, Germany
19+
*/
20+
21+
22+
package com.arangodb.entity.arangosearch.analyzer;
23+
24+
import com.arangodb.entity.arangosearch.AnalyzerType;
25+
26+
import java.util.Objects;
27+
28+
/**
29+
* An Analyzer capable of breaking up a GeoJSON object or coordinate array in [longitude, latitude] order into a set of
30+
* indexable tokens for further usage with ArangoSearch Geo functions.
31+
* <p>
32+
* The Analyzer is similar to {@link GeoJSONAnalyzer}, but it internally uses a format for storing the geo-spatial data
33+
* that is more efficient. You can choose between different formats to make a tradeoff between the size on disk, the
34+
* precision, and query performance.
35+
*
36+
* @author Michele Rastelli
37+
* @see <a href= "https://www.arangodb.com/docs/stable/analyzers.html#geo_s2">API Documentation</a>
38+
* @since ArangoDB 3.10.5
39+
*/
40+
public final class GeoS2Analyzer extends SearchAnalyzer {
41+
private GeoS2AnalyzerProperties properties;
42+
43+
public GeoS2Analyzer() {
44+
setType(AnalyzerType.geo_s2);
45+
}
46+
47+
public GeoS2AnalyzerProperties getProperties() {
48+
return properties;
49+
}
50+
51+
public void setProperties(GeoS2AnalyzerProperties properties) {
52+
this.properties = properties;
53+
}
54+
55+
@Override
56+
public boolean equals(Object o) {
57+
if (this == o) return true;
58+
if (o == null || getClass() != o.getClass()) return false;
59+
if (!super.equals(o)) return false;
60+
GeoS2Analyzer that = (GeoS2Analyzer) o;
61+
return Objects.equals(properties, that.properties);
62+
}
63+
64+
@Override
65+
public int hashCode() {
66+
return Objects.hash(super.hashCode(), properties);
67+
}
68+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,119 @@
1+
/*
2+
* DISCLAIMER
3+
*
4+
* Copyright 2016 ArangoDB GmbH, Cologne, Germany
5+
*
6+
* Licensed under the Apache License, Version 2.0 (the "License");
7+
* you may not use this file except in compliance with the License.
8+
* You may obtain a copy of the License at
9+
*
10+
* http://www.apache.org/licenses/LICENSE-2.0
11+
*
12+
* Unless required by applicable law or agreed to in writing, software
13+
* distributed under the License is distributed on an "AS IS" BASIS,
14+
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
15+
* See the License for the specific language governing permissions and
16+
* limitations under the License.
17+
*
18+
* Copyright holder is ArangoDB GmbH, Cologne, Germany
19+
*/
20+
21+
package com.arangodb.entity.arangosearch.analyzer;
22+
23+
24+
import java.util.Objects;
25+
26+
/**
27+
* @author Michele Rastelli
28+
*/
29+
public final class GeoS2AnalyzerProperties {
30+
31+
private GeoS2AnalyzerType type;
32+
private GeoAnalyzerOptions options;
33+
private GeoS2Format format;
34+
35+
public GeoS2AnalyzerType getType() {
36+
return type;
37+
}
38+
39+
public void setType(GeoS2AnalyzerType type) {
40+
this.type = type;
41+
}
42+
43+
/**
44+
* @return Options for fine-tuning geo queries {@link GeoS2AnalyzerProperties}. These options should generally
45+
* remain unchanged.
46+
*/
47+
public GeoAnalyzerOptions getOptions() {
48+
return options;
49+
}
50+
51+
public void setOptions(GeoAnalyzerOptions options) {
52+
this.options = options;
53+
}
54+
55+
/**
56+
* @return The internal binary representation to use for storing the geo-spatial data in an index.
57+
*/
58+
public GeoS2Format getFormat() {
59+
return format;
60+
}
61+
62+
public void setFormat(GeoS2Format format) {
63+
this.format = format;
64+
}
65+
66+
@Override
67+
public boolean equals(Object o) {
68+
if (this == o) return true;
69+
if (o == null || getClass() != o.getClass()) return false;
70+
GeoS2AnalyzerProperties that = (GeoS2AnalyzerProperties) o;
71+
return type == that.type && Objects.equals(options, that.options) && format == that.format;
72+
}
73+
74+
@Override
75+
public int hashCode() {
76+
return Objects.hash(type, options, format);
77+
}
78+
79+
public enum GeoS2AnalyzerType {
80+
81+
/**
82+
* (default): index all GeoJSON geometry types (Point, Polygon etc.)
83+
*/
84+
shape,
85+
86+
/**
87+
* compute and only index the centroid of the input geometry
88+
*/
89+
centroid,
90+
91+
/**
92+
* only index GeoJSON objects of type Point, ignore all other geometry types
93+
*/
94+
point
95+
}
96+
97+
public enum GeoS2Format {
98+
/**
99+
* Store each latitude and longitude value as an 8-byte floating-point value (16 bytes per coordinate pair).
100+
* This format preserves numeric values exactly and is more compact than the VelocyPack format used by
101+
* {@link GeoJSONAnalyzer}. (default)
102+
*/
103+
latLngDouble,
104+
105+
/**
106+
* Store each latitude and longitude value as an 4-byte integer value (8 bytes per coordinate pair). This is the
107+
* most compact format but the precision is limited to approximately 1 to 10 centimeters.
108+
*/
109+
latLngInt,
110+
111+
/**
112+
* Store each longitude-latitude pair in the native format of Google S2 which is used for geo-spatial
113+
* calculations (24 bytes per coordinate pair). This is not a particular compact format but it reduces the
114+
* number of computations necessary when you execute geo-spatial queries. This format preserves numeric values
115+
* exactly.
116+
*/
117+
s2Point
118+
}
119+
}

core/src/main/java/com/arangodb/entity/arangosearch/analyzer/SearchAnalyzer.java

+1
Original file line numberDiff line numberDiff line change
@@ -48,6 +48,7 @@
4848
@JsonSubTypes.Type(name = "aql", value = AQLAnalyzer.class),
4949
@JsonSubTypes.Type(name = "geojson", value = GeoJSONAnalyzer.class),
5050
@JsonSubTypes.Type(name = "geopoint", value = GeoPointAnalyzer.class),
51+
@JsonSubTypes.Type(name = "geo_s2", value = GeoS2Analyzer.class),
5152
@JsonSubTypes.Type(name = "segmentation", value = SegmentationAnalyzer.class),
5253
@JsonSubTypes.Type(name = "collation", value = CollationAnalyzer.class),
5354
@JsonSubTypes.Type(name = "classification", value = ClassificationAnalyzer.class),

driver/src/test/java/com/arangodb/ArangoSearchTest.java

+30
Original file line numberDiff line numberDiff line change
@@ -846,6 +846,36 @@ void geoJsonAnalyzer(ArangoDatabase db) {
846846
}
847847

848848

849+
@ParameterizedTest(name = "{index}")
850+
@MethodSource("dbs")
851+
void geoS2Analyzer(ArangoDatabase db) {
852+
assumeTrue(isEnterprise());
853+
assumeTrue(isAtLeastVersion(3, 10, 5));
854+
855+
GeoAnalyzerOptions options = new GeoAnalyzerOptions();
856+
options.setMaxLevel(10);
857+
options.setMaxCells(11);
858+
options.setMinLevel(8);
859+
860+
GeoS2AnalyzerProperties properties = new GeoS2AnalyzerProperties();
861+
properties.setOptions(options);
862+
properties.setType(GeoS2AnalyzerProperties.GeoS2AnalyzerType.point);
863+
properties.setFormat(GeoS2AnalyzerProperties.GeoS2Format.s2Point);
864+
865+
Set<AnalyzerFeature> features = new HashSet<>();
866+
features.add(AnalyzerFeature.frequency);
867+
features.add(AnalyzerFeature.norm);
868+
features.add(AnalyzerFeature.position);
869+
870+
GeoS2Analyzer geoS2Analyzer = new GeoS2Analyzer();
871+
geoS2Analyzer.setName("test-" + UUID.randomUUID());
872+
geoS2Analyzer.setProperties(properties);
873+
geoS2Analyzer.setFeatures(features);
874+
875+
createGetAndDeleteTypedAnalyzer(db, geoS2Analyzer);
876+
}
877+
878+
849879
@ParameterizedTest(name = "{index}")
850880
@MethodSource("dbs")
851881
void geoPointAnalyzer(ArangoDatabase db) {

http/src/main/java/com/arangodb/http/HttpCommunication.java

+2
Original file line numberDiff line numberDiff line change
@@ -30,6 +30,7 @@
3030
import com.arangodb.internal.serde.InternalSerde;
3131
import com.arangodb.internal.util.HostUtils;
3232
import com.arangodb.internal.util.RequestUtils;
33+
import com.arangodb.internal.util.ResponseUtils;
3334
import org.slf4j.Logger;
3435
import org.slf4j.LoggerFactory;
3536

@@ -82,6 +83,7 @@ private InternalResponse execute(final InternalRequest request, final HostHandle
8283
String body = response.getBody() == null ? "" : serde.toJsonString(response.getBody());
8384
LOGGER.debug("Received Response [id={}]: {} {}", reqId, response, body);
8485
}
86+
ResponseUtils.checkError(serde, response);
8587
hostHandler.success();
8688
hostHandler.confirm();
8789
return response;

http/src/main/java/com/arangodb/http/HttpConnection.java

-9
Original file line numberDiff line numberDiff line change
@@ -25,9 +25,7 @@
2525
import com.arangodb.internal.config.ArangoConfig;
2626
import com.arangodb.internal.net.Connection;
2727
import com.arangodb.internal.serde.ContentTypeFactory;
28-
import com.arangodb.internal.serde.InternalSerde;
2928
import com.arangodb.internal.util.EncodeUtils;
30-
import com.arangodb.internal.util.ResponseUtils;
3129
import com.arangodb.internal.InternalRequest;
3230
import com.arangodb.internal.RequestType;
3331
import com.arangodb.internal.InternalResponse;
@@ -75,7 +73,6 @@ public class HttpConnection implements Connection {
7573
private static final String CONTENT_TYPE_VPACK = "application/x-velocypack";
7674
private static final String USER_AGENT = getUserAgent();
7775
private static final AtomicInteger THREAD_COUNT = new AtomicInteger();
78-
private final InternalSerde serde;
7976
private final ContentType contentType;
8077
private String auth;
8178
private final WebClient client;
@@ -88,7 +85,6 @@ private static String getUserAgent() {
8885

8986
HttpConnection(final ArangoConfig config, final HostDescription host) {
9087
super();
91-
serde = config.getInternalSerde();
9288
Protocol protocol = config.getProtocol();
9389
contentType = ContentTypeFactory.of(protocol);
9490
timeout = config.getTimeout();
@@ -244,7 +240,6 @@ public InternalResponse execute(final InternalRequest request) throws IOExceptio
244240
throw new IOException(cause);
245241
}
246242
}
247-
checkError(resp);
248243
return resp;
249244
}
250245

@@ -295,10 +290,6 @@ private InternalResponse buildResponse(final HttpResponse<Buffer> httpResponse)
295290
return response;
296291
}
297292

298-
protected void checkError(final InternalResponse response) {
299-
ResponseUtils.checkError(serde, response);
300-
}
301-
302293
@Override
303294
public void setJwt(String jwt) {
304295
if (jwt != null) {

0 commit comments

Comments
 (0)