Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Emulate "fields" option on older versions #77749

Merged
merged 42 commits into from
Sep 21, 2021
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
42 commits
Select commit Hold shift + click to select a range
11392de
WIP: Emulate "fields" on older versions
Jul 21, 2021
813d011
Adding ccs testing
Jul 28, 2021
950f987
Move ccs testing to own qa project
Jul 29, 2021
83fdb97
Prevent potential NPE
Jul 29, 2021
a05f7b6
Fix warnings and mute known failing test
Jul 30, 2021
ac9a8bc
Fix failing tests that disconnect clusters
Jul 30, 2021
f9ee226
adding cases to FieldsOptionEmulationIT
Jul 30, 2021
7631238
Use noop adapter instead of nulling instances
Jul 30, 2021
8859c21
small cleanups
Jul 30, 2021
74fd52e
Add request flag to only enable bwc behaviour explicitely
Aug 2, 2021
6c648ea
iter
Aug 3, 2021
8f8f339
fix tests
Aug 3, 2021
2a4a823
checkstyle
Aug 3, 2021
d0c04e3
Merge branch '7.x' into ccs-fields-bridge
Aug 3, 2021
83811fe
Add unit testing, pull out FieldsOptionSourceAdapter into its own class
Aug 4, 2021
6560e40
Merge branch '7.x' into ccs-fields-bridge
Aug 5, 2021
00c14c8
Adding testing around querying into ojects
Aug 5, 2021
b331460
Merge branch '7.x' into ccs-fields-bridge
Aug 30, 2021
c7d90cf
Update request serialization logic
Aug 30, 2021
3e5ff12
WIP refactoring test infra
Sep 3, 2021
4768d41
Merge branch '7.x' into ccs-fields-bridge
elasticmachine Sep 6, 2021
55343f0
Add test for multivalued and object fields
Sep 6, 2021
06bf196
Merge branch '7.x' into ccs-fields-bridge
Sep 8, 2021
7faf8fc
rename method arg and remove 'enable' option from rest spec
Sep 8, 2021
c65b4e5
Add test for multi-field behaviour
Sep 8, 2021
ad061b5
Merge branch '7.x' into ccs-fields-bridge
Sep 9, 2021
8915654
Remove need for wrapping query result
Sep 9, 2021
3b82d1d
Adapter only needs connection version
Sep 9, 2021
f9df7dd
Rework and simplify adapter class
Sep 10, 2021
61ee322
Fix FetchSearchPhaseTests due to insufficient mocking
Sep 10, 2021
a5f419e
Fix tests
Sep 10, 2021
29f56dc
Don't support ccs remote reduce for fields Option emulation
Sep 13, 2021
c8fcdec
iter
Sep 13, 2021
1279272
Adress review comments
Sep 13, 2021
fda2fec
checkstyle
Sep 13, 2021
9694309
fix test
Sep 13, 2021
1f2e256
Merge branch '7.x' into ccs-fields-bridge
Sep 14, 2021
ac34a44
tidying up
Sep 14, 2021
cf32aa1
remove field emulation flag serialization
Sep 14, 2021
51624f8
iter
Sep 15, 2021
35ba198
Merge branch '7.x' into ccs-fields-emulation-ctd
Sep 20, 2021
961a85a
add vaidation error if emulation is enabled for dfs_query_then_fetch
Sep 20, 2021
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
61 changes: 61 additions & 0 deletions qa/ccs-old-version-remote-cluster/build.gradle
Original file line number Diff line number Diff line change
@@ -0,0 +1,61 @@
/*
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
* or more contributor license agreements. Licensed under the Elastic License
* 2.0 and the Server Side Public License, v 1; you may not use this file except
* in compliance with, at your election, the Elastic License 2.0 or the Server
* Side Public License, v 1.
*/

import org.elasticsearch.gradle.Version
import org.elasticsearch.gradle.internal.info.BuildParams
import org.elasticsearch.gradle.testclusters.StandaloneRestIntegTestTask

apply plugin: 'elasticsearch.internal-testclusters'
apply plugin: 'elasticsearch.standalone-rest-test'
apply plugin: 'elasticsearch.bwc-test'
apply plugin: 'elasticsearch.rest-resources'

dependencies {
testImplementation project(':client:rest-high-level')
}

for (Version bwcVersion : BuildParams.bwcVersions.wireCompatible) {
String baseName = "v${bwcVersion}"
String bwcVersionStr = "${bwcVersion}"

testClusters {
"${baseName}-local" {
numberOfNodes = 2
versions = [project.version]
setting 'cluster.remote.node.attr', 'gateway'
setting 'xpack.security.enabled', 'false'
}
"${baseName}-remote" {
numberOfNodes = 2
versions = [bwcVersionStr]
firstNode.setting 'node.attr.gateway', 'true'
lastNode.setting 'node.attr.gateway', 'true'
setting 'xpack.security.enabled', 'false'
}
}

tasks.withType(StandaloneRestIntegTestTask).matching { it.name.startsWith("${baseName}#") }.configureEach {
useCluster testClusters."${baseName}-local"
useCluster testClusters."${baseName}-remote"
systemProperty 'tests.upgrade_from_version', bwcVersionStr.replace('-SNAPSHOT', '')

doFirst {
nonInputProperties.systemProperty('tests.rest.cluster', "${-> testClusters."${baseName}-local".allHttpSocketURI.join(",")}")
nonInputProperties.systemProperty('tests.rest.remote_cluster', "${-> testClusters."${baseName}-remote".allHttpSocketURI.join(",")}")
}
}

tasks.register("${baseName}#testBWCEmulation", StandaloneRestIntegTestTask) {
dependsOn "processTestResources"
mustRunAfter("precommit")
}

tasks.register(bwcTaskName(bwcVersion)) {
dependsOn tasks.named("${baseName}#testBWCEmulation")
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,181 @@
/*
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
* or more contributor license agreements. Licensed under the Elastic License
* 2.0 and the Server Side Public License, v 1; you may not use this file except
* in compliance with, at your election, the Elastic License 2.0 or the Server
* Side Public License, v 1.
*/

/*
* Licensed to Elasticsearch under one or more contributor
* license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright
* ownership. Elasticsearch licenses this file to you under
* the Apache License, Version 2.0 (the "License"); you may
* not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*/

package org.elasticsearch.upgrades;

import org.apache.http.HttpHost;
import org.apache.logging.log4j.LogManager;
import org.apache.logging.log4j.Logger;
import org.elasticsearch.Version;
import org.elasticsearch.action.admin.indices.delete.DeleteIndexRequest;
import org.elasticsearch.action.admin.indices.refresh.RefreshRequest;
import org.elasticsearch.action.search.SearchResponse;
import org.elasticsearch.client.Request;
import org.elasticsearch.client.RequestOptions;
import org.elasticsearch.client.Response;
import org.elasticsearch.client.ResponseException;
import org.elasticsearch.client.RestClient;
import org.elasticsearch.client.RestHighLevelClient;
import org.elasticsearch.client.indices.CreateIndexRequest;
import org.elasticsearch.cluster.metadata.IndexMetadata;
import org.elasticsearch.common.document.DocumentField;
import org.elasticsearch.common.settings.Settings;
import org.elasticsearch.common.settings.Settings.Builder;
import org.elasticsearch.common.xcontent.DeprecationHandler;
import org.elasticsearch.common.xcontent.NamedXContentRegistry;
import org.elasticsearch.common.xcontent.XContentParser;
import org.elasticsearch.common.xcontent.json.JsonXContent;
import org.elasticsearch.rest.action.document.RestIndexAction;
import org.elasticsearch.search.SearchHit;
import org.elasticsearch.test.hamcrest.ElasticsearchAssertions;
import org.elasticsearch.test.rest.AbstractCCSRestTestCase;

import java.io.IOException;
import java.util.List;
import java.util.Map;

import static org.hamcrest.Matchers.containsString;
import static org.hamcrest.Matchers.equalTo;
import static org.hamcrest.Matchers.hasSize;

/**
* This test ensure that we emulate the "fields" option when the local cluster supports it but the remote
* cluster is running an older compatible version.
*/
public class CCSFieldsOptionEmulationIT extends AbstractCCSRestTestCase {

private static final Logger LOGGER = LogManager.getLogger(CCSFieldsOptionEmulationIT.class);
private static final Version UPGRADE_FROM_VERSION = Version.fromString(System.getProperty("tests.upgrade_from_version"));
private static final String CLUSTER_ALIAS = "remote_cluster";

static int indexDocs(RestHighLevelClient client, String index, int numDocs, boolean expectWarnings) throws IOException {
for (int i = 0; i < numDocs; i++) {
Request indexDoc = new Request("PUT", index + "/type/" + i);
indexDoc.setJsonEntity("{\"field\": \"f" + i + "\", \"array\": [1, 2, 3] , \"obj\": { \"innerObj\" : \"foo\" } }");
if (expectWarnings) {
indexDoc.setOptions(expectWarnings(RestIndexAction.TYPES_DEPRECATION_MESSAGE));
}
client.getLowLevelClient().performRequest(indexDoc);
}
client.indices().refresh(new RefreshRequest(index), RequestOptions.DEFAULT);
return numDocs;
}

static RestHighLevelClient newLocalClient(Logger logger) {
final List<HttpHost> hosts = parseHosts("tests.rest.cluster");
final int index = random().nextInt(hosts.size());
logger.info("Using client node {}", index);
return new RestHighLevelClient(RestClient.builder(hosts.get(index)));
}

static RestHighLevelClient newRemoteClient() {
return new RestHighLevelClient(RestClient.builder(randomFrom(parseHosts("tests.rest.remote_cluster"))));
}

public void testFieldsOptionEmulation() throws Exception {
String localIndex = "test_bwc_fields_index";
String remoteIndex = "test_bwc_fields_remote_index";
try (RestHighLevelClient localClient = newLocalClient(LOGGER);
RestHighLevelClient remoteClient = newRemoteClient()) {
localClient.indices().create(new CreateIndexRequest(localIndex)
.settings(Settings.builder().put(IndexMetadata.SETTING_NUMBER_OF_SHARDS, between(1, 5))),
RequestOptions.DEFAULT);
int localNumDocs = indexDocs(localClient, localIndex, between(10, 20), true);

Builder remoteIndexSettings = Settings.builder().put(IndexMetadata.SETTING_NUMBER_OF_SHARDS, between(1, 5));
remoteClient.indices().create(new CreateIndexRequest(remoteIndex)
.settings(remoteIndexSettings),
RequestOptions.DEFAULT);
boolean expectRemoteIndexWarnings = UPGRADE_FROM_VERSION.onOrAfter(Version.V_7_0_0);
int remoteNumDocs = indexDocs(remoteClient, remoteIndex, between(10, 20), expectRemoteIndexWarnings);
int expectedHitCount = localNumDocs + remoteNumDocs;

List<Node> remoteNodes = getNodes(remoteClient.getLowLevelClient());
assertThat(remoteNodes, hasSize(2));
configureRemoteClusters(getNodes(remoteClient.getLowLevelClient()), CLUSTER_ALIAS, UPGRADE_FROM_VERSION, LOGGER);
RestClient lowLevelClient = localClient.getLowLevelClient();
for (String minimizeRoundTrips : new String[] { "true", "false" }) {
Request request = new Request("POST", "/_search");
request.addParameter("index", localIndex + "," + CLUSTER_ALIAS + ":" + remoteIndex);
request.addParameter("ccs_minimize_roundtrips", minimizeRoundTrips);
request.addParameter("enable_fields_emulation", "true");
request.setJsonEntity("{\"_source\": false, \"fields\": [\"*\"] , \"size\": " + expectedHitCount + "}");
Response response = lowLevelClient.performRequest(request);
try (
XContentParser parser = JsonXContent.jsonXContent.createParser(
NamedXContentRegistry.EMPTY,
DeprecationHandler.THROW_UNSUPPORTED_OPERATION,
response.getEntity().getContent()
)
) {
SearchResponse searchResponse = SearchResponse.fromXContent(parser);
ElasticsearchAssertions.assertNoFailures(searchResponse);
ElasticsearchAssertions.assertHitCount(searchResponse, expectedHitCount);
SearchHit[] hits = searchResponse.getHits().getHits();
for (SearchHit hit : hits) {
assertFalse("No source in hit expected but was: " + hit.toString(), hit.hasSource());
Map<String, DocumentField> fields = hit.getFields();
assertNotNull(fields);
assertNotNull("Field `field` not found, hit was: " + hit.toString(), fields.get("field"));
if (hit.getIndex().equals(localIndex) || UPGRADE_FROM_VERSION.onOrAfter(Version.V_7_10_0)) {
assertNotNull("Field `field.keyword` not found, hit was: " + hit.toString(), fields.get("field.keyword"));
} else {
// we won't be able to get multi-field for remote indices below V7.10
assertNull(
"Field `field.keyword` should not be returned, hit was: " + hit.toString(),
fields.get("field.keyword")
);
}
DocumentField arrayField = fields.get("array");
assertNotNull("Field `array` not found, hit was: " + hit.toString(), arrayField);
assertEquals(3, ((List<?>) arrayField.getValues()).size());
assertNull("Object fields should be flattened by the fields API", fields.get("obj"));
assertEquals(1, fields.get("obj.innerObj").getValues().size());
assertEquals("foo", fields.get("obj.innerObj").getValue());
}
}
}

// also check validation of request
Request request = new Request("POST", "/_search");
request.addParameter("index", localIndex + "," + CLUSTER_ALIAS + ":" + remoteIndex);
request.addParameter("enable_fields_emulation", "true");
request.addParameter("search_type", "dfs_query_then_fetch");
request.setJsonEntity("{\"_source\": false, \"fields\": [\"*\"] , \"size\": " + expectedHitCount + "}");
final ResponseException ex = expectThrows(ResponseException.class, () -> lowLevelClient.performRequest(request));
assertThat(ex.getResponse().getStatusLine().getStatusCode(), equalTo(400));
assertThat(ex.getMessage(), containsString("Validation Failed"));
assertThat(
ex.getMessage(),
containsString("[enable_fields_emulation] cannot be used with [dfs_query_then_fetch] search type.")
);

localClient.indices().delete(new DeleteIndexRequest(localIndex), RequestOptions.DEFAULT);
remoteClient.indices().delete(new DeleteIndexRequest(remoteIndex), RequestOptions.DEFAULT);
}
}
}
Loading