From d764285ccf657882ed0575a5ae51bad24bc728b0 Mon Sep 17 00:00:00 2001 From: Matthew Wells Date: Tue, 27 Jun 2023 14:14:46 -0700 Subject: [PATCH] Fix CSV/RAW output header being application/json rather than plain/text (#1779) * Fix CI (#1760) * Fix ML-commons missing dependency. Signed-off-by: Yury-Fridlyand * Fix `mockito` dependency. Signed-off-by: Yury-Fridlyand * Revert changes in `:opensearch` since it is not needed anymore. Signed-off-by: Yury-Fridlyand --------- Signed-off-by: Yury-Fridlyand Signed-off-by: Matthew Wells * Fix CSV/RAW outputting wrong format (#279) * Fixed bug where CSV/RAW outputs as JSON rather than plain text Signed-off-by: Matthew Wells --------- Signed-off-by: Yury-Fridlyand Signed-off-by: Matthew Wells Co-authored-by: Yury-Fridlyand (cherry picked from commit 1ec696d201752a3913c325d710b7ab25e25226b7) --- .../org/opensearch/sql/sql/CsvFormatIT.java | 17 ++++++++++++++ .../org/opensearch/sql/sql/RawFormatIT.java | 16 ++++++++++++++ .../sql/sql/SimpleQueryStringIT.java | 20 +++++++++++++++++ .../sql/legacy/plugin/RestSQLQueryAction.java | 13 ++++++----- .../format/FlatResponseFormatter.java | 6 +++++ .../format/JsonResponseFormatter.java | 6 +++++ .../response/format/ResponseFormatter.java | 7 ++++++ .../format/CommandResponseFormatterTest.java | 7 ++++++ .../format/CsvResponseFormatterTest.java | 6 +++++ .../format/RawResponseFormatterTest.java | 22 ++++++++++++------- .../VisualizationResponseFormatterTest.java | 7 ++++++ 11 files changed, 113 insertions(+), 14 deletions(-) diff --git a/integ-test/src/test/java/org/opensearch/sql/sql/CsvFormatIT.java b/integ-test/src/test/java/org/opensearch/sql/sql/CsvFormatIT.java index 782e2e22b5..aa2737cbac 100644 --- a/integ-test/src/test/java/org/opensearch/sql/sql/CsvFormatIT.java +++ b/integ-test/src/test/java/org/opensearch/sql/sql/CsvFormatIT.java @@ -7,10 +7,14 @@ package org.opensearch.sql.sql; import static org.opensearch.sql.legacy.TestsConstants.TEST_INDEX_BANK_CSV_SANITIZE; +import static org.opensearch.sql.protocol.response.format.FlatResponseFormatter.CONTENT_TYPE; import java.io.IOException; import java.util.Locale; + import org.junit.Test; +import org.opensearch.client.Request; +import org.opensearch.client.Response; import org.opensearch.sql.common.utils.StringUtils; import org.opensearch.sql.legacy.SQLIntegTestCase; @@ -49,4 +53,17 @@ public void escapeSanitizeTest() { + "\",Elinor\",\"Ratliff,,,\"%n"), result); } + + @Test + public void contentHeaderTest() throws IOException { + String query = String.format(Locale.ROOT, "SELECT firstname, lastname FROM %s", TEST_INDEX_BANK_CSV_SANITIZE); + String requestBody = makeRequest(query); + + Request sqlRequest = new Request("POST", "/_plugins/_sql?format=csv"); + sqlRequest.setJsonEntity(requestBody); + + Response response = client().performRequest(sqlRequest); + + assertEquals(response.getEntity().getContentType(), CONTENT_TYPE); + } } diff --git a/integ-test/src/test/java/org/opensearch/sql/sql/RawFormatIT.java b/integ-test/src/test/java/org/opensearch/sql/sql/RawFormatIT.java index 8cba86647c..b040b97136 100644 --- a/integ-test/src/test/java/org/opensearch/sql/sql/RawFormatIT.java +++ b/integ-test/src/test/java/org/opensearch/sql/sql/RawFormatIT.java @@ -6,11 +6,15 @@ package org.opensearch.sql.sql; +import static org.opensearch.sql.legacy.TestsConstants.TEST_INDEX_BANK_CSV_SANITIZE; import static org.opensearch.sql.legacy.TestsConstants.TEST_INDEX_BANK_RAW_SANITIZE; +import static org.opensearch.sql.protocol.response.format.FlatResponseFormatter.CONTENT_TYPE; import java.io.IOException; import java.util.Locale; import org.junit.Test; +import org.opensearch.client.Request; +import org.opensearch.client.Response; import org.opensearch.sql.common.utils.StringUtils; import org.opensearch.sql.legacy.SQLIntegTestCase; @@ -35,4 +39,16 @@ public void rawFormatWithPipeFieldTest() { result); } + @Test + public void contentHeaderTest() throws IOException { + String query = String.format(Locale.ROOT, "SELECT firstname, lastname FROM %s", TEST_INDEX_BANK_RAW_SANITIZE); + String requestBody = makeRequest(query); + + Request sqlRequest = new Request("POST", "/_plugins/_sql?format=raw"); + sqlRequest.setJsonEntity(requestBody); + + Response response = client().performRequest(sqlRequest); + + assertEquals(response.getEntity().getContentType(), CONTENT_TYPE); + } } diff --git a/integ-test/src/test/java/org/opensearch/sql/sql/SimpleQueryStringIT.java b/integ-test/src/test/java/org/opensearch/sql/sql/SimpleQueryStringIT.java index a9e600121f..42a3c99a13 100644 --- a/integ-test/src/test/java/org/opensearch/sql/sql/SimpleQueryStringIT.java +++ b/integ-test/src/test/java/org/opensearch/sql/sql/SimpleQueryStringIT.java @@ -5,11 +5,17 @@ package org.opensearch.sql.sql; +import static org.opensearch.sql.legacy.TestsConstants.TEST_INDEX_BANK_CSV_SANITIZE; import static org.opensearch.sql.legacy.TestsConstants.TEST_INDEX_BEER; +import static org.opensearch.sql.protocol.response.format.JsonResponseFormatter.CONTENT_TYPE; import java.io.IOException; +import java.util.Locale; + import org.json.JSONObject; import org.junit.Test; +import org.opensearch.client.Request; +import org.opensearch.client.Response; import org.opensearch.sql.legacy.SQLIntegTestCase; public class SimpleQueryStringIT extends SQLIntegTestCase { @@ -61,4 +67,18 @@ public void verify_wildcard_test() throws IOException { var result = new JSONObject(executeQuery(query, "jdbc")); assertEquals(10, result.getInt("total")); } + + @Test + public void contentHeaderTest() throws IOException { + String query = "SELECT Id FROM " + TEST_INDEX_BEER + + " WHERE simple_query_string([\\\"Tags\\\" ^ 1.5, Title, 'Body' 4.2], 'taste')"; + String requestBody = makeRequest(query); + + Request sqlRequest = new Request("POST", "/_plugins/_sql"); + sqlRequest.setJsonEntity(requestBody); + + Response response = client().performRequest(sqlRequest); + + assertEquals(response.getEntity().getContentType(), CONTENT_TYPE); + } } diff --git a/legacy/src/main/java/org/opensearch/sql/legacy/plugin/RestSQLQueryAction.java b/legacy/src/main/java/org/opensearch/sql/legacy/plugin/RestSQLQueryAction.java index 37cbba4adf..bbc1c293be 100644 --- a/legacy/src/main/java/org/opensearch/sql/legacy/plugin/RestSQLQueryAction.java +++ b/legacy/src/main/java/org/opensearch/sql/legacy/plugin/RestSQLQueryAction.java @@ -141,15 +141,16 @@ public void onFailure(Exception e) { private ResponseListener createExplainResponseListener( RestChannel channel, BiConsumer errorHandler) { - return new ResponseListener() { + return new ResponseListener<>() { @Override public void onResponse(ExplainResponse response) { - sendResponse(channel, OK, new JsonResponseFormatter(PRETTY) { + JsonResponseFormatter formatter = new JsonResponseFormatter<>(PRETTY) { @Override protected Object buildJsonObject(ExplainResponse response) { return response; } - }.format(response)); + }; + sendResponse(channel, OK, formatter.format(response), formatter.contentType()); } @Override @@ -180,7 +181,7 @@ private ResponseListener createQueryResponseListener( public void onResponse(QueryResponse response) { sendResponse(channel, OK, formatter.format(new QueryResult(response.getSchema(), response.getResults(), - response.getCursor()))); + response.getCursor())), formatter.contentType()); } @Override @@ -190,9 +191,9 @@ public void onFailure(Exception e) { }; } - private void sendResponse(RestChannel channel, RestStatus status, String content) { + private void sendResponse(RestChannel channel, RestStatus status, String content, String contentType) { channel.sendResponse(new BytesRestResponse( - status, "application/json; charset=UTF-8", content)); + status, contentType, content)); } private static void logAndPublishMetrics(Exception e) { diff --git a/protocol/src/main/java/org/opensearch/sql/protocol/response/format/FlatResponseFormatter.java b/protocol/src/main/java/org/opensearch/sql/protocol/response/format/FlatResponseFormatter.java index d4d00c4535..0575647dad 100644 --- a/protocol/src/main/java/org/opensearch/sql/protocol/response/format/FlatResponseFormatter.java +++ b/protocol/src/main/java/org/opensearch/sql/protocol/response/format/FlatResponseFormatter.java @@ -23,6 +23,8 @@ public abstract class FlatResponseFormatter implements ResponseFormatter SENSITIVE_CHAR = ImmutableSet.of("=", "+", "-", "@"); + public static final String CONTENT_TYPE = "plain/text; charset=UTF-8"; + private boolean sanitize = false; public FlatResponseFormatter(String seperator, boolean sanitize) { @@ -30,6 +32,10 @@ public FlatResponseFormatter(String seperator, boolean sanitize) { this.sanitize = sanitize; } + public String contentType() { + return CONTENT_TYPE; + } + @Override public String format(QueryResult response) { FlatResult result = new FlatResult(response, sanitize); diff --git a/protocol/src/main/java/org/opensearch/sql/protocol/response/format/JsonResponseFormatter.java b/protocol/src/main/java/org/opensearch/sql/protocol/response/format/JsonResponseFormatter.java index 03e060925d..810a7d0c2d 100644 --- a/protocol/src/main/java/org/opensearch/sql/protocol/response/format/JsonResponseFormatter.java +++ b/protocol/src/main/java/org/opensearch/sql/protocol/response/format/JsonResponseFormatter.java @@ -36,6 +36,8 @@ public enum Style { */ private final Style style; + public static final String CONTENT_TYPE = "application/json; charset=UTF-8"; + @Override public String format(R response) { return jsonify(buildJsonObject(response)); @@ -47,6 +49,10 @@ public String format(Throwable t) { (style == PRETTY) ? prettyFormat(t) : compactFormat(t)); } + public String contentType() { + return CONTENT_TYPE; + } + /** * Build JSON object to generate response json string. * diff --git a/protocol/src/main/java/org/opensearch/sql/protocol/response/format/ResponseFormatter.java b/protocol/src/main/java/org/opensearch/sql/protocol/response/format/ResponseFormatter.java index 4618a4f80d..6d9cc093c5 100644 --- a/protocol/src/main/java/org/opensearch/sql/protocol/response/format/ResponseFormatter.java +++ b/protocol/src/main/java/org/opensearch/sql/protocol/response/format/ResponseFormatter.java @@ -27,4 +27,11 @@ public interface ResponseFormatter { */ String format(Throwable t); + /** + * Getter for the content type header of the response. + * + * @return string + */ + String contentType(); + } diff --git a/protocol/src/test/java/org/opensearch/sql/protocol/response/format/CommandResponseFormatterTest.java b/protocol/src/test/java/org/opensearch/sql/protocol/response/format/CommandResponseFormatterTest.java index a3052324fe..85efbab369 100644 --- a/protocol/src/test/java/org/opensearch/sql/protocol/response/format/CommandResponseFormatterTest.java +++ b/protocol/src/test/java/org/opensearch/sql/protocol/response/format/CommandResponseFormatterTest.java @@ -10,6 +10,7 @@ import static org.opensearch.sql.data.model.ExprValueUtils.tupleValue; import static org.opensearch.sql.data.type.ExprCoreType.INTEGER; import static org.opensearch.sql.data.type.ExprCoreType.STRING; +import static org.opensearch.sql.protocol.response.format.JsonResponseFormatter.CONTENT_TYPE; import static org.opensearch.sql.protocol.response.format.JsonResponseFormatter.Style.PRETTY; import com.google.common.collect.ImmutableList; @@ -56,4 +57,10 @@ public void formats_error_as_default_formatter() { assertEquals(new JdbcResponseFormatter(PRETTY).format(exception), new CommandResponseFormatter().format(exception)); } + + @Test + void testContentType() { + var formatter = new CommandResponseFormatter(); + assertEquals(formatter.contentType(), CONTENT_TYPE); + } } diff --git a/protocol/src/test/java/org/opensearch/sql/protocol/response/format/CsvResponseFormatterTest.java b/protocol/src/test/java/org/opensearch/sql/protocol/response/format/CsvResponseFormatterTest.java index 7008b51fa6..82b4f372b3 100644 --- a/protocol/src/test/java/org/opensearch/sql/protocol/response/format/CsvResponseFormatterTest.java +++ b/protocol/src/test/java/org/opensearch/sql/protocol/response/format/CsvResponseFormatterTest.java @@ -14,6 +14,7 @@ import static org.opensearch.sql.data.model.ExprValueUtils.tupleValue; import static org.opensearch.sql.data.type.ExprCoreType.INTEGER; import static org.opensearch.sql.data.type.ExprCoreType.STRING; +import static org.opensearch.sql.protocol.response.format.FlatResponseFormatter.CONTENT_TYPE; import com.google.common.collect.ImmutableList; import com.google.common.collect.ImmutableMap; @@ -130,4 +131,9 @@ void replaceNullValues() { assertEquals(format(expected), formatter.format(response)); } + @Test + void testContentType() { + assertEquals(formatter.contentType(), CONTENT_TYPE); + } + } diff --git a/protocol/src/test/java/org/opensearch/sql/protocol/response/format/RawResponseFormatterTest.java b/protocol/src/test/java/org/opensearch/sql/protocol/response/format/RawResponseFormatterTest.java index 24b5a4431d..b33a4f216a 100644 --- a/protocol/src/test/java/org/opensearch/sql/protocol/response/format/RawResponseFormatterTest.java +++ b/protocol/src/test/java/org/opensearch/sql/protocol/response/format/RawResponseFormatterTest.java @@ -14,6 +14,7 @@ import static org.opensearch.sql.data.model.ExprValueUtils.tupleValue; import static org.opensearch.sql.data.type.ExprCoreType.INTEGER; import static org.opensearch.sql.data.type.ExprCoreType.STRING; +import static org.opensearch.sql.protocol.response.format.FlatResponseFormatter.CONTENT_TYPE; import com.google.common.collect.ImmutableList; import com.google.common.collect.ImmutableMap; @@ -27,7 +28,7 @@ * Unit test for {@link FlatResponseFormatter}. */ public class RawResponseFormatterTest { - private FlatResponseFormatter rawFormater = new RawResponseFormatter(); + private FlatResponseFormatter rawFormatter = new RawResponseFormatter(); @Test void formatResponse() { @@ -38,7 +39,7 @@ void formatResponse() { tupleValue(ImmutableMap.of("name", "John", "age", 20)), tupleValue(ImmutableMap.of("name", "Smith", "age", 30)))); String expected = "name|age%nJohn|20%nSmith|30"; - assertEquals(format(expected), rawFormater.format(response)); + assertEquals(format(expected), rawFormatter.format(response)); } @Test @@ -53,7 +54,7 @@ void sanitizeHeaders() { "=firstname", "John", "+lastname", "Smith", "-city", "Seattle", "@age", 20)))); String expected = "=firstname|+lastname|-city|@age%n" + "John|Smith|Seattle|20"; - assertEquals(format(expected), rawFormater.format(response)); + assertEquals(format(expected), rawFormatter.format(response)); } @Test @@ -74,7 +75,7 @@ void sanitizeData() { + "-Seattle%n" + "@Seattle%n" + "Seattle="; - assertEquals(format(expected), rawFormater.format(response)); + assertEquals(format(expected), rawFormatter.format(response)); } @Test @@ -86,7 +87,7 @@ void quoteIfRequired() { tupleValue(ImmutableMap.of("na|me", "John|Smith", "||age", "30|||")))); String expected = "\"na|me\"|\"||age\"%n" + "\"John|Smith\"|\"30|||\""; - assertEquals(format(expected), rawFormater.format(response)); + assertEquals(format(expected), rawFormatter.format(response)); } @Test @@ -94,7 +95,7 @@ void formatError() { Throwable t = new RuntimeException("This is an exception"); String expected = "{\n \"type\": \"RuntimeException\",\n \"reason\": \"This is an exception\"\n}"; - assertEquals(expected, rawFormater.format(t)); + assertEquals(expected, rawFormatter.format(t)); } @Test @@ -121,7 +122,7 @@ void senstiveCharater() { String expected = "city%n" + "@Seattle%n" + "++Seattle"; - assertEquals(format(expected), rawFormater.format(response)); + assertEquals(format(expected), rawFormatter.format(response)); } @Test @@ -153,7 +154,12 @@ void replaceNullValues() { + "John|Seattle%n" + "|Seattle%n" + "John|"; - assertEquals(format(expected), rawFormater.format(response)); + assertEquals(format(expected), rawFormatter.format(response)); + } + + @Test + void testContentType() { + assertEquals(rawFormatter.contentType(), CONTENT_TYPE); } } diff --git a/protocol/src/test/java/org/opensearch/sql/protocol/response/format/VisualizationResponseFormatterTest.java b/protocol/src/test/java/org/opensearch/sql/protocol/response/format/VisualizationResponseFormatterTest.java index 88a26aeb6b..f501a53d64 100644 --- a/protocol/src/test/java/org/opensearch/sql/protocol/response/format/VisualizationResponseFormatterTest.java +++ b/protocol/src/test/java/org/opensearch/sql/protocol/response/format/VisualizationResponseFormatterTest.java @@ -11,6 +11,7 @@ import static org.opensearch.sql.data.model.ExprValueUtils.tupleValue; import static org.opensearch.sql.data.type.ExprCoreType.INTEGER; import static org.opensearch.sql.data.type.ExprCoreType.STRING; +import static org.opensearch.sql.protocol.response.format.JsonResponseFormatter.CONTENT_TYPE; import com.google.common.collect.ImmutableList; import com.google.common.collect.ImmutableMap; @@ -187,4 +188,10 @@ private static void assertJsonEquals(String expected, String actual) { JsonParser.parseString(expected), JsonParser.parseString(actual)); } + + @Test + void testContentType() { + var formatter = new CommandResponseFormatter(); + assertEquals(formatter.contentType(), CONTENT_TYPE); + } }