From 1100cc7dfec9afb2cbb927263cfe408f7e5003b7 Mon Sep 17 00:00:00 2001 From: Sicheng Song Date: Tue, 11 Jul 2023 23:44:49 +0000 Subject: [PATCH 1/4] Add more UT for remote inference classes Signed-off-by: Sicheng Song --- .../MLConnectorDeleteRequestTests.java | 100 ++++++++++++ .../connector/MLConnectorGetRequestTests.java | 98 ++++++++++++ .../MLConnectorGetResponseTests.java | 107 +++++++++++++ .../MLCreateConnectorInputTests.java | 9 ++ .../MLCreateConnectorRequestTest.java | 1 + .../MLCreateConnectorRequestTests.java | 149 ++++++++++++++++++ ...va => MLCreateConnectorResponseTests.java} | 2 +- .../model/MLModelDeleteRequestTest.java | 20 +++ .../model/MLModelGetRequestTest.java | 17 ++ .../model/MLModelGetResponseTest.java | 36 +++++ .../MLRegisterModelGroupRequestTest.java | 13 +- 11 files changed, 544 insertions(+), 8 deletions(-) create mode 100644 common/src/test/java/org/opensearch/ml/common/transport/connector/MLConnectorDeleteRequestTests.java create mode 100644 common/src/test/java/org/opensearch/ml/common/transport/connector/MLConnectorGetRequestTests.java create mode 100644 common/src/test/java/org/opensearch/ml/common/transport/connector/MLConnectorGetResponseTests.java create mode 100644 common/src/test/java/org/opensearch/ml/common/transport/connector/MLCreateConnectorInputTests.java create mode 100644 common/src/test/java/org/opensearch/ml/common/transport/connector/MLCreateConnectorRequestTests.java rename common/src/test/java/org/opensearch/ml/common/transport/connector/{MLCreateConnectorResponseTest.java => MLCreateConnectorResponseTests.java} (96%) diff --git a/common/src/test/java/org/opensearch/ml/common/transport/connector/MLConnectorDeleteRequestTests.java b/common/src/test/java/org/opensearch/ml/common/transport/connector/MLConnectorDeleteRequestTests.java new file mode 100644 index 0000000000..27bb438599 --- /dev/null +++ b/common/src/test/java/org/opensearch/ml/common/transport/connector/MLConnectorDeleteRequestTests.java @@ -0,0 +1,100 @@ +/* + * Copyright OpenSearch Contributors + * SPDX-License-Identifier: Apache-2.0 + */ + +package org.opensearch.ml.common.transport.connector; + +import org.junit.Before; +import org.junit.Test; +import org.opensearch.action.ActionRequest; +import org.opensearch.action.ActionRequestValidationException; +import org.opensearch.common.io.stream.BytesStreamOutput; +import org.opensearch.common.io.stream.StreamOutput; + +import java.io.IOException; +import java.io.UncheckedIOException; + +import static org.junit.Assert.assertEquals; +import static org.junit.Assert.assertNotSame; +import static org.junit.Assert.assertNull; +import static org.junit.Assert.assertSame; + +public class MLConnectorDeleteRequestTests { + private String connectorId; + + @Before + public void setUp() { + connectorId = "test-connector-id"; + } + + @Test + public void writeTo_Success() throws IOException { + MLConnectorDeleteRequest mlConnectorDeleteRequest = MLConnectorDeleteRequest.builder() + .connectorId(connectorId).build(); + BytesStreamOutput bytesStreamOutput = new BytesStreamOutput(); + mlConnectorDeleteRequest.writeTo(bytesStreamOutput); + MLConnectorDeleteRequest parsedConnector = new MLConnectorDeleteRequest(bytesStreamOutput.bytes().streamInput()); + assertEquals(parsedConnector.getConnectorId(), connectorId); + } + + @Test + public void valid_Exception_NullConnectorId() { + MLConnectorDeleteRequest mlConnectorDeleteRequest = MLConnectorDeleteRequest.builder().build(); + ActionRequestValidationException exception = mlConnectorDeleteRequest.validate(); + assertEquals("Validation Failed: 1: ML connector id can't be null;", exception.getMessage()); + } + + @Test + public void validate_Success() { + MLConnectorDeleteRequest mlConnectorDeleteRequest = MLConnectorDeleteRequest.builder() + .connectorId(connectorId).build(); + ActionRequestValidationException actionRequestValidationException = mlConnectorDeleteRequest.validate(); + assertNull(actionRequestValidationException); + } + + @Test + public void fromActionRequest_Success() { + MLConnectorDeleteRequest mlConnectorDeleteRequest = MLConnectorDeleteRequest.builder() + .connectorId(connectorId).build(); + ActionRequest actionRequest = new ActionRequest() { + @Override + public ActionRequestValidationException validate() { + return null; + } + + @Override + public void writeTo(StreamOutput out) throws IOException { + mlConnectorDeleteRequest.writeTo(out); + } + }; + MLConnectorDeleteRequest parsedConnector = MLConnectorDeleteRequest.fromActionRequest(actionRequest); + assertNotSame(parsedConnector, mlConnectorDeleteRequest); + assertEquals(parsedConnector.getConnectorId(), connectorId); + } + + @Test(expected = UncheckedIOException.class) + public void fromActionRequest_IOException() { + ActionRequest actionRequest = new ActionRequest() { + @Override + public ActionRequestValidationException validate() { + return null; + } + + @Override + public void writeTo(StreamOutput out) throws IOException { + throw new IOException(); + } + }; + MLConnectorDeleteRequest.fromActionRequest(actionRequest); + } + + @Test + public void fromActionRequestWithConnectorDeleteRequest_Success() { + MLConnectorDeleteRequest mlConnectorDeleteRequest = MLConnectorDeleteRequest.builder() + .connectorId(connectorId).build(); + MLConnectorDeleteRequest mlConnectorDeleteRequestFromActionRequest = MLConnectorDeleteRequest.fromActionRequest(mlConnectorDeleteRequest); + assertSame(mlConnectorDeleteRequest, mlConnectorDeleteRequestFromActionRequest); + assertEquals(mlConnectorDeleteRequest.getConnectorId(), mlConnectorDeleteRequestFromActionRequest.getConnectorId()); + } +} diff --git a/common/src/test/java/org/opensearch/ml/common/transport/connector/MLConnectorGetRequestTests.java b/common/src/test/java/org/opensearch/ml/common/transport/connector/MLConnectorGetRequestTests.java new file mode 100644 index 0000000000..0113aca7d6 --- /dev/null +++ b/common/src/test/java/org/opensearch/ml/common/transport/connector/MLConnectorGetRequestTests.java @@ -0,0 +1,98 @@ +/* + * Copyright OpenSearch Contributors + * SPDX-License-Identifier: Apache-2.0 + */ + + +package org.opensearch.ml.common.transport.connector; + +import java.io.IOException; +import java.io.UncheckedIOException; + +import org.junit.Before; +import org.junit.Test; +import org.opensearch.action.ActionRequest; +import org.opensearch.action.ActionRequestValidationException; +import org.opensearch.common.io.stream.BytesStreamOutput; +import org.opensearch.common.io.stream.StreamOutput; + +import static org.junit.Assert.assertEquals; +import static org.junit.Assert.assertNotSame; +import static org.junit.Assert.assertNull; +import static org.junit.Assert.assertSame; + +public class MLConnectorGetRequestTests { + private String connectorId; + + @Before + public void setUp() { + connectorId = "test-connector-id"; + } + + @Test + public void writeTo_Success() throws IOException { + MLConnectorGetRequest mlConnectorGetRequest = MLConnectorGetRequest.builder().connectorId(connectorId).build(); + BytesStreamOutput bytesStreamOutput = new BytesStreamOutput(); + mlConnectorGetRequest.writeTo(bytesStreamOutput); + MLConnectorGetRequest parsedConnector = new MLConnectorGetRequest(bytesStreamOutput.bytes().streamInput()); + assertEquals(connectorId, parsedConnector.getConnectorId()); + } + + @Test + public void fromActionRequest_Success() { + MLConnectorGetRequest mlConnectorGetRequest = MLConnectorGetRequest.builder().connectorId(connectorId).build(); + ActionRequest actionRequest = new ActionRequest() { + @Override + public ActionRequestValidationException validate() { + return null; + } + + @Override + public void writeTo(StreamOutput out) throws IOException { + mlConnectorGetRequest.writeTo(out); + } + }; + MLConnectorGetRequest mlConnectorGetRequestFromActionRequest = MLConnectorGetRequest.fromActionRequest(actionRequest); + assertNotSame(mlConnectorGetRequest, mlConnectorGetRequestFromActionRequest); + assertEquals(mlConnectorGetRequest.getConnectorId(), mlConnectorGetRequestFromActionRequest.getConnectorId()); + } + + @Test(expected = UncheckedIOException.class) + public void fromActionRequest_IOException() { + ActionRequest actionRequest = new ActionRequest() { + @Override + public ActionRequestValidationException validate() { + return null; + } + + @Override + public void writeTo(StreamOutput out) throws IOException { + throw new IOException(); + } + }; + MLConnectorGetRequest.fromActionRequest(actionRequest); + } + + @Test + public void fromActionRequestWithMLConnectorGetRequest_Success() { + MLConnectorGetRequest mlConnectorGetRequest = MLConnectorGetRequest.builder().connectorId(connectorId).build(); + MLConnectorGetRequest mlConnectorGetRequestFromActionRequest = MLConnectorGetRequest.fromActionRequest(mlConnectorGetRequest); + assertSame(mlConnectorGetRequest, mlConnectorGetRequestFromActionRequest); + assertEquals(mlConnectorGetRequest.getConnectorId(), mlConnectorGetRequestFromActionRequest.getConnectorId()); + } + + @Test + public void validate_Exception_NullConnctorId() { + MLConnectorGetRequest mlConnectorGetRequest = MLConnectorGetRequest.builder().build(); + ActionRequestValidationException actionRequestValidationException = mlConnectorGetRequest.validate(); + assertEquals("Validation Failed: 1: ML connector id can't be null;", actionRequestValidationException.getMessage()); + } + + @Test + public void validate_Success() { + MLConnectorGetRequest mlConnectorGetRequest = MLConnectorGetRequest.builder().connectorId(connectorId).build(); + ActionRequestValidationException actionRequestValidationException = mlConnectorGetRequest.validate(); + assertNull(actionRequestValidationException); + } +} + diff --git a/common/src/test/java/org/opensearch/ml/common/transport/connector/MLConnectorGetResponseTests.java b/common/src/test/java/org/opensearch/ml/common/transport/connector/MLConnectorGetResponseTests.java new file mode 100644 index 0000000000..69a19255e8 --- /dev/null +++ b/common/src/test/java/org/opensearch/ml/common/transport/connector/MLConnectorGetResponseTests.java @@ -0,0 +1,107 @@ +/* + * Copyright OpenSearch Contributors + * SPDX-License-Identifier: Apache-2.0 + */ + +package org.opensearch.ml.common.transport.connector; + +import org.junit.Before; +import org.junit.Test; +import org.opensearch.action.ActionResponse; +import org.opensearch.common.Strings; +import org.opensearch.common.io.stream.BytesStreamOutput; +import org.opensearch.common.io.stream.StreamOutput; +import org.opensearch.common.xcontent.XContentFactory; +import org.opensearch.common.xcontent.XContentType; +import org.opensearch.core.xcontent.ToXContent; +import org.opensearch.core.xcontent.XContentBuilder; +import org.opensearch.ml.common.connector.Connector; +import org.opensearch.ml.common.connector.HttpConnectorTest; + +import java.io.IOException; +import java.io.UncheckedIOException; + +import static org.junit.Assert.assertEquals; +import static org.junit.Assert.assertNotEquals; +import static org.junit.Assert.assertNotNull; +import static org.junit.Assert.assertNotSame; +import static org.junit.Assert.assertSame; + +public class MLConnectorGetResponseTests { + Connector connector; + + @Before + public void setUp() { + connector = HttpConnectorTest.createHttpConnector(); + } + + @Test + public void writeTo_Success() throws IOException { + BytesStreamOutput bytesStreamOutput = new BytesStreamOutput(); + MLConnectorGetResponse response = MLConnectorGetResponse.builder().mlConnector(connector).build(); + response.writeTo(bytesStreamOutput); + MLConnectorGetResponse parsedResponse = new MLConnectorGetResponse(bytesStreamOutput.bytes().streamInput()); + assertNotEquals(response, parsedResponse); + assertNotSame(response.mlConnector, parsedResponse.mlConnector); + assertEquals(response.mlConnector, parsedResponse.mlConnector); + assertEquals(response.mlConnector.getName(), parsedResponse.mlConnector.getName()); + assertEquals(response.mlConnector.getAccess(), parsedResponse.mlConnector.getAccess()); + assertEquals(response.mlConnector.getProtocol(), parsedResponse.mlConnector.getProtocol()); + assertEquals(response.mlConnector.getDecryptedHeaders(), parsedResponse.mlConnector.getDecryptedHeaders()); + assertEquals(response.mlConnector.getBackendRoles(), parsedResponse.mlConnector.getBackendRoles()); + assertEquals(response.mlConnector.getActions(), parsedResponse.mlConnector.getActions()); + assertEquals(response.mlConnector.getParameters(), parsedResponse.mlConnector.getParameters()); + } + + @Test + public void toXContentTest() throws IOException { + MLConnectorGetResponse mlConnectorGetResponse = MLConnectorGetResponse.builder().mlConnector(connector).build(); + XContentBuilder builder = XContentFactory.contentBuilder(XContentType.JSON); + mlConnectorGetResponse.toXContent(builder, ToXContent.EMPTY_PARAMS); + assertNotNull(builder); + String jsonStr = Strings.toString(builder); + assertEquals("{\"name\":\"test_connector_name\"," + + "\"version\":\"1\",\"description\":\"this is a test connector\",\"protocol\":\"http\"," + + "\"parameters\":{\"input\":\"test input value\"},\"credential\":{\"key\":\"test_key_value\"}," + + "\"actions\":[{\"action_type\":\"PREDICT\",\"method\":\"POST\",\"url\":\"https://test.com\"," + + "\"headers\":{\"api_key\":\"${credential.key}\"}," + + "\"request_body\":\"{\\\"input\\\": \\\"${parameters.input}\\\"}\"," + + "\"pre_process_function\":\"connector.pre_process.openai.embedding\"," + + "\"post_process_function\":\"connector.post_process.openai.embedding\"}]," + + "\"backend_roles\":[\"role1\",\"role2\"]," + + "\"access\":\"public\"}", jsonStr); + } + + @Test + public void fromActionResponseWithMLConnectorGetResponse_Success() { + MLConnectorGetResponse mlConnectorGetResponse = MLConnectorGetResponse.builder().mlConnector(connector).build(); + MLConnectorGetResponse mlConnectorGetResponseFromActionResponse = MLConnectorGetResponse.fromActionResponse(mlConnectorGetResponse); + assertSame(mlConnectorGetResponse, mlConnectorGetResponseFromActionResponse); + assertEquals(mlConnectorGetResponse.mlConnector, mlConnectorGetResponseFromActionResponse.mlConnector); + } + + @Test + public void fromActionResponse_Success() { + MLConnectorGetResponse mlConnectorGetResponse = MLConnectorGetResponse.builder().mlConnector(connector).build(); + ActionResponse actionResponse = new ActionResponse() { + @Override + public void writeTo(StreamOutput out) throws IOException { + mlConnectorGetResponse.writeTo(out); + } + }; + MLConnectorGetResponse mlConnectorGetResponseFromActionResponse = MLConnectorGetResponse.fromActionResponse(actionResponse); + assertNotSame(mlConnectorGetResponse, mlConnectorGetResponseFromActionResponse); + assertEquals(mlConnectorGetResponse.mlConnector, mlConnectorGetResponseFromActionResponse.mlConnector); + } + + @Test(expected = UncheckedIOException.class) + public void fromActionResponse_IOException() { + ActionResponse actionResponse = new ActionResponse() { + @Override + public void writeTo(StreamOutput out) throws IOException { + throw new IOException(); + } + }; + MLConnectorGetResponse.fromActionResponse(actionResponse); + } +} diff --git a/common/src/test/java/org/opensearch/ml/common/transport/connector/MLCreateConnectorInputTests.java b/common/src/test/java/org/opensearch/ml/common/transport/connector/MLCreateConnectorInputTests.java new file mode 100644 index 0000000000..879ac0a585 --- /dev/null +++ b/common/src/test/java/org/opensearch/ml/common/transport/connector/MLCreateConnectorInputTests.java @@ -0,0 +1,9 @@ +/* + * Copyright OpenSearch Contributors + * SPDX-License-Identifier: Apache-2.0 + */ + +package org.opensearch.ml.common.transport.connector; + +public class MLCreateConnectorInputTests { +} diff --git a/common/src/test/java/org/opensearch/ml/common/transport/connector/MLCreateConnectorRequestTest.java b/common/src/test/java/org/opensearch/ml/common/transport/connector/MLCreateConnectorRequestTest.java index 4fb636a6e7..0debb02ee6 100644 --- a/common/src/test/java/org/opensearch/ml/common/transport/connector/MLCreateConnectorRequestTest.java +++ b/common/src/test/java/org/opensearch/ml/common/transport/connector/MLCreateConnectorRequestTest.java @@ -35,6 +35,7 @@ public void readFromStream() throws IOException { .protocol("http") .version("1") .description("test") + .addAllBackendRoles(true) .build(); MLCreateConnectorRequest request = new MLCreateConnectorRequest(input); BytesStreamOutput output = new BytesStreamOutput(); diff --git a/common/src/test/java/org/opensearch/ml/common/transport/connector/MLCreateConnectorRequestTests.java b/common/src/test/java/org/opensearch/ml/common/transport/connector/MLCreateConnectorRequestTests.java new file mode 100644 index 0000000000..368a30acc6 --- /dev/null +++ b/common/src/test/java/org/opensearch/ml/common/transport/connector/MLCreateConnectorRequestTests.java @@ -0,0 +1,149 @@ +/* + * Copyright OpenSearch Contributors + * SPDX-License-Identifier: Apache-2.0 + */ + +package org.opensearch.ml.common.transport.connector; + +import java.io.IOException; +import java.io.UncheckedIOException; +import java.util.Arrays; +import java.util.HashMap; +import java.util.List; +import java.util.Map; + +import org.junit.Before; +import org.junit.Ignore; +import org.junit.Test; +import org.opensearch.action.ActionRequest; +import org.opensearch.action.ActionRequestValidationException; +import org.opensearch.common.io.stream.BytesStreamOutput; +import org.opensearch.common.io.stream.StreamOutput; +import org.opensearch.ml.common.AccessMode; +import org.opensearch.ml.common.connector.ConnectorAction; +import org.opensearch.ml.common.connector.MLPostProcessFunction; +import org.opensearch.ml.common.connector.MLPreProcessFunction; + +import static org.junit.Assert.assertEquals; +import static org.junit.Assert.assertNotEquals; +import static org.junit.Assert.assertNotSame; +import static org.junit.Assert.assertNull; +import static org.junit.Assert.assertSame; + +public class MLCreateConnectorRequestTests { + private MLCreateConnectorInput mlCreateConnectorInput; + + @Before + public void setUp(){ + ConnectorAction.ActionType actionType = ConnectorAction.ActionType.PREDICT; + String method = "POST"; + String url = "https://test.com"; + Map headers = new HashMap<>(); + headers.put("api_key", "${credential.key}"); + String mlCreateConnectorRequestBody = "{\"input\": \"${parameters.input}\"}"; + String preProcessFunction = MLPreProcessFunction.TEXT_DOCS_TO_OPENAI_EMBEDDING_INPUT; + String postProcessFunction = MLPostProcessFunction.OPENAI_EMBEDDING; + ConnectorAction action = new ConnectorAction(actionType, method, url, headers, mlCreateConnectorRequestBody, preProcessFunction, postProcessFunction); + + // java.lang.IllegalStateException: unexpected byte [0x6f] will be thrown if we specify backendRoles field. + mlCreateConnectorInput = MLCreateConnectorInput.builder() + .name("test_connector_name") + .description("this is a test connector") + .version("1") + .protocol("http") + .parameters(Map.of("input", "test input value")) + .credential(Map.of("key", "test_key_value")) + .actions(List.of(action)) + .access(AccessMode.PUBLIC) + .addAllBackendRoles(true) + .build(); + } + + @Test + public void writeTo_Success() throws IOException { + MLCreateConnectorRequest mlCreateConnectorRequest = MLCreateConnectorRequest.builder().mlCreateConnectorInput(mlCreateConnectorInput).build(); + BytesStreamOutput output = new BytesStreamOutput(); + mlCreateConnectorRequest.writeTo(output); + MLCreateConnectorRequest parsedRequest = new MLCreateConnectorRequest(output.bytes().streamInput()); + assertEquals(mlCreateConnectorRequest.getMlCreateConnectorInput().getName(), parsedRequest.getMlCreateConnectorInput().getName()); + assertEquals(mlCreateConnectorRequest.getMlCreateConnectorInput().getAccess(), parsedRequest.getMlCreateConnectorInput().getAccess()); + assertEquals(mlCreateConnectorRequest.getMlCreateConnectorInput().getProtocol(), parsedRequest.getMlCreateConnectorInput().getProtocol()); + assertEquals(mlCreateConnectorRequest.getMlCreateConnectorInput().getBackendRoles(), parsedRequest.getMlCreateConnectorInput().getBackendRoles()); + assertEquals(mlCreateConnectorRequest.getMlCreateConnectorInput().getActions(), parsedRequest.getMlCreateConnectorInput().getActions()); + assertEquals(mlCreateConnectorRequest.getMlCreateConnectorInput().getParameters(), parsedRequest.getMlCreateConnectorInput().getParameters()); + } + + @Test + public void validate_Success() { + MLCreateConnectorRequest mlCreateConnectorRequest = MLCreateConnectorRequest.builder() + .mlCreateConnectorInput(mlCreateConnectorInput) + .build(); + + assertNull(mlCreateConnectorRequest.validate()); + } + + @Test + public void validate_Exception_NullMLRegisterModelGroupInput() { + MLCreateConnectorRequest mlCreateConnectorRequest = MLCreateConnectorRequest.builder() + .build(); + ActionRequestValidationException exception = mlCreateConnectorRequest.validate(); + assertEquals("Validation Failed: 1: ML Connector input can't be null;", exception.getMessage()); + } + + @Test + // MLCreateConnectorInput check its parameters when created, so exception is not thrown here + public void validate_Exception_NullMLModelName() { + mlCreateConnectorInput.setName(null); + MLCreateConnectorRequest mlCreateConnectorRequest = MLCreateConnectorRequest.builder() + .mlCreateConnectorInput(mlCreateConnectorInput) + .build(); + + assertNull(mlCreateConnectorRequest.validate()); + assertNull(mlCreateConnectorRequest.getMlCreateConnectorInput().getName()); + } + + @Test + public void fromActionRequest_Success_WithMLRegisterModelRequest() { + MLCreateConnectorRequest mlCreateConnectorRequest = MLCreateConnectorRequest.builder() + .mlCreateConnectorInput(mlCreateConnectorInput) + .build(); + assertSame(MLCreateConnectorRequest.fromActionRequest(mlCreateConnectorRequest), mlCreateConnectorRequest); + } + + @Test + public void fromActionRequest_Success_WithNonMLRegisterModelRequest() { + MLCreateConnectorRequest mlCreateConnectorRequest = MLCreateConnectorRequest.builder() + .mlCreateConnectorInput(mlCreateConnectorInput) + .build(); + ActionRequest actionRequest = new ActionRequest() { + @Override + public ActionRequestValidationException validate() { + return null; + } + + @Override + public void writeTo(StreamOutput out) throws IOException { + mlCreateConnectorRequest.writeTo(out); + } + }; + MLCreateConnectorRequest result = MLCreateConnectorRequest.fromActionRequest(actionRequest); + assertNotSame(result, mlCreateConnectorRequest); + assertEquals(mlCreateConnectorRequest.getMlCreateConnectorInput().getName(), result.getMlCreateConnectorInput().getName()); + } + + @Test(expected = UncheckedIOException.class) + public void fromActionRequest_IOException() { + ActionRequest actionRequest = new ActionRequest() { + @Override + public ActionRequestValidationException validate() { + return null; + } + + @Override + public void writeTo(StreamOutput out) throws IOException { + throw new IOException(); + } + }; + MLCreateConnectorRequest.fromActionRequest(actionRequest); + } +} diff --git a/common/src/test/java/org/opensearch/ml/common/transport/connector/MLCreateConnectorResponseTest.java b/common/src/test/java/org/opensearch/ml/common/transport/connector/MLCreateConnectorResponseTests.java similarity index 96% rename from common/src/test/java/org/opensearch/ml/common/transport/connector/MLCreateConnectorResponseTest.java rename to common/src/test/java/org/opensearch/ml/common/transport/connector/MLCreateConnectorResponseTests.java index 4a829d16f7..8d58047980 100644 --- a/common/src/test/java/org/opensearch/ml/common/transport/connector/MLCreateConnectorResponseTest.java +++ b/common/src/test/java/org/opensearch/ml/common/transport/connector/MLCreateConnectorResponseTests.java @@ -15,7 +15,7 @@ import java.io.IOException; -public class MLCreateConnectorResponseTest { +public class MLCreateConnectorResponseTests { @Test public void toXContent() throws IOException { diff --git a/common/src/test/java/org/opensearch/ml/common/transport/model/MLModelDeleteRequestTest.java b/common/src/test/java/org/opensearch/ml/common/transport/model/MLModelDeleteRequestTest.java index b8a4f45b3e..ddc053ac8e 100644 --- a/common/src/test/java/org/opensearch/ml/common/transport/model/MLModelDeleteRequestTest.java +++ b/common/src/test/java/org/opensearch/ml/common/transport/model/MLModelDeleteRequestTest.java @@ -17,6 +17,8 @@ import static org.junit.Assert.assertEquals; import static org.junit.Assert.assertNotSame; +import static org.junit.Assert.assertNull; +import static org.junit.Assert.assertSame; public class MLModelDeleteRequestTest { private String modelId; @@ -36,6 +38,14 @@ public void writeTo_Success() throws IOException { assertEquals(parsedModel.getModelId(), modelId); } + @Test + public void validate_Success() { + MLModelDeleteRequest mlModelDeleteRequest = MLModelDeleteRequest.builder() + .modelId(modelId).build(); + ActionRequestValidationException actionRequestValidationException = mlModelDeleteRequest.validate(); + assertNull(actionRequestValidationException); + } + @Test public void validate_Exception_NullModelId() { MLModelDeleteRequest mlModelDeleteRequest = MLModelDeleteRequest.builder().build(); @@ -79,4 +89,14 @@ public void writeTo(StreamOutput out) throws IOException { }; MLModelDeleteRequest.fromActionRequest(actionRequest); } + + + @Test + public void fromActionRequestWithModelDeleteRequest_Success() { + MLModelDeleteRequest mlModelDeleteRequest = MLModelDeleteRequest.builder() + .modelId(modelId).build(); + MLModelDeleteRequest mlModelDeleteRequestFromActionRequest = MLModelDeleteRequest.fromActionRequest(mlModelDeleteRequest); + assertSame(mlModelDeleteRequest, mlModelDeleteRequestFromActionRequest); + assertEquals(mlModelDeleteRequest.getModelId(), mlModelDeleteRequestFromActionRequest.getModelId()); + } } diff --git a/common/src/test/java/org/opensearch/ml/common/transport/model/MLModelGetRequestTest.java b/common/src/test/java/org/opensearch/ml/common/transport/model/MLModelGetRequestTest.java index 94def0df1d..53b40cf0cc 100644 --- a/common/src/test/java/org/opensearch/ml/common/transport/model/MLModelGetRequestTest.java +++ b/common/src/test/java/org/opensearch/ml/common/transport/model/MLModelGetRequestTest.java @@ -17,6 +17,8 @@ import static org.junit.Assert.assertEquals; import static org.junit.Assert.assertNotSame; +import static org.junit.Assert.assertNull; +import static org.junit.Assert.assertSame; public class MLModelGetRequestTest { private String modelId; @@ -79,4 +81,19 @@ public void writeTo(StreamOutput out) throws IOException { }; MLModelGetRequest.fromActionRequest(actionRequest); } + + @Test + public void validate_Success() { + MLModelGetRequest mlModelGetRequest = MLModelGetRequest.builder().modelId(modelId).build(); + ActionRequestValidationException actionRequestValidationException = mlModelGetRequest.validate(); + assertNull(actionRequestValidationException); + } + + @Test + public void fromActionRequestWithMLModelGetRequest_Success() { + MLModelGetRequest mlModelGetRequest = MLModelGetRequest.builder().modelId(modelId).build(); + MLModelGetRequest mlModelGetRequestFromActionRequest = MLModelGetRequest.fromActionRequest(mlModelGetRequest); + assertSame(mlModelGetRequest, mlModelGetRequestFromActionRequest); + assertEquals(mlModelGetRequest.getModelId(), mlModelGetRequestFromActionRequest.getModelId()); + } } diff --git a/common/src/test/java/org/opensearch/ml/common/transport/model/MLModelGetResponseTest.java b/common/src/test/java/org/opensearch/ml/common/transport/model/MLModelGetResponseTest.java index da9d1869de..c7d68e9cc5 100644 --- a/common/src/test/java/org/opensearch/ml/common/transport/model/MLModelGetResponseTest.java +++ b/common/src/test/java/org/opensearch/ml/common/transport/model/MLModelGetResponseTest.java @@ -7,8 +7,10 @@ import org.junit.Before; import org.junit.Test; +import org.opensearch.action.ActionResponse; import org.opensearch.common.Strings; import org.opensearch.common.io.stream.BytesStreamOutput; +import org.opensearch.common.io.stream.StreamOutput; import org.opensearch.common.xcontent.XContentFactory; import org.opensearch.common.xcontent.XContentType; import org.opensearch.commons.authuser.User; @@ -19,6 +21,7 @@ import org.opensearch.ml.common.model.MLModelState; import java.io.IOException; +import java.io.UncheckedIOException; import static org.junit.Assert.*; import static org.junit.Assert.assertEquals; @@ -66,4 +69,37 @@ public void toXContentTest() throws IOException { "\"model_content\":\"content\"," + "\"user\":{\"name\":\"\",\"backend_roles\":[],\"roles\":[],\"custom_attribute_names\":[],\"user_requested_tenant\":null},\"model_state\":\"TRAINED\"}", jsonStr); } + + @Test + public void fromActionResponseWithMLModelGetResponse_Success() { + MLModelGetResponse mlModelGetResponse = MLModelGetResponse.builder().mlModel(mlModel).build(); + MLModelGetResponse mlModelGetResponseFromActionResponse = MLModelGetResponse.fromActionResponse(mlModelGetResponse); + assertSame(mlModelGetResponse, mlModelGetResponseFromActionResponse); + assertEquals(mlModelGetResponse.mlModel, mlModelGetResponseFromActionResponse.mlModel); + } + + @Test + public void fromActionResponse_Success() { + MLModelGetResponse mlModelGetResponse = MLModelGetResponse.builder().mlModel(mlModel).build(); + ActionResponse actionResponse = new ActionResponse() { + @Override + public void writeTo(StreamOutput out) throws IOException { + mlModelGetResponse.writeTo(out); + } + }; + MLModelGetResponse mlModelGetResponseFromActionResponse = MLModelGetResponse.fromActionResponse(actionResponse); + assertNotSame(mlModelGetResponse, mlModelGetResponseFromActionResponse); + assertNotEquals(mlModelGetResponse.mlModel, mlModelGetResponseFromActionResponse.mlModel); + } + + @Test(expected = UncheckedIOException.class) + public void fromActionResponse_IOException() { + ActionResponse actionResponse = new ActionResponse() { + @Override + public void writeTo(StreamOutput out) throws IOException { + throw new IOException(); + } + }; + MLModelGetResponse.fromActionResponse(actionResponse); + } } diff --git a/common/src/test/java/org/opensearch/ml/common/transport/model_group/MLRegisterModelGroupRequestTest.java b/common/src/test/java/org/opensearch/ml/common/transport/model_group/MLRegisterModelGroupRequestTest.java index 6fe149126f..f06f23c8ba 100644 --- a/common/src/test/java/org/opensearch/ml/common/transport/model_group/MLRegisterModelGroupRequestTest.java +++ b/common/src/test/java/org/opensearch/ml/common/transport/model_group/MLRegisterModelGroupRequestTest.java @@ -35,18 +35,17 @@ public void setUp(){ @Test public void writeTo_Success() throws IOException { - MLRegisterModelGroupRequest request = MLRegisterModelGroupRequest.builder() .registerModelGroupInput(mlRegisterModelGroupInput) .build(); BytesStreamOutput bytesStreamOutput = new BytesStreamOutput(); request.writeTo(bytesStreamOutput); - request = new MLRegisterModelGroupRequest(bytesStreamOutput.bytes().streamInput()); - assertEquals("name", request.getRegisterModelGroupInput().getName()); - assertEquals("description", request.getRegisterModelGroupInput().getDescription()); - assertEquals("IT", request.getRegisterModelGroupInput().getBackendRoles().get(0)); - assertEquals(AccessMode.RESTRICTED, request.getRegisterModelGroupInput().getModelAccessMode()); - assertEquals(true, request.getRegisterModelGroupInput().getIsAddAllBackendRoles()); + MLRegisterModelGroupRequest parsedRequest = new MLRegisterModelGroupRequest(bytesStreamOutput.bytes().streamInput()); + assertEquals(request.getRegisterModelGroupInput().getName(), parsedRequest.getRegisterModelGroupInput().getName()); + assertEquals(request.getRegisterModelGroupInput().getDescription(), parsedRequest.getRegisterModelGroupInput().getDescription()); + assertEquals(request.getRegisterModelGroupInput().getBackendRoles().get(0), parsedRequest.getRegisterModelGroupInput().getBackendRoles().get(0)); + assertEquals(request.getRegisterModelGroupInput().getModelAccessMode(), parsedRequest.getRegisterModelGroupInput().getModelAccessMode()); + assertEquals(request.getRegisterModelGroupInput().getIsAddAllBackendRoles() ,parsedRequest.getRegisterModelGroupInput().getIsAddAllBackendRoles()); } @Test From baf40312f3e8c36497ec00704bdac962a43cddce Mon Sep 17 00:00:00 2001 From: Sicheng Song Date: Wed, 12 Jul 2023 00:07:23 +0000 Subject: [PATCH 2/4] Add UTs Signed-off-by: Sicheng Song --- .../MLCreateConnectorInputTests.java | 61 ++++++++++++++++ .../MLCreateConnectorRequestTest.java | 73 ------------------- .../MLCreateConnectorRequestTests.java | 15 ---- 3 files changed, 61 insertions(+), 88 deletions(-) delete mode 100644 common/src/test/java/org/opensearch/ml/common/transport/connector/MLCreateConnectorRequestTest.java diff --git a/common/src/test/java/org/opensearch/ml/common/transport/connector/MLCreateConnectorInputTests.java b/common/src/test/java/org/opensearch/ml/common/transport/connector/MLCreateConnectorInputTests.java index 879ac0a585..4de9198785 100644 --- a/common/src/test/java/org/opensearch/ml/common/transport/connector/MLCreateConnectorInputTests.java +++ b/common/src/test/java/org/opensearch/ml/common/transport/connector/MLCreateConnectorInputTests.java @@ -5,5 +5,66 @@ package org.opensearch.ml.common.transport.connector; +import java.io.IOException; +import java.io.UncheckedIOException; +import java.util.HashMap; +import java.util.List; +import java.util.Map; + +import org.junit.Before; +import org.junit.Test; +import org.opensearch.action.ActionRequest; +import org.opensearch.action.ActionRequestValidationException; +import org.opensearch.common.io.stream.BytesStreamOutput; +import org.opensearch.common.io.stream.StreamOutput; +import org.opensearch.ml.common.AccessMode; +import org.opensearch.ml.common.connector.ConnectorAction; +import org.opensearch.ml.common.connector.MLPostProcessFunction; +import org.opensearch.ml.common.connector.MLPreProcessFunction; + +import static org.junit.Assert.assertEquals; +import static org.junit.Assert.assertNotSame; +import static org.junit.Assert.assertNull; +import static org.junit.Assert.assertSame; + public class MLCreateConnectorInputTests { + private MLCreateConnectorInput mlCreateConnectorInput; + + @Before + public void setUp(){ + ConnectorAction.ActionType actionType = ConnectorAction.ActionType.PREDICT; + String method = "POST"; + String url = "https://test.com"; + Map headers = new HashMap<>(); + headers.put("api_key", "${credential.key}"); + String mlCreateConnectorRequestBody = "{\"input\": \"${parameters.input}\"}"; + String preProcessFunction = MLPreProcessFunction.TEXT_DOCS_TO_OPENAI_EMBEDDING_INPUT; + String postProcessFunction = MLPostProcessFunction.OPENAI_EMBEDDING; + ConnectorAction action = new ConnectorAction(actionType, method, url, headers, mlCreateConnectorRequestBody, preProcessFunction, postProcessFunction); + + // java.lang.IllegalStateException: unexpected byte [0x6f] will be thrown if we specify backendRoles field. + mlCreateConnectorInput = MLCreateConnectorInput.builder() + .name("test_connector_name") + .description("this is a test connector") + .version("1") + .protocol("http") + .parameters(Map.of("input", "test input value")) + .credential(Map.of("key", "test_key_value")) + .actions(List.of(action)) + .access(AccessMode.PUBLIC) + .addAllBackendRoles(true) + .build(); + } + + @Test + // MLCreateConnectorInput check its parameters when created, so exception is not thrown here + public void validate_Exception_NullMLModelName() { + mlCreateConnectorInput.setName(null); + MLCreateConnectorRequest mlCreateConnectorRequest = MLCreateConnectorRequest.builder() + .mlCreateConnectorInput(mlCreateConnectorInput) + .build(); + + assertNull(mlCreateConnectorRequest.validate()); + assertNull(mlCreateConnectorRequest.getMlCreateConnectorInput().getName()); + } } diff --git a/common/src/test/java/org/opensearch/ml/common/transport/connector/MLCreateConnectorRequestTest.java b/common/src/test/java/org/opensearch/ml/common/transport/connector/MLCreateConnectorRequestTest.java deleted file mode 100644 index 0debb02ee6..0000000000 --- a/common/src/test/java/org/opensearch/ml/common/transport/connector/MLCreateConnectorRequestTest.java +++ /dev/null @@ -1,73 +0,0 @@ -/* - * Copyright OpenSearch Contributors - * SPDX-License-Identifier: Apache-2.0 - */ - -package org.opensearch.ml.common.transport.connector; - -import org.junit.Assert; -import org.junit.Rule; -import org.junit.Test; -import org.junit.rules.ExpectedException; -import org.opensearch.action.ActionRequest; -import org.opensearch.action.ActionRequestValidationException; -import org.opensearch.common.io.stream.BytesStreamOutput; - -import java.io.IOException; -import java.io.UncheckedIOException; - -public class MLCreateConnectorRequestTest { - - @Rule - public ExpectedException exceptionRule = ExpectedException.none(); - - @Test - public void validate_nullInput() { - MLCreateConnectorRequest request = new MLCreateConnectorRequest((MLCreateConnectorInput)null); - ActionRequestValidationException exception = request.validate(); - Assert.assertTrue(exception.getMessage().contains("ML Connector input can't be null")); - } - - @Test - public void readFromStream() throws IOException { - MLCreateConnectorInput input = MLCreateConnectorInput.builder() - .name("test_connector") - .protocol("http") - .version("1") - .description("test") - .addAllBackendRoles(true) - .build(); - MLCreateConnectorRequest request = new MLCreateConnectorRequest(input); - BytesStreamOutput output = new BytesStreamOutput(); - request.writeTo(output); - MLCreateConnectorRequest request2 = new MLCreateConnectorRequest(output.bytes().streamInput()); - Assert.assertEquals("test_connector", request2.getMlCreateConnectorInput().getName()); - Assert.assertEquals("http", request2.getMlCreateConnectorInput().getProtocol()); - Assert.assertEquals("1", request2.getMlCreateConnectorInput().getVersion()); - Assert.assertEquals("test", request2.getMlCreateConnectorInput().getDescription()); - } - - @Test - public void fromActionRequest() { - MLCreateConnectorInput input = MLCreateConnectorInput.builder() - .name("test_connector") - .protocol("http") - .version("1") - .description("test") - .build(); - ActionRequest request = new MLCreateConnectorRequest(input); - MLCreateConnectorRequest request2 = MLCreateConnectorRequest.fromActionRequest(request); - Assert.assertEquals("test_connector", request2.getMlCreateConnectorInput().getName()); - Assert.assertEquals("http", request2.getMlCreateConnectorInput().getProtocol()); - Assert.assertEquals("1", request2.getMlCreateConnectorInput().getVersion()); - Assert.assertEquals("test", request2.getMlCreateConnectorInput().getDescription()); - } - - @Test - public void fromActionRequest_Exception() { - exceptionRule.expect(UncheckedIOException.class); - exceptionRule.expectMessage("Failed to parse ActionRequest into MLCreateConnectorRequest"); - ActionRequest request = new MLConnectorGetRequest("test_id", true); - MLCreateConnectorRequest.fromActionRequest(request); - } -} diff --git a/common/src/test/java/org/opensearch/ml/common/transport/connector/MLCreateConnectorRequestTests.java b/common/src/test/java/org/opensearch/ml/common/transport/connector/MLCreateConnectorRequestTests.java index 368a30acc6..581a4bdf33 100644 --- a/common/src/test/java/org/opensearch/ml/common/transport/connector/MLCreateConnectorRequestTests.java +++ b/common/src/test/java/org/opensearch/ml/common/transport/connector/MLCreateConnectorRequestTests.java @@ -7,7 +7,6 @@ import java.io.IOException; import java.io.UncheckedIOException; -import java.util.Arrays; import java.util.HashMap; import java.util.List; import java.util.Map; @@ -25,7 +24,6 @@ import org.opensearch.ml.common.connector.MLPreProcessFunction; import static org.junit.Assert.assertEquals; -import static org.junit.Assert.assertNotEquals; import static org.junit.Assert.assertNotSame; import static org.junit.Assert.assertNull; import static org.junit.Assert.assertSame; @@ -45,7 +43,6 @@ public void setUp(){ String postProcessFunction = MLPostProcessFunction.OPENAI_EMBEDDING; ConnectorAction action = new ConnectorAction(actionType, method, url, headers, mlCreateConnectorRequestBody, preProcessFunction, postProcessFunction); - // java.lang.IllegalStateException: unexpected byte [0x6f] will be thrown if we specify backendRoles field. mlCreateConnectorInput = MLCreateConnectorInput.builder() .name("test_connector_name") .description("this is a test connector") @@ -90,18 +87,6 @@ public void validate_Exception_NullMLRegisterModelGroupInput() { assertEquals("Validation Failed: 1: ML Connector input can't be null;", exception.getMessage()); } - @Test - // MLCreateConnectorInput check its parameters when created, so exception is not thrown here - public void validate_Exception_NullMLModelName() { - mlCreateConnectorInput.setName(null); - MLCreateConnectorRequest mlCreateConnectorRequest = MLCreateConnectorRequest.builder() - .mlCreateConnectorInput(mlCreateConnectorInput) - .build(); - - assertNull(mlCreateConnectorRequest.validate()); - assertNull(mlCreateConnectorRequest.getMlCreateConnectorInput().getName()); - } - @Test public void fromActionRequest_Success_WithMLRegisterModelRequest() { MLCreateConnectorRequest mlCreateConnectorRequest = MLCreateConnectorRequest.builder() From 2cf62e03f75a4c024d8565cbe94205ad092db0d8 Mon Sep 17 00:00:00 2001 From: Sicheng Song Date: Wed, 12 Jul 2023 01:24:15 +0000 Subject: [PATCH 3/4] fix a bug in MLCreateConnectorInput class Signed-off-by: Sicheng Song --- common/build.gradle | 4 ++-- .../common/transport/connector/MLCreateConnectorInput.java | 2 +- .../transport/connector/MLCreateConnectorInputTests.java | 5 +++-- .../transport/connector/MLCreateConnectorRequestTests.java | 4 +++- 4 files changed, 9 insertions(+), 6 deletions(-) diff --git a/common/build.gradle b/common/build.gradle index 9a01a80b83..3aeb644d42 100644 --- a/common/build.gradle +++ b/common/build.gradle @@ -36,11 +36,11 @@ jacocoTestCoverageVerification { rule { limit { counter = 'LINE' - minimum = 0.6 //TODO: add more test to meet the coverage bar 0.9 + minimum = 0.8 //TODO: add more test to meet the coverage bar 0.9 } limit { counter = 'BRANCH' - minimum = 0.5 //TODO: add more test to meet the coverage bar 0.9 + minimum = 0.7 //TODO: add more test to meet the coverage bar 0.9 } } } diff --git a/common/src/main/java/org/opensearch/ml/common/transport/connector/MLCreateConnectorInput.java b/common/src/main/java/org/opensearch/ml/common/transport/connector/MLCreateConnectorInput.java index aec2d1b0b6..43568afb05 100644 --- a/common/src/main/java/org/opensearch/ml/common/transport/connector/MLCreateConnectorInput.java +++ b/common/src/main/java/org/opensearch/ml/common/transport/connector/MLCreateConnectorInput.java @@ -227,7 +227,7 @@ public void writeTo(StreamOutput output) throws IOException { } if (!CollectionUtils.isEmpty(backendRoles)) { output.writeBoolean(true); - output.writeOptionalStringCollection(backendRoles); + output.writeStringCollection(backendRoles); } else { output.writeBoolean(false); } diff --git a/common/src/test/java/org/opensearch/ml/common/transport/connector/MLCreateConnectorInputTests.java b/common/src/test/java/org/opensearch/ml/common/transport/connector/MLCreateConnectorInputTests.java index 4de9198785..81a6e36b64 100644 --- a/common/src/test/java/org/opensearch/ml/common/transport/connector/MLCreateConnectorInputTests.java +++ b/common/src/test/java/org/opensearch/ml/common/transport/connector/MLCreateConnectorInputTests.java @@ -7,6 +7,7 @@ import java.io.IOException; import java.io.UncheckedIOException; +import java.util.Arrays; import java.util.HashMap; import java.util.List; import java.util.Map; @@ -42,7 +43,6 @@ public void setUp(){ String postProcessFunction = MLPostProcessFunction.OPENAI_EMBEDDING; ConnectorAction action = new ConnectorAction(actionType, method, url, headers, mlCreateConnectorRequestBody, preProcessFunction, postProcessFunction); - // java.lang.IllegalStateException: unexpected byte [0x6f] will be thrown if we specify backendRoles field. mlCreateConnectorInput = MLCreateConnectorInput.builder() .name("test_connector_name") .description("this is a test connector") @@ -52,7 +52,8 @@ public void setUp(){ .credential(Map.of("key", "test_key_value")) .actions(List.of(action)) .access(AccessMode.PUBLIC) - .addAllBackendRoles(true) + .backendRoles(Arrays.asList("role1", "role2")) + .addAllBackendRoles(false) .build(); } diff --git a/common/src/test/java/org/opensearch/ml/common/transport/connector/MLCreateConnectorRequestTests.java b/common/src/test/java/org/opensearch/ml/common/transport/connector/MLCreateConnectorRequestTests.java index 581a4bdf33..6fe82d7f2b 100644 --- a/common/src/test/java/org/opensearch/ml/common/transport/connector/MLCreateConnectorRequestTests.java +++ b/common/src/test/java/org/opensearch/ml/common/transport/connector/MLCreateConnectorRequestTests.java @@ -7,6 +7,7 @@ import java.io.IOException; import java.io.UncheckedIOException; +import java.util.Arrays; import java.util.HashMap; import java.util.List; import java.util.Map; @@ -52,7 +53,8 @@ public void setUp(){ .credential(Map.of("key", "test_key_value")) .actions(List.of(action)) .access(AccessMode.PUBLIC) - .addAllBackendRoles(true) + .backendRoles(Arrays.asList("role1", "role2")) + .addAllBackendRoles(false) .build(); } From 6bdd594b7c34e28a5c2a9fd75660d8105c3e960e Mon Sep 17 00:00:00 2001 From: Sicheng Song Date: Wed, 12 Jul 2023 03:01:53 +0000 Subject: [PATCH 4/4] Add more UTs Signed-off-by: Sicheng Song --- .../MLCreateConnectorInputTests.java | 160 +++++++++++++++++- 1 file changed, 153 insertions(+), 7 deletions(-) diff --git a/common/src/test/java/org/opensearch/ml/common/transport/connector/MLCreateConnectorInputTests.java b/common/src/test/java/org/opensearch/ml/common/transport/connector/MLCreateConnectorInputTests.java index 81a6e36b64..c037da1529 100644 --- a/common/src/test/java/org/opensearch/ml/common/transport/connector/MLCreateConnectorInputTests.java +++ b/common/src/test/java/org/opensearch/ml/common/transport/connector/MLCreateConnectorInputTests.java @@ -8,28 +8,59 @@ import java.io.IOException; import java.io.UncheckedIOException; import java.util.Arrays; +import java.util.Collections; import java.util.HashMap; import java.util.List; import java.util.Map; +import java.util.function.Consumer; import org.junit.Before; +import org.junit.Rule; import org.junit.Test; +import org.junit.rules.ExpectedException; import org.opensearch.action.ActionRequest; import org.opensearch.action.ActionRequestValidationException; +import org.opensearch.common.Strings; import org.opensearch.common.io.stream.BytesStreamOutput; +import org.opensearch.common.io.stream.StreamInput; import org.opensearch.common.io.stream.StreamOutput; +import org.opensearch.common.settings.Settings; +import org.opensearch.common.xcontent.LoggingDeprecationHandler; +import org.opensearch.common.xcontent.XContentFactory; +import org.opensearch.common.xcontent.XContentType; +import org.opensearch.core.xcontent.NamedXContentRegistry; +import org.opensearch.core.xcontent.ToXContent; +import org.opensearch.core.xcontent.XContentBuilder; +import org.opensearch.core.xcontent.XContentParser; import org.opensearch.ml.common.AccessMode; import org.opensearch.ml.common.connector.ConnectorAction; import org.opensearch.ml.common.connector.MLPostProcessFunction; import org.opensearch.ml.common.connector.MLPreProcessFunction; +import org.opensearch.search.SearchModule; import static org.junit.Assert.assertEquals; +import static org.junit.Assert.assertNotNull; import static org.junit.Assert.assertNotSame; import static org.junit.Assert.assertNull; import static org.junit.Assert.assertSame; +import static org.junit.Assert.assertTrue; public class MLCreateConnectorInputTests { private MLCreateConnectorInput mlCreateConnectorInput; + private MLCreateConnectorInput mlCreateDryRunConnectorInput; + + @Rule + public final ExpectedException exceptionRule = ExpectedException.none(); + private final String expectedInputStr = "{\"name\":\"test_connector_name\"," + + "\"description\":\"this is a test connector\",\"version\":\"1\",\"protocol\":\"http\"," + + "\"parameters\":{\"input\":\"test input value\"},\"credential\":{\"key\":\"test_key_value\"}," + + "\"actions\":[{\"action_type\":\"PREDICT\",\"method\":\"POST\",\"url\":\"https://test.com\"," + + "\"headers\":{\"api_key\":\"${credential.key}\"}," + + "\"request_body\":\"{\\\"input\\\": \\\"${parameters.input}\\\"}\"," + + "\"pre_process_function\":\"connector.pre_process.openai.embedding\"," + + "\"post_process_function\":\"connector.post_process.openai.embedding\"}]," + + "\"backend_roles\":[\"role1\",\"role2\"],\"add_all_backend_roles\":false," + + "\"access_mode\":\"PUBLIC\"}"; @Before public void setUp(){ @@ -55,17 +86,132 @@ public void setUp(){ .backendRoles(Arrays.asList("role1", "role2")) .addAllBackendRoles(false) .build(); + + mlCreateDryRunConnectorInput = MLCreateConnectorInput.builder() + .dryRun(true) + .build(); } @Test - // MLCreateConnectorInput check its parameters when created, so exception is not thrown here - public void validate_Exception_NullMLModelName() { - mlCreateConnectorInput.setName(null); - MLCreateConnectorRequest mlCreateConnectorRequest = MLCreateConnectorRequest.builder() - .mlCreateConnectorInput(mlCreateConnectorInput) + public void constructorMLCreateConnectorInput_NullName() { + exceptionRule.expect(IllegalArgumentException.class); + exceptionRule.expectMessage("Connector name is null"); + MLCreateConnectorInput.builder() + .name(null) + .description("this is a test connector") + .version("1") + .protocol("http") + .parameters(Map.of("input", "test input value")) + .credential(Map.of("key", "test_key_value")) + .actions(List.of()) + .access(AccessMode.PUBLIC) + .backendRoles(Arrays.asList("role1", "role2")) + .addAllBackendRoles(false) .build(); + } - assertNull(mlCreateConnectorRequest.validate()); - assertNull(mlCreateConnectorRequest.getMlCreateConnectorInput().getName()); + @Test + public void constructorMLCreateConnectorInput_NullVersion() { + exceptionRule.expect(IllegalArgumentException.class); + exceptionRule.expectMessage("Connector version is null"); + MLCreateConnectorInput.builder() + .name("test_connector_name") + .description("this is a test connector") + .version(null) + .protocol("http") + .parameters(Map.of("input", "test input value")) + .credential(Map.of("key", "test_key_value")) + .actions(List.of()) + .access(AccessMode.PUBLIC) + .backendRoles(Arrays.asList("role1", "role2")) + .addAllBackendRoles(false) + .build(); } + + @Test + public void constructorMLCreateConnectorInput_NullProtocol() { + exceptionRule.expect(IllegalArgumentException.class); + exceptionRule.expectMessage("Connector protocol is null"); + MLCreateConnectorInput.builder() + .name("test_connector_name") + .description("this is a test connector") + .version("1") + .protocol(null) + .parameters(Map.of("input", "test input value")) + .credential(Map.of("key", "test_key_value")) + .actions(List.of()) + .access(AccessMode.PUBLIC) + .backendRoles(Arrays.asList("role1", "role2")) + .addAllBackendRoles(false) + .build(); + } + + @Test + public void testToXContent_FullFields() throws Exception { + XContentBuilder builder = XContentFactory.contentBuilder(XContentType.JSON); + mlCreateConnectorInput.toXContent(builder, ToXContent.EMPTY_PARAMS); + assertNotNull(builder); + String jsonStr = Strings.toString(builder); + assertEquals(expectedInputStr, jsonStr); + } + + @Test + public void testToXContent_NullFields() throws Exception { + XContentBuilder builder = XContentFactory.contentBuilder(XContentType.JSON); + mlCreateDryRunConnectorInput.toXContent(builder, ToXContent.EMPTY_PARAMS); + assertNotNull(builder); + String jsonStr = Strings.toString(builder); + assertEquals("{}", jsonStr); + } + + @Test + public void testParse() throws Exception { + testParseFromJsonString(expectedInputStr, parsedInput -> { + assertEquals("test_connector_name", parsedInput.getName()); + }); + } + + @Test + public void testParseWithDryRun() throws Exception { + String expectedInputStrWithDryRun = "{\"dry_run\":true}"; + testParseFromJsonString(expectedInputStrWithDryRun, parsedInput -> { + assertNull(parsedInput.getName()); + assertTrue(parsedInput.isDryRun()); + }); + } + + @Test + public void readInputStream_Success() throws IOException { + readInputStream(mlCreateConnectorInput, parsedInput -> assertEquals(mlCreateConnectorInput.getName(), parsedInput.getName())); + } + + @Test + public void readInputStream_SuccessWithNullFields() throws IOException { + MLCreateConnectorInput mlCreateMinimalConnectorInput = MLCreateConnectorInput.builder() + .name("test_connector_name") + .version("1") + .protocol("http") + .build(); + readInputStream(mlCreateMinimalConnectorInput, parsedInput -> { + assertEquals(mlCreateMinimalConnectorInput.getName(), parsedInput.getName()); + assertNull(parsedInput.getActions()); + }); + } + + private void testParseFromJsonString(String expectedInputString, Consumer verify) throws Exception { + XContentParser parser = XContentType.JSON.xContent().createParser(new NamedXContentRegistry(new SearchModule(Settings.EMPTY, + Collections.emptyList()).getNamedXContents()), LoggingDeprecationHandler.INSTANCE, expectedInputString); + parser.nextToken(); + MLCreateConnectorInput parsedInput = MLCreateConnectorInput.parse(parser); + verify.accept(parsedInput); + } + + private void readInputStream(MLCreateConnectorInput input, Consumer verify) throws IOException { + BytesStreamOutput bytesStreamOutput = new BytesStreamOutput(); + input.writeTo(bytesStreamOutput); + StreamInput streamInput = bytesStreamOutput.bytes().streamInput(); + MLCreateConnectorInput parsedInput = new MLCreateConnectorInput(streamInput); + verify.accept(parsedInput); + } + }