From 3917fd6c2a562e82675d3b21fc76a0e09e2fc169 Mon Sep 17 00:00:00 2001 From: John Mazanec Date: Tue, 5 Apr 2022 17:25:16 -0400 Subject: [PATCH] Allow null value for params in method mappings (#354) Allow user to input null value for parameters field for KNNMethodContext and MethodComponentContext. Adds tests to reproduce Mapping Parsing Error when the parameters take the value null as well as BWC tests. Signed-off-by: John Mazanec (cherry picked from commit b08127c2c4dcd1844072302e838f44897c8f18f5) --- .../knn/index/KNNMethodContext.java | 7 +- .../knn/index/MethodComponentContext.java | 72 ++++++++++++++----- .../knn/index/KNNMethodContextTests.java | 15 ++++ .../index/MethodComponentContextTests.java | 43 +++++++++++ 4 files changed, 117 insertions(+), 20 deletions(-) diff --git a/src/main/java/org/opensearch/knn/index/KNNMethodContext.java b/src/main/java/org/opensearch/knn/index/KNNMethodContext.java index 68f21ecac..1a268007c 100644 --- a/src/main/java/org/opensearch/knn/index/KNNMethodContext.java +++ b/src/main/java/org/opensearch/knn/index/KNNMethodContext.java @@ -42,7 +42,7 @@ */ public class KNNMethodContext implements ToXContentFragment, Writeable { - private static Logger logger = LogManager.getLogger(KNNMethodContext.class); + private static final Logger logger = LogManager.getLogger(KNNMethodContext.class); private static KNNMethodContext defaultInstance = null; @@ -191,6 +191,11 @@ public static KNNMethodContext parse(Object in) { name = (String) value; } else if (PARAMETERS.equals(key)) { + if (value == null) { + parameters = null; + continue; + } + if (!(value instanceof Map)) { throw new MapperParsingException("Unable to parse parameters for main method component"); } diff --git a/src/main/java/org/opensearch/knn/index/MethodComponentContext.java b/src/main/java/org/opensearch/knn/index/MethodComponentContext.java index 196ea1938..85db38a53 100644 --- a/src/main/java/org/opensearch/knn/index/MethodComponentContext.java +++ b/src/main/java/org/opensearch/knn/index/MethodComponentContext.java @@ -21,6 +21,7 @@ import org.opensearch.index.mapper.MapperParsingException; import java.io.IOException; +import java.util.Collections; import java.util.HashMap; import java.util.Map; import java.util.stream.Collectors; @@ -38,10 +39,10 @@ */ public class MethodComponentContext implements ToXContentFragment, Writeable { - private static Logger logger = LogManager.getLogger(MethodComponentContext.class); + private static final Logger logger = LogManager.getLogger(MethodComponentContext.class); - private String name; - private Map parameters; + private final String name; + private final Map parameters; /** * Constructor @@ -62,7 +63,15 @@ public MethodComponentContext(String name, Map parameters) { */ public MethodComponentContext(StreamInput in) throws IOException { this.name = in.readString(); - this.parameters = in.readMap(StreamInput::readString, new ParameterMapValueReader()); + + // Due to backwards compatibility issue, parameters could be null. To prevent any null pointer exceptions, + // do not read if their are no bytes left is null. Make sure this is in sync with the fellow read method. For + // more information, refer to https://github.com/opensearch-project/k-NN/issues/353. + if (in.available() > 0) { + this.parameters = in.readMap(StreamInput::readString, new ParameterMapValueReader()); + } else { + this.parameters = null; + } } /** @@ -94,6 +103,11 @@ public static MethodComponentContext parse(Object in) { } name = (String) value; } else if (PARAMETERS.equals(key)) { + if (value == null) { + parameters = null; + continue; + } + if (!(value instanceof Map)) { throw new MapperParsingException("Unable to parse parameters for method component"); } @@ -126,22 +140,30 @@ public static MethodComponentContext parse(Object in) { @Override public XContentBuilder toXContent(XContentBuilder builder, Params params) throws IOException { builder.field(NAME, name); - builder.startObject(PARAMETERS); - parameters.forEach((key, value) -> { - try { - if (value instanceof MethodComponentContext) { - builder.startObject(key); - ((MethodComponentContext) value).toXContent(builder, params); - builder.endObject(); - } else { - builder.field(key, value); + // Due to backwards compatibility issue, parameters could be null. To prevent any null pointer exceptions, + // we just create the null field. If parameters are not null, we created a nested structure. For more + // information, refer to https://github.com/opensearch-project/k-NN/issues/353. + if (parameters == null) { + builder.field(PARAMETERS, (String) null); + } else { + builder.startObject(PARAMETERS); + parameters.forEach((key, value) -> { + try { + if (value instanceof MethodComponentContext) { + builder.startObject(key); + ((MethodComponentContext) value).toXContent(builder, params); + builder.endObject(); + } else { + builder.field(key, value); + } + } catch (IOException ioe) { + throw new RuntimeException("Unable to generate xcontent for method component"); } - } catch (IOException ioe) { - throw new RuntimeException("Unable to generate xcontent for method component"); - } - }); - builder.endObject(); + }); + builder.endObject(); + } + return builder; } @@ -179,13 +201,25 @@ public String getName() { * @return parameters */ public Map getParameters() { + // Due to backwards compatibility issue, parameters could be null. To prevent any null pointer exceptions, + // return an empty map if parameters is null. For more information, refer to + // https://github.com/opensearch-project/k-NN/issues/353. + if (parameters == null) { + return Collections.emptyMap(); + } return parameters; } @Override public void writeTo(StreamOutput out) throws IOException { out.writeString(this.name); - out.writeMap(this.parameters, StreamOutput::writeString, new ParameterMapValueWriter()); + + // Due to backwards compatibility issue, parameters could be null. To prevent any null pointer exceptions, + // do not write if parameters is null. Make sure this is in sync with the fellow read method. For more + // information, refer to https://github.com/opensearch-project/k-NN/issues/353. + if (this.parameters != null) { + out.writeMap(this.parameters, StreamOutput::writeString, new ParameterMapValueWriter()); + } } // Because the generic StreamOutput writeMap method can only write generic values, we need to create a custom one diff --git a/src/test/java/org/opensearch/knn/index/KNNMethodContextTests.java b/src/test/java/org/opensearch/knn/index/KNNMethodContextTests.java index 92dbd3eff..b0b584f56 100644 --- a/src/test/java/org/opensearch/knn/index/KNNMethodContextTests.java +++ b/src/test/java/org/opensearch/knn/index/KNNMethodContextTests.java @@ -261,6 +261,21 @@ public void testParse_invalid() throws IOException { expectThrows(MapperParsingException.class, () -> MethodComponentContext.parse(in7)); } + /** + * Test context method parsing when parameters are set to null + */ + public void testParse_nullParameters() throws IOException { + String methodName = "test-method"; + XContentBuilder xContentBuilder = XContentFactory.jsonBuilder() + .startObject() + .field(NAME, methodName) + .field(PARAMETERS, (String) null) + .endObject(); + Map in = xContentBuilderToMap(xContentBuilder); + KNNMethodContext knnMethodContext = KNNMethodContext.parse(in); + assertTrue(knnMethodContext.getMethodComponent().getParameters().isEmpty()); + } + /** * Test context method parsing when input is valid */ diff --git a/src/test/java/org/opensearch/knn/index/MethodComponentContextTests.java b/src/test/java/org/opensearch/knn/index/MethodComponentContextTests.java index bf2c0fc3b..befcce17d 100644 --- a/src/test/java/org/opensearch/knn/index/MethodComponentContextTests.java +++ b/src/test/java/org/opensearch/knn/index/MethodComponentContextTests.java @@ -51,6 +51,13 @@ public void testStreams() throws IOException { MethodComponentContext copy = new MethodComponentContext(streamOutput.bytes().streamInput()); assertEquals(original, copy); + + // Check that everything works when streams are null + original = new MethodComponentContext(name, null); + streamOutput = new BytesStreamOutput(); + original.writeTo(streamOutput); + copy = new MethodComponentContext(streamOutput.bytes().streamInput()); + assertEquals(original, copy); } /** @@ -115,6 +122,25 @@ public void testGetParameters() throws IOException { MethodComponentContext methodContext = new MethodComponentContext(name, params); assertEquals(paramVal1, methodContext.getParameters().get(paramKey1)); assertEquals(paramVal2, methodContext.getParameters().get(paramKey2)); + + // When parameters are null, an empty map should be returned + methodContext = new MethodComponentContext(name, null); + assertTrue(methodContext.getParameters().isEmpty()); + } + + /** + * Test method component context parsing when parameters are set to null + */ + public void testParse_nullParameters() throws IOException { + String name = "test-name"; + XContentBuilder xContentBuilder = XContentFactory.jsonBuilder() + .startObject() + .field(NAME, name) + .field(PARAMETERS, (String) null) + .endObject(); + Map in = xContentBuilderToMap(xContentBuilder); + MethodComponentContext methodComponentContext = MethodComponentContext.parse(in); + assertTrue(methodComponentContext.getParameters().isEmpty()); } /** @@ -213,6 +239,15 @@ public void testToXContent() throws IOException { assertEquals(paramVal1, paramMap.get(paramKey1)); assertEquals(paramVal2, paramMap.get(paramKey2)); + + // Check when parameters are null + xContentBuilder = XContentFactory.jsonBuilder().startObject().field(NAME, name).field(PARAMETERS, (String) null).endObject(); + in = xContentBuilderToMap(xContentBuilder); + methodContext = MethodComponentContext.parse(in); + builder = XContentFactory.jsonBuilder().startObject(); + builder = methodContext.toXContent(builder, ToXContent.EMPTY_PARAMS).endObject(); + out = xContentBuilderToMap(builder); + assertNull(out.get(PARAMETERS)); } public void testEquals() { @@ -233,11 +268,15 @@ public void testEquals() { MethodComponentContext methodContext1 = new MethodComponentContext(name1, parameters1); MethodComponentContext methodContext2 = new MethodComponentContext(name1, parameters1); MethodComponentContext methodContext3 = new MethodComponentContext(name2, parameters2); + MethodComponentContext methodContext4 = new MethodComponentContext(name2, null); + MethodComponentContext methodContext5 = new MethodComponentContext(name2, null); assertEquals(methodContext1, methodContext1); assertEquals(methodContext1, methodContext2); assertNotEquals(methodContext1, methodContext3); assertNotEquals(methodContext1, null); + assertNotEquals(methodContext2, methodContext4); + assertEquals(methodContext4, methodContext5); } public void testHashCode() { @@ -257,9 +296,13 @@ public void testHashCode() { MethodComponentContext methodContext1 = new MethodComponentContext(name1, parameters1); MethodComponentContext methodContext2 = new MethodComponentContext(name1, parameters1); MethodComponentContext methodContext3 = new MethodComponentContext(name2, parameters2); + MethodComponentContext methodContext4 = new MethodComponentContext(name2, null); + MethodComponentContext methodContext5 = new MethodComponentContext(name2, null); assertEquals(methodContext1.hashCode(), methodContext1.hashCode()); assertEquals(methodContext1.hashCode(), methodContext2.hashCode()); assertNotEquals(methodContext1.hashCode(), methodContext3.hashCode()); + assertNotEquals(methodContext1.hashCode(), methodContext4.hashCode()); + assertEquals(methodContext4.hashCode(), methodContext5.hashCode()); } }