diff --git a/TESTING.asciidoc b/TESTING.asciidoc index 3b11e2ce3106e..3229b7e768137 100644 --- a/TESTING.asciidoc +++ b/TESTING.asciidoc @@ -340,13 +340,13 @@ be downloaded again unless they have been updated to a new version. + . Run the tests with `./gradlew packagingTest`. This will cause Gradle to build the tar, zip, and deb packages and all the plugins. It will then run the tests -on ubuntu-1404 and centos-7. We chose those two distributions as the default +on ubuntu-1604 and centos-7. We chose those two distributions as the default because they cover deb and rpm packaging and SyvVinit and systemd. You can choose which boxes to test by setting the `-Pvagrant.boxes` project property. All of the valid options for this property are: -* `sample` - The default, only chooses ubuntu-1404 and centos-7 +* `sample` - The default, only chooses ubuntu-1604 and centos-7 * List of box names, comma separated (e.g. `oel-7,fedora-28`) - Chooses exactly the boxes listed. * `linux-all` - All linux boxes. * `windows-all` - All Windows boxes. If there are any Windows boxes which do not @@ -364,11 +364,10 @@ will remain running and you'll have to stop them manually with `./gradlew stop` All the regular vagrant commands should just work so you can get a shell in a VM running trusty by running -`vagrant up ubuntu-1404 --provider virtualbox && vagrant ssh ubuntu-1404`. +`vagrant up ubuntu-1604 --provider virtualbox && vagrant ssh ubuntu-1604`. These are the linux flavors supported, all of which we provide images for -* ubuntu-1404 aka trusty * ubuntu-1604 aka xenial * ubuntu-1804 aka bionic beaver * debian-8 aka jessie @@ -422,13 +421,13 @@ It's important to think of VMs like cattle. If they become lame you just shoot them and let vagrant reprovision them. Say you've hosed your precise VM: ---------------------------------------------------- -vagrant ssh ubuntu-1404 -c 'sudo rm -rf /bin'; echo oops +vagrant ssh ubuntu-1604 -c 'sudo rm -rf /bin'; echo oops ---------------------------------------------------- All you've got to do to get another one is ---------------------------------------------- -vagrant destroy -f ubuntu-1404 && vagrant up ubuntu-1404 --provider virtualbox +vagrant destroy -f ubuntu-1604 && vagrant up ubuntu-1604 --provider virtualbox ---------------------------------------------- The whole process takes a minute and a half on a modern laptop, two and a half diff --git a/Vagrantfile b/Vagrantfile index 4624172b02ada..4fbf1beeb04e9 100644 --- a/Vagrantfile +++ b/Vagrantfile @@ -46,12 +46,6 @@ Vagrant.configure(2) do |config| PROJECT_DIR = ENV['VAGRANT_PROJECT_DIR'] || Dir.pwd config.vm.synced_folder PROJECT_DIR, '/project' - 'ubuntu-1404'.tap do |box| - config.vm.define box, define_opts do |config| - config.vm.box = 'elastic/ubuntu-14.04-x86_64' - deb_common config, box - end - end 'ubuntu-1604'.tap do |box| config.vm.define box, define_opts do |config| config.vm.box = 'elastic/ubuntu-16.04-x86_64' diff --git a/buildSrc/src/main/groovy/org/elasticsearch/gradle/vagrant/VagrantTestPlugin.groovy b/buildSrc/src/main/groovy/org/elasticsearch/gradle/vagrant/VagrantTestPlugin.groovy index 4bdef1ff6fd30..763b5509772af 100644 --- a/buildSrc/src/main/groovy/org/elasticsearch/gradle/vagrant/VagrantTestPlugin.groovy +++ b/buildSrc/src/main/groovy/org/elasticsearch/gradle/vagrant/VagrantTestPlugin.groovy @@ -31,7 +31,6 @@ class VagrantTestPlugin implements Plugin { 'oel-7', 'opensuse-42', 'sles-12', - 'ubuntu-1404', 'ubuntu-1604', 'ubuntu-1804' ]) @@ -48,7 +47,7 @@ class VagrantTestPlugin implements Plugin { /** Boxes used when sampling the tests **/ static final List SAMPLE = unmodifiableList([ 'centos-7', - 'ubuntu-1404' + 'ubuntu-1604' ]) /** All distributions to bring into test VM, whether or not they are used **/ diff --git a/client/rest-high-level/src/main/java/org/elasticsearch/client/dataframe/transforms/DataFrameTransformState.java b/client/rest-high-level/src/main/java/org/elasticsearch/client/dataframe/transforms/DataFrameTransformState.java index fd191bb600ca6..248ee9a18f53f 100644 --- a/client/rest-high-level/src/main/java/org/elasticsearch/client/dataframe/transforms/DataFrameTransformState.java +++ b/client/rest-high-level/src/main/java/org/elasticsearch/client/dataframe/transforms/DataFrameTransformState.java @@ -43,14 +43,16 @@ public class DataFrameTransformState { private static final ParseField TASK_STATE = new ParseField("task_state"); private static final ParseField CURRENT_POSITION = new ParseField("current_position"); private static final ParseField GENERATION = new ParseField("generation"); + private static final ParseField REASON = new ParseField("reason"); @SuppressWarnings("unchecked") public static final ConstructingObjectParser PARSER = - new ConstructingObjectParser<>("data_frame_transform_state", + new ConstructingObjectParser<>("data_frame_transform_state", true, args -> new DataFrameTransformState((DataFrameTransformTaskState) args[0], (IndexerState) args[1], (HashMap) args[2], - (long) args[3])); + (long) args[3], + (String) args[4])); static { PARSER.declareField(constructorArg(), @@ -68,6 +70,7 @@ public class DataFrameTransformState { throw new IllegalArgumentException("Unsupported token [" + p.currentToken() + "]"); }, CURRENT_POSITION, ObjectParser.ValueType.VALUE_OBJECT_ARRAY); PARSER.declareLong(ConstructingObjectParser.optionalConstructorArg(), GENERATION); + PARSER.declareString(ConstructingObjectParser.optionalConstructorArg(), REASON); } public static DataFrameTransformState fromXContent(XContentParser parser) throws IOException { @@ -78,15 +81,18 @@ public static DataFrameTransformState fromXContent(XContentParser parser) throws private final IndexerState indexerState; private final long generation; private final SortedMap currentPosition; + private final String reason; public DataFrameTransformState(DataFrameTransformTaskState taskState, IndexerState indexerState, @Nullable Map position, - long generation) { + long generation, + @Nullable String reason) { this.taskState = taskState; this.indexerState = indexerState; this.currentPosition = position == null ? null : Collections.unmodifiableSortedMap(new TreeMap<>(position)); this.generation = generation; + this.reason = reason; } public IndexerState getIndexerState() { @@ -106,6 +112,11 @@ public long getGeneration() { return generation; } + @Nullable + public String getReason() { + return reason; + } + @Override public boolean equals(Object other) { if (this == other) { @@ -121,11 +132,13 @@ public boolean equals(Object other) { return Objects.equals(this.taskState, that.taskState) && Objects.equals(this.indexerState, that.indexerState) && Objects.equals(this.currentPosition, that.currentPosition) && - this.generation == that.generation; + this.generation == that.generation && + Objects.equals(this.reason, that.reason); } @Override public int hashCode() { - return Objects.hash(taskState, indexerState, currentPosition, generation); + return Objects.hash(taskState, indexerState, currentPosition, generation, reason); } + } diff --git a/client/rest-high-level/src/test/java/org/elasticsearch/client/DataFrameTransformIT.java b/client/rest-high-level/src/test/java/org/elasticsearch/client/DataFrameTransformIT.java index e8724cc071dae..3e564a86207ba 100644 --- a/client/rest-high-level/src/test/java/org/elasticsearch/client/DataFrameTransformIT.java +++ b/client/rest-high-level/src/test/java/org/elasticsearch/client/DataFrameTransformIT.java @@ -358,6 +358,7 @@ public void testGetStats() throws Exception { DataFrameTransformStateAndStats stateAndStats = response.getTransformsStateAndStats().get(0); assertEquals(IndexerState.STARTED, stateAndStats.getTransformState().getIndexerState()); assertEquals(DataFrameTransformTaskState.STARTED, stateAndStats.getTransformState().getTaskState()); + assertEquals(null, stateAndStats.getTransformState().getReason()); assertNotEquals(zeroIndexerStats, stateAndStats.getTransformStats()); }); } diff --git a/client/rest-high-level/src/test/java/org/elasticsearch/client/dataframe/transforms/DataFrameTransformStateTests.java b/client/rest-high-level/src/test/java/org/elasticsearch/client/dataframe/transforms/DataFrameTransformStateTests.java index 17dc388948167..fa1ac4202f9dd 100644 --- a/client/rest-high-level/src/test/java/org/elasticsearch/client/dataframe/transforms/DataFrameTransformStateTests.java +++ b/client/rest-high-level/src/test/java/org/elasticsearch/client/dataframe/transforms/DataFrameTransformStateTests.java @@ -36,7 +36,8 @@ public void testFromXContent() throws IOException { DataFrameTransformStateTests::randomDataFrameTransformState, DataFrameTransformStateTests::toXContent, DataFrameTransformState::fromXContent) - .supportsUnknownFields(false) + .supportsUnknownFields(true) + .randomFieldsExcludeFilter(field -> field.equals("current_position")) .test(); } @@ -44,7 +45,8 @@ public static DataFrameTransformState randomDataFrameTransformState() { return new DataFrameTransformState(randomFrom(DataFrameTransformTaskState.values()), randomFrom(IndexerState.values()), randomPositionMap(), - randomLongBetween(0,10)); + randomLongBetween(0,10), + randomBoolean() ? null : randomAlphaOfLength(10)); } public static void toXContent(DataFrameTransformState state, XContentBuilder builder) throws IOException { @@ -55,6 +57,9 @@ public static void toXContent(DataFrameTransformState state, XContentBuilder bui builder.field("current_position", state.getPosition()); } builder.field("generation", state.getGeneration()); + if (state.getReason() != null) { + builder.field("reason", state.getReason()); + } builder.endObject(); } diff --git a/distribution/src/bin/elasticsearch-env.bat b/distribution/src/bin/elasticsearch-env.bat index bd34880e40ece..f1cdc2fd22457 100644 --- a/distribution/src/bin/elasticsearch-env.bat +++ b/distribution/src/bin/elasticsearch-env.bat @@ -65,5 +65,5 @@ rem check the Java version %JAVA% -cp "%ES_CLASSPATH%" "org.elasticsearch.tools.java_version_checker.JavaVersionChecker" || exit /b 1 if not defined ES_TMPDIR ( - for /f "tokens=* usebackq" %%a in (`"%JAVA% -cp "!ES_CLASSPATH!" "org.elasticsearch.tools.launchers.TempDirectory""`) do set ES_TMPDIR=%%a + for /f "tokens=* usebackq" %%a in (`CALL %JAVA% -cp "!ES_CLASSPATH!" "org.elasticsearch.tools.launchers.TempDirectory"`) do set ES_TMPDIR=%%a ) diff --git a/distribution/src/bin/elasticsearch.bat b/distribution/src/bin/elasticsearch.bat index 7df6f19fc0765..ecbbad826e797 100644 --- a/distribution/src/bin/elasticsearch.bat +++ b/distribution/src/bin/elasticsearch.bat @@ -41,9 +41,9 @@ IF ERRORLEVEL 1 ( EXIT /B %ERRORLEVEL% ) -set "ES_JVM_OPTIONS=%ES_PATH_CONF%\jvm.options" +set ES_JVM_OPTIONS=%ES_PATH_CONF%\jvm.options @setlocal -for /F "usebackq delims=" %%a in (`"%JAVA% -cp "!ES_CLASSPATH!" "org.elasticsearch.tools.launchers.JvmOptionsParser" "!ES_JVM_OPTIONS!" || echo jvm_options_parser_failed"`) do set JVM_OPTIONS=%%a +for /F "usebackq delims=" %%a in (`CALL %JAVA% -cp "!ES_CLASSPATH!" "org.elasticsearch.tools.launchers.JvmOptionsParser" "!ES_JVM_OPTIONS!" ^|^| echo jvm_options_parser_failed`) do set JVM_OPTIONS=%%a @endlocal & set "MAYBE_JVM_OPTIONS_PARSER_FAILED=%JVM_OPTIONS%" & set ES_JAVA_OPTS=%JVM_OPTIONS:${ES_TMPDIR}=!ES_TMPDIR!% %ES_JAVA_OPTS% if "%MAYBE_JVM_OPTIONS_PARSER_FAILED%" == "jvm_options_parser_failed" ( diff --git a/docs/java-api/docs/update.asciidoc b/docs/java-api/docs/update.asciidoc index 1c2211be9ba13..0935c9f11eca4 100644 --- a/docs/java-api/docs/update.asciidoc +++ b/docs/java-api/docs/update.asciidoc @@ -22,7 +22,9 @@ Or you can use `prepareUpdate()` method: [source,java] -------------------------------------------------- client.prepareUpdate("ttl", "doc", "1") - .setScript(new Script("ctx._source.gender = \"male\"" <1> , ScriptService.ScriptType.INLINE, null, null)) + .setScript(new Script( + "ctx._source.gender = \"male\"", <1> + ScriptService.ScriptType.INLINE, null, null)) .get(); client.prepareUpdate("ttl", "doc", "1") diff --git a/docs/java-api/index.asciidoc b/docs/java-api/index.asciidoc index e5eb2a6b02062..d85ad8350e9cc 100644 --- a/docs/java-api/index.asciidoc +++ b/docs/java-api/index.asciidoc @@ -1,8 +1,8 @@ -[[java-api]] = Java API include::../Versions.asciidoc[] +[[java-api]] [preface] == Preface diff --git a/docs/painless/painless-casting.asciidoc b/docs/painless/painless-casting.asciidoc index 4bcd14cbfc6a1..b742fbd115591 100644 --- a/docs/painless/painless-casting.asciidoc +++ b/docs/painless/painless-casting.asciidoc @@ -338,6 +338,28 @@ Use the cast operator to convert a <> value into a explicit cast `String "s"` to `char s` -> `char s`; store `char s` to `c` +[[character-string-casting]] +==== Character to String Casting + +Use the cast operator to convert a <> value into a +<> value. + +*Examples* + +* Casting a `String` reference into a `char` type value. ++ +[source,Painless] +---- +<1> char c = 65; +<2> String s = (String)c; +---- +<1> declare `char c`; + store `char 65` to `c`; +<2> declare `String s` + load from `c` -> `char A`; + explicit cast `char A` to `String "A"` -> `String "A"`; + store `String "A"` to `s` + [[boxing-unboxing]] ==== Boxing and Unboxing @@ -464,61 +486,51 @@ based on the type the `def` value represents. The following tables show all allowed casts. Read the tables row by row, where the original type is shown in the first column, and each subsequent column -indicates whether a cast to the specified target type is implicit (I), explicit -(E), or is not allowed (-). +indicates whether a cast to the specified target type is implicit (I), +explicit (E), boxed/unboxed for methods only (A), a reference type cast (@), +or is not allowed (-). See <> +for allowed reference type casts. *Primitive/Reference Types* -[cols="<3,^1,^1,^1,^1,^1,^1,^1,^1,^1,^1,^1,^1,^1,^1,^1,^1,^1,^1,^1"] +[cols="<3,^1,^1,^1,^1,^1,^1,^1,^1,^1,^1,^1,^1,^1,^1,^1,^1,^1,^1,^1,^1,^1"] |==== -| | o | b | s | c | i | j | f | d | O | B | S | C | I | L | F | D | T | R | def -| boolean ( o ) | | - | - | - | - | - | - | - | - | - | - | - | - | - | - | - | - | - | I -| byte ( b ) | - | | I | I | I | I | I | I | - | - | - | - | - | - | - | - | - | - | I -| short ( s ) | - | E | | E | I | I | I | I | - | - | - | - | - | - | - | - | - | - | I -| char ( c ) | - | E | E | | I | I | I | I | - | - | - | - | - | - | - | - | E | - | I -| int ( i ) | - | E | E | E | | I | I | I | - | - | - | - | - | - | - | - | - | - | I -| long ( j ) | - | E | E | E | E | | I | I | - | - | - | - | - | - | - | - | - | - | I -| float ( f ) | - | E | E | E | E | E | | I | - | - | - | - | - | - | - | - | - | - | I -| double ( d ) | - | E | E | E | E | E | E | | - | - | - | - | - | - | - | - | - | - | I -| Boolean ( O ) | - | - | - | - | - | - | - | - | - | - | - | | - | - | - | - | - | - | I -| Byte ( B ) | - | - | - | - | - | - | - | - | - | | - | - | - | - | - | - | - | - | I -| Short ( S ) | - | - | - | - | - | - | - | - | - | - | | - | - | - | - | - | - | - | I -| Character ( C ) | - | - | - | - | - | - | - | - | - | - | - | | - | - | - | - | - | - | I -| Integer ( I ) | - | - | - | - | - | - | - | - | - | - | - | - | | - | - | - | - | - | I -| Long ( L ) | - | - | - | - | - | - | - | - | - | - | - | - | - | | - | - | - | - | I -| Float ( F ) | - | - | - | - | - | - | - | - | - | - | - | - | - | - | | - | - | - | I -| Double ( D ) | - | - | - | - | - | - | - | - | - | - | - | - | - | - | - | | - | - | I -| String ( T ) | - | - | - | E | - | - | - | - | - | - | - | - | - | - | - | - | | - | I -| Reference ( R ) | - | - | - | - | - | - | - | - | - | - | - | - | - | - | - | - | - | @ | I +| | O | N | T | b | y | s | c | i | j | f | d | B | Y | S | C | I | J | F | D | R | def +| Object ( O ) | | @ | @ | - | - | - | - | - | - | - | - | @ | @ | @ | @ | @ | @ | @ | @ | @ | I +| Number ( N ) | I | | - | - | - | - | - | - | - | - | - | - | @ | @ | - | @ | @ | @ | @ | @ | I +| String ( T ) | I | - | | - | - | - | - | - | - | - | - | - | - | - | E | - | - | - | - | - | I +| boolean ( b ) | A | - | - | | - | - | - | - | - | - | - | A | - | - | - | - | - | - | - | - | I +| byte ( y ) | A | A | - | - | | I | E | I | I | I | I | - | A | A | - | A | A | A | A | - | I +| short ( s ) | A | A | - | - | E | | E | I | I | I | I | - | - | A | - | A | A | A | A | - | I +| char ( c ) | A | - | E | - | E | E | | I | I | I | I | - | - | - | A | A | A | A | A | - | I +| int ( i ) | A | A | - | - | E | E | E | | I | I | I | - | - | - | - | A | A | A | A | - | I +| long ( j ) | A | A | - | - | E | E | E | E | | I | I | - | - | - | - | - | A | A | A | - | I +| float ( f ) | A | A | - | - | E | E | E | E | E | | I | - | - | - | - | - | - | A | A | - | I +| double ( d ) | A | A | - | - | E | E | E | E | E | E | | - | - | - | - | - | - | - | A | - | I +| Boolean ( B ) | A | - | - | A | - | - | - | - | - | - | - | | - | - | - | - | - | - | - | @ | I +| Byte ( Y ) | A | I | - | - | A | A | - | A | A | A | A | - | | A | - | A | A | A | A | @ | I +| Short ( S ) | A | I | - | - | - | A | - | A | A | A | A | - | - | | - | A | A | A | A | @ | I +| Character ( C ) | A | - | - | - | - | - | A | A | A | A | A | - | - | - | | A | A | A | A | @ | I +| Integer ( I ) | A | - | - | - | - | - | - | A | A | A | A | - | - | - | - | | A | A | A | @ | I +| Long ( J ) | A | - | - | - | - | - | - | - | A | A | A | - | - | - | - | - | | A | A | @ | I +| Float ( F ) | A | - | - | - | - | - | - | - | - | A | A | - | - | - | - | - | - | | A | @ | I +| Double ( D ) | A | - | - | - | - | - | - | - | - | - | A | - | - | - | - | - | - | - | | @ | I +| Reference ( R ) | I | @ | @ | - | - | - | - | - | - | - | - | @ | @ | @ | @ | @ | @ | @ | @ | @ | I |==== -@ See <> for allowed reference - type casts. - *`def` Type* -[cols="<3,^1,^1,^1,^1,^1,^1,^1,^1,^1,^1,^1,^1,^1,^1,^1,^1,^1,^1,^1"] +[cols="<3,^1,^1,^1,^1,^1,^1,^1,^1,^1,^1,^1,^1,^1,^1,^1,^1,^1,^1,^1,^1"] |==== -| | o | b | s | c | i | j | f | d | O | B | S | C | I | L | F | D | T | R | def -| def as boolean | I | - | - | - | - | - | - | - | I | - | - | - | - | - | - | - | - | - | -| def as byte | - | I | I | I | I | I | I | I | - | I | I | I | I | I | I | I | - | - | -| def as short | - | E | I | E | I | I | I | I | - | E | I | E | I | I | I | I | - | - | -| def as char | - | E | E | I | I | I | I | I | - | E | E | I | I | I | I | I | E | - | -| def as int | - | E | E | E | I | I | I | I | - | E | E | E | I | I | I | I | - | - | -| def as long | - | E | E | E | E | I | I | I | - | E | E | E | E | I | I | I | - | - | -| def as float | - | E | E | E | E | E | I | I | - | E | E | E | E | E | I | I | - | - | -| def as double | - | E | E | E | E | E | E | I | - | E | E | E | E | E | E | I | - | - | -| def as Boolean | I | - | - | - | - | - | - | - | I | - | - | - | | - | - | - | - | - | -| def as Byte | - | I | I | I | I | I | I | I | - | I | I | I | I | I | I | I | - | - | -| def as Short | - | E | I | E | I | I | I | I | - | E | I | E | I | I | I | I | - | - | -| def as Character | - | E | E | I | I | I | I | I | - | E | E | I | I | I | I | I | - | - | -| def as Integer | - | E | E | E | I | I | I | I | - | E | E | E | I | I | I | I | - | - | -| def as Long | - | E | E | E | E | I | I | I | - | E | E | E | E | I | I | I | - | - | -| def as Float | - | E | E | E | E | E | I | I | - | E | E | E | E | E | I | I | - | - | -| def as Double | - | E | E | E | E | E | E | I | - | E | E | E | E | E | E | I | - | - | -| def as String | - | - | - | E | - | - | - | - | - | - | - | - | - | - | - | - | I | - | -| def as Reference | - | - | - | - | - | - | - | - | - | - | - | - | - | - | - | - | - | @ | +| | O | N | T | b | y | s | c | i | j | f | d | B | Y | S | C | I | J | F | D | R +| def as String | I | - | I | - | - | - | E | - | - | - | - | - | - | - | E | - | - | - | - | @ +| def as boolean/Boolean | I | - | - | I | - | - | - | - | - | - | - | I | - | - | - | - | - | - | - | @ +| def as byte/Byte | I | - | - | - | I | I | E | I | I | I | I | - | I | I | E | I | I | I | I | @ +| def as short/Short | I | - | - | - | E | I | E | I | I | I | I | - | E | I | E | I | I | I | I | @ +| def as char/Character | I | - | - | - | E | E | I | I | I | I | I | - | E | E | I | I | I | I | I | @ +| def as int/Integer | I | - | - | - | E | E | E | I | I | I | I | - | E | E | E | I | I | I | I | @ +| def as long/Long | I | - | - | - | E | E | E | E | I | I | I | - | E | E | E | E | I | I | I | @ +| def as float/Float | I | - | - | - | E | E | E | E | E | I | I | - | E | E | E | E | E | I | I | @ +| def as double/Double | I | - | - | - | E | E | E | E | E | E | I | - | E | E | E | E | E | E | I | @ +| def as Reference | @ | @ | @ | - | - | - | - | - | - | - | - | @ | @ | @ | @ | @ | @ | @ | @ | @ |==== - -@ See <> for allowed reference - type casts. diff --git a/docs/reference/analysis/tokenfilters/synonym-graph-tokenfilter.asciidoc b/docs/reference/analysis/tokenfilters/synonym-graph-tokenfilter.asciidoc index fa228abd74ac0..785eb77f2c65e 100644 --- a/docs/reference/analysis/tokenfilters/synonym-graph-tokenfilter.asciidoc +++ b/docs/reference/analysis/tokenfilters/synonym-graph-tokenfilter.asciidoc @@ -69,10 +69,10 @@ PUT /test_index } }, "filter" : { - "my_stop": { - "type" : "stop", - "stopwords": ["bar"] - }, + "my_stop": { + "type" : "stop", + "stopwords": ["bar"] + }, "synonym_graph" : { "type" : "synonym_graph", "lenient": true, diff --git a/docs/reference/analysis/tokenfilters/synonym-tokenfilter.asciidoc b/docs/reference/analysis/tokenfilters/synonym-tokenfilter.asciidoc index 715abdde6331d..87c99f6f38683 100644 --- a/docs/reference/analysis/tokenfilters/synonym-tokenfilter.asciidoc +++ b/docs/reference/analysis/tokenfilters/synonym-tokenfilter.asciidoc @@ -58,10 +58,10 @@ PUT /test_index } }, "filter" : { - "my_stop": { - "type" : "stop", - "stopwords": ["bar"] - }, + "my_stop": { + "type" : "stop", + "stopwords": ["bar"] + }, "synonym" : { "type" : "synonym", "lenient": true, diff --git a/docs/reference/how-to/indexing-speed.asciidoc b/docs/reference/how-to/indexing-speed.asciidoc index 164c853271711..1c8b989779d6c 100644 --- a/docs/reference/how-to/indexing-speed.asciidoc +++ b/docs/reference/how-to/indexing-speed.asciidoc @@ -36,12 +36,24 @@ number of workers is. This can be tested by progressively increasing the number of workers until either I/O or CPU is saturated on the cluster. [float] -=== Increase the refresh interval - -The default <> is `1s`, which -forces Elasticsearch to create a new segment every second. -Increasing this value (to say, `30s`) will allow larger segments to flush and -decreases future merge pressure. +=== Unset or increase the refresh interval + +The operation that consists of making changes visible to search - called a +<> - is costly, and calling it often while there is +ongoing indexing activity can hurt indexing speed. + +By default, Elasticsearch runs this operation every second, but only on +indices that have received one search request or more in the last 30 seconds. +This is the optimal configuration if you have no or very little search traffic +(e.g. less than one search request every 5 minutes) and want to optimize for +indexing speed. + +On the other hand, if your index experiences regular search requests, this +default behavior means that Elasticsearch will refresh your index every 1 +second. If you can afford to increase the amount of time between when a document +gets indexed and when it becomes visible, increasing the +<> to a larger value, e.g. +`30s`, might help improve indexing speed. [float] === Disable refresh and replicas for initial loads diff --git a/docs/reference/query-dsl/multi-match-query.asciidoc b/docs/reference/query-dsl/multi-match-query.asciidoc index b8fbb61a950d0..703abd42ee621 100644 --- a/docs/reference/query-dsl/multi-match-query.asciidoc +++ b/docs/reference/query-dsl/multi-match-query.asciidoc @@ -85,11 +85,11 @@ parameter, which can be set to: were one big field. Looks for each word in *any* field. See <>. -`phrase`:: Runs a `match_phrase` query on each field and uses the `_score` +`phrase`:: Runs a `match_phrase` query on each field and uses the `_score` from the best field. See <>. -`phrase_prefix`:: Runs a `match_phrase_prefix` query on each field and - combines the `_score` from each field. See <>. +`phrase_prefix`:: Runs a `match_phrase_prefix` query on each field and uses + the `_score` from the best field. See <>. `bool_prefix`:: Creates a `match_bool_prefix` query on each field and combines the `_score` from each field. See diff --git a/docs/reference/sql/functions/grouping.asciidoc b/docs/reference/sql/functions/grouping.asciidoc index d3a57f0d3a81a..742f072dbd039 100644 --- a/docs/reference/sql/functions/grouping.asciidoc +++ b/docs/reference/sql/functions/grouping.asciidoc @@ -80,3 +80,7 @@ When the histogram in SQL is applied on **DATE** type instead of **DATETIME**, t the multiple of a day. E.g.: for `HISTOGRAM(CAST(birth_date AS DATE), INTERVAL '2 3:04' DAY TO MINUTE)` the interval actually used will be `INTERVAL '2' DAY`. If the interval specified is less than 1 day, e.g.: `HISTOGRAM(CAST(birth_date AS DATE), INTERVAL '20' HOUR)` then the interval used will be `INTERVAL '1' DAY`. + +[IMPORTANT] +Histogram in SQL cannot be applied applied on **TIME** type. +E.g.: `HISTOGRAM(CAST(birth_date AS TIME), INTERVAL '10' MINUTES)` is currently not supported. diff --git a/docs/reference/sql/language/data-types.asciidoc b/docs/reference/sql/language/data-types.asciidoc index 42e5c842a4187..6c2304993c9d2 100644 --- a/docs/reference/sql/language/data-types.asciidoc +++ b/docs/reference/sql/language/data-types.asciidoc @@ -44,9 +44,10 @@ s|SQL precision Most of {es} <> are available in {es-sql}, as indicated above. As one can see, all of {es} <> are mapped to the data type with the same name in {es-sql}, with the exception of **date** data type which is mapped to **datetime** in {es-sql}. -This is to avoid confusion with the ANSI SQL **DATE** (date only) type, which is also supported by {es-sql} -in queries (with the use of <>/<>), -but doesn't correspond to an actual mapping in {es} (see the <> below). +This is to avoid confusion with the ANSI SQL types **DATE** (date only) and **TIME** (time only), which are also +supported by {es-sql} in queries (with the use of +<>/<>), but don't correspond to an +actual mapping in {es} (see the <> below). Obviously, not all types in {es} have an equivalent in SQL and vice-versa hence why, {es-sql} uses the data type _particularities_ of the former over the latter as ultimately {es} is the backing store. @@ -66,6 +67,7 @@ s|SQL precision | date | 24 +| time | 18 | interval_year | 7 | interval_month | 7 | interval_day | 23 diff --git a/docs/reference/sql/limitations.asciidoc b/docs/reference/sql/limitations.asciidoc index e8c99901e27c1..e2a538cd08571 100644 --- a/docs/reference/sql/limitations.asciidoc +++ b/docs/reference/sql/limitations.asciidoc @@ -113,3 +113,28 @@ FROM (SELECT ...) WHERE [simple_condition]`, this is currently **un-supported**. Using `FIRST` and `LAST` in the `HAVING` clause is not supported. The same applies to <> and <> when their target column is of type <> as they are internally translated to `FIRST` and `LAST`. + +[float] +=== Using TIME data type in GROUP BY or <> + +Using `TIME` data type as a grouping key is currently not supported. For example: + +[source, sql] +------------------------------------------------------------- +SELECT count(*) FROM test GROUP BY CAST(date_created AS TIME); +------------------------------------------------------------- + +On the other hand, it can still be used if it's wrapped with a scalar function that returns another data type, +for example: + +[source, sql] +------------------------------------------------------------- +SELECT count(*) FROM test GROUP BY MINUTE((CAST(date_created AS TIME)); +------------------------------------------------------------- + +`TIME` data type is also currently not supported in histogram grouping function. For example: + +[source, sql] +------------------------------------------------------------- +SELECT HISTOGRAM(CAST(birth_date AS TIME), INTERVAL '10' MINUTES) as h, COUNT(*) FROM t GROUP BY h +------------------------------------------------------------- diff --git a/modules/reindex/src/main/java/org/elasticsearch/index/reindex/remote/RemoteRequestBuilders.java b/modules/reindex/src/main/java/org/elasticsearch/index/reindex/remote/RemoteRequestBuilders.java index d20be74798066..2423de5fd704a 100644 --- a/modules/reindex/src/main/java/org/elasticsearch/index/reindex/remote/RemoteRequestBuilders.java +++ b/modules/reindex/src/main/java/org/elasticsearch/index/reindex/remote/RemoteRequestBuilders.java @@ -38,6 +38,10 @@ import org.elasticsearch.search.sort.SortBuilder; import java.io.IOException; +import java.io.UnsupportedEncodingException; +import java.net.URLEncoder; +import java.util.Arrays; +import java.util.stream.Collectors; import static org.elasticsearch.common.unit.TimeValue.timeValueMillis; @@ -54,8 +58,8 @@ private RemoteRequestBuilders() {} static Request initialSearch(SearchRequest searchRequest, BytesReference query, Version remoteVersion) { // It is nasty to build paths with StringBuilder but we'll be careful.... StringBuilder path = new StringBuilder("/"); - addIndexesOrTypes(path, "Index", searchRequest.indices()); - addIndexesOrTypes(path, "Type", searchRequest.types()); + addIndices(path, searchRequest.indices()); + addTypes(path, searchRequest.types()); path.append("_search"); Request request = new Request("POST", path.toString()); @@ -158,14 +162,34 @@ static Request initialSearch(SearchRequest searchRequest, BytesReference query, return request; } - private static void addIndexesOrTypes(StringBuilder path, String name, String[] indicesOrTypes) { - if (indicesOrTypes == null || indicesOrTypes.length == 0) { + private static void addIndices(StringBuilder path, String[] indices) { + if (indices == null || indices.length == 0) { return; } - for (String indexOrType : indicesOrTypes) { - checkIndexOrType(name, indexOrType); + + path.append(Arrays.stream(indices).map(RemoteRequestBuilders::encodeIndex).collect(Collectors.joining(","))).append('/'); + } + + private static String encodeIndex(String s) { + if (s.contains("%")) { // already encoded, pass-through to allow this in mixed version clusters + checkIndexOrType("Index", s); + return s; + } + try { + return URLEncoder.encode(s, "utf-8"); + } catch (UnsupportedEncodingException e) { + throw new RuntimeException(e); + } + } + + private static void addTypes(StringBuilder path, String[] types) { + if (types == null || types.length == 0) { + return; + } + for (String indexOrType : types) { + checkIndexOrType("Type", indexOrType); } - path.append(Strings.arrayToCommaDelimitedString(indicesOrTypes)).append('/'); + path.append(Strings.arrayToCommaDelimitedString(types)).append('/'); } private static void checkIndexOrType(String name, String indexOrType) { diff --git a/modules/reindex/src/test/java/org/elasticsearch/index/reindex/remote/RemoteRequestBuildersTests.java b/modules/reindex/src/test/java/org/elasticsearch/index/reindex/remote/RemoteRequestBuildersTests.java index 0f985fd37016a..eb6192a043160 100644 --- a/modules/reindex/src/test/java/org/elasticsearch/index/reindex/remote/RemoteRequestBuildersTests.java +++ b/modules/reindex/src/test/java/org/elasticsearch/index/reindex/remote/RemoteRequestBuildersTests.java @@ -68,19 +68,26 @@ public void testIntialSearchPath() { searchRequest.indices("a", "b"); searchRequest.types("c", "d"); assertEquals("/a,b/c,d/_search", initialSearch(searchRequest, query, remoteVersion).getEndpoint()); - searchRequest.indices("cat,"); - expectBadStartRequest(searchRequest, "Index", ",", "cat,"); - searchRequest.indices("cat,", "dog"); - expectBadStartRequest(searchRequest, "Index", ",", "cat,"); - searchRequest.indices("dog", "cat,"); - expectBadStartRequest(searchRequest, "Index", ",", "cat,"); + assertEquals("/cat%2C/c,d/_search", initialSearch(searchRequest, query, remoteVersion).getEndpoint()); searchRequest.indices("cat/"); - expectBadStartRequest(searchRequest, "Index", "/", "cat/"); + assertEquals("/cat%2F/c,d/_search", initialSearch(searchRequest, query, remoteVersion).getEndpoint()); searchRequest.indices("cat/", "dog"); - expectBadStartRequest(searchRequest, "Index", "/", "cat/"); - searchRequest.indices("dog", "cat/"); - expectBadStartRequest(searchRequest, "Index", "/", "cat/"); + assertEquals("/cat%2F,dog/c,d/_search", initialSearch(searchRequest, query, remoteVersion).getEndpoint()); + // test a specific date math + all characters that need escaping. + searchRequest.indices("", "<>/{}|+:,"); + assertEquals("/%3Ccat%7Bnow%2Fd%7D%3E,%3C%3E%2F%7B%7D%7C%2B%3A%2C/c,d/_search", + initialSearch(searchRequest, query, remoteVersion).getEndpoint()); + + // pass-through if already escaped. + searchRequest.indices("%2f", "%3a"); + assertEquals("/%2f,%3a/c,d/_search", initialSearch(searchRequest, query, remoteVersion).getEndpoint()); + + // do not allow , and / if already escaped. + searchRequest.indices("%2fcat,"); + expectBadStartRequest(searchRequest, "Index", ",", "%2fcat,"); + searchRequest.indices("%3ccat/"); + expectBadStartRequest(searchRequest, "Index", "/", "%3ccat/"); searchRequest.indices("ok"); searchRequest.types("cat,"); diff --git a/plugins/repository-s3/src/main/java/org/elasticsearch/repositories/s3/S3Repository.java b/plugins/repository-s3/src/main/java/org/elasticsearch/repositories/s3/S3Repository.java index 522f15661bd64..f9d6ada5da38f 100644 --- a/plugins/repository-s3/src/main/java/org/elasticsearch/repositories/s3/S3Repository.java +++ b/plugins/repository-s3/src/main/java/org/elasticsearch/repositories/s3/S3Repository.java @@ -157,8 +157,6 @@ class S3Repository extends BlobStoreRepository { private final String cannedACL; - private final RepositoryMetaData repositoryMetaData; - /** * Constructs an s3 backed repository */ @@ -169,8 +167,6 @@ class S3Repository extends BlobStoreRepository { super(metadata, settings, namedXContentRegistry); this.service = service; - this.repositoryMetaData = metadata; - // Parse and validate the user's S3 Storage Class setting this.bucket = BUCKET_SETTING.get(metadata.settings()); if (bucket == null) { @@ -216,7 +212,7 @@ class S3Repository extends BlobStoreRepository { @Override protected S3BlobStore createBlobStore() { - return new S3BlobStore(service, bucket, serverSideEncryption, bufferSize, cannedACL, storageClass, repositoryMetaData); + return new S3BlobStore(service, bucket, serverSideEncryption, bufferSize, cannedACL, storageClass, metadata); } // only use for testing diff --git a/qa/rolling-upgrade/src/test/java/org/elasticsearch/upgrades/RecoveryIT.java b/qa/rolling-upgrade/src/test/java/org/elasticsearch/upgrades/RecoveryIT.java index f8246ca6dcd24..1b2eb7064f04c 100644 --- a/qa/rolling-upgrade/src/test/java/org/elasticsearch/upgrades/RecoveryIT.java +++ b/qa/rolling-upgrade/src/test/java/org/elasticsearch/upgrades/RecoveryIT.java @@ -121,6 +121,7 @@ protected void doRun() throws Exception { return future; } + @AwaitsFix(bugUrl = "https://github.com/elastic/elasticsearch/issues/40731") public void testRecoveryWithConcurrentIndexing() throws Exception { final String index = "recovery_with_concurrent_indexing"; Response response = client().performRequest(new Request("GET", "_nodes")); diff --git a/qa/vagrant/src/main/java/org/elasticsearch/packaging/test/ArchiveTestCase.java b/qa/vagrant/src/main/java/org/elasticsearch/packaging/test/ArchiveTestCase.java index af22861a90aa8..ba0484046adfd 100644 --- a/qa/vagrant/src/main/java/org/elasticsearch/packaging/test/ArchiveTestCase.java +++ b/qa/vagrant/src/main/java/org/elasticsearch/packaging/test/ArchiveTestCase.java @@ -20,21 +20,24 @@ package org.elasticsearch.packaging.test; import com.carrotsearch.randomizedtesting.annotations.TestCaseOrdering; +import com.carrotsearch.randomizedtesting.generators.RandomStrings; import org.apache.http.client.fluent.Request; import org.elasticsearch.packaging.util.Archives; import org.elasticsearch.packaging.util.Distribution; +import org.elasticsearch.packaging.util.FileUtils; import org.elasticsearch.packaging.util.Installation; import org.elasticsearch.packaging.util.Platforms; import org.elasticsearch.packaging.util.ServerUtils; import org.elasticsearch.packaging.util.Shell; import org.elasticsearch.packaging.util.Shell.Result; -import java.io.IOException; import java.nio.charset.StandardCharsets; import java.nio.file.Files; import java.nio.file.Path; +import java.nio.file.Paths; import java.util.stream.Stream; +import static com.carrotsearch.randomizedtesting.RandomizedTest.getRandom; import static org.elasticsearch.packaging.util.Archives.ARCHIVE_OWNER; import static org.elasticsearch.packaging.util.Archives.installArchive; import static org.elasticsearch.packaging.util.Archives.verifyArchiveInstallation; @@ -49,6 +52,7 @@ import static org.elasticsearch.packaging.util.FileUtils.rm; import static org.elasticsearch.packaging.util.ServerUtils.makeRequest; import static org.hamcrest.CoreMatchers.containsString; +import static org.hamcrest.CoreMatchers.equalTo; import static org.hamcrest.CoreMatchers.is; import static org.hamcrest.CoreMatchers.not; import static org.hamcrest.CoreMatchers.notNullValue; @@ -62,12 +66,12 @@ @TestCaseOrdering(TestCaseOrdering.AlphabeticOrder.class) public abstract class ArchiveTestCase extends PackagingTestCase { - public void test10Install() { + public void test10Install() throws Exception { installation = installArchive(distribution()); verifyArchiveInstallation(installation, distribution()); } - public void test20PluginsListWithNoPlugins() { + public void test20PluginsListWithNoPlugins() throws Exception { assumeThat(installation, is(notNullValue())); final Installation.Executables bin = installation.executables(); @@ -77,7 +81,7 @@ public void test20PluginsListWithNoPlugins() { assertThat(r.stdout, isEmptyString()); } - public void test30NoJava() { + public void test30NoJava() throws Exception { assumeThat(installation, is(notNullValue())); final Installation.Executables bin = installation.executables(); @@ -101,7 +105,7 @@ public void test30NoJava() { } } - public void test40CreateKeystoreManually() { + public void test40CreateKeystoreManually() throws Exception { assumeThat(installation, is(notNullValue())); final Installation.Executables bin = installation.executables(); @@ -134,7 +138,7 @@ public void test40CreateKeystoreManually() { }); } - public void test50StartAndStop() throws IOException { + public void test50StartAndStop() throws Exception { assumeThat(installation, is(notNullValue())); // cleanup from previous test @@ -152,7 +156,7 @@ public void test50StartAndStop() throws IOException { Archives.stopElasticsearch(installation); } - public void assertRunsWithJavaHome() throws IOException { + public void assertRunsWithJavaHome() throws Exception { Shell sh = newShell(); Platforms.onLinux(() -> { @@ -173,13 +177,13 @@ public void assertRunsWithJavaHome() throws IOException { assertThat(new String(Files.readAllBytes(log), StandardCharsets.UTF_8), containsString(systemJavaHome)); } - public void test51JavaHomeOverride() throws IOException { + public void test51JavaHomeOverride() throws Exception { assumeThat(installation, is(notNullValue())); assertRunsWithJavaHome(); } - public void test52BundledJdkRemoved() throws IOException { + public void test52BundledJdkRemoved() throws Exception { assumeThat(installation, is(notNullValue())); assumeThat(distribution().hasJdk, is(true)); @@ -192,7 +196,63 @@ public void test52BundledJdkRemoved() throws IOException { } } - public void test60AutoCreateKeystore() { + public void test53JavaHomeWithSpecialCharacters() throws Exception { + assumeThat(installation, is(notNullValue())); + + Platforms.onWindows(() -> { + final Shell sh = new Shell(); + try { + // once windows 2012 is no longer supported and powershell 5.0 is always available we can change this command + sh.run("cmd /c mklink /D 'C:\\Program Files (x86)\\java' $Env:JAVA_HOME"); + + sh.getEnv().put("JAVA_HOME", "C:\\Program Files (x86)\\java"); + + //verify ES can start, stop and run plugin list + Archives.runElasticsearch(installation, sh); + + Archives.stopElasticsearch(installation); + + String pluginListCommand = installation.bin + "/elasticsearch-plugin list"; + Result result = sh.run(pluginListCommand); + assertThat(result.exitCode, equalTo(0)); + + } finally { + //clean up sym link + sh.run("cmd /c del /F /Q 'C:\\Program Files (x86)\\java' "); + } + }); + + Platforms.onLinux(() -> { + final Shell sh = new Shell(); + // Create temporary directory with a space and link to java binary. + // Use it as java_home + String nameWithSpace = RandomStrings.randomAsciiAlphanumOfLength(getRandom(), 10) + "java home"; + String test_java_home = FileUtils.mkdir(Paths.get("/home",ARCHIVE_OWNER, nameWithSpace)).toAbsolutePath().toString(); + try { + final String systemJavaHome = sh.run("echo $SYSTEM_JAVA_HOME").stdout.trim(); + final String java = systemJavaHome + "/bin/java"; + + sh.run("mkdir -p \"" + test_java_home + "/bin\""); + sh.run("ln -s \"" + java + "\" \"" + test_java_home + "/bin/java\""); + sh.run("chown -R " + ARCHIVE_OWNER + ":" + ARCHIVE_OWNER + " \"" + test_java_home + "\""); + + sh.getEnv().put("JAVA_HOME", test_java_home); + + //verify ES can start, stop and run plugin list + Archives.runElasticsearch(installation, sh); + + Archives.stopElasticsearch(installation); + + String pluginListCommand = installation.bin + "/elasticsearch-plugin list"; + Result result = sh.run(pluginListCommand); + assertThat(result.exitCode, equalTo(0)); + } finally { + FileUtils.rm(Paths.get("\"" + test_java_home + "\"")); + } + }); + } + + public void test60AutoCreateKeystore() throws Exception { assumeThat(installation, is(notNullValue())); assertThat(installation.config("elasticsearch.keystore"), file(File, ARCHIVE_OWNER, ARCHIVE_OWNER, p660)); @@ -211,7 +271,7 @@ public void test60AutoCreateKeystore() { }); } - public void test70CustomPathConfAndJvmOptions() throws IOException { + public void test70CustomPathConfAndJvmOptions() throws Exception { assumeThat(installation, is(notNullValue())); final Path tempConf = getTempDir().resolve("esconf-alternate"); @@ -260,7 +320,7 @@ public void test70CustomPathConfAndJvmOptions() throws IOException { } } - public void test80RelativePathConf() throws IOException { + public void test80RelativePathConf() throws Exception { assumeThat(installation, is(notNullValue())); final Path temp = getTempDir().resolve("esconf-alternate"); @@ -304,7 +364,7 @@ public void test80RelativePathConf() throws IOException { } } - public void test90SecurityCliPackaging() { + public void test90SecurityCliPackaging() throws Exception { assumeThat(installation, is(notNullValue())); final Installation.Executables bin = installation.executables(); @@ -328,7 +388,7 @@ public void test90SecurityCliPackaging() { } } - public void test91ElasticsearchShardCliPackaging() { + public void test91ElasticsearchShardCliPackaging() throws Exception { assumeThat(installation, is(notNullValue())); final Installation.Executables bin = installation.executables(); @@ -345,7 +405,7 @@ public void test91ElasticsearchShardCliPackaging() { } } - public void test92ElasticsearchNodeCliPackaging() { + public void test92ElasticsearchNodeCliPackaging() throws Exception { assumeThat(installation, is(notNullValue())); final Installation.Executables bin = installation.executables(); @@ -363,7 +423,7 @@ public void test92ElasticsearchNodeCliPackaging() { } } - public void test93ElasticsearchNodeCustomDataPathAndNotEsHomeWorkDir() throws IOException { + public void test93ElasticsearchNodeCustomDataPathAndNotEsHomeWorkDir() throws Exception { assumeThat(installation, is(notNullValue())); Path relativeDataPath = installation.data.relativize(installation.home); diff --git a/qa/vagrant/src/main/java/org/elasticsearch/packaging/test/DebPreservationTestCase.java b/qa/vagrant/src/main/java/org/elasticsearch/packaging/test/DebPreservationTestCase.java index 1b2b891da4513..12597ae8b4de2 100644 --- a/qa/vagrant/src/main/java/org/elasticsearch/packaging/test/DebPreservationTestCase.java +++ b/qa/vagrant/src/main/java/org/elasticsearch/packaging/test/DebPreservationTestCase.java @@ -26,7 +26,6 @@ import org.junit.Before; import org.junit.BeforeClass; -import java.io.IOException; import java.nio.file.Files; import java.nio.file.Paths; @@ -54,7 +53,7 @@ public abstract class DebPreservationTestCase extends PackagingTestCase { protected abstract Distribution distribution(); @BeforeClass - public static void cleanup() { + public static void cleanup() throws Exception { installation = null; cleanEverything(); } @@ -65,14 +64,14 @@ public void onlyCompatibleDistributions() { assumeTrue("only compatible distributions", distribution().packaging.compatible); } - public void test10Install() throws IOException { + public void test10Install() throws Exception { assertRemoved(distribution()); installation = install(distribution()); assertInstalled(distribution()); verifyPackageInstallation(installation, distribution(), newShell()); } - public void test20Remove() { + public void test20Remove() throws Exception { assumeThat(installation, is(notNullValue())); remove(distribution()); @@ -117,7 +116,7 @@ public void test20Remove() { assertTrue(Files.exists(installation.envFile)); } - public void test30Purge() { + public void test30Purge() throws Exception { assumeThat(installation, is(notNullValue())); final Shell sh = new Shell(); diff --git a/qa/vagrant/src/main/java/org/elasticsearch/packaging/test/PackageTestCase.java b/qa/vagrant/src/main/java/org/elasticsearch/packaging/test/PackageTestCase.java index 458359b299e75..c664e28931087 100644 --- a/qa/vagrant/src/main/java/org/elasticsearch/packaging/test/PackageTestCase.java +++ b/qa/vagrant/src/main/java/org/elasticsearch/packaging/test/PackageTestCase.java @@ -20,6 +20,7 @@ package org.elasticsearch.packaging.test; import com.carrotsearch.randomizedtesting.annotations.TestCaseOrdering; +import com.carrotsearch.randomizedtesting.generators.RandomStrings; import org.apache.http.client.fluent.Request; import org.elasticsearch.packaging.util.FileUtils; import org.elasticsearch.packaging.util.Shell; @@ -27,7 +28,6 @@ import org.hamcrest.CoreMatchers; import org.junit.Before; -import java.io.IOException; import java.nio.charset.StandardCharsets; import java.nio.file.Files; import java.nio.file.Path; @@ -36,6 +36,7 @@ import java.util.regex.Matcher; import java.util.regex.Pattern; +import static com.carrotsearch.randomizedtesting.RandomizedTest.getRandom; import static org.elasticsearch.packaging.util.FileUtils.append; import static org.elasticsearch.packaging.util.FileUtils.assertPathsDontExist; import static org.elasticsearch.packaging.util.FileUtils.assertPathsExist; @@ -72,19 +73,19 @@ public abstract class PackageTestCase extends PackagingTestCase { private Shell sh; @Before - public void onlyCompatibleDistributions() { + public void onlyCompatibleDistributions() throws Exception { assumeTrue("only compatible distributions", distribution().packaging.compatible); sh = newShell(); } - public void test10InstallPackage() throws IOException { + public void test10InstallPackage() throws Exception { assertRemoved(distribution()); installation = install(distribution()); assertInstalled(distribution()); verifyPackageInstallation(installation, distribution(), sh); } - public void test20PluginsCommandWhenNoPlugins() { + public void test20PluginsCommandWhenNoPlugins() throws Exception { assumeThat(installation, is(notNullValue())); assertThat(sh.run(installation.bin("elasticsearch-plugin") + " list").stdout, isEmptyString()); @@ -104,7 +105,7 @@ public void test31InstallDoesNotStartServer() { assertThat(sh.run("ps aux").stdout, not(containsString("org.elasticsearch.bootstrap.Elasticsearch"))); } - public void assertRunsWithJavaHome() throws IOException { + public void assertRunsWithJavaHome() throws Exception { String systemJavaHome = sh.run("echo $SYSTEM_JAVA_HOME").stdout.trim(); byte[] originalEnvFile = Files.readAllBytes(installation.envFile); try { @@ -121,7 +122,7 @@ public void assertRunsWithJavaHome() throws IOException { assertThat(new String(Files.readAllBytes(log), StandardCharsets.UTF_8), containsString(systemJavaHome)); } - public void test32JavaHomeOverride() throws IOException { + public void test32JavaHomeOverride() throws Exception { assumeThat(installation, is(notNullValue())); // we always run with java home when no bundled jdk is included, so this test would be repetitive assumeThat(distribution().hasJdk, is(true)); @@ -129,7 +130,7 @@ public void test32JavaHomeOverride() throws IOException { assertRunsWithJavaHome(); } - public void test42BundledJdkRemoved() throws IOException { + public void test42BundledJdkRemoved() throws Exception { assumeThat(installation, is(notNullValue())); assumeThat(distribution().hasJdk, is(true)); @@ -142,7 +143,7 @@ public void test42BundledJdkRemoved() throws IOException { } } - public void test40StartServer() throws IOException { + public void test40StartServer() throws Exception { String start = sh.runIgnoreExitCode("date ").stdout.trim(); assumeThat(installation, is(notNullValue())); @@ -159,7 +160,7 @@ public void test40StartServer() throws IOException { verifyPackageInstallation(installation, distribution(), sh); // check startup script didn't change permissions } - public void test50Remove() { + public void test50Remove() throws Exception { assumeThat(installation, is(notNullValue())); remove(distribution()); @@ -209,7 +210,7 @@ public void test50Remove() { assertFalse(Files.exists(SYSTEMD_SERVICE)); } - public void test60Reinstall() throws IOException { + public void test60Reinstall() throws Exception { assumeThat(installation, is(notNullValue())); installation = install(distribution()); @@ -220,7 +221,7 @@ public void test60Reinstall() throws IOException { assertRemoved(distribution()); } - public void test70RestartServer() throws IOException { + public void test70RestartServer() throws Exception { try { installation = install(distribution()); assertInstalled(distribution()); @@ -235,7 +236,7 @@ public void test70RestartServer() throws IOException { } - public void test72TestRuntimeDirectory() throws IOException { + public void test72TestRuntimeDirectory() throws Exception { try { installation = install(distribution()); FileUtils.rm(installation.pidDir); @@ -247,7 +248,7 @@ public void test72TestRuntimeDirectory() throws IOException { } } - public void test73gcLogsExist() throws IOException { + public void test73gcLogsExist() throws Exception { installation = install(distribution()); startElasticsearch(sh); // it can be gc.log or gc.log.0.current @@ -264,7 +265,7 @@ public void test73gcLogsExist() throws IOException { * # but it should not block ES from starting * # see https://github.com/elastic/elasticsearch/issues/11594 */ - public void test80DeletePID_DIRandRestart() throws IOException { + public void test80DeletePID_DIRandRestart() throws Exception { assumeTrue(isSystemd()); rm(installation.pidDir); @@ -280,7 +281,7 @@ public void test80DeletePID_DIRandRestart() throws IOException { stopElasticsearch(sh); } - public void test81CustomPathConfAndJvmOptions() throws IOException { + public void test81CustomPathConfAndJvmOptions() throws Exception { assumeTrue(isSystemd()); assumeThat(installation, is(notNullValue())); @@ -291,8 +292,9 @@ public void test81CustomPathConfAndJvmOptions() throws IOException { // The custom config directory is not under /tmp or /var/tmp because // systemd's private temp directory functionally means different // processes can have different views of what's in these directories - String temp = sh.runIgnoreExitCode("mktemp -p /etc -d").stdout.trim(); - final Path tempConf = Paths.get(temp); + String randomName = RandomStrings.randomAsciiAlphanumOfLength(getRandom(), 10); + sh.run("mkdir /etc/"+randomName); + final Path tempConf = Paths.get("/etc/"+randomName); try { mkdir(tempConf); @@ -331,7 +333,7 @@ public void test81CustomPathConfAndJvmOptions() throws IOException { } } - public void test82SystemdMask() throws IOException { + public void test82SystemdMask() throws Exception { try { assumeTrue(isSystemd()); @@ -345,7 +347,7 @@ public void test82SystemdMask() throws IOException { } } - public void test83serviceFileSetsLimits() throws IOException { + public void test83serviceFileSetsLimits() throws Exception { // Limits are changed on systemd platforms only assumeTrue(isSystemd()); diff --git a/qa/vagrant/src/main/java/org/elasticsearch/packaging/test/PackagingTestCase.java b/qa/vagrant/src/main/java/org/elasticsearch/packaging/test/PackagingTestCase.java index 7cb860e617eb0..bd7738aeac4ac 100644 --- a/qa/vagrant/src/main/java/org/elasticsearch/packaging/test/PackagingTestCase.java +++ b/qa/vagrant/src/main/java/org/elasticsearch/packaging/test/PackagingTestCase.java @@ -64,7 +64,7 @@ public void setup() { protected static Installation installation; @BeforeClass - public static void cleanup() { + public static void cleanup() throws Exception { installation = null; cleanEverything(); } @@ -72,7 +72,7 @@ public static void cleanup() { /** The {@link Distribution} that should be tested in this case */ protected abstract Distribution distribution(); - protected Shell newShell() { + protected Shell newShell() throws Exception { Shell sh = new Shell(); if (distribution().hasJdk == false) { Platforms.onLinux(() -> { diff --git a/qa/vagrant/src/main/java/org/elasticsearch/packaging/test/RpmPreservationTestCase.java b/qa/vagrant/src/main/java/org/elasticsearch/packaging/test/RpmPreservationTestCase.java index 5cfc10b110afb..7b6ac039fc55c 100644 --- a/qa/vagrant/src/main/java/org/elasticsearch/packaging/test/RpmPreservationTestCase.java +++ b/qa/vagrant/src/main/java/org/elasticsearch/packaging/test/RpmPreservationTestCase.java @@ -26,7 +26,6 @@ import org.junit.Before; import org.junit.BeforeClass; -import java.io.IOException; import java.nio.file.Files; import java.nio.file.Path; import java.util.stream.Stream; @@ -56,7 +55,7 @@ public abstract class RpmPreservationTestCase extends PackagingTestCase { protected abstract Distribution distribution(); @BeforeClass - public static void cleanup() { + public static void cleanup() throws Exception { installation = null; cleanEverything(); } @@ -67,14 +66,14 @@ public void onlyCompatibleDistributions() { assumeTrue("only compatible distributions", distribution().packaging.compatible); } - public void test10Install() throws IOException { + public void test10Install() throws Exception { assertRemoved(distribution()); installation = install(distribution()); assertInstalled(distribution()); verifyPackageInstallation(installation, distribution(), newShell()); } - public void test20Remove() { + public void test20Remove() throws Exception { assumeThat(installation, is(notNullValue())); remove(distribution()); @@ -89,7 +88,7 @@ public void test20Remove() { assertFalse(Files.exists(installation.envFile)); } - public void test30PreserveConfig() throws IOException { + public void test30PreserveConfig() throws Exception { final Shell sh = new Shell(); installation = install(distribution()); diff --git a/qa/vagrant/src/main/java/org/elasticsearch/packaging/test/WindowsServiceTestCase.java b/qa/vagrant/src/main/java/org/elasticsearch/packaging/test/WindowsServiceTestCase.java index 08f54096e073d..57eaf13fe9e94 100644 --- a/qa/vagrant/src/main/java/org/elasticsearch/packaging/test/WindowsServiceTestCase.java +++ b/qa/vagrant/src/main/java/org/elasticsearch/packaging/test/WindowsServiceTestCase.java @@ -102,7 +102,7 @@ private void assertExit(Result result, String script, int exitCode) { } } - public void test10InstallArchive() { + public void test10InstallArchive() throws Exception { installation = installArchive(distribution()); verifyArchiveInstallation(installation, distribution()); serviceScript = installation.bin("elasticsearch-service.bat").toString(); diff --git a/qa/vagrant/src/main/java/org/elasticsearch/packaging/util/Archives.java b/qa/vagrant/src/main/java/org/elasticsearch/packaging/util/Archives.java index c8ddda2dc4f37..e557b47fb8912 100644 --- a/qa/vagrant/src/main/java/org/elasticsearch/packaging/util/Archives.java +++ b/qa/vagrant/src/main/java/org/elasticsearch/packaging/util/Archives.java @@ -19,7 +19,6 @@ package org.elasticsearch.packaging.util; -import java.io.IOException; import java.nio.file.Files; import java.nio.file.Path; import java.nio.file.Paths; @@ -37,15 +36,14 @@ import static org.elasticsearch.packaging.util.FileUtils.getDefaultArchiveInstallPath; import static org.elasticsearch.packaging.util.FileUtils.getDistributionFile; import static org.elasticsearch.packaging.util.FileUtils.lsGlob; - import static org.elasticsearch.packaging.util.FileUtils.mv; import static org.elasticsearch.packaging.util.FileUtils.slurp; import static org.elasticsearch.packaging.util.Platforms.isDPKG; import static org.hamcrest.MatcherAssert.assertThat; import static org.hamcrest.Matchers.isEmptyOrNullString; -import static org.hamcrest.core.Is.is; -import static org.hamcrest.collection.IsEmptyCollection.empty; import static org.hamcrest.collection.IsCollectionWithSize.hasSize; +import static org.hamcrest.collection.IsEmptyCollection.empty; +import static org.hamcrest.core.Is.is; import static org.hamcrest.core.IsNot.not; import static org.junit.Assert.assertTrue; @@ -59,11 +57,11 @@ public class Archives { ? "vagrant" : "elasticsearch"; - public static Installation installArchive(Distribution distribution) { + public static Installation installArchive(Distribution distribution) throws Exception { return installArchive(distribution, getDefaultArchiveInstallPath(), getCurrentVersion()); } - public static Installation installArchive(Distribution distribution, Path fullInstallPath, String version) { + public static Installation installArchive(Distribution distribution, Path fullInstallPath, String version) throws Exception { final Shell sh = new Shell(); final Path distributionFile = getDistributionFile(distribution); @@ -255,7 +253,7 @@ private static void verifyDefaultInstallation(Installation es, Distribution dist ).forEach(configFile -> assertThat(es.config(configFile), file(File, owner, owner, p660))); } - public static void runElasticsearch(Installation installation, Shell sh) throws IOException { + public static void runElasticsearch(Installation installation, Shell sh) throws Exception { final Path pidFile = installation.home.resolve("elasticsearch.pid"); final Installation.Executables bin = installation.executables(); @@ -305,7 +303,7 @@ public static void runElasticsearch(Installation installation, Shell sh) throws Platforms.onWindows(() -> sh.run("Get-Process -Id " + pid)); } - public static void stopElasticsearch(Installation installation) { + public static void stopElasticsearch(Installation installation) throws Exception { Path pidFile = installation.home.resolve("elasticsearch.pid"); assertTrue(Files.exists(pidFile)); String pid = slurp(pidFile).trim(); diff --git a/qa/vagrant/src/main/java/org/elasticsearch/packaging/util/Cleanup.java b/qa/vagrant/src/main/java/org/elasticsearch/packaging/util/Cleanup.java index fda61e9fb36e5..f9b98d58ccacc 100644 --- a/qa/vagrant/src/main/java/org/elasticsearch/packaging/util/Cleanup.java +++ b/qa/vagrant/src/main/java/org/elasticsearch/packaging/util/Cleanup.java @@ -50,7 +50,7 @@ public class Cleanup { // todo private static final List ELASTICSEARCH_FILES_WINDOWS = Collections.emptyList(); - public static void cleanEverything() { + public static void cleanEverything() throws Exception { final Shell sh = new Shell(); // kill elasticsearch processes diff --git a/qa/vagrant/src/main/java/org/elasticsearch/packaging/util/Packages.java b/qa/vagrant/src/main/java/org/elasticsearch/packaging/util/Packages.java index afa7e371c2c55..c5dcc34af882f 100644 --- a/qa/vagrant/src/main/java/org/elasticsearch/packaging/util/Packages.java +++ b/qa/vagrant/src/main/java/org/elasticsearch/packaging/util/Packages.java @@ -54,14 +54,14 @@ public class Packages { public static final Path SYSVINIT_SCRIPT = Paths.get("/etc/init.d/elasticsearch"); public static final Path SYSTEMD_SERVICE = Paths.get("/usr/lib/systemd/system/elasticsearch.service"); - public static void assertInstalled(Distribution distribution) { + public static void assertInstalled(Distribution distribution) throws Exception { final Result status = packageStatus(distribution); assertThat(status.exitCode, is(0)); Platforms.onDPKG(() -> assertFalse(Pattern.compile("(?m)^Status:.+deinstall ok").matcher(status.stdout).find())); } - public static void assertRemoved(Distribution distribution) { + public static void assertRemoved(Distribution distribution) throws Exception { final Result status = packageStatus(distribution); Platforms.onRPM(() -> assertThat(status.exitCode, is(1))); @@ -133,7 +133,7 @@ public static Result runInstallCommand(Distribution distribution, String version } } - public static void remove(Distribution distribution) { + public static void remove(Distribution distribution) throws Exception { final Shell sh = new Shell(); Platforms.onRPM(() -> { diff --git a/qa/vagrant/src/main/java/org/elasticsearch/packaging/util/Platforms.java b/qa/vagrant/src/main/java/org/elasticsearch/packaging/util/Platforms.java index dbac9c88d26c9..6258c1336b2fc 100644 --- a/qa/vagrant/src/main/java/org/elasticsearch/packaging/util/Platforms.java +++ b/qa/vagrant/src/main/java/org/elasticsearch/packaging/util/Platforms.java @@ -65,25 +65,25 @@ public static boolean isSysVInit() { return new Shell().runIgnoreExitCode("which service").isSuccess(); } - public static void onWindows(PlatformAction action) { + public static void onWindows(PlatformAction action) throws Exception { if (WINDOWS) { action.run(); } } - public static void onLinux(PlatformAction action) { + public static void onLinux(PlatformAction action) throws Exception { if (LINUX) { action.run(); } } - public static void onRPM(PlatformAction action) { + public static void onRPM(PlatformAction action) throws Exception { if (isRPM()) { action.run(); } } - public static void onDPKG(PlatformAction action) { + public static void onDPKG(PlatformAction action) throws Exception { if (isDPKG()) { action.run(); } @@ -94,6 +94,6 @@ public static void onDPKG(PlatformAction action) { */ @FunctionalInterface public interface PlatformAction { - void run(); + void run() throws Exception; } } diff --git a/server/src/main/java/org/elasticsearch/Build.java b/server/src/main/java/org/elasticsearch/Build.java index dcf827091f54e..be37c56837d70 100644 --- a/server/src/main/java/org/elasticsearch/Build.java +++ b/server/src/main/java/org/elasticsearch/Build.java @@ -241,7 +241,13 @@ public static void writeBuild(Build build, StreamOutput out) throws IOException out.writeString(build.flavor().displayName()); } if (out.getVersion().onOrAfter(Version.V_6_3_0)) { - out.writeString(build.type().displayName()); + final Type buildType; + if (out.getVersion().before(Version.V_6_7_0) && build.type() == Type.DOCKER) { + buildType = Type.TAR; + } else { + buildType = build.type(); + } + out.writeString(buildType.displayName()); } out.writeString(build.shortHash()); out.writeString(build.date()); diff --git a/server/src/main/java/org/elasticsearch/common/time/DateFormatters.java b/server/src/main/java/org/elasticsearch/common/time/DateFormatters.java index 2379b4f00c2bf..027d360153f2b 100644 --- a/server/src/main/java/org/elasticsearch/common/time/DateFormatters.java +++ b/server/src/main/java/org/elasticsearch/common/time/DateFormatters.java @@ -456,7 +456,7 @@ public class DateFormatters { .appendValue(HOUR_OF_DAY, 2, 2, SignStyle.NOT_NEGATIVE) .appendValue(MINUTE_OF_HOUR, 2, 2, SignStyle.NOT_NEGATIVE) .appendValue(SECOND_OF_MINUTE, 2, 2, SignStyle.NOT_NEGATIVE) - .appendFraction(NANO_OF_SECOND, 3, 9, true) + .appendFraction(NANO_OF_SECOND, 1, 9, true) .appendZoneOrOffsetId() .toFormatter(Locale.ROOT), new DateTimeFormatterBuilder() @@ -465,7 +465,7 @@ public class DateFormatters { .appendValue(HOUR_OF_DAY, 2, 2, SignStyle.NOT_NEGATIVE) .appendValue(MINUTE_OF_HOUR, 2, 2, SignStyle.NOT_NEGATIVE) .appendValue(SECOND_OF_MINUTE, 2, 2, SignStyle.NOT_NEGATIVE) - .appendFraction(NANO_OF_SECOND, 3, 9, true) + .appendFraction(NANO_OF_SECOND, 1, 9, true) .append(TIME_ZONE_FORMATTER_NO_COLON) .toFormatter(Locale.ROOT) ); @@ -517,12 +517,20 @@ public class DateFormatters { private static final DateFormatter STRICT_HOUR_MINUTE_SECOND = new JavaDateFormatter("strict_hour_minute_second", STRICT_HOUR_MINUTE_SECOND_FORMATTER); + private static final DateTimeFormatter STRICT_DATE_PRINTER = new DateTimeFormatterBuilder() + .append(STRICT_YEAR_MONTH_DAY_FORMATTER) + .appendLiteral('T') + .append(STRICT_HOUR_MINUTE_SECOND_FORMATTER) + .appendFraction(NANO_OF_SECOND, 3, 9, true) + .appendOffset("+HH:MM", "Z") + .toFormatter(Locale.ROOT); + private static final DateTimeFormatter STRICT_DATE_FORMATTER = new DateTimeFormatterBuilder() .append(STRICT_YEAR_MONTH_DAY_FORMATTER) .appendLiteral('T') .append(STRICT_HOUR_MINUTE_SECOND_FORMATTER) .optionalStart() - .appendFraction(NANO_OF_SECOND, 3, 9, true) + .appendFraction(NANO_OF_SECOND, 1, 9, true) .optionalEnd() .toFormatter(Locale.ROOT); @@ -530,8 +538,7 @@ public class DateFormatters { * Returns a formatter that combines a full date and time, separated by a 'T' * (yyyy-MM-dd'T'HH:mm:ss.SSSZZ). */ - private static final DateFormatter STRICT_DATE_TIME = new JavaDateFormatter("strict_date_time", - new DateTimeFormatterBuilder().append(STRICT_DATE_FORMATTER).appendOffset("+HH:MM", "Z").toFormatter(Locale.ROOT), + private static final DateFormatter STRICT_DATE_TIME = new JavaDateFormatter("strict_date_time", STRICT_DATE_PRINTER, new DateTimeFormatterBuilder().append(STRICT_DATE_FORMATTER).appendZoneOrOffsetId().toFormatter(Locale.ROOT), new DateTimeFormatterBuilder().append(STRICT_DATE_FORMATTER).append(TIME_ZONE_FORMATTER_NO_COLON).toFormatter(Locale.ROOT) ); @@ -653,7 +660,7 @@ public class DateFormatters { private static final DateFormatter STRICT_HOUR_MINUTE = new JavaDateFormatter("strict_hour_minute", DateTimeFormatter.ofPattern("HH:mm", Locale.ROOT)); - private static final DateTimeFormatter STRICT_ORDINAL_DATE_TIME_FORMATTER_BASE = new DateTimeFormatterBuilder() + private static final DateTimeFormatter STRICT_ORDINAL_DATE_TIME_PRINTER = new DateTimeFormatterBuilder() .appendValue(ChronoField.YEAR, 4, 10, SignStyle.EXCEEDS_PAD) .appendLiteral('-') .appendValue(DAY_OF_YEAR, 3, 3, SignStyle.NOT_NEGATIVE) @@ -666,12 +673,25 @@ public class DateFormatters { .optionalEnd() .toFormatter(Locale.ROOT); + private static final DateTimeFormatter STRICT_ORDINAL_DATE_TIME_FORMATTER_BASE = new DateTimeFormatterBuilder() + .appendValue(ChronoField.YEAR, 4, 10, SignStyle.EXCEEDS_PAD) + .appendLiteral('-') + .appendValue(DAY_OF_YEAR, 3, 3, SignStyle.NOT_NEGATIVE) + .appendLiteral('T') + .appendPattern("HH:mm") + .optionalStart() + .appendLiteral(':') + .appendValue(SECOND_OF_MINUTE, 2, 2, SignStyle.NOT_NEGATIVE) + .appendFraction(NANO_OF_SECOND, 1, 9, true) + .optionalEnd() + .toFormatter(Locale.ROOT); + /* * Returns a formatter for a full ordinal date and time, using a four * digit year and three digit dayOfYear (yyyy-DDD'T'HH:mm:ss.SSSZZ). */ private static final DateFormatter STRICT_ORDINAL_DATE_TIME = new JavaDateFormatter("strict_ordinal_date_time", - new DateTimeFormatterBuilder().append(STRICT_ORDINAL_DATE_TIME_FORMATTER_BASE) + new DateTimeFormatterBuilder().append(STRICT_ORDINAL_DATE_TIME_PRINTER) .appendOffset("+HH:MM", "Z").toFormatter(Locale.ROOT), new DateTimeFormatterBuilder().append(STRICT_ORDINAL_DATE_TIME_FORMATTER_BASE) .appendZoneOrOffsetId().toFormatter(Locale.ROOT), @@ -1198,7 +1218,7 @@ public class DateFormatters { * digit year and three digit dayOfYear (yyyy-DDD'T'HH:mm:ss.SSSZZ). */ private static final DateFormatter ORDINAL_DATE_TIME = new JavaDateFormatter("ordinal_date_time", - new DateTimeFormatterBuilder().append(STRICT_ORDINAL_DATE_TIME_FORMATTER_BASE) + new DateTimeFormatterBuilder().append(STRICT_ORDINAL_DATE_TIME_PRINTER) .appendOffset("+HH:MM", "Z").toFormatter(Locale.ROOT), new DateTimeFormatterBuilder().append(ORDINAL_DATE_TIME_FORMATTER_BASE) .appendZoneOrOffsetId().toFormatter(Locale.ROOT), diff --git a/server/src/main/java/org/elasticsearch/index/shard/IndexShard.java b/server/src/main/java/org/elasticsearch/index/shard/IndexShard.java index 97d1939c1b292..ff1922a231dc8 100644 --- a/server/src/main/java/org/elasticsearch/index/shard/IndexShard.java +++ b/server/src/main/java/org/elasticsearch/index/shard/IndexShard.java @@ -947,9 +947,7 @@ public FlushStats flushStats() { public DocsStats docStats() { readAllowed(); - DocsStats docsStats = getEngine().docStats(); - markSearcherAccessed(); - return docsStats; + return getEngine().docStats(); } /** @@ -1028,11 +1026,7 @@ public TranslogStats translogStats() { public CompletionStats completionStats(String... fields) { readAllowed(); try { - CompletionStats stats = getEngine().completionStats(fields); - // we don't wait for a pending refreshes here since it's a stats call instead we mark it as accessed only which will cause - // the next scheduled refresh to go through and refresh the stats as well - markSearcherAccessed(); - return stats; + return getEngine().completionStats(fields); } catch (IOException e) { throw new UncheckedIOException(e); } diff --git a/server/src/main/java/org/elasticsearch/repositories/blobstore/BlobStoreRepository.java b/server/src/main/java/org/elasticsearch/repositories/blobstore/BlobStoreRepository.java index a73626351d875..253501a542b20 100644 --- a/server/src/main/java/org/elasticsearch/repositories/blobstore/BlobStoreRepository.java +++ b/server/src/main/java/org/elasticsearch/repositories/blobstore/BlobStoreRepository.java @@ -135,11 +135,11 @@ * |- Ac1342-B_x/ - data for index "foo" which was assigned the unique id of Ac1342-B_x in the repository * | |- meta-20131010.dat - JSON Serialized IndexMetaData for index "foo" * | |- 0/ - data for shard "0" of index "foo" - * | | |- __1 \ - * | | |- __2 | - * | | |- __3 |- files from different segments see snapshot-* for their mappings to real segment files - * | | |- __4 | - * | | |- __5 / + * | | |- __1 \ (files with numeric names were created by older ES versions) + * | | |- __2 | + * | | |- __VPO5oDMVT5y4Akv8T_AO_A |- files from different segments see snap-* for their mappings to real segment files + * | | |- __1gbJy18wS_2kv1qI7FgKuQ | + * | | |- __R8JvZAHlSMyMXyZc2SS8Zg / * | | ..... * | | |- snap-20131010.dat - JSON serialized BlobStoreIndexShardSnapshot for snapshot "20131010" * | | |- snap-20131011.dat - JSON serialized BlobStoreIndexShardSnapshot for snapshot "20131011" @@ -162,8 +162,6 @@ public abstract class BlobStoreRepository extends AbstractLifecycleComponent imp protected final RepositoryMetaData metadata; - protected final NamedXContentRegistry namedXContentRegistry; - private static final int BUFFER_SIZE = 4096; private static final String SNAPSHOT_PREFIX = "snap-"; @@ -213,11 +211,11 @@ public abstract class BlobStoreRepository extends AbstractLifecycleComponent imp private final CounterMetric restoreRateLimitingTimeInNanos = new CounterMetric(); - private ChecksumBlobStoreFormat globalMetaDataFormat; + private final ChecksumBlobStoreFormat globalMetaDataFormat; - private ChecksumBlobStoreFormat indexMetaDataFormat; + private final ChecksumBlobStoreFormat indexMetaDataFormat; - private ChecksumBlobStoreFormat snapshotFormat; + private final ChecksumBlobStoreFormat snapshotFormat; private final boolean readOnly; @@ -240,17 +238,21 @@ protected BlobStoreRepository(RepositoryMetaData metadata, Settings settings, NamedXContentRegistry namedXContentRegistry) { this.settings = settings; this.metadata = metadata; - this.namedXContentRegistry = namedXContentRegistry; this.compress = COMPRESS_SETTING.get(metadata.settings()); snapshotRateLimiter = getRateLimiter(metadata.settings(), "max_snapshot_bytes_per_sec", new ByteSizeValue(40, ByteSizeUnit.MB)); restoreRateLimiter = getRateLimiter(metadata.settings(), "max_restore_bytes_per_sec", new ByteSizeValue(40, ByteSizeUnit.MB)); readOnly = metadata.settings().getAsBoolean("readonly", false); - indexShardSnapshotFormat = new ChecksumBlobStoreFormat<>(SNAPSHOT_CODEC, SNAPSHOT_NAME_FORMAT, BlobStoreIndexShardSnapshot::fromXContent, namedXContentRegistry, compress); indexShardSnapshotsFormat = new ChecksumBlobStoreFormat<>(SNAPSHOT_INDEX_CODEC, SNAPSHOT_INDEX_NAME_FORMAT, BlobStoreIndexShardSnapshots::fromXContent, namedXContentRegistry, compress); + globalMetaDataFormat = new ChecksumBlobStoreFormat<>(METADATA_CODEC, METADATA_NAME_FORMAT, + MetaData::fromXContent, namedXContentRegistry, compress); + indexMetaDataFormat = new ChecksumBlobStoreFormat<>(INDEX_METADATA_CODEC, METADATA_NAME_FORMAT, + IndexMetaData::fromXContent, namedXContentRegistry, compress); + snapshotFormat = new ChecksumBlobStoreFormat<>(SNAPSHOT_CODEC, SNAPSHOT_NAME_FORMAT, + SnapshotInfo::fromXContentInternal, namedXContentRegistry, compress); } @Override @@ -259,12 +261,6 @@ protected void doStart() { if (chunkSize != null && chunkSize.getBytes() <= 0) { throw new IllegalArgumentException("the chunk size cannot be negative: [" + chunkSize + "]"); } - globalMetaDataFormat = new ChecksumBlobStoreFormat<>(METADATA_CODEC, METADATA_NAME_FORMAT, - MetaData::fromXContent, namedXContentRegistry, compress); - indexMetaDataFormat = new ChecksumBlobStoreFormat<>(INDEX_METADATA_CODEC, METADATA_NAME_FORMAT, - IndexMetaData::fromXContent, namedXContentRegistry, compress); - snapshotFormat = new ChecksumBlobStoreFormat<>(SNAPSHOT_CODEC, SNAPSHOT_NAME_FORMAT, - SnapshotInfo::fromXContentInternal, namedXContentRegistry, compress); } @Override @@ -1069,41 +1065,6 @@ protected void finalize(final List snapshots, } } - /** - * Generates blob name - * - * @param generation the blob number - * @return the blob name - */ - protected String fileNameFromGeneration(long generation) { - return DATA_BLOB_PREFIX + Long.toString(generation, Character.MAX_RADIX); - } - - /** - * Finds the next available blob number - * - * @param blobs list of blobs in the repository - * @return next available blob number - */ - protected long findLatestFileNameGeneration(Map blobs) { - long generation = -1; - for (String name : blobs.keySet()) { - if (!name.startsWith(DATA_BLOB_PREFIX)) { - continue; - } - name = canonicalName(name); - try { - long currentGen = Long.parseLong(name.substring(DATA_BLOB_PREFIX.length()), Character.MAX_RADIX); - if (currentGen > generation) { - generation = currentGen; - } - } catch (NumberFormatException e) { - logger.warn("file [{}] does not conform to the '{}' schema", name, DATA_BLOB_PREFIX); - } - } - return generation; - } - /** * Loads all available snapshots in the repository * @@ -1196,7 +1157,6 @@ public void snapshot(final IndexCommit snapshotIndexCommit) { throw new IndexShardSnapshotFailedException(shardId, "failed to list blobs", e); } - long generation = findLatestFileNameGeneration(blobs); Tuple tuple = buildBlobStoreIndexShardSnapshots(blobs); BlobStoreIndexShardSnapshots snapshots = tuple.v1(); int fileListGeneration = tuple.v2(); @@ -1264,7 +1224,7 @@ public void snapshot(final IndexCommit snapshotIndexCommit) { indexIncrementalSize += md.length(); // create a new FileInfo BlobStoreIndexShardSnapshot.FileInfo snapshotFileInfo = - new BlobStoreIndexShardSnapshot.FileInfo(fileNameFromGeneration(++generation), md, chunkSize()); + new BlobStoreIndexShardSnapshot.FileInfo(DATA_BLOB_PREFIX + UUIDs.randomBase64UUID(), md, chunkSize()); indexCommitPointFiles.add(snapshotFileInfo); filesToSnapshot.add(snapshotFileInfo); } else { diff --git a/server/src/main/java/org/elasticsearch/repositories/fs/FsRepository.java b/server/src/main/java/org/elasticsearch/repositories/fs/FsRepository.java index e3e986c1eca9a..d2a27cc2bb3ef 100644 --- a/server/src/main/java/org/elasticsearch/repositories/fs/FsRepository.java +++ b/server/src/main/java/org/elasticsearch/repositories/fs/FsRepository.java @@ -63,7 +63,7 @@ public class FsRepository extends BlobStoreRepository { new ByteSizeValue(Long.MAX_VALUE), new ByteSizeValue(5), new ByteSizeValue(Long.MAX_VALUE), Property.NodeScope); private final Environment environment; - private ByteSizeValue chunkSize; + private final ByteSizeValue chunkSize; private final BlobPath basePath; diff --git a/server/src/test/java/org/elasticsearch/BuildTests.java b/server/src/test/java/org/elasticsearch/BuildTests.java index 1f99a1f4542b5..12af1d31841cf 100644 --- a/server/src/test/java/org/elasticsearch/BuildTests.java +++ b/server/src/test/java/org/elasticsearch/BuildTests.java @@ -20,15 +20,23 @@ package org.elasticsearch; import org.elasticsearch.common.io.FileSystemUtils; +import org.elasticsearch.common.io.stream.StreamInput; +import org.elasticsearch.common.io.stream.StreamOutput; +import org.elasticsearch.common.io.stream.Writeable; import org.elasticsearch.test.ESTestCase; +import org.elasticsearch.test.EqualsHashCodeTestUtils; import java.io.IOException; import java.io.InputStream; import java.net.URL; import java.util.Arrays; +import java.util.List; +import java.util.Objects; import java.util.Set; import java.util.stream.Collectors; +import static org.hamcrest.Matchers.equalTo; + public class BuildTests extends ESTestCase { /** Asking for the jar metadata should not throw exception in tests, no matter how configured */ @@ -115,4 +123,103 @@ public void testEqualsAndHashCode() { ); assertNotEquals(build, differentVersion); } + + private static class WriteableBuild implements Writeable { + private final Build build; + + WriteableBuild(StreamInput in) throws IOException { + build = Build.readBuild(in); + } + + WriteableBuild(Build build) { + this.build = build; + } + + @Override + public void writeTo(StreamOutput out) throws IOException { + Build.writeBuild(build, out); + } + + @Override + public boolean equals(Object o) { + if (this == o) return true; + if (o == null || getClass() != o.getClass()) return false; + WriteableBuild that = (WriteableBuild) o; + return build.equals(that.build); + } + + @Override + public int hashCode() { + return Objects.hash(build); + } + } + + private static String randomStringExcept(final String s) { + return randomAlphaOfLength(13 - s.length()); + } + + public void testSerialization() { + EqualsHashCodeTestUtils.checkEqualsAndHashCode(new WriteableBuild(new Build( + randomFrom(Build.Flavor.values()), randomFrom(Build.Type.values()), + randomAlphaOfLength(6), randomAlphaOfLength(6), randomBoolean(), randomAlphaOfLength(6))), + b -> copyWriteable(b, writableRegistry(), WriteableBuild::new, Version.CURRENT), + b -> { + switch (randomIntBetween(1, 6)) { + case 1: + return new WriteableBuild(new Build( + randomValueOtherThan(b.build.flavor(), () -> randomFrom(Build.Flavor.values())), b.build.type(), + b.build.shortHash(), b.build.date(), b.build.isSnapshot(), b.build.getQualifiedVersion())); + case 2: + return new WriteableBuild(new Build(b.build.flavor(), + randomValueOtherThan(b.build.type(), () -> randomFrom(Build.Type.values())), + b.build.shortHash(), b.build.date(), b.build.isSnapshot(), b.build.getQualifiedVersion())); + case 3: + return new WriteableBuild(new Build(b.build.flavor(), b.build.type(), + randomStringExcept(b.build.shortHash()), b.build.date(), b.build.isSnapshot(), b.build.getQualifiedVersion())); + case 4: + return new WriteableBuild(new Build(b.build.flavor(), b.build.type(), + b.build.shortHash(), randomStringExcept(b.build.date()), b.build.isSnapshot(), b.build.getQualifiedVersion())); + case 5: + return new WriteableBuild(new Build(b.build.flavor(), b.build.type(), + b.build.shortHash(), b.build.date(), b.build.isSnapshot() == false, b.build.getQualifiedVersion())); + case 6: + return new WriteableBuild(new Build(b.build.flavor(), b.build.type(), + b.build.shortHash(), b.build.date(), b.build.isSnapshot(), randomStringExcept(b.build.getQualifiedVersion()))); + } + throw new AssertionError(); + }); + } + + public void testSerializationBWC() throws IOException { + final WriteableBuild dockerBuild = new WriteableBuild(new Build(randomFrom(Build.Flavor.values()), Build.Type.DOCKER, + randomAlphaOfLength(6), randomAlphaOfLength(6), randomBoolean(), randomAlphaOfLength(6))); + + final List versions = Version.getDeclaredVersions(Version.class); + final Version pre63Version = randomFrom(versions.stream().filter(v -> v.before(Version.V_6_3_0)).collect(Collectors.toList())); + final Version post63Pre67Version = randomFrom(versions.stream() + .filter(v -> v.onOrAfter(Version.V_6_3_0) && v.before(Version.V_6_7_0)).collect(Collectors.toList())); + final Version post67Pre70Version = randomFrom(versions.stream() + .filter(v -> v.onOrAfter(Version.V_6_7_0) && v.before(Version.V_7_0_0)).collect(Collectors.toList())); + final Version post70Version = randomFrom(versions.stream().filter(v -> v.onOrAfter(Version.V_7_0_0)).collect(Collectors.toList())); + + final WriteableBuild pre63 = copyWriteable(dockerBuild, writableRegistry(), WriteableBuild::new, pre63Version); + final WriteableBuild post63pre67 = copyWriteable(dockerBuild, writableRegistry(), WriteableBuild::new, post63Pre67Version); + final WriteableBuild post67pre70 = copyWriteable(dockerBuild, writableRegistry(), WriteableBuild::new, post67Pre70Version); + final WriteableBuild post70 = copyWriteable(dockerBuild, writableRegistry(), WriteableBuild::new, post70Version); + + assertThat(pre63.build.flavor(), equalTo(Build.Flavor.OSS)); + assertThat(post63pre67.build.flavor(), equalTo(dockerBuild.build.flavor())); + assertThat(post67pre70.build.flavor(), equalTo(dockerBuild.build.flavor())); + assertThat(post70.build.flavor(), equalTo(dockerBuild.build.flavor())); + + assertThat(pre63.build.type(), equalTo(Build.Type.UNKNOWN)); + assertThat(post63pre67.build.type(), equalTo(Build.Type.TAR)); + assertThat(post67pre70.build.type(), equalTo(dockerBuild.build.type())); + assertThat(post70.build.type(), equalTo(dockerBuild.build.type())); + + assertThat(pre63.build.getQualifiedVersion(), equalTo(pre63Version.toString())); + assertThat(post63pre67.build.getQualifiedVersion(), equalTo(post63Pre67Version.toString())); + assertThat(post67pre70.build.getQualifiedVersion(), equalTo(post67Pre70Version.toString())); + assertThat(post70.build.getQualifiedVersion(), equalTo(dockerBuild.build.getQualifiedVersion())); + } } diff --git a/server/src/test/java/org/elasticsearch/cluster/metadata/MetaDataMappingServiceTests.java b/server/src/test/java/org/elasticsearch/cluster/metadata/MetaDataMappingServiceTests.java index d7e9767d7a14d..81fd5ec7fc83d 100644 --- a/server/src/test/java/org/elasticsearch/cluster/metadata/MetaDataMappingServiceTests.java +++ b/server/src/test/java/org/elasticsearch/cluster/metadata/MetaDataMappingServiceTests.java @@ -26,6 +26,7 @@ import org.elasticsearch.common.compress.CompressedXContent; import org.elasticsearch.index.Index; import org.elasticsearch.index.IndexService; +import org.elasticsearch.index.mapper.MapperService; import org.elasticsearch.plugins.Plugin; import org.elasticsearch.test.ESSingleNodeTestCase; import org.elasticsearch.test.InternalSettingsPlugin; @@ -114,4 +115,23 @@ public void testMappingVersionUnchanged() throws Exception { assertThat(result.resultingState.metaData().index("test").getMappingVersion(), equalTo(previousVersion)); } + public void testMappingUpdateAccepts_docAsType() throws Exception { + final IndexService indexService = createIndex("test", + client().admin().indices().prepareCreate("test").addMapping("my_type")); + final MetaDataMappingService mappingService = getInstanceFromNode(MetaDataMappingService.class); + final ClusterService clusterService = getInstanceFromNode(ClusterService.class); + final PutMappingClusterStateUpdateRequest request = new PutMappingClusterStateUpdateRequest() + .type(MapperService.SINGLE_MAPPING_NAME); + request.indices(new Index[] {indexService.index()}); + request.source("{ \"properties\": { \"foo\": { \"type\": \"keyword\" } }}"); + final ClusterStateTaskExecutor.ClusterTasksResult result = + mappingService.putMappingExecutor.execute(clusterService.state(), Collections.singletonList(request)); + assertThat(result.executionResults.size(), equalTo(1)); + assertTrue(result.executionResults.values().iterator().next().isSuccess()); + MappingMetaData mappingMetaData = result.resultingState.metaData().index("test").mapping(); + assertEquals("my_type", mappingMetaData.type()); + assertEquals(Collections.singletonMap("properties", + Collections.singletonMap("foo", + Collections.singletonMap("type", "keyword"))), mappingMetaData.sourceAsMap()); + } } diff --git a/server/src/test/java/org/elasticsearch/common/joda/JavaJodaTimeDuellingTests.java b/server/src/test/java/org/elasticsearch/common/joda/JavaJodaTimeDuellingTests.java index 5798b5f799203..c3a541fe87ec2 100644 --- a/server/src/test/java/org/elasticsearch/common/joda/JavaJodaTimeDuellingTests.java +++ b/server/src/test/java/org/elasticsearch/common/joda/JavaJodaTimeDuellingTests.java @@ -100,6 +100,7 @@ public void testDuellingFormatsValidParsing() { assertSameDate("20181126T121212+01:00", "basic_date_time_no_millis"); assertSameDate("20181126T121212+0100", "basic_date_time_no_millis"); assertSameDate("2018363", "basic_ordinal_date"); + assertSameDate("2018363T121212.1Z", "basic_ordinal_date_time"); assertSameDate("2018363T121212.123Z", "basic_ordinal_date_time"); assertSameDate("2018363T121212.123456789Z", "basic_ordinal_date_time"); assertSameDate("2018363T121212.123+0100", "basic_ordinal_date_time"); @@ -107,15 +108,19 @@ public void testDuellingFormatsValidParsing() { assertSameDate("2018363T121212Z", "basic_ordinal_date_time_no_millis"); assertSameDate("2018363T121212+0100", "basic_ordinal_date_time_no_millis"); assertSameDate("2018363T121212+01:00", "basic_ordinal_date_time_no_millis"); + assertSameDate("121212.1Z", "basic_time"); assertSameDate("121212.123Z", "basic_time"); assertSameDate("121212.123456789Z", "basic_time"); + assertSameDate("121212.1+0100", "basic_time"); assertSameDate("121212.123+0100", "basic_time"); assertSameDate("121212.123+01:00", "basic_time"); assertSameDate("121212Z", "basic_time_no_millis"); assertSameDate("121212+0100", "basic_time_no_millis"); assertSameDate("121212+01:00", "basic_time_no_millis"); + assertSameDate("T121212.1Z", "basic_t_time"); assertSameDate("T121212.123Z", "basic_t_time"); assertSameDate("T121212.123456789Z", "basic_t_time"); + assertSameDate("T121212.1+0100", "basic_t_time"); assertSameDate("T121212.123+0100", "basic_t_time"); assertSameDate("T121212.123+01:00", "basic_t_time"); assertSameDate("T121212Z", "basic_t_time_no_millis"); @@ -124,6 +129,7 @@ public void testDuellingFormatsValidParsing() { assertSameDate("2018W313", "basic_week_date"); assertSameDate("1W313", "basic_week_date"); assertSameDate("18W313", "basic_week_date"); + assertSameDate("2018W313T121212.1Z", "basic_week_date_time"); assertSameDate("2018W313T121212.123Z", "basic_week_date_time"); assertSameDate("2018W313T121212.123456789Z", "basic_week_date_time"); assertSameDate("2018W313T121212.123+0100", "basic_week_date_time"); @@ -145,8 +151,10 @@ public void testDuellingFormatsValidParsing() { assertSameDate("2018-12-31T12:12:12", "date_hour_minute_second"); assertSameDate("2018-12-31T12:12:1", "date_hour_minute_second"); + assertSameDate("2018-12-31T12:12:12.1", "date_hour_minute_second_fraction"); assertSameDate("2018-12-31T12:12:12.123", "date_hour_minute_second_fraction"); assertSameDate("2018-12-31T12:12:12.123456789", "date_hour_minute_second_fraction"); + assertSameDate("2018-12-31T12:12:12.1", "date_hour_minute_second_millis"); assertSameDate("2018-12-31T12:12:12.123", "date_hour_minute_second_millis"); assertParseException("2018-12-31T12:12:12.123456789", "date_hour_minute_second_millis"); assertSameDate("2018-12-31T12:12:12.1", "date_hour_minute_second_millis"); @@ -157,11 +165,14 @@ public void testDuellingFormatsValidParsing() { assertSameDate("2018-05-30T20", "date_optional_time"); assertSameDate("2018-05-30T20:21", "date_optional_time"); assertSameDate("2018-05-30T20:21:23", "date_optional_time"); + assertSameDate("2018-05-30T20:21:23.1", "date_optional_time"); assertSameDate("2018-05-30T20:21:23.123", "date_optional_time"); assertSameDate("2018-05-30T20:21:23.123456789", "date_optional_time"); assertSameDate("2018-05-30T20:21:23.123Z", "date_optional_time"); assertSameDate("2018-05-30T20:21:23.123456789Z", "date_optional_time"); + assertSameDate("2018-05-30T20:21:23.1+0100", "date_optional_time"); assertSameDate("2018-05-30T20:21:23.123+0100", "date_optional_time"); + assertSameDate("2018-05-30T20:21:23.1+01:00", "date_optional_time"); assertSameDate("2018-05-30T20:21:23.123+01:00", "date_optional_time"); assertSameDate("2018-12-1", "date_optional_time"); assertSameDate("2018-12-31T10:15:30", "date_optional_time"); @@ -169,17 +180,23 @@ public void testDuellingFormatsValidParsing() { assertSameDate("2018-12-31T10:5:30", "date_optional_time"); assertSameDate("2018-12-31T1:15:30", "date_optional_time"); + assertSameDate("2018-12-31T10:15:30.1Z", "date_time"); assertSameDate("2018-12-31T10:15:30.123Z", "date_time"); assertSameDate("2018-12-31T10:15:30.123456789Z", "date_time"); + assertSameDate("2018-12-31T10:15:30.1+0100", "date_time"); assertSameDate("2018-12-31T10:15:30.123+0100", "date_time"); assertSameDate("2018-12-31T10:15:30.123+01:00", "date_time"); + assertSameDate("2018-12-31T10:15:30.1+01:00", "date_time"); assertSameDate("2018-12-31T10:15:30.11Z", "date_time"); assertSameDate("2018-12-31T10:15:30.11+0100", "date_time"); assertSameDate("2018-12-31T10:15:30.11+01:00", "date_time"); + assertSameDate("2018-12-31T10:15:3.1Z", "date_time"); assertSameDate("2018-12-31T10:15:3.123Z", "date_time"); assertSameDate("2018-12-31T10:15:3.123456789Z", "date_time"); + assertSameDate("2018-12-31T10:15:3.1+0100", "date_time"); assertSameDate("2018-12-31T10:15:3.123+0100", "date_time"); assertSameDate("2018-12-31T10:15:3.123+01:00", "date_time"); + assertSameDate("2018-12-31T10:15:3.1+01:00", "date_time"); assertSameDate("2018-12-31T10:15:30Z", "date_time_no_millis"); assertSameDate("2018-12-31T10:15:30+0100", "date_time_no_millis"); @@ -218,10 +235,12 @@ public void testDuellingFormatsValidParsing() { assertSameDate("2018-128", "ordinal_date"); assertSameDate("2018-1", "ordinal_date"); + assertSameDate("2018-128T10:15:30.1Z", "ordinal_date_time"); assertSameDate("2018-128T10:15:30.123Z", "ordinal_date_time"); assertSameDate("2018-128T10:15:30.123456789Z", "ordinal_date_time"); assertSameDate("2018-128T10:15:30.123+0100", "ordinal_date_time"); assertSameDate("2018-128T10:15:30.123+01:00", "ordinal_date_time"); + assertSameDate("2018-1T10:15:30.1Z", "ordinal_date_time"); assertSameDate("2018-1T10:15:30.123Z", "ordinal_date_time"); assertSameDate("2018-1T10:15:30.123456789Z", "ordinal_date_time"); assertSameDate("2018-1T10:15:30.123+0100", "ordinal_date_time"); @@ -234,16 +253,20 @@ public void testDuellingFormatsValidParsing() { assertSameDate("2018-1T10:15:30+0100", "ordinal_date_time_no_millis"); assertSameDate("2018-1T10:15:30+01:00", "ordinal_date_time_no_millis"); + assertSameDate("10:15:30.1Z", "time"); assertSameDate("10:15:30.123Z", "time"); assertSameDate("10:15:30.123456789Z", "time"); assertSameDate("10:15:30.123+0100", "time"); assertSameDate("10:15:30.123+01:00", "time"); + assertSameDate("1:15:30.1Z", "time"); assertSameDate("1:15:30.123Z", "time"); assertSameDate("1:15:30.123+0100", "time"); assertSameDate("1:15:30.123+01:00", "time"); + assertSameDate("10:1:30.1Z", "time"); assertSameDate("10:1:30.123Z", "time"); assertSameDate("10:1:30.123+0100", "time"); assertSameDate("10:1:30.123+01:00", "time"); + assertSameDate("10:15:3.1Z", "time"); assertSameDate("10:15:3.123Z", "time"); assertSameDate("10:15:3.123+0100", "time"); assertSameDate("10:15:3.123+01:00", "time"); @@ -267,10 +290,13 @@ public void testDuellingFormatsValidParsing() { assertSameDate("10:15:3+01:00", "time_no_millis"); assertParseException("10:15:3", "time_no_millis"); + assertSameDate("T10:15:30.1Z", "t_time"); assertSameDate("T10:15:30.123Z", "t_time"); assertSameDate("T10:15:30.123456789Z", "t_time"); + assertSameDate("T10:15:30.1+0100", "t_time"); assertSameDate("T10:15:30.123+0100", "t_time"); assertSameDate("T10:15:30.123+01:00", "t_time"); + assertSameDate("T10:15:30.1+01:00", "t_time"); assertSameDate("T1:15:30.123Z", "t_time"); assertSameDate("T1:15:30.123+0100", "t_time"); assertSameDate("T1:15:30.123+01:00", "t_time"); @@ -305,12 +331,18 @@ public void testDuellingFormatsValidParsing() { "Cannot parse \"2012-W1-8\": Value 8 for dayOfWeek must be in the range [1,7]"); assertJavaTimeParseException("2012-W1-8", "week_date"); + assertSameDate("2012-W48-6T10:15:30.1Z", "week_date_time"); assertSameDate("2012-W48-6T10:15:30.123Z", "week_date_time"); assertSameDate("2012-W48-6T10:15:30.123456789Z", "week_date_time"); + assertSameDate("2012-W48-6T10:15:30.1+0100", "week_date_time"); assertSameDate("2012-W48-6T10:15:30.123+0100", "week_date_time"); + assertSameDate("2012-W48-6T10:15:30.1+01:00", "week_date_time"); assertSameDate("2012-W48-6T10:15:30.123+01:00", "week_date_time"); + assertSameDate("2012-W1-6T10:15:30.1Z", "week_date_time"); assertSameDate("2012-W1-6T10:15:30.123Z", "week_date_time"); + assertSameDate("2012-W1-6T10:15:30.1+0100", "week_date_time"); assertSameDate("2012-W1-6T10:15:30.123+0100", "week_date_time"); + assertSameDate("2012-W1-6T10:15:30.1+01:00", "week_date_time"); assertSameDate("2012-W1-6T10:15:30.123+01:00", "week_date_time"); assertSameDate("2012-W48-6T10:15:30Z", "week_date_time_no_millis"); @@ -357,9 +389,12 @@ public void testExceptionWhenCompositeParsingFails(){ public void testDuelingStrictParsing() { assertSameDate("2018W313", "strict_basic_week_date"); assertParseException("18W313", "strict_basic_week_date"); + assertSameDate("2018W313T121212.1Z", "strict_basic_week_date_time"); assertSameDate("2018W313T121212.123Z", "strict_basic_week_date_time"); assertSameDate("2018W313T121212.123456789Z", "strict_basic_week_date_time"); + assertSameDate("2018W313T121212.1+0100", "strict_basic_week_date_time"); assertSameDate("2018W313T121212.123+0100", "strict_basic_week_date_time"); + assertSameDate("2018W313T121212.1+01:00", "strict_basic_week_date_time"); assertSameDate("2018W313T121212.123+01:00", "strict_basic_week_date_time"); assertParseException("2018W313T12128.123Z", "strict_basic_week_date_time"); assertParseException("2018W313T12128.123456789Z", "strict_basic_week_date_time"); @@ -387,6 +422,7 @@ public void testDuelingStrictParsing() { assertParseException("2018-12-31T8:3", "strict_date_hour_minute"); assertSameDate("2018-12-31T12:12:12", "strict_date_hour_minute_second"); assertParseException("2018-12-31T12:12:1", "strict_date_hour_minute_second"); + assertSameDate("2018-12-31T12:12:12.1", "strict_date_hour_minute_second_fraction"); assertSameDate("2018-12-31T12:12:12.123", "strict_date_hour_minute_second_fraction"); assertSameDate("2018-12-31T12:12:12.123456789", "strict_date_hour_minute_second_fraction"); assertSameDate("2018-12-31T12:12:12.123", "strict_date_hour_minute_second_millis"); @@ -407,9 +443,12 @@ public void testDuelingStrictParsing() { assertParseException("2018-12-31T10:5:30", "strict_date_optional_time"); assertParseException("2018-12-31T9:15:30", "strict_date_optional_time"); assertSameDate("2015-01-04T00:00Z", "strict_date_optional_time"); + assertSameDate("2018-12-31T10:15:30.1Z", "strict_date_time"); assertSameDate("2018-12-31T10:15:30.123Z", "strict_date_time"); assertSameDate("2018-12-31T10:15:30.123456789Z", "strict_date_time"); + assertSameDate("2018-12-31T10:15:30.1+0100", "strict_date_time"); assertSameDate("2018-12-31T10:15:30.123+0100", "strict_date_time"); + assertSameDate("2018-12-31T10:15:30.1+01:00", "strict_date_time"); assertSameDate("2018-12-31T10:15:30.123+01:00", "strict_date_time"); assertSameDate("2018-12-31T10:15:30.11Z", "strict_date_time"); assertSameDate("2018-12-31T10:15:30.11+0100", "strict_date_time"); @@ -442,9 +481,12 @@ public void testDuelingStrictParsing() { assertSameDate("2018-128", "strict_ordinal_date"); assertParseException("2018-1", "strict_ordinal_date"); + assertSameDate("2018-128T10:15:30.1Z", "strict_ordinal_date_time"); assertSameDate("2018-128T10:15:30.123Z", "strict_ordinal_date_time"); assertSameDate("2018-128T10:15:30.123456789Z", "strict_ordinal_date_time"); + assertSameDate("2018-128T10:15:30.1+0100", "strict_ordinal_date_time"); assertSameDate("2018-128T10:15:30.123+0100", "strict_ordinal_date_time"); + assertSameDate("2018-128T10:15:30.1+01:00", "strict_ordinal_date_time"); assertSameDate("2018-128T10:15:30.123+01:00", "strict_ordinal_date_time"); assertParseException("2018-1T10:15:30.123Z", "strict_ordinal_date_time"); @@ -453,6 +495,7 @@ public void testDuelingStrictParsing() { assertSameDate("2018-128T10:15:30+01:00", "strict_ordinal_date_time_no_millis"); assertParseException("2018-1T10:15:30Z", "strict_ordinal_date_time_no_millis"); + assertSameDate("10:15:30.1Z", "strict_time"); assertSameDate("10:15:30.123Z", "strict_time"); assertSameDate("10:15:30.123456789Z", "strict_time"); assertSameDate("10:15:30.123+0100", "strict_time"); @@ -474,9 +517,12 @@ public void testDuelingStrictParsing() { assertParseException("10:15:3Z", "strict_time_no_millis"); assertParseException("10:15:3", "strict_time_no_millis"); + assertSameDate("T10:15:30.1Z", "strict_t_time"); assertSameDate("T10:15:30.123Z", "strict_t_time"); assertSameDate("T10:15:30.123456789Z", "strict_t_time"); + assertSameDate("T10:15:30.1+0100", "strict_t_time"); assertSameDate("T10:15:30.123+0100", "strict_t_time"); + assertSameDate("T10:15:30.1+01:00", "strict_t_time"); assertSameDate("T10:15:30.123+01:00", "strict_t_time"); assertParseException("T1:15:30.123Z", "strict_t_time"); assertParseException("T10:1:30.123Z", "strict_t_time"); @@ -505,9 +551,12 @@ public void testDuelingStrictParsing() { "Cannot parse \"2012-W01-8\": Value 8 for dayOfWeek must be in the range [1,7]"); assertJavaTimeParseException("2012-W01-8", "strict_week_date"); + assertSameDate("2012-W48-6T10:15:30.1Z", "strict_week_date_time"); assertSameDate("2012-W48-6T10:15:30.123Z", "strict_week_date_time"); assertSameDate("2012-W48-6T10:15:30.123456789Z", "strict_week_date_time"); + assertSameDate("2012-W48-6T10:15:30.1+0100", "strict_week_date_time"); assertSameDate("2012-W48-6T10:15:30.123+0100", "strict_week_date_time"); + assertSameDate("2012-W48-6T10:15:30.1+01:00", "strict_week_date_time"); assertSameDate("2012-W48-6T10:15:30.123+01:00", "strict_week_date_time"); assertParseException("2012-W1-6T10:15:30.123Z", "strict_week_date_time"); diff --git a/server/src/test/java/org/elasticsearch/index/shard/IndexShardTests.java b/server/src/test/java/org/elasticsearch/index/shard/IndexShardTests.java index 67106b04f8deb..da474e8d770b2 100644 --- a/server/src/test/java/org/elasticsearch/index/shard/IndexShardTests.java +++ b/server/src/test/java/org/elasticsearch/index/shard/IndexShardTests.java @@ -2767,7 +2767,7 @@ public void testCompletionStatsMarksSearcherAccessed() throws Exception { }); long prevAccessTime = shard.getLastSearcherAccess(); indexShard.completionStats(); - assertThat("searcher was not marked as accessed", shard.getLastSearcherAccess(), greaterThan(prevAccessTime)); + assertThat("searcher was marked as accessed", shard.getLastSearcherAccess(), equalTo(prevAccessTime)); } finally { closeShards(indexShard); } @@ -2797,7 +2797,7 @@ public void testDocStats() throws Exception { }); long prevAccessTime = shard.getLastSearcherAccess(); final DocsStats docsStats = indexShard.docStats(); - assertThat("searcher was not marked as accessed", shard.getLastSearcherAccess(), greaterThan(prevAccessTime)); + assertThat("searcher was marked as accessed", shard.getLastSearcherAccess(), equalTo(prevAccessTime)); assertThat(docsStats.getCount(), equalTo(numDocs)); try (Engine.Searcher searcher = indexShard.acquireSearcher("test")) { assertTrue(searcher.reader().numDocs() <= docsStats.getCount()); diff --git a/x-pack/plugin/core/src/main/java/org/elasticsearch/xpack/core/dataframe/DataFrameField.java b/x-pack/plugin/core/src/main/java/org/elasticsearch/xpack/core/dataframe/DataFrameField.java index 73e639cec5e1d..154bd206ade66 100644 --- a/x-pack/plugin/core/src/main/java/org/elasticsearch/xpack/core/dataframe/DataFrameField.java +++ b/x-pack/plugin/core/src/main/java/org/elasticsearch/xpack/core/dataframe/DataFrameField.java @@ -33,7 +33,7 @@ public final class DataFrameField { public static final String REST_BASE_PATH = "/_data_frame/"; public static final String REST_BASE_PATH_TRANSFORMS = REST_BASE_PATH + "transforms/"; public static final String REST_BASE_PATH_TRANSFORMS_BY_ID = REST_BASE_PATH_TRANSFORMS + "{id}/"; - public static final String DATA_FRAME_TRANSFORM_AUDIT_ID_FIELD = "transform_id"; + public static final String TRANSFORM_ID = "transform_id"; // note: this is used to match tasks public static final String PERSISTENT_TASK_DESCRIPTION_PREFIX = "data_frame_"; diff --git a/x-pack/plugin/core/src/main/java/org/elasticsearch/xpack/core/dataframe/DataFrameMessages.java b/x-pack/plugin/core/src/main/java/org/elasticsearch/xpack/core/dataframe/DataFrameMessages.java index 7fd633a1e9d0e..dbe789ca3aebf 100644 --- a/x-pack/plugin/core/src/main/java/org/elasticsearch/xpack/core/dataframe/DataFrameMessages.java +++ b/x-pack/plugin/core/src/main/java/org/elasticsearch/xpack/core/dataframe/DataFrameMessages.java @@ -29,12 +29,16 @@ public class DataFrameMessages { public static final String DATA_FRAME_CONFIG_INVALID = "Data frame transform configuration is invalid [{0}]"; public static final String REST_DATA_FRAME_FAILED_TO_SERIALIZE_TRANSFORM = "Failed to serialise transform [{0}]"; + public static final String DATA_FRAME_FAILED_TO_PERSIST_STATS = "Failed to persist data frame statistics for transform [{0}]"; + public static final String DATA_FRAME_UNKNOWN_TRANSFORM_STATS = "Statistics for transform [{0}] could not be found"; public static final String FAILED_TO_CREATE_DESTINATION_INDEX = "Could not create destination index [{0}] for transform [{1}]"; public static final String FAILED_TO_LOAD_TRANSFORM_CONFIGURATION = "Failed to load data frame transform configuration for transform [{0}]"; public static final String FAILED_TO_PARSE_TRANSFORM_CONFIGURATION = "Failed to parse transform configuration for data frame transform [{0}]"; + public static final String FAILED_TO_PARSE_TRANSFORM_STATISTICS_CONFIGURATION = + "Failed to parse transform statistics for data frame transform [{0}]"; public static final String DATA_FRAME_TRANSFORM_CONFIGURATION_NO_TRANSFORM = "Data frame transform configuration must specify exactly 1 function"; public static final String DATA_FRAME_TRANSFORM_CONFIGURATION_PIVOT_NO_GROUP_BY = diff --git a/x-pack/plugin/core/src/main/java/org/elasticsearch/xpack/core/dataframe/action/GetDataFrameTransformsStatsAction.java b/x-pack/plugin/core/src/main/java/org/elasticsearch/xpack/core/dataframe/action/GetDataFrameTransformsStatsAction.java index f0e92aa36db2f..d91f7a1a06964 100644 --- a/x-pack/plugin/core/src/main/java/org/elasticsearch/xpack/core/dataframe/action/GetDataFrameTransformsStatsAction.java +++ b/x-pack/plugin/core/src/main/java/org/elasticsearch/xpack/core/dataframe/action/GetDataFrameTransformsStatsAction.java @@ -6,9 +6,9 @@ package org.elasticsearch.xpack.core.dataframe.action; +import org.elasticsearch.ElasticsearchException; import org.elasticsearch.action.Action; import org.elasticsearch.action.ActionRequestValidationException; -import org.elasticsearch.action.FailedNodeException; import org.elasticsearch.action.TaskOperationFailure; import org.elasticsearch.action.support.tasks.BaseTasksRequest; import org.elasticsearch.action.support.tasks.BaseTasksResponse; @@ -20,14 +20,18 @@ import org.elasticsearch.common.xcontent.ToXContentObject; import org.elasticsearch.common.xcontent.XContentBuilder; import org.elasticsearch.tasks.Task; +import org.elasticsearch.xpack.core.action.util.PageParams; import org.elasticsearch.xpack.core.dataframe.DataFrameField; import org.elasticsearch.xpack.core.dataframe.transforms.DataFrameTransformStateAndStats; import java.io.IOException; +import java.util.ArrayList; import java.util.Collections; import java.util.List; import java.util.Objects; +import static org.elasticsearch.action.ValidateActions.addValidationError; + public class GetDataFrameTransformsStatsAction extends Action { public static final GetDataFrameTransformsStatsAction INSTANCE = new GetDataFrameTransformsStatsAction(); @@ -43,6 +47,11 @@ public Response newResponse() { public static class Request extends BaseTasksRequest { private String id; + private PageParams pageParams = PageParams.defaultParams(); + + public static final int MAX_SIZE_RETURN = 1000; + // used internally to expand the queried id expression + private List expandedIds = Collections.emptyList(); public Request(String id) { if (Strings.isNullOrEmpty(id) || id.equals("*")) { @@ -55,36 +64,58 @@ public Request(String id) { public Request(StreamInput in) throws IOException { super(in); id = in.readString(); + expandedIds = in.readList(StreamInput::readString); + pageParams = in.readOptionalWriteable(PageParams::new); } @Override public boolean match(Task task) { - // If we are retrieving all the transforms, the task description does not contain the id - if (id.equals(MetaData.ALL)) { - return task.getDescription().startsWith(DataFrameField.PERSISTENT_TASK_DESCRIPTION_PREFIX); - } - // Otherwise find the task by ID - return task.getDescription().equals(DataFrameField.PERSISTENT_TASK_DESCRIPTION_PREFIX + id); + // Only get tasks that we have expanded to + return expandedIds.stream() + .anyMatch(transformId -> task.getDescription().equals(DataFrameField.PERSISTENT_TASK_DESCRIPTION_PREFIX + transformId)); } public String getId() { return id; } + public List getExpandedIds() { + return expandedIds; + } + + public void setExpandedIds(List expandedIds) { + this.expandedIds = Collections.unmodifiableList(new ArrayList<>(expandedIds)); + } + + public final void setPageParams(PageParams pageParams) { + this.pageParams = pageParams; + } + + public final PageParams getPageParams() { + return pageParams; + } + @Override public void writeTo(StreamOutput out) throws IOException { super.writeTo(out); out.writeString(id); + out.writeStringCollection(expandedIds); + out.writeOptionalWriteable(pageParams); } @Override public ActionRequestValidationException validate() { - return null; + ActionRequestValidationException exception = null; + if (getPageParams() != null && getPageParams().getSize() > MAX_SIZE_RETURN) { + exception = addValidationError("Param [" + PageParams.SIZE.getPreferredName() + + "] has a max acceptable value of [" + MAX_SIZE_RETURN + "]", exception); + } + return exception; } @Override public int hashCode() { - return Objects.hash(id); + return Objects.hash(id, pageParams); } @Override @@ -96,7 +127,7 @@ public boolean equals(Object obj) { return false; } Request other = (Request) obj; - return Objects.equals(id, other.id); + return Objects.equals(id, other.id) && Objects.equals(pageParams, other.pageParams); } } @@ -109,7 +140,7 @@ public Response(List transformsStateAndStats) { } public Response(List transformsStateAndStats, List taskFailures, - List nodeFailures) { + List nodeFailures) { super(taskFailures, nodeFailures); this.transformsStateAndStats = transformsStateAndStats; } diff --git a/x-pack/plugin/core/src/main/java/org/elasticsearch/xpack/core/dataframe/notifications/DataFrameAuditMessage.java b/x-pack/plugin/core/src/main/java/org/elasticsearch/xpack/core/dataframe/notifications/DataFrameAuditMessage.java index 7dab9be6ab3cc..e6ac6cbc57b15 100644 --- a/x-pack/plugin/core/src/main/java/org/elasticsearch/xpack/core/dataframe/notifications/DataFrameAuditMessage.java +++ b/x-pack/plugin/core/src/main/java/org/elasticsearch/xpack/core/dataframe/notifications/DataFrameAuditMessage.java @@ -11,17 +11,17 @@ import org.elasticsearch.common.xcontent.XContentParser; import org.elasticsearch.xpack.core.common.notifications.AbstractAuditMessage; import org.elasticsearch.xpack.core.common.notifications.Level; +import org.elasticsearch.xpack.core.dataframe.DataFrameField; import org.elasticsearch.xpack.core.ml.utils.time.TimeUtils; import java.util.Date; import static org.elasticsearch.common.xcontent.ConstructingObjectParser.constructorArg; import static org.elasticsearch.common.xcontent.ConstructingObjectParser.optionalConstructorArg; -import static org.elasticsearch.xpack.core.dataframe.DataFrameField.DATA_FRAME_TRANSFORM_AUDIT_ID_FIELD; public class DataFrameAuditMessage extends AbstractAuditMessage { - private static final ParseField TRANSFORM_ID = new ParseField(DATA_FRAME_TRANSFORM_AUDIT_ID_FIELD); + private static final ParseField TRANSFORM_ID = new ParseField(DataFrameField.TRANSFORM_ID); public static final ConstructingObjectParser PARSER = new ConstructingObjectParser<>( "data_frame_audit_message", true, diff --git a/x-pack/plugin/core/src/main/java/org/elasticsearch/xpack/core/dataframe/transforms/DataFrameIndexerTransformStats.java b/x-pack/plugin/core/src/main/java/org/elasticsearch/xpack/core/dataframe/transforms/DataFrameIndexerTransformStats.java index 9bb654b31d702..05cb6dea33ed9 100644 --- a/x-pack/plugin/core/src/main/java/org/elasticsearch/xpack/core/dataframe/transforms/DataFrameIndexerTransformStats.java +++ b/x-pack/plugin/core/src/main/java/org/elasticsearch/xpack/core/dataframe/transforms/DataFrameIndexerTransformStats.java @@ -6,35 +6,43 @@ package org.elasticsearch.xpack.core.dataframe.transforms; +import org.elasticsearch.common.Nullable; import org.elasticsearch.common.ParseField; import org.elasticsearch.common.io.stream.StreamInput; +import org.elasticsearch.common.io.stream.StreamOutput; import org.elasticsearch.common.xcontent.ConstructingObjectParser; import org.elasticsearch.common.xcontent.XContentBuilder; import org.elasticsearch.common.xcontent.XContentParser; +import org.elasticsearch.xpack.core.dataframe.DataFrameField; import org.elasticsearch.xpack.core.indexing.IndexerJobStats; import java.io.IOException; +import java.util.Objects; import static org.elasticsearch.common.xcontent.ConstructingObjectParser.constructorArg; +import static org.elasticsearch.common.xcontent.ConstructingObjectParser.optionalConstructorArg; public class DataFrameIndexerTransformStats extends IndexerJobStats { - private static final String NAME = "data_frame_indexer_transform_stats"; - private static ParseField NUM_PAGES = new ParseField("pages_processed"); - private static ParseField NUM_INPUT_DOCUMENTS = new ParseField("documents_processed"); - private static ParseField NUM_OUTPUT_DOCUMENTS = new ParseField("documents_indexed"); - private static ParseField NUM_INVOCATIONS = new ParseField("trigger_count"); - private static ParseField INDEX_TIME_IN_MS = new ParseField("index_time_in_ms"); - private static ParseField SEARCH_TIME_IN_MS = new ParseField("search_time_in_ms"); - private static ParseField INDEX_TOTAL = new ParseField("index_total"); - private static ParseField SEARCH_TOTAL = new ParseField("search_total"); - private static ParseField SEARCH_FAILURES = new ParseField("search_failures"); - private static ParseField INDEX_FAILURES = new ParseField("index_failures"); + private static final String DEFAULT_TRANSFORM_ID = "_all"; + + public static final String NAME = "data_frame_indexer_transform_stats"; + public static ParseField NUM_PAGES = new ParseField("pages_processed"); + public static ParseField NUM_INPUT_DOCUMENTS = new ParseField("documents_processed"); + public static ParseField NUM_OUTPUT_DOCUMENTS = new ParseField("documents_indexed"); + public static ParseField NUM_INVOCATIONS = new ParseField("trigger_count"); + public static ParseField INDEX_TIME_IN_MS = new ParseField("index_time_in_ms"); + public static ParseField SEARCH_TIME_IN_MS = new ParseField("search_time_in_ms"); + public static ParseField INDEX_TOTAL = new ParseField("index_total"); + public static ParseField SEARCH_TOTAL = new ParseField("search_total"); + public static ParseField SEARCH_FAILURES = new ParseField("search_failures"); + public static ParseField INDEX_FAILURES = new ParseField("index_failures"); public static final ConstructingObjectParser PARSER = new ConstructingObjectParser<>( - NAME, args -> new DataFrameIndexerTransformStats((long) args[0], (long) args[1], (long) args[2], - (long) args[3], (long) args[4], (long) args[5], (long) args[6], (long) args[7], (long) args[8], (long) args[9])); + NAME, args -> new DataFrameIndexerTransformStats((String) args[0], (long) args[1], (long) args[2], (long) args[3], + (long) args[4], (long) args[5], (long) args[6], (long) args[7], (long) args[8], (long) args[9], (long) args[10])); static { + PARSER.declareString(optionalConstructorArg(), DataFrameField.ID); PARSER.declareLong(constructorArg(), NUM_PAGES); PARSER.declareLong(constructorArg(), NUM_INPUT_DOCUMENTS); PARSER.declareLong(constructorArg(), NUM_OUTPUT_DOCUMENTS); @@ -45,20 +53,72 @@ public class DataFrameIndexerTransformStats extends IndexerJobStats { PARSER.declareLong(constructorArg(), SEARCH_TOTAL); PARSER.declareLong(constructorArg(), INDEX_FAILURES); PARSER.declareLong(constructorArg(), SEARCH_FAILURES); + PARSER.declareString(optionalConstructorArg(), DataFrameField.INDEX_DOC_TYPE); + } + + private final String transformId; + + /** + * Certain situations call for a default transform ID, e.g. when merging many different transforms for statistics gather. + * + * The returned stats object cannot be stored in the index as the transformId does not refer to a real transform configuration + * + * @return new DataFrameIndexerTransformStats with empty stats and a default transform ID + */ + public static DataFrameIndexerTransformStats withDefaultTransformId() { + return new DataFrameIndexerTransformStats(DEFAULT_TRANSFORM_ID); } - public DataFrameIndexerTransformStats() { + public static DataFrameIndexerTransformStats withDefaultTransformId(long numPages, long numInputDocuments, long numOutputDocuments, + long numInvocations, long indexTime, long searchTime, + long indexTotal, long searchTotal, long indexFailures, + long searchFailures) { + return new DataFrameIndexerTransformStats(DEFAULT_TRANSFORM_ID, numPages, numInputDocuments, + numOutputDocuments, numInvocations, indexTime, searchTime, indexTotal, searchTotal, + indexFailures, searchFailures); + } + + public DataFrameIndexerTransformStats(String transformId) { super(); + this.transformId = Objects.requireNonNull(transformId, "parameter transformId must not be null"); + } + + public DataFrameIndexerTransformStats(String transformId, long numPages, long numInputDocuments, long numOutputDocuments, + long numInvocations, long indexTime, long searchTime, long indexTotal, long searchTotal, + long indexFailures, long searchFailures) { + super(numPages, numInputDocuments, numOutputDocuments, numInvocations, indexTime, searchTime, indexTotal, searchTotal, + indexFailures, searchFailures); + this.transformId = Objects.requireNonNull(transformId, "parameter transformId must not be null"); } - public DataFrameIndexerTransformStats(long numPages, long numInputDocuments, long numOuputDocuments, long numInvocations, - long indexTime, long searchTime, long indexTotal, long searchTotal, long indexFailures, long searchFailures) { - super(numPages, numInputDocuments, numOuputDocuments, numInvocations, indexTime, searchTime, indexTotal, searchTotal, indexFailures, - searchFailures); + public DataFrameIndexerTransformStats(DataFrameIndexerTransformStats other) { + this(other.transformId, other.numPages, other.numInputDocuments, other.numOuputDocuments, other.numInvocations, + other.indexTime, other.searchTime, other.indexTotal, other.searchTotal, other.indexFailures, other.searchFailures); } public DataFrameIndexerTransformStats(StreamInput in) throws IOException { super(in); + transformId = in.readString(); + } + + @Override + public void writeTo(StreamOutput out) throws IOException { + super.writeTo(out); + out.writeString(transformId); + } + + /** + * Get the persisted stats document name from the Data Frame Transformer Id. + * + * @return The id of document the where the transform stats are persisted + */ + public static String documentId(String transformId) { + return NAME + "-" + transformId; + } + + @Nullable + public String getTransformId() { + return transformId; } @Override @@ -74,11 +134,22 @@ public XContentBuilder toXContent(XContentBuilder builder, Params params) throws builder.field(SEARCH_TIME_IN_MS.getPreferredName(), searchTime); builder.field(SEARCH_TOTAL.getPreferredName(), searchTotal); builder.field(SEARCH_FAILURES.getPreferredName(), searchFailures); + if (params.paramAsBoolean(DataFrameField.FOR_INTERNAL_STORAGE, false)) { + // If we are storing something, it should have a valid transform ID. + if (transformId.equals(DEFAULT_TRANSFORM_ID)) { + throw new IllegalArgumentException("when storing transform statistics, a valid transform id must be provided"); + } + builder.field(DataFrameField.ID.getPreferredName(), transformId); + builder.field(DataFrameField.INDEX_DOC_TYPE.getPreferredName(), NAME); + } builder.endObject(); return builder; } public DataFrameIndexerTransformStats merge(DataFrameIndexerTransformStats other) { + // We should probably not merge two sets of stats unless one is an accumulation object (i.e. with the default transform id) + // or the stats are referencing the same transform + assert transformId.equals(DEFAULT_TRANSFORM_ID) || this.transformId.equals(other.transformId); numPages += other.numPages; numInputDocuments += other.numInputDocuments; numOuputDocuments += other.numOuputDocuments; @@ -93,6 +164,37 @@ public DataFrameIndexerTransformStats merge(DataFrameIndexerTransformStats other return this; } + @Override + public boolean equals(Object other) { + if (this == other) { + return true; + } + + if (other == null || getClass() != other.getClass()) { + return false; + } + + DataFrameIndexerTransformStats that = (DataFrameIndexerTransformStats) other; + + return Objects.equals(this.transformId, that.transformId) + && Objects.equals(this.numPages, that.numPages) + && Objects.equals(this.numInputDocuments, that.numInputDocuments) + && Objects.equals(this.numOuputDocuments, that.numOuputDocuments) + && Objects.equals(this.numInvocations, that.numInvocations) + && Objects.equals(this.indexTime, that.indexTime) + && Objects.equals(this.searchTime, that.searchTime) + && Objects.equals(this.indexFailures, that.indexFailures) + && Objects.equals(this.searchFailures, that.searchFailures) + && Objects.equals(this.indexTotal, that.indexTotal) + && Objects.equals(this.searchTotal, that.searchTotal); + } + + @Override + public int hashCode() { + return Objects.hash(transformId, numPages, numInputDocuments, numOuputDocuments, numInvocations, + indexTime, searchTime, indexFailures, searchFailures, indexTotal, searchTotal); + } + public static DataFrameIndexerTransformStats fromXContent(XContentParser parser) { try { return PARSER.parse(parser, null); diff --git a/x-pack/plugin/core/src/main/java/org/elasticsearch/xpack/core/dataframe/transforms/DataFrameTransformStateAndStats.java b/x-pack/plugin/core/src/main/java/org/elasticsearch/xpack/core/dataframe/transforms/DataFrameTransformStateAndStats.java index e155998aa2e08..dc58be9e6638a 100644 --- a/x-pack/plugin/core/src/main/java/org/elasticsearch/xpack/core/dataframe/transforms/DataFrameTransformStateAndStats.java +++ b/x-pack/plugin/core/src/main/java/org/elasticsearch/xpack/core/dataframe/transforms/DataFrameTransformStateAndStats.java @@ -40,9 +40,13 @@ public class DataFrameTransformStateAndStats implements Writeable, ToXContentObj } public static DataFrameTransformStateAndStats initialStateAndStats(String id) { + return initialStateAndStats(id, new DataFrameIndexerTransformStats(id)); + } + + public static DataFrameTransformStateAndStats initialStateAndStats(String id, DataFrameIndexerTransformStats indexerTransformStats) { return new DataFrameTransformStateAndStats(id, new DataFrameTransformState(DataFrameTransformTaskState.STOPPED, IndexerState.STOPPED, null, 0L, null), - new DataFrameIndexerTransformStats()); + indexerTransformStats); } public DataFrameTransformStateAndStats(String id, DataFrameTransformState state, DataFrameIndexerTransformStats stats) { @@ -62,7 +66,7 @@ public XContentBuilder toXContent(XContentBuilder builder, Params params) throws builder.startObject(); builder.field(DataFrameField.ID.getPreferredName(), id); builder.field(STATE_FIELD.getPreferredName(), transformState); - builder.field(DataFrameField.STATS_FIELD.getPreferredName(), transformStats); + builder.field(DataFrameField.STATS_FIELD.getPreferredName(), transformStats, params); builder.endObject(); return builder; } diff --git a/x-pack/plugin/core/src/main/java/org/elasticsearch/xpack/core/security/SecurityFeatureSetUsage.java b/x-pack/plugin/core/src/main/java/org/elasticsearch/xpack/core/security/SecurityFeatureSetUsage.java index a06eacefcf81f..618414426f085 100644 --- a/x-pack/plugin/core/src/main/java/org/elasticsearch/xpack/core/security/SecurityFeatureSetUsage.java +++ b/x-pack/plugin/core/src/main/java/org/elasticsearch/xpack/core/security/SecurityFeatureSetUsage.java @@ -107,6 +107,10 @@ protected void innerXContent(XContentBuilder builder, Params params) throws IOEx builder.field(AUDIT_XFIELD, auditUsage); builder.field(IP_FILTER_XFIELD, ipFilterUsage); builder.field(ANONYMOUS_XFIELD, anonymousUsage); + } else if (sslUsage.isEmpty() == false) { + // A trial (or basic) license can have SSL without security. + // This is because security defaults to disabled on that license, but that dynamic-default does not disable SSL. + builder.field(SSL_XFIELD, sslUsage); } } diff --git a/x-pack/plugin/core/src/main/java/org/elasticsearch/xpack/core/security/action/rolemapping/PutRoleMappingRequest.java b/x-pack/plugin/core/src/main/java/org/elasticsearch/xpack/core/security/action/rolemapping/PutRoleMappingRequest.java index 168adaa111658..ae036b63162f0 100644 --- a/x-pack/plugin/core/src/main/java/org/elasticsearch/xpack/core/security/action/rolemapping/PutRoleMappingRequest.java +++ b/x-pack/plugin/core/src/main/java/org/elasticsearch/xpack/core/security/action/rolemapping/PutRoleMappingRequest.java @@ -137,7 +137,7 @@ public void readFrom(StreamInput in) throws IOException { this.name = in.readString(); this.enabled = in.readBoolean(); this.roles = in.readStringList(); - if (in.getVersion().onOrAfter(Version.V_8_0_0)) { + if (in.getVersion().onOrAfter(Version.V_7_1_0)) { this.roleTemplates = in.readList(TemplateRoleName::new); } this.rules = ExpressionParser.readExpression(in); @@ -151,7 +151,7 @@ public void writeTo(StreamOutput out) throws IOException { out.writeString(name); out.writeBoolean(enabled); out.writeStringCollection(roles); - if (out.getVersion().onOrAfter(Version.V_8_0_0)) { + if (out.getVersion().onOrAfter(Version.V_7_1_0)) { out.writeList(roleTemplates); } ExpressionParser.writeExpression(rules, out); diff --git a/x-pack/plugin/core/src/main/java/org/elasticsearch/xpack/core/security/authc/support/mapper/ExpressionRoleMapping.java b/x-pack/plugin/core/src/main/java/org/elasticsearch/xpack/core/security/authc/support/mapper/ExpressionRoleMapping.java index dd5fb08fa14b7..1564b46760ce4 100644 --- a/x-pack/plugin/core/src/main/java/org/elasticsearch/xpack/core/security/authc/support/mapper/ExpressionRoleMapping.java +++ b/x-pack/plugin/core/src/main/java/org/elasticsearch/xpack/core/security/authc/support/mapper/ExpressionRoleMapping.java @@ -91,7 +91,7 @@ public ExpressionRoleMapping(StreamInput in) throws IOException { this.name = in.readString(); this.enabled = in.readBoolean(); this.roles = in.readStringList(); - if (in.getVersion().onOrAfter(Version.V_8_0_0)) { + if (in.getVersion().onOrAfter(Version.V_7_1_0)) { this.roleTemplates = in.readList(TemplateRoleName::new); } else { this.roleTemplates = Collections.emptyList(); @@ -105,7 +105,7 @@ public void writeTo(StreamOutput out) throws IOException { out.writeString(name); out.writeBoolean(enabled); out.writeStringCollection(roles); - if (out.getVersion().onOrAfter(Version.V_8_0_0)) { + if (out.getVersion().onOrAfter(Version.V_7_1_0)) { out.writeList(roleTemplates); } ExpressionParser.writeExpression(expression, out); diff --git a/x-pack/plugin/core/src/test/java/org/elasticsearch/xpack/core/dataframe/transforms/DataFrameIndexerTransformStatsTests.java b/x-pack/plugin/core/src/test/java/org/elasticsearch/xpack/core/dataframe/transforms/DataFrameIndexerTransformStatsTests.java index 45b44882d7861..54fc5d5d45dc5 100644 --- a/x-pack/plugin/core/src/test/java/org/elasticsearch/xpack/core/dataframe/transforms/DataFrameIndexerTransformStatsTests.java +++ b/x-pack/plugin/core/src/test/java/org/elasticsearch/xpack/core/dataframe/transforms/DataFrameIndexerTransformStatsTests.java @@ -7,12 +7,19 @@ package org.elasticsearch.xpack.core.dataframe.transforms; import org.elasticsearch.common.io.stream.Writeable; +import org.elasticsearch.common.xcontent.ToXContent; import org.elasticsearch.common.xcontent.XContentParser; import org.elasticsearch.test.AbstractSerializingTestCase; +import org.elasticsearch.xpack.core.dataframe.DataFrameField; import java.io.IOException; +import java.util.Collections; public class DataFrameIndexerTransformStatsTests extends AbstractSerializingTestCase { + + protected static ToXContent.Params TO_XCONTENT_PARAMS = new ToXContent.MapParams( + Collections.singletonMap(DataFrameField.FOR_INTERNAL_STORAGE, "true")); + @Override protected DataFrameIndexerTransformStats createTestInstance() { return randomStats(); @@ -29,21 +36,32 @@ protected DataFrameIndexerTransformStats doParseInstance(XContentParser parser) } public static DataFrameIndexerTransformStats randomStats() { - return new DataFrameIndexerTransformStats(randomLongBetween(0L, 10000L), randomLongBetween(0L, 10000L), - randomLongBetween(0L, 10000L), randomLongBetween(0L, 10000L), randomLongBetween(0L, 10000L), randomLongBetween(0L, 10000L), - randomLongBetween(0L, 10000L), randomLongBetween(0L, 10000L), randomLongBetween(0L, 10000L), randomLongBetween(0L, 10000L)); + return randomStats(randomAlphaOfLength(10)); + } + + public static DataFrameIndexerTransformStats randomStats(String transformId) { + return new DataFrameIndexerTransformStats(transformId, randomLongBetween(10L, 10000L), + randomLongBetween(0L, 10000L), randomLongBetween(0L, 10000L), randomLongBetween(0L, 10000L), randomLongBetween(0L, 10000L), + randomLongBetween(0L, 10000L), randomLongBetween(0L, 10000L), randomLongBetween(0L, 10000L), randomLongBetween(0L, 10000L), + randomLongBetween(0L, 10000L)); + } + + @Override + protected ToXContent.Params getToXContentParams() { + return TO_XCONTENT_PARAMS; } public void testMerge() throws IOException { - DataFrameIndexerTransformStats emptyStats = new DataFrameIndexerTransformStats(); - DataFrameIndexerTransformStats randomStats = randomStats(); + String transformId = randomAlphaOfLength(10); + DataFrameIndexerTransformStats emptyStats = new DataFrameIndexerTransformStats(transformId); + DataFrameIndexerTransformStats randomStats = randomStats(transformId); assertEquals(randomStats, emptyStats.merge(randomStats)); assertEquals(randomStats, randomStats.merge(emptyStats)); DataFrameIndexerTransformStats randomStatsClone = copyInstance(randomStats); - DataFrameIndexerTransformStats trippleRandomStats = new DataFrameIndexerTransformStats(3 * randomStats.getNumPages(), + DataFrameIndexerTransformStats trippleRandomStats = new DataFrameIndexerTransformStats(transformId, 3 * randomStats.getNumPages(), 3 * randomStats.getNumDocuments(), 3 * randomStats.getOutputDocuments(), 3 * randomStats.getNumInvocations(), 3 * randomStats.getIndexTime(), 3 * randomStats.getSearchTime(), 3 * randomStats.getIndexTotal(), 3 * randomStats.getSearchTotal(), 3 * randomStats.getIndexFailures(), 3 * randomStats.getSearchFailures()); diff --git a/x-pack/plugin/core/src/test/java/org/elasticsearch/xpack/core/dataframe/transforms/DataFrameTransformStateAndStatsTests.java b/x-pack/plugin/core/src/test/java/org/elasticsearch/xpack/core/dataframe/transforms/DataFrameTransformStateAndStatsTests.java index 266967e27b903..4f80d0d0b453c 100644 --- a/x-pack/plugin/core/src/test/java/org/elasticsearch/xpack/core/dataframe/transforms/DataFrameTransformStateAndStatsTests.java +++ b/x-pack/plugin/core/src/test/java/org/elasticsearch/xpack/core/dataframe/transforms/DataFrameTransformStateAndStatsTests.java @@ -7,22 +7,26 @@ package org.elasticsearch.xpack.core.dataframe.transforms; import org.elasticsearch.common.io.stream.Writeable.Reader; +import org.elasticsearch.common.xcontent.ToXContent; import org.elasticsearch.common.xcontent.XContentParser; +import org.elasticsearch.xpack.core.dataframe.DataFrameField; import java.io.IOException; +import java.util.Collections; public class DataFrameTransformStateAndStatsTests extends AbstractSerializingDataFrameTestCase { + protected static ToXContent.Params TO_XCONTENT_PARAMS = new ToXContent.MapParams( + Collections.singletonMap(DataFrameField.FOR_INTERNAL_STORAGE, "true")); + public static DataFrameTransformStateAndStats randomDataFrameTransformStateAndStats(String id) { return new DataFrameTransformStateAndStats(id, DataFrameTransformStateTests.randomDataFrameTransformState(), - DataFrameIndexerTransformStatsTests.randomStats()); + DataFrameIndexerTransformStatsTests.randomStats(id)); } public static DataFrameTransformStateAndStats randomDataFrameTransformStateAndStats() { - return new DataFrameTransformStateAndStats(randomAlphaOfLengthBetween(1, 10), - DataFrameTransformStateTests.randomDataFrameTransformState(), - DataFrameIndexerTransformStatsTests.randomStats()); + return randomDataFrameTransformStateAndStats(randomAlphaOfLengthBetween(1, 10)); } @Override @@ -30,6 +34,13 @@ protected DataFrameTransformStateAndStats doParseInstance(XContentParser parser) return DataFrameTransformStateAndStats.PARSER.apply(parser, null); } + @Override + // Setting params for internal storage so that we can check XContent equivalence as + // DataFrameIndexerTransformStats does not write the ID to the XContentObject unless it is for internal storage + protected ToXContent.Params getToXContentParams() { + return TO_XCONTENT_PARAMS; + } + @Override protected DataFrameTransformStateAndStats createTestInstance() { return randomDataFrameTransformStateAndStats(); diff --git a/x-pack/plugin/core/src/test/java/org/elasticsearch/xpack/core/dataframe/transforms/DataFrameTransformStateHlrcTests.java b/x-pack/plugin/core/src/test/java/org/elasticsearch/xpack/core/dataframe/transforms/DataFrameTransformStateHlrcTests.java new file mode 100644 index 0000000000000..4f1c6b1f7615e --- /dev/null +++ b/x-pack/plugin/core/src/test/java/org/elasticsearch/xpack/core/dataframe/transforms/DataFrameTransformStateHlrcTests.java @@ -0,0 +1,51 @@ +/* + * Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one + * or more contributor license agreements. Licensed under the Elastic License; + * you may not use this file except in compliance with the Elastic License. + */ + +package org.elasticsearch.xpack.core.dataframe.transforms; + +import org.elasticsearch.common.xcontent.XContentParser; +import org.elasticsearch.protocol.AbstractHlrcXContentTestCase; +import org.elasticsearch.xpack.core.indexing.IndexerState; + +import java.io.IOException; +import java.util.function.Predicate; + +public class DataFrameTransformStateHlrcTests extends AbstractHlrcXContentTestCase { + + @Override + public org.elasticsearch.client.dataframe.transforms.DataFrameTransformState doHlrcParseInstance(XContentParser parser) + throws IOException { + return org.elasticsearch.client.dataframe.transforms.DataFrameTransformState.fromXContent(parser); + } + + @Override + public DataFrameTransformState convertHlrcToInternal(org.elasticsearch.client.dataframe.transforms.DataFrameTransformState instance) { + return new DataFrameTransformState(DataFrameTransformTaskState.fromString(instance.getTaskState().value()), + IndexerState.fromString(instance.getIndexerState().value()), + instance.getPosition(), instance.getGeneration(), instance.getReason()); + } + + @Override + protected DataFrameTransformState createTestInstance() { + return DataFrameTransformStateTests.randomDataFrameTransformState(); + } + + @Override + protected DataFrameTransformState doParseInstance(XContentParser parser) throws IOException { + return DataFrameTransformState.fromXContent(parser); + } + + @Override + protected boolean supportsUnknownFields() { + return true; + } + + @Override + protected Predicate getRandomFieldsExcludeFilter() { + return field -> field.equals("current_position"); + } +} diff --git a/x-pack/plugin/data-frame/qa/single-node-tests/src/test/java/org/elasticsearch/xpack/dataframe/integration/DataFrameGetAndGetStatsIT.java b/x-pack/plugin/data-frame/qa/single-node-tests/src/test/java/org/elasticsearch/xpack/dataframe/integration/DataFrameGetAndGetStatsIT.java index 590d81c6e08b5..72176933b2a73 100644 --- a/x-pack/plugin/data-frame/qa/single-node-tests/src/test/java/org/elasticsearch/xpack/dataframe/integration/DataFrameGetAndGetStatsIT.java +++ b/x-pack/plugin/data-frame/qa/single-node-tests/src/test/java/org/elasticsearch/xpack/dataframe/integration/DataFrameGetAndGetStatsIT.java @@ -8,13 +8,17 @@ import org.elasticsearch.client.Request; import org.elasticsearch.common.xcontent.support.XContentMapValues; +import org.elasticsearch.xpack.core.dataframe.DataFrameField; +import org.junit.After; import org.junit.Before; import java.io.IOException; import java.util.Collections; +import java.util.List; import java.util.Map; import static org.elasticsearch.xpack.core.security.authc.support.UsernamePasswordToken.basicAuthHeaderValue; +import static org.hamcrest.Matchers.greaterThan; public class DataFrameGetAndGetStatsIT extends DataFrameRestTestCase { @@ -47,6 +51,11 @@ public void createIndexes() throws IOException { setupUser(TEST_ADMIN_USER_NAME, Collections.singletonList("data_frame_transforms_admin")); } + @After + public void clearOutTransforms() throws Exception { + wipeDataFrameTransforms(); + } + public void testGetAndGetStats() throws Exception { createPivotReviewsTransform("pivot_1", "pivot_reviews_1", null); createPivotReviewsTransform("pivot_2", "pivot_reviews_2", null); @@ -67,6 +76,12 @@ public void testGetAndGetStats() throws Exception { getRequest = createRequestWithAuth("GET", DATAFRAME_ENDPOINT + "*/_stats", authHeader); stats = entityAsMap(client().performRequest(getRequest)); assertEquals(2, XContentMapValues.extractValue("count", stats)); + getRequest = createRequestWithAuth("GET", DATAFRAME_ENDPOINT + "pivot_1,pivot_2/_stats", authHeader); + stats = entityAsMap(client().performRequest(getRequest)); + assertEquals(2, XContentMapValues.extractValue("count", stats)); + getRequest = createRequestWithAuth("GET", DATAFRAME_ENDPOINT + "pivot_*/_stats", authHeader); + stats = entityAsMap(client().performRequest(getRequest)); + assertEquals(2, XContentMapValues.extractValue("count", stats)); // only pivot_1 getRequest = createRequestWithAuth("GET", DATAFRAME_ENDPOINT + "pivot_1/_stats", authHeader); @@ -89,4 +104,35 @@ public void testGetAndGetStats() throws Exception { transforms = entityAsMap(client().performRequest(getRequest)); assertEquals(1, XContentMapValues.extractValue("count", transforms)); } + + @SuppressWarnings("unchecked") + public void testGetPersistedStatsWithoutTask() throws Exception { + createPivotReviewsTransform("pivot_stats_1", "pivot_reviews_stats_1", null); + startAndWaitForTransform("pivot_stats_1", "pivot_reviews_stats_1"); + stopDataFrameTransform("pivot_stats_1", false); + + // Get rid of the first transform task, but keep the configuration + client().performRequest(new Request("POST", "_tasks/_cancel?actions="+DataFrameField.TASK_NAME+"*")); + + // Verify that the task is gone + Map tasks = + entityAsMap(client().performRequest(new Request("GET", "_tasks?actions="+DataFrameField.TASK_NAME+"*"))); + assertTrue(((Map)XContentMapValues.extractValue("nodes", tasks)).isEmpty()); + + createPivotReviewsTransform("pivot_stats_2", "pivot_reviews_stats_2", null); + startAndWaitForTransform("pivot_stats_2", "pivot_reviews_stats_2"); + + Request getRequest = createRequestWithAuth("GET", DATAFRAME_ENDPOINT + "_stats", BASIC_AUTH_VALUE_DATA_FRAME_ADMIN); + Map stats = entityAsMap(client().performRequest(getRequest)); + assertEquals(2, XContentMapValues.extractValue("count", stats)); + List> transformsStats = (List>)XContentMapValues.extractValue("transforms", stats); + // Verify that both transforms, the one with the task and the one without have statistics + for (Map transformStats : transformsStats) { + Map stat = (Map)transformStats.get("stats"); + assertThat(((Integer)stat.get("documents_processed")), greaterThan(0)); + assertThat(((Integer)stat.get("search_time_in_ms")), greaterThan(0)); + assertThat(((Integer)stat.get("search_total")), greaterThan(0)); + assertThat(((Integer)stat.get("pages_processed")), greaterThan(0)); + } + } } diff --git a/x-pack/plugin/data-frame/qa/single-node-tests/src/test/java/org/elasticsearch/xpack/dataframe/integration/DataFramePivotRestIT.java b/x-pack/plugin/data-frame/qa/single-node-tests/src/test/java/org/elasticsearch/xpack/dataframe/integration/DataFramePivotRestIT.java index 99c08f1a50583..95daf11f674d3 100644 --- a/x-pack/plugin/data-frame/qa/single-node-tests/src/test/java/org/elasticsearch/xpack/dataframe/integration/DataFramePivotRestIT.java +++ b/x-pack/plugin/data-frame/qa/single-node-tests/src/test/java/org/elasticsearch/xpack/dataframe/integration/DataFramePivotRestIT.java @@ -268,7 +268,6 @@ public void testPreviewTransform() throws Exception { }); } - @AwaitsFix(bugUrl = "https://github.com/elastic/elasticsearch/issues/40537") public void testPivotWithMaxOnDateField() throws Exception { String transformId = "simpleDateHistogramPivotWithMaxTime"; String dataFrameIndex = "pivot_reviews_via_date_histogram_with_max_time"; @@ -312,7 +311,7 @@ public void testPivotWithMaxOnDateField() throws Exception { Map searchResult = getAsMap(dataFrameIndex + "/_search?q=by_day:2017-01-15"); String actual = (String) ((List) XContentMapValues.extractValue("hits.hits._source.timestamp", searchResult)).get(0); // Do `containsString` as actual ending timestamp is indeterminate due to how data is generated - assertThat(actual, containsString("2017-01-15T20:")); + assertThat(actual, containsString("2017-01-15T")); } private void assertOnePivotValue(String query, double expected) throws IOException { diff --git a/x-pack/plugin/data-frame/qa/single-node-tests/src/test/java/org/elasticsearch/xpack/dataframe/integration/DataFrameUsageIT.java b/x-pack/plugin/data-frame/qa/single-node-tests/src/test/java/org/elasticsearch/xpack/dataframe/integration/DataFrameUsageIT.java index 5fcdba603eebb..24ce173b37567 100644 --- a/x-pack/plugin/data-frame/qa/single-node-tests/src/test/java/org/elasticsearch/xpack/dataframe/integration/DataFrameUsageIT.java +++ b/x-pack/plugin/data-frame/qa/single-node-tests/src/test/java/org/elasticsearch/xpack/dataframe/integration/DataFrameUsageIT.java @@ -9,11 +9,19 @@ import org.elasticsearch.client.Request; import org.elasticsearch.client.Response; import org.elasticsearch.common.xcontent.support.XContentMapValues; +import org.elasticsearch.xpack.core.dataframe.DataFrameField; +import org.elasticsearch.xpack.core.dataframe.transforms.DataFrameIndexerTransformStats; +import org.elasticsearch.xpack.dataframe.persistence.DataFrameInternalIndex; import org.junit.Before; import java.io.IOException; +import java.util.HashMap; +import java.util.List; import java.util.Map; +import static org.elasticsearch.xpack.core.dataframe.DataFrameField.INDEX_DOC_TYPE; +import static org.elasticsearch.xpack.dataframe.DataFrameFeatureSet.PROVIDED_STATS; + public class DataFrameUsageIT extends DataFrameRestTestCase { private boolean indicesCreated = false; @@ -45,22 +53,63 @@ public void testUsage() throws Exception { assertEquals(null, XContentMapValues.extractValue("data_frame.transforms", usageAsMap)); assertEquals(null, XContentMapValues.extractValue("data_frame.stats", usageAsMap)); - // create a transform + // create transforms createPivotReviewsTransform("test_usage", "pivot_reviews", null); + createPivotReviewsTransform("test_usage_no_task", "pivot_reviews_no_task", null); + createPivotReviewsTransform("test_usage_no_stats_or_task", "pivot_reviews_no_stats_or_task", null); usageResponse = client().performRequest(new Request("GET", "_xpack/usage")); usageAsMap = entityAsMap(usageResponse); - assertEquals(1, XContentMapValues.extractValue("data_frame.transforms._all", usageAsMap)); - assertEquals(1, XContentMapValues.extractValue("data_frame.transforms.stopped", usageAsMap)); + assertEquals(3, XContentMapValues.extractValue("data_frame.transforms._all", usageAsMap)); + assertEquals(3, XContentMapValues.extractValue("data_frame.transforms.stopped", usageAsMap)); + + startAndWaitForTransform("test_usage_no_task", "pivot_reviews_no_task"); + stopDataFrameTransform("test_usage_no_task", false); + // Remove the task, we should still have the transform and its stat doc + client().performRequest(new Request("POST", "_tasks/_cancel?actions="+ DataFrameField.TASK_NAME+"*")); - // TODO remove as soon as stats are stored in an index instead of ClusterState with the task startAndWaitForTransform("test_usage", "pivot_reviews"); + Request statsExistsRequest = new Request("GET", + DataFrameInternalIndex.INDEX_NAME+"/_search?q=" + + INDEX_DOC_TYPE.getPreferredName() + ":" + + DataFrameIndexerTransformStats.NAME); + // Verify that we have our two stats documents + assertBusy(() -> { + Map hasStatsMap = entityAsMap(client().performRequest(statsExistsRequest)); + assertEquals(2, XContentMapValues.extractValue("hits.total.value", hasStatsMap)); + }); + + Request getRequest = new Request("GET", DATAFRAME_ENDPOINT + "test_usage/_stats"); + Map stats = entityAsMap(client().performRequest(getRequest)); + Map expectedStats = new HashMap<>(); + for(String statName : PROVIDED_STATS) { + @SuppressWarnings("unchecked") + List specificStatistic = ((List)XContentMapValues.extractValue("transforms.stats." + statName, stats)); + assertNotNull(specificStatistic); + Integer statistic = (specificStatistic).get(0); + expectedStats.put(statName, statistic); + } + + getRequest = new Request("GET", DATAFRAME_ENDPOINT + "test_usage_no_task/_stats"); + stats = entityAsMap(client().performRequest(getRequest)); + for(String statName : PROVIDED_STATS) { + @SuppressWarnings("unchecked") + List specificStatistic = ((List)XContentMapValues.extractValue("transforms.stats." + statName, stats)); + assertNotNull(specificStatistic); + Integer statistic = (specificStatistic).get(0); + expectedStats.merge(statName, statistic, Integer::sum); + } + + usageResponse = client().performRequest(new Request("GET", "_xpack/usage")); usageAsMap = entityAsMap(usageResponse); // we should see some stats - assertEquals(1, XContentMapValues.extractValue("data_frame.transforms._all", usageAsMap)); + assertEquals(3, XContentMapValues.extractValue("data_frame.transforms._all", usageAsMap)); assertEquals(1, XContentMapValues.extractValue("data_frame.transforms.started", usageAsMap)); - assertEquals(0, XContentMapValues.extractValue("data_frame.stats.index_failures", usageAsMap)); + assertEquals(2, XContentMapValues.extractValue("data_frame.transforms.stopped", usageAsMap)); + for(String statName : PROVIDED_STATS) { + assertEquals(expectedStats.get(statName), XContentMapValues.extractValue("data_frame.stats."+statName, usageAsMap)); + } } } diff --git a/x-pack/plugin/data-frame/src/main/java/org/elasticsearch/xpack/dataframe/DataFrameFeatureSet.java b/x-pack/plugin/data-frame/src/main/java/org/elasticsearch/xpack/dataframe/DataFrameFeatureSet.java index 2cb7d59475444..029fe88766df5 100644 --- a/x-pack/plugin/data-frame/src/main/java/org/elasticsearch/xpack/dataframe/DataFrameFeatureSet.java +++ b/x-pack/plugin/data-frame/src/main/java/org/elasticsearch/xpack/dataframe/DataFrameFeatureSet.java @@ -6,21 +6,41 @@ package org.elasticsearch.xpack.dataframe; +import org.apache.logging.log4j.LogManager; +import org.apache.logging.log4j.Logger; +import org.elasticsearch.ResourceNotFoundException; import org.elasticsearch.action.ActionListener; +import org.elasticsearch.action.search.SearchRequest; +import org.elasticsearch.action.search.SearchRequestBuilder; +import org.elasticsearch.action.search.SearchResponse; import org.elasticsearch.client.Client; -import org.elasticsearch.cluster.metadata.MetaData; +import org.elasticsearch.cluster.service.ClusterService; import org.elasticsearch.common.Nullable; import org.elasticsearch.common.inject.Inject; import org.elasticsearch.common.settings.Settings; +import org.elasticsearch.index.query.QueryBuilder; +import org.elasticsearch.index.query.QueryBuilders; import org.elasticsearch.license.XPackLicenseState; +import org.elasticsearch.persistent.PersistentTasksCustomMetaData; +import org.elasticsearch.search.aggregations.Aggregation; +import org.elasticsearch.search.aggregations.AggregationBuilders; +import org.elasticsearch.search.aggregations.metrics.NumericMetricsAggregation; +import org.elasticsearch.xpack.core.ClientHelper; import org.elasticsearch.xpack.core.XPackFeatureSet; import org.elasticsearch.xpack.core.XPackField; import org.elasticsearch.xpack.core.XPackSettings; import org.elasticsearch.xpack.core.dataframe.DataFrameFeatureSetUsage; +import org.elasticsearch.xpack.core.dataframe.DataFrameField; import org.elasticsearch.xpack.core.dataframe.transforms.DataFrameIndexerTransformStats; -import org.elasticsearch.xpack.core.dataframe.action.GetDataFrameTransformsStatsAction; -import org.elasticsearch.xpack.core.dataframe.transforms.DataFrameTransformStateAndStats; - +import org.elasticsearch.xpack.core.dataframe.transforms.DataFrameTransform; +import org.elasticsearch.xpack.core.dataframe.transforms.DataFrameTransformConfig; +import org.elasticsearch.xpack.core.dataframe.transforms.DataFrameTransformState; +import org.elasticsearch.xpack.core.dataframe.transforms.DataFrameTransformTaskState; +import org.elasticsearch.xpack.dataframe.persistence.DataFrameInternalIndex; + +import java.util.ArrayList; +import java.util.Arrays; +import java.util.Collection; import java.util.Collections; import java.util.HashMap; import java.util.List; @@ -33,11 +53,28 @@ public class DataFrameFeatureSet implements XPackFeatureSet { private final boolean enabled; private final Client client; private final XPackLicenseState licenseState; + private final ClusterService clusterService; + + private static final Logger logger = LogManager.getLogger(DataFrameFeatureSet.class); + + public static final String[] PROVIDED_STATS = new String[] { + DataFrameIndexerTransformStats.NUM_PAGES.getPreferredName(), + DataFrameIndexerTransformStats.NUM_INPUT_DOCUMENTS.getPreferredName(), + DataFrameIndexerTransformStats.NUM_OUTPUT_DOCUMENTS.getPreferredName(), + DataFrameIndexerTransformStats.NUM_INVOCATIONS.getPreferredName(), + DataFrameIndexerTransformStats.INDEX_TIME_IN_MS.getPreferredName(), + DataFrameIndexerTransformStats.SEARCH_TIME_IN_MS.getPreferredName(), + DataFrameIndexerTransformStats.INDEX_TOTAL.getPreferredName(), + DataFrameIndexerTransformStats.SEARCH_TOTAL.getPreferredName(), + DataFrameIndexerTransformStats.INDEX_FAILURES.getPreferredName(), + DataFrameIndexerTransformStats.SEARCH_FAILURES.getPreferredName(), + }; @Inject - public DataFrameFeatureSet(Settings settings, Client client, @Nullable XPackLicenseState licenseState) { + public DataFrameFeatureSet(Settings settings, ClusterService clusterService, Client client, @Nullable XPackLicenseState licenseState) { this.enabled = XPackSettings.DATA_FRAME_ENABLED.get(settings); this.client = Objects.requireNonNull(client); + this.clusterService = Objects.requireNonNull(clusterService); this.licenseState = licenseState; } @@ -69,30 +106,127 @@ public Map nativeCodeInfo() { @Override public void usage(ActionListener listener) { if (enabled == false) { - listener.onResponse( - new DataFrameFeatureSetUsage(available(), enabled(), Collections.emptyMap(), new DataFrameIndexerTransformStats())); + listener.onResponse(new DataFrameFeatureSetUsage(available(), + enabled(), + Collections.emptyMap(), + DataFrameIndexerTransformStats.withDefaultTransformId())); return; } - final GetDataFrameTransformsStatsAction.Request transformStatsRequest = new GetDataFrameTransformsStatsAction.Request(MetaData.ALL); - client.execute(GetDataFrameTransformsStatsAction.INSTANCE, - transformStatsRequest, - ActionListener.wrap(transformStatsResponse -> - listener.onResponse(createUsage(available(), enabled(), transformStatsResponse.getTransformsStateAndStats())), - listener::onFailure)); + PersistentTasksCustomMetaData taskMetadata = PersistentTasksCustomMetaData.getPersistentTasksCustomMetaData(clusterService.state()); + Collection> dataFrameTasks = taskMetadata == null ? + Collections.emptyList() : + taskMetadata.findTasks(DataFrameTransform.NAME, (t) -> true); + final int taskCount = dataFrameTasks.size(); + final Map transformsCountByState = new HashMap<>(); + for(PersistentTasksCustomMetaData.PersistentTask dataFrameTask : dataFrameTasks) { + DataFrameTransformState state = (DataFrameTransformState)dataFrameTask.getState(); + transformsCountByState.merge(state.getTaskState().value(), 1L, Long::sum); + } + + ActionListener totalStatsListener = ActionListener.wrap( + statSummations -> listener.onResponse(new DataFrameFeatureSetUsage(available(), + enabled(), + transformsCountByState, + statSummations)), + listener::onFailure + ); + + ActionListener totalTransformCountListener = ActionListener.wrap( + transformCountSuccess -> { + if (transformCountSuccess.getShardFailures().length > 0) { + logger.error("total transform count search returned shard failures: {}", + Arrays.toString(transformCountSuccess.getShardFailures())); + } + long totalTransforms = transformCountSuccess.getHits().getTotalHits().value; + if (totalTransforms == 0) { + listener.onResponse(new DataFrameFeatureSetUsage(available(), + enabled(), + transformsCountByState, + DataFrameIndexerTransformStats.withDefaultTransformId())); + return; + } + transformsCountByState.merge(DataFrameTransformTaskState.STOPPED.value(), totalTransforms - taskCount, Long::sum); + getStatisticSummations(client, totalStatsListener); + }, + transformCountFailure -> { + if (transformCountFailure instanceof ResourceNotFoundException) { + getStatisticSummations(client, totalStatsListener); + } else { + listener.onFailure(transformCountFailure); + } + } + ); + + SearchRequest totalTransformCount = client.prepareSearch(DataFrameInternalIndex.INDEX_NAME) + .setTrackTotalHits(true) + .setQuery(QueryBuilders.constantScoreQuery(QueryBuilders.boolQuery() + .filter(QueryBuilders.termQuery(DataFrameField.INDEX_DOC_TYPE.getPreferredName(), DataFrameTransformConfig.NAME)))) + .request(); + + ClientHelper.executeAsyncWithOrigin(client.threadPool().getThreadContext(), + ClientHelper.DATA_FRAME_ORIGIN, + totalTransformCount, + totalTransformCountListener, + client::search); } - static DataFrameFeatureSetUsage createUsage(boolean available, - boolean enabled, - List transformsStateAndStats) { + static DataFrameIndexerTransformStats parseSearchAggs(SearchResponse searchResponse) { + List statisticsList = new ArrayList<>(PROVIDED_STATS.length); - Map transformsCountByState = new HashMap<>(); - DataFrameIndexerTransformStats accumulatedStats = new DataFrameIndexerTransformStats(); - transformsStateAndStats.forEach(singleResult -> { - transformsCountByState.merge(singleResult.getTransformState().getIndexerState().value(), 1L, Long::sum); - accumulatedStats.merge(singleResult.getTransformStats()); - }); + for(String statName : PROVIDED_STATS) { + Aggregation agg = searchResponse.getAggregations().get(statName); + if (agg instanceof NumericMetricsAggregation.SingleValue) { + statisticsList.add((long)((NumericMetricsAggregation.SingleValue)agg).value()); + } else { + statisticsList.add(0L); + } + } + return DataFrameIndexerTransformStats.withDefaultTransformId(statisticsList.get(0), // numPages + statisticsList.get(1), // numInputDocuments + statisticsList.get(2), // numOutputDocuments + statisticsList.get(3), // numInvocations + statisticsList.get(4), // indexTime + statisticsList.get(5), // searchTime + statisticsList.get(6), // indexTotal + statisticsList.get(7), // searchTotal + statisticsList.get(8), // indexFailures + statisticsList.get(9)); // searchFailures + } + + static void getStatisticSummations(Client client, ActionListener statsListener) { + QueryBuilder queryBuilder = QueryBuilders.constantScoreQuery(QueryBuilders.boolQuery() + .filter(QueryBuilders.termQuery(DataFrameField.INDEX_DOC_TYPE.getPreferredName(), + DataFrameIndexerTransformStats.NAME))); + + SearchRequestBuilder requestBuilder = client.prepareSearch(DataFrameInternalIndex.INDEX_NAME) + .setSize(0) + .setQuery(queryBuilder); + + for(String statName : PROVIDED_STATS) { + requestBuilder.addAggregation(AggregationBuilders.sum(statName).field(statName)); + } - return new DataFrameFeatureSetUsage(available, enabled, transformsCountByState, accumulatedStats); + ActionListener getStatisticSummationsListener = ActionListener.wrap( + searchResponse -> { + if (searchResponse.getShardFailures().length > 0) { + logger.error("statistics summations search returned shard failures: {}", + Arrays.toString(searchResponse.getShardFailures())); + } + statsListener.onResponse(parseSearchAggs(searchResponse)); + }, + failure -> { + if (failure instanceof ResourceNotFoundException) { + statsListener.onResponse(DataFrameIndexerTransformStats.withDefaultTransformId()); + } else { + statsListener.onFailure(failure); + } + } + ); + ClientHelper.executeAsyncWithOrigin(client.threadPool().getThreadContext(), + ClientHelper.DATA_FRAME_ORIGIN, + requestBuilder.request(), + getStatisticSummationsListener, + client::search); } } diff --git a/x-pack/plugin/data-frame/src/main/java/org/elasticsearch/xpack/dataframe/action/TransportGetDataFrameTransformsStatsAction.java b/x-pack/plugin/data-frame/src/main/java/org/elasticsearch/xpack/dataframe/action/TransportGetDataFrameTransformsStatsAction.java index b751279abf233..c2f64dd66a583 100644 --- a/x-pack/plugin/data-frame/src/main/java/org/elasticsearch/xpack/dataframe/action/TransportGetDataFrameTransformsStatsAction.java +++ b/x-pack/plugin/data-frame/src/main/java/org/elasticsearch/xpack/dataframe/action/TransportGetDataFrameTransformsStatsAction.java @@ -6,17 +6,21 @@ package org.elasticsearch.xpack.dataframe.action; +import org.apache.logging.log4j.LogManager; +import org.apache.logging.log4j.Logger; +import org.elasticsearch.ElasticsearchException; import org.elasticsearch.ElasticsearchParseException; import org.elasticsearch.ResourceNotFoundException; import org.elasticsearch.action.ActionListener; import org.elasticsearch.action.ActionListenerResponseHandler; import org.elasticsearch.action.FailedNodeException; import org.elasticsearch.action.TaskOperationFailure; +import org.elasticsearch.action.search.SearchRequest; +import org.elasticsearch.action.search.SearchResponse; import org.elasticsearch.action.support.ActionFilters; import org.elasticsearch.action.support.tasks.TransportTasksAction; import org.elasticsearch.client.Client; import org.elasticsearch.cluster.ClusterState; -import org.elasticsearch.cluster.metadata.MetaData; import org.elasticsearch.cluster.node.DiscoveryNodes; import org.elasticsearch.cluster.service.ClusterService; import org.elasticsearch.common.bytes.BytesReference; @@ -27,28 +31,29 @@ import org.elasticsearch.common.xcontent.XContentParser; import org.elasticsearch.common.xcontent.XContentType; import org.elasticsearch.discovery.MasterNotDiscoveredException; +import org.elasticsearch.index.IndexNotFoundException; import org.elasticsearch.index.query.QueryBuilder; import org.elasticsearch.index.query.QueryBuilders; import org.elasticsearch.search.SearchHit; -import org.elasticsearch.search.fetch.subphase.FetchSourceContext; import org.elasticsearch.search.sort.SortOrder; import org.elasticsearch.tasks.Task; import org.elasticsearch.threadpool.ThreadPool; import org.elasticsearch.transport.TransportService; +import org.elasticsearch.xpack.core.ClientHelper; import org.elasticsearch.xpack.core.dataframe.DataFrameField; import org.elasticsearch.xpack.core.dataframe.action.GetDataFrameTransformsStatsAction; import org.elasticsearch.xpack.core.dataframe.action.GetDataFrameTransformsStatsAction.Request; import org.elasticsearch.xpack.core.dataframe.action.GetDataFrameTransformsStatsAction.Response; -import org.elasticsearch.xpack.core.dataframe.transforms.DataFrameTransformConfig; +import org.elasticsearch.xpack.core.dataframe.transforms.DataFrameIndexerTransformStats; import org.elasticsearch.xpack.core.dataframe.transforms.DataFrameTransformStateAndStats; import org.elasticsearch.xpack.dataframe.persistence.DataFrameInternalIndex; -import org.elasticsearch.xpack.dataframe.persistence.DataFramePersistentTaskUtils; import org.elasticsearch.xpack.dataframe.persistence.DataFrameTransformsConfigManager; import org.elasticsearch.xpack.dataframe.transforms.DataFrameTransformTask; -import org.elasticsearch.xpack.dataframe.util.BatchedDataIterator; import java.io.IOException; import java.io.InputStream; +import java.util.ArrayList; +import java.util.Arrays; import java.util.Collections; import java.util.Comparator; import java.util.HashSet; @@ -62,6 +67,8 @@ public class TransportGetDataFrameTransformsStatsAction extends GetDataFrameTransformsStatsAction.Response, GetDataFrameTransformsStatsAction.Response> { + private static final Logger logger = LogManager.getLogger(TransportGetDataFrameTransformsStatsAction.class); + private final Client client; private final DataFrameTransformsConfigManager dataFrameTransformsConfigManager; @Inject @@ -88,8 +95,6 @@ protected Response newResponse(Request request, List tasks, List listener) { List transformsStateAndStats = Collections.emptyList(); - assert task.getTransformId().equals(request.getId()) || request.getId().equals(MetaData.ALL); - // Little extra insurance, make sure we only return transforms that aren't cancelled if (task.isCancelled() == false) { DataFrameTransformStateAndStats transformStateAndStats = new DataFrameTransformStateAndStats(task.getTransformId(), @@ -101,139 +106,115 @@ protected void taskOperation(Request request, DataFrameTransformTask task, Actio } @Override - // TODO gather stats from docs when moved out of allocated task - protected void doExecute(Task task, Request request, ActionListener listener) { + protected void doExecute(Task task, Request request, ActionListener finalListener) { final ClusterState state = clusterService.state(); final DiscoveryNodes nodes = state.nodes(); - if (nodes.isLocalNodeElectedMaster()) { - if (DataFramePersistentTaskUtils.stateHasDataFrameTransforms(request.getId(), state)) { - ActionListener transformStatsListener = ActionListener.wrap( - response -> collectStatsForTransformsWithoutTasks(request, response, listener), - listener::onFailure - ); - super.doExecute(task, request, transformStatsListener); - } else { - // If we don't have any tasks, pass empty collection to this method - collectStatsForTransformsWithoutTasks(request, new Response(Collections.emptyList()), listener); - } - + dataFrameTransformsConfigManager.expandTransformIds(request.getId(), request.getPageParams(), ActionListener.wrap( + ids -> { + request.setExpandedIds(ids); + super.doExecute(task, request, ActionListener.wrap( + response -> collectStatsForTransformsWithoutTasks(request, response, finalListener), + finalListener::onFailure + )); + }, + e -> { + // If the index to search, or the individual config is not there, just return empty + if (e instanceof ResourceNotFoundException) { + finalListener.onResponse(new Response(Collections.emptyList())); + } else { + finalListener.onFailure(e); + } + } + )); } else { // Delegates GetTransforms to elected master node, so it becomes the coordinating node. // Non-master nodes may have a stale cluster state that shows transforms which are cancelled // on the master, which makes testing difficult. if (nodes.getMasterNode() == null) { - listener.onFailure(new MasterNotDiscoveredException("no known master nodes")); + finalListener.onFailure(new MasterNotDiscoveredException("no known master nodes")); } else { transportService.sendRequest(nodes.getMasterNode(), actionName, request, - new ActionListenerResponseHandler<>(listener, Response::new)); + new ActionListenerResponseHandler<>(finalListener, Response::new)); } } } - // TODO correct when we start storing stats in docs, right now, just return STOPPED and empty stats private void collectStatsForTransformsWithoutTasks(Request request, Response response, ActionListener listener) { - if (request.getId().equals(MetaData.ALL) == false) { - // If we did not find any tasks && this is NOT for ALL, verify that the single config exists, and return as stopped - // Empty other wise - if (response.getTransformsStateAndStats().isEmpty()) { - dataFrameTransformsConfigManager.getTransformConfiguration(request.getId(), ActionListener.wrap( - config -> - listener.onResponse( - new Response(Collections.singletonList(DataFrameTransformStateAndStats.initialStateAndStats(config.getId())))), - exception -> { - if (exception instanceof ResourceNotFoundException) { - listener.onResponse(new Response(Collections.emptyList())); - } else { - listener.onFailure(exception); - } - } - )); - } else { - // If it was not ALL && we DO have stored stats, simply return those as we found them all, since we only support 1 or all - listener.onResponse(response); - } + // We gathered all there is, no need to continue + if (request.getExpandedIds().size() == response.getTransformsStateAndStats().size()) { + listener.onResponse(response); return; } - // We only do this mass collection if we are getting ALL tasks - TransformIdCollector collector = new TransformIdCollector(); - collector.execute(ActionListener.wrap( - allIds -> { - response.getTransformsStateAndStats().forEach( - tsas -> allIds.remove(tsas.getId()) - ); - List statsWithoutTasks = allIds.stream() - .map(DataFrameTransformStateAndStats::initialStateAndStats) - .collect(Collectors.toList()); - statsWithoutTasks.addAll(response.getTransformsStateAndStats()); - statsWithoutTasks.sort(Comparator.comparing(DataFrameTransformStateAndStats::getId)); - listener.onResponse(new Response(statsWithoutTasks)); - }, - listener::onFailure - )); - } - - /** - * This class recursively queries a scroll search over all transform_ids and puts them in a set - */ - private class TransformIdCollector extends BatchedDataIterator> { - - private final Set ids = new HashSet<>(); - TransformIdCollector() { - super(client, DataFrameInternalIndex.INDEX_NAME); - } - void execute(final ActionListener> finalListener) { - if (this.hasNext()) { - next(ActionListener.wrap( - setOfIds -> execute(finalListener), - finalListener::onFailure - )); - } else { - finalListener.onResponse(ids); - } - } + Set transformsWithoutTasks = new HashSet<>(request.getExpandedIds()); + transformsWithoutTasks.removeAll(response.getTransformsStateAndStats().stream().map(DataFrameTransformStateAndStats::getId) + .collect(Collectors.toList())); + + // Small assurance that we are at least below the max. Terms search has a hard limit of 10k, we should at least be below that. + assert transformsWithoutTasks.size() <= Request.MAX_SIZE_RETURN; + + ActionListener searchStatsListener = ActionListener.wrap( + searchResponse -> { + List nodeFailures = new ArrayList<>(response.getNodeFailures()); + if (searchResponse.getShardFailures().length > 0) { + String msg = "transform statistics document search returned shard failures: " + + Arrays.toString(searchResponse.getShardFailures()); + logger.error(msg); + nodeFailures.add(new ElasticsearchException(msg)); + } + List allStateAndStats = response.getTransformsStateAndStats(); + for(SearchHit hit : searchResponse.getHits().getHits()) { + BytesReference source = hit.getSourceRef(); + try { + DataFrameIndexerTransformStats stats = parseFromSource(source); + allStateAndStats.add(DataFrameTransformStateAndStats.initialStateAndStats(stats.getTransformId(), stats)); + transformsWithoutTasks.remove(stats.getTransformId()); + } catch (IOException e) { + listener.onFailure(new ElasticsearchParseException("Could not parse data frame transform stats", e)); + return; + } + } + transformsWithoutTasks.forEach(transformId -> + allStateAndStats.add(DataFrameTransformStateAndStats.initialStateAndStats(transformId))); - @Override - protected QueryBuilder getQuery() { - return QueryBuilders.boolQuery() - .filter(QueryBuilders.termQuery(DataFrameField.INDEX_DOC_TYPE.getPreferredName(), DataFrameTransformConfig.NAME)); - } + // Any transform in collection could NOT have a task, so, even though the list is initially sorted + // it can easily become arbitrarily ordered based on which transforms don't have a task or stats docs + allStateAndStats.sort(Comparator.comparing(DataFrameTransformStateAndStats::getId)); - @Override - protected String map(SearchHit hit) { - BytesReference source = hit.getSourceRef(); - try (InputStream stream = source.streamInput(); - XContentParser parser = XContentFactory.xContent(XContentType.JSON).createParser(NamedXContentRegistry.EMPTY, - LoggingDeprecationHandler.INSTANCE, stream)) { - return (String)parser.map().get(DataFrameField.ID.getPreferredName()); - } catch (IOException e) { - throw new ElasticsearchParseException("failed to parse bucket", e); + listener.onResponse(new Response(allStateAndStats, response.getTaskFailures(), nodeFailures)); + }, + e -> { + if (e instanceof IndexNotFoundException) { + listener.onResponse(response); + } else { + listener.onFailure(e); + } } - } + ); - @Override - protected Set getCollection() { - return ids; - } + QueryBuilder builder = QueryBuilders.constantScoreQuery(QueryBuilders.boolQuery() + .filter(QueryBuilders.termsQuery(DataFrameField.ID.getPreferredName(), transformsWithoutTasks)) + .filter(QueryBuilders.termQuery(DataFrameField.INDEX_DOC_TYPE.getPreferredName(), DataFrameIndexerTransformStats.NAME))); - @Override - protected SortOrder sortOrder() { - return SortOrder.ASC; - } + SearchRequest searchRequest = client.prepareSearch(DataFrameInternalIndex.INDEX_NAME) + .addSort(DataFrameField.ID.getPreferredName(), SortOrder.ASC) + .setQuery(builder) + .request(); - @Override - protected String sortField() { - return DataFrameField.ID.getPreferredName(); - } + ClientHelper.executeAsyncWithOrigin(client.threadPool().getThreadContext(), + ClientHelper.DATA_FRAME_ORIGIN, + searchRequest, + searchStatsListener, client::search); + } - @Override - protected FetchSourceContext getFetchSourceContext() { - return new FetchSourceContext(true, new String[]{DataFrameField.ID.getPreferredName()}, new String[]{}); + private static DataFrameIndexerTransformStats parseFromSource(BytesReference source) throws IOException { + try (InputStream stream = source.streamInput(); + XContentParser parser = XContentFactory.xContent(XContentType.JSON) + .createParser(NamedXContentRegistry.EMPTY, LoggingDeprecationHandler.INSTANCE, stream)) { + return DataFrameIndexerTransformStats.PARSER.apply(parser, null); } } - - } diff --git a/x-pack/plugin/data-frame/src/main/java/org/elasticsearch/xpack/dataframe/action/TransportPreviewDataFrameTransformAction.java b/x-pack/plugin/data-frame/src/main/java/org/elasticsearch/xpack/dataframe/action/TransportPreviewDataFrameTransformAction.java index 78f6823034811..63b2ed720c0be 100644 --- a/x-pack/plugin/data-frame/src/main/java/org/elasticsearch/xpack/dataframe/action/TransportPreviewDataFrameTransformAction.java +++ b/x-pack/plugin/data-frame/src/main/java/org/elasticsearch/xpack/dataframe/action/TransportPreviewDataFrameTransformAction.java @@ -81,7 +81,7 @@ private void getPreview(Pivot pivot, ActionListener>> l ActionListener.wrap( r -> { final CompositeAggregation agg = r.getAggregations().get(COMPOSITE_AGGREGATION_NAME); - DataFrameIndexerTransformStats stats = new DataFrameIndexerTransformStats(); + DataFrameIndexerTransformStats stats = DataFrameIndexerTransformStats.withDefaultTransformId(); // remove all internal fields List> results = pivot.extractResults(agg, deducedMappings, stats) .map(record -> { diff --git a/x-pack/plugin/data-frame/src/main/java/org/elasticsearch/xpack/dataframe/persistence/DataFrameInternalIndex.java b/x-pack/plugin/data-frame/src/main/java/org/elasticsearch/xpack/dataframe/persistence/DataFrameInternalIndex.java index 2905e4c225793..26847c4881c3f 100644 --- a/x-pack/plugin/data-frame/src/main/java/org/elasticsearch/xpack/dataframe/persistence/DataFrameInternalIndex.java +++ b/x-pack/plugin/data-frame/src/main/java/org/elasticsearch/xpack/dataframe/persistence/DataFrameInternalIndex.java @@ -15,6 +15,7 @@ import org.elasticsearch.index.mapper.MapperService; import org.elasticsearch.xpack.core.common.notifications.AbstractAuditMessage; import org.elasticsearch.xpack.core.dataframe.DataFrameField; +import org.elasticsearch.xpack.core.dataframe.transforms.DataFrameIndexerTransformStats; import org.elasticsearch.xpack.core.dataframe.transforms.DestConfig; import org.elasticsearch.xpack.core.dataframe.transforms.SourceConfig; @@ -23,7 +24,7 @@ import static org.elasticsearch.common.xcontent.XContentFactory.jsonBuilder; import static org.elasticsearch.index.mapper.MapperService.SINGLE_MAPPING_NAME; -import static org.elasticsearch.xpack.core.dataframe.DataFrameField.DATA_FRAME_TRANSFORM_AUDIT_ID_FIELD; +import static org.elasticsearch.xpack.core.dataframe.DataFrameField.TRANSFORM_ID; public final class DataFrameInternalIndex { @@ -49,6 +50,7 @@ public final class DataFrameInternalIndex { // data types public static final String DOUBLE = "double"; + public static final String LONG = "long"; public static final String KEYWORD = "keyword"; public static IndexTemplateMetaData getIndexTemplateMetaData() throws IOException { @@ -83,7 +85,7 @@ private static XContentBuilder auditMappings() throws IOException { addMetaInformation(builder); builder.field(DYNAMIC, "false"); builder.startObject(PROPERTIES) - .startObject(DATA_FRAME_TRANSFORM_AUDIT_ID_FIELD) + .startObject(TRANSFORM_ID) .field(TYPE, KEYWORD) .endObject() .startObject(AbstractAuditMessage.LEVEL.getPreferredName()) @@ -125,7 +127,8 @@ private static XContentBuilder mappings() throws IOException { builder.startObject(DataFrameField.INDEX_DOC_TYPE.getPreferredName()).field(TYPE, KEYWORD).endObject(); // add the schema for transform configurations addDataFrameTransformsConfigMappings(builder); - + // add the schema for transform stats + addDataFrameTransformsStatsMappings(builder); // end type builder.endObject(); // end properties @@ -135,6 +138,41 @@ private static XContentBuilder mappings() throws IOException { return builder; } + + private static XContentBuilder addDataFrameTransformsStatsMappings(XContentBuilder builder) throws IOException { + return builder + .startObject(DataFrameIndexerTransformStats.NUM_PAGES.getPreferredName()) + .field(TYPE, LONG) + .endObject() + .startObject(DataFrameIndexerTransformStats.NUM_INPUT_DOCUMENTS.getPreferredName()) + .field(TYPE, LONG) + .endObject() + .startObject(DataFrameIndexerTransformStats.NUM_OUTPUT_DOCUMENTS.getPreferredName()) + .field(TYPE, LONG) + .endObject() + .startObject(DataFrameIndexerTransformStats.NUM_INVOCATIONS.getPreferredName()) + .field(TYPE, LONG) + .endObject() + .startObject(DataFrameIndexerTransformStats.INDEX_TIME_IN_MS.getPreferredName()) + .field(TYPE, LONG) + .endObject() + .startObject(DataFrameIndexerTransformStats.SEARCH_TIME_IN_MS.getPreferredName()) + .field(TYPE, LONG) + .endObject() + .startObject(DataFrameIndexerTransformStats.INDEX_TOTAL.getPreferredName()) + .field(TYPE, LONG) + .endObject() + .startObject(DataFrameIndexerTransformStats.SEARCH_TOTAL.getPreferredName()) + .field(TYPE, LONG) + .endObject() + .startObject(DataFrameIndexerTransformStats.SEARCH_FAILURES.getPreferredName()) + .field(TYPE, LONG) + .endObject() + .startObject(DataFrameIndexerTransformStats.INDEX_FAILURES.getPreferredName()) + .field(TYPE, LONG) + .endObject(); + } + private static XContentBuilder addDataFrameTransformsConfigMappings(XContentBuilder builder) throws IOException { return builder .startObject(DataFrameField.ID.getPreferredName()) diff --git a/x-pack/plugin/data-frame/src/main/java/org/elasticsearch/xpack/dataframe/persistence/DataFrameTransformsConfigManager.java b/x-pack/plugin/data-frame/src/main/java/org/elasticsearch/xpack/dataframe/persistence/DataFrameTransformsConfigManager.java index 1392e95e79a5d..a19a0b65d85f1 100644 --- a/x-pack/plugin/data-frame/src/main/java/org/elasticsearch/xpack/dataframe/persistence/DataFrameTransformsConfigManager.java +++ b/x-pack/plugin/data-frame/src/main/java/org/elasticsearch/xpack/dataframe/persistence/DataFrameTransformsConfigManager.java @@ -17,9 +17,13 @@ import org.elasticsearch.action.get.GetRequest; import org.elasticsearch.action.index.IndexAction; import org.elasticsearch.action.index.IndexRequest; +import org.elasticsearch.action.search.SearchRequest; +import org.elasticsearch.action.search.SearchResponse; import org.elasticsearch.action.support.WriteRequest; import org.elasticsearch.client.Client; +import org.elasticsearch.common.Strings; import org.elasticsearch.common.bytes.BytesReference; +import org.elasticsearch.common.regex.Regex; import org.elasticsearch.common.xcontent.LoggingDeprecationHandler; import org.elasticsearch.common.xcontent.NamedXContentRegistry; import org.elasticsearch.common.xcontent.ToXContent; @@ -29,18 +33,26 @@ import org.elasticsearch.common.xcontent.XContentType; import org.elasticsearch.index.IndexNotFoundException; import org.elasticsearch.index.engine.VersionConflictEngineException; +import org.elasticsearch.index.query.BoolQueryBuilder; import org.elasticsearch.index.query.QueryBuilder; import org.elasticsearch.index.query.QueryBuilders; import org.elasticsearch.index.reindex.DeleteByQueryAction; import org.elasticsearch.index.reindex.DeleteByQueryRequest; +import org.elasticsearch.search.SearchHit; +import org.elasticsearch.search.sort.SortOrder; +import org.elasticsearch.xpack.core.action.util.ExpandedIdsMatcher; +import org.elasticsearch.xpack.core.action.util.PageParams; import org.elasticsearch.xpack.core.dataframe.DataFrameField; import org.elasticsearch.xpack.core.dataframe.DataFrameMessages; +import org.elasticsearch.xpack.core.dataframe.transforms.DataFrameIndexerTransformStats; import org.elasticsearch.xpack.core.dataframe.transforms.DataFrameTransformCheckpoint; import org.elasticsearch.xpack.core.dataframe.transforms.DataFrameTransformConfig; import java.io.IOException; import java.io.InputStream; +import java.util.ArrayList; import java.util.Collections; +import java.util.List; import java.util.Map; import static org.elasticsearch.xpack.core.ClientHelper.DATA_FRAME_ORIGIN; @@ -172,6 +184,61 @@ public void getTransformConfiguration(String transformId, ActionListener> foundIdsListener) { + String[] idTokens = ExpandedIdsMatcher.tokenizeExpression(transformIdsExpression); + QueryBuilder queryBuilder = buildQueryFromTokenizedIds(idTokens, DataFrameTransformConfig.NAME); + + SearchRequest request = client.prepareSearch(DataFrameInternalIndex.INDEX_NAME) + .addSort(DataFrameField.ID.getPreferredName(), SortOrder.ASC) + .setFrom(pageParams.getFrom()) + .setSize(pageParams.getSize()) + .setQuery(queryBuilder) + // We only care about the `id` field, small optimization + .setFetchSource(DataFrameField.ID.getPreferredName(), "") + .request(); + + final ExpandedIdsMatcher requiredMatches = new ExpandedIdsMatcher(idTokens, true); + + executeAsyncWithOrigin(client.threadPool().getThreadContext(), DATA_FRAME_ORIGIN, request, + ActionListener.wrap( + searchResponse -> { + List ids = new ArrayList<>(searchResponse.getHits().getHits().length); + for (SearchHit hit : searchResponse.getHits().getHits()) { + BytesReference source = hit.getSourceRef(); + try (InputStream stream = source.streamInput(); + XContentParser parser = XContentFactory.xContent(XContentType.JSON).createParser(NamedXContentRegistry.EMPTY, + LoggingDeprecationHandler.INSTANCE, stream)) { + ids.add((String) parser.map().get(DataFrameField.ID.getPreferredName())); + } catch (IOException e) { + foundIdsListener.onFailure(new ElasticsearchParseException("failed to parse search hit for ids", e)); + return; + } + } + requiredMatches.filterMatchedIds(ids); + if (requiredMatches.hasUnmatchedIds()) { + // some required Ids were not found + foundIdsListener.onFailure( + new ResourceNotFoundException( + DataFrameMessages.getMessage(DataFrameMessages.REST_DATA_FRAME_UNKNOWN_TRANSFORM, + requiredMatches.unmatchedIdsString()))); + return; + } + foundIdsListener.onResponse(ids); + }, + foundIdsListener::onFailure + ), client::search); + } + /** * This deletes the configuration and all other documents corresponding to the transform id (e.g. checkpoints). * @@ -206,6 +273,58 @@ public void deleteTransform(String transformId, ActionListener listener })); } + public void putOrUpdateTransformStats(DataFrameIndexerTransformStats stats, ActionListener listener) { + try (XContentBuilder builder = XContentFactory.jsonBuilder()) { + XContentBuilder source = stats.toXContent(builder, new ToXContent.MapParams(TO_XCONTENT_PARAMS)); + + IndexRequest indexRequest = new IndexRequest(DataFrameInternalIndex.INDEX_NAME) + .setRefreshPolicy(WriteRequest.RefreshPolicy.IMMEDIATE) + .id(DataFrameIndexerTransformStats.documentId(stats.getTransformId())) + .source(source); + + executeAsyncWithOrigin(client, DATA_FRAME_ORIGIN, IndexAction.INSTANCE, indexRequest, ActionListener.wrap( + r -> listener.onResponse(true), + e -> listener.onFailure(new RuntimeException( + DataFrameMessages.getMessage(DataFrameMessages.DATA_FRAME_FAILED_TO_PERSIST_STATS, stats.getTransformId()), + e)) + )); + } catch (IOException e) { + // not expected to happen but for the sake of completeness + listener.onFailure(new ElasticsearchParseException( + DataFrameMessages.getMessage(DataFrameMessages.DATA_FRAME_FAILED_TO_PERSIST_STATS, stats.getTransformId()), + e)); + } + } + + public void getTransformStats(String transformId, ActionListener resultListener) { + GetRequest getRequest = new GetRequest(DataFrameInternalIndex.INDEX_NAME, DataFrameIndexerTransformStats.documentId(transformId)); + executeAsyncWithOrigin(client, DATA_FRAME_ORIGIN, GetAction.INSTANCE, getRequest, ActionListener.wrap(getResponse -> { + + if (getResponse.isExists() == false) { + resultListener.onFailure(new ResourceNotFoundException( + DataFrameMessages.getMessage(DataFrameMessages.DATA_FRAME_UNKNOWN_TRANSFORM_STATS, transformId))); + return; + } + BytesReference source = getResponse.getSourceAsBytesRef(); + try (InputStream stream = source.streamInput(); + XContentParser parser = XContentFactory.xContent(XContentType.JSON) + .createParser(xContentRegistry, LoggingDeprecationHandler.INSTANCE, stream)) { + resultListener.onResponse(DataFrameIndexerTransformStats.fromXContent(parser)); + } catch (Exception e) { + logger.error( + DataFrameMessages.getMessage(DataFrameMessages.FAILED_TO_PARSE_TRANSFORM_STATISTICS_CONFIGURATION, transformId), e); + resultListener.onFailure(e); + } + }, e -> { + if (e instanceof ResourceNotFoundException) { + resultListener.onFailure(new ResourceNotFoundException( + DataFrameMessages.getMessage(DataFrameMessages.DATA_FRAME_UNKNOWN_TRANSFORM_STATS, transformId))); + } else { + resultListener.onFailure(e); + } + })); + } + private void parseTransformLenientlyFromSource(BytesReference source, String transformId, ActionListener transformListener) { try (InputStream stream = source.streamInput(); @@ -229,4 +348,28 @@ private void parseCheckpointsLenientlyFromSource(BytesReference source, String t transformListener.onFailure(e); } } + + private QueryBuilder buildQueryFromTokenizedIds(String[] idTokens, String resourceName) { + BoolQueryBuilder queryBuilder = QueryBuilders.boolQuery() + .filter(QueryBuilders.termQuery(DataFrameField.INDEX_DOC_TYPE.getPreferredName(), resourceName)); + if (Strings.isAllOrWildcard(idTokens) == false) { + List terms = new ArrayList<>(); + BoolQueryBuilder shouldQueries = new BoolQueryBuilder(); + for (String token : idTokens) { + if (Regex.isSimpleMatchPattern(token)) { + shouldQueries.should(QueryBuilders.wildcardQuery(DataFrameField.ID.getPreferredName(), token)); + } else { + terms.add(token); + } + } + if (terms.isEmpty() == false) { + shouldQueries.should(QueryBuilders.termsQuery(DataFrameField.ID.getPreferredName(), terms)); + } + + if (shouldQueries.should().isEmpty() == false) { + queryBuilder.filter(shouldQueries); + } + } + return QueryBuilders.constantScoreQuery(queryBuilder); + } } diff --git a/x-pack/plugin/data-frame/src/main/java/org/elasticsearch/xpack/dataframe/rest/action/RestGetDataFrameTransformsStatsAction.java b/x-pack/plugin/data-frame/src/main/java/org/elasticsearch/xpack/dataframe/rest/action/RestGetDataFrameTransformsStatsAction.java index 0609c2d499fc7..87cc13edbc329 100644 --- a/x-pack/plugin/data-frame/src/main/java/org/elasticsearch/xpack/dataframe/rest/action/RestGetDataFrameTransformsStatsAction.java +++ b/x-pack/plugin/data-frame/src/main/java/org/elasticsearch/xpack/dataframe/rest/action/RestGetDataFrameTransformsStatsAction.java @@ -12,6 +12,7 @@ import org.elasticsearch.rest.RestController; import org.elasticsearch.rest.RestRequest; import org.elasticsearch.rest.action.RestToXContentListener; +import org.elasticsearch.xpack.core.action.util.PageParams; import org.elasticsearch.xpack.core.dataframe.DataFrameField; import org.elasticsearch.xpack.core.dataframe.action.GetDataFrameTransformsStatsAction; @@ -27,6 +28,11 @@ public RestGetDataFrameTransformsStatsAction(Settings settings, RestController c protected RestChannelConsumer prepareRequest(RestRequest restRequest, NodeClient client) { String id = restRequest.param(DataFrameField.ID.getPreferredName()); GetDataFrameTransformsStatsAction.Request request = new GetDataFrameTransformsStatsAction.Request(id); + if (restRequest.hasParam(PageParams.FROM.getPreferredName()) || restRequest.hasParam(PageParams.SIZE.getPreferredName())) { + request.setPageParams( + new PageParams(restRequest.paramAsInt(PageParams.FROM.getPreferredName(), PageParams.DEFAULT_FROM), + restRequest.paramAsInt(PageParams.SIZE.getPreferredName(), PageParams.DEFAULT_SIZE))); + } return channel -> client.execute(GetDataFrameTransformsStatsAction.INSTANCE, request, new RestToXContentListener<>(channel)); } diff --git a/x-pack/plugin/data-frame/src/main/java/org/elasticsearch/xpack/dataframe/transforms/DataFrameIndexer.java b/x-pack/plugin/data-frame/src/main/java/org/elasticsearch/xpack/dataframe/transforms/DataFrameIndexer.java index bb07722ddeed0..090a9c9cfccc0 100644 --- a/x-pack/plugin/data-frame/src/main/java/org/elasticsearch/xpack/dataframe/transforms/DataFrameIndexer.java +++ b/x-pack/plugin/data-frame/src/main/java/org/elasticsearch/xpack/dataframe/transforms/DataFrameIndexer.java @@ -39,8 +39,11 @@ public abstract class DataFrameIndexer extends AsyncTwoPhaseIndexer initialState, Map initialPosition) { - super(executor, initialState, initialPosition, new DataFrameIndexerTransformStats()); + public DataFrameIndexer(Executor executor, + AtomicReference initialState, + Map initialPosition, + DataFrameIndexerTransformStats jobStats) { + super(executor, initialState, initialPosition, jobStats); } protected abstract DataFrameTransformConfig getConfig(); diff --git a/x-pack/plugin/data-frame/src/main/java/org/elasticsearch/xpack/dataframe/transforms/DataFrameTransformPersistentTasksExecutor.java b/x-pack/plugin/data-frame/src/main/java/org/elasticsearch/xpack/dataframe/transforms/DataFrameTransformPersistentTasksExecutor.java index d53354db2aa70..e3c27fd21fe03 100644 --- a/x-pack/plugin/data-frame/src/main/java/org/elasticsearch/xpack/dataframe/transforms/DataFrameTransformPersistentTasksExecutor.java +++ b/x-pack/plugin/data-frame/src/main/java/org/elasticsearch/xpack/dataframe/transforms/DataFrameTransformPersistentTasksExecutor.java @@ -8,6 +8,8 @@ import org.apache.logging.log4j.LogManager; import org.apache.logging.log4j.Logger; +import org.elasticsearch.ResourceNotFoundException; +import org.elasticsearch.action.ActionListener; import org.elasticsearch.client.Client; import org.elasticsearch.common.Nullable; import org.elasticsearch.persistent.AllocatedPersistentTask; @@ -60,18 +62,33 @@ protected void nodeOperation(AllocatedPersistentTask task, @Nullable DataFrameTr DataFrameTransformTask buildTask = (DataFrameTransformTask) task; SchedulerEngine.Job schedulerJob = new SchedulerEngine.Job( DataFrameTransformTask.SCHEDULE_NAME + "_" + params.getId(), next()); - DataFrameTransformState transformState = (DataFrameTransformState) state; if (transformState != null && transformState.getTaskState() == DataFrameTransformTaskState.FAILED) { logger.warn("Tried to start failed transform [" + params.getId() + "] failure reason: " + transformState.getReason()); return; } + transformsConfigManager.getTransformStats(params.getId(), ActionListener.wrap( + stats -> { + // Initialize with the previously recorded stats + buildTask.initializePreviousStats(stats); + scheduleTask(buildTask, schedulerJob, params.getId()); + }, + error -> { + if (error instanceof ResourceNotFoundException == false) { + logger.error("Unable to load previously persisted statistics for transform [" + params.getId() + "]", error); + } + scheduleTask(buildTask, schedulerJob, params.getId()); + } + )); + } + + private void scheduleTask(DataFrameTransformTask buildTask, SchedulerEngine.Job schedulerJob, String id) { // Note that while the task is added to the scheduler here, the internal state will prevent // it from doing any work until the task is "started" via the StartTransform api schedulerEngine.register(buildTask); schedulerEngine.add(schedulerJob); - logger.info("Data frame transform [" + params.getId() + "] created."); + logger.info("Data frame transform [" + id + "] created."); } static SchedulerEngine.Schedule next() { diff --git a/x-pack/plugin/data-frame/src/main/java/org/elasticsearch/xpack/dataframe/transforms/DataFrameTransformTask.java b/x-pack/plugin/data-frame/src/main/java/org/elasticsearch/xpack/dataframe/transforms/DataFrameTransformTask.java index b8bc2870307aa..23884afec3348 100644 --- a/x-pack/plugin/data-frame/src/main/java/org/elasticsearch/xpack/dataframe/transforms/DataFrameTransformTask.java +++ b/x-pack/plugin/data-frame/src/main/java/org/elasticsearch/xpack/dataframe/transforms/DataFrameTransformTask.java @@ -64,6 +64,7 @@ public class DataFrameTransformTask extends AllocatedPersistentTask implements S private final ThreadPool threadPool; private final DataFrameIndexer indexer; private final Auditor auditor; + private final DataFrameIndexerTransformStats previousStats; private final AtomicReference taskState; private final AtomicReference stateReason; @@ -110,6 +111,7 @@ public DataFrameTransformTask(long id, String type, String action, TaskId parent this.indexer = new ClientDataFrameIndexer(transform.getId(), transformsConfigManager, transformsCheckpointService, new AtomicReference<>(initialState), initialPosition, client, auditor); this.generation = new AtomicReference<>(initialGeneration); + this.previousStats = new DataFrameIndexerTransformStats(transform.getId()); this.taskState = new AtomicReference<>(initialTaskState); this.stateReason = new AtomicReference<>(initialReason); this.failureCount = new AtomicInteger(0); @@ -131,8 +133,12 @@ public DataFrameTransformState getState() { return new DataFrameTransformState(taskState.get(), indexer.getState(), indexer.getPosition(), generation.get(), stateReason.get()); } + void initializePreviousStats(DataFrameIndexerTransformStats stats) { + previousStats.merge(stats); + } + public DataFrameIndexerTransformStats getStats() { - return indexer.getStats(); + return new DataFrameIndexerTransformStats(previousStats).merge(indexer.getStats()); } public long getGeneration() { @@ -297,6 +303,7 @@ protected class ClientDataFrameIndexer extends DataFrameIndexer { private final DataFrameTransformsCheckpointService transformsCheckpointService; private final String transformId; private final Auditor auditor; + private volatile DataFrameIndexerTransformStats previouslyPersistedStats = null; // Keeps track of the last exception that was written to our audit, keeps us from spamming the audit index private volatile String lastAuditedExceptionMessage = null; private Map fieldMappings = null; @@ -307,7 +314,8 @@ public ClientDataFrameIndexer(String transformId, DataFrameTransformsConfigManag DataFrameTransformsCheckpointService transformsCheckpointService, AtomicReference initialState, Map initialPosition, Client client, Auditor auditor) { - super(threadPool.executor(ThreadPool.Names.GENERIC), initialState, initialPosition); + super(threadPool.executor(ThreadPool.Names.GENERIC), initialState, initialPosition, + new DataFrameIndexerTransformStats(transformId)); this.transformId = transformId; this.transformsConfigManager = transformsConfigManager; this.transformsCheckpointService = transformsCheckpointService; @@ -422,7 +430,39 @@ protected void doSaveState(IndexerState indexerState, Map positi generation.get(), stateReason.get()); logger.info("Updating persistent state of transform [" + transform.getId() + "] to [" + state.toString() + "]"); - persistStateToClusterState(state, ActionListener.wrap(t -> next.run(), e -> next.run())); + + // Persisting stats when we call `doSaveState` should be ok as we only call it on a state transition and + // only every-so-often when doing the bulk indexing calls. See AsyncTwoPhaseIndexer#onBulkResponse for current periodicity + ActionListener> updateClusterStateListener = ActionListener.wrap( + task -> { + // Make a copy of the previousStats so that they are not constantly updated when `merge` is called + DataFrameIndexerTransformStats tempStats = new DataFrameIndexerTransformStats(previousStats).merge(getStats()); + + // Only persist the stats if something has actually changed + if (previouslyPersistedStats == null || previouslyPersistedStats.equals(tempStats) == false) { + transformsConfigManager.putOrUpdateTransformStats(tempStats, + ActionListener.wrap( + r -> { + previouslyPersistedStats = tempStats; + next.run(); + }, + statsExc -> { + logger.error("Updating stats of transform [" + transform.getId() + "] failed", statsExc); + next.run(); + } + )); + // The stats that we have previously written to the doc is the same as as it is now, no need to update it + } else { + next.run(); + } + }, + exc -> { + logger.error("Updating persistent state of transform [" + transform.getId() + "] failed", exc); + next.run(); + } + ); + + persistStateToClusterState(state, updateClusterStateListener); } @Override diff --git a/x-pack/plugin/data-frame/src/test/java/org/elasticsearch/xpack/dataframe/DataFrameFeatureSetTests.java b/x-pack/plugin/data-frame/src/test/java/org/elasticsearch/xpack/dataframe/DataFrameFeatureSetTests.java index bea7e5b0148e0..f70cbb8b277c2 100644 --- a/x-pack/plugin/data-frame/src/test/java/org/elasticsearch/xpack/dataframe/DataFrameFeatureSetTests.java +++ b/x-pack/plugin/data-frame/src/test/java/org/elasticsearch/xpack/dataframe/DataFrameFeatureSetTests.java @@ -6,8 +6,10 @@ package org.elasticsearch.xpack.dataframe; +import org.elasticsearch.action.search.SearchResponse; import org.elasticsearch.action.support.PlainActionFuture; import org.elasticsearch.client.Client; +import org.elasticsearch.cluster.service.ClusterService; import org.elasticsearch.common.settings.Settings; import org.elasticsearch.common.xcontent.ToXContent; import org.elasticsearch.common.xcontent.XContentBuilder; @@ -15,25 +17,24 @@ import org.elasticsearch.common.xcontent.XContentParser; import org.elasticsearch.common.xcontent.support.XContentMapValues; import org.elasticsearch.license.XPackLicenseState; +import org.elasticsearch.search.aggregations.Aggregation; +import org.elasticsearch.search.aggregations.Aggregations; +import org.elasticsearch.search.aggregations.metrics.NumericMetricsAggregation; import org.elasticsearch.test.ESTestCase; import org.elasticsearch.xpack.core.XPackFeatureSet; import org.elasticsearch.xpack.core.XPackFeatureSet.Usage; -import org.elasticsearch.xpack.core.dataframe.DataFrameFeatureSetUsage; import org.elasticsearch.xpack.core.dataframe.transforms.DataFrameIndexerTransformStats; -import org.elasticsearch.xpack.core.dataframe.transforms.DataFrameTransformConfig; -import org.elasticsearch.xpack.core.dataframe.transforms.DataFrameTransformConfigTests; -import org.elasticsearch.xpack.core.dataframe.transforms.DataFrameTransformStateAndStats; -import org.elasticsearch.xpack.core.dataframe.transforms.DataFrameTransformStateAndStatsTests; import org.junit.Before; import java.io.IOException; import java.util.ArrayList; -import java.util.HashMap; +import java.util.Collections; import java.util.List; import java.util.Map; import java.util.concurrent.ExecutionException; -import static java.lang.Math.toIntExact; +import static org.elasticsearch.xpack.dataframe.DataFrameFeatureSet.PROVIDED_STATS; +import static org.hamcrest.CoreMatchers.equalTo; import static org.hamcrest.core.Is.is; import static org.mockito.Mockito.mock; import static org.mockito.Mockito.when; @@ -47,7 +48,10 @@ public void init() { } public void testAvailable() { - DataFrameFeatureSet featureSet = new DataFrameFeatureSet(Settings.EMPTY, mock(Client.class), licenseState); + DataFrameFeatureSet featureSet = new DataFrameFeatureSet(Settings.EMPTY, + mock(ClusterService.class), + mock(Client.class), + licenseState); boolean available = randomBoolean(); when(licenseState.isDataFrameAllowed()).thenReturn(available); assertThat(featureSet.available(), is(available)); @@ -57,89 +61,67 @@ public void testEnabledSetting() { boolean enabled = randomBoolean(); Settings.Builder settings = Settings.builder(); settings.put("xpack.data_frame.enabled", enabled); - DataFrameFeatureSet featureSet = new DataFrameFeatureSet(settings.build(), mock(Client.class), licenseState); + DataFrameFeatureSet featureSet = new DataFrameFeatureSet(settings.build(), + mock(ClusterService.class), + mock(Client.class), + licenseState); assertThat(featureSet.enabled(), is(enabled)); } public void testEnabledDefault() { - DataFrameFeatureSet featureSet = new DataFrameFeatureSet(Settings.EMPTY, mock(Client.class), licenseState); + DataFrameFeatureSet featureSet = new DataFrameFeatureSet(Settings.EMPTY, + mock(ClusterService.class), + mock(Client.class), + licenseState); assertTrue(featureSet.enabled()); } - public void testUsage() throws IOException { - List transformsStateAndStats = new ArrayList<>(); - int count = randomIntBetween(0, 10); - int uniqueId = 0; - for (int i = 0; i < count; ++i) { - transformsStateAndStats.add( - DataFrameTransformStateAndStatsTests.randomDataFrameTransformStateAndStats("df-" + Integer.toString(uniqueId++))); + public void testParseSearchAggs() { + Aggregations emptyAggs = new Aggregations(Collections.emptyList()); + SearchResponse withEmptyAggs = mock(SearchResponse.class); + when(withEmptyAggs.getAggregations()).thenReturn(emptyAggs); + + assertThat(DataFrameFeatureSet.parseSearchAggs(withEmptyAggs), equalTo(DataFrameIndexerTransformStats.withDefaultTransformId())); + + DataFrameIndexerTransformStats expectedStats = new DataFrameIndexerTransformStats("_all", + 1, // numPages + 2, // numInputDocuments + 3, // numOutputDocuments + 4, // numInvocations + 5, // indexTime + 6, // searchTime + 7, // indexTotal + 8, // searchTotal + 9, // indexFailures + 10); // searchFailures + + int currentStat = 1; + List aggs = new ArrayList<>(PROVIDED_STATS.length); + for (String statName : PROVIDED_STATS) { + aggs.add(buildAgg(statName, (double) currentStat++)); } + Aggregations aggregations = new Aggregations(aggs); + SearchResponse withAggs = mock(SearchResponse.class); + when(withAggs.getAggregations()).thenReturn(aggregations); - count = randomIntBetween(0, 10); - List transformConfigWithoutTasks = new ArrayList<>(); - for (int i = 0; i < count; ++i) { - transformConfigWithoutTasks.add( - DataFrameTransformConfigTests.randomDataFrameTransformConfig("df-" + Integer.toString(uniqueId++))); - } - - List transformConfigWithTasks = - new ArrayList<>(transformsStateAndStats.size() + transformConfigWithoutTasks.size()); - - transformsStateAndStats.forEach(stats -> - transformConfigWithTasks.add(DataFrameTransformConfigTests.randomDataFrameTransformConfig(stats.getId()))); - transformConfigWithoutTasks.forEach(withoutTask -> - transformsStateAndStats.add(DataFrameTransformStateAndStats.initialStateAndStats(withoutTask.getId()))); - - boolean enabled = randomBoolean(); - boolean available = randomBoolean(); - DataFrameFeatureSetUsage usage = DataFrameFeatureSet.createUsage(available, - enabled, - transformsStateAndStats); - - assertEquals(enabled, usage.enabled()); - try (XContentBuilder builder = XContentFactory.jsonBuilder()) { - usage.toXContent(builder, ToXContent.EMPTY_PARAMS); + assertThat(DataFrameFeatureSet.parseSearchAggs(withAggs), equalTo(expectedStats)); + } - XContentParser parser = createParser(builder); - Map usageAsMap = parser.map(); - assertEquals(available, (boolean) XContentMapValues.extractValue("available", usageAsMap)); - - if (transformsStateAndStats.isEmpty() && transformConfigWithoutTasks.isEmpty()) { - // no transforms, no stats - assertEquals(null, XContentMapValues.extractValue("transforms", usageAsMap)); - assertEquals(null, XContentMapValues.extractValue("stats", usageAsMap)); - } else { - assertEquals(transformsStateAndStats.size(), XContentMapValues.extractValue("transforms._all", usageAsMap)); - - Map stateCounts = new HashMap<>(); - transformsStateAndStats.stream() - .map(x -> x.getTransformState().getIndexerState().value()) - .forEach(x -> stateCounts.merge(x, 1, Integer::sum)); - stateCounts.forEach((k, v) -> assertEquals(v, XContentMapValues.extractValue("transforms." + k, usageAsMap))); - - // use default constructed stats object for assertions if transformsStateAndStats is empty - DataFrameIndexerTransformStats combinedStats = new DataFrameIndexerTransformStats(); - if (transformsStateAndStats.isEmpty() == false) { - combinedStats = transformsStateAndStats.stream().map(x -> x.getTransformStats()).reduce((l, r) -> l.merge(r)).get(); - } - - assertEquals(toIntExact(combinedStats.getIndexFailures()), - XContentMapValues.extractValue("stats.index_failures", usageAsMap)); - assertEquals(toIntExact(combinedStats.getIndexTotal()), - XContentMapValues.extractValue("stats.index_total", usageAsMap)); - assertEquals(toIntExact(combinedStats.getSearchTime()), - XContentMapValues.extractValue("stats.search_time_in_ms", usageAsMap)); - assertEquals(toIntExact(combinedStats.getNumDocuments()), - XContentMapValues.extractValue("stats.documents_processed", usageAsMap)); - } - } + private static Aggregation buildAgg(String name, double value) { + NumericMetricsAggregation.SingleValue agg = mock(NumericMetricsAggregation.SingleValue.class); + when(agg.getName()).thenReturn(name); + when(agg.value()).thenReturn(value); + return agg; } public void testUsageDisabled() throws IOException, InterruptedException, ExecutionException { when(licenseState.isDataFrameAllowed()).thenReturn(true); Settings.Builder settings = Settings.builder(); settings.put("xpack.data_frame.enabled", false); - DataFrameFeatureSet featureSet = new DataFrameFeatureSet(settings.build(), mock(Client.class), licenseState); + DataFrameFeatureSet featureSet = new DataFrameFeatureSet(settings.build(), + mock(ClusterService.class), + mock(Client.class), + licenseState); PlainActionFuture future = new PlainActionFuture<>(); featureSet.usage(future); XPackFeatureSet.Usage usage = future.get(); diff --git a/x-pack/plugin/data-frame/src/test/java/org/elasticsearch/xpack/dataframe/checkpoint/DataFrameTransformsCheckpointServiceTests.java b/x-pack/plugin/data-frame/src/test/java/org/elasticsearch/xpack/dataframe/checkpoint/DataFrameTransformsCheckpointServiceTests.java index 0868315165cdc..9cc2769e7d149 100644 --- a/x-pack/plugin/data-frame/src/test/java/org/elasticsearch/xpack/dataframe/checkpoint/DataFrameTransformsCheckpointServiceTests.java +++ b/x-pack/plugin/data-frame/src/test/java/org/elasticsearch/xpack/dataframe/checkpoint/DataFrameTransformsCheckpointServiceTests.java @@ -82,7 +82,6 @@ public void testExtractIndexCheckpointsLostPrimaries() { } } - @AwaitsFix(bugUrl = "https://github.com/elastic/elasticsearch/issues/40368") public void testExtractIndexCheckpointsInconsistentGlobalCheckpoints() { Map expectedCheckpoints = new HashMap<>(); Set indices = randomUserIndices(); @@ -161,7 +160,7 @@ private static ShardStats[] createRandomShardStats(Map expectedC long globalCheckpoint = randomBoolean() ? localCheckpoint : randomLongBetween(0L, 100000000L); long maxSeqNo = Math.max(localCheckpoint, globalCheckpoint); - SeqNoStats seqNoStats = new SeqNoStats(maxSeqNo, localCheckpoint, globalCheckpoint); + final SeqNoStats validSeqNoStats = new SeqNoStats(maxSeqNo, localCheckpoint, globalCheckpoint); checkpoints.add(globalCheckpoint); for (int replica = 0; replica < numShardCopies; replica++) { @@ -194,10 +193,16 @@ private static ShardStats[] createRandomShardStats(Map expectedC if (inconsistentReplica == replica) { // overwrite - seqNoStats = new SeqNoStats(maxSeqNo, localCheckpoint, globalCheckpoint + randomLongBetween(10L, 100L)); + SeqNoStats invalidSeqNoStats = + new SeqNoStats(maxSeqNo, localCheckpoint, globalCheckpoint + randomLongBetween(10L, 100L)); + shardStats.add( + new ShardStats(shardRouting, + new ShardPath(false, path, path, shardId), stats, null, invalidSeqNoStats, null)); + } else { + shardStats.add( + new ShardStats(shardRouting, + new ShardPath(false, path, path, shardId), stats, null, validSeqNoStats, null)); } - - shardStats.add(new ShardStats(shardRouting, new ShardPath(false, path, path, shardId), stats, null, seqNoStats, null)); } } diff --git a/x-pack/plugin/data-frame/src/test/java/org/elasticsearch/xpack/dataframe/persistence/DataFrameTransformsConfigManagerTests.java b/x-pack/plugin/data-frame/src/test/java/org/elasticsearch/xpack/dataframe/persistence/DataFrameTransformsConfigManagerTests.java index f9c5d405fe665..8fe384553bcec 100644 --- a/x-pack/plugin/data-frame/src/test/java/org/elasticsearch/xpack/dataframe/persistence/DataFrameTransformsConfigManagerTests.java +++ b/x-pack/plugin/data-frame/src/test/java/org/elasticsearch/xpack/dataframe/persistence/DataFrameTransformsConfigManagerTests.java @@ -8,6 +8,7 @@ import org.elasticsearch.ResourceAlreadyExistsException; import org.elasticsearch.ResourceNotFoundException; +import org.elasticsearch.xpack.core.action.util.PageParams; import org.elasticsearch.xpack.core.dataframe.DataFrameMessages; import org.elasticsearch.xpack.core.dataframe.transforms.DataFrameTransformCheckpoint; import org.elasticsearch.xpack.core.dataframe.transforms.DataFrameTransformCheckpointTests; @@ -15,6 +16,13 @@ import org.elasticsearch.xpack.core.dataframe.transforms.DataFrameTransformConfigTests; import org.junit.Before; +import java.util.Arrays; +import java.util.Collections; +import java.util.List; + +import static org.hamcrest.Matchers.equalTo; +import static org.hamcrest.Matchers.instanceOf; + public class DataFrameTransformsConfigManagerTests extends DataFrameSingleNodeTestCase { private DataFrameTransformsConfigManager transformsConfigManager; @@ -128,4 +136,84 @@ public void testCreateReadDeleteCheckPoint() throws InterruptedException { assertAsync(listener -> transformsConfigManager.getTransformCheckpoint(checkpoint.getTransformId(), checkpoint.getCheckpoint(), listener), DataFrameTransformCheckpoint.EMPTY, null, null); } + + public void testExpandIds() throws Exception { + DataFrameTransformConfig transformConfig1 = DataFrameTransformConfigTests.randomDataFrameTransformConfig("transform1_expand"); + DataFrameTransformConfig transformConfig2 = DataFrameTransformConfigTests.randomDataFrameTransformConfig("transform2_expand"); + DataFrameTransformConfig transformConfig3 = DataFrameTransformConfigTests.randomDataFrameTransformConfig("transform3_expand"); + + // create transform + assertAsync(listener -> transformsConfigManager.putTransformConfiguration(transformConfig1, listener), true, null, null); + assertAsync(listener -> transformsConfigManager.putTransformConfiguration(transformConfig2, listener), true, null, null); + assertAsync(listener -> transformsConfigManager.putTransformConfiguration(transformConfig3, listener), true, null, null); + + + // expand 1 id + assertAsync(listener -> + transformsConfigManager.expandTransformIds(transformConfig1.getId(), + PageParams.defaultParams(), + listener), + Collections.singletonList("transform1_expand"), + null, + null); + + // expand 2 ids explicitly + assertAsync(listener -> + transformsConfigManager.expandTransformIds("transform1_expand,transform2_expand", + PageParams.defaultParams(), + listener), + Arrays.asList("transform1_expand", "transform2_expand"), + null, + null); + + // expand 3 ids wildcard and explicit + assertAsync(listener -> + transformsConfigManager.expandTransformIds("transform1*,transform2_expand,transform3_expand", + PageParams.defaultParams(), + listener), + Arrays.asList("transform1_expand", "transform2_expand", "transform3_expand"), + null, + null); + + // expand 3 ids _all + assertAsync(listener -> + transformsConfigManager.expandTransformIds("_all", + PageParams.defaultParams(), + listener), + Arrays.asList("transform1_expand", "transform2_expand", "transform3_expand"), + null, + null); + + // expand 1 id _all with pagination + assertAsync(listener -> + transformsConfigManager.expandTransformIds("_all", + new PageParams(0, 1), + listener), + Collections.singletonList("transform1_expand"), + null, + null); + + // expand 2 later ids _all with pagination + assertAsync(listener -> + transformsConfigManager.expandTransformIds("_all", + new PageParams(1, 2), + listener), + Arrays.asList("transform2_expand", "transform3_expand"), + null, + null); + + // expand 1 id explicitly that does not exist + assertAsync(listener -> + transformsConfigManager.expandTransformIds("unknown,unknown2", + new PageParams(1, 2), + listener), + (List)null, + null, + e -> { + assertThat(e, instanceOf(ResourceNotFoundException.class)); + assertThat(e.getMessage(), + equalTo(DataFrameMessages.getMessage(DataFrameMessages.REST_DATA_FRAME_UNKNOWN_TRANSFORM, "unknown,unknown2"))); + }); + + } } diff --git a/x-pack/plugin/data-frame/src/test/java/org/elasticsearch/xpack/dataframe/transforms/pivot/AggregationResultUtilsTests.java b/x-pack/plugin/data-frame/src/test/java/org/elasticsearch/xpack/dataframe/transforms/pivot/AggregationResultUtilsTests.java index eedf6264f348b..c2c22dc6ffad4 100644 --- a/x-pack/plugin/data-frame/src/test/java/org/elasticsearch/xpack/dataframe/transforms/pivot/AggregationResultUtilsTests.java +++ b/x-pack/plugin/data-frame/src/test/java/org/elasticsearch/xpack/dataframe/transforms/pivot/AggregationResultUtilsTests.java @@ -501,7 +501,7 @@ aggTypedName, asMap( "value", 122.55), DOC_COUNT, 44) )); - DataFrameIndexerTransformStats stats = new DataFrameIndexerTransformStats(); + DataFrameIndexerTransformStats stats = DataFrameIndexerTransformStats.withDefaultTransformId(); Map fieldTypeMap = asStringMap( aggName, "double", @@ -534,7 +534,7 @@ aggTypedName, asMap( private void executeTest(GroupConfig groups, Collection aggregationBuilders, Map input, Map fieldTypeMap, List> expected, long expectedDocCounts) throws IOException { - DataFrameIndexerTransformStats stats = new DataFrameIndexerTransformStats(); + DataFrameIndexerTransformStats stats = DataFrameIndexerTransformStats.withDefaultTransformId(); XContentBuilder builder = XContentFactory.contentBuilder(randomFrom(XContentType.values())); builder.map(input); diff --git a/x-pack/plugin/ilm/src/test/java/org/elasticsearch/xpack/indexlifecycle/IndexLifecycleRunnerTests.java b/x-pack/plugin/ilm/src/test/java/org/elasticsearch/xpack/indexlifecycle/IndexLifecycleRunnerTests.java index 565a9723c8c6c..511d0e5be1ab9 100644 --- a/x-pack/plugin/ilm/src/test/java/org/elasticsearch/xpack/indexlifecycle/IndexLifecycleRunnerTests.java +++ b/x-pack/plugin/ilm/src/test/java/org/elasticsearch/xpack/indexlifecycle/IndexLifecycleRunnerTests.java @@ -385,14 +385,12 @@ public void testRunStateChangePolicyWithAsyncActionNextStep() throws Exception { ClusterState before = clusterService.state(); CountDownLatch latch = new CountDownLatch(1); step.setLatch(latch); + CountDownLatch asyncLatch = new CountDownLatch(1); + nextStep.setLatch(asyncLatch); runner.runPolicyAfterStateChange(policyName, indexMetaData); // Wait for the cluster state action step awaitLatch(latch, 5, TimeUnit.SECONDS); - - CountDownLatch asyncLatch = new CountDownLatch(1); - nextStep.setLatch(asyncLatch); - // Wait for the async action step awaitLatch(asyncLatch, 5, TimeUnit.SECONDS); ClusterState after = clusterService.state(); diff --git a/x-pack/plugin/rollup/src/main/java/org/elasticsearch/xpack/rollup/action/TransportRollupSearchAction.java b/x-pack/plugin/rollup/src/main/java/org/elasticsearch/xpack/rollup/action/TransportRollupSearchAction.java index 610275705eef8..e85a92c061366 100644 --- a/x-pack/plugin/rollup/src/main/java/org/elasticsearch/xpack/rollup/action/TransportRollupSearchAction.java +++ b/x-pack/plugin/rollup/src/main/java/org/elasticsearch/xpack/rollup/action/TransportRollupSearchAction.java @@ -56,17 +56,14 @@ import org.elasticsearch.xpack.core.rollup.RollupField; import org.elasticsearch.xpack.core.rollup.action.RollupJobCaps; import org.elasticsearch.xpack.core.rollup.action.RollupSearchAction; -import org.elasticsearch.xpack.core.rollup.job.DateHistogramGroupConfig; import org.elasticsearch.xpack.rollup.Rollup; import org.elasticsearch.xpack.rollup.RollupJobIdentifierUtils; import org.elasticsearch.xpack.rollup.RollupRequestTranslator; import org.elasticsearch.xpack.rollup.RollupResponseTranslator; -import org.joda.time.DateTimeZone; import java.io.IOException; import java.util.ArrayList; import java.util.Arrays; -import java.util.Collections; import java.util.HashSet; import java.util.List; import java.util.Objects; @@ -286,11 +283,8 @@ static QueryBuilder rewriteQuery(QueryBuilder builder, Set jobCap } else if (builder.getWriteableName().equals(RangeQueryBuilder.NAME)) { RangeQueryBuilder range = (RangeQueryBuilder) builder; String fieldName = range.fieldName(); - // Many range queries don't include the timezone because the default is UTC, but the query - // builder will return null so we need to set it here - String timeZone = range.timeZone() == null ? DateTimeZone.UTC.toString() : range.timeZone(); - String rewrittenFieldName = rewriteFieldName(jobCaps, RangeQueryBuilder.NAME, fieldName, timeZone); + String rewrittenFieldName = rewriteFieldName(jobCaps, RangeQueryBuilder.NAME, fieldName); RangeQueryBuilder rewritten = new RangeQueryBuilder(rewrittenFieldName) .from(range.from()) .to(range.to()) @@ -306,12 +300,12 @@ static QueryBuilder rewriteQuery(QueryBuilder builder, Set jobCap } else if (builder.getWriteableName().equals(TermQueryBuilder.NAME)) { TermQueryBuilder term = (TermQueryBuilder) builder; String fieldName = term.fieldName(); - String rewrittenFieldName = rewriteFieldName(jobCaps, TermQueryBuilder.NAME, fieldName, null); + String rewrittenFieldName = rewriteFieldName(jobCaps, TermQueryBuilder.NAME, fieldName); return new TermQueryBuilder(rewrittenFieldName, term.value()); } else if (builder.getWriteableName().equals(TermsQueryBuilder.NAME)) { TermsQueryBuilder terms = (TermsQueryBuilder) builder; String fieldName = terms.fieldName(); - String rewrittenFieldName = rewriteFieldName(jobCaps, TermQueryBuilder.NAME, fieldName, null); + String rewrittenFieldName = rewriteFieldName(jobCaps, TermQueryBuilder.NAME, fieldName); return new TermsQueryBuilder(rewrittenFieldName, terms.values()); } else if (builder.getWriteableName().equals(MatchAllQueryBuilder.NAME)) { // no-op @@ -321,11 +315,7 @@ static QueryBuilder rewriteQuery(QueryBuilder builder, Set jobCap } } - private static String rewriteFieldName(Set jobCaps, - String builderName, - String fieldName, - String timeZone) { - List incompatibleTimeZones = timeZone == null ? Collections.emptyList() : new ArrayList<>(); + private static String rewriteFieldName(Set jobCaps, String builderName, String fieldName) { List rewrittenFieldNames = jobCaps.stream() // We only care about job caps that have the query's target field .filter(caps -> caps.getFieldCaps().keySet().contains(fieldName)) @@ -335,17 +325,7 @@ private static String rewriteFieldName(Set jobCaps, // For now, we only allow filtering on grouping fields .filter(agg -> { String type = (String)agg.get(RollupField.AGG); - - // If the cap is for a date_histo, and the query is a range, the timezones need to match - if (type.equals(DateHistogramAggregationBuilder.NAME) && timeZone != null) { - boolean matchingTZ = ((String)agg.get(DateHistogramGroupConfig.TIME_ZONE)) - .equalsIgnoreCase(timeZone); - if (matchingTZ == false) { - incompatibleTimeZones.add((String)agg.get(DateHistogramGroupConfig.TIME_ZONE)); - } - return matchingTZ; - } - // Otherwise just make sure it's one of the three groups + // make sure it's one of the three groups return type.equals(TermsAggregationBuilder.NAME) || type.equals(DateHistogramAggregationBuilder.NAME) || type.equals(HistogramAggregationBuilder.NAME); @@ -363,14 +343,8 @@ private static String rewriteFieldName(Set jobCaps, .distinct() .collect(ArrayList::new, List::addAll, List::addAll); if (rewrittenFieldNames.isEmpty()) { - if (incompatibleTimeZones.isEmpty()) { - throw new IllegalArgumentException("Field [" + fieldName + "] in [" + builderName + throw new IllegalArgumentException("Field [" + fieldName + "] in [" + builderName + "] query is not available in selected rollup indices, cannot query."); - } else { - throw new IllegalArgumentException("Field [" + fieldName + "] in [" + builderName - + "] query was found in rollup indices, but requested timezone is not compatible. Options include: " - + incompatibleTimeZones); - } } else if (rewrittenFieldNames.size() > 1) { throw new IllegalArgumentException("Ambiguous field name resolution when mapping to rolled fields. Field name [" + fieldName + "] was mapped to: [" + Strings.collectionToDelimitedString(rewrittenFieldNames, ",") + "]."); diff --git a/x-pack/plugin/rollup/src/test/java/org/elasticsearch/xpack/rollup/action/SearchActionTests.java b/x-pack/plugin/rollup/src/test/java/org/elasticsearch/xpack/rollup/action/SearchActionTests.java index 0032b5a88a563..5a851d17e5eaf 100644 --- a/x-pack/plugin/rollup/src/test/java/org/elasticsearch/xpack/rollup/action/SearchActionTests.java +++ b/x-pack/plugin/rollup/src/test/java/org/elasticsearch/xpack/rollup/action/SearchActionTests.java @@ -140,16 +140,15 @@ public void testRangeNullTimeZone() { assertThat(((RangeQueryBuilder)rewritten).fieldName(), equalTo("foo.date_histogram.timestamp")); } - public void testRangeWrongTZ() { + public void testRangeDifferentTZ() { final GroupConfig groupConfig = new GroupConfig(new DateHistogramGroupConfig("foo", new DateHistogramInterval("1h"), null, "UTC")); final RollupJobConfig config = new RollupJobConfig("foo", "index", "rollup", "*/5 * * * * ?", 10, groupConfig, emptyList(), null); RollupJobCaps cap = new RollupJobCaps(config); Set caps = new HashSet<>(); caps.add(cap); - Exception e = expectThrows(IllegalArgumentException.class, - () -> TransportRollupSearchAction.rewriteQuery(new RangeQueryBuilder("foo").gt(1).timeZone("CET"), caps)); - assertThat(e.getMessage(), equalTo("Field [foo] in [range] query was found in rollup indices, but requested timezone is not " + - "compatible. Options include: [UTC]")); + QueryBuilder rewritten = TransportRollupSearchAction.rewriteQuery(new RangeQueryBuilder("foo").gt(1).timeZone("CET"), caps); + assertThat(rewritten, instanceOf(RangeQueryBuilder.class)); + assertThat(((RangeQueryBuilder)rewritten).fieldName(), equalTo("foo.date_histogram.timestamp")); } public void testTermQuery() { diff --git a/x-pack/plugin/security/src/main/java/org/elasticsearch/xpack/security/SecurityFeatureSet.java b/x-pack/plugin/security/src/main/java/org/elasticsearch/xpack/security/SecurityFeatureSet.java index 2e5832d0834e7..ce7e54a3cb29d 100644 --- a/x-pack/plugin/security/src/main/java/org/elasticsearch/xpack/security/SecurityFeatureSet.java +++ b/x-pack/plugin/security/src/main/java/org/elasticsearch/xpack/security/SecurityFeatureSet.java @@ -150,10 +150,18 @@ public void usage(ActionListener listener) { } static Map sslUsage(Settings settings) { - Map map = new HashMap<>(2); - map.put("http", singletonMap("enabled", HTTP_SSL_ENABLED.get(settings))); - map.put("transport", singletonMap("enabled", TRANSPORT_SSL_ENABLED.get(settings))); - return map; + // If security has been explicitly disabled in the settings, then SSL is also explicitly disabled, and we don't want to report + // these http/transport settings as they would be misleading (they could report `true` even though they were ignored) + // But, if security has not been explicitly configured, but has defaulted to off due to the current license type, + // then these SSL settings are still respected (that is SSL might be enabled, while the rest of security is disabled). + if (XPackSettings.SECURITY_ENABLED.get(settings)) { + Map map = new HashMap<>(2); + map.put("http", singletonMap("enabled", HTTP_SSL_ENABLED.get(settings))); + map.put("transport", singletonMap("enabled", TRANSPORT_SSL_ENABLED.get(settings))); + return map; + } else { + return Collections.emptyMap(); + } } static Map tokenServiceUsage(Settings settings) { diff --git a/x-pack/plugin/security/src/test/java/org/elasticsearch/xpack/security/SecurityFeatureSetTests.java b/x-pack/plugin/security/src/test/java/org/elasticsearch/xpack/security/SecurityFeatureSetTests.java index 146dc78698eca..2fc2ea8865d91 100644 --- a/x-pack/plugin/security/src/test/java/org/elasticsearch/xpack/security/SecurityFeatureSetTests.java +++ b/x-pack/plugin/security/src/test/java/org/elasticsearch/xpack/security/SecurityFeatureSetTests.java @@ -28,6 +28,7 @@ import org.elasticsearch.xpack.security.transport.filter.IPFilter; import org.junit.Before; +import java.io.IOException; import java.util.Arrays; import java.util.Collections; import java.util.HashMap; @@ -126,29 +127,10 @@ public void testUsage() throws Exception { final boolean rolesStoreEnabled = randomBoolean(); - doAnswer(invocationOnMock -> { - ActionListener> listener = (ActionListener>) invocationOnMock.getArguments()[0]; - if (rolesStoreEnabled) { - listener.onResponse(Collections.singletonMap("count", 1)); - } else { - listener.onResponse(Collections.emptyMap()); - } - return Void.TYPE; - }).when(rolesStore).usageStats(any(ActionListener.class)); + configureRoleStoreUsage(rolesStoreEnabled); final boolean roleMappingStoreEnabled = randomBoolean(); - doAnswer(invocationOnMock -> { - ActionListener> listener = (ActionListener) invocationOnMock.getArguments()[0]; - if (roleMappingStoreEnabled) { - final Map map = new HashMap<>(); - map.put("size", 12L); - map.put("enabled", 10L); - listener.onResponse(map); - } else { - listener.onResponse(Collections.emptyMap()); - } - return Void.TYPE; - }).when(roleMappingStore).usageStats(any(ActionListener.class)); + configureRoleMappingStoreUsage(roleMappingStoreEnabled); Map realmsUsageStats = new HashMap<>(); for (int i = 0; i < 5; i++) { @@ -158,11 +140,7 @@ public void testUsage() throws Exception { realmUsage.put("key2", Arrays.asList(i)); realmUsage.put("key3", Arrays.asList(i % 2 == 0)); } - doAnswer(invocationOnMock -> { - ActionListener> listener = (ActionListener) invocationOnMock.getArguments()[0]; - listener.onResponse(realmsUsageStats); - return Void.TYPE; - }).when(realms).usageStats(any(ActionListener.class)); + configureRealmsUsage(realmsUsageStats); final boolean anonymousEnabled = randomBoolean(); if (anonymousEnabled) { @@ -182,11 +160,7 @@ public void testUsage() throws Exception { assertThat(usage.name(), is(XPackField.SECURITY)); assertThat(usage.enabled(), is(enabled)); assertThat(usage.available(), is(authcAuthzAvailable)); - XContentSource source; - try (XContentBuilder builder = XContentFactory.jsonBuilder()) { - usage.toXContent(builder, ToXContent.EMPTY_PARAMS); - source = new XContentSource(builder); - } + XContentSource source = getXContentSource(usage); if (enabled) { if (authcAuthzAvailable) { @@ -251,4 +225,101 @@ public void testUsage() throws Exception { } } } + + public void testUsageOnTrialLicenseWithSecurityDisabledByDefault() throws Exception { + when(licenseState.isSecurityAvailable()).thenReturn(true); + when(licenseState.isSecurityDisabledByTrialLicense()).thenReturn(true); + + Settings.Builder settings = Settings.builder().put(this.settings); + + final boolean httpSSLEnabled = randomBoolean(); + settings.put("xpack.security.http.ssl.enabled", httpSSLEnabled); + final boolean transportSSLEnabled = randomBoolean(); + settings.put("xpack.security.transport.ssl.enabled", transportSSLEnabled); + + final boolean auditingEnabled = randomBoolean(); + settings.put(XPackSettings.AUDIT_ENABLED.getKey(), auditingEnabled); + + final boolean rolesStoreEnabled = randomBoolean(); + configureRoleStoreUsage(rolesStoreEnabled); + + final boolean roleMappingStoreEnabled = randomBoolean(); + configureRoleMappingStoreUsage(roleMappingStoreEnabled); + + configureRealmsUsage(Collections.emptyMap()); + + SecurityFeatureSet featureSet = new SecurityFeatureSet(settings.build(), licenseState, + realms, rolesStore, roleMappingStore, ipFilter); + PlainActionFuture future = new PlainActionFuture<>(); + featureSet.usage(future); + XPackFeatureSet.Usage securityUsage = future.get(); + BytesStreamOutput out = new BytesStreamOutput(); + securityUsage.writeTo(out); + XPackFeatureSet.Usage serializedUsage = new SecurityFeatureSetUsage(out.bytes().streamInput()); + for (XPackFeatureSet.Usage usage : Arrays.asList(securityUsage, serializedUsage)) { + assertThat(usage, is(notNullValue())); + assertThat(usage.name(), is(XPackField.SECURITY)); + assertThat(usage.enabled(), is(false)); + assertThat(usage.available(), is(true)); + XContentSource source = getXContentSource(usage); + + // check SSL : This is permitted even though security has been dynamically disabled by the trial license. + assertThat(source.getValue("ssl"), is(notNullValue())); + assertThat(source.getValue("ssl.http.enabled"), is(httpSSLEnabled)); + assertThat(source.getValue("ssl.transport.enabled"), is(transportSSLEnabled)); + + // everything else is missing because security is disabled + assertThat(source.getValue("realms"), is(nullValue())); + assertThat(source.getValue("token_service"), is(nullValue())); + assertThat(source.getValue("api_key_service"), is(nullValue())); + assertThat(source.getValue("audit"), is(nullValue())); + assertThat(source.getValue("anonymous"), is(nullValue())); + assertThat(source.getValue("ipfilter"), is(nullValue())); + assertThat(source.getValue("roles"), is(nullValue())); + } + } + + private XContentSource getXContentSource(XPackFeatureSet.Usage usage) throws IOException { + XContentSource source; + try (XContentBuilder builder = XContentFactory.jsonBuilder()) { + usage.toXContent(builder, ToXContent.EMPTY_PARAMS); + source = new XContentSource(builder); + } + return source; + } + + private void configureRealmsUsage(Map realmsUsageStats) { + doAnswer(invocationOnMock -> { + ActionListener> listener = (ActionListener) invocationOnMock.getArguments()[0]; + listener.onResponse(realmsUsageStats); + return Void.TYPE; + }).when(realms).usageStats(any(ActionListener.class)); + } + + private void configureRoleStoreUsage(boolean rolesStoreEnabled) { + doAnswer(invocationOnMock -> { + ActionListener> listener = (ActionListener>) invocationOnMock.getArguments()[0]; + if (rolesStoreEnabled) { + listener.onResponse(Collections.singletonMap("count", 1)); + } else { + listener.onResponse(Collections.emptyMap()); + } + return Void.TYPE; + }).when(rolesStore).usageStats(any(ActionListener.class)); + } + + private void configureRoleMappingStoreUsage(boolean roleMappingStoreEnabled) { + doAnswer(invocationOnMock -> { + ActionListener> listener = (ActionListener) invocationOnMock.getArguments()[0]; + if (roleMappingStoreEnabled) { + final Map map = new HashMap<>(); + map.put("size", 12L); + map.put("enabled", 10L); + listener.onResponse(map); + } else { + listener.onResponse(Collections.emptyMap()); + } + return Void.TYPE; + }).when(roleMappingStore).usageStats(any(ActionListener.class)); + } } diff --git a/x-pack/plugin/security/src/test/java/org/elasticsearch/xpack/security/authc/support/mapper/ExpressionRoleMappingTests.java b/x-pack/plugin/security/src/test/java/org/elasticsearch/xpack/security/authc/support/mapper/ExpressionRoleMappingTests.java index 276d8a333f796..57db60051194e 100644 --- a/x-pack/plugin/security/src/test/java/org/elasticsearch/xpack/security/authc/support/mapper/ExpressionRoleMappingTests.java +++ b/x-pack/plugin/security/src/test/java/org/elasticsearch/xpack/security/authc/support/mapper/ExpressionRoleMappingTests.java @@ -251,7 +251,7 @@ public void testToXContentWithTemplates() throws Exception { public void testSerialization() throws Exception { final ExpressionRoleMapping original = randomRoleMapping(true); - final Version version = VersionUtils.randomVersionBetween(random(), Version.V_8_0_0, null); + final Version version = VersionUtils.randomVersionBetween(random(), Version.V_7_1_0, null); BytesStreamOutput output = new BytesStreamOutput(); output.setVersion(version); original.writeTo(output); diff --git a/x-pack/plugin/sql/jdbc/src/main/java/org/elasticsearch/xpack/sql/jdbc/EsType.java b/x-pack/plugin/sql/jdbc/src/main/java/org/elasticsearch/xpack/sql/jdbc/EsType.java index 75a165a1077c0..51a03dad70b55 100644 --- a/x-pack/plugin/sql/jdbc/src/main/java/org/elasticsearch/xpack/sql/jdbc/EsType.java +++ b/x-pack/plugin/sql/jdbc/src/main/java/org/elasticsearch/xpack/sql/jdbc/EsType.java @@ -29,6 +29,7 @@ public enum EsType implements SQLType { NESTED( Types.STRUCT), BINARY( Types.VARBINARY), DATE( Types.DATE), + TIME( Types.TIME), DATETIME( Types.TIMESTAMP), IP( Types.VARCHAR), INTERVAL_YEAR( ExtraTypes.INTERVAL_YEAR), diff --git a/x-pack/plugin/sql/jdbc/src/main/java/org/elasticsearch/xpack/sql/jdbc/JdbcDateUtils.java b/x-pack/plugin/sql/jdbc/src/main/java/org/elasticsearch/xpack/sql/jdbc/JdbcDateUtils.java index c0f2e6e46ea03..7b8b3ebed4906 100644 --- a/x-pack/plugin/sql/jdbc/src/main/java/org/elasticsearch/xpack/sql/jdbc/JdbcDateUtils.java +++ b/x-pack/plugin/sql/jdbc/src/main/java/org/elasticsearch/xpack/sql/jdbc/JdbcDateUtils.java @@ -10,17 +10,12 @@ import java.sql.Time; import java.sql.Timestamp; import java.time.LocalDate; +import java.time.OffsetTime; import java.time.ZonedDateTime; -import java.time.format.DateTimeFormatter; -import java.time.format.DateTimeFormatterBuilder; -import java.util.Locale; import java.util.function.Function; -import static java.time.format.DateTimeFormatter.ISO_LOCAL_DATE; -import static java.time.temporal.ChronoField.HOUR_OF_DAY; -import static java.time.temporal.ChronoField.MILLI_OF_SECOND; -import static java.time.temporal.ChronoField.MINUTE_OF_HOUR; -import static java.time.temporal.ChronoField.SECOND_OF_MINUTE; +import static org.elasticsearch.xpack.sql.proto.StringUtils.ISO_DATE_WITH_MILLIS; +import static org.elasticsearch.xpack.sql.proto.StringUtils.ISO_TIME_WITH_MILLIS; /** * JDBC specific datetime specific utility methods. Because of lack of visibility, this class borrows code @@ -30,29 +25,21 @@ final class JdbcDateUtils { private JdbcDateUtils() {} + // In Java 8 LocalDate.EPOCH is not available, introduced with later Java versions private static final LocalDate EPOCH = LocalDate.of(1970, 1, 1); - static final DateTimeFormatter ISO_WITH_MILLIS = new DateTimeFormatterBuilder() - .parseCaseInsensitive() - .append(ISO_LOCAL_DATE) - .appendLiteral('T') - .appendValue(HOUR_OF_DAY, 2) - .appendLiteral(':') - .appendValue(MINUTE_OF_HOUR, 2) - .appendLiteral(':') - .appendValue(SECOND_OF_MINUTE, 2) - .appendFraction(MILLI_OF_SECOND, 3, 3, true) - .appendOffsetId() - .toFormatter(Locale.ROOT); - private static ZonedDateTime asDateTime(String date) { - return ISO_WITH_MILLIS.parse(date, ZonedDateTime::from); + return ISO_DATE_WITH_MILLIS.parse(date, ZonedDateTime::from); } - static long asMillisSinceEpoch(String date) { + static long dateTimeAsMillisSinceEpoch(String date) { return asDateTime(date).toInstant().toEpochMilli(); } + static long timeAsMillisSinceEpoch(String date) { + return ISO_TIME_WITH_MILLIS.parse(date, OffsetTime::from).atDate(EPOCH).toInstant().toEpochMilli(); + } + static Date asDate(String date) { ZonedDateTime zdt = asDateTime(date); return new Date(zdt.toLocalDate().atStartOfDay(zdt.getZone()).toInstant().toEpochMilli()); @@ -63,14 +50,22 @@ static Time asTime(String date) { return new Time(zdt.toLocalTime().atDate(EPOCH).atZone(zdt.getZone()).toInstant().toEpochMilli()); } + static Time timeAsTime(String date) { + OffsetTime ot = ISO_TIME_WITH_MILLIS.parse(date, OffsetTime::from); + return new Time(ot.atDate(EPOCH).toInstant().toEpochMilli()); + } + static Timestamp asTimestamp(long millisSinceEpoch) { return new Timestamp(millisSinceEpoch); } static Timestamp asTimestamp(String date) { - return new Timestamp(asMillisSinceEpoch(date)); + return new Timestamp(dateTimeAsMillisSinceEpoch(date)); } + static Timestamp timeAsTimestamp(String date) { + return new Timestamp(timeAsMillisSinceEpoch(date)); + } /* * Handles the value received as parameter, as either String (a ZonedDateTime formatted in ISO 8601 standard with millis) - * date fields being returned formatted like this. Or a Long value, in case of Histograms. diff --git a/x-pack/plugin/sql/jdbc/src/main/java/org/elasticsearch/xpack/sql/jdbc/JdbcResultSet.java b/x-pack/plugin/sql/jdbc/src/main/java/org/elasticsearch/xpack/sql/jdbc/JdbcResultSet.java index 9b1fcb48901a7..1d2489fc6d50d 100644 --- a/x-pack/plugin/sql/jdbc/src/main/java/org/elasticsearch/xpack/sql/jdbc/JdbcResultSet.java +++ b/x-pack/plugin/sql/jdbc/src/main/java/org/elasticsearch/xpack/sql/jdbc/JdbcResultSet.java @@ -33,8 +33,15 @@ import java.util.function.Function; import static java.lang.String.format; +import static org.elasticsearch.xpack.sql.jdbc.EsType.DATE; +import static org.elasticsearch.xpack.sql.jdbc.EsType.DATETIME; +import static org.elasticsearch.xpack.sql.jdbc.EsType.TIME; import static org.elasticsearch.xpack.sql.jdbc.JdbcDateUtils.asDateTimeField; -import static org.elasticsearch.xpack.sql.jdbc.JdbcDateUtils.asMillisSinceEpoch; +import static org.elasticsearch.xpack.sql.jdbc.JdbcDateUtils.dateTimeAsMillisSinceEpoch; +import static org.elasticsearch.xpack.sql.jdbc.JdbcDateUtils.asTimestamp; +import static org.elasticsearch.xpack.sql.jdbc.JdbcDateUtils.timeAsMillisSinceEpoch; +import static org.elasticsearch.xpack.sql.jdbc.JdbcDateUtils.timeAsTime; +import static org.elasticsearch.xpack.sql.jdbc.JdbcDateUtils.timeAsTimestamp; class JdbcResultSet implements ResultSet, JdbcWrapper { @@ -251,17 +258,20 @@ private Long dateTimeAsMillis(int columnIndex) throws SQLException { // TODO: the B6 appendix of the jdbc spec does mention CHAR, VARCHAR, LONGVARCHAR, DATE, TIMESTAMP as supported // jdbc types that should be handled by getDate and getTime methods. From all of those we support VARCHAR and // TIMESTAMP. Should we consider the VARCHAR conversion as a later enhancement? - if (EsType.DATETIME == type) { + if (DATETIME == type) { // the cursor can return an Integer if the date-since-epoch is small enough, XContentParser (Jackson) will // return the "smallest" data type for numbers when parsing // TODO: this should probably be handled server side if (val == null) { return null; } - return asDateTimeField(val, JdbcDateUtils::asMillisSinceEpoch, Function.identity()); + return asDateTimeField(val, JdbcDateUtils::dateTimeAsMillisSinceEpoch, Function.identity()); } - if (EsType.DATE == type) { - return asMillisSinceEpoch(val.toString()); + if (DATE == type) { + return dateTimeAsMillisSinceEpoch(val.toString()); + } + if (TIME == type) { + return timeAsMillisSinceEpoch(val.toString()); } return val == null ? null : (Long) val; } catch (ClassCastException cce) { @@ -277,10 +287,15 @@ private Date asDate(int columnIndex) throws SQLException { return null; } + EsType type = columnType(columnIndex); + if (type == TIME) { + return new Date(0L); + } + try { + return JdbcDateUtils.asDate(val.toString()); } catch (Exception e) { - EsType type = columnType(columnIndex); throw new SQLException( format(Locale.ROOT, "Unable to convert value [%.128s] of type [%s] to a Date", val, type.getName()), e); } @@ -294,11 +309,14 @@ private Time asTime(int columnIndex) throws SQLException { } EsType type = columnType(columnIndex); - if (type == EsType.DATE) { + if (type == DATE) { return new Time(0L); } try { + if (type == TIME) { + return timeAsTime(val.toString()); + } return JdbcDateUtils.asTime(val.toString()); } catch (Exception e) { throw new SQLException( @@ -313,13 +331,16 @@ private Timestamp asTimeStamp(int columnIndex) throws SQLException { return null; } + EsType type = columnType(columnIndex); try { if (val instanceof Number) { - return JdbcDateUtils.asTimestamp(((Number) val).longValue()); + return asTimestamp(((Number) val).longValue()); + } + if (type == TIME) { + return timeAsTimestamp(val.toString()); } - return JdbcDateUtils.asTimestamp(val.toString()); + return asTimestamp(val.toString()); } catch (Exception e) { - EsType type = columnType(columnIndex); throw new SQLException( format(Locale.ROOT, "Unable to convert value [%.128s] of type [%s] to a Timestamp", val, type.getName()), e); } @@ -342,7 +363,7 @@ public Date getDate(String columnLabel, Calendar cal) throws SQLException { @Override public Time getTime(int columnIndex, Calendar cal) throws SQLException { EsType type = columnType(columnIndex); - if (type == EsType.DATE) { + if (type == DATE) { return new Time(0L); } return TypeConverter.convertTime(dateTimeAsMillis(columnIndex), safeCalendar(cal)); diff --git a/x-pack/plugin/sql/jdbc/src/main/java/org/elasticsearch/xpack/sql/jdbc/TypeConverter.java b/x-pack/plugin/sql/jdbc/src/main/java/org/elasticsearch/xpack/sql/jdbc/TypeConverter.java index c7f591d4e67e2..dc1f77c808b2b 100644 --- a/x-pack/plugin/sql/jdbc/src/main/java/org/elasticsearch/xpack/sql/jdbc/TypeConverter.java +++ b/x-pack/plugin/sql/jdbc/src/main/java/org/elasticsearch/xpack/sql/jdbc/TypeConverter.java @@ -38,6 +38,10 @@ import static java.util.Calendar.MONTH; import static java.util.Calendar.SECOND; import static java.util.Calendar.YEAR; +import static org.elasticsearch.xpack.sql.jdbc.EsType.DATE; +import static org.elasticsearch.xpack.sql.jdbc.EsType.DATETIME; +import static org.elasticsearch.xpack.sql.jdbc.EsType.TIME; +import static org.elasticsearch.xpack.sql.jdbc.JdbcDateUtils.asDateTimeField; /** * Conversion utilities for conversion of JDBC types to Java type and back @@ -218,9 +222,11 @@ static Object convert(Object v, EsType columnType, String typeString) throws SQL case FLOAT: return floatValue(v); // Float might be represented as string for infinity and NaN values case DATE: - return JdbcDateUtils.asDateTimeField(v, JdbcDateUtils::asDate, Date::new); + return asDateTimeField(v, JdbcDateUtils::asDate, Date::new); + case TIME: + return asDateTimeField(v, JdbcDateUtils::asTime, Time::new); case DATETIME: - return JdbcDateUtils.asDateTimeField(v, JdbcDateUtils::asTimestamp, Timestamp::new); + return asDateTimeField(v, JdbcDateUtils::asTimestamp, Timestamp::new); case INTERVAL_YEAR: case INTERVAL_MONTH: case INTERVAL_YEAR_TO_MONTH: @@ -482,25 +488,34 @@ private static Double asDouble(Object val, EsType columnType, String typeString) } private static Date asDate(Object val, EsType columnType, String typeString) throws SQLException { - if (columnType == EsType.DATETIME || columnType == EsType.DATE) { - return JdbcDateUtils.asDateTimeField(val, JdbcDateUtils::asDate, Date::new); + if (columnType == DATETIME || columnType == DATE) { + return asDateTimeField(val, JdbcDateUtils::asDate, Date::new); + } + if (columnType == TIME) { + return new Date(0L); } return failConversion(val, columnType, typeString, Date.class); } private static Time asTime(Object val, EsType columnType, String typeString) throws SQLException { - if (columnType == EsType.DATETIME) { - return JdbcDateUtils.asDateTimeField(val, JdbcDateUtils::asTime, Time::new); + if (columnType == DATETIME) { + return asDateTimeField(val, JdbcDateUtils::asTime, Time::new); + } + if (columnType == TIME) { + return asDateTimeField(val, JdbcDateUtils::timeAsTime, Time::new); } - if (columnType == EsType.DATE) { + if (columnType == DATE) { return new Time(0L); } return failConversion(val, columnType, typeString, Time.class); } private static Timestamp asTimestamp(Object val, EsType columnType, String typeString) throws SQLException { - if (columnType == EsType.DATETIME || columnType == EsType.DATE) { - return JdbcDateUtils.asDateTimeField(val, JdbcDateUtils::asTimestamp, Timestamp::new); + if (columnType == DATETIME || columnType == DATE) { + return asDateTimeField(val, JdbcDateUtils::asTimestamp, Timestamp::new); + } + if (columnType == TIME) { + return asDateTimeField(val, JdbcDateUtils::timeAsTimestamp, Timestamp::new); } return failConversion(val, columnType, typeString, Timestamp.class); } diff --git a/x-pack/plugin/sql/qa/src/main/java/org/elasticsearch/xpack/sql/qa/SqlProtocolTestCase.java b/x-pack/plugin/sql/qa/src/main/java/org/elasticsearch/xpack/sql/qa/SqlProtocolTestCase.java index 1a47bb0add85b..bf34558b404f5 100644 --- a/x-pack/plugin/sql/qa/src/main/java/org/elasticsearch/xpack/sql/qa/SqlProtocolTestCase.java +++ b/x-pack/plugin/sql/qa/src/main/java/org/elasticsearch/xpack/sql/qa/SqlProtocolTestCase.java @@ -72,6 +72,18 @@ public void testDateTimes() throws IOException { "datetime", "1119-01-15T12:37:29.000Z", 24); assertQuery("SELECT CAST(CAST('-26853765751000' AS BIGINT) AS DATETIME)", "CAST(CAST('-26853765751000' AS BIGINT) AS DATETIME)", "datetime", "1119-01-15T12:37:29.000Z", 24); + + assertQuery("SELECT CAST('2019-01-14' AS DATE)", "CAST('2019-01-14' AS DATE)", + "date", "2019-01-14T00:00:00.000Z", 24); + assertQuery("SELECT CAST(-26853765751000 AS DATE)", "CAST(-26853765751000 AS DATE)", + "date", "1119-01-15T00:00:00.000Z", 24); + + assertQuery("SELECT CAST('12:29:25.123Z' AS TIME)", "CAST('12:29:25.123Z' AS TIME)", + "time", "12:29:25.123Z", 18); + assertQuery("SELECT CAST('12:29:25.123456789+05:00' AS TIME)", "CAST('12:29:25.123456789+05:00' AS TIME)", + "time", "12:29:25.123+05:00", 18); + assertQuery("SELECT CAST(-26853765751000 AS TIME)", "CAST(-26853765751000 AS TIME)", + "time", "12:37:29.000Z", 18); } public void testIPs() throws IOException { diff --git a/x-pack/plugin/sql/qa/src/main/java/org/elasticsearch/xpack/sql/qa/jdbc/CsvSpecTestCase.java b/x-pack/plugin/sql/qa/src/main/java/org/elasticsearch/xpack/sql/qa/jdbc/CsvSpecTestCase.java index 7db6faefb57c3..7f5077f15a885 100644 --- a/x-pack/plugin/sql/qa/src/main/java/org/elasticsearch/xpack/sql/qa/jdbc/CsvSpecTestCase.java +++ b/x-pack/plugin/sql/qa/src/main/java/org/elasticsearch/xpack/sql/qa/jdbc/CsvSpecTestCase.java @@ -6,7 +6,6 @@ package org.elasticsearch.xpack.sql.qa.jdbc; import com.carrotsearch.randomizedtesting.annotations.ParametersFactory; - import org.apache.logging.log4j.Logger; import org.elasticsearch.xpack.sql.qa.jdbc.CsvTestUtils.CsvTestCase; diff --git a/x-pack/plugin/sql/qa/src/main/java/org/elasticsearch/xpack/sql/qa/jdbc/JdbcAssert.java b/x-pack/plugin/sql/qa/src/main/java/org/elasticsearch/xpack/sql/qa/jdbc/JdbcAssert.java index 76456f2d7e64b..76894fc5a53d5 100644 --- a/x-pack/plugin/sql/qa/src/main/java/org/elasticsearch/xpack/sql/qa/jdbc/JdbcAssert.java +++ b/x-pack/plugin/sql/qa/src/main/java/org/elasticsearch/xpack/sql/qa/jdbc/JdbcAssert.java @@ -7,7 +7,6 @@ package org.elasticsearch.xpack.sql.qa.jdbc; import com.carrotsearch.hppc.IntObjectHashMap; - import org.apache.logging.log4j.Logger; import org.elasticsearch.geo.geometry.Geometry; import org.elasticsearch.geo.geometry.Point; @@ -223,6 +222,9 @@ private static void doAssertResultSetData(ResultSet expected, ResultSet actual, case "Date": columnClassName = "java.sql.Date"; break; + case "Time": + columnClassName = "java.sql.Time"; + break; case "Timestamp": columnClassName = "java.sql.Timestamp"; break; diff --git a/x-pack/plugin/sql/qa/src/main/java/org/elasticsearch/xpack/sql/qa/jdbc/ResultSetTestCase.java b/x-pack/plugin/sql/qa/src/main/java/org/elasticsearch/xpack/sql/qa/jdbc/ResultSetTestCase.java index b8cd81e39f545..3d65769a9b8d7 100644 --- a/x-pack/plugin/sql/qa/src/main/java/org/elasticsearch/xpack/sql/qa/jdbc/ResultSetTestCase.java +++ b/x-pack/plugin/sql/qa/src/main/java/org/elasticsearch/xpack/sql/qa/jdbc/ResultSetTestCase.java @@ -1151,6 +1151,35 @@ public void testGetDateType() throws Exception { assertEquals(expectedTimestamp, results.getObject("date", java.sql.Timestamp.class)); }); } + + public void testGetTimeType() throws Exception { + createIndex("test"); + updateMapping("test", builder -> builder.startObject("test_date").field("type", "date").endObject()); + + // 2018-03-12 17:20:30.123 UTC + Long timeInMillis = 1520875230123L; + index("test", "1", builder -> builder.field("test_date", timeInMillis)); + + // UTC +10 hours + String timeZoneId1 = "Etc/GMT-10"; + + doWithQueryAndTimezone("SELECT CAST(test_date AS TIME) as time FROM test", timeZoneId1, results -> { + results.next(); + + java.sql.Date expectedDate = new java.sql.Date(0L); + assertEquals(expectedDate, results.getDate("time")); + assertEquals(expectedDate, results.getObject("time", java.sql.Date.class)); + + java.sql.Time expectedTime = asTime(timeInMillis, ZoneId.of("Etc/GMT-10")); + assertEquals(expectedTime, results.getTime("time")); + assertEquals(expectedTime, results.getObject("time", java.sql.Time.class)); + + java.sql.Timestamp expectedTimestamp = new java.sql.Timestamp(expectedTime.getTime()); + assertEquals(expectedTimestamp, results.getTimestamp("time")); + assertEquals(expectedTimestamp, results.getObject("time", java.sql.Timestamp.class)); + }); + } + public void testValidGetObjectCalls() throws Exception { createIndex("test"); updateMappingForNumericValuesTests("test"); diff --git a/x-pack/plugin/sql/qa/src/main/resources/time.csv-spec b/x-pack/plugin/sql/qa/src/main/resources/time.csv-spec new file mode 100644 index 0000000000000..cba8dd29a75a1 --- /dev/null +++ b/x-pack/plugin/sql/qa/src/main/resources/time.csv-spec @@ -0,0 +1,83 @@ +// +// TIME +// + +// AwaitsFix: https://github.com/elastic/elasticsearch/issues/40717 +//timeExtractTimeParts +//SELECT +//SECOND(CAST(birth_date AS TIME)) d, +//MINUTE(CAST(birth_date AS TIME)) m, +//HOUR(CAST(birth_date AS TIME)) h +//FROM "test_emp" WHERE emp_no < 10010 ORDER BY emp_no; +// +// d:i | m:i | h:i +//0 |0 |0 +//0 |0 |0 +//0 |0 |0 +//0 |0 |0 +//0 |0 |0 +//0 |0 |0 +//0 |0 |0 +//0 |0 |0 +//0 |0 |0 +//; + +timeAsFilter +SELECT birth_date, last_name FROM "test_emp" WHERE birth_date::TIME = CAST('00:00:00' AS TIME) ORDER BY emp_no LIMIT 5; + + birth_date:ts | last_name:s +1953-09-02 00:00:00Z | Facello +1964-06-02 00:00:00Z | Simmel +1959-12-03 00:00:00Z | Bamford +1954-05-01 00:00:00Z | Koblick +1955-01-21 00:00:00Z | Maliniak +; + +timeAsFilter_NoMatch +SELECT count(*) FROM "test_emp" WHERE birth_date::TIME = CAST('12:34:56.789' AS TIME); + + count(*):l +0 +; + +timeAsOrderBy +SELECT last_name FROM "test_emp" ORDER BY birth_date::TIME, emp_no LIMIT 5; + +last_name:s +Meriste +Lenart +Stamatiou +Tzvieli +Casley +; + +timeAndFunctionAsGroupingKey +SELECT HOUR(CAST(birth_date AS TIME)) AS m, CAST(SUM(emp_no) AS INT) s FROM test_emp GROUP BY m ORDER BY m LIMIT 5; + + m:i | s:i +null |100445 +0 |904605 +; + +// AwaitsFix: https://github.com/elastic/elasticsearch/issues/40717 +//timeAsHavingFilter +//SELECT MINUTE_OF_HOUR(MAX(birth_date)::TIME + INTERVAL 10 MINUTES) as minute, gender FROM test_emp GROUP BY gender HAVING CAST(MAX(birth_date) AS TIME) = CAST('00:00:00.000' AS TIME) ORDER BY gender; +// +//minute:i | gender:s +//10 | null +//10 | F +//10 | M +//; + +timeAsHavingFilterNoMatch +SELECT MINUTE_OF_HOUR(MAX(birth_date)::TIME) as minute, gender FROM test_emp GROUP BY gender HAVING CAST(MAX(birth_date) AS TIME) > CAST('00:00:00.000' AS TIME); + +minute:i | gender:s +; + +timeAndInterval +SELECT HOUR(CAST('10:11:12.345' AS TIME) + INTERVAL '20' HOURS) AS h, SECOND(INTERVAL '40' SECONDS + CAST('10:11:12.345' AS TIME)) AS m; + +h:i | m:i +6 | 52 +; diff --git a/x-pack/plugin/sql/sql-proto/src/main/java/org/elasticsearch/xpack/sql/proto/StringUtils.java b/x-pack/plugin/sql/sql-proto/src/main/java/org/elasticsearch/xpack/sql/proto/StringUtils.java index 93c2e2f743c97..bd90354e5f98c 100644 --- a/x-pack/plugin/sql/sql-proto/src/main/java/org/elasticsearch/xpack/sql/proto/StringUtils.java +++ b/x-pack/plugin/sql/sql-proto/src/main/java/org/elasticsearch/xpack/sql/proto/StringUtils.java @@ -8,6 +8,7 @@ import java.sql.Timestamp; import java.time.Duration; +import java.time.OffsetTime; import java.time.Period; import java.time.ZonedDateTime; import java.time.format.DateTimeFormatter; @@ -26,7 +27,7 @@ public final class StringUtils { public static final String EMPTY = ""; - private static final DateTimeFormatter ISO_WITH_MILLIS = new DateTimeFormatterBuilder() + public static final DateTimeFormatter ISO_DATE_WITH_MILLIS = new DateTimeFormatterBuilder() .parseCaseInsensitive() .append(ISO_LOCAL_DATE) .appendLiteral('T') @@ -39,6 +40,17 @@ public final class StringUtils { .appendOffsetId() .toFormatter(Locale.ROOT); + public static final DateTimeFormatter ISO_TIME_WITH_MILLIS = new DateTimeFormatterBuilder() + .parseCaseInsensitive() + .appendValue(HOUR_OF_DAY, 2) + .appendLiteral(':') + .appendValue(MINUTE_OF_HOUR, 2) + .appendLiteral(':') + .appendValue(SECOND_OF_MINUTE, 2) + .appendFraction(MILLI_OF_SECOND, 3, 3, true) + .appendOffsetId() + .toFormatter(Locale.ROOT); + private static final int SECONDS_PER_MINUTE = 60; private static final int SECONDS_PER_HOUR = SECONDS_PER_MINUTE * 60; private static final int SECONDS_PER_DAY = SECONDS_PER_HOUR * 24; @@ -49,16 +61,18 @@ public static String toString(Object value) { if (value == null) { return "null"; } - + + if (value instanceof ZonedDateTime) { + return ((ZonedDateTime) value).format(ISO_DATE_WITH_MILLIS); + } + if (value instanceof OffsetTime) { + return ((OffsetTime) value).format(ISO_TIME_WITH_MILLIS); + } if (value instanceof Timestamp) { Timestamp ts = (Timestamp) value; return ts.toInstant().toString(); } - if (value instanceof ZonedDateTime) { - return ((ZonedDateTime) value).format(ISO_WITH_MILLIS); - } - // handle intervals // YEAR/MONTH/YEAR TO MONTH -> YEAR TO MONTH if (value instanceof Period) { @@ -112,4 +126,4 @@ public static String toString(Object value) { private static String indent(long timeUnit) { return timeUnit < 10 ? "0" + timeUnit : Long.toString(timeUnit); } -} \ No newline at end of file +} diff --git a/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/analysis/analyzer/Verifier.java b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/analysis/analyzer/Verifier.java index 47c53e772d5dd..bade2d44c8af3 100644 --- a/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/analysis/analyzer/Verifier.java +++ b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/analysis/analyzer/Verifier.java @@ -298,7 +298,8 @@ private static boolean checkGroupBy(LogicalPlan p, Set localFailures, return checkGroupByInexactField(p, localFailures) && checkGroupByAgg(p, localFailures, resolvedFunctions) && checkGroupByOrder(p, localFailures, groupingFailures) - && checkGroupByHaving(p, localFailures, groupingFailures, resolvedFunctions); + && checkGroupByHaving(p, localFailures, groupingFailures, resolvedFunctions) + && checkGroupByTime(p, localFailures); } // check whether an orderBy failed or if it occurs on a non-key @@ -473,14 +474,30 @@ private static boolean checkGroupByInexactField(LogicalPlan p, Set loca a.groupings().forEach(e -> e.forEachUp(c -> { EsField.Exact exact = c.getExactInfo(); if (exact.hasExact() == false) { - localFailures.add(fail(c, "Field of data type [" + c.dataType().typeName + "] cannot be used for grouping; " + - exact.errorMsg())); + localFailures.add(fail(c, "Field [" + c.sourceText() + "] of data type [" + c.dataType().typeName + "] " + + "cannot be used for grouping; " + exact.errorMsg())); } }, FieldAttribute.class)); } return true; } + private static boolean checkGroupByTime(LogicalPlan p, Set localFailures) { + if (p instanceof Aggregate) { + Aggregate a = (Aggregate) p; + + // TIME data type is not allowed for grouping key + // https://github.com/elastic/elasticsearch/issues/40639 + a.groupings().forEach(f -> { + if (f.dataType().isTimeBased()) { + localFailures.add(fail(f, "Function [" + f.sourceText() + "] with data type [" + f.dataType().typeName + + "] " + "cannot be used for grouping")); + } + }); + } + return true; + } + // check whether plain columns specified in an agg are mentioned in the group-by private static boolean checkGroupByAgg(LogicalPlan p, Set localFailures, Map functions) { if (p instanceof Aggregate) { diff --git a/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/execution/search/extractor/TopHitsAggExtractor.java b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/execution/search/extractor/TopHitsAggExtractor.java index b541df7e81a81..14d50f7c9a09b 100644 --- a/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/execution/search/extractor/TopHitsAggExtractor.java +++ b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/execution/search/extractor/TopHitsAggExtractor.java @@ -75,6 +75,8 @@ public Object extract(Bucket bucket) { Object value = agg.getHits().getAt(0).getFields().values().iterator().next().getValue(); if (fieldDataType.isDateBased()) { return DateUtils.asDateTime(Long.parseLong(value.toString()), zoneId); + } else if (fieldDataType.isTimeBased()) { + return DateUtils.asTimeOnly(Long.parseLong(value.toString()), zoneId); } else { return value; } diff --git a/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/TypeResolutions.java b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/TypeResolutions.java index f7c644ddf2702..d382dad83a19d 100644 --- a/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/TypeResolutions.java +++ b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/TypeResolutions.java @@ -43,8 +43,18 @@ public static TypeResolution isDate(Expression e, String operationName, ParamOrd return isType(e, DataType::isDateBased, operationName, paramOrd, "date", "datetime"); } + public static TypeResolution isDateOrTime(Expression e, String operationName, ParamOrdinal paramOrd) { + return isType(e, DataType::isDateOrTimeBased, operationName, paramOrd, "date", "time", "datetime"); + } + public static TypeResolution isNumericOrDate(Expression e, String operationName, ParamOrdinal paramOrd) { - return isType(e, dt -> dt.isNumeric() || dt.isDateBased(), operationName, paramOrd, "date", "datetime", "numeric"); + return isType(e, dt -> dt.isNumeric() || dt.isDateBased(), operationName, paramOrd, + "date", "datetime", "numeric"); + } + + public static TypeResolution isNumericOrDateOrTime(Expression e, String operationName, ParamOrdinal paramOrd) { + return isType(e, dt -> dt.isNumeric() || dt.isDateOrTimeBased(), operationName, paramOrd, + "date", "time", "datetime", "numeric"); } diff --git a/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/aggregate/Max.java b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/aggregate/Max.java index 5827083343a0f..eaf2d798f6d2d 100644 --- a/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/aggregate/Max.java +++ b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/aggregate/Max.java @@ -14,7 +14,7 @@ import java.util.List; import static org.elasticsearch.xpack.sql.expression.TypeResolutions.isExact; -import static org.elasticsearch.xpack.sql.expression.TypeResolutions.isNumericOrDate; +import static org.elasticsearch.xpack.sql.expression.TypeResolutions.isNumericOrDateOrTime; /** * Find the maximum value in matching documents. @@ -50,7 +50,7 @@ protected TypeResolution resolveType() { if (field().dataType().isString()) { return isExact(field(), sourceText(), ParamOrdinal.DEFAULT); } else { - return isNumericOrDate(field(), sourceText(), ParamOrdinal.DEFAULT); + return isNumericOrDateOrTime(field(), sourceText(), ParamOrdinal.DEFAULT); } } } diff --git a/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/aggregate/Min.java b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/aggregate/Min.java index e64774fe8e720..f195517335883 100644 --- a/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/aggregate/Min.java +++ b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/aggregate/Min.java @@ -14,7 +14,7 @@ import java.util.List; import static org.elasticsearch.xpack.sql.expression.TypeResolutions.isExact; -import static org.elasticsearch.xpack.sql.expression.TypeResolutions.isNumericOrDate; +import static org.elasticsearch.xpack.sql.expression.TypeResolutions.isNumericOrDateOrTime; /** * Find the minimum value in matched documents. @@ -53,7 +53,7 @@ protected TypeResolution resolveType() { if (field().dataType().isString()) { return isExact(field(), sourceText(), ParamOrdinal.DEFAULT); } else { - return isNumericOrDate(field(), sourceText(), ParamOrdinal.DEFAULT); + return isNumericOrDateOrTime(field(), sourceText(), ParamOrdinal.DEFAULT); } } } diff --git a/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/scalar/Processors.java b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/scalar/Processors.java index 34320f8ab17b6..4f0beba79b6cd 100644 --- a/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/scalar/Processors.java +++ b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/scalar/Processors.java @@ -14,6 +14,7 @@ import org.elasticsearch.xpack.sql.expression.function.scalar.geo.GeoProcessor; import org.elasticsearch.xpack.sql.expression.function.scalar.geo.StDistanceProcessor; import org.elasticsearch.xpack.sql.expression.function.scalar.geo.StWkttosqlProcessor; +import org.elasticsearch.xpack.sql.expression.function.scalar.datetime.TimeProcessor; import org.elasticsearch.xpack.sql.expression.function.scalar.math.BinaryMathProcessor; import org.elasticsearch.xpack.sql.expression.function.scalar.math.BinaryOptionalMathProcessor; import org.elasticsearch.xpack.sql.expression.function.scalar.math.MathProcessor; @@ -81,6 +82,7 @@ public static List getNamedWriteables() { // datetime entries.add(new Entry(Processor.class, DateTimeProcessor.NAME, DateTimeProcessor::new)); + entries.add(new Entry(Processor.class, TimeProcessor.NAME, TimeProcessor::new)); entries.add(new Entry(Processor.class, NamedDateTimeProcessor.NAME, NamedDateTimeProcessor::new)); entries.add(new Entry(Processor.class, NonIsoDateTimeProcessor.NAME, NonIsoDateTimeProcessor::new)); entries.add(new Entry(Processor.class, QuarterProcessor.NAME, QuarterProcessor::new)); diff --git a/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/BaseDateTimeFunction.java b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/BaseDateTimeFunction.java index cae78a42e55e9..bda86183fff02 100644 --- a/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/BaseDateTimeFunction.java +++ b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/BaseDateTimeFunction.java @@ -13,13 +13,12 @@ import org.elasticsearch.xpack.sql.tree.Source; import java.time.ZoneId; -import java.time.ZonedDateTime; import java.util.Objects; import static org.elasticsearch.xpack.sql.expression.TypeResolutions.isDate; abstract class BaseDateTimeFunction extends UnaryScalarFunction { - + private final ZoneId zoneId; BaseDateTimeFunction(Source source, Expression field, ZoneId zoneId) { @@ -50,17 +49,9 @@ public boolean foldable() { @Override public Object fold() { - ZonedDateTime folded = (ZonedDateTime) field().fold(); - if (folded == null) { - return null; - } - - return doFold(folded.withZoneSameInstant(zoneId)); + return makeProcessor().process(field().fold()); } - protected abstract Object doFold(ZonedDateTime dateTime); - - @Override public boolean equals(Object obj) { if (obj == null || obj.getClass() != getClass()) { @@ -68,7 +59,7 @@ public boolean equals(Object obj) { } BaseDateTimeFunction other = (BaseDateTimeFunction) obj; return Objects.equals(other.field(), field()) - && Objects.equals(other.zoneId(), zoneId()); + && Objects.equals(other.zoneId(), zoneId()); } @Override diff --git a/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/BaseDateTimeProcessor.java b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/BaseDateTimeProcessor.java index 608057cf235d5..ddab74aa927aa 100644 --- a/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/BaseDateTimeProcessor.java +++ b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/BaseDateTimeProcessor.java @@ -18,11 +18,11 @@ public abstract class BaseDateTimeProcessor implements Processor { private final ZoneId zoneId; - + BaseDateTimeProcessor(ZoneId zoneId) { this.zoneId = zoneId; } - + BaseDateTimeProcessor(StreamInput in) throws IOException { zoneId = ZoneId.of(in.readString()); } @@ -31,7 +31,7 @@ public abstract class BaseDateTimeProcessor implements Processor { public void writeTo(StreamOutput out) throws IOException { out.writeString(zoneId.getId()); } - + ZoneId zoneId() { return zoneId; } @@ -43,11 +43,11 @@ public Object process(Object input) { } if (!(input instanceof ZonedDateTime)) { - throw new SqlIllegalArgumentException("A date is required; received {}", input); + throw new SqlIllegalArgumentException("A [date], a [time] or a [datetime] is required; received {}", input); } return doProcess(((ZonedDateTime) input).withZoneSameInstant(zoneId)); } abstract Object doProcess(ZonedDateTime dateTime); -} \ No newline at end of file +} diff --git a/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/DateTimeFunction.java b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/DateTimeFunction.java index 9a55548c921bb..d314056ea64e8 100644 --- a/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/DateTimeFunction.java +++ b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/DateTimeFunction.java @@ -16,6 +16,7 @@ import java.time.ZoneId; import java.time.ZonedDateTime; import java.time.temporal.ChronoField; +import java.time.temporal.Temporal; import static org.elasticsearch.xpack.sql.expression.gen.script.ParamsBuilder.paramsBuilder; @@ -28,17 +29,12 @@ public abstract class DateTimeFunction extends BaseDateTimeFunction { this.extractor = extractor; } - @Override - protected Object doFold(ZonedDateTime dateTime) { - return dateTimeChrono(dateTime, extractor.chronoField()); - } - public static Integer dateTimeChrono(ZonedDateTime dateTime, String tzId, String chronoName) { ZonedDateTime zdt = dateTime.withZoneSameInstant(ZoneId.of(tzId)); return dateTimeChrono(zdt, ChronoField.valueOf(chronoName)); } - private static Integer dateTimeChrono(ZonedDateTime dateTime, ChronoField field) { + protected static Integer dateTimeChrono(Temporal dateTime, ChronoField field) { return Integer.valueOf(dateTime.get(field)); } @@ -68,4 +64,8 @@ public DataType dataType() { // used for applying ranges public abstract String dateTimeFormat(); -} \ No newline at end of file + + protected DateTimeExtractor extractor() { + return extractor; + } +} diff --git a/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/DateTimeProcessor.java b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/DateTimeProcessor.java index 5357462fdd6a3..4a39914729951 100644 --- a/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/DateTimeProcessor.java +++ b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/DateTimeProcessor.java @@ -9,6 +9,7 @@ import org.elasticsearch.common.io.stream.StreamOutput; import java.io.IOException; +import java.time.OffsetTime; import java.time.ZoneId; import java.time.ZonedDateTime; import java.time.temporal.ChronoField; @@ -38,6 +39,10 @@ public int extract(ZonedDateTime dt) { return dt.get(field); } + public int extract(OffsetTime time) { + return time.get(field); + } + public ChronoField chronoField() { return field; } @@ -95,4 +100,4 @@ public boolean equals(Object obj) { public String toString() { return extractor.toString(); } -} \ No newline at end of file +} diff --git a/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/HourOfDay.java b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/HourOfDay.java index 32de1179965f6..c15a730e25b3f 100644 --- a/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/HourOfDay.java +++ b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/HourOfDay.java @@ -7,15 +7,15 @@ import org.elasticsearch.xpack.sql.expression.Expression; import org.elasticsearch.xpack.sql.expression.function.scalar.datetime.DateTimeProcessor.DateTimeExtractor; -import org.elasticsearch.xpack.sql.tree.Source; import org.elasticsearch.xpack.sql.tree.NodeInfo.NodeCtor2; +import org.elasticsearch.xpack.sql.tree.Source; import java.time.ZoneId; /** * Extract the hour of the day from a datetime. */ -public class HourOfDay extends DateTimeFunction { +public class HourOfDay extends TimeFunction { public HourOfDay(Source source, Expression field, ZoneId zoneId) { super(source, field, zoneId, DateTimeExtractor.HOUR_OF_DAY); } diff --git a/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/MinuteOfDay.java b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/MinuteOfDay.java index 1ef450b3f650d..823de40034feb 100644 --- a/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/MinuteOfDay.java +++ b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/MinuteOfDay.java @@ -7,15 +7,15 @@ import org.elasticsearch.xpack.sql.expression.Expression; import org.elasticsearch.xpack.sql.expression.function.scalar.datetime.DateTimeProcessor.DateTimeExtractor; -import org.elasticsearch.xpack.sql.tree.Source; import org.elasticsearch.xpack.sql.tree.NodeInfo.NodeCtor2; +import org.elasticsearch.xpack.sql.tree.Source; import java.time.ZoneId; /** * Extract the minute of the day from a datetime. */ -public class MinuteOfDay extends DateTimeFunction { +public class MinuteOfDay extends TimeFunction { public MinuteOfDay(Source source, Expression field, ZoneId zoneId) { super(source, field, zoneId, DateTimeExtractor.MINUTE_OF_DAY); diff --git a/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/MinuteOfHour.java b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/MinuteOfHour.java index 9c4cf4884b3f6..1136b858a7227 100644 --- a/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/MinuteOfHour.java +++ b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/MinuteOfHour.java @@ -7,15 +7,15 @@ import org.elasticsearch.xpack.sql.expression.Expression; import org.elasticsearch.xpack.sql.expression.function.scalar.datetime.DateTimeProcessor.DateTimeExtractor; -import org.elasticsearch.xpack.sql.tree.Source; import org.elasticsearch.xpack.sql.tree.NodeInfo.NodeCtor2; +import org.elasticsearch.xpack.sql.tree.Source; import java.time.ZoneId; /** * Exract the minute of the hour from a datetime. */ -public class MinuteOfHour extends DateTimeFunction { +public class MinuteOfHour extends TimeFunction { public MinuteOfHour(Source source, Expression field, ZoneId zoneId) { super(source, field, zoneId, DateTimeExtractor.MINUTE_OF_HOUR); } diff --git a/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/NamedDateTimeFunction.java b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/NamedDateTimeFunction.java index b5d7305d2bbd2..35397df5ef4aa 100644 --- a/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/NamedDateTimeFunction.java +++ b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/NamedDateTimeFunction.java @@ -15,7 +15,6 @@ import org.elasticsearch.xpack.sql.util.StringUtils; import java.time.ZoneId; -import java.time.ZonedDateTime; import java.util.Locale; import static java.lang.String.format; @@ -33,11 +32,6 @@ abstract class NamedDateTimeFunction extends BaseDateTimeFunction { this.nameExtractor = nameExtractor; } - @Override - protected Object doFold(ZonedDateTime dateTime) { - return nameExtractor.extract(dateTime); - } - @Override public ScriptTemplate scriptWithField(FieldAttribute field) { return new ScriptTemplate( @@ -58,4 +52,4 @@ protected Processor makeProcessor() { public DataType dataType() { return DataType.KEYWORD; } -} \ No newline at end of file +} diff --git a/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/NonIsoDateTimeFunction.java b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/NonIsoDateTimeFunction.java index 4d5fb4ad91efd..1aee57ae80bc3 100644 --- a/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/NonIsoDateTimeFunction.java +++ b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/NonIsoDateTimeFunction.java @@ -15,7 +15,6 @@ import org.elasticsearch.xpack.sql.util.StringUtils; import java.time.ZoneId; -import java.time.ZonedDateTime; import java.util.Locale; import static java.lang.String.format; @@ -33,11 +32,6 @@ abstract class NonIsoDateTimeFunction extends BaseDateTimeFunction { this.extractor = extractor; } - @Override - protected Object doFold(ZonedDateTime dateTime) { - return extractor.extract(dateTime); - } - @Override public ScriptTemplate scriptWithField(FieldAttribute field) { return new ScriptTemplate( @@ -58,4 +52,4 @@ protected Processor makeProcessor() { public DataType dataType() { return DataType.INTEGER; } -} \ No newline at end of file +} diff --git a/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/Quarter.java b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/Quarter.java index 4837b7c4a8603..275e7181bc312 100644 --- a/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/Quarter.java +++ b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/Quarter.java @@ -10,14 +10,12 @@ import org.elasticsearch.xpack.sql.expression.FieldAttribute; import org.elasticsearch.xpack.sql.expression.gen.processor.Processor; import org.elasticsearch.xpack.sql.expression.gen.script.ScriptTemplate; -import org.elasticsearch.xpack.sql.tree.Source; import org.elasticsearch.xpack.sql.tree.NodeInfo.NodeCtor2; +import org.elasticsearch.xpack.sql.tree.Source; import org.elasticsearch.xpack.sql.type.DataType; import java.time.ZoneId; -import java.time.ZonedDateTime; -import static org.elasticsearch.xpack.sql.expression.function.scalar.datetime.QuarterProcessor.quarter; import static org.elasticsearch.xpack.sql.expression.gen.script.ParamsBuilder.paramsBuilder; public class Quarter extends BaseDateTimeFunction { @@ -26,11 +24,6 @@ public Quarter(Source source, Expression field, ZoneId zoneId) { super(source, field, zoneId); } - @Override - protected Object doFold(ZonedDateTime dateTime) { - return quarter(dateTime); - } - @Override public ScriptTemplate scriptWithField(FieldAttribute field) { return new ScriptTemplate(formatTemplate("{sql}.quarter(doc[{}].value, {})"), @@ -60,4 +53,4 @@ protected Processor makeProcessor() { public DataType dataType() { return DataType.INTEGER; } -} \ No newline at end of file +} diff --git a/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/SecondOfMinute.java b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/SecondOfMinute.java index 4b7c354f412d9..fb83191f5bcbd 100644 --- a/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/SecondOfMinute.java +++ b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/SecondOfMinute.java @@ -7,15 +7,15 @@ import org.elasticsearch.xpack.sql.expression.Expression; import org.elasticsearch.xpack.sql.expression.function.scalar.datetime.DateTimeProcessor.DateTimeExtractor; -import org.elasticsearch.xpack.sql.tree.Source; import org.elasticsearch.xpack.sql.tree.NodeInfo.NodeCtor2; +import org.elasticsearch.xpack.sql.tree.Source; import java.time.ZoneId; /** * Extract the second of the minute from a datetime. */ -public class SecondOfMinute extends DateTimeFunction { +public class SecondOfMinute extends TimeFunction { public SecondOfMinute(Source source, Expression field, ZoneId zoneId) { super(source, field, zoneId, DateTimeExtractor.SECOND_OF_MINUTE); } diff --git a/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/TimeFunction.java b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/TimeFunction.java new file mode 100644 index 0000000000000..857d8fada5bd4 --- /dev/null +++ b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/TimeFunction.java @@ -0,0 +1,40 @@ +/* + * Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one + * or more contributor license agreements. Licensed under the Elastic License; + * you may not use this file except in compliance with the Elastic License. + */ +package org.elasticsearch.xpack.sql.expression.function.scalar.datetime; + +import org.elasticsearch.xpack.sql.expression.Expression; +import org.elasticsearch.xpack.sql.expression.Expressions; +import org.elasticsearch.xpack.sql.expression.function.scalar.datetime.DateTimeProcessor.DateTimeExtractor; +import org.elasticsearch.xpack.sql.expression.gen.processor.Processor; +import org.elasticsearch.xpack.sql.tree.Source; + +import java.time.OffsetTime; +import java.time.ZoneId; +import java.time.temporal.ChronoField; + +import static org.elasticsearch.xpack.sql.expression.TypeResolutions.isDateOrTime; +import static org.elasticsearch.xpack.sql.util.DateUtils.asTimeAtZone; + +public abstract class TimeFunction extends DateTimeFunction { + + TimeFunction(Source source, Expression field, ZoneId zoneId, DateTimeExtractor extractor) { + super(source, field, zoneId, extractor); + } + + public static Integer dateTimeChrono(OffsetTime time, String tzId, String chronoName) { + return dateTimeChrono(asTimeAtZone(time, ZoneId.of(tzId)), ChronoField.valueOf(chronoName)); + } + + @Override + protected TypeResolution resolveType() { + return isDateOrTime(field(), sourceText(), Expressions.ParamOrdinal.DEFAULT); + } + + @Override + protected Processor makeProcessor() { + return new TimeProcessor(extractor(), zoneId()); + } +} diff --git a/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/TimeProcessor.java b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/TimeProcessor.java new file mode 100644 index 0000000000000..a263d83dcb195 --- /dev/null +++ b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/TimeProcessor.java @@ -0,0 +1,56 @@ +/* + * Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one + * or more contributor license agreements. Licensed under the Elastic License; + * you may not use this file except in compliance with the Elastic License. + */ +package org.elasticsearch.xpack.sql.expression.function.scalar.datetime; + +import org.elasticsearch.common.io.stream.StreamInput; + +import java.io.IOException; +import java.time.OffsetTime; +import java.time.ZoneId; +import java.util.Objects; + +import static org.elasticsearch.xpack.sql.util.DateUtils.asTimeAtZone; + +public class TimeProcessor extends DateTimeProcessor { + + + public static final String NAME = "time"; + + public TimeProcessor(DateTimeExtractor extractor, ZoneId zoneId) { + super(extractor, zoneId); + } + + public TimeProcessor(StreamInput in) throws IOException { + super(in); + } + + @Override + public Object process(Object input) { + if (input instanceof OffsetTime) { + return doProcess(asTimeAtZone((OffsetTime) input, zoneId())); + } + return super.process(input); + } + + private Object doProcess(OffsetTime time) { + return extractor().extract(time); + } + + @Override + public int hashCode() { + return Objects.hash(extractor(), zoneId()); + } + + @Override + public boolean equals(Object obj) { + if (obj == null || obj.getClass() != getClass()) { + return false; + } + TimeProcessor other = (TimeProcessor) obj; + return Objects.equals(extractor(), other.extractor()) + && Objects.equals(zoneId(), other.zoneId()); + } +} diff --git a/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/scalar/whitelist/InternalSqlScriptUtils.java b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/scalar/whitelist/InternalSqlScriptUtils.java index 89836c0d22d0a..e12d8d33e0d3e 100644 --- a/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/scalar/whitelist/InternalSqlScriptUtils.java +++ b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/function/scalar/whitelist/InternalSqlScriptUtils.java @@ -17,6 +17,7 @@ import org.elasticsearch.xpack.sql.expression.function.scalar.geo.GeoShape; import org.elasticsearch.xpack.sql.expression.function.scalar.geo.StDistanceProcessor; import org.elasticsearch.xpack.sql.expression.function.scalar.geo.StWkttosqlProcessor; +import org.elasticsearch.xpack.sql.expression.function.scalar.datetime.TimeFunction; import org.elasticsearch.xpack.sql.expression.function.scalar.math.BinaryMathProcessor.BinaryMathOperation; import org.elasticsearch.xpack.sql.expression.function.scalar.math.BinaryOptionalMathProcessor.BinaryOptionalMathOperation; import org.elasticsearch.xpack.sql.expression.function.scalar.math.MathProcessor.MathOperation; @@ -46,6 +47,7 @@ import org.elasticsearch.xpack.sql.util.StringUtils; import java.time.Duration; +import java.time.OffsetTime; import java.time.Period; import java.time.ZonedDateTime; import java.util.List; @@ -321,6 +323,9 @@ public static Integer dateTimeChrono(Object dateTime, String tzId, String chrono if (dateTime == null || tzId == null || chronoName == null) { return null; } + if (dateTime instanceof OffsetTime) { + return TimeFunction.dateTimeChrono((OffsetTime) dateTime, tzId, chronoName); + } return DateTimeFunction.dateTimeChrono(asDateTime(dateTime), tzId, chronoName); } @@ -401,6 +406,10 @@ public static IntervalYearMonth intervalYearMonth(String text, String typeName) return new IntervalYearMonth(Period.parse(text), DataType.fromTypeName(typeName)); } + public static OffsetTime asTime(String time) { + return OffsetTime.parse(time); + } + // // String functions // diff --git a/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/gen/script/ScriptWeaver.java b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/gen/script/ScriptWeaver.java index f9717865c69f8..223e22b2a33ba 100644 --- a/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/gen/script/ScriptWeaver.java +++ b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/gen/script/ScriptWeaver.java @@ -21,6 +21,7 @@ import org.elasticsearch.xpack.sql.type.DataType; import org.elasticsearch.xpack.sql.util.DateUtils; +import java.time.OffsetTime; import java.time.ZonedDateTime; import static org.elasticsearch.xpack.sql.expression.gen.script.ParamsBuilder.paramsBuilder; @@ -81,12 +82,19 @@ default ScriptTemplate scriptWithFoldable(Expression foldable) { return new ScriptTemplate(processScript("{sql}.intervalYearMonth({},{})"), paramsBuilder().variable(iym.interval().toString()).variable(iym.dataType().name()).build(), dataType()); - } else if (fold instanceof IntervalDayTime) { + } + if (fold instanceof IntervalDayTime) { IntervalDayTime idt = (IntervalDayTime) fold; return new ScriptTemplate(processScript("{sql}.intervalDayTime({},{})"), paramsBuilder().variable(idt.interval().toString()).variable(idt.dataType().name()).build(), dataType()); } + if (fold instanceof OffsetTime) { + OffsetTime ot = (OffsetTime) fold; + return new ScriptTemplate(processScript("{sql}.asTime({})"), + paramsBuilder().variable(ot.toString()).build(), + dataType()); + } if (fold instanceof GeoShape) { GeoShape geoShape = (GeoShape) fold; diff --git a/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/predicate/operator/arithmetic/Arithmetics.java b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/predicate/operator/arithmetic/Arithmetics.java index d66fb7df2ba5d..33a4f8c0e5603 100644 --- a/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/predicate/operator/arithmetic/Arithmetics.java +++ b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/predicate/operator/arithmetic/Arithmetics.java @@ -6,14 +6,24 @@ package org.elasticsearch.xpack.sql.expression.predicate.operator.arithmetic; import java.time.Duration; +import java.time.OffsetTime; import java.time.Period; -import java.time.ZonedDateTime; +import java.time.temporal.Temporal; + +import static org.elasticsearch.xpack.sql.util.DateUtils.DAY_IN_MILLIS; /** * Arithmetic operation using the type widening rules of the JLS 5.6.2 namely * widen to double or float or long or int in this order. */ -public abstract class Arithmetics { +public final class Arithmetics { + + private Arithmetics() {} + + private enum IntervalOperation { + ADD, + SUB + } static Number add(Number l, Number r) { if (l == null || r == null) { @@ -33,20 +43,12 @@ static Number add(Number l, Number r) { return Integer.valueOf(Math.addExact(l.intValue(), r.intValue())); } - static ZonedDateTime add(ZonedDateTime l, Period r) { - if (l == null || r == null) { - return null; - } - - return l.plus(r); + static Temporal add(Temporal l, Period r) { + return periodArithmetics(l, r, IntervalOperation.ADD); } - static ZonedDateTime add(ZonedDateTime l, Duration r) { - if (l == null || r == null) { - return null; - } - - return l.plus(r); + static Temporal add(Temporal l, Duration r) { + return durationArithmetics(l, r, IntervalOperation.ADD); } static Number sub(Number l, Number r) { @@ -67,20 +69,12 @@ static Number sub(Number l, Number r) { return Integer.valueOf(Math.subtractExact(l.intValue(), r.intValue())); } - static ZonedDateTime sub(ZonedDateTime l, Period r) { - if (l == null || r == null) { - return null; - } - - return l.minus(r); + static Temporal sub(Temporal l, Period r) { + return periodArithmetics(l, r, IntervalOperation.SUB); } - static ZonedDateTime sub(ZonedDateTime l, Duration r) { - if (l == null || r == null) { - return null; - } - - return l.minus(r); + static Temporal sub(Temporal l, Duration r) { + return durationArithmetics(l, r, IntervalOperation.SUB); } static Number mul(Number l, Number r) { @@ -162,4 +156,36 @@ static Number negate(Number n) { return Integer.valueOf(Math.negateExact(n.intValue())); } + + private static Temporal periodArithmetics(Temporal l, Period r, IntervalOperation operation) { + if (l == null || r == null) { + return null; + } + + if (l instanceof OffsetTime) { + return l; + } + + if (operation == IntervalOperation.ADD) { + return l.plus(r); + } else { + return l.minus(r); + } + } + + private static Temporal durationArithmetics(Temporal l, Duration r, IntervalOperation operation) { + if (l == null || r == null) { + return null; + } + + if (l instanceof OffsetTime) { + r = Duration.ofMillis(r.toMillis() % DAY_IN_MILLIS); + } + + if (operation == IntervalOperation.ADD) { + return l.plus(r); + } else { + return l.minus(r); + } + } } diff --git a/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/predicate/operator/arithmetic/BinaryArithmeticProcessor.java b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/predicate/operator/arithmetic/BinaryArithmeticProcessor.java index a0fd57e30d0ca..b6bfaa4acb63d 100644 --- a/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/predicate/operator/arithmetic/BinaryArithmeticProcessor.java +++ b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/predicate/operator/arithmetic/BinaryArithmeticProcessor.java @@ -17,7 +17,9 @@ import org.elasticsearch.xpack.sql.expression.predicate.operator.arithmetic.BinaryArithmeticProcessor.BinaryArithmeticOperation; import java.io.IOException; +import java.time.OffsetTime; import java.time.ZonedDateTime; +import java.time.temporal.Temporal; import java.util.function.BiFunction; public class BinaryArithmeticProcessor extends FunctionalBinaryProcessor { @@ -41,17 +43,17 @@ public enum BinaryArithmeticOperation implements PredicateBiFunction) { + if ((r instanceof ZonedDateTime || r instanceof OffsetTime) && l instanceof Interval) { throw new SqlIllegalArgumentException("Cannot subtract a date from an interval; do you mean the reverse?"); } @@ -181,4 +183,4 @@ protected Object doProcess(Object left, Object right) { // this should not occur throw new SqlIllegalArgumentException("Cannot perform arithmetic operation due to arguments"); } -} \ No newline at end of file +} diff --git a/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/predicate/operator/arithmetic/Sub.java b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/predicate/operator/arithmetic/Sub.java index cad2d7ffa625a..ee3ca6aa6773b 100644 --- a/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/predicate/operator/arithmetic/Sub.java +++ b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/expression/predicate/operator/arithmetic/Sub.java @@ -34,7 +34,7 @@ protected Sub replaceChildren(Expression newLeft, Expression newRight) { @Override protected TypeResolution resolveWithIntervals() { - if (right().dataType().isDateBased() && DataTypes.isInterval(left().dataType())) { + if ((right().dataType().isDateOrTimeBased()) && DataTypes.isInterval(left().dataType())) { return new TypeResolution(format(null, "Cannot subtract a {}[{}] from an interval[{}]; do you mean the reverse?", right().dataType().typeName, right().source().text(), left().source().text())); } diff --git a/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/parser/ExpressionBuilder.java b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/parser/ExpressionBuilder.java index c3d5ba2228467..ee5ff0300b22b 100644 --- a/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/parser/ExpressionBuilder.java +++ b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/parser/ExpressionBuilder.java @@ -114,7 +114,6 @@ import org.elasticsearch.xpack.sql.util.StringUtils; import java.time.Duration; -import java.time.LocalTime; import java.time.Period; import java.time.format.DateTimeParseException; import java.time.temporal.TemporalAmount; @@ -124,11 +123,11 @@ import java.util.Map; import java.util.StringJoiner; -import static java.time.format.DateTimeFormatter.ISO_LOCAL_TIME; import static java.util.Collections.emptyList; import static java.util.Collections.singletonList; import static org.elasticsearch.xpack.sql.type.DataTypeConversion.conversionFor; import static org.elasticsearch.xpack.sql.util.DateUtils.asDateOnly; +import static org.elasticsearch.xpack.sql.util.DateUtils.asTimeOnly; import static org.elasticsearch.xpack.sql.util.DateUtils.ofEscapedLiteral; abstract class ExpressionBuilder extends IdentifierBuilder { @@ -768,14 +767,11 @@ public Literal visitTimeEscapedLiteral(TimeEscapedLiteralContext ctx) { Source source = source(ctx); // parse HH:mm:ss - LocalTime lt = null; try { - lt = LocalTime.parse(string, ISO_LOCAL_TIME); + return new Literal(source, asTimeOnly(string), DataType.TIME); } catch (DateTimeParseException ex) { throw new ParsingException(source, "Invalid time received; {}", ex.getMessage()); } - - throw new SqlIllegalArgumentException("Time (only) literals are not supported; a date component is required as well"); } @Override diff --git a/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/planner/QueryTranslator.java b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/planner/QueryTranslator.java index e34d94e187649..7e5516810d92a 100644 --- a/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/planner/QueryTranslator.java +++ b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/planner/QueryTranslator.java @@ -296,7 +296,7 @@ else if (exp instanceof GroupingFunction) { if (h.dataType() == DATE) { intervalAsMillis = DateUtils.minDayInterval(intervalAsMillis); } - // TODO: set timezone + if (field instanceof FieldAttribute) { key = new GroupByDateHistogram(aggId, nameOf(field), intervalAsMillis, h.zoneId()); } else if (field instanceof Function) { diff --git a/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/querydsl/agg/GroupByKey.java b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/querydsl/agg/GroupByKey.java index 9638a1bd305d2..df207269eec0f 100644 --- a/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/querydsl/agg/GroupByKey.java +++ b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/querydsl/agg/GroupByKey.java @@ -41,6 +41,8 @@ public final CompositeValuesSourceBuilder asValueSource() { builder.valueType(ValueType.STRING); } else if (script.outputType() == DataType.DATE) { builder.valueType(ValueType.LONG); + } else if (script.outputType() == DataType.TIME) { + builder.valueType(ValueType.LONG); } else if (script.outputType() == DataType.DATETIME) { builder.valueType(ValueType.LONG); } else if (script.outputType() == DataType.BOOLEAN) { diff --git a/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/type/DataType.java b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/type/DataType.java index 8dcbaa593cedd..370b660109461 100644 --- a/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/type/DataType.java +++ b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/type/DataType.java @@ -45,6 +45,7 @@ public enum DataType { NESTED( "nested", JDBCType.STRUCT, -1, 0, 0, false, false, false), BINARY( "binary", JDBCType.VARBINARY, -1, Integer.MAX_VALUE, Integer.MAX_VALUE, false, false, false), DATE( JDBCType.DATE, Long.BYTES, 24, 24, false, false, true), + TIME( JDBCType.TIME, Long.BYTES, 3, 18, false, false, true), // since ODBC and JDBC interpret precision for Date as display size // the precision is 23 (number of chars in ISO8601 with millis) + Z (the UTC timezone) // see https://github.com/elastic/elasticsearch/issues/30386#issuecomment-386807288 @@ -106,7 +107,7 @@ public enum DataType { // Date ODBC_TO_ES.put("SQL_DATE", DATE); - ODBC_TO_ES.put("SQL_TIME", DATETIME); + ODBC_TO_ES.put("SQL_TIME", TIME); ODBC_TO_ES.put("SQL_TIMESTAMP", DATETIME); // Intervals @@ -259,6 +260,14 @@ public boolean isGeo() { public boolean isDateBased() { return this == DATE || this == DATETIME; } + + public boolean isTimeBased() { + return this == TIME; + } + + public boolean isDateOrTimeBased() { + return isDateBased() || isTimeBased(); + } public static DataType fromOdbcType(String odbcType) { return ODBC_TO_ES.get(odbcType); @@ -284,6 +293,6 @@ public static DataType fromTypeName(String esType) { } public String format() { - return isDateBased() ? DateUtils.DATE_PARSE_FORMAT : null; + return isDateOrTimeBased() ? DateUtils.DATE_PARSE_FORMAT : null; } } diff --git a/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/type/DataTypeConversion.java b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/type/DataTypeConversion.java index 9dbb2a3abb6f6..40a03e26eb0ef 100644 --- a/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/type/DataTypeConversion.java +++ b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/type/DataTypeConversion.java @@ -10,6 +10,7 @@ import org.elasticsearch.xpack.sql.SqlIllegalArgumentException; import org.elasticsearch.xpack.sql.util.DateUtils; +import java.time.OffsetTime; import java.time.ZonedDateTime; import java.time.format.DateTimeParseException; import java.util.Locale; @@ -22,6 +23,7 @@ import static org.elasticsearch.xpack.sql.type.DataType.DATETIME; import static org.elasticsearch.xpack.sql.type.DataType.LONG; import static org.elasticsearch.xpack.sql.type.DataType.NULL; +import static org.elasticsearch.xpack.sql.type.DataType.TIME; /** * Conversions from one Elasticsearch data type to another Elasticsearch data types. @@ -87,8 +89,24 @@ public static DataType commonType(DataType left, DataType right) { return right; } } - if (left == DATETIME) { + if (left == TIME) { if (right == DATE) { + return DATETIME; + } + if (DataTypes.isInterval(right)) { + return left; + } + } + if (right == TIME) { + if (left == DATE) { + return DATETIME; + } + if (DataTypes.isInterval(left)) { + return right; + } + } + if (left == DATETIME) { + if (right == DATE || right == TIME) { return left; } if (DataTypes.isInterval(right)) { @@ -96,7 +114,7 @@ public static DataType commonType(DataType left, DataType right) { } } if (right == DATETIME) { - if (left == DATE) { + if (left == DATE || left == TIME) { return right; } if (DataTypes.isInterval(left)) { @@ -144,7 +162,7 @@ public static Conversion conversionFor(DataType from, DataType to) { Conversion conversion = conversion(from, to); if (conversion == null) { - throw new SqlIllegalArgumentException("cannot convert from [" + from + "] to [" + to + "]"); + throw new SqlIllegalArgumentException("cannot convert from [" + from.typeName + "] to [" + to.typeName + "]"); } return conversion; } @@ -170,6 +188,8 @@ private static Conversion conversion(DataType from, DataType to) { return conversionToDouble(from); case DATE: return conversionToDate(from); + case TIME: + return conversionToTime(from); case DATETIME: return conversionToDateTime(from); case BOOLEAN: @@ -184,6 +204,9 @@ private static Conversion conversionToString(DataType from) { if (from == DATE) { return Conversion.DATE_TO_STRING; } + if (from == TIME) { + return Conversion.TIME_TO_STRING; + } if (from == DATETIME) { return Conversion.DATETIME_TO_STRING; } @@ -213,6 +236,9 @@ private static Conversion conversionToLong(DataType from) { if (from == DATE) { return Conversion.DATE_TO_LONG; } + if (from == TIME) { + return Conversion.TIME_TO_LONG; + } if (from == DATETIME) { return Conversion.DATETIME_TO_LONG; } @@ -235,6 +261,9 @@ private static Conversion conversionToInt(DataType from) { if (from == DATE) { return Conversion.DATE_TO_INT; } + if (from == TIME) { + return Conversion.TIME_TO_INT; + } if (from == DATETIME) { return Conversion.DATETIME_TO_INT; } @@ -257,6 +286,9 @@ private static Conversion conversionToShort(DataType from) { if (from == DATE) { return Conversion.DATE_TO_SHORT; } + if (from == TIME) { + return Conversion.TIME_TO_SHORT; + } if (from == DATETIME) { return Conversion.DATETIME_TO_SHORT; } @@ -279,6 +311,9 @@ private static Conversion conversionToByte(DataType from) { if (from == DATE) { return Conversion.DATE_TO_BYTE; } + if (from == TIME) { + return Conversion.TIME_TO_BYTE; + } if (from == DATETIME) { return Conversion.DATETIME_TO_BYTE; } @@ -301,6 +336,9 @@ private static Conversion conversionToFloat(DataType from) { if (from == DATE) { return Conversion.DATE_TO_FLOAT; } + if (from == TIME) { + return Conversion.TIME_TO_FLOAT; + } if (from == DATETIME) { return Conversion.DATETIME_TO_FLOAT; } @@ -323,6 +361,9 @@ private static Conversion conversionToDouble(DataType from) { if (from == DATE) { return Conversion.DATE_TO_DOUBLE; } + if (from == TIME) { + return Conversion.TIME_TO_DOUBLE; + } if (from == DATETIME) { return Conversion.DATETIME_TO_DOUBLE; } @@ -348,6 +389,28 @@ private static Conversion conversionToDate(DataType from) { return null; } + private static Conversion conversionToTime(DataType from) { + if (from.isRational()) { + return Conversion.RATIONAL_TO_TIME; + } + if (from.isInteger()) { + return Conversion.INTEGER_TO_TIME; + } + if (from == BOOLEAN) { + return Conversion.BOOL_TO_TIME; // We emit an int here which is ok because of Java's casting rules + } + if (from.isString()) { + return Conversion.STRING_TO_TIME; + } + if (from == DATE) { + return Conversion.DATE_TO_TIME; + } + if (from == DATETIME) { + return Conversion.DATETIME_TO_TIME; + } + return null; + } + private static Conversion conversionToDateTime(DataType from) { if (from.isRational()) { return Conversion.RATIONAL_TO_DATETIME; @@ -377,6 +440,9 @@ private static Conversion conversionToBoolean(DataType from) { if (from == DATE) { return Conversion.DATE_TO_BOOLEAN; } + if (from == TIME) { + return Conversion.TIME_TO_BOOLEAN; + } if (from == DATETIME) { return Conversion.DATETIME_TO_BOOLEAN; } @@ -456,6 +522,7 @@ public enum Conversion { NULL(value -> null), DATE_TO_STRING(o -> DateUtils.toDateString((ZonedDateTime) o)), + TIME_TO_STRING(o -> DateUtils.toTimeString((OffsetTime) o)), DATETIME_TO_STRING(o -> DateUtils.toString((ZonedDateTime) o)), OTHER_TO_STRING(String::valueOf), @@ -463,6 +530,7 @@ public enum Conversion { INTEGER_TO_LONG(fromLong(value -> value)), STRING_TO_LONG(fromString(Long::valueOf, "long")), DATE_TO_LONG(fromDateTime(value -> value)), + TIME_TO_LONG(fromTime(value -> value)), DATETIME_TO_LONG(fromDateTime(value -> value)), RATIONAL_TO_INT(fromDouble(value -> safeToInt(safeToLong(value)))), @@ -470,6 +538,7 @@ public enum Conversion { BOOL_TO_INT(fromBool(value -> value ? 1 : 0)), STRING_TO_INT(fromString(Integer::valueOf, "integer")), DATE_TO_INT(fromDateTime(DataTypeConversion::safeToInt)), + TIME_TO_INT(fromTime(DataTypeConversion::safeToInt)), DATETIME_TO_INT(fromDateTime(DataTypeConversion::safeToInt)), RATIONAL_TO_SHORT(fromDouble(value -> safeToShort(safeToLong(value)))), @@ -477,6 +546,7 @@ public enum Conversion { BOOL_TO_SHORT(fromBool(value -> value ? (short) 1 : (short) 0)), STRING_TO_SHORT(fromString(Short::valueOf, "short")), DATE_TO_SHORT(fromDateTime(DataTypeConversion::safeToShort)), + TIME_TO_SHORT(fromTime(DataTypeConversion::safeToShort)), DATETIME_TO_SHORT(fromDateTime(DataTypeConversion::safeToShort)), RATIONAL_TO_BYTE(fromDouble(value -> safeToByte(safeToLong(value)))), @@ -484,6 +554,7 @@ public enum Conversion { BOOL_TO_BYTE(fromBool(value -> value ? (byte) 1 : (byte) 0)), STRING_TO_BYTE(fromString(Byte::valueOf, "byte")), DATE_TO_BYTE(fromDateTime(DataTypeConversion::safeToByte)), + TIME_TO_BYTE(fromTime(DataTypeConversion::safeToByte)), DATETIME_TO_BYTE(fromDateTime(DataTypeConversion::safeToByte)), // TODO floating point conversions are lossy but conversions to integer conversions are not. Are we ok with that? @@ -492,6 +563,7 @@ public enum Conversion { BOOL_TO_FLOAT(fromBool(value -> value ? 1f : 0f)), STRING_TO_FLOAT(fromString(Float::valueOf, "float")), DATE_TO_FLOAT(fromDateTime(value -> (float) value)), + TIME_TO_FLOAT(fromTime(value -> (float) value)), DATETIME_TO_FLOAT(fromDateTime(value -> (float) value)), RATIONAL_TO_DOUBLE(fromDouble(Double::valueOf)), @@ -499,6 +571,7 @@ public enum Conversion { BOOL_TO_DOUBLE(fromBool(value -> value ? 1d : 0d)), STRING_TO_DOUBLE(fromString(Double::valueOf, "double")), DATE_TO_DOUBLE(fromDateTime(Double::valueOf)), + TIME_TO_DOUBLE(fromTime(Double::valueOf)), DATETIME_TO_DOUBLE(fromDateTime(Double::valueOf)), RATIONAL_TO_DATE(toDate(RATIONAL_TO_LONG)), @@ -507,6 +580,13 @@ public enum Conversion { STRING_TO_DATE(fromString(DateUtils::asDateOnly, "date")), DATETIME_TO_DATE(fromDatetimeToDate()), + RATIONAL_TO_TIME(toTime(RATIONAL_TO_LONG)), + INTEGER_TO_TIME(toTime(INTEGER_TO_LONG)), + BOOL_TO_TIME(toTime(BOOL_TO_INT)), + STRING_TO_TIME(fromString(DateUtils::asTimeOnly, "time")), + DATE_TO_TIME(fromDatetimeToTime()), + DATETIME_TO_TIME(fromDatetimeToTime()), + RATIONAL_TO_DATETIME(toDateTime(RATIONAL_TO_LONG)), INTEGER_TO_DATETIME(toDateTime(INTEGER_TO_LONG)), BOOL_TO_DATETIME(toDateTime(BOOL_TO_INT)), @@ -516,6 +596,7 @@ public enum Conversion { NUMERIC_TO_BOOLEAN(fromLong(value -> value != 0)), STRING_TO_BOOLEAN(fromString(DataTypeConversion::convertToBoolean, "boolean")), DATE_TO_BOOLEAN(fromDateTime(value -> value != 0)), + TIME_TO_BOOLEAN(fromTime(value -> value != 0)), DATETIME_TO_BOOLEAN(fromDateTime(value -> value != 0)), BOOL_TO_LONG(fromBool(value -> value ? 1L : 0L)), @@ -557,22 +638,34 @@ private static Function fromBool(Function conve return (Object l) -> converter.apply(((Boolean) l)); } - private static Function fromDateTime(Function converter) { - return l -> converter.apply(((ZonedDateTime) l).toInstant().toEpochMilli()); + private static Function fromTime(Function converter) { + return l -> converter.apply(((OffsetTime) l).atDate(DateUtils.EPOCH).toInstant().toEpochMilli()); } - private static Function toDateTime(Conversion conversion) { - return l -> DateUtils.asDateTime(((Number) conversion.convert(l)).longValue()); + private static Function fromDateTime(Function converter) { + return l -> converter.apply(((ZonedDateTime) l).toInstant().toEpochMilli()); } private static Function toDate(Conversion conversion) { return l -> DateUtils.asDateOnly(((Number) conversion.convert(l)).longValue()); } + private static Function toTime(Conversion conversion) { + return l -> DateUtils.asTimeOnly(((Number) conversion.convert(l)).longValue()); + } + + private static Function toDateTime(Conversion conversion) { + return l -> DateUtils.asDateTime(((Number) conversion.convert(l)).longValue()); + } + private static Function fromDatetimeToDate() { return l -> DateUtils.asDateOnly((ZonedDateTime) l); } + private static Function fromDatetimeToTime() { + return l -> ((ZonedDateTime) l).toOffsetDateTime().toOffsetTime(); + } + public Object convert(Object l) { if (l == null) { return null; diff --git a/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/type/DataTypes.java b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/type/DataTypes.java index cc902bcb077be..5e5ed25efdc71 100644 --- a/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/type/DataTypes.java +++ b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/type/DataTypes.java @@ -8,6 +8,8 @@ import org.elasticsearch.xpack.sql.SqlIllegalArgumentException; import org.elasticsearch.xpack.sql.expression.function.scalar.geo.GeoShape; import org.elasticsearch.xpack.sql.expression.literal.Interval; + +import java.time.OffsetTime; import java.time.ZonedDateTime; import static org.elasticsearch.xpack.sql.type.DataType.BOOLEAN; @@ -27,6 +29,7 @@ import static org.elasticsearch.xpack.sql.type.DataType.LONG; import static org.elasticsearch.xpack.sql.type.DataType.NULL; import static org.elasticsearch.xpack.sql.type.DataType.SHORT; +import static org.elasticsearch.xpack.sql.type.DataType.TIME; import static org.elasticsearch.xpack.sql.type.DataType.UNSUPPORTED; import static org.elasticsearch.xpack.sql.type.DataType.fromTypeName; @@ -67,6 +70,9 @@ public static DataType fromJava(Object value) { if (value instanceof Short) { return SHORT; } + if (value instanceof OffsetTime) { + return TIME; + } if (value instanceof ZonedDateTime) { return DATETIME; } diff --git a/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/util/DateUtils.java b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/util/DateUtils.java index 38db3cbe131cf..45072f7f480b1 100644 --- a/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/util/DateUtils.java +++ b/x-pack/plugin/sql/src/main/java/org/elasticsearch/xpack/sql/util/DateUtils.java @@ -12,6 +12,7 @@ import java.time.Instant; import java.time.LocalDate; +import java.time.OffsetTime; import java.time.ZoneId; import java.time.ZonedDateTime; import java.time.format.DateTimeFormatter; @@ -19,11 +20,15 @@ import static java.time.format.DateTimeFormatter.ISO_LOCAL_DATE; import static java.time.format.DateTimeFormatter.ISO_LOCAL_TIME; +import static java.time.format.DateTimeFormatter.ISO_TIME; public final class DateUtils { public static final ZoneId UTC = ZoneId.of("Z"); public static final String DATE_PARSE_FORMAT = "epoch_millis"; + // In Java 8 LocalDate.EPOCH is not available, introduced with later Java versions + public static final LocalDate EPOCH = LocalDate.of(1970, 1, 1); + public static final long DAY_IN_MILLIS = 60 * 60 * 24 * 1000L; private static final DateTimeFormatter DATE_TIME_ESCAPED_LITERAL_FORMATTER = new DateTimeFormatterBuilder() .append(ISO_LOCAL_DATE) @@ -33,8 +38,6 @@ public final class DateUtils { private static final DateFormatter UTC_DATE_TIME_FORMATTER = DateFormatter.forPattern("date_optional_time").withZone(UTC); - private static final long DAY_IN_MILLIS = 60 * 60 * 24 * 1000L; - private DateUtils() {} /** @@ -44,6 +47,24 @@ public static ZonedDateTime asDateOnly(long millis) { return ZonedDateTime.ofInstant(Instant.ofEpochMilli(millis), UTC).toLocalDate().atStartOfDay(UTC); } + /** + * Creates an date for SQL TIME type from the millis since epoch. + */ + public static OffsetTime asTimeOnly(long millis) { + return OffsetTime.ofInstant(Instant.ofEpochMilli(millis % DAY_IN_MILLIS), UTC); + } + + /** + * Creates an date for SQL TIME type from the millis since epoch. + */ + public static OffsetTime asTimeOnly(long millis, ZoneId zoneId) { + return OffsetTime.ofInstant(Instant.ofEpochMilli(millis % DAY_IN_MILLIS), zoneId); + } + + public static OffsetTime asTimeAtZone(OffsetTime time, ZoneId zonedId) { + return time.atDate(DateUtils.EPOCH).atZoneSameInstant(zonedId).toOffsetDateTime().toOffsetTime(); + } + /** * Creates a datetime from the millis since epoch (thus the time-zone is UTC). */ @@ -69,6 +90,10 @@ public static ZonedDateTime asDateOnly(ZonedDateTime zdt) { return zdt.toLocalDate().atStartOfDay(zdt.getZone()); } + public static OffsetTime asTimeOnly(String timeFormat) { + return DateFormatters.from(ISO_TIME.parse(timeFormat)).toOffsetDateTime().toOffsetTime(); + } + /** * Parses the given string into a DateTime using UTC as a default timezone. */ @@ -88,6 +113,10 @@ public static String toDateString(ZonedDateTime date) { return date.format(ISO_LOCAL_DATE); } + public static String toTimeString(OffsetTime time) { + return time.format(ISO_LOCAL_TIME); + } + public static long minDayInterval(long l) { if (l < DAY_IN_MILLIS ) { return DAY_IN_MILLIS; diff --git a/x-pack/plugin/sql/src/main/resources/org/elasticsearch/xpack/sql/plugin/sql_whitelist.txt b/x-pack/plugin/sql/src/main/resources/org/elasticsearch/xpack/sql/plugin/sql_whitelist.txt index 325048e3e0643..e3f2f15826dda 100644 --- a/x-pack/plugin/sql/src/main/resources/org/elasticsearch/xpack/sql/plugin/sql_whitelist.txt +++ b/x-pack/plugin/sql/src/main/resources/org/elasticsearch/xpack/sql/plugin/sql_whitelist.txt @@ -19,6 +19,9 @@ class org.elasticsearch.xpack.sql.expression.literal.IntervalDayTime { class org.elasticsearch.xpack.sql.expression.literal.IntervalYearMonth { } +class java.time.OffsetTime { +} + class org.elasticsearch.xpack.sql.expression.function.scalar.whitelist.InternalSqlScriptUtils { # @@ -114,6 +117,7 @@ class org.elasticsearch.xpack.sql.expression.function.scalar.whitelist.InternalS IntervalDayTime intervalDayTime(String, String) IntervalYearMonth intervalYearMonth(String, String) ZonedDateTime asDateTime(Object) + OffsetTime asTime(String) # # ASCII Functions diff --git a/x-pack/plugin/sql/src/test/java/org/elasticsearch/xpack/sql/analysis/analyzer/VerifierErrorMessagesTests.java b/x-pack/plugin/sql/src/test/java/org/elasticsearch/xpack/sql/analysis/analyzer/VerifierErrorMessagesTests.java index c76ddffe437c6..3b1e8da318ff2 100644 --- a/x-pack/plugin/sql/src/test/java/org/elasticsearch/xpack/sql/analysis/analyzer/VerifierErrorMessagesTests.java +++ b/x-pack/plugin/sql/src/test/java/org/elasticsearch/xpack/sql/analysis/analyzer/VerifierErrorMessagesTests.java @@ -203,10 +203,42 @@ public void testExtractNonDateTime() { assertEquals("1:8: Invalid datetime field [ABS]. Use any datetime function.", error("SELECT EXTRACT(ABS FROM date) FROM test")); } + public void testValidDateTimeFunctionsOnTime() { + accept("SELECT HOUR_OF_DAY(CAST(date AS TIME)) FROM test"); + accept("SELECT MINUTE_OF_HOUR(CAST(date AS TIME)) FROM test"); + accept("SELECT MINUTE_OF_DAY(CAST(date AS TIME)) FROM test"); + accept("SELECT SECOND_OF_MINUTE(CAST(date AS TIME)) FROM test"); + } + + public void testInvalidDateTimeFunctionsOnTime() { + assertEquals("1:8: argument of [DAY_OF_YEAR(CAST(date AS TIME))] must be [date or datetime], " + + "found value [CAST(date AS TIME)] type [time]", + error("SELECT DAY_OF_YEAR(CAST(date AS TIME)) FROM test")); + } + + public void testGroupByOnTimeNotAllowed() { + assertEquals("1:36: Function [CAST(date AS TIME)] with data type [time] cannot be used for grouping", + error("SELECT count(*) FROM test GROUP BY CAST(date AS TIME)")); + } + + public void testGroupByOnTimeWrappedWithScalar() { + accept("SELECT count(*) FROM test GROUP BY MINUTE(CAST(date AS TIME))"); + } + + public void testHistogramOnTimeNotAllowed() { + assertEquals("1:8: first argument of [HISTOGRAM] must be [date, datetime or numeric], " + + "found value [CAST(date AS TIME)] type [time]", + error("SELECT HISTOGRAM(CAST(date AS TIME), INTERVAL 1 MONTH), COUNT(*) FROM test GROUP BY 1")); + } + public void testSubtractFromInterval() { assertEquals("1:8: Cannot subtract a datetime[CAST('2000-01-01' AS DATETIME)] " + "from an interval[INTERVAL 1 MONTH]; do you mean the reverse?", error("SELECT INTERVAL 1 MONTH - CAST('2000-01-01' AS DATETIME)")); + + assertEquals("1:8: Cannot subtract a time[CAST('12:23:56.789' AS TIME)] " + + "from an interval[INTERVAL 1 MONTH]; do you mean the reverse?", + error("SELECT INTERVAL 1 MONTH - CAST('12:23:56.789' AS TIME)")); } public void testMultipleColumns() { @@ -293,7 +325,7 @@ public void testStarOnNested() { } public void testGroupByOnInexact() { - assertEquals("1:36: Field of data type [text] cannot be used for grouping; " + + assertEquals("1:36: Field [text] of data type [text] cannot be used for grouping; " + "No keyword/multi-field defined exact matches for [text]; define one or use MATCH/QUERY instead", error("SELECT COUNT(*) FROM test GROUP BY text")); } diff --git a/x-pack/plugin/sql/src/test/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/DateTimeProcessorTests.java b/x-pack/plugin/sql/src/test/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/DateTimeProcessorTests.java index 03f9c949d2992..6e248fb7794df 100644 --- a/x-pack/plugin/sql/src/test/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/DateTimeProcessorTests.java +++ b/x-pack/plugin/sql/src/test/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/DateTimeProcessorTests.java @@ -7,17 +7,20 @@ import org.elasticsearch.common.io.stream.Writeable.Reader; import org.elasticsearch.test.AbstractWireSerializingTestCase; +import org.elasticsearch.xpack.sql.SqlIllegalArgumentException; import org.elasticsearch.xpack.sql.expression.function.scalar.datetime.DateTimeProcessor.DateTimeExtractor; -import java.io.IOException; +import java.time.OffsetTime; +import java.time.ZoneId; import static org.elasticsearch.xpack.sql.expression.function.scalar.datetime.DateTimeTestUtils.dateTime; import static org.elasticsearch.xpack.sql.util.DateUtils.UTC; +import static org.hamcrest.Matchers.startsWith; public class DateTimeProcessorTests extends AbstractWireSerializingTestCase { public static DateTimeProcessor randomDateTimeProcessor() { - return new DateTimeProcessor(randomFrom(DateTimeExtractor.values()), UTC); + return new DateTimeProcessor(randomFrom(DateTimeExtractor.values()), randomZone()); } @Override @@ -31,12 +34,12 @@ protected Reader instanceReader() { } @Override - protected DateTimeProcessor mutateInstance(DateTimeProcessor instance) throws IOException { + protected DateTimeProcessor mutateInstance(DateTimeProcessor instance) { DateTimeExtractor replaced = randomValueOtherThan(instance.extractor(), () -> randomFrom(DateTimeExtractor.values())); - return new DateTimeProcessor(replaced, UTC); + return new DateTimeProcessor(replaced, randomZone()); } - public void testApply() { + public void testApply_withTimezoneUTC() { DateTimeProcessor proc = new DateTimeProcessor(DateTimeExtractor.YEAR, UTC); assertEquals(1970, proc.process(dateTime(0L))); assertEquals(2017, proc.process(dateTime(2017, 01, 02, 10, 10))); @@ -46,4 +49,21 @@ public void testApply() { assertEquals(2, proc.process(dateTime(2017, 01, 02, 10, 10))); assertEquals(31, proc.process(dateTime(2017, 01, 31, 10, 10))); } + + public void testApply_withTimezoneOtherThanUTC() { + ZoneId zoneId = ZoneId.of("Etc/GMT-10"); + DateTimeProcessor proc = new DateTimeProcessor(DateTimeExtractor.YEAR, zoneId); + assertEquals(2018, proc.process(dateTime(2017, 12, 31, 18, 10))); + + proc = new DateTimeProcessor(DateTimeExtractor.DAY_OF_MONTH, zoneId); + assertEquals(1, proc.process(dateTime(2017, 12, 31, 20, 30))); + } + + public void testFailOnTime() { + DateTimeProcessor proc = new DateTimeProcessor(DateTimeExtractor.YEAR, UTC); + SqlIllegalArgumentException e = expectThrows(SqlIllegalArgumentException.class, () -> { + proc.process(OffsetTime.now(UTC)); + }); + assertThat(e.getMessage(), startsWith("A [date], a [time] or a [datetime] is required; received ")); + } } diff --git a/x-pack/plugin/sql/src/test/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/DateTimeTestUtils.java b/x-pack/plugin/sql/src/test/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/DateTimeTestUtils.java index 4323cce234c54..13215eb41aebc 100644 --- a/x-pack/plugin/sql/src/test/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/DateTimeTestUtils.java +++ b/x-pack/plugin/sql/src/test/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/DateTimeTestUtils.java @@ -8,6 +8,8 @@ import org.elasticsearch.xpack.sql.util.DateUtils; +import java.time.OffsetTime; +import java.time.ZoneOffset; import java.time.ZonedDateTime; public class DateTimeTestUtils { @@ -25,4 +27,12 @@ public static ZonedDateTime dateTime(long millisSinceEpoch) { public static ZonedDateTime date(long millisSinceEpoch) { return DateUtils.asDateOnly(millisSinceEpoch); } + + public static OffsetTime time(long millisSinceEpoch) { + return DateUtils.asTimeOnly(millisSinceEpoch); + } + + public static OffsetTime time(int hour, int minute, int second, int nano) { + return OffsetTime.of(hour, minute, second, nano, ZoneOffset.UTC); + } } diff --git a/x-pack/plugin/sql/src/test/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/TimeProcessorTests.java b/x-pack/plugin/sql/src/test/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/TimeProcessorTests.java new file mode 100644 index 0000000000000..65b2cde2d0a69 --- /dev/null +++ b/x-pack/plugin/sql/src/test/java/org/elasticsearch/xpack/sql/expression/function/scalar/datetime/TimeProcessorTests.java @@ -0,0 +1,77 @@ +/* + * Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one + * or more contributor license agreements. Licensed under the Elastic License; + * you may not use this file except in compliance with the Elastic License. + */ +package org.elasticsearch.xpack.sql.expression.function.scalar.datetime; + +import org.elasticsearch.common.io.stream.Writeable.Reader; +import org.elasticsearch.test.AbstractWireSerializingTestCase; +import org.elasticsearch.xpack.sql.expression.function.scalar.datetime.DateTimeProcessor.DateTimeExtractor; + +import java.time.ZoneId; + +import static org.elasticsearch.xpack.sql.expression.function.scalar.datetime.DateTimeTestUtils.time; +import static org.elasticsearch.xpack.sql.util.DateUtils.UTC; + +public class TimeProcessorTests extends AbstractWireSerializingTestCase { + + public static TimeProcessor randomTimeProcessor() { + return new TimeProcessor(randomFrom(DateTimeExtractor.values()), randomZone()); + } + + @Override + protected TimeProcessor createTestInstance() { + return randomTimeProcessor(); + } + + @Override + protected Reader instanceReader() { + return TimeProcessor::new; + } + + @Override + protected TimeProcessor mutateInstance(TimeProcessor instance) { + DateTimeExtractor replaced = randomValueOtherThan(instance.extractor(), () -> randomFrom(DateTimeExtractor.values())); + return new TimeProcessor(replaced, randomZone()); + } + + public void testApply_withTimeZoneUTC() { + TimeProcessor proc = new TimeProcessor(DateTimeExtractor.SECOND_OF_MINUTE, UTC); + assertEquals(0, proc.process(time(0L))); + assertEquals(2, proc.process(time(2345L))); + + proc = new TimeProcessor(DateTimeExtractor.MINUTE_OF_DAY, UTC); + assertEquals(0, proc.process(time(0L))); + assertEquals(620, proc.process(time(10, 20, 30, 123456789))); + + proc = new TimeProcessor(DateTimeExtractor.MINUTE_OF_HOUR, UTC); + assertEquals(0, proc.process(time(0L))); + assertEquals(20, proc.process(time(10, 20, 30, 123456789))); + + proc = new TimeProcessor(DateTimeExtractor.HOUR_OF_DAY, UTC); + assertEquals(0, proc.process(time(0L))); + assertEquals(10, proc.process(time(10, 20, 30, 123456789))); + } + + public void testApply_withTimeZoneOtherThanUTC() { + ZoneId zoneId = ZoneId.of("Etc/GMT-10"); + + TimeProcessor proc = new TimeProcessor(DateTimeExtractor.SECOND_OF_MINUTE, zoneId); + assertEquals(0, proc.process(time(0L))); + assertEquals(2, proc.process(time(2345L))); + + proc = new TimeProcessor(DateTimeExtractor.MINUTE_OF_DAY, zoneId); + assertEquals(600, proc.process(time(0L))); + assertEquals(1220, proc.process(time(10, 20, 30, 123456789))); + + proc = new TimeProcessor(DateTimeExtractor.MINUTE_OF_HOUR, zoneId); + assertEquals(0, proc.process(time(0L))); + assertEquals(20, proc.process(time(10, 20, 30, 123456789))); + + proc = new TimeProcessor(DateTimeExtractor.HOUR_OF_DAY, zoneId); + assertEquals(10, proc.process(time(0L))); + assertEquals(20, proc.process(time(10, 20, 30, 123456789)));; + assertEquals(4, proc.process(time(18, 20, 30, 123456789))); + } +} diff --git a/x-pack/plugin/sql/src/test/java/org/elasticsearch/xpack/sql/expression/predicate/operator/arithmetic/BinaryArithmeticTests.java b/x-pack/plugin/sql/src/test/java/org/elasticsearch/xpack/sql/expression/predicate/operator/arithmetic/BinaryArithmeticTests.java index 696f999b0b051..1c4b0697f959e 100644 --- a/x-pack/plugin/sql/src/test/java/org/elasticsearch/xpack/sql/expression/predicate/operator/arithmetic/BinaryArithmeticTests.java +++ b/x-pack/plugin/sql/src/test/java/org/elasticsearch/xpack/sql/expression/predicate/operator/arithmetic/BinaryArithmeticTests.java @@ -15,6 +15,7 @@ import org.elasticsearch.xpack.sql.util.DateUtils; import java.time.Duration; +import java.time.OffsetTime; import java.time.Period; import java.time.ZonedDateTime; import java.time.temporal.TemporalAmount; @@ -104,6 +105,33 @@ public void testAddDayTimeIntervalToDateTimeReverse() { assertEquals(L(now.plus(t)), L(x)); } + public void testAddYearMonthIntervalToTime() { + OffsetTime now = OffsetTime.now(DateUtils.UTC); + Literal l = L(now); + TemporalAmount t = Period.ofYears(100).plusMonths(50); + Literal r = interval(t, INTERVAL_HOUR); + OffsetTime x = add(l, r); + assertEquals(L(now), L(x)); + } + + public void testAddDayTimeIntervalToTime() { + OffsetTime now = OffsetTime.now(DateUtils.UTC); + Literal l = L(now); + TemporalAmount t = Duration.ofHours(32); + Literal r = interval(Duration.ofHours(32), INTERVAL_HOUR); + OffsetTime x = add(l, r); + assertEquals(L(now.plus(t)), L(x)); + } + + public void testAddDayTimeIntervalToTimeReverse() { + OffsetTime now = OffsetTime.now(DateUtils.UTC); + Literal l = L(now); + TemporalAmount t = Duration.ofHours(45); + Literal r = interval(Duration.ofHours(45), INTERVAL_HOUR); + OffsetTime x = add(r, l); + assertEquals(L(now.plus(t)), L(x)); + } + public void testAddNumberToIntervalIllegal() { Literal r = interval(Duration.ofHours(2), INTERVAL_HOUR); SqlIllegalArgumentException expect = expectThrows(SqlIllegalArgumentException.class, () -> add(r, L(1))); @@ -142,12 +170,6 @@ public void testSubYearMonthIntervalToDateTimeIllegal() { assertEquals("Cannot subtract a date from an interval; do you mean the reverse?", ex.getMessage()); } - public void testSubNumberFromIntervalIllegal() { - Literal r = interval(Duration.ofHours(2), INTERVAL_HOUR); - SqlIllegalArgumentException expect = expectThrows(SqlIllegalArgumentException.class, () -> sub(r, L(1))); - assertEquals("Cannot compute [-] between [IntervalDayTime] [Integer]", expect.getMessage()); - } - public void testSubDayTimeIntervalToDateTime() { ZonedDateTime now = ZonedDateTime.now(DateUtils.UTC); Literal l = L(now); @@ -157,7 +179,40 @@ public void testSubDayTimeIntervalToDateTime() { assertEquals(L(now.minus(t)), L(x)); } - public void testMulIntervalNumber() throws Exception { + public void testSubYearMonthIntervalToTime() { + OffsetTime now = OffsetTime.now(DateUtils.UTC); + Literal l = L(now); + TemporalAmount t = Period.ofYears(100).plusMonths(50); + Literal r = interval(t, INTERVAL_HOUR); + OffsetTime x = sub(l, r); + assertEquals(L(now), L(x)); + } + + public void testSubYearMonthIntervalToTimeIllegal() { + OffsetTime now = OffsetTime.now(DateUtils.UTC); + Literal l = L(now); + TemporalAmount t = Period.ofYears(100).plusMonths(50); + Literal r = interval(t, INTERVAL_HOUR); + SqlIllegalArgumentException ex = expectThrows(SqlIllegalArgumentException.class, () -> sub(r, l)); + assertEquals("Cannot subtract a date from an interval; do you mean the reverse?", ex.getMessage()); + } + + public void testSubDayTimeIntervalToTime() { + OffsetTime now = OffsetTime.now(DateUtils.UTC); + Literal l = L(now); + TemporalAmount t = Duration.ofHours(36); + Literal r = interval(Duration.ofHours(36), INTERVAL_HOUR); + OffsetTime x = sub(l, r); + assertEquals(L(now.minus(t)), L(x)); + } + + public void testSubNumberFromIntervalIllegal() { + Literal r = interval(Duration.ofHours(2), INTERVAL_HOUR); + SqlIllegalArgumentException expect = expectThrows(SqlIllegalArgumentException.class, () -> sub(r, L(1))); + assertEquals("Cannot compute [-] between [IntervalDayTime] [Integer]", expect.getMessage()); + } + + public void testMulIntervalNumber() { Literal l = interval(Duration.ofHours(2), INTERVAL_HOUR); IntervalDayTime interval = mul(l, -1); assertEquals(INTERVAL_HOUR, interval.dataType()); @@ -165,7 +220,7 @@ public void testMulIntervalNumber() throws Exception { assertEquals(Duration.ofHours(2).negated(), p); } - public void testMulNumberInterval() throws Exception { + public void testMulNumberInterval() { Literal r = interval(Period.ofYears(1), INTERVAL_YEAR); IntervalYearMonth interval = mul(-2, r); assertEquals(INTERVAL_YEAR, interval.dataType()); diff --git a/x-pack/plugin/sql/src/test/java/org/elasticsearch/xpack/sql/parser/EscapedFunctionsTests.java b/x-pack/plugin/sql/src/test/java/org/elasticsearch/xpack/sql/parser/EscapedFunctionsTests.java index bc3ea049c242e..6fd4611a43416 100644 --- a/x-pack/plugin/sql/src/test/java/org/elasticsearch/xpack/sql/parser/EscapedFunctionsTests.java +++ b/x-pack/plugin/sql/src/test/java/org/elasticsearch/xpack/sql/parser/EscapedFunctionsTests.java @@ -6,7 +6,6 @@ package org.elasticsearch.xpack.sql.parser; import org.elasticsearch.test.ESTestCase; -import org.elasticsearch.xpack.sql.SqlIllegalArgumentException; import org.elasticsearch.xpack.sql.expression.Expression; import org.elasticsearch.xpack.sql.expression.Literal; import org.elasticsearch.xpack.sql.expression.UnresolvedAttribute; @@ -180,9 +179,9 @@ public void testDateLiteralValidation() { ex.getMessage()); } - public void testTimeLiteralUnsupported() { - SqlIllegalArgumentException ex = expectThrows(SqlIllegalArgumentException.class, () -> timeLiteral("10:10:10")); - assertThat(ex.getMessage(), is("Time (only) literals are not supported; a date component is required as well")); + public void testTimeLiteral() { + Literal l = timeLiteral("12:23:56"); + assertThat(l.dataType(), is(DataType.TIME)); } public void testTimeLiteralValidation() { diff --git a/x-pack/plugin/sql/src/test/java/org/elasticsearch/xpack/sql/plan/logical/command/sys/SysParserTests.java b/x-pack/plugin/sql/src/test/java/org/elasticsearch/xpack/sql/plan/logical/command/sys/SysParserTests.java index 2436e9dc341c8..1472a24a18a45 100644 --- a/x-pack/plugin/sql/src/test/java/org/elasticsearch/xpack/sql/plan/logical/command/sys/SysParserTests.java +++ b/x-pack/plugin/sql/src/test/java/org/elasticsearch/xpack/sql/plan/logical/command/sys/SysParserTests.java @@ -57,11 +57,11 @@ private Tuple sql(String sql) { return new Tuple<>(cmd, session); } - public void testSysTypes() throws Exception { + public void testSysTypes() { Command cmd = sql("SYS TYPES").v1(); List names = asList("BYTE", "LONG", "BINARY", "NULL", "INTEGER", "SHORT", "HALF_FLOAT", "FLOAT", "DOUBLE", "SCALED_FLOAT", - "KEYWORD", "TEXT", "IP", "BOOLEAN", "DATE", "DATETIME", + "KEYWORD", "TEXT", "IP", "BOOLEAN", "DATE", "TIME", "DATETIME", "INTERVAL_YEAR", "INTERVAL_MONTH", "INTERVAL_DAY", "INTERVAL_HOUR", "INTERVAL_MINUTE", "INTERVAL_SECOND", "INTERVAL_YEAR_TO_MONTH", "INTERVAL_DAY_TO_HOUR", "INTERVAL_DAY_TO_MINUTE", "INTERVAL_DAY_TO_SECOND", "INTERVAL_HOUR_TO_MINUTE", "INTERVAL_HOUR_TO_SECOND", "INTERVAL_MINUTE_TO_SECOND", @@ -86,11 +86,11 @@ public void testSysTypes() throws Exception { }, ex -> fail(ex.getMessage()))); } - public void testSysColsNoArgs() throws Exception { + public void testSysColsNoArgs() { runSysColumns("SYS COLUMNS"); } - public void testSysColumnEmptyCatalog() throws Exception { + public void testSysColumnEmptyCatalog() { Tuple sql = sql("SYS COLUMNS CATALOG '' TABLE LIKE '%' LIKE '%'"); sql.v1().execute(sql.v2(), ActionListener.wrap(r -> { @@ -99,7 +99,7 @@ public void testSysColumnEmptyCatalog() throws Exception { }, ex -> fail(ex.getMessage()))); } - public void testSysColsTableOnlyCatalog() throws Exception { + public void testSysColsTableOnlyCatalog() { Tuple sql = sql("SYS COLUMNS CATALOG 'catalog'"); sql.v1().execute(sql.v2(), ActionListener.wrap(r -> { @@ -108,20 +108,20 @@ public void testSysColsTableOnlyCatalog() throws Exception { }, ex -> fail(ex.getMessage()))); } - public void testSysColsTableOnlyPattern() throws Exception { + public void testSysColsTableOnlyPattern() { runSysColumns("SYS COLUMNS TABLE LIKE 'test'"); } - public void testSysColsColOnlyPattern() throws Exception { + public void testSysColsColOnlyPattern() { runSysColumns("SYS COLUMNS LIKE '%'"); } - public void testSysColsTableAndColsPattern() throws Exception { + public void testSysColsTableAndColsPattern() { runSysColumns("SYS COLUMNS TABLE LIKE 'test' LIKE '%'"); } - private void runSysColumns(String commandVariation) throws Exception { + private void runSysColumns(String commandVariation) { Tuple sql = sql(commandVariation); List names = asList("bool", "int", diff --git a/x-pack/plugin/sql/src/test/java/org/elasticsearch/xpack/sql/plan/logical/command/sys/SysTypesTests.java b/x-pack/plugin/sql/src/test/java/org/elasticsearch/xpack/sql/plan/logical/command/sys/SysTypesTests.java index 981d58d150010..805268dd5b687 100644 --- a/x-pack/plugin/sql/src/test/java/org/elasticsearch/xpack/sql/plan/logical/command/sys/SysTypesTests.java +++ b/x-pack/plugin/sql/src/test/java/org/elasticsearch/xpack/sql/plan/logical/command/sys/SysTypesTests.java @@ -44,7 +44,7 @@ public void testSysTypes() { Command cmd = sql("SYS TYPES").v1(); List names = asList("BYTE", "LONG", "BINARY", "NULL", "INTEGER", "SHORT", "HALF_FLOAT", "FLOAT", "DOUBLE", "SCALED_FLOAT", - "KEYWORD", "TEXT", "IP", "BOOLEAN", "DATE", "DATETIME", + "KEYWORD", "TEXT", "IP", "BOOLEAN", "DATE", "TIME", "DATETIME", "INTERVAL_YEAR", "INTERVAL_MONTH", "INTERVAL_DAY", "INTERVAL_HOUR", "INTERVAL_MINUTE", "INTERVAL_SECOND", "INTERVAL_YEAR_TO_MONTH", "INTERVAL_DAY_TO_HOUR", "INTERVAL_DAY_TO_MINUTE", "INTERVAL_DAY_TO_SECOND", "INTERVAL_HOUR_TO_MINUTE", "INTERVAL_HOUR_TO_SECOND", "INTERVAL_MINUTE_TO_SECOND", diff --git a/x-pack/plugin/sql/src/test/java/org/elasticsearch/xpack/sql/planner/QueryTranslatorTests.java b/x-pack/plugin/sql/src/test/java/org/elasticsearch/xpack/sql/planner/QueryTranslatorTests.java index b83fa33ab8d56..04bb20d7252b7 100644 --- a/x-pack/plugin/sql/src/test/java/org/elasticsearch/xpack/sql/planner/QueryTranslatorTests.java +++ b/x-pack/plugin/sql/src/test/java/org/elasticsearch/xpack/sql/planner/QueryTranslatorTests.java @@ -753,7 +753,7 @@ public void testGroupByHistogramWithDateTruncateIntervalToDayMultiples() { assertEquals(259200000L, ((GroupByDateHistogram) eqe.queryContainer().aggs().groups().get(0)).interval()); } } - + public void testCountAndCountDistinctFolding() { PhysicalPlan p = optimizeAndPlan("SELECT COUNT(DISTINCT keyword) dkey, COUNT(keyword) key FROM test"); assertEquals(EsQueryExec.class, p.getClass()); @@ -898,57 +898,56 @@ public void testTopHitsAggregationWithTwoArgs() { } } - - public void testGlobalCountInImplicitGroupByForcesTrackHits() throws Exception { + public void testGlobalCountInImplicitGroupByForcesTrackHits() { PhysicalPlan p = optimizeAndPlan("SELECT COUNT(*) FROM test"); assertEquals(EsQueryExec.class, p.getClass()); EsQueryExec eqe = (EsQueryExec) p; assertTrue("Should be tracking hits", eqe.queryContainer().shouldTrackHits()); } - public void testGlobalCountAllInImplicitGroupByForcesTrackHits() throws Exception { + public void testGlobalCountAllInImplicitGroupByForcesTrackHits() { PhysicalPlan p = optimizeAndPlan("SELECT COUNT(ALL *) FROM test"); assertEquals(EsQueryExec.class, p.getClass()); EsQueryExec eqe = (EsQueryExec) p; assertTrue("Should be tracking hits", eqe.queryContainer().shouldTrackHits()); } - public void testGlobalCountInSpecificGroupByDoesNotForceTrackHits() throws Exception { + public void testGlobalCountInSpecificGroupByDoesNotForceTrackHits() { PhysicalPlan p = optimizeAndPlan("SELECT COUNT(*) FROM test GROUP BY int"); assertEquals(EsQueryExec.class, p.getClass()); EsQueryExec eqe = (EsQueryExec) p; assertFalse("Should NOT be tracking hits", eqe.queryContainer().shouldTrackHits()); } - public void testFieldAllCountDoesNotTrackHits() throws Exception { + public void testFieldAllCountDoesNotTrackHits() { PhysicalPlan p = optimizeAndPlan("SELECT COUNT(ALL int) FROM test"); assertEquals(EsQueryExec.class, p.getClass()); EsQueryExec eqe = (EsQueryExec) p; assertFalse("Should NOT be tracking hits", eqe.queryContainer().shouldTrackHits()); } - public void testFieldCountDoesNotTrackHits() throws Exception { + public void testFieldCountDoesNotTrackHits() { PhysicalPlan p = optimizeAndPlan("SELECT COUNT(int) FROM test"); assertEquals(EsQueryExec.class, p.getClass()); EsQueryExec eqe = (EsQueryExec) p; assertFalse("Should NOT be tracking hits", eqe.queryContainer().shouldTrackHits()); } - public void testDistinctCountDoesNotTrackHits() throws Exception { + public void testDistinctCountDoesNotTrackHits() { PhysicalPlan p = optimizeAndPlan("SELECT COUNT(DISTINCT int) FROM test"); assertEquals(EsQueryExec.class, p.getClass()); EsQueryExec eqe = (EsQueryExec) p; assertFalse("Should NOT be tracking hits", eqe.queryContainer().shouldTrackHits()); } - public void testNoCountDoesNotTrackHits() throws Exception { + public void testNoCountDoesNotTrackHits() { PhysicalPlan p = optimizeAndPlan("SELECT int FROM test"); assertEquals(EsQueryExec.class, p.getClass()); EsQueryExec eqe = (EsQueryExec) p; assertFalse("Should NOT be tracking hits", eqe.queryContainer().shouldTrackHits()); } - public void testZonedDateTimeInScripts() throws Exception { + public void testZonedDateTimeInScripts() { PhysicalPlan p = optimizeAndPlan( "SELECT date FROM test WHERE date + INTERVAL 1 YEAR > CAST('2019-03-11T12:34:56.000Z' AS DATETIME)"); assertEquals(EsQueryExec.class, p.getClass()); diff --git a/x-pack/plugin/sql/src/test/java/org/elasticsearch/xpack/sql/type/DataTypeConversionTests.java b/x-pack/plugin/sql/src/test/java/org/elasticsearch/xpack/sql/type/DataTypeConversionTests.java index a72f9ee7f1244..d44f69393f12b 100644 --- a/x-pack/plugin/sql/src/test/java/org/elasticsearch/xpack/sql/type/DataTypeConversionTests.java +++ b/x-pack/plugin/sql/src/test/java/org/elasticsearch/xpack/sql/type/DataTypeConversionTests.java @@ -11,14 +11,16 @@ import org.elasticsearch.xpack.sql.tree.Location; import org.elasticsearch.xpack.sql.tree.Source; import org.elasticsearch.xpack.sql.type.DataTypeConversion.Conversion; -import org.elasticsearch.xpack.sql.util.DateUtils; +import java.time.OffsetTime; +import java.time.ZoneOffset; import java.time.ZonedDateTime; import java.util.stream.Collectors; import java.util.stream.Stream; import static org.elasticsearch.xpack.sql.expression.function.scalar.datetime.DateTimeTestUtils.date; import static org.elasticsearch.xpack.sql.expression.function.scalar.datetime.DateTimeTestUtils.dateTime; +import static org.elasticsearch.xpack.sql.expression.function.scalar.datetime.DateTimeTestUtils.time; import static org.elasticsearch.xpack.sql.type.DataType.BOOLEAN; import static org.elasticsearch.xpack.sql.type.DataType.BYTE; import static org.elasticsearch.xpack.sql.type.DataType.DATE; @@ -38,12 +40,15 @@ import static org.elasticsearch.xpack.sql.type.DataType.NULL; import static org.elasticsearch.xpack.sql.type.DataType.SHORT; import static org.elasticsearch.xpack.sql.type.DataType.TEXT; +import static org.elasticsearch.xpack.sql.type.DataType.TIME; import static org.elasticsearch.xpack.sql.type.DataType.UNSUPPORTED; import static org.elasticsearch.xpack.sql.type.DataType.fromTypeName; import static org.elasticsearch.xpack.sql.type.DataType.values; import static org.elasticsearch.xpack.sql.type.DataTypeConversion.commonType; import static org.elasticsearch.xpack.sql.type.DataTypeConversion.conversionFor; +import static org.elasticsearch.xpack.sql.util.DateUtils.asDateOnly; import static org.elasticsearch.xpack.sql.util.DateUtils.asDateTime; +import static org.elasticsearch.xpack.sql.util.DateUtils.asTimeOnly; public class DataTypeConversionTests extends ESTestCase { @@ -58,8 +63,16 @@ public void testConversionToString() { { Conversion conversion = conversionFor(DATE, to); assertNull(conversion.convert(null)); - assertEquals("1973-11-29", conversion.convert(DateUtils.asDateOnly(123456789101L))); - assertEquals("1966-02-02", conversion.convert(DateUtils.asDateOnly(-123456789101L))); + assertEquals("1973-11-29", conversion.convert(asDateOnly(123456789101L))); + assertEquals("1966-02-02", conversion.convert(asDateOnly(-123456789101L))); + } + { + Conversion conversion = conversionFor(TIME, to); + assertNull(conversion.convert(null)); + assertEquals("00:02:03.456", conversion.convert(asTimeOnly(123456L))); + assertEquals("21:33:09.101", conversion.convert(asTimeOnly(123456789101L))); + assertEquals("23:57:56.544", conversion.convert(asTimeOnly(-123456L))); + assertEquals("02:26:50.899", conversion.convert(asTimeOnly(-123456789101L))); } { Conversion conversion = conversionFor(DATETIME, to); @@ -98,8 +111,16 @@ public void testConversionToLong() { { Conversion conversion = conversionFor(DATE, to); assertNull(conversion.convert(null)); - assertEquals(123379200000L, conversion.convert(DateUtils.asDateOnly(123456789101L))); - assertEquals(-123465600000L, conversion.convert(DateUtils.asDateOnly(-123456789101L))); + assertEquals(123379200000L, conversion.convert(asDateOnly(123456789101L))); + assertEquals(-123465600000L, conversion.convert(asDateOnly(-123456789101L))); + } + { + Conversion conversion = conversionFor(TIME, to); + assertNull(conversion.convert(null)); + assertEquals(123456L, conversion.convert(asTimeOnly(123456L))); + assertEquals(77589101L, conversion.convert(asTimeOnly(123456789101L))); + assertEquals(86276544L, conversion.convert(asTimeOnly(-123456L))); + assertEquals(8810899L, conversion.convert(asTimeOnly(-123456789101L))); } { Conversion conversion = conversionFor(DATETIME, to); @@ -140,6 +161,10 @@ public void testConversionToDate() { assertEquals(date(1), conversion.convert(true)); assertEquals(date(0), conversion.convert(false)); } + { + Exception e = expectThrows(SqlIllegalArgumentException.class, () -> conversionFor(TIME, to)); + assertEquals("cannot convert from [time] to [date]", e.getMessage()); + } { Conversion conversion = conversionFor(DATETIME, to); assertNull(conversion.convert(null)); @@ -160,12 +185,67 @@ public void testConversionToDate() { ZonedDateTime zdt = org.elasticsearch.common.time.DateUtils.nowWithMillisResolution(); Conversion forward = conversionFor(DATE, KEYWORD); Conversion back = conversionFor(KEYWORD, DATE); - assertEquals(DateUtils.asDateOnly(zdt), back.convert(forward.convert(zdt))); + assertEquals(asDateOnly(zdt), back.convert(forward.convert(zdt))); Exception e = expectThrows(SqlIllegalArgumentException.class, () -> conversion.convert("0xff")); assertEquals("cannot cast [0xff] to [date]: Text '0xff' could not be parsed at index 0", e.getMessage()); } } + public void testConversionToTime() { + DataType to = TIME; + { + Conversion conversion = conversionFor(DOUBLE, to); + assertNull(conversion.convert(null)); + assertEquals(time(10L), conversion.convert(10.0)); + assertEquals(time(10L), conversion.convert(10.1)); + assertEquals(time(11L), conversion.convert(10.6)); + Exception e = expectThrows(SqlIllegalArgumentException.class, () -> conversion.convert(Double.MAX_VALUE)); + assertEquals("[" + Double.MAX_VALUE + "] out of [long] range", e.getMessage()); + } + { + Conversion conversion = conversionFor(INTEGER, to); + assertNull(conversion.convert(null)); + assertEquals(time(10L), conversion.convert(10)); + assertEquals(time(-134L), conversion.convert(-134)); + } + { + Conversion conversion = conversionFor(BOOLEAN, to); + assertNull(conversion.convert(null)); + assertEquals(time(1), conversion.convert(true)); + assertEquals(time(0), conversion.convert(false)); + } + { + Conversion conversion = conversionFor(DATE, to); + assertNull(conversion.convert(null)); + assertEquals(time(123379200000L), conversion.convert(asDateOnly(123456789101L))); + assertEquals(time(-123465600000L), conversion.convert(asDateOnly(-123456789101L))); + } + { + Conversion conversion = conversionFor(DATETIME, to); + assertNull(conversion.convert(null)); + assertEquals(time(77589101L), conversion.convert(asDateTime(123456789101L))); + assertEquals(time(8810899L), conversion.convert(asDateTime(-123456789101L))); + } + { + Conversion conversion = conversionFor(KEYWORD, to); + assertNull(conversion.convert(null)); + + assertEquals(time(0L), conversion.convert("00:00:00Z")); + assertEquals(time(1000L), conversion.convert("00:00:01Z")); + assertEquals(time(1234L), conversion.convert("00:00:01.234Z")); + assertEquals(time(63296789L).withOffsetSameInstant(ZoneOffset.ofHours(-5)), conversion.convert("12:34:56.789-05:00")); + + // double check back and forth conversion + OffsetTime ot = org.elasticsearch.common.time.DateUtils.nowWithMillisResolution().toOffsetDateTime().toOffsetTime(); + Conversion forward = conversionFor(TIME, KEYWORD); + Conversion back = conversionFor(KEYWORD, TIME); + assertEquals(ot, back.convert(forward.convert(ot))); + Exception e = expectThrows(SqlIllegalArgumentException.class, () -> conversion.convert("0xff")); + assertEquals("cannot cast [0xff] to [time]: Text '0xff' could not be parsed at index 0", + e.getMessage()); + } + } + public void testConversionToDateTime() { DataType to = DATETIME; { @@ -192,8 +272,12 @@ public void testConversionToDateTime() { { Conversion conversion = conversionFor(DATE, to); assertNull(conversion.convert(null)); - assertEquals(dateTime(123379200000L), conversion.convert(DateUtils.asDateOnly(123456789101L))); - assertEquals(dateTime(-123465600000L), conversion.convert(DateUtils.asDateOnly(-123456789101L))); + assertEquals(dateTime(123379200000L), conversion.convert(asDateOnly(123456789101L))); + assertEquals(dateTime(-123465600000L), conversion.convert(asDateOnly(-123456789101L))); + } + { + Exception e = expectThrows(SqlIllegalArgumentException.class, () -> conversionFor(TIME, to)); + assertEquals("cannot convert from [time] to [datetime]", e.getMessage()); } { Conversion conversion = conversionFor(KEYWORD, to); @@ -217,6 +301,58 @@ public void testConversionToDateTime() { } } + public void testConversionToFloat() { + DataType to = FLOAT; + { + Conversion conversion = conversionFor(DOUBLE, to); + assertNull(conversion.convert(null)); + assertEquals(10.0f, (float) conversion.convert(10.0d), 0.00001); + assertEquals(10.1f, (float) conversion.convert(10.1d), 0.00001); + assertEquals(10.6f, (float) conversion.convert(10.6d), 0.00001); + } + { + Conversion conversion = conversionFor(INTEGER, to); + assertNull(conversion.convert(null)); + assertEquals(10.0f, (float) conversion.convert(10), 0.00001); + assertEquals(-134.0f, (float) conversion.convert(-134), 0.00001); + } + { + Conversion conversion = conversionFor(BOOLEAN, to); + assertNull(conversion.convert(null)); + assertEquals(1.0f, (float) conversion.convert(true), 0); + assertEquals(0.0f, (float) conversion.convert(false), 0); + } + { + Conversion conversion = conversionFor(DATE, to); + assertNull(conversion.convert(null)); + assertEquals(1.233792E11f, (float) conversion.convert(asDateOnly(123456789101L)), 0); + assertEquals(-1.234656E11f, (float) conversion.convert(asDateOnly(-123456789101L)), 0); + } + { + Conversion conversion = conversionFor(TIME, to); + assertNull(conversion.convert(null)); + assertEquals(123456.0f, (float) conversion.convert(asTimeOnly(123456L)), 0); + assertEquals(7.7589104E7f, (float) conversion.convert(asTimeOnly(123456789101L)), 0); + assertEquals(8.6276544E7f, (float) conversion.convert(asTimeOnly(-123456L)), 0); + assertEquals(8810899.0f, (float) conversion.convert(asTimeOnly(-123456789101L)), 0); + } + { + Conversion conversion = conversionFor(DATETIME, to); + assertNull(conversion.convert(null)); + assertEquals(1.23456789101E11f, (float) conversion.convert(asDateTime(123456789101L)), 0); + assertEquals(-1.23456789101E11f, (float) conversion.convert(asDateTime(-123456789101L)), 0); + } + { + Conversion conversion = conversionFor(KEYWORD, to); + assertNull(conversion.convert(null)); + assertEquals(1.0f, (float) conversion.convert("1"), 0); + assertEquals(0.0f, (float) conversion.convert("-0"), 0); + assertEquals(12.776f, (float) conversion.convert("12.776"), 0.00001); + Exception e = expectThrows(SqlIllegalArgumentException.class, () -> conversion.convert("0xff")); + assertEquals("cannot cast [0xff] to [float]", e.getMessage()); + } + } + public void testConversionToDouble() { DataType to = DOUBLE; { @@ -241,8 +377,16 @@ public void testConversionToDouble() { { Conversion conversion = conversionFor(DATE, to); assertNull(conversion.convert(null)); - assertEquals(1.233792E11, (double) conversion.convert(DateUtils.asDateOnly(123456789101L)), 0); - assertEquals(-1.234656E11, (double) conversion.convert(DateUtils.asDateOnly(-123456789101L)), 0); + assertEquals(1.233792E11, (double) conversion.convert(asDateOnly(123456789101L)), 0); + assertEquals(-1.234656E11, (double) conversion.convert(asDateOnly(-123456789101L)), 0); + } + { + Conversion conversion = conversionFor(TIME, to); + assertNull(conversion.convert(null)); + assertEquals(123456.0, (double) conversion.convert(asTimeOnly(123456L)), 0); + assertEquals(7.7589101E7, (double) conversion.convert(asTimeOnly(123456789101L)), 0); + assertEquals(8.6276544E7, (double) conversion.convert(asTimeOnly(-123456L)), 0); + assertEquals(8810899.0, (double) conversion.convert(asTimeOnly(-123456789101L)), 0); } { Conversion conversion = conversionFor(DATETIME, to); @@ -294,9 +438,16 @@ public void testConversionToBoolean() { { Conversion conversion = conversionFor(DATE, to); assertNull(conversion.convert(null)); - assertEquals(true, conversion.convert(DateUtils.asDateOnly(123456789101L))); - assertEquals(true, conversion.convert(DateUtils.asDateOnly(-123456789101L))); - assertEquals(false, conversion.convert(DateUtils.asDateOnly(0L))); + assertEquals(true, conversion.convert(asDateOnly(123456789101L))); + assertEquals(true, conversion.convert(asDateOnly(-123456789101L))); + assertEquals(false, conversion.convert(asDateOnly(0L))); + } + { + Conversion conversion = conversionFor(TIME, to); + assertNull(conversion.convert(null)); + assertEquals(true, conversion.convert(asTimeOnly(123456789101L))); + assertEquals(true, conversion.convert(asTimeOnly(-123456789101L))); + assertEquals(false, conversion.convert(asTimeOnly(0L))); } { Conversion conversion = conversionFor(DATETIME, to); @@ -343,20 +494,29 @@ public void testConversionToInt() { { Conversion conversion = conversionFor(DATE, to); assertNull(conversion.convert(null)); - assertEquals(0, conversion.convert(DateUtils.asDateOnly(12345678L))); - assertEquals(86400000, conversion.convert(DateUtils.asDateOnly(123456789L))); - assertEquals(172800000, conversion.convert(DateUtils.asDateOnly(223456789L))); - assertEquals(-172800000, conversion.convert(DateUtils.asDateOnly(-123456789L))); - Exception e = expectThrows(SqlIllegalArgumentException.class, () -> conversion.convert(DateUtils.asDateOnly(Long.MAX_VALUE))); + assertEquals(0, conversion.convert(asDateOnly(12345678L))); + assertEquals(86400000, conversion.convert(asDateOnly(123456789L))); + assertEquals(172800000, conversion.convert(asDateOnly(223456789L))); + assertEquals(-172800000, conversion.convert(asDateOnly(-123456789L))); + Exception e = expectThrows(SqlIllegalArgumentException.class, () -> conversion.convert(asDateOnly(Long.MAX_VALUE))); assertEquals("[9223372036828800000] out of [integer] range", e.getMessage()); } + { + Conversion conversion = conversionFor(TIME, to); + assertNull(conversion.convert(null)); + assertEquals(123456, conversion.convert(asTimeOnly(123456L))); + assertEquals(77589101, conversion.convert(asTimeOnly(123456789101L))); + assertEquals(86276544, conversion.convert(asTimeOnly(-123456L))); + assertEquals(8810899, conversion.convert(asTimeOnly(-123456789101L))); + assertEquals(25975807, conversion.convert(asTimeOnly(Long.MAX_VALUE))); + } { Conversion conversion = conversionFor(DATETIME, to); assertNull(conversion.convert(null)); - assertEquals(12345678, conversion.convert(DateUtils.asDateTime(12345678L))); - assertEquals(223456789, conversion.convert(DateUtils.asDateTime(223456789L))); - assertEquals(-123456789, conversion.convert(DateUtils.asDateTime(-123456789L))); - Exception e = expectThrows(SqlIllegalArgumentException.class, () -> conversion.convert(DateUtils.asDateTime(Long.MAX_VALUE))); + assertEquals(12345678, conversion.convert(asDateTime(12345678L))); + assertEquals(223456789, conversion.convert(asDateTime(223456789L))); + assertEquals(-123456789, conversion.convert(asDateTime(-123456789L))); + Exception e = expectThrows(SqlIllegalArgumentException.class, () -> conversion.convert(asDateTime(Long.MAX_VALUE))); assertEquals("[" + Long.MAX_VALUE + "] out of [integer] range", e.getMessage()); } } @@ -375,17 +535,26 @@ public void testConversionToShort() { { Conversion conversion = conversionFor(DATE, to); assertNull(conversion.convert(null)); - assertEquals((short) 0, conversion.convert(DateUtils.asDateOnly(12345678L))); - Exception e = expectThrows(SqlIllegalArgumentException.class, () -> conversion.convert(DateUtils.asDateOnly(123456789L))); + assertEquals((short) 0, conversion.convert(asDateOnly(12345678L))); + Exception e = expectThrows(SqlIllegalArgumentException.class, () -> conversion.convert(asDateOnly(123456789L))); assertEquals("[86400000] out of [short] range", e.getMessage()); } + { + Conversion conversion = conversionFor(TIME, to); + assertNull(conversion.convert(null)); + assertEquals((short) 12345, conversion.convert(asTimeOnly(12345L))); + Exception e1 = expectThrows(SqlIllegalArgumentException.class, () -> conversion.convert(asTimeOnly(-123456789L))); + assertEquals("[49343211] out of [short] range", e1.getMessage()); + Exception e2 = expectThrows(SqlIllegalArgumentException.class, () -> conversion.convert(asTimeOnly(123456789L))); + assertEquals("[37056789] out of [short] range", e2.getMessage()); + } { Conversion conversion = conversionFor(DATETIME, to); assertNull(conversion.convert(null)); - assertEquals((short) 12345, conversion.convert(DateUtils.asDateTime(12345L))); - assertEquals((short) -12345, conversion.convert(DateUtils.asDateTime(-12345L))); + assertEquals((short) 12345, conversion.convert(asDateTime(12345L))); + assertEquals((short) -12345, conversion.convert(asDateTime(-12345L))); Exception e = expectThrows(SqlIllegalArgumentException.class, - () -> conversion.convert(DateUtils.asDateTime(Integer.MAX_VALUE))); + () -> conversion.convert(asDateTime(Integer.MAX_VALUE))); assertEquals("[" + Integer.MAX_VALUE + "] out of [short] range", e.getMessage()); } } @@ -404,17 +573,26 @@ public void testConversionToByte() { { Conversion conversion = conversionFor(DATE, to); assertNull(conversion.convert(null)); - assertEquals((byte) 0, conversion.convert(DateUtils.asDateOnly(12345678L))); - Exception e = expectThrows(SqlIllegalArgumentException.class, () -> conversion.convert(DateUtils.asDateOnly(123456789L))); + assertEquals((byte) 0, conversion.convert(asDateOnly(12345678L))); + Exception e = expectThrows(SqlIllegalArgumentException.class, () -> conversion.convert(asDateOnly(123456789L))); assertEquals("[86400000] out of [byte] range", e.getMessage()); } + { + Conversion conversion = conversionFor(TIME, to); + assertNull(conversion.convert(null)); + assertEquals((byte) 123, conversion.convert(asTimeOnly(123L))); + Exception e1 = expectThrows(SqlIllegalArgumentException.class, () -> conversion.convert(asTimeOnly(-123L))); + assertEquals("[86399877] out of [byte] range", e1.getMessage()); + Exception e2 = expectThrows(SqlIllegalArgumentException.class, () -> conversion.convert(asTimeOnly(123456789L))); + assertEquals("[37056789] out of [byte] range", e2.getMessage()); + } { Conversion conversion = conversionFor(DATETIME, to); assertNull(conversion.convert(null)); - assertEquals((byte) 123, conversion.convert(DateUtils.asDateTime(123L))); - assertEquals((byte) -123, conversion.convert(DateUtils.asDateTime(-123L))); + assertEquals((byte) 123, conversion.convert(asDateTime(123L))); + assertEquals((byte) -123, conversion.convert(asDateTime(-123L))); Exception e = expectThrows(SqlIllegalArgumentException.class, - () -> conversion.convert(DateUtils.asDateTime(Integer.MAX_VALUE))); + () -> conversion.convert(asDateTime(Integer.MAX_VALUE))); assertEquals("[" + Integer.MAX_VALUE + "] out of [byte] range", e.getMessage()); } } @@ -453,10 +631,16 @@ public void testCommonType() { // dates/datetimes and intervals assertEquals(DATETIME, commonType(DATE, DATETIME)); assertEquals(DATETIME, commonType(DATETIME, DATE)); + assertEquals(DATETIME, commonType(TIME, DATETIME)); + assertEquals(DATETIME, commonType(DATETIME, TIME)); assertEquals(DATETIME, commonType(DATETIME, randomInterval())); assertEquals(DATETIME, commonType(randomInterval(), DATETIME)); + assertEquals(DATETIME, commonType(DATE, TIME)); + assertEquals(DATETIME, commonType(TIME, DATE)); assertEquals(DATE, commonType(DATE, randomInterval())); assertEquals(DATE, commonType(randomInterval(), DATE)); + assertEquals(TIME, commonType(TIME, randomInterval())); + assertEquals(TIME, commonType(randomInterval(), TIME)); assertEquals(INTERVAL_YEAR_TO_MONTH, commonType(INTERVAL_YEAR_TO_MONTH, INTERVAL_MONTH)); assertEquals(INTERVAL_HOUR_TO_SECOND, commonType(INTERVAL_HOUR_TO_MINUTE, INTERVAL_HOUR_TO_SECOND)); @@ -474,7 +658,7 @@ public void testEsDataTypes() { public void testConversionToUnsupported() { Exception e = expectThrows(SqlIllegalArgumentException.class, () -> conversionFor(INTEGER, UNSUPPORTED)); - assertEquals("cannot convert from [INTEGER] to [UNSUPPORTED]", e.getMessage()); + assertEquals("cannot convert from [integer] to [unsupported]", e.getMessage()); } public void testStringToIp() { diff --git a/x-pack/plugin/src/test/resources/rest-api-spec/api/data_frame.get_data_frame_transform_stats.json b/x-pack/plugin/src/test/resources/rest-api-spec/api/data_frame.get_data_frame_transform_stats.json index 6b4529650dd8a..b100c7e0a2471 100644 --- a/x-pack/plugin/src/test/resources/rest-api-spec/api/data_frame.get_data_frame_transform_stats.json +++ b/x-pack/plugin/src/test/resources/rest-api-spec/api/data_frame.get_data_frame_transform_stats.json @@ -11,6 +11,18 @@ "required": false, "description": "The id of the transform for which to get stats. '_all' or '*' implies all transforms" } + }, + "params": { + "from": { + "type": "number", + "required": false, + "description": "skips a number of transform stats, defaults to 0" + }, + "size": { + "type": "number", + "required": false, + "description": "specifies a max number of transform stats to get, defaults to 100" + } } }, "body": null diff --git a/x-pack/plugin/src/test/resources/rest-api-spec/test/data_frame/transforms_stats.yml b/x-pack/plugin/src/test/resources/rest-api-spec/test/data_frame/transforms_stats.yml index ac6aca4f35d4e..88f1e43fd118b 100644 --- a/x-pack/plugin/src/test/resources/rest-api-spec/test/data_frame/transforms_stats.yml +++ b/x-pack/plugin/src/test/resources/rest-api-spec/test/data_frame/transforms_stats.yml @@ -83,29 +83,66 @@ teardown: } } - do: - data_frame.start_data_frame_transform: - transform_id: "airline-transform-stats-dos" + data_frame.put_data_frame_transform: + transform_id: "airline-transform-stats-the-third" + body: > + { + "source": { "index": "airline-data" }, + "dest": { "index": "airline-data-by-airline-stats-the-third" }, + "pivot": { + "group_by": { "airline": {"terms": {"field": "airline"}}}, + "aggs": {"avg_response": {"avg": {"field": "responsetime"}}} + } + } - do: data_frame.get_data_frame_transform_stats: transform_id: "*" - - match: { count: 2 } + - match: { count: 3 } - match: { transforms.0.id: "airline-transform-stats" } - match: { transforms.1.id: "airline-transform-stats-dos" } + - match: { transforms.2.id: "airline-transform-stats-the-third" } - do: data_frame.get_data_frame_transform_stats: transform_id: "_all" - - match: { count: 2 } + - match: { count: 3 } - match: { transforms.0.id: "airline-transform-stats" } - match: { transforms.1.id: "airline-transform-stats-dos" } + - match: { transforms.2.id: "airline-transform-stats-the-third" } - do: - data_frame.stop_data_frame_transform: - transform_id: "airline-transform-stats-dos" + data_frame.get_data_frame_transform_stats: + transform_id: "airline-transform-stats-dos,airline-transform-stats-the*" + - match: { count: 2 } + - match: { transforms.0.id: "airline-transform-stats-dos" } + - match: { transforms.1.id: "airline-transform-stats-the-third" } + + - do: + data_frame.get_data_frame_transform_stats: + transform_id: "_all" + from: 0 + size: 1 + - match: { count: 1 } + - match: { transforms.0.id: "airline-transform-stats" } + + - do: + data_frame.get_data_frame_transform_stats: + transform_id: "_all" + from: 1 + size: 2 + - match: { count: 2 } + - match: { transforms.0.id: "airline-transform-stats-dos" } + - match: { transforms.1.id: "airline-transform-stats-the-third" } + - do: data_frame.delete_data_frame_transform: transform_id: "airline-transform-stats-dos" + - do: + data_frame.delete_data_frame_transform: + transform_id: "airline-transform-stats-the-third" + + --- "Test get multiple transform stats where one does not have a task": - do: diff --git a/x-pack/plugin/watcher/src/test/java/org/elasticsearch/xpack/watcher/test/integration/WatchMetadataTests.java b/x-pack/plugin/watcher/src/test/java/org/elasticsearch/xpack/watcher/test/integration/WatchMetadataTests.java index 1e2c1ddbc64f1..aff3a62c12cf1 100644 --- a/x-pack/plugin/watcher/src/test/java/org/elasticsearch/xpack/watcher/test/integration/WatchMetadataTests.java +++ b/x-pack/plugin/watcher/src/test/java/org/elasticsearch/xpack/watcher/test/integration/WatchMetadataTests.java @@ -38,6 +38,7 @@ public class WatchMetadataTests extends AbstractWatcherIntegrationTestCase { + @AwaitsFix(bugUrl = "https://github.com/elastic/elasticsearch/issues/40631") public void testWatchMetadata() throws Exception { Map metadata = new HashMap<>(); metadata.put("foo", "bar"); diff --git a/x-pack/qa/rolling-upgrade/src/test/resources/rest-api-spec/test/upgraded_cluster/40_ml_datafeed_crud.yml b/x-pack/qa/rolling-upgrade/src/test/resources/rest-api-spec/test/upgraded_cluster/40_ml_datafeed_crud.yml index 6ca05d86b1c4d..cee6af0df76ad 100644 --- a/x-pack/qa/rolling-upgrade/src/test/resources/rest-api-spec/test/upgraded_cluster/40_ml_datafeed_crud.yml +++ b/x-pack/qa/rolling-upgrade/src/test/resources/rest-api-spec/test/upgraded_cluster/40_ml_datafeed_crud.yml @@ -5,6 +5,9 @@ setup: wait_for_nodes: 3 # wait for long enough that we give delayed unassigned shards to stop being delayed timeout: 70s + +--- +"Test old and mixed cluster datafeeds without aggs": - do: indices.create: index: airline-data @@ -13,8 +16,7 @@ setup: properties: time: type: date ---- -"Test old and mixed cluster datafeeds without aggs": + - do: ml.get_datafeeds: datafeed_id: old-cluster-datafeed-without-aggs @@ -45,8 +47,73 @@ setup: - match: { datafeeds.0.state: "stopped"} - is_false: datafeeds.0.node + - do: + ml.open_job: + job_id: old-cluster-datafeed-job-without-aggs + + - do: + ml.start_datafeed: + datafeed_id: old-cluster-datafeed-without-aggs + start: 0 + + - do: + ml.stop_datafeed: + datafeed_id: old-cluster-datafeed-without-aggs + + - do: + ml.close_job: + job_id: old-cluster-datafeed-job-without-aggs + + - do: + ml.delete_datafeed: + datafeed_id: old-cluster-datafeed-without-aggs + + - do: + ml.delete_job: + job_id: old-cluster-datafeed-job-without-aggs + - match: { acknowledged: true } + + - do: + ml.open_job: + job_id: mixed-cluster-datafeed-job-without-aggs + + - do: + ml.start_datafeed: + datafeed_id: mixed-cluster-datafeed-without-aggs + start: 0 + + - do: + ml.stop_datafeed: + datafeed_id: mixed-cluster-datafeed-without-aggs + + - do: + ml.close_job: + job_id: mixed-cluster-datafeed-job-without-aggs + + - do: + ml.delete_datafeed: + datafeed_id: mixed-cluster-datafeed-without-aggs + + - do: + ml.delete_job: + job_id: mixed-cluster-datafeed-job-without-aggs + - match: { acknowledged: true } + + - do: + indices.delete: + index: airline-data + --- "Test old and mixed cluster datafeeds with aggs": + - do: + indices.create: + index: airline-data + body: + mappings: + properties: + time: + type: date + - do: ml.get_datafeeds: datafeed_id: old-cluster-datafeed-with-aggs @@ -81,52 +148,56 @@ setup: - do: ml.open_job: - job_id: old-cluster-datafeed-job + job_id: old-cluster-datafeed-job-with-aggs - do: ml.start_datafeed: - datafeed_id: old-cluster-datafeed + datafeed_id: old-cluster-datafeed-with-aggs start: 0 - do: ml.stop_datafeed: - datafeed_id: old-cluster-datafeed + datafeed_id: old-cluster-datafeed-with-aggs - do: ml.close_job: - job_id: old-cluster-datafeed-job + job_id: old-cluster-datafeed-job-with-aggs - do: ml.delete_datafeed: - datafeed_id: old-cluster-datafeed + datafeed_id: old-cluster-datafeed-with-aggs - do: ml.delete_job: - job_id: old-cluster-datafeed-job + job_id: old-cluster-datafeed-job-with-aggs - match: { acknowledged: true } - do: ml.open_job: - job_id: mixed-cluster-datafeed-job + job_id: mixed-cluster-datafeed-job-with-aggs - do: ml.start_datafeed: - datafeed_id: mixed-cluster-datafeed + datafeed_id: mixed-cluster-datafeed-with-aggs start: 0 - do: ml.stop_datafeed: - datafeed_id: mixed-cluster-datafeed + datafeed_id: mixed-cluster-datafeed-with-aggs - do: ml.close_job: - job_id: mixed-cluster-datafeed-job + job_id: mixed-cluster-datafeed-job-with-aggs - do: ml.delete_datafeed: - datafeed_id: mixed-cluster-datafeed + datafeed_id: mixed-cluster-datafeed-with-aggs - do: ml.delete_job: - job_id: mixed-cluster-datafeed-job + job_id: mixed-cluster-datafeed-job-with-aggs - match: { acknowledged: true } + + - do: + indices.delete: + index: airline-data