SQL: Add multi_value_field_leniency inside FieldHitExtractor (#40113)
For cases where fields can have multi values, allow the behavior to be customized through a dedicated configuration field. By default this will be enabled on the drivers so that existing datasets work instead of throwing an exception. For regular SQL usage, the behavior is false so that the user is aware of the underlying data. Fix #39700 (cherry picked from commit 2b351571961f172fd59290ee079126bbd081ceaf)
This commit is contained in:
parent
b8ad337234
commit
076a68007c
|
@ -122,6 +122,10 @@ Query timeout (in seconds). That is the maximum amount of time waiting for a que
|
||||||
|
|
||||||
`proxy.socks`:: SOCKS proxy host name
|
`proxy.socks`:: SOCKS proxy host name
|
||||||
|
|
||||||
|
[float]
|
||||||
|
==== Mapping
|
||||||
|
`field.multi.value.leniency` (default `true`):: Whether to be lenient and return the first value for fields with multiple values (true) or throw an exception.
|
||||||
|
|
||||||
[float]
|
[float]
|
||||||
==== Additional
|
==== Additional
|
||||||
|
|
||||||
|
|
|
@ -356,6 +356,10 @@ More information available https://docs.oracle.com/javase/8/docs/api/java/time/Z
|
||||||
|false
|
|false
|
||||||
|Return the results in a columnar fashion, rather than row-based fashion. Valid for `json`, `yaml`, `cbor` and `smile`.
|
|Return the results in a columnar fashion, rather than row-based fashion. Valid for `json`, `yaml`, `cbor` and `smile`.
|
||||||
|
|
||||||
|
|field_multi_value_leniency
|
||||||
|
|false
|
||||||
|
|Throw an exception when encountering multiple values for a field (default) or be lenient and return the first value from the list (without any guarantees of what that will be - typically the first in natural ascending order).
|
||||||
|
|
||||||
|===
|
|===
|
||||||
|
|
||||||
Do note that most parameters (outside the timeout and `columnar` ones) make sense only during the initial query - any follow-up pagination request only requires the `cursor` parameter as explained in the <<sql-pagination, pagination>> chapter.
|
Do note that most parameters (outside the timeout and `columnar` ones) make sense only during the initial query - any follow-up pagination request only requires the `cursor` parameter as explained in the <<sql-pagination, pagination>> chapter.
|
||||||
|
|
|
@ -60,6 +60,8 @@ pagination taking place on the **root nested document and not on its inner hits*
|
||||||
|
|
||||||
Array fields are not supported due to the "invisible" way in which {es} handles an array of values: the mapping doesn't indicate whether
|
Array fields are not supported due to the "invisible" way in which {es} handles an array of values: the mapping doesn't indicate whether
|
||||||
a field is an array (has multiple values) or not, so without reading all the data, {es-sql} cannot know whether a field is a single or multi value.
|
a field is an array (has multiple values) or not, so without reading all the data, {es-sql} cannot know whether a field is a single or multi value.
|
||||||
|
When multiple values are returned for a field, by default, {es-sql} will throw an exception. However, it is possible to change this behavior through `field_multi_value_leniency` parameter in REST (disabled by default) or
|
||||||
|
`field.multi.value.leniency` in drivers (enabled by default).
|
||||||
|
|
||||||
[float]
|
[float]
|
||||||
=== Sorting by aggregation
|
=== Sorting by aggregation
|
||||||
|
|
|
@ -47,20 +47,25 @@ class JdbcConfiguration extends ConnectionConfiguration {
|
||||||
// can be out/err/url
|
// can be out/err/url
|
||||||
static final String DEBUG_OUTPUT_DEFAULT = "err";
|
static final String DEBUG_OUTPUT_DEFAULT = "err";
|
||||||
|
|
||||||
public static final String TIME_ZONE = "timezone";
|
static final String TIME_ZONE = "timezone";
|
||||||
// follow the JDBC spec and use the JVM default...
|
// follow the JDBC spec and use the JVM default...
|
||||||
// to avoid inconsistency, the default is picked up once at startup and reused across connections
|
// to avoid inconsistency, the default is picked up once at startup and reused across connections
|
||||||
// to cater to the principle of least surprise
|
// to cater to the principle of least surprise
|
||||||
// really, the way to move forward is to specify a calendar or the timezone manually
|
// really, the way to move forward is to specify a calendar or the timezone manually
|
||||||
static final String TIME_ZONE_DEFAULT = TimeZone.getDefault().getID();
|
static final String TIME_ZONE_DEFAULT = TimeZone.getDefault().getID();
|
||||||
|
|
||||||
|
static final String FIELD_MULTI_VALUE_LENIENCY = "field.multi.value.leniency";
|
||||||
|
static final String FIELD_MULTI_VALUE_LENIENCY_DEFAULT = "true";
|
||||||
|
|
||||||
|
|
||||||
// options that don't change at runtime
|
// options that don't change at runtime
|
||||||
private static final Set<String> OPTION_NAMES = new LinkedHashSet<>(Arrays.asList(TIME_ZONE, DEBUG, DEBUG_OUTPUT));
|
private static final Set<String> OPTION_NAMES = new LinkedHashSet<>(
|
||||||
|
Arrays.asList(TIME_ZONE, FIELD_MULTI_VALUE_LENIENCY, DEBUG, DEBUG_OUTPUT));
|
||||||
|
|
||||||
static {
|
static {
|
||||||
// trigger version initialization
|
// trigger version initialization
|
||||||
// typically this should have already happened but in case the
|
// typically this should have already happened but in case the
|
||||||
// JdbcDriver/JdbcDataSource are not used and the impl. classes used directly
|
// EsDriver/EsDataSource are not used and the impl. classes used directly
|
||||||
// this covers that case
|
// this covers that case
|
||||||
Version.CURRENT.toString();
|
Version.CURRENT.toString();
|
||||||
}
|
}
|
||||||
|
@ -71,6 +76,7 @@ class JdbcConfiguration extends ConnectionConfiguration {
|
||||||
|
|
||||||
// mutable ones
|
// mutable ones
|
||||||
private ZoneId zoneId;
|
private ZoneId zoneId;
|
||||||
|
private boolean fieldMultiValueLeniency;
|
||||||
|
|
||||||
public static JdbcConfiguration create(String u, Properties props, int loginTimeoutSeconds) throws JdbcSQLException {
|
public static JdbcConfiguration create(String u, Properties props, int loginTimeoutSeconds) throws JdbcSQLException {
|
||||||
URI uri = parseUrl(u);
|
URI uri = parseUrl(u);
|
||||||
|
@ -151,6 +157,8 @@ class JdbcConfiguration extends ConnectionConfiguration {
|
||||||
|
|
||||||
this.zoneId = parseValue(TIME_ZONE, props.getProperty(TIME_ZONE, TIME_ZONE_DEFAULT),
|
this.zoneId = parseValue(TIME_ZONE, props.getProperty(TIME_ZONE, TIME_ZONE_DEFAULT),
|
||||||
s -> TimeZone.getTimeZone(s).toZoneId().normalized());
|
s -> TimeZone.getTimeZone(s).toZoneId().normalized());
|
||||||
|
this.fieldMultiValueLeniency = parseValue(FIELD_MULTI_VALUE_LENIENCY,
|
||||||
|
props.getProperty(FIELD_MULTI_VALUE_LENIENCY, FIELD_MULTI_VALUE_LENIENCY_DEFAULT), Boolean::parseBoolean);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -174,6 +182,10 @@ class JdbcConfiguration extends ConnectionConfiguration {
|
||||||
return zoneId != null ? TimeZone.getTimeZone(zoneId) : null;
|
return zoneId != null ? TimeZone.getTimeZone(zoneId) : null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public boolean fieldMultiValueLeniency() {
|
||||||
|
return fieldMultiValueLeniency;
|
||||||
|
}
|
||||||
|
|
||||||
public static boolean canAccept(String url) {
|
public static boolean canAccept(String url) {
|
||||||
return (StringUtils.hasText(url) && url.trim().startsWith(JdbcConfiguration.URL_PREFIX));
|
return (StringUtils.hasText(url) && url.trim().startsWith(JdbcConfiguration.URL_PREFIX));
|
||||||
}
|
}
|
||||||
|
|
|
@ -49,10 +49,15 @@ class JdbcHttpClient {
|
||||||
|
|
||||||
Cursor query(String sql, List<SqlTypedParamValue> params, RequestMeta meta) throws SQLException {
|
Cursor query(String sql, List<SqlTypedParamValue> params, RequestMeta meta) throws SQLException {
|
||||||
int fetch = meta.fetchSize() > 0 ? meta.fetchSize() : conCfg.pageSize();
|
int fetch = meta.fetchSize() > 0 ? meta.fetchSize() : conCfg.pageSize();
|
||||||
SqlQueryRequest sqlRequest = new SqlQueryRequest(sql, params, null, conCfg.zoneId(),
|
SqlQueryRequest sqlRequest = new SqlQueryRequest(sql, params, conCfg.zoneId(),
|
||||||
fetch,
|
fetch,
|
||||||
TimeValue.timeValueMillis(meta.timeoutInMs()), TimeValue.timeValueMillis(meta.queryTimeoutInMs()),
|
TimeValue.timeValueMillis(meta.timeoutInMs()),
|
||||||
false, new RequestInfo(Mode.JDBC));
|
TimeValue.timeValueMillis(meta.queryTimeoutInMs()),
|
||||||
|
null,
|
||||||
|
Boolean.FALSE,
|
||||||
|
null,
|
||||||
|
new RequestInfo(Mode.JDBC),
|
||||||
|
conCfg.fieldMultiValueLeniency());
|
||||||
SqlQueryResponse response = httpClient.query(sqlRequest);
|
SqlQueryResponse response = httpClient.query(sqlRequest);
|
||||||
return new DefaultCursor(this, response.cursor(), toJdbcColumnInfo(response.columns()), response.rows(), meta);
|
return new DefaultCursor(this, response.cursor(), toJdbcColumnInfo(response.columns()), response.rows(), meta);
|
||||||
}
|
}
|
||||||
|
|
|
@ -50,9 +50,33 @@ public abstract class JdbcIntegrationTestCase extends ESRestTestCase {
|
||||||
}
|
}
|
||||||
|
|
||||||
public Connection esJdbc() throws SQLException {
|
public Connection esJdbc() throws SQLException {
|
||||||
return randomBoolean() ? useDriverManager() : useDataSource();
|
return esJdbc(connectionProperties());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public Connection esJdbc(Properties props) throws SQLException {
|
||||||
|
return createConnection(props);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected Connection createConnection(Properties connectionProperties) throws SQLException {
|
||||||
|
String elasticsearchAddress = getProtocol() + "://" + elasticsearchAddress();
|
||||||
|
String address = "jdbc:es://" + elasticsearchAddress;
|
||||||
|
Connection connection = null;
|
||||||
|
if (randomBoolean()) {
|
||||||
|
connection = DriverManager.getConnection(address, connectionProperties);
|
||||||
|
} else {
|
||||||
|
EsDataSource dataSource = new EsDataSource();
|
||||||
|
dataSource.setUrl(address);
|
||||||
|
dataSource.setProperties(connectionProperties);
|
||||||
|
connection = dataSource.getConnection();
|
||||||
|
}
|
||||||
|
|
||||||
|
assertNotNull("The timezone should be specified", connectionProperties.getProperty("timezone"));
|
||||||
|
return connection;
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// methods below are used inside the documentation only
|
||||||
|
//
|
||||||
protected Connection useDriverManager() throws SQLException {
|
protected Connection useDriverManager() throws SQLException {
|
||||||
String elasticsearchAddress = getProtocol() + "://" + elasticsearchAddress();
|
String elasticsearchAddress = getProtocol() + "://" + elasticsearchAddress();
|
||||||
// tag::connect-dm
|
// tag::connect-dm
|
||||||
|
@ -114,6 +138,8 @@ public abstract class JdbcIntegrationTestCase extends ESRestTestCase {
|
||||||
protected Properties connectionProperties() {
|
protected Properties connectionProperties() {
|
||||||
Properties connectionProperties = new Properties();
|
Properties connectionProperties = new Properties();
|
||||||
connectionProperties.put("timezone", randomKnownTimeZone());
|
connectionProperties.put("timezone", randomKnownTimeZone());
|
||||||
|
// in the tests, don't be lenient towards multi values
|
||||||
|
connectionProperties.put("field.multi.value.leniency", "false");
|
||||||
return connectionProperties;
|
return connectionProperties;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -9,11 +9,11 @@ import org.elasticsearch.client.Request;
|
||||||
import org.elasticsearch.common.CheckedBiFunction;
|
import org.elasticsearch.common.CheckedBiFunction;
|
||||||
import org.elasticsearch.common.CheckedConsumer;
|
import org.elasticsearch.common.CheckedConsumer;
|
||||||
import org.elasticsearch.common.CheckedFunction;
|
import org.elasticsearch.common.CheckedFunction;
|
||||||
|
import org.elasticsearch.common.CheckedSupplier;
|
||||||
import org.elasticsearch.common.Strings;
|
import org.elasticsearch.common.Strings;
|
||||||
import org.elasticsearch.common.collect.Tuple;
|
import org.elasticsearch.common.collect.Tuple;
|
||||||
import org.elasticsearch.common.xcontent.XContentBuilder;
|
import org.elasticsearch.common.xcontent.XContentBuilder;
|
||||||
import org.elasticsearch.common.xcontent.json.JsonXContent;
|
import org.elasticsearch.common.xcontent.json.JsonXContent;
|
||||||
import org.elasticsearch.xpack.sql.jdbc.EsDataSource;
|
|
||||||
import org.elasticsearch.xpack.sql.jdbc.EsType;
|
import org.elasticsearch.xpack.sql.jdbc.EsType;
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
|
@ -22,7 +22,6 @@ import java.io.Reader;
|
||||||
import java.sql.Blob;
|
import java.sql.Blob;
|
||||||
import java.sql.Clob;
|
import java.sql.Clob;
|
||||||
import java.sql.Connection;
|
import java.sql.Connection;
|
||||||
import java.sql.DriverManager;
|
|
||||||
import java.sql.NClob;
|
import java.sql.NClob;
|
||||||
import java.sql.PreparedStatement;
|
import java.sql.PreparedStatement;
|
||||||
import java.sql.ResultSet;
|
import java.sql.ResultSet;
|
||||||
|
@ -80,6 +79,34 @@ public class ResultSetTestCase extends JdbcIntegrationTestCase {
|
||||||
dateTimeTestingFields.put(new Tuple<String, Object>("test_keyword", "true"), EsType.KEYWORD);
|
dateTimeTestingFields.put(new Tuple<String, Object>("test_keyword", "true"), EsType.KEYWORD);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void testMultiValueFieldWithMultiValueLeniencyEnabled() throws Exception {
|
||||||
|
createTestDataForMultiValueTests();
|
||||||
|
|
||||||
|
doWithQuery(() -> esWithLeniency(true), "SELECT int, keyword FROM test", (results) -> {
|
||||||
|
results.next();
|
||||||
|
Object number = results.getObject(1);
|
||||||
|
Object string = results.getObject(2);
|
||||||
|
assertEquals(-10, number);
|
||||||
|
assertEquals("-10", string);
|
||||||
|
assertFalse(results.next());
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
public void testMultiValueFieldWithMultiValueLeniencyDisabled() throws Exception {
|
||||||
|
createTestDataForMultiValueTests();
|
||||||
|
|
||||||
|
SQLException expected = expectThrows(SQLException.class,
|
||||||
|
() -> doWithQuery(() -> esWithLeniency(false), "SELECT int, keyword FROM test", (results) -> {
|
||||||
|
}));
|
||||||
|
assertTrue(expected.getMessage().contains("Arrays (returned by [int]) are not supported"));
|
||||||
|
|
||||||
|
// default has multi value disabled
|
||||||
|
expected = expectThrows(SQLException.class,
|
||||||
|
() -> doWithQuery(() -> esJdbc(), "SELECT int, keyword FROM test", (results) -> {
|
||||||
|
}));
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
// Byte values testing
|
// Byte values testing
|
||||||
public void testGettingValidByteWithoutCasting() throws Exception {
|
public void testGettingValidByteWithoutCasting() throws Exception {
|
||||||
byte random1 = randomByte();
|
byte random1 = randomByte();
|
||||||
|
@ -1132,7 +1159,7 @@ public class ResultSetTestCase extends JdbcIntegrationTestCase {
|
||||||
/*
|
/*
|
||||||
* Checks StackOverflowError fix for https://github.com/elastic/elasticsearch/pull/31735
|
* Checks StackOverflowError fix for https://github.com/elastic/elasticsearch/pull/31735
|
||||||
*/
|
*/
|
||||||
public void testNoInfiniteRecursiveGetObjectCalls() throws SQLException, IOException {
|
public void testNoInfiniteRecursiveGetObjectCalls() throws Exception {
|
||||||
index("library", "1", builder -> {
|
index("library", "1", builder -> {
|
||||||
builder.field("name", "Don Quixote");
|
builder.field("name", "Don Quixote");
|
||||||
builder.field("page_count", 1072);
|
builder.field("page_count", 1072);
|
||||||
|
@ -1303,17 +1330,16 @@ public class ResultSetTestCase extends JdbcIntegrationTestCase {
|
||||||
}
|
}
|
||||||
|
|
||||||
private void doWithQuery(String query, CheckedConsumer<ResultSet, SQLException> consumer) throws SQLException {
|
private void doWithQuery(String query, CheckedConsumer<ResultSet, SQLException> consumer) throws SQLException {
|
||||||
try (Connection connection = esJdbc()) {
|
doWithQuery(() -> esJdbc(), query, consumer);
|
||||||
try (PreparedStatement statement = connection.prepareStatement(query)) {
|
|
||||||
try (ResultSet results = statement.executeQuery()) {
|
|
||||||
consumer.accept(results);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private void doWithQueryAndTimezone(String query, String tz, CheckedConsumer<ResultSet, SQLException> consumer) throws SQLException {
|
private void doWithQueryAndTimezone(String query, String tz, CheckedConsumer<ResultSet, SQLException> consumer) throws SQLException {
|
||||||
try (Connection connection = esJdbc(tz)) {
|
doWithQuery(() -> esJdbc(tz), query, consumer);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void doWithQuery(CheckedSupplier<Connection, SQLException> con, String query, CheckedConsumer<ResultSet, SQLException> consumer)
|
||||||
|
throws SQLException {
|
||||||
|
try (Connection connection = con.get()) {
|
||||||
try (PreparedStatement statement = connection.prepareStatement(query)) {
|
try (PreparedStatement statement = connection.prepareStatement(query)) {
|
||||||
try (ResultSet results = statement.executeQuery()) {
|
try (ResultSet results = statement.executeQuery()) {
|
||||||
consumer.accept(results);
|
consumer.accept(results);
|
||||||
|
@ -1355,7 +1381,29 @@ public class ResultSetTestCase extends JdbcIntegrationTestCase {
|
||||||
client().performRequest(request);
|
client().performRequest(request);
|
||||||
}
|
}
|
||||||
|
|
||||||
private void createTestDataForByteValueTests(byte random1, byte random2, byte random3) throws Exception, IOException {
|
private void createTestDataForMultiValueTests() throws Exception {
|
||||||
|
createIndex("test");
|
||||||
|
updateMapping("test", builder -> {
|
||||||
|
builder.startObject("int").field("type", "integer").endObject();
|
||||||
|
builder.startObject("keyword").field("type", "keyword").endObject();
|
||||||
|
});
|
||||||
|
|
||||||
|
Integer[] values = randomArray(3, 15, s -> new Integer[s], () -> Integer.valueOf(randomInt(50)));
|
||||||
|
// add the minimal value in the middle yet the test will pick it up since the results are sorted
|
||||||
|
values[2] = Integer.valueOf(-10);
|
||||||
|
|
||||||
|
String[] stringValues = new String[values.length];
|
||||||
|
for (int i = 0; i < values.length; i++) {
|
||||||
|
stringValues[i] = String.valueOf(values[i]);
|
||||||
|
}
|
||||||
|
|
||||||
|
index("test", "1", builder -> {
|
||||||
|
builder.array("int", (Object[]) values);
|
||||||
|
builder.array("keyword", stringValues);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
private void createTestDataForByteValueTests(byte random1, byte random2, byte random3) throws Exception {
|
||||||
createIndex("test");
|
createIndex("test");
|
||||||
updateMapping("test", builder -> {
|
updateMapping("test", builder -> {
|
||||||
builder.startObject("test_byte").field("type", "byte").endObject();
|
builder.startObject("test_byte").field("type", "byte").endObject();
|
||||||
|
@ -1373,7 +1421,7 @@ public class ResultSetTestCase extends JdbcIntegrationTestCase {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
private void createTestDataForShortValueTests(short random1, short random2, short random3) throws Exception, IOException {
|
private void createTestDataForShortValueTests(short random1, short random2, short random3) throws Exception {
|
||||||
createIndex("test");
|
createIndex("test");
|
||||||
updateMapping("test", builder -> {
|
updateMapping("test", builder -> {
|
||||||
builder.startObject("test_short").field("type", "short").endObject();
|
builder.startObject("test_short").field("type", "short").endObject();
|
||||||
|
@ -1391,7 +1439,7 @@ public class ResultSetTestCase extends JdbcIntegrationTestCase {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
private void createTestDataForIntegerValueTests(int random1, int random2, int random3) throws Exception, IOException {
|
private void createTestDataForIntegerValueTests(int random1, int random2, int random3) throws Exception {
|
||||||
createIndex("test");
|
createIndex("test");
|
||||||
updateMapping("test", builder -> {
|
updateMapping("test", builder -> {
|
||||||
builder.startObject("test_integer").field("type", "integer").endObject();
|
builder.startObject("test_integer").field("type", "integer").endObject();
|
||||||
|
@ -1409,7 +1457,7 @@ public class ResultSetTestCase extends JdbcIntegrationTestCase {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
private void createTestDataForLongValueTests(long random1, long random2, long random3) throws Exception, IOException {
|
private void createTestDataForLongValueTests(long random1, long random2, long random3) throws Exception {
|
||||||
createIndex("test");
|
createIndex("test");
|
||||||
updateMapping("test", builder -> {
|
updateMapping("test", builder -> {
|
||||||
builder.startObject("test_long").field("type", "long").endObject();
|
builder.startObject("test_long").field("type", "long").endObject();
|
||||||
|
@ -1427,7 +1475,7 @@ public class ResultSetTestCase extends JdbcIntegrationTestCase {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
private void createTestDataForDoubleValueTests(double random1, double random2, double random3) throws Exception, IOException {
|
private void createTestDataForDoubleValueTests(double random1, double random2, double random3) throws Exception {
|
||||||
createIndex("test");
|
createIndex("test");
|
||||||
updateMapping("test", builder -> {
|
updateMapping("test", builder -> {
|
||||||
builder.startObject("test_double").field("type", "double").endObject();
|
builder.startObject("test_double").field("type", "double").endObject();
|
||||||
|
@ -1445,7 +1493,7 @@ public class ResultSetTestCase extends JdbcIntegrationTestCase {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
private void createTestDataForFloatValueTests(float random1, float random2, float random3) throws Exception, IOException {
|
private void createTestDataForFloatValueTests(float random1, float random2, float random3) throws Exception {
|
||||||
createIndex("test");
|
createIndex("test");
|
||||||
updateMapping("test", builder -> {
|
updateMapping("test", builder -> {
|
||||||
builder.startObject("test_float").field("type", "float").endObject();
|
builder.startObject("test_float").field("type", "float").endObject();
|
||||||
|
@ -1481,7 +1529,7 @@ public class ResultSetTestCase extends JdbcIntegrationTestCase {
|
||||||
* Creates test data for all numeric get* methods. All values random and different from the other numeric fields already generated.
|
* Creates test data for all numeric get* methods. All values random and different from the other numeric fields already generated.
|
||||||
* It returns a map containing the field name and its randomly generated value to be later used in checking the returned values.
|
* It returns a map containing the field name and its randomly generated value to be later used in checking the returned values.
|
||||||
*/
|
*/
|
||||||
private Map<String,Number> createTestDataForNumericValueTypes(Supplier<Number> randomGenerator) throws Exception, IOException {
|
private Map<String, Number> createTestDataForNumericValueTypes(Supplier<Number> randomGenerator) throws Exception {
|
||||||
Map<String,Number> map = new HashMap<>();
|
Map<String,Number> map = new HashMap<>();
|
||||||
createIndex("test");
|
createIndex("test");
|
||||||
updateMappingForNumericValuesTests("test");
|
updateMappingForNumericValuesTests("test");
|
||||||
|
@ -1575,31 +1623,19 @@ public class ResultSetTestCase extends JdbcIntegrationTestCase {
|
||||||
}
|
}
|
||||||
|
|
||||||
private Connection esJdbc(String timeZoneId) throws SQLException {
|
private Connection esJdbc(String timeZoneId) throws SQLException {
|
||||||
return randomBoolean() ? useDriverManager(timeZoneId) : useDataSource(timeZoneId);
|
|
||||||
}
|
|
||||||
|
|
||||||
private Connection useDriverManager(String timeZoneId) throws SQLException {
|
|
||||||
String elasticsearchAddress = getProtocol() + "://" + elasticsearchAddress();
|
|
||||||
String address = "jdbc:es://" + elasticsearchAddress;
|
|
||||||
Properties connectionProperties = connectionProperties();
|
Properties connectionProperties = connectionProperties();
|
||||||
connectionProperties.put(JDBC_TIMEZONE, timeZoneId);
|
connectionProperties.put(JDBC_TIMEZONE, timeZoneId);
|
||||||
Connection connection = DriverManager.getConnection(address, connectionProperties);
|
Connection connection = esJdbc(connectionProperties);
|
||||||
|
|
||||||
assertNotNull("The timezone should be specified", connectionProperties.getProperty(JDBC_TIMEZONE));
|
assertNotNull("The timezone should be specified", connectionProperties.getProperty(JDBC_TIMEZONE));
|
||||||
return connection;
|
return connection;
|
||||||
}
|
}
|
||||||
|
|
||||||
private Connection useDataSource(String timeZoneId) throws SQLException {
|
private Connection esWithLeniency(boolean multiValueLeniency) throws SQLException {
|
||||||
String elasticsearchAddress = getProtocol() + "://" + elasticsearchAddress();
|
String property = "field.multi.value.leniency";
|
||||||
EsDataSource dataSource = new EsDataSource();
|
|
||||||
String address = "jdbc:es://" + elasticsearchAddress;
|
|
||||||
dataSource.setUrl(address);
|
|
||||||
Properties connectionProperties = connectionProperties();
|
Properties connectionProperties = connectionProperties();
|
||||||
connectionProperties.put(JDBC_TIMEZONE, timeZoneId);
|
connectionProperties.setProperty(property, Boolean.toString(multiValueLeniency));
|
||||||
dataSource.setProperties(connectionProperties);
|
Connection connection = esJdbc(connectionProperties);
|
||||||
Connection connection = dataSource.getConnection();
|
assertNotNull("The leniency should be specified", connectionProperties.getProperty(property));
|
||||||
|
|
||||||
assertNotNull("The timezone should be specified", connectionProperties.getProperty(JDBC_TIMEZONE));
|
|
||||||
return connection;
|
return connection;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -57,7 +57,7 @@ public abstract class AbstractSqlQueryRequest extends AbstractSqlRequest impleme
|
||||||
}
|
}
|
||||||
|
|
||||||
public AbstractSqlQueryRequest(String query, List<SqlTypedParamValue> params, QueryBuilder filter, ZoneId zoneId,
|
public AbstractSqlQueryRequest(String query, List<SqlTypedParamValue> params, QueryBuilder filter, ZoneId zoneId,
|
||||||
int fetchSize, TimeValue requestTimeout, TimeValue pageTimeout, RequestInfo requestInfo) {
|
int fetchSize, TimeValue requestTimeout, TimeValue pageTimeout, RequestInfo requestInfo) {
|
||||||
super(requestInfo);
|
super(requestInfo);
|
||||||
this.query = query;
|
this.query = query;
|
||||||
this.params = params;
|
this.params = params;
|
||||||
|
|
|
@ -15,6 +15,7 @@ import org.elasticsearch.common.xcontent.ObjectParser;
|
||||||
import org.elasticsearch.common.xcontent.XContentBuilder;
|
import org.elasticsearch.common.xcontent.XContentBuilder;
|
||||||
import org.elasticsearch.common.xcontent.XContentParser;
|
import org.elasticsearch.common.xcontent.XContentParser;
|
||||||
import org.elasticsearch.index.query.QueryBuilder;
|
import org.elasticsearch.index.query.QueryBuilder;
|
||||||
|
import org.elasticsearch.xpack.sql.proto.Protocol;
|
||||||
import org.elasticsearch.xpack.sql.proto.RequestInfo;
|
import org.elasticsearch.xpack.sql.proto.RequestInfo;
|
||||||
import org.elasticsearch.xpack.sql.proto.SqlTypedParamValue;
|
import org.elasticsearch.xpack.sql.proto.SqlTypedParamValue;
|
||||||
|
|
||||||
|
@ -31,10 +32,13 @@ import static org.elasticsearch.action.ValidateActions.addValidationError;
|
||||||
public class SqlQueryRequest extends AbstractSqlQueryRequest {
|
public class SqlQueryRequest extends AbstractSqlQueryRequest {
|
||||||
private static final ObjectParser<SqlQueryRequest, Void> PARSER = objectParser(SqlQueryRequest::new);
|
private static final ObjectParser<SqlQueryRequest, Void> PARSER = objectParser(SqlQueryRequest::new);
|
||||||
static final ParseField COLUMNAR = new ParseField("columnar");
|
static final ParseField COLUMNAR = new ParseField("columnar");
|
||||||
|
static final ParseField FIELD_MULTI_VALUE_LENIENCY = new ParseField("field_multi_value_leniency");
|
||||||
|
|
||||||
|
|
||||||
static {
|
static {
|
||||||
PARSER.declareString(SqlQueryRequest::cursor, CURSOR);
|
PARSER.declareString(SqlQueryRequest::cursor, CURSOR);
|
||||||
PARSER.declareBoolean(SqlQueryRequest::columnar, COLUMNAR);
|
PARSER.declareBoolean(SqlQueryRequest::columnar, COLUMNAR);
|
||||||
|
PARSER.declareBoolean(SqlQueryRequest::fieldMultiValueLeniency, FIELD_MULTI_VALUE_LENIENCY);
|
||||||
}
|
}
|
||||||
|
|
||||||
private String cursor = "";
|
private String cursor = "";
|
||||||
|
@ -43,6 +47,7 @@ public class SqlQueryRequest extends AbstractSqlQueryRequest {
|
||||||
* See {@code SqlTranslateRequest.toXContent}
|
* See {@code SqlTranslateRequest.toXContent}
|
||||||
*/
|
*/
|
||||||
private Boolean columnar = Boolean.FALSE;
|
private Boolean columnar = Boolean.FALSE;
|
||||||
|
private boolean fieldMultiValueLeniency = Protocol.FIELD_MULTI_VALUE_LENIENCY;
|
||||||
|
|
||||||
public SqlQueryRequest() {
|
public SqlQueryRequest() {
|
||||||
super();
|
super();
|
||||||
|
@ -50,10 +55,11 @@ public class SqlQueryRequest extends AbstractSqlQueryRequest {
|
||||||
|
|
||||||
public SqlQueryRequest(String query, List<SqlTypedParamValue> params, QueryBuilder filter, ZoneId zoneId,
|
public SqlQueryRequest(String query, List<SqlTypedParamValue> params, QueryBuilder filter, ZoneId zoneId,
|
||||||
int fetchSize, TimeValue requestTimeout, TimeValue pageTimeout, Boolean columnar,
|
int fetchSize, TimeValue requestTimeout, TimeValue pageTimeout, Boolean columnar,
|
||||||
String cursor, RequestInfo requestInfo) {
|
String cursor, RequestInfo requestInfo, boolean fieldMultiValueLeniency) {
|
||||||
super(query, params, filter, zoneId, fetchSize, requestTimeout, pageTimeout, requestInfo);
|
super(query, params, filter, zoneId, fetchSize, requestTimeout, pageTimeout, requestInfo);
|
||||||
this.cursor = cursor;
|
this.cursor = cursor;
|
||||||
this.columnar = columnar;
|
this.columnar = columnar;
|
||||||
|
this.fieldMultiValueLeniency = fieldMultiValueLeniency;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -99,10 +105,21 @@ public class SqlQueryRequest extends AbstractSqlQueryRequest {
|
||||||
return this;
|
return this;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
public SqlQueryRequest fieldMultiValueLeniency(boolean leniency) {
|
||||||
|
this.fieldMultiValueLeniency = leniency;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean fieldMultiValueLeniency() {
|
||||||
|
return fieldMultiValueLeniency;
|
||||||
|
}
|
||||||
|
|
||||||
public SqlQueryRequest(StreamInput in) throws IOException {
|
public SqlQueryRequest(StreamInput in) throws IOException {
|
||||||
super(in);
|
super(in);
|
||||||
cursor = in.readString();
|
cursor = in.readString();
|
||||||
columnar = in.readOptionalBoolean();
|
columnar = in.readOptionalBoolean();
|
||||||
|
fieldMultiValueLeniency = in.readBoolean();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -110,6 +127,7 @@ public class SqlQueryRequest extends AbstractSqlQueryRequest {
|
||||||
super.writeTo(out);
|
super.writeTo(out);
|
||||||
out.writeString(cursor);
|
out.writeString(cursor);
|
||||||
out.writeOptionalBoolean(columnar);
|
out.writeOptionalBoolean(columnar);
|
||||||
|
out.writeBoolean(fieldMultiValueLeniency);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -121,7 +139,8 @@ public class SqlQueryRequest extends AbstractSqlQueryRequest {
|
||||||
public boolean equals(Object obj) {
|
public boolean equals(Object obj) {
|
||||||
return super.equals(obj)
|
return super.equals(obj)
|
||||||
&& Objects.equals(cursor, ((SqlQueryRequest) obj).cursor)
|
&& Objects.equals(cursor, ((SqlQueryRequest) obj).cursor)
|
||||||
&& Objects.equals(columnar, ((SqlQueryRequest) obj).columnar);
|
&& Objects.equals(columnar, ((SqlQueryRequest) obj).columnar)
|
||||||
|
&& fieldMultiValueLeniency == ((SqlQueryRequest) obj).fieldMultiValueLeniency;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -133,7 +152,8 @@ public class SqlQueryRequest extends AbstractSqlQueryRequest {
|
||||||
public XContentBuilder toXContent(XContentBuilder builder, Params params) throws IOException {
|
public XContentBuilder toXContent(XContentBuilder builder, Params params) throws IOException {
|
||||||
// This is needed just to test round-trip compatibility with proto.SqlQueryRequest
|
// This is needed just to test round-trip compatibility with proto.SqlQueryRequest
|
||||||
return new org.elasticsearch.xpack.sql.proto.SqlQueryRequest(query(), params(), zoneId(), fetchSize(), requestTimeout(),
|
return new org.elasticsearch.xpack.sql.proto.SqlQueryRequest(query(), params(), zoneId(), fetchSize(), requestTimeout(),
|
||||||
pageTimeout(), filter(), columnar(), cursor(), requestInfo()).toXContent(builder, params);
|
pageTimeout(), filter(), columnar(), cursor(), requestInfo(), fieldMultiValueLeniency())
|
||||||
|
.toXContent(builder, params);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static SqlQueryRequest fromXContent(XContentParser parser) {
|
public static SqlQueryRequest fromXContent(XContentParser parser) {
|
||||||
|
|
|
@ -25,14 +25,15 @@ public class SqlQueryRequestBuilder extends ActionRequestBuilder<SqlQueryRequest
|
||||||
|
|
||||||
public SqlQueryRequestBuilder(ElasticsearchClient client, SqlQueryAction action) {
|
public SqlQueryRequestBuilder(ElasticsearchClient client, SqlQueryAction action) {
|
||||||
this(client, action, "", Collections.emptyList(), null, Protocol.TIME_ZONE, Protocol.FETCH_SIZE, Protocol.REQUEST_TIMEOUT,
|
this(client, action, "", Collections.emptyList(), null, Protocol.TIME_ZONE, Protocol.FETCH_SIZE, Protocol.REQUEST_TIMEOUT,
|
||||||
Protocol.PAGE_TIMEOUT, false, "", new RequestInfo(Mode.PLAIN));
|
Protocol.PAGE_TIMEOUT, false, "", new RequestInfo(Mode.PLAIN), Protocol.FIELD_MULTI_VALUE_LENIENCY);
|
||||||
}
|
}
|
||||||
|
|
||||||
public SqlQueryRequestBuilder(ElasticsearchClient client, SqlQueryAction action, String query, List<SqlTypedParamValue> params,
|
public SqlQueryRequestBuilder(ElasticsearchClient client, SqlQueryAction action, String query, List<SqlTypedParamValue> params,
|
||||||
QueryBuilder filter, ZoneId zoneId, int fetchSize, TimeValue requestTimeout,
|
QueryBuilder filter, ZoneId zoneId, int fetchSize, TimeValue requestTimeout,
|
||||||
TimeValue pageTimeout, boolean columnar, String nextPageInfo, RequestInfo requestInfo) {
|
TimeValue pageTimeout, boolean columnar, String nextPageInfo, RequestInfo requestInfo,
|
||||||
|
boolean multiValueFieldLeniency) {
|
||||||
super(client, action, new SqlQueryRequest(query, params, filter, zoneId, fetchSize, requestTimeout, pageTimeout, columnar,
|
super(client, action, new SqlQueryRequest(query, params, filter, zoneId, fetchSize, requestTimeout, pageTimeout, columnar,
|
||||||
nextPageInfo, requestInfo));
|
nextPageInfo, requestInfo, multiValueFieldLeniency));
|
||||||
}
|
}
|
||||||
|
|
||||||
public SqlQueryRequestBuilder query(String query) {
|
public SqlQueryRequestBuilder query(String query) {
|
||||||
|
@ -84,4 +85,9 @@ public class SqlQueryRequestBuilder extends ActionRequestBuilder<SqlQueryRequest
|
||||||
request.fetchSize(fetchSize);
|
request.fetchSize(fetchSize);
|
||||||
return this;
|
return this;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public SqlQueryRequestBuilder multiValueFieldLeniency(boolean lenient) {
|
||||||
|
request.fieldMultiValueLeniency(lenient);
|
||||||
|
return this;
|
||||||
|
}
|
||||||
}
|
}
|
|
@ -64,8 +64,12 @@ public class SqlTranslateRequest extends AbstractSqlQueryRequest {
|
||||||
@Override
|
@Override
|
||||||
public XContentBuilder toXContent(XContentBuilder builder, Params params) throws IOException {
|
public XContentBuilder toXContent(XContentBuilder builder, Params params) throws IOException {
|
||||||
// This is needed just to test parsing of SqlTranslateRequest, so we can reuse SqlQuerySerialization
|
// This is needed just to test parsing of SqlTranslateRequest, so we can reuse SqlQuerySerialization
|
||||||
return new SqlQueryRequest(query(), params(), zoneId(), fetchSize(), requestTimeout(),
|
return new SqlQueryRequest(query(), params(), zoneId(), fetchSize(), requestTimeout(), pageTimeout(),
|
||||||
pageTimeout(), filter(), null, null, requestInfo()).toXContent(builder, params);
|
filter(),
|
||||||
|
null,
|
||||||
|
null,
|
||||||
|
requestInfo(),
|
||||||
|
false).toXContent(builder, params);
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -55,7 +55,8 @@ public class SqlQueryRequestTests extends AbstractSerializingTestCase<SqlQueryRe
|
||||||
protected SqlQueryRequest createTestInstance() {
|
protected SqlQueryRequest createTestInstance() {
|
||||||
return new SqlQueryRequest(randomAlphaOfLength(10), randomParameters(), SqlTestUtils.randomFilterOrNull(random()),
|
return new SqlQueryRequest(randomAlphaOfLength(10), randomParameters(), SqlTestUtils.randomFilterOrNull(random()),
|
||||||
randomZone(), between(1, Integer.MAX_VALUE), randomTV(),
|
randomZone(), between(1, Integer.MAX_VALUE), randomTV(),
|
||||||
randomTV(), randomBoolean(), randomAlphaOfLength(10), requestInfo
|
randomTV(), randomBoolean(), randomAlphaOfLength(10), requestInfo,
|
||||||
|
randomBoolean()
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -114,7 +115,7 @@ public class SqlQueryRequestTests extends AbstractSerializingTestCase<SqlQueryRe
|
||||||
);
|
);
|
||||||
SqlQueryRequest newRequest = new SqlQueryRequest(instance.query(), instance.params(), instance.filter(),
|
SqlQueryRequest newRequest = new SqlQueryRequest(instance.query(), instance.params(), instance.filter(),
|
||||||
instance.zoneId(), instance.fetchSize(), instance.requestTimeout(), instance.pageTimeout(), instance.columnar(),
|
instance.zoneId(), instance.fetchSize(), instance.requestTimeout(), instance.pageTimeout(), instance.columnar(),
|
||||||
instance.cursor(), instance.requestInfo());
|
instance.cursor(), instance.requestInfo(), instance.fieldMultiValueLeniency());
|
||||||
mutator.accept(newRequest);
|
mutator.accept(newRequest);
|
||||||
return newRequest;
|
return newRequest;
|
||||||
}
|
}
|
||||||
|
|
|
@ -25,7 +25,7 @@ public class ServerQueryCliCommand extends AbstractServerCliCommand {
|
||||||
BasicFormatter formatter;
|
BasicFormatter formatter;
|
||||||
String data;
|
String data;
|
||||||
try {
|
try {
|
||||||
response = cliClient.queryInit(line, cliSession.getFetchSize());
|
response = cliClient.basicQuery(line, cliSession.getFetchSize());
|
||||||
formatter = new BasicFormatter(response.columns(), response.rows(), CLI);
|
formatter = new BasicFormatter(response.columns(), response.rows(), CLI);
|
||||||
data = formatter.formatWithHeader(response.columns(), response.rows());
|
data = formatter.formatWithHeader(response.columns(), response.rows());
|
||||||
while (true) {
|
while (true) {
|
||||||
|
|
|
@ -31,11 +31,11 @@ public class ServerQueryCliCommandTests extends ESTestCase {
|
||||||
TestTerminal testTerminal = new TestTerminal();
|
TestTerminal testTerminal = new TestTerminal();
|
||||||
HttpClient client = mock(HttpClient.class);
|
HttpClient client = mock(HttpClient.class);
|
||||||
CliSession cliSession = new CliSession(client);
|
CliSession cliSession = new CliSession(client);
|
||||||
when(client.queryInit("blah", 1000)).thenThrow(new SQLException("test exception"));
|
when(client.basicQuery("blah", 1000)).thenThrow(new SQLException("test exception"));
|
||||||
ServerQueryCliCommand cliCommand = new ServerQueryCliCommand();
|
ServerQueryCliCommand cliCommand = new ServerQueryCliCommand();
|
||||||
assertTrue(cliCommand.handle(testTerminal, cliSession, "blah"));
|
assertTrue(cliCommand.handle(testTerminal, cliSession, "blah"));
|
||||||
assertEquals("<b>Bad request [</b><i>test exception</i><b>]</b>\n", testTerminal.toString());
|
assertEquals("<b>Bad request [</b><i>test exception</i><b>]</b>\n", testTerminal.toString());
|
||||||
verify(client, times(1)).queryInit(eq("blah"), eq(1000));
|
verify(client, times(1)).basicQuery(eq("blah"), eq(1000));
|
||||||
verifyNoMoreInteractions(client);
|
verifyNoMoreInteractions(client);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -44,11 +44,11 @@ public class ServerQueryCliCommandTests extends ESTestCase {
|
||||||
HttpClient client = mock(HttpClient.class);
|
HttpClient client = mock(HttpClient.class);
|
||||||
CliSession cliSession = new CliSession(client);
|
CliSession cliSession = new CliSession(client);
|
||||||
cliSession.setFetchSize(10);
|
cliSession.setFetchSize(10);
|
||||||
when(client.queryInit("test query", 10)).thenReturn(fakeResponse("", true, "foo"));
|
when(client.basicQuery("test query", 10)).thenReturn(fakeResponse("", true, "foo"));
|
||||||
ServerQueryCliCommand cliCommand = new ServerQueryCliCommand();
|
ServerQueryCliCommand cliCommand = new ServerQueryCliCommand();
|
||||||
assertTrue(cliCommand.handle(testTerminal, cliSession, "test query"));
|
assertTrue(cliCommand.handle(testTerminal, cliSession, "test query"));
|
||||||
assertEquals(" field \n---------------\nfoo \n<flush/>", testTerminal.toString());
|
assertEquals(" field \n---------------\nfoo \n<flush/>", testTerminal.toString());
|
||||||
verify(client, times(1)).queryInit(eq("test query"), eq(10));
|
verify(client, times(1)).basicQuery(eq("test query"), eq(10));
|
||||||
verifyNoMoreInteractions(client);
|
verifyNoMoreInteractions(client);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -57,14 +57,14 @@ public class ServerQueryCliCommandTests extends ESTestCase {
|
||||||
HttpClient client = mock(HttpClient.class);
|
HttpClient client = mock(HttpClient.class);
|
||||||
CliSession cliSession = new CliSession(client);
|
CliSession cliSession = new CliSession(client);
|
||||||
cliSession.setFetchSize(10);
|
cliSession.setFetchSize(10);
|
||||||
when(client.queryInit("test query", 10)).thenReturn(fakeResponse("my_cursor1", true, "first"));
|
when(client.basicQuery("test query", 10)).thenReturn(fakeResponse("my_cursor1", true, "first"));
|
||||||
when(client.nextPage("my_cursor1")).thenReturn(fakeResponse("my_cursor2", false, "second"));
|
when(client.nextPage("my_cursor1")).thenReturn(fakeResponse("my_cursor2", false, "second"));
|
||||||
when(client.nextPage("my_cursor2")).thenReturn(fakeResponse("", false, "third"));
|
when(client.nextPage("my_cursor2")).thenReturn(fakeResponse("", false, "third"));
|
||||||
ServerQueryCliCommand cliCommand = new ServerQueryCliCommand();
|
ServerQueryCliCommand cliCommand = new ServerQueryCliCommand();
|
||||||
assertTrue(cliCommand.handle(testTerminal, cliSession, "test query"));
|
assertTrue(cliCommand.handle(testTerminal, cliSession, "test query"));
|
||||||
assertEquals(" field \n---------------\nfirst \nsecond \nthird \n<flush/>",
|
assertEquals(" field \n---------------\nfirst \nsecond \nthird \n<flush/>",
|
||||||
testTerminal.toString());
|
testTerminal.toString());
|
||||||
verify(client, times(1)).queryInit(eq("test query"), eq(10));
|
verify(client, times(1)).basicQuery(eq("test query"), eq(10));
|
||||||
verify(client, times(2)).nextPage(any());
|
verify(client, times(2)).nextPage(any());
|
||||||
verifyNoMoreInteractions(client);
|
verifyNoMoreInteractions(client);
|
||||||
}
|
}
|
||||||
|
@ -76,13 +76,13 @@ public class ServerQueryCliCommandTests extends ESTestCase {
|
||||||
cliSession.setFetchSize(15);
|
cliSession.setFetchSize(15);
|
||||||
// Set a separator
|
// Set a separator
|
||||||
cliSession.setFetchSeparator("-----");
|
cliSession.setFetchSeparator("-----");
|
||||||
when(client.queryInit("test query", 15)).thenReturn(fakeResponse("my_cursor1", true, "first"));
|
when(client.basicQuery("test query", 15)).thenReturn(fakeResponse("my_cursor1", true, "first"));
|
||||||
when(client.nextPage("my_cursor1")).thenReturn(fakeResponse("", false, "second"));
|
when(client.nextPage("my_cursor1")).thenReturn(fakeResponse("", false, "second"));
|
||||||
ServerQueryCliCommand cliCommand = new ServerQueryCliCommand();
|
ServerQueryCliCommand cliCommand = new ServerQueryCliCommand();
|
||||||
assertTrue(cliCommand.handle(testTerminal, cliSession, "test query"));
|
assertTrue(cliCommand.handle(testTerminal, cliSession, "test query"));
|
||||||
assertEquals(" field \n---------------\nfirst \n-----\nsecond \n<flush/>",
|
assertEquals(" field \n---------------\nfirst \n-----\nsecond \n<flush/>",
|
||||||
testTerminal.toString());
|
testTerminal.toString());
|
||||||
verify(client, times(1)).queryInit(eq("test query"), eq(15));
|
verify(client, times(1)).basicQuery(eq("test query"), eq(15));
|
||||||
verify(client, times(1)).nextPage(any());
|
verify(client, times(1)).nextPage(any());
|
||||||
verifyNoMoreInteractions(client);
|
verifyNoMoreInteractions(client);
|
||||||
}
|
}
|
||||||
|
@ -92,14 +92,14 @@ public class ServerQueryCliCommandTests extends ESTestCase {
|
||||||
HttpClient client = mock(HttpClient.class);
|
HttpClient client = mock(HttpClient.class);
|
||||||
CliSession cliSession = new CliSession(client);
|
CliSession cliSession = new CliSession(client);
|
||||||
cliSession.setFetchSize(15);
|
cliSession.setFetchSize(15);
|
||||||
when(client.queryInit("test query", 15)).thenReturn(fakeResponse("my_cursor1", true, "first"));
|
when(client.basicQuery("test query", 15)).thenReturn(fakeResponse("my_cursor1", true, "first"));
|
||||||
when(client.nextPage("my_cursor1")).thenThrow(new SQLException("test exception"));
|
when(client.nextPage("my_cursor1")).thenThrow(new SQLException("test exception"));
|
||||||
when(client.queryClose("my_cursor1", Mode.CLI)).thenReturn(true);
|
when(client.queryClose("my_cursor1", Mode.CLI)).thenReturn(true);
|
||||||
ServerQueryCliCommand cliCommand = new ServerQueryCliCommand();
|
ServerQueryCliCommand cliCommand = new ServerQueryCliCommand();
|
||||||
assertTrue(cliCommand.handle(testTerminal, cliSession, "test query"));
|
assertTrue(cliCommand.handle(testTerminal, cliSession, "test query"));
|
||||||
assertEquals(" field \n---------------\nfirst \n" +
|
assertEquals(" field \n---------------\nfirst \n" +
|
||||||
"<b>Bad request [</b><i>test exception</i><b>]</b>\n", testTerminal.toString());
|
"<b>Bad request [</b><i>test exception</i><b>]</b>\n", testTerminal.toString());
|
||||||
verify(client, times(1)).queryInit(eq("test query"), eq(15));
|
verify(client, times(1)).basicQuery(eq("test query"), eq(15));
|
||||||
verify(client, times(1)).nextPage(any());
|
verify(client, times(1)).nextPage(any());
|
||||||
verify(client, times(1)).queryClose(eq("my_cursor1"), eq(Mode.CLI));
|
verify(client, times(1)).queryClose(eq("my_cursor1"), eq(Mode.CLI));
|
||||||
verifyNoMoreInteractions(client);
|
verifyNoMoreInteractions(client);
|
||||||
|
|
|
@ -32,7 +32,6 @@ import java.io.InputStream;
|
||||||
import java.security.AccessController;
|
import java.security.AccessController;
|
||||||
import java.security.PrivilegedAction;
|
import java.security.PrivilegedAction;
|
||||||
import java.sql.SQLException;
|
import java.sql.SQLException;
|
||||||
import java.time.ZoneId;
|
|
||||||
import java.util.Collections;
|
import java.util.Collections;
|
||||||
import java.util.function.Function;
|
import java.util.function.Function;
|
||||||
|
|
||||||
|
@ -61,12 +60,18 @@ public class HttpClient {
|
||||||
return get("/", MainResponse::fromXContent);
|
return get("/", MainResponse::fromXContent);
|
||||||
}
|
}
|
||||||
|
|
||||||
public SqlQueryResponse queryInit(String query, int fetchSize) throws SQLException {
|
public SqlQueryResponse basicQuery(String query, int fetchSize) throws SQLException {
|
||||||
// TODO allow customizing the time zone - this is what session set/reset/get should be about
|
// TODO allow customizing the time zone - this is what session set/reset/get should be about
|
||||||
// method called only from CLI
|
// method called only from CLI
|
||||||
SqlQueryRequest sqlRequest = new SqlQueryRequest(query, Collections.emptyList(), null, ZoneId.of("Z"),
|
SqlQueryRequest sqlRequest = new SqlQueryRequest(query, Collections.emptyList(), Protocol.TIME_ZONE,
|
||||||
fetchSize, TimeValue.timeValueMillis(cfg.queryTimeout()), TimeValue.timeValueMillis(cfg.pageTimeout()),
|
fetchSize,
|
||||||
false, new RequestInfo(Mode.CLI));
|
TimeValue.timeValueMillis(cfg.queryTimeout()),
|
||||||
|
TimeValue.timeValueMillis(cfg.pageTimeout()),
|
||||||
|
null,
|
||||||
|
Boolean.FALSE,
|
||||||
|
null,
|
||||||
|
new RequestInfo(Mode.CLI),
|
||||||
|
false);
|
||||||
return query(sqlRequest);
|
return query(sqlRequest);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -22,6 +22,7 @@ public final class Protocol {
|
||||||
public static final int FETCH_SIZE = 1000;
|
public static final int FETCH_SIZE = 1000;
|
||||||
public static final TimeValue REQUEST_TIMEOUT = TimeValue.timeValueSeconds(90);
|
public static final TimeValue REQUEST_TIMEOUT = TimeValue.timeValueSeconds(90);
|
||||||
public static final TimeValue PAGE_TIMEOUT = TimeValue.timeValueSeconds(45);
|
public static final TimeValue PAGE_TIMEOUT = TimeValue.timeValueSeconds(45);
|
||||||
|
public static final boolean FIELD_MULTI_VALUE_LENIENCY = false;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* SQL-related endpoints
|
* SQL-related endpoints
|
||||||
|
|
|
@ -32,11 +32,13 @@ public class SqlQueryRequest extends AbstractSqlRequest {
|
||||||
private final ToXContent filter;
|
private final ToXContent filter;
|
||||||
private final Boolean columnar;
|
private final Boolean columnar;
|
||||||
private final List<SqlTypedParamValue> params;
|
private final List<SqlTypedParamValue> params;
|
||||||
|
private final boolean fieldMultiValueLeniency;
|
||||||
|
|
||||||
|
|
||||||
public SqlQueryRequest(String query, List<SqlTypedParamValue> params, ZoneId zoneId, int fetchSize,
|
public SqlQueryRequest(String query, List<SqlTypedParamValue> params, ZoneId zoneId, int fetchSize,
|
||||||
TimeValue requestTimeout, TimeValue pageTimeout, ToXContent filter, Boolean columnar,
|
TimeValue requestTimeout, TimeValue pageTimeout, ToXContent filter, Boolean columnar,
|
||||||
String cursor, RequestInfo requestInfo) {
|
String cursor, RequestInfo requestInfo,
|
||||||
|
boolean fieldMultiValueLeniency) {
|
||||||
super(requestInfo);
|
super(requestInfo);
|
||||||
this.query = query;
|
this.query = query;
|
||||||
this.params = params;
|
this.params = params;
|
||||||
|
@ -47,16 +49,12 @@ public class SqlQueryRequest extends AbstractSqlRequest {
|
||||||
this.filter = filter;
|
this.filter = filter;
|
||||||
this.columnar = columnar;
|
this.columnar = columnar;
|
||||||
this.cursor = cursor;
|
this.cursor = cursor;
|
||||||
}
|
this.fieldMultiValueLeniency = fieldMultiValueLeniency;
|
||||||
|
|
||||||
public SqlQueryRequest(String query, List<SqlTypedParamValue> params, ToXContent filter, ZoneId zoneId,
|
|
||||||
int fetchSize, TimeValue requestTimeout, TimeValue pageTimeout, boolean columnar, RequestInfo requestInfo) {
|
|
||||||
this(query, params, zoneId, fetchSize, requestTimeout, pageTimeout, filter, columnar, null, requestInfo);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public SqlQueryRequest(String cursor, TimeValue requestTimeout, TimeValue pageTimeout, RequestInfo requestInfo) {
|
public SqlQueryRequest(String cursor, TimeValue requestTimeout, TimeValue pageTimeout, RequestInfo requestInfo) {
|
||||||
this("", Collections.emptyList(), Protocol.TIME_ZONE, Protocol.FETCH_SIZE, requestTimeout, pageTimeout,
|
this("", Collections.emptyList(), Protocol.TIME_ZONE, Protocol.FETCH_SIZE, requestTimeout, pageTimeout,
|
||||||
null, false, cursor, requestInfo);
|
null, false, cursor, requestInfo, Protocol.FIELD_MULTI_VALUE_LENIENCY);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -125,6 +123,10 @@ public class SqlQueryRequest extends AbstractSqlRequest {
|
||||||
return columnar;
|
return columnar;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public boolean fieldMultiValueLeniency() {
|
||||||
|
return fieldMultiValueLeniency;
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean equals(Object o) {
|
public boolean equals(Object o) {
|
||||||
if (this == o) {
|
if (this == o) {
|
||||||
|
@ -145,12 +147,14 @@ public class SqlQueryRequest extends AbstractSqlRequest {
|
||||||
Objects.equals(pageTimeout, that.pageTimeout) &&
|
Objects.equals(pageTimeout, that.pageTimeout) &&
|
||||||
Objects.equals(filter, that.filter) &&
|
Objects.equals(filter, that.filter) &&
|
||||||
Objects.equals(columnar, that.columnar) &&
|
Objects.equals(columnar, that.columnar) &&
|
||||||
Objects.equals(cursor, that.cursor);
|
Objects.equals(cursor, that.cursor) &&
|
||||||
|
fieldMultiValueLeniency == that.fieldMultiValueLeniency;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public int hashCode() {
|
public int hashCode() {
|
||||||
return Objects.hash(super.hashCode(), query, zoneId, fetchSize, requestTimeout, pageTimeout, filter, columnar, cursor);
|
return Objects.hash(super.hashCode(), query, zoneId, fetchSize, requestTimeout, pageTimeout,
|
||||||
|
filter, columnar, cursor, fieldMultiValueLeniency);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -188,6 +192,9 @@ public class SqlQueryRequest extends AbstractSqlRequest {
|
||||||
if (columnar != null) {
|
if (columnar != null) {
|
||||||
builder.field("columnar", columnar);
|
builder.field("columnar", columnar);
|
||||||
}
|
}
|
||||||
|
if (fieldMultiValueLeniency) {
|
||||||
|
builder.field("field_multi_value_leniency", fieldMultiValueLeniency);
|
||||||
|
}
|
||||||
if (cursor != null) {
|
if (cursor != null) {
|
||||||
builder.field("cursor", cursor);
|
builder.field("cursor", cursor);
|
||||||
}
|
}
|
||||||
|
|
|
@ -118,13 +118,13 @@ public class Querier {
|
||||||
|
|
||||||
if (query.isAggsOnly()) {
|
if (query.isAggsOnly()) {
|
||||||
if (query.aggs().useImplicitGroupBy()) {
|
if (query.aggs().useImplicitGroupBy()) {
|
||||||
l = new ImplicitGroupActionListener(listener, client, timeout, output, query, search);
|
l = new ImplicitGroupActionListener(listener, client, cfg, output, query, search);
|
||||||
} else {
|
} else {
|
||||||
l = new CompositeActionListener(listener, client, timeout, output, query, search);
|
l = new CompositeActionListener(listener, client, cfg, output, query, search);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
search.scroll(keepAlive);
|
search.scroll(keepAlive);
|
||||||
l = new ScrollActionListener(listener, client, timeout, output, query);
|
l = new ScrollActionListener(listener, client, cfg, output, query);
|
||||||
}
|
}
|
||||||
|
|
||||||
client.search(search, l);
|
client.search(search, l);
|
||||||
|
@ -309,9 +309,9 @@ public class Querier {
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
ImplicitGroupActionListener(ActionListener<SchemaRowSet> listener, Client client, TimeValue keepAlive, List<Attribute> output,
|
ImplicitGroupActionListener(ActionListener<SchemaRowSet> listener, Client client, Configuration cfg, List<Attribute> output,
|
||||||
QueryContainer query, SearchRequest request) {
|
QueryContainer query, SearchRequest request) {
|
||||||
super(listener, client, keepAlive, output, query, request);
|
super(listener, client, cfg, output, query, request);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -360,9 +360,9 @@ public class Querier {
|
||||||
*/
|
*/
|
||||||
static class CompositeActionListener extends BaseAggActionListener {
|
static class CompositeActionListener extends BaseAggActionListener {
|
||||||
|
|
||||||
CompositeActionListener(ActionListener<SchemaRowSet> listener, Client client, TimeValue keepAlive,
|
CompositeActionListener(ActionListener<SchemaRowSet> listener, Client client, Configuration cfg,
|
||||||
List<Attribute> output, QueryContainer query, SearchRequest request) {
|
List<Attribute> output, QueryContainer query, SearchRequest request) {
|
||||||
super(listener, client, keepAlive, output, query, request);
|
super(listener, client, cfg, output, query, request);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -404,9 +404,9 @@ public class Querier {
|
||||||
final SearchRequest request;
|
final SearchRequest request;
|
||||||
final BitSet mask;
|
final BitSet mask;
|
||||||
|
|
||||||
BaseAggActionListener(ActionListener<SchemaRowSet> listener, Client client, TimeValue keepAlive, List<Attribute> output,
|
BaseAggActionListener(ActionListener<SchemaRowSet> listener, Client client, Configuration cfg, List<Attribute> output,
|
||||||
QueryContainer query, SearchRequest request) {
|
QueryContainer query, SearchRequest request) {
|
||||||
super(listener, client, keepAlive, output);
|
super(listener, client, cfg, output);
|
||||||
|
|
||||||
this.query = query;
|
this.query = query;
|
||||||
this.request = request;
|
this.request = request;
|
||||||
|
@ -467,12 +467,14 @@ public class Querier {
|
||||||
static class ScrollActionListener extends BaseActionListener {
|
static class ScrollActionListener extends BaseActionListener {
|
||||||
private final QueryContainer query;
|
private final QueryContainer query;
|
||||||
private final BitSet mask;
|
private final BitSet mask;
|
||||||
|
private final boolean multiValueFieldLeniency;
|
||||||
|
|
||||||
ScrollActionListener(ActionListener<SchemaRowSet> listener, Client client, TimeValue keepAlive,
|
ScrollActionListener(ActionListener<SchemaRowSet> listener, Client client, Configuration cfg,
|
||||||
List<Attribute> output, QueryContainer query) {
|
List<Attribute> output, QueryContainer query) {
|
||||||
super(listener, client, keepAlive, output);
|
super(listener, client, cfg, output);
|
||||||
this.query = query;
|
this.query = query;
|
||||||
this.mask = query.columnMask(output);
|
this.mask = query.columnMask(output);
|
||||||
|
this.multiValueFieldLeniency = cfg.multiValueFieldLeniency();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -516,12 +518,12 @@ public class Querier {
|
||||||
private HitExtractor createExtractor(FieldExtraction ref) {
|
private HitExtractor createExtractor(FieldExtraction ref) {
|
||||||
if (ref instanceof SearchHitFieldRef) {
|
if (ref instanceof SearchHitFieldRef) {
|
||||||
SearchHitFieldRef f = (SearchHitFieldRef) ref;
|
SearchHitFieldRef f = (SearchHitFieldRef) ref;
|
||||||
return new FieldHitExtractor(f.name(), f.getDataType(), f.useDocValue(), f.hitName());
|
return new FieldHitExtractor(f.name(), f.getDataType(), f.useDocValue(), f.hitName(), multiValueFieldLeniency);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (ref instanceof ScriptFieldRef) {
|
if (ref instanceof ScriptFieldRef) {
|
||||||
ScriptFieldRef f = (ScriptFieldRef) ref;
|
ScriptFieldRef f = (ScriptFieldRef) ref;
|
||||||
return new FieldHitExtractor(f.name(), null, true);
|
return new FieldHitExtractor(f.name(), null, true, multiValueFieldLeniency);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (ref instanceof ComputedRef) {
|
if (ref instanceof ComputedRef) {
|
||||||
|
@ -558,14 +560,16 @@ public class Querier {
|
||||||
final ActionListener<SchemaRowSet> listener;
|
final ActionListener<SchemaRowSet> listener;
|
||||||
|
|
||||||
final Client client;
|
final Client client;
|
||||||
|
final Configuration cfg;
|
||||||
final TimeValue keepAlive;
|
final TimeValue keepAlive;
|
||||||
final Schema schema;
|
final Schema schema;
|
||||||
|
|
||||||
BaseActionListener(ActionListener<SchemaRowSet> listener, Client client, TimeValue keepAlive, List<Attribute> output) {
|
BaseActionListener(ActionListener<SchemaRowSet> listener, Client client, Configuration cfg, List<Attribute> output) {
|
||||||
this.listener = listener;
|
this.listener = listener;
|
||||||
|
|
||||||
this.client = client;
|
this.client = client;
|
||||||
this.keepAlive = keepAlive;
|
this.cfg = cfg;
|
||||||
|
this.keepAlive = cfg.requestTimeout();
|
||||||
this.schema = Rows.schema(output);
|
this.schema = Rows.schema(output);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -29,8 +29,6 @@ import java.util.StringJoiner;
|
||||||
*/
|
*/
|
||||||
public class FieldHitExtractor implements HitExtractor {
|
public class FieldHitExtractor implements HitExtractor {
|
||||||
|
|
||||||
private static final boolean ARRAYS_LENIENCY = false;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Stands for {@code field}. We try to use short names for {@link HitExtractor}s
|
* Stands for {@code field}. We try to use short names for {@link HitExtractor}s
|
||||||
* to save a few bytes when when we send them back to the user.
|
* to save a few bytes when when we send them back to the user.
|
||||||
|
@ -48,16 +46,22 @@ public class FieldHitExtractor implements HitExtractor {
|
||||||
private final String fieldName, hitName;
|
private final String fieldName, hitName;
|
||||||
private final DataType dataType;
|
private final DataType dataType;
|
||||||
private final boolean useDocValue;
|
private final boolean useDocValue;
|
||||||
|
private final boolean arrayLeniency;
|
||||||
private final String[] path;
|
private final String[] path;
|
||||||
|
|
||||||
public FieldHitExtractor(String name, DataType dataType, boolean useDocValue) {
|
public FieldHitExtractor(String name, DataType dataType, boolean useDocValue) {
|
||||||
this(name, dataType, useDocValue, null);
|
this(name, dataType, useDocValue, null, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
public FieldHitExtractor(String name, DataType dataType, boolean useDocValue, String hitName) {
|
public FieldHitExtractor(String name, DataType dataType, boolean useDocValue, boolean arrayLeniency) {
|
||||||
|
this(name, dataType, useDocValue, null, arrayLeniency);
|
||||||
|
}
|
||||||
|
|
||||||
|
public FieldHitExtractor(String name, DataType dataType, boolean useDocValue, String hitName, boolean arrayLeniency) {
|
||||||
this.fieldName = name;
|
this.fieldName = name;
|
||||||
this.dataType = dataType;
|
this.dataType = dataType;
|
||||||
this.useDocValue = useDocValue;
|
this.useDocValue = useDocValue;
|
||||||
|
this.arrayLeniency = arrayLeniency;
|
||||||
this.hitName = hitName;
|
this.hitName = hitName;
|
||||||
|
|
||||||
if (hitName != null) {
|
if (hitName != null) {
|
||||||
|
@ -75,6 +79,7 @@ public class FieldHitExtractor implements HitExtractor {
|
||||||
dataType = esType != null ? DataType.fromTypeName(esType) : null;
|
dataType = esType != null ? DataType.fromTypeName(esType) : null;
|
||||||
useDocValue = in.readBoolean();
|
useDocValue = in.readBoolean();
|
||||||
hitName = in.readOptionalString();
|
hitName = in.readOptionalString();
|
||||||
|
arrayLeniency = in.readBoolean();
|
||||||
path = sourcePath(fieldName, useDocValue, hitName);
|
path = sourcePath(fieldName, useDocValue, hitName);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -89,6 +94,7 @@ public class FieldHitExtractor implements HitExtractor {
|
||||||
out.writeOptionalString(dataType == null ? null : dataType.typeName);
|
out.writeOptionalString(dataType == null ? null : dataType.typeName);
|
||||||
out.writeBoolean(useDocValue);
|
out.writeBoolean(useDocValue);
|
||||||
out.writeOptionalString(hitName);
|
out.writeOptionalString(hitName);
|
||||||
|
out.writeBoolean(arrayLeniency);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -117,7 +123,7 @@ public class FieldHitExtractor implements HitExtractor {
|
||||||
if (list.isEmpty()) {
|
if (list.isEmpty()) {
|
||||||
return null;
|
return null;
|
||||||
} else {
|
} else {
|
||||||
if (ARRAYS_LENIENCY || list.size() == 1) {
|
if (arrayLeniency || list.size() == 1) {
|
||||||
return unwrapMultiValue(list.get(0));
|
return unwrapMultiValue(list.get(0));
|
||||||
} else {
|
} else {
|
||||||
throw new SqlIllegalArgumentException("Arrays (returned by [{}]) are not supported", fieldName);
|
throw new SqlIllegalArgumentException("Arrays (returned by [{}]) are not supported", fieldName);
|
||||||
|
@ -222,11 +228,12 @@ public class FieldHitExtractor implements HitExtractor {
|
||||||
FieldHitExtractor other = (FieldHitExtractor) obj;
|
FieldHitExtractor other = (FieldHitExtractor) obj;
|
||||||
return fieldName.equals(other.fieldName)
|
return fieldName.equals(other.fieldName)
|
||||||
&& hitName.equals(other.hitName)
|
&& hitName.equals(other.hitName)
|
||||||
&& useDocValue == other.useDocValue;
|
&& useDocValue == other.useDocValue
|
||||||
|
&& arrayLeniency == other.arrayLeniency;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public int hashCode() {
|
public int hashCode() {
|
||||||
return Objects.hash(fieldName, useDocValue, hitName);
|
return Objects.hash(fieldName, useDocValue, hitName, arrayLeniency);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -47,7 +47,7 @@ public class TransportSqlClearCursorAction extends HandledTransportAction<SqlCle
|
||||||
Cursor cursor = Cursors.decodeFromString(request.getCursor());
|
Cursor cursor = Cursors.decodeFromString(request.getCursor());
|
||||||
planExecutor.cleanCursor(
|
planExecutor.cleanCursor(
|
||||||
new Configuration(DateUtils.UTC, Protocol.FETCH_SIZE, Protocol.REQUEST_TIMEOUT, Protocol.PAGE_TIMEOUT, null,
|
new Configuration(DateUtils.UTC, Protocol.FETCH_SIZE, Protocol.REQUEST_TIMEOUT, Protocol.PAGE_TIMEOUT, null,
|
||||||
request.mode(), StringUtils.EMPTY, StringUtils.EMPTY, StringUtils.EMPTY),
|
request.mode(), StringUtils.EMPTY, StringUtils.EMPTY, StringUtils.EMPTY, Protocol.FIELD_MULTI_VALUE_LENIENCY),
|
||||||
cursor, ActionListener.wrap(
|
cursor, ActionListener.wrap(
|
||||||
success -> listener.onResponse(new SqlClearCursorResponse(success)), listener::onFailure));
|
success -> listener.onResponse(new SqlClearCursorResponse(success)), listener::onFailure));
|
||||||
}
|
}
|
||||||
|
|
|
@ -72,7 +72,7 @@ public class TransportSqlQueryAction extends HandledTransportAction<SqlQueryRequ
|
||||||
// The configuration is always created however when dealing with the next page, only the timeouts are relevant
|
// The configuration is always created however when dealing with the next page, only the timeouts are relevant
|
||||||
// the rest having default values (since the query is already created)
|
// the rest having default values (since the query is already created)
|
||||||
Configuration cfg = new Configuration(request.zoneId(), request.fetchSize(), request.requestTimeout(), request.pageTimeout(),
|
Configuration cfg = new Configuration(request.zoneId(), request.fetchSize(), request.requestTimeout(), request.pageTimeout(),
|
||||||
request.filter(), request.mode(), request.clientId(), username, clusterName);
|
request.filter(), request.mode(), request.clientId(), username, clusterName, request.fieldMultiValueLeniency());
|
||||||
|
|
||||||
if (Strings.hasText(request.cursor()) == false) {
|
if (Strings.hasText(request.cursor()) == false) {
|
||||||
planExecutor.sql(cfg, request.query(), request.params(),
|
planExecutor.sql(cfg, request.query(), request.params(),
|
||||||
|
|
|
@ -21,6 +21,7 @@ import org.elasticsearch.xpack.sql.action.SqlTranslateAction;
|
||||||
import org.elasticsearch.xpack.sql.action.SqlTranslateRequest;
|
import org.elasticsearch.xpack.sql.action.SqlTranslateRequest;
|
||||||
import org.elasticsearch.xpack.sql.action.SqlTranslateResponse;
|
import org.elasticsearch.xpack.sql.action.SqlTranslateResponse;
|
||||||
import org.elasticsearch.xpack.sql.execution.PlanExecutor;
|
import org.elasticsearch.xpack.sql.execution.PlanExecutor;
|
||||||
|
import org.elasticsearch.xpack.sql.proto.Protocol;
|
||||||
import org.elasticsearch.xpack.sql.session.Configuration;
|
import org.elasticsearch.xpack.sql.session.Configuration;
|
||||||
|
|
||||||
import static org.elasticsearch.xpack.sql.plugin.Transports.clusterName;
|
import static org.elasticsearch.xpack.sql.plugin.Transports.clusterName;
|
||||||
|
@ -55,7 +56,7 @@ public class TransportSqlTranslateAction extends HandledTransportAction<SqlTrans
|
||||||
Configuration cfg = new Configuration(request.zoneId(), request.fetchSize(),
|
Configuration cfg = new Configuration(request.zoneId(), request.fetchSize(),
|
||||||
request.requestTimeout(), request.pageTimeout(), request.filter(),
|
request.requestTimeout(), request.pageTimeout(), request.filter(),
|
||||||
request.mode(), request.clientId(),
|
request.mode(), request.clientId(),
|
||||||
username(securityContext), clusterName(clusterService));
|
username(securityContext), clusterName(clusterService), Protocol.FIELD_MULTI_VALUE_LENIENCY);
|
||||||
|
|
||||||
planExecutor.searchSource(cfg, request.query(), request.params(), ActionListener.wrap(
|
planExecutor.searchSource(cfg, request.query(), request.params(), ActionListener.wrap(
|
||||||
searchSourceBuilder -> listener.onResponse(new SqlTranslateResponse(searchSourceBuilder)), listener::onFailure));
|
searchSourceBuilder -> listener.onResponse(new SqlTranslateResponse(searchSourceBuilder)), listener::onFailure));
|
||||||
|
|
|
@ -23,6 +23,7 @@ public class Configuration {
|
||||||
private final String clientId;
|
private final String clientId;
|
||||||
private final String username;
|
private final String username;
|
||||||
private final String clusterName;
|
private final String clusterName;
|
||||||
|
private final boolean multiValueFieldLeniency;
|
||||||
private final ZonedDateTime now;
|
private final ZonedDateTime now;
|
||||||
|
|
||||||
@Nullable
|
@Nullable
|
||||||
|
@ -30,7 +31,8 @@ public class Configuration {
|
||||||
|
|
||||||
public Configuration(ZoneId zi, int pageSize, TimeValue requestTimeout, TimeValue pageTimeout, QueryBuilder filter,
|
public Configuration(ZoneId zi, int pageSize, TimeValue requestTimeout, TimeValue pageTimeout, QueryBuilder filter,
|
||||||
Mode mode, String clientId,
|
Mode mode, String clientId,
|
||||||
String username, String clusterName) {
|
String username, String clusterName,
|
||||||
|
boolean multiValueFieldLeniency) {
|
||||||
this.zoneId = zi.normalized();
|
this.zoneId = zi.normalized();
|
||||||
this.pageSize = pageSize;
|
this.pageSize = pageSize;
|
||||||
this.requestTimeout = requestTimeout;
|
this.requestTimeout = requestTimeout;
|
||||||
|
@ -40,6 +42,7 @@ public class Configuration {
|
||||||
this.clientId = clientId;
|
this.clientId = clientId;
|
||||||
this.username = username;
|
this.username = username;
|
||||||
this.clusterName = clusterName;
|
this.clusterName = clusterName;
|
||||||
|
this.multiValueFieldLeniency = multiValueFieldLeniency;
|
||||||
this.now = ZonedDateTime.now(zoneId);
|
this.now = ZonedDateTime.now(zoneId);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -81,4 +84,8 @@ public class Configuration {
|
||||||
public ZonedDateTime now() {
|
public ZonedDateTime now() {
|
||||||
return now;
|
return now;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public boolean multiValueFieldLeniency() {
|
||||||
|
return multiValueFieldLeniency;
|
||||||
|
}
|
||||||
}
|
}
|
|
@ -30,7 +30,7 @@ public class TestUtils {
|
||||||
|
|
||||||
public static final Configuration TEST_CFG = new Configuration(DateUtils.UTC, Protocol.FETCH_SIZE,
|
public static final Configuration TEST_CFG = new Configuration(DateUtils.UTC, Protocol.FETCH_SIZE,
|
||||||
Protocol.REQUEST_TIMEOUT, Protocol.PAGE_TIMEOUT, null, Mode.PLAIN,
|
Protocol.REQUEST_TIMEOUT, Protocol.PAGE_TIMEOUT, null, Mode.PLAIN,
|
||||||
null, null, null);
|
null, null, null, false);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns the current UTC date-time with milliseconds precision.
|
* Returns the current UTC date-time with milliseconds precision.
|
||||||
|
@ -54,7 +54,8 @@ public class TestUtils {
|
||||||
randomFrom(Mode.values()),
|
randomFrom(Mode.values()),
|
||||||
randomAlphaOfLength(10),
|
randomAlphaOfLength(10),
|
||||||
randomAlphaOfLength(10),
|
randomAlphaOfLength(10),
|
||||||
randomAlphaOfLength(10));
|
randomAlphaOfLength(10),
|
||||||
|
false);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static Configuration randomConfiguration(ZoneId providedZoneId) {
|
public static Configuration randomConfiguration(ZoneId providedZoneId) {
|
||||||
|
@ -66,7 +67,8 @@ public class TestUtils {
|
||||||
randomFrom(Mode.values()),
|
randomFrom(Mode.values()),
|
||||||
randomAlphaOfLength(10),
|
randomAlphaOfLength(10),
|
||||||
randomAlphaOfLength(10),
|
randomAlphaOfLength(10),
|
||||||
randomAlphaOfLength(10));
|
randomAlphaOfLength(10),
|
||||||
|
false);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -36,7 +36,7 @@ public class FieldHitExtractorTests extends AbstractWireSerializingTestCase<Fiel
|
||||||
public static FieldHitExtractor randomFieldHitExtractor() {
|
public static FieldHitExtractor randomFieldHitExtractor() {
|
||||||
String hitName = randomAlphaOfLength(5);
|
String hitName = randomAlphaOfLength(5);
|
||||||
String name = randomAlphaOfLength(5) + "." + hitName;
|
String name = randomAlphaOfLength(5) + "." + hitName;
|
||||||
return new FieldHitExtractor(name, null, randomBoolean(), hitName);
|
return new FieldHitExtractor(name, null, randomBoolean(), hitName, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -51,7 +51,7 @@ public class FieldHitExtractorTests extends AbstractWireSerializingTestCase<Fiel
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected FieldHitExtractor mutateInstance(FieldHitExtractor instance) {
|
protected FieldHitExtractor mutateInstance(FieldHitExtractor instance) {
|
||||||
return new FieldHitExtractor(instance.fieldName() + "mutated", null, true, instance.hitName());
|
return new FieldHitExtractor(instance.fieldName() + "mutated", null, true, instance.hitName(), false);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void testGetDottedValueWithDocValues() {
|
public void testGetDottedValueWithDocValues() {
|
||||||
|
@ -174,7 +174,7 @@ public class FieldHitExtractorTests extends AbstractWireSerializingTestCase<Fiel
|
||||||
}
|
}
|
||||||
|
|
||||||
public void testToString() {
|
public void testToString() {
|
||||||
assertEquals("hit.field@hit", new FieldHitExtractor("hit.field", null, true, "hit").toString());
|
assertEquals("hit.field@hit", new FieldHitExtractor("hit.field", null, true, "hit", false).toString());
|
||||||
}
|
}
|
||||||
|
|
||||||
public void testMultiValuedDocValue() {
|
public void testMultiValuedDocValue() {
|
||||||
|
@ -240,6 +240,14 @@ public class FieldHitExtractorTests extends AbstractWireSerializingTestCase<Fiel
|
||||||
assertThat(ex.getMessage(), is("Arrays (returned by [a]) are not supported"));
|
assertThat(ex.getMessage(), is("Arrays (returned by [a]) are not supported"));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void testMultiValuedSourceAllowed() {
|
||||||
|
FieldHitExtractor fe = new FieldHitExtractor("a", null, false, true);
|
||||||
|
Object valueA = randomValue();
|
||||||
|
Object valueB = randomValue();
|
||||||
|
Map<String, Object> map = singletonMap("a", asList(valueA, valueB));
|
||||||
|
assertEquals(valueA, fe.extractFromSource(map));
|
||||||
|
}
|
||||||
|
|
||||||
public void testFieldWithDots() {
|
public void testFieldWithDots() {
|
||||||
FieldHitExtractor fe = new FieldHitExtractor("a.b", null, false);
|
FieldHitExtractor fe = new FieldHitExtractor("a.b", null, false);
|
||||||
Object value = randomValue();
|
Object value = randomValue();
|
||||||
|
|
|
@ -31,7 +31,7 @@ public class DatabaseFunctionTests extends ESTestCase {
|
||||||
new Configuration(DateUtils.UTC, Protocol.FETCH_SIZE, Protocol.REQUEST_TIMEOUT,
|
new Configuration(DateUtils.UTC, Protocol.FETCH_SIZE, Protocol.REQUEST_TIMEOUT,
|
||||||
Protocol.PAGE_TIMEOUT, null,
|
Protocol.PAGE_TIMEOUT, null,
|
||||||
randomFrom(Mode.values()), randomAlphaOfLength(10),
|
randomFrom(Mode.values()), randomAlphaOfLength(10),
|
||||||
null, clusterName),
|
null, clusterName, randomBoolean()),
|
||||||
new FunctionRegistry(),
|
new FunctionRegistry(),
|
||||||
IndexResolution.valid(test),
|
IndexResolution.valid(test),
|
||||||
new Verifier(new Metrics())
|
new Verifier(new Metrics())
|
||||||
|
|
|
@ -30,7 +30,7 @@ public class UserFunctionTests extends ESTestCase {
|
||||||
new Configuration(DateUtils.UTC, Protocol.FETCH_SIZE, Protocol.REQUEST_TIMEOUT,
|
new Configuration(DateUtils.UTC, Protocol.FETCH_SIZE, Protocol.REQUEST_TIMEOUT,
|
||||||
Protocol.PAGE_TIMEOUT, null,
|
Protocol.PAGE_TIMEOUT, null,
|
||||||
randomFrom(Mode.values()), randomAlphaOfLength(10),
|
randomFrom(Mode.values()), randomAlphaOfLength(10),
|
||||||
null, randomAlphaOfLengthBetween(1, 15)),
|
null, randomAlphaOfLengthBetween(1, 15), randomBoolean()),
|
||||||
new FunctionRegistry(),
|
new FunctionRegistry(),
|
||||||
IndexResolution.valid(test),
|
IndexResolution.valid(test),
|
||||||
new Verifier(new Metrics())
|
new Verifier(new Metrics())
|
||||||
|
|
Loading…
Reference in New Issue