Fix AnalyzeAction response serialization (#44284)
Currently we loose information about whether a token list in an AnalyzeAction response is null or an empty list, because we write a 0 value to the stream in both cases and deserialize to a null value on the receiving side. This change fixes this so we write an additional flag indicating whether the value is null or not, followed by the size of the list and its content. Closes #44078
This commit is contained in:
parent
b40b6dd542
commit
835b7a120d
|
@ -22,13 +22,10 @@ package org.elasticsearch.client.indices;
|
||||||
import org.elasticsearch.action.admin.indices.analyze.AnalyzeAction;
|
import org.elasticsearch.action.admin.indices.analyze.AnalyzeAction;
|
||||||
import org.elasticsearch.client.AbstractResponseTestCase;
|
import org.elasticsearch.client.AbstractResponseTestCase;
|
||||||
import org.elasticsearch.common.xcontent.XContentParser;
|
import org.elasticsearch.common.xcontent.XContentParser;
|
||||||
|
import org.elasticsearch.test.RandomObjects;
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.util.ArrayList;
|
|
||||||
import java.util.Arrays;
|
import java.util.Arrays;
|
||||||
import java.util.HashMap;
|
|
||||||
import java.util.List;
|
|
||||||
import java.util.Map;
|
|
||||||
|
|
||||||
public class AnalyzeResponseTests extends AbstractResponseTestCase<AnalyzeAction.Response, AnalyzeResponse> {
|
public class AnalyzeResponseTests extends AbstractResponseTestCase<AnalyzeAction.Response, AnalyzeResponse> {
|
||||||
|
|
||||||
|
@ -37,7 +34,7 @@ public class AnalyzeResponseTests extends AbstractResponseTestCase<AnalyzeAction
|
||||||
int tokenCount = randomIntBetween(1, 30);
|
int tokenCount = randomIntBetween(1, 30);
|
||||||
AnalyzeAction.AnalyzeToken[] tokens = new AnalyzeAction.AnalyzeToken[tokenCount];
|
AnalyzeAction.AnalyzeToken[] tokens = new AnalyzeAction.AnalyzeToken[tokenCount];
|
||||||
for (int i = 0; i < tokenCount; i++) {
|
for (int i = 0; i < tokenCount; i++) {
|
||||||
tokens[i] = randomToken();
|
tokens[i] = RandomObjects.randomToken(random());
|
||||||
}
|
}
|
||||||
if (randomBoolean()) {
|
if (randomBoolean()) {
|
||||||
AnalyzeAction.CharFilteredText[] charfilters = null;
|
AnalyzeAction.CharFilteredText[] charfilters = null;
|
||||||
|
@ -62,45 +59,6 @@ public class AnalyzeResponseTests extends AbstractResponseTestCase<AnalyzeAction
|
||||||
return new AnalyzeAction.Response(Arrays.asList(tokens), null);
|
return new AnalyzeAction.Response(Arrays.asList(tokens), null);
|
||||||
}
|
}
|
||||||
|
|
||||||
private AnalyzeAction.AnalyzeToken randomToken() {
|
|
||||||
String token = randomAlphaOfLengthBetween(1, 20);
|
|
||||||
int position = randomIntBetween(0, 1000);
|
|
||||||
int startOffset = randomIntBetween(0, 1000);
|
|
||||||
int endOffset = randomIntBetween(0, 1000);
|
|
||||||
int posLength = randomIntBetween(1, 5);
|
|
||||||
String type = randomAlphaOfLengthBetween(1, 20);
|
|
||||||
Map<String, Object> extras = new HashMap<>();
|
|
||||||
if (randomBoolean()) {
|
|
||||||
int entryCount = randomInt(6);
|
|
||||||
for (int i = 0; i < entryCount; i++) {
|
|
||||||
switch (randomInt(6)) {
|
|
||||||
case 0:
|
|
||||||
case 1:
|
|
||||||
case 2:
|
|
||||||
case 3:
|
|
||||||
String key = randomAlphaOfLength(5);
|
|
||||||
String value = randomAlphaOfLength(10);
|
|
||||||
extras.put(key, value);
|
|
||||||
break;
|
|
||||||
case 4:
|
|
||||||
String objkey = randomAlphaOfLength(5);
|
|
||||||
Map<String, String> obj = new HashMap<>();
|
|
||||||
obj.put(randomAlphaOfLength(5), randomAlphaOfLength(10));
|
|
||||||
extras.put(objkey, obj);
|
|
||||||
break;
|
|
||||||
case 5:
|
|
||||||
String listkey = randomAlphaOfLength(5);
|
|
||||||
List<String> list = new ArrayList<>();
|
|
||||||
list.add(randomAlphaOfLength(4));
|
|
||||||
list.add(randomAlphaOfLength(6));
|
|
||||||
extras.put(listkey, list);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return new AnalyzeAction.AnalyzeToken(token, position, startOffset, endOffset, posLength, type, extras);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected AnalyzeResponse doParseToClientInstance(XContentParser parser) throws IOException {
|
protected AnalyzeResponse doParseToClientInstance(XContentParser parser) throws IOException {
|
||||||
return AnalyzeResponse.fromXContent(parser);
|
return AnalyzeResponse.fromXContent(parser);
|
||||||
|
|
|
@ -19,9 +19,10 @@
|
||||||
|
|
||||||
package org.elasticsearch.action.admin.indices.analyze;
|
package org.elasticsearch.action.admin.indices.analyze;
|
||||||
|
|
||||||
import org.elasticsearch.action.ActionType;
|
import org.elasticsearch.Version;
|
||||||
import org.elasticsearch.action.ActionRequestValidationException;
|
import org.elasticsearch.action.ActionRequestValidationException;
|
||||||
import org.elasticsearch.action.ActionResponse;
|
import org.elasticsearch.action.ActionResponse;
|
||||||
|
import org.elasticsearch.action.ActionType;
|
||||||
import org.elasticsearch.action.support.single.shard.SingleShardRequest;
|
import org.elasticsearch.action.support.single.shard.SingleShardRequest;
|
||||||
import org.elasticsearch.common.ParseField;
|
import org.elasticsearch.common.ParseField;
|
||||||
import org.elasticsearch.common.Strings;
|
import org.elasticsearch.common.Strings;
|
||||||
|
@ -292,22 +293,29 @@ public class AnalyzeAction extends ActionType<AnalyzeAction.Response> {
|
||||||
private final List<AnalyzeToken> tokens;
|
private final List<AnalyzeToken> tokens;
|
||||||
|
|
||||||
public Response(List<AnalyzeToken> tokens, DetailAnalyzeResponse detail) {
|
public Response(List<AnalyzeToken> tokens, DetailAnalyzeResponse detail) {
|
||||||
|
if (tokens == null && detail == null) {
|
||||||
|
throw new IllegalArgumentException("Neither token nor detail set on AnalysisAction.Response");
|
||||||
|
}
|
||||||
this.tokens = tokens;
|
this.tokens = tokens;
|
||||||
this.detail = detail;
|
this.detail = detail;
|
||||||
}
|
}
|
||||||
|
|
||||||
public Response(StreamInput in) throws IOException {
|
public Response(StreamInput in) throws IOException {
|
||||||
super.readFrom(in);
|
super.readFrom(in);
|
||||||
|
if (in.getVersion().onOrAfter(Version.V_7_3_0)) {
|
||||||
|
AnalyzeToken[] tokenArray = in.readOptionalArray(AnalyzeToken::new, AnalyzeToken[]::new);
|
||||||
|
tokens = tokenArray != null ? Arrays.asList(tokenArray) : null;
|
||||||
|
} else {
|
||||||
int size = in.readVInt();
|
int size = in.readVInt();
|
||||||
if (size > 0) {
|
if (size > 0) {
|
||||||
tokens = new ArrayList<>(size);
|
tokens = new ArrayList<>(size);
|
||||||
for (int i = 0; i < size; i++) {
|
for (int i = 0; i < size; i++) {
|
||||||
tokens.add(new AnalyzeToken(in));
|
tokens.add(new AnalyzeToken(in));
|
||||||
}
|
}
|
||||||
}
|
} else {
|
||||||
else {
|
|
||||||
tokens = null;
|
tokens = null;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
detail = in.readOptionalWriteable(DetailAnalyzeResponse::new);
|
detail = in.readOptionalWriteable(DetailAnalyzeResponse::new);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -346,6 +354,13 @@ public class AnalyzeAction extends ActionType<AnalyzeAction.Response> {
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void writeTo(StreamOutput out) throws IOException {
|
public void writeTo(StreamOutput out) throws IOException {
|
||||||
|
if (out.getVersion().onOrAfter(Version.V_7_3_0)) {
|
||||||
|
AnalyzeToken[] tokenArray = null;
|
||||||
|
if (tokens != null) {
|
||||||
|
tokenArray = tokens.toArray(new AnalyzeToken[0]);
|
||||||
|
}
|
||||||
|
out.writeOptionalArray(tokenArray);
|
||||||
|
} else {
|
||||||
if (tokens != null) {
|
if (tokens != null) {
|
||||||
out.writeVInt(tokens.size());
|
out.writeVInt(tokens.size());
|
||||||
for (AnalyzeToken token : tokens) {
|
for (AnalyzeToken token : tokens) {
|
||||||
|
@ -354,13 +369,18 @@ public class AnalyzeAction extends ActionType<AnalyzeAction.Response> {
|
||||||
} else {
|
} else {
|
||||||
out.writeVInt(0);
|
out.writeVInt(0);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
out.writeOptionalWriteable(detail);
|
out.writeOptionalWriteable(detail);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean equals(Object o) {
|
public boolean equals(Object o) {
|
||||||
if (this == o) return true;
|
if (this == o) {
|
||||||
if (o == null || getClass() != o.getClass()) return false;
|
return true;
|
||||||
|
}
|
||||||
|
if (o == null || getClass() != o.getClass()) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
Response that = (Response) o;
|
Response that = (Response) o;
|
||||||
return Objects.equals(detail, that.detail) &&
|
return Objects.equals(detail, that.detail) &&
|
||||||
Objects.equals(tokens, that.tokens);
|
Objects.equals(tokens, that.tokens);
|
||||||
|
@ -401,8 +421,12 @@ public class AnalyzeAction extends ActionType<AnalyzeAction.Response> {
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean equals(Object o) {
|
public boolean equals(Object o) {
|
||||||
if (this == o) return true;
|
if (this == o) {
|
||||||
if (o == null || getClass() != o.getClass()) return false;
|
return true;
|
||||||
|
}
|
||||||
|
if (o == null || getClass() != o.getClass()) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
AnalyzeToken that = (AnalyzeToken) o;
|
AnalyzeToken that = (AnalyzeToken) o;
|
||||||
return startOffset == that.startOffset &&
|
return startOffset == that.startOffset &&
|
||||||
endOffset == that.endOffset &&
|
endOffset == that.endOffset &&
|
||||||
|
@ -582,8 +606,12 @@ public class AnalyzeAction extends ActionType<AnalyzeAction.Response> {
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean equals(Object o) {
|
public boolean equals(Object o) {
|
||||||
if (this == o) return true;
|
if (this == o) {
|
||||||
if (o == null || getClass() != o.getClass()) return false;
|
return true;
|
||||||
|
}
|
||||||
|
if (o == null || getClass() != o.getClass()) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
DetailAnalyzeResponse that = (DetailAnalyzeResponse) o;
|
DetailAnalyzeResponse that = (DetailAnalyzeResponse) o;
|
||||||
return customAnalyzer == that.customAnalyzer &&
|
return customAnalyzer == that.customAnalyzer &&
|
||||||
Objects.equals(analyzer, that.analyzer) &&
|
Objects.equals(analyzer, that.analyzer) &&
|
||||||
|
@ -669,8 +697,12 @@ public class AnalyzeAction extends ActionType<AnalyzeAction.Response> {
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean equals(Object o) {
|
public boolean equals(Object o) {
|
||||||
if (this == o) return true;
|
if (this == o) {
|
||||||
if (o == null || getClass() != o.getClass()) return false;
|
return true;
|
||||||
|
}
|
||||||
|
if (o == null || getClass() != o.getClass()) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
AnalyzeTokenList that = (AnalyzeTokenList) o;
|
AnalyzeTokenList that = (AnalyzeTokenList) o;
|
||||||
return Objects.equals(name, that.name) &&
|
return Objects.equals(name, that.name) &&
|
||||||
Arrays.equals(tokens, that.tokens);
|
Arrays.equals(tokens, that.tokens);
|
||||||
|
@ -690,17 +722,20 @@ public class AnalyzeAction extends ActionType<AnalyzeAction.Response> {
|
||||||
|
|
||||||
AnalyzeTokenList(StreamInput in) throws IOException {
|
AnalyzeTokenList(StreamInput in) throws IOException {
|
||||||
name = in.readString();
|
name = in.readString();
|
||||||
|
if (in.getVersion().onOrAfter(Version.V_7_3_0)) {
|
||||||
|
tokens = in.readOptionalArray(AnalyzeToken::new, AnalyzeToken[]::new);
|
||||||
|
} else {
|
||||||
int size = in.readVInt();
|
int size = in.readVInt();
|
||||||
if (size > 0) {
|
if (size > 0) {
|
||||||
tokens = new AnalyzeToken[size];
|
tokens = new AnalyzeToken[size];
|
||||||
for (int i = 0; i < size; i++) {
|
for (int i = 0; i < size; i++) {
|
||||||
tokens[i] = new AnalyzeToken(in);
|
tokens[i] = new AnalyzeToken(in);
|
||||||
}
|
}
|
||||||
}
|
} else {
|
||||||
else {
|
|
||||||
tokens = null;
|
tokens = null;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
public String getName() {
|
public String getName() {
|
||||||
return name;
|
return name;
|
||||||
|
@ -732,6 +767,9 @@ public class AnalyzeAction extends ActionType<AnalyzeAction.Response> {
|
||||||
@Override
|
@Override
|
||||||
public void writeTo(StreamOutput out) throws IOException {
|
public void writeTo(StreamOutput out) throws IOException {
|
||||||
out.writeString(name);
|
out.writeString(name);
|
||||||
|
if (out.getVersion().onOrAfter(Version.V_7_3_0)) {
|
||||||
|
out.writeOptionalArray(tokens);
|
||||||
|
} else {
|
||||||
if (tokens != null) {
|
if (tokens != null) {
|
||||||
out.writeVInt(tokens.length);
|
out.writeVInt(tokens.length);
|
||||||
for (AnalyzeToken token : tokens) {
|
for (AnalyzeToken token : tokens) {
|
||||||
|
@ -742,6 +780,7 @@ public class AnalyzeAction extends ActionType<AnalyzeAction.Response> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
public static class CharFilteredText implements Writeable, ToXContentObject {
|
public static class CharFilteredText implements Writeable, ToXContentObject {
|
||||||
private final String name;
|
private final String name;
|
||||||
|
@ -789,8 +828,12 @@ public class AnalyzeAction extends ActionType<AnalyzeAction.Response> {
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean equals(Object o) {
|
public boolean equals(Object o) {
|
||||||
if (this == o) return true;
|
if (this == o) {
|
||||||
if (o == null || getClass() != o.getClass()) return false;
|
return true;
|
||||||
|
}
|
||||||
|
if (o == null || getClass() != o.getClass()) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
CharFilteredText that = (CharFilteredText) o;
|
CharFilteredText that = (CharFilteredText) o;
|
||||||
return Objects.equals(name, that.name) &&
|
return Objects.equals(name, that.name) &&
|
||||||
Arrays.equals(texts, that.texts);
|
Arrays.equals(texts, that.texts);
|
||||||
|
|
|
@ -19,20 +19,25 @@
|
||||||
|
|
||||||
package org.elasticsearch.action.admin.indices.analyze;
|
package org.elasticsearch.action.admin.indices.analyze;
|
||||||
|
|
||||||
|
import org.elasticsearch.action.admin.indices.analyze.AnalyzeAction.AnalyzeToken;
|
||||||
import org.elasticsearch.common.bytes.BytesReference;
|
import org.elasticsearch.common.bytes.BytesReference;
|
||||||
|
import org.elasticsearch.common.io.stream.Writeable.Reader;
|
||||||
import org.elasticsearch.common.xcontent.ToXContent;
|
import org.elasticsearch.common.xcontent.ToXContent;
|
||||||
import org.elasticsearch.common.xcontent.XContentBuilder;
|
import org.elasticsearch.common.xcontent.XContentBuilder;
|
||||||
import org.elasticsearch.common.xcontent.XContentHelper;
|
import org.elasticsearch.common.xcontent.XContentHelper;
|
||||||
import org.elasticsearch.common.xcontent.json.JsonXContent;
|
import org.elasticsearch.common.xcontent.json.JsonXContent;
|
||||||
import org.elasticsearch.test.ESTestCase;
|
import org.elasticsearch.test.AbstractWireSerializingTestCase;
|
||||||
|
import org.elasticsearch.test.RandomObjects;
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.Arrays;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
|
|
||||||
import static org.hamcrest.Matchers.equalTo;
|
import static org.hamcrest.Matchers.equalTo;
|
||||||
|
|
||||||
public class AnalyzeResponseTests extends ESTestCase {
|
public class AnalyzeResponseTests extends AbstractWireSerializingTestCase<AnalyzeAction.Response> {
|
||||||
|
|
||||||
@SuppressWarnings("unchecked")
|
@SuppressWarnings("unchecked")
|
||||||
public void testNullResponseToXContent() throws IOException {
|
public void testNullResponseToXContent() throws IOException {
|
||||||
|
@ -59,6 +64,64 @@ public class AnalyzeResponseTests extends ESTestCase {
|
||||||
assertThat(nullTokens.size(), equalTo(0));
|
assertThat(nullTokens.size(), equalTo(0));
|
||||||
assertThat(name, equalTo(nameValue));
|
assertThat(name, equalTo(nameValue));
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void testConstructorArgs() {
|
||||||
|
IllegalArgumentException ex = expectThrows(IllegalArgumentException.class, () -> new AnalyzeAction.Response(null, null));
|
||||||
|
assertEquals("Neither token nor detail set on AnalysisAction.Response", ex.getMessage());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected AnalyzeAction.Response createTestInstance() {
|
||||||
|
int tokenCount = randomIntBetween(0, 30);
|
||||||
|
AnalyzeAction.AnalyzeToken[] tokens = new AnalyzeAction.AnalyzeToken[tokenCount];
|
||||||
|
for (int i = 0; i < tokenCount; i++) {
|
||||||
|
tokens[i] = RandomObjects.randomToken(random());
|
||||||
|
}
|
||||||
|
if (randomBoolean()) {
|
||||||
|
AnalyzeAction.CharFilteredText[] charfilters = null;
|
||||||
|
AnalyzeAction.AnalyzeTokenList[] tokenfilters = null;
|
||||||
|
if (randomBoolean()) {
|
||||||
|
charfilters = new AnalyzeAction.CharFilteredText[]{
|
||||||
|
new AnalyzeAction.CharFilteredText("my_charfilter", new String[]{"one two"})
|
||||||
|
};
|
||||||
|
}
|
||||||
|
if (randomBoolean()) {
|
||||||
|
tokenfilters = new AnalyzeAction.AnalyzeTokenList[]{
|
||||||
|
new AnalyzeAction.AnalyzeTokenList("my_tokenfilter_1", tokens),
|
||||||
|
new AnalyzeAction.AnalyzeTokenList("my_tokenfilter_2", tokens)
|
||||||
|
};
|
||||||
|
}
|
||||||
|
AnalyzeAction.DetailAnalyzeResponse dar = new AnalyzeAction.DetailAnalyzeResponse(
|
||||||
|
charfilters,
|
||||||
|
new AnalyzeAction.AnalyzeTokenList("my_tokenizer", tokens),
|
||||||
|
tokenfilters);
|
||||||
|
return new AnalyzeAction.Response(null, dar);
|
||||||
|
}
|
||||||
|
return new AnalyzeAction.Response(Arrays.asList(tokens), null);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Either add a token to the token list or change the details token list name
|
||||||
|
*/
|
||||||
|
@Override
|
||||||
|
protected AnalyzeAction.Response mutateInstance(AnalyzeAction.Response instance) throws IOException {
|
||||||
|
if (instance.getTokens() != null) {
|
||||||
|
List<AnalyzeToken> extendedList = new ArrayList<>(instance.getTokens());
|
||||||
|
extendedList.add(RandomObjects.randomToken(random()));
|
||||||
|
return new AnalyzeAction.Response(extendedList, null);
|
||||||
|
} else {
|
||||||
|
AnalyzeToken[] tokens = instance.detail().tokenizer().getTokens();
|
||||||
|
return new AnalyzeAction.Response(null, new AnalyzeAction.DetailAnalyzeResponse(
|
||||||
|
instance.detail().charfilters(),
|
||||||
|
new AnalyzeAction.AnalyzeTokenList("my_other_tokenizer", tokens),
|
||||||
|
instance.detail().tokenfilters()));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected Reader<AnalyzeAction.Response> instanceReader() {
|
||||||
|
return AnalyzeAction.Response::new;
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
|
@ -24,6 +24,7 @@ import org.elasticsearch.action.admin.indices.analyze.AnalyzeRequestBuilder;
|
||||||
import org.elasticsearch.common.settings.Settings;
|
import org.elasticsearch.common.settings.Settings;
|
||||||
import org.elasticsearch.plugins.Plugin;
|
import org.elasticsearch.plugins.Plugin;
|
||||||
import org.elasticsearch.test.ESIntegTestCase;
|
import org.elasticsearch.test.ESIntegTestCase;
|
||||||
|
import org.elasticsearch.test.ESIntegTestCase.ClusterScope;
|
||||||
import org.elasticsearch.test.MockKeywordPlugin;
|
import org.elasticsearch.test.MockKeywordPlugin;
|
||||||
import org.hamcrest.core.IsNull;
|
import org.hamcrest.core.IsNull;
|
||||||
|
|
||||||
|
@ -41,6 +42,7 @@ import static org.hamcrest.Matchers.hasSize;
|
||||||
import static org.hamcrest.Matchers.is;
|
import static org.hamcrest.Matchers.is;
|
||||||
import static org.hamcrest.Matchers.startsWith;
|
import static org.hamcrest.Matchers.startsWith;
|
||||||
|
|
||||||
|
@ClusterScope(minNumDataNodes = 2)
|
||||||
public class AnalyzeActionIT extends ESIntegTestCase {
|
public class AnalyzeActionIT extends ESIntegTestCase {
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -387,5 +389,16 @@ public class AnalyzeActionIT extends ESIntegTestCase {
|
||||||
assertThat(token.getPositionLength(), equalTo(1));
|
assertThat(token.getPositionLength(), equalTo(1));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Input text that doesn't produce tokens should return an empty token list
|
||||||
|
*/
|
||||||
|
public void testZeroTokenAnalysis() throws IOException {
|
||||||
|
assertAcked(prepareCreate("test"));
|
||||||
|
ensureGreen("test");
|
||||||
|
|
||||||
|
AnalyzeAction.Response analyzeResponse = client().admin().indices().prepareAnalyze("test", ".").get();
|
||||||
|
assertNotNull(analyzeResponse.getTokens());
|
||||||
|
assertThat(analyzeResponse.getTokens().size(), equalTo(0));
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -19,9 +19,14 @@
|
||||||
|
|
||||||
package org.elasticsearch.test;
|
package org.elasticsearch.test;
|
||||||
|
|
||||||
|
import com.carrotsearch.randomizedtesting.RandomizedTest;
|
||||||
|
import com.carrotsearch.randomizedtesting.generators.RandomNumbers;
|
||||||
import com.carrotsearch.randomizedtesting.generators.RandomPicks;
|
import com.carrotsearch.randomizedtesting.generators.RandomPicks;
|
||||||
import com.carrotsearch.randomizedtesting.generators.RandomStrings;
|
import com.carrotsearch.randomizedtesting.generators.RandomStrings;
|
||||||
|
|
||||||
import org.elasticsearch.ElasticsearchException;
|
import org.elasticsearch.ElasticsearchException;
|
||||||
|
import org.elasticsearch.action.admin.indices.analyze.AnalyzeAction;
|
||||||
|
import org.elasticsearch.action.admin.indices.analyze.AnalyzeAction.AnalyzeToken;
|
||||||
import org.elasticsearch.action.support.replication.ReplicationResponse.ShardInfo;
|
import org.elasticsearch.action.support.replication.ReplicationResponse.ShardInfo;
|
||||||
import org.elasticsearch.action.support.replication.ReplicationResponse.ShardInfo.Failure;
|
import org.elasticsearch.action.support.replication.ReplicationResponse.ShardInfo.Failure;
|
||||||
import org.elasticsearch.cluster.block.ClusterBlockException;
|
import org.elasticsearch.cluster.block.ClusterBlockException;
|
||||||
|
@ -43,7 +48,9 @@ import java.io.IOException;
|
||||||
import java.nio.charset.StandardCharsets;
|
import java.nio.charset.StandardCharsets;
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.Base64;
|
import java.util.Base64;
|
||||||
|
import java.util.HashMap;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
import java.util.Map;
|
||||||
import java.util.Random;
|
import java.util.Random;
|
||||||
|
|
||||||
import static com.carrotsearch.randomizedtesting.generators.RandomNumbers.randomIntBetween;
|
import static com.carrotsearch.randomizedtesting.generators.RandomNumbers.randomIntBetween;
|
||||||
|
@ -341,4 +348,43 @@ public final class RandomObjects {
|
||||||
|
|
||||||
return Tuple.tuple(actual, expected);
|
return Tuple.tuple(actual, expected);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static AnalyzeToken randomToken(Random random) {
|
||||||
|
String token = RandomStrings.randomAsciiLettersOfLengthBetween(random, 1, 20);
|
||||||
|
int position = RandomizedTest.randomIntBetween(0, 1000);
|
||||||
|
int startOffset = RandomizedTest.randomIntBetween(0, 1000);
|
||||||
|
int endOffset = RandomizedTest.randomIntBetween(0, 1000);
|
||||||
|
int posLength = RandomizedTest.randomIntBetween(1, 5);
|
||||||
|
String type = RandomStrings.randomAsciiLettersOfLengthBetween(random, 1, 20);
|
||||||
|
Map<String, Object> extras = new HashMap<>();
|
||||||
|
if (random.nextBoolean()) {
|
||||||
|
int entryCount = RandomNumbers.randomIntBetween(random, 0, 6);
|
||||||
|
for (int i = 0; i < entryCount; i++) {
|
||||||
|
switch (RandomNumbers.randomIntBetween(random, 0, 6)) {
|
||||||
|
case 0:
|
||||||
|
case 1:
|
||||||
|
case 2:
|
||||||
|
case 3:
|
||||||
|
String key = RandomStrings.randomAsciiLettersOfLength(random, 5);
|
||||||
|
String value = RandomStrings.randomAsciiLettersOfLength(random, 10);
|
||||||
|
extras.put(key, value);
|
||||||
|
break;
|
||||||
|
case 4:
|
||||||
|
String objkey = RandomStrings.randomAsciiLettersOfLength(random, 5);
|
||||||
|
Map<String, String> obj = new HashMap<>();
|
||||||
|
obj.put(RandomStrings.randomAsciiLettersOfLength(random, 5), RandomStrings.randomAsciiLettersOfLength(random, 10));
|
||||||
|
extras.put(objkey, obj);
|
||||||
|
break;
|
||||||
|
case 5:
|
||||||
|
String listkey = RandomStrings.randomAsciiLettersOfLength(random, 5);
|
||||||
|
List<String> list = new ArrayList<>();
|
||||||
|
list.add(RandomStrings.randomAsciiLettersOfLength(random, 4));
|
||||||
|
list.add(RandomStrings.randomAsciiLettersOfLength(random, 6));
|
||||||
|
extras.put(listkey, list);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return new AnalyzeAction.AnalyzeToken(token, position, startOffset, endOffset, posLength, type, extras);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue