From d005c76bccaa3e1b841ac814ddf817dc0bf38db1 Mon Sep 17 00:00:00 2001 From: Kirill Ostanin Date: Wed, 18 Oct 2023 13:11:41 +0400 Subject: [PATCH 1/5] Add support neural query type Signed-off-by: Kirill_Ostanin --- .../_types/query_dsl/NeuralQuery.java | 206 ++++++++++++++++++ .../opensearch/_types/query_dsl/Query.java | 30 +++ .../_types/query_dsl/QueryBuilders.java | 7 + .../client/opensearch/model/VariantsTest.java | 44 ++++ 4 files changed, 287 insertions(+) create mode 100644 java-client/src/main/java/org/opensearch/client/opensearch/_types/query_dsl/NeuralQuery.java diff --git a/java-client/src/main/java/org/opensearch/client/opensearch/_types/query_dsl/NeuralQuery.java b/java-client/src/main/java/org/opensearch/client/opensearch/_types/query_dsl/NeuralQuery.java new file mode 100644 index 0000000000..60952db0f0 --- /dev/null +++ b/java-client/src/main/java/org/opensearch/client/opensearch/_types/query_dsl/NeuralQuery.java @@ -0,0 +1,206 @@ +/* + * SPDX-License-Identifier: Apache-2.0 + * + * The OpenSearch Contributors require contributions made to + * this file be licensed under the Apache-2.0 license or a + * compatible open source license. + */ + +package org.opensearch.client.opensearch._types.query_dsl; + +import jakarta.json.stream.JsonGenerator; +import org.opensearch.client.json.JsonpDeserializable; +import org.opensearch.client.json.JsonpDeserializer; +import org.opensearch.client.json.JsonpMapper; +import org.opensearch.client.json.ObjectBuilderDeserializer; +import org.opensearch.client.json.ObjectDeserializer; +import org.opensearch.client.util.ApiTypeHelper; +import org.opensearch.client.util.ObjectBuilder; + +import javax.annotation.Nullable; +import java.util.function.Function; + +@JsonpDeserializable +public class NeuralQuery extends QueryBase implements QueryVariant { + + private final String field; + private final String queryText; + private final int k; + @Nullable + private final String modelId; + + + private NeuralQuery(NeuralQuery.Builder builder) { + super(builder); + + this.field = ApiTypeHelper.requireNonNull(builder.field, this, "field"); + this.queryText = ApiTypeHelper.requireNonNull(builder.queryText, this, "queryText"); + this.k = ApiTypeHelper.requireNonNull(builder.k, this, "k"); + this.modelId = builder.modelId; + } + + public static NeuralQuery of(Function> fn) { + return fn.apply(new NeuralQuery.Builder()).build(); + } + + /** + * Query variant kind. + * + * @return The query variant kind. + */ + @Override + public Query.Kind _queryKind() { + return null; + } + + /** + * Required - The target field. + * + * @return The target field. + */ + public final String field() { + return this.field; + } + + /** + * Required - Search query text. + * + * @return Search query text. + */ + public final String queryText() { + return this.queryText; + } + + /** + * Required - The number of neighbors to return. + * + * @return The number of neighbors to return. + */ + public final int k() { + return this.k; + } + + /** + * Builder for {@link NeuralQuery}. + */ + + /** + * Optional - The model_id field if the default model for the index or field is set. + * Required - The model_id field if there is no default model set for the index or field. + * + * @return The model_id field. + */ + @Nullable + public final String modelId() { + return this.modelId; + } + + @Override + protected void serializeInternal(JsonGenerator generator, JsonpMapper mapper) { + generator.writeStartObject(this.field); + + super.serializeInternal(generator, mapper); + + generator.write("query_text", this.queryText); + + if (this.modelId != null) { + generator.write("model_id", this.modelId); + } + + generator.write("k", this.k); + + generator.writeEnd(); + } + + /** + * Builder for {@link NeuralQuery}. + */ + public static class Builder extends QueryBase.AbstractBuilder implements ObjectBuilder { + @Nullable + private String field; + @Nullable + private String queryText; + @Nullable + private String modelId; + @Nullable + private Integer k; + + /** + * Required - The target field. + * + * @param field The target field. + * @return This builder. + */ + public NeuralQuery.Builder field(@Nullable String field) { + this.field = field; + return this; + } + + /** + * Required - Search query text. + * + * @param queryText Search query text. + * @return This builder. + */ + public NeuralQuery.Builder queryText(@Nullable String queryText) { + this.queryText = queryText; + return this; + } + + /** + * Optional - The model_id field if the default model for the index or field is set. + * Required - The model_id field if there is no default model set for the index or field. + * + * @param modelId The model_id field. + * @return This builder. + */ + public NeuralQuery.Builder modelId(@Nullable String modelId) { + this.modelId = modelId; + return this; + } + + /** + * Required - The number of neighbors to return. + * + * @param k The number of neighbors to return. + * @return This builder. + */ + public NeuralQuery.Builder k(@Nullable Integer k) { + this.k = k; + return this; + } + + @Override + protected NeuralQuery.Builder self() { + return this; + } + + /** + * Builds a {@link NeuralQuery}. + * + * @return The built {@link NeuralQuery}. + */ + @Override + public NeuralQuery build() { + _checkSingleUse(); + + return new NeuralQuery(this); + } + } + + public static final JsonpDeserializer _DESERIALIZER = ObjectBuilderDeserializer.lazy( + NeuralQuery.Builder::new, + NeuralQuery::setupNeuralQueryDeserializer + ); + + protected static void setupNeuralQueryDeserializer(ObjectDeserializer op) { + setupQueryBaseDeserializer(op); + + op.add(NeuralQuery.Builder::queryText, JsonpDeserializer.stringDeserializer(), "query_text"); + op.add(NeuralQuery.Builder::modelId, JsonpDeserializer.stringDeserializer(), "model_id"); + op.add(NeuralQuery.Builder::k, JsonpDeserializer.integerDeserializer(), "k"); + + + op.setKey(NeuralQuery.Builder::field, JsonpDeserializer.stringDeserializer()); + } +} diff --git a/java-client/src/main/java/org/opensearch/client/opensearch/_types/query_dsl/Query.java b/java-client/src/main/java/org/opensearch/client/opensearch/_types/query_dsl/Query.java index b1d39da0a3..7413df3593 100644 --- a/java-client/src/main/java/org/opensearch/client/opensearch/_types/query_dsl/Query.java +++ b/java-client/src/main/java/org/opensearch/client/opensearch/_types/query_dsl/Query.java @@ -118,6 +118,8 @@ public enum Kind implements JsonEnum { Nested("nested"), + Neural("neural"), + ParentId("parent_id"), Percolate("percolate"), @@ -706,6 +708,23 @@ public NestedQuery nested() { return TaggedUnionUtils.get(this, Kind.Nested); } + /** + * Is this variant instance of kind {@code neural}? + */ + public boolean isNeural() { + return _kind == Kind.Neural; + } + + /** + * Get the {@code neural} variant value. + * + * @throws IllegalStateException + * if the current variant is not of the {@code neural} kind. + */ + public NeuralQuery neural() { + return TaggedUnionUtils.get(this, Kind.Neural); + } + /** * Is this variant instance of kind {@code parent_id}? */ @@ -1450,6 +1469,16 @@ public ObjectBuilder nested(Function neural(NeuralQuery v) { + this._kind = Kind.Neural; + this._value = v; + return this; + } + + public ObjectBuilder neural(Function> fn) { + return this.neural(fn.apply(new NeuralQuery.Builder()).build()); + } + public ObjectBuilder parentId(ParentIdQuery v) { this._kind = Kind.ParentId; this._value = v; @@ -1747,6 +1776,7 @@ protected static void setupQueryDeserializer(ObjectDeserializer op) { op.add(Builder::moreLikeThis, MoreLikeThisQuery._DESERIALIZER, "more_like_this"); op.add(Builder::multiMatch, MultiMatchQuery._DESERIALIZER, "multi_match"); op.add(Builder::nested, NestedQuery._DESERIALIZER, "nested"); + op.add(Builder::neural, NeuralQuery._DESERIALIZER, "neural"); op.add(Builder::parentId, ParentIdQuery._DESERIALIZER, "parent_id"); op.add(Builder::percolate, PercolateQuery._DESERIALIZER, "percolate"); op.add(Builder::pinned, PinnedQuery._DESERIALIZER, "pinned"); diff --git a/java-client/src/main/java/org/opensearch/client/opensearch/_types/query_dsl/QueryBuilders.java b/java-client/src/main/java/org/opensearch/client/opensearch/_types/query_dsl/QueryBuilders.java index dc6db50515..f165ac7060 100644 --- a/java-client/src/main/java/org/opensearch/client/opensearch/_types/query_dsl/QueryBuilders.java +++ b/java-client/src/main/java/org/opensearch/client/opensearch/_types/query_dsl/QueryBuilders.java @@ -254,6 +254,13 @@ public static NestedQuery.Builder nested() { return new NestedQuery.Builder(); } + /** + * Creates a builder for the {@link NeuralQuery nested} {@code Query} variant. + */ + public static NeuralQuery.Builder neural() { + return new NeuralQuery.Builder(); + } + /** * Creates a builder for the {@link ParentIdQuery parent_id} {@code Query} * variant. diff --git a/java-client/src/test/java/org/opensearch/client/opensearch/model/VariantsTest.java b/java-client/src/test/java/org/opensearch/client/opensearch/model/VariantsTest.java index b511f4ec1d..dd731c6829 100644 --- a/java-client/src/test/java/org/opensearch/client/opensearch/model/VariantsTest.java +++ b/java-client/src/test/java/org/opensearch/client/opensearch/model/VariantsTest.java @@ -205,4 +205,48 @@ public void testNestedVariantsWithContainerProperties() { assertEquals("m1 value", search.aggregations().get("agg1").meta().get("m1").to(String.class)); assertEquals("m2 value", search.aggregations().get("agg1").meta().get("m2").to(String.class)); } + + @Test + public void testNeuralQuery() { + + SearchRequest searchRequest = SearchRequest.of( + s -> s.query( + q -> q.neural(n -> n.field("passage_embedding") + .queryText("Hi world") + .modelId("bQ1J8ooBpBj3wT4HVUsb") + .k(100) + ) + ) + ); + + assertEquals("passage_embedding", searchRequest.query().neural().field()); + assertEquals("Hi world", searchRequest.query().neural().queryText()); + assertEquals("bQ1J8ooBpBj3wT4HVUsb", searchRequest.query().neural().modelId()); + assertEquals(100, searchRequest.query().neural().k()); + } + + @Test + public void testNeuralQueryFromJson() { + + String json = "{\n" + + " \"from\": 0,\n" + + " \"size\": 100,\n" + + " \"query\": {\n" + + " \"neural\": {\n" + + " \"passage_embedding\": {\n" + + " \"query_text\": \"Hi world\",\n" + + " \"model_id\": \"bQ1J8ooBpBj3wT4HVUsb\",\n" + + " \"k\": 100\n" + + " }\n" + + " }\n" + + " }\n" + + "}"; + + SearchRequest searchRequest = ModelTestCase.fromJson(json, SearchRequest.class, mapper); + + assertEquals("passage_embedding", searchRequest.query().neural().field()); + assertEquals("Hi world", searchRequest.query().neural().queryText()); + assertEquals("bQ1J8ooBpBj3wT4HVUsb", searchRequest.query().neural().modelId()); + assertEquals(100, searchRequest.query().neural().k()); + } } From 50889eb4ace4d1296f7df13bf6216c0c89957b04 Mon Sep 17 00:00:00 2001 From: Kirill Ostanin Date: Mon, 23 Oct 2023 11:54:23 +0400 Subject: [PATCH 2/5] fix spotless Signed-off-by: Kirill Ostanin --- .../_types/query_dsl/NeuralQuery.java | 11 +++--- .../client/opensearch/model/VariantsTest.java | 34 ++++++++----------- 2 files changed, 18 insertions(+), 27 deletions(-) diff --git a/java-client/src/main/java/org/opensearch/client/opensearch/_types/query_dsl/NeuralQuery.java b/java-client/src/main/java/org/opensearch/client/opensearch/_types/query_dsl/NeuralQuery.java index 60952db0f0..e1041db04e 100644 --- a/java-client/src/main/java/org/opensearch/client/opensearch/_types/query_dsl/NeuralQuery.java +++ b/java-client/src/main/java/org/opensearch/client/opensearch/_types/query_dsl/NeuralQuery.java @@ -9,6 +9,8 @@ package org.opensearch.client.opensearch._types.query_dsl; import jakarta.json.stream.JsonGenerator; +import java.util.function.Function; +import javax.annotation.Nullable; import org.opensearch.client.json.JsonpDeserializable; import org.opensearch.client.json.JsonpDeserializer; import org.opensearch.client.json.JsonpMapper; @@ -17,9 +19,6 @@ import org.opensearch.client.util.ApiTypeHelper; import org.opensearch.client.util.ObjectBuilder; -import javax.annotation.Nullable; -import java.util.function.Function; - @JsonpDeserializable public class NeuralQuery extends QueryBase implements QueryVariant { @@ -29,7 +28,6 @@ public class NeuralQuery extends QueryBase implements QueryVariant { @Nullable private final String modelId; - private NeuralQuery(NeuralQuery.Builder builder) { super(builder); @@ -189,8 +187,8 @@ public NeuralQuery build() { } public static final JsonpDeserializer _DESERIALIZER = ObjectBuilderDeserializer.lazy( - NeuralQuery.Builder::new, - NeuralQuery::setupNeuralQueryDeserializer + NeuralQuery.Builder::new, + NeuralQuery::setupNeuralQueryDeserializer ); protected static void setupNeuralQueryDeserializer(ObjectDeserializer op) { @@ -200,7 +198,6 @@ protected static void setupNeuralQueryDeserializer(ObjectDeserializer s.query( - q -> q.neural(n -> n.field("passage_embedding") - .queryText("Hi world") - .modelId("bQ1J8ooBpBj3wT4HVUsb") - .k(100) - ) - ) + s -> s.query(q -> q.neural(n -> n.field("passage_embedding").queryText("Hi world").modelId("bQ1J8ooBpBj3wT4HVUsb").k(100))) ); assertEquals("passage_embedding", searchRequest.query().neural().field()); @@ -228,19 +222,19 @@ public void testNeuralQuery() { @Test public void testNeuralQueryFromJson() { - String json = "{\n" + - " \"from\": 0,\n" + - " \"size\": 100,\n" + - " \"query\": {\n" + - " \"neural\": {\n" + - " \"passage_embedding\": {\n" + - " \"query_text\": \"Hi world\",\n" + - " \"model_id\": \"bQ1J8ooBpBj3wT4HVUsb\",\n" + - " \"k\": 100\n" + - " }\n" + - " }\n" + - " }\n" + - "}"; + String json = "{\n" + + " \"from\": 0,\n" + + " \"size\": 100,\n" + + " \"query\": {\n" + + " \"neural\": {\n" + + " \"passage_embedding\": {\n" + + " \"query_text\": \"Hi world\",\n" + + " \"model_id\": \"bQ1J8ooBpBj3wT4HVUsb\",\n" + + " \"k\": 100\n" + + " }\n" + + " }\n" + + " }\n" + + "}"; SearchRequest searchRequest = ModelTestCase.fromJson(json, SearchRequest.class, mapper); From acfda84b2a4995b33cd3133a7f4f9973b35a8315 Mon Sep 17 00:00:00 2001 From: Kirill Ostanin Date: Mon, 23 Oct 2023 12:01:37 +0400 Subject: [PATCH 3/5] Update CHANGELOG.md Signed-off-by: Kirill Ostanin --- CHANGELOG.md | 1 + 1 file changed, 1 insertion(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index dac2b79946..5d9dd2716c 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -49,6 +49,7 @@ This section is for maintaining a changelog for all breaking changes for the cli - Added support for "script_fields" in multi search request ([#632](https://github.com/opensearch-project/opensearch-java/pull/632)) - Added size attribute to MultiTermsAggregation ([#627](https://github.com/opensearch-project/opensearch-java/pull/627)) - Added version increment workflow that executes after release ([#664](https://github.com/opensearch-project/opensearch-java/pull/664)) +- Added support for neural query type ([#674](https://github.com/opensearch-project/opensearch-java/pull/674)) ### Dependencies - Bumps `org.ajoberstar.grgit:grgit-gradle` from 5.0.0 to 5.2.0 From 3b93a1b8288e90e8b66f948186855bc231f33322 Mon Sep 17 00:00:00 2001 From: Kirill Ostanin Date: Tue, 24 Oct 2023 11:09:01 +0400 Subject: [PATCH 4/5] refactoring Signed-off-by: Kirill Ostanin --- .../client/opensearch/_types/query_dsl/NeuralQuery.java | 7 ++----- 1 file changed, 2 insertions(+), 5 deletions(-) diff --git a/java-client/src/main/java/org/opensearch/client/opensearch/_types/query_dsl/NeuralQuery.java b/java-client/src/main/java/org/opensearch/client/opensearch/_types/query_dsl/NeuralQuery.java index e1041db04e..083200fe81 100644 --- a/java-client/src/main/java/org/opensearch/client/opensearch/_types/query_dsl/NeuralQuery.java +++ b/java-client/src/main/java/org/opensearch/client/opensearch/_types/query_dsl/NeuralQuery.java @@ -48,7 +48,7 @@ public static NeuralQuery of(Function implements ObjectBuilder { - @Nullable private String field; - @Nullable private String queryText; + private Integer k; @Nullable private String modelId; - @Nullable - private Integer k; /** * Required - The target field. From 65183f160a5abfe9d7202138039e051fd5540dd9 Mon Sep 17 00:00:00 2001 From: Kirill Ostanin Date: Tue, 24 Oct 2023 12:53:19 +0400 Subject: [PATCH 5/5] attempt to restart integration tests Signed-off-by: Kirill Ostanin --- .../org/opensearch/client/opensearch/model/VariantsTest.java | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/java-client/src/test/java/org/opensearch/client/opensearch/model/VariantsTest.java b/java-client/src/test/java/org/opensearch/client/opensearch/model/VariantsTest.java index 0f3e265c76..e504050c19 100644 --- a/java-client/src/test/java/org/opensearch/client/opensearch/model/VariantsTest.java +++ b/java-client/src/test/java/org/opensearch/client/opensearch/model/VariantsTest.java @@ -228,7 +228,7 @@ public void testNeuralQueryFromJson() { + " \"query\": {\n" + " \"neural\": {\n" + " \"passage_embedding\": {\n" - + " \"query_text\": \"Hi world\",\n" + + " \"query_text\": \"Hi world!\",\n" + " \"model_id\": \"bQ1J8ooBpBj3wT4HVUsb\",\n" + " \"k\": 100\n" + " }\n" @@ -239,7 +239,7 @@ public void testNeuralQueryFromJson() { SearchRequest searchRequest = ModelTestCase.fromJson(json, SearchRequest.class, mapper); assertEquals("passage_embedding", searchRequest.query().neural().field()); - assertEquals("Hi world", searchRequest.query().neural().queryText()); + assertEquals("Hi world!", searchRequest.query().neural().queryText()); assertEquals("bQ1J8ooBpBj3wT4HVUsb", searchRequest.query().neural().modelId()); assertEquals(100, searchRequest.query().neural().k()); }