MoreLikeThis API: Search documents that are "like" the specified document, closes #45.

This commit is contained in:
kimchy 2010-02-27 03:57:36 +02:00
parent c30d790609
commit 8b36281d60
19 changed files with 529 additions and 80 deletions

View File

@ -22,12 +22,18 @@ package org.elasticsearch.action.mlt;
import org.elasticsearch.action.ActionRequest; import org.elasticsearch.action.ActionRequest;
import org.elasticsearch.action.ActionRequestValidationException; import org.elasticsearch.action.ActionRequestValidationException;
import org.elasticsearch.action.Actions; import org.elasticsearch.action.Actions;
import org.elasticsearch.action.search.SearchType;
import org.elasticsearch.search.Scroll;
import org.elasticsearch.search.builder.SearchSourceBuilder;
import org.elasticsearch.util.Bytes;
import org.elasticsearch.util.Strings; import org.elasticsearch.util.Strings;
import java.io.DataInput; import java.io.DataInput;
import java.io.DataOutput; import java.io.DataOutput;
import java.io.IOException; import java.io.IOException;
import static org.elasticsearch.search.Scroll.*;
/** /**
* @author kimchy (shay.banon) * @author kimchy (shay.banon)
*/ */
@ -52,6 +58,14 @@ public class MoreLikeThisRequest implements ActionRequest {
private Boolean boostTerms = null; private Boolean boostTerms = null;
private float boostTermsFactor = -1; private float boostTermsFactor = -1;
private SearchType searchType = SearchType.DEFAULT;
private String searchQueryHint;
private String[] searchIndices;
private String[] searchTypes;
private Scroll searchScroll;
private byte[] searchSource;
private boolean threadedListener = false; private boolean threadedListener = false;
public MoreLikeThisRequest() { public MoreLikeThisRequest() {
@ -87,6 +101,11 @@ public class MoreLikeThisRequest implements ActionRequest {
return this.fields; return this.fields;
} }
public MoreLikeThisRequest fields(String... fields) {
this.fields = fields;
return this;
}
public MoreLikeThisRequest percentTermsToMatch(float percentTermsToMatch) { public MoreLikeThisRequest percentTermsToMatch(float percentTermsToMatch) {
this.percentTermsToMatch = percentTermsToMatch; this.percentTermsToMatch = percentTermsToMatch;
return this; return this;
@ -159,7 +178,7 @@ public class MoreLikeThisRequest implements ActionRequest {
return this.maxWordLen; return this.maxWordLen;
} }
public MoreLikeThisRequest boostTerms(boolean boostTerms) { public MoreLikeThisRequest boostTerms(Boolean boostTerms) {
this.boostTerms = boostTerms; this.boostTerms = boostTerms;
return this; return this;
} }
@ -177,6 +196,75 @@ public class MoreLikeThisRequest implements ActionRequest {
return this.boostTermsFactor; return this.boostTermsFactor;
} }
public MoreLikeThisRequest searchSource(SearchSourceBuilder sourceBuilder) {
return searchSource(sourceBuilder.build());
}
public MoreLikeThisRequest searchSource(byte[] searchSource) {
this.searchSource = searchSource;
return this;
}
public byte[] searchSource() {
return this.searchSource;
}
/**
* Sets the search type of the mlt search query.
*/
public MoreLikeThisRequest searchType(SearchType searchType) {
this.searchType = searchType;
return this;
}
public SearchType searchType() {
return this.searchType;
}
/**
* Sets the indices the resulting mlt query will run against. If not set, will run
* against the index the document was fetched from.
*/
public MoreLikeThisRequest searchIndices(String... searchIndices) {
this.searchIndices = searchIndices;
return this;
}
public String[] searchIndices() {
return this.searchIndices;
}
/**
* Sets the types the resulting mlt query will run against. If not set, will run
* against the type of the document fetched.
*/
public MoreLikeThisRequest searchTypes(String... searchTypes) {
this.searchTypes = searchTypes;
return this;
}
public String[] searchTypes() {
return this.searchTypes;
}
public MoreLikeThisRequest searchQueryHint(String searchQueryHint) {
this.searchQueryHint = searchQueryHint;
return this;
}
public String searchQueryHint() {
return this.searchQueryHint;
}
public MoreLikeThisRequest searchScroll(Scroll searchScroll) {
this.searchScroll = searchScroll;
return this;
}
public Scroll searchScroll() {
return this.searchScroll;
}
@Override public ActionRequestValidationException validate() { @Override public ActionRequestValidationException validate() {
ActionRequestValidationException validationException = null; ActionRequestValidationException validationException = null;
if (index == null) { if (index == null) {
@ -233,6 +321,42 @@ public class MoreLikeThisRequest implements ActionRequest {
boostTerms = in.readBoolean(); boostTerms = in.readBoolean();
} }
boostTermsFactor = in.readFloat(); boostTermsFactor = in.readFloat();
searchType = SearchType.fromId(in.readByte());
if (in.readBoolean()) {
searchQueryHint = in.readUTF();
}
size = in.readInt();
if (size == -1) {
searchIndices = null;
} else if (size == 0) {
searchIndices = Strings.EMPTY_ARRAY;
} else {
searchIndices = new String[size];
for (int i = 0; i < size; i++) {
searchIndices[i] = in.readUTF();
}
}
size = in.readInt();
if (size == -1) {
searchTypes = null;
} else if (size == 0) {
searchTypes = Strings.EMPTY_ARRAY;
} else {
searchTypes = new String[size];
for (int i = 0; i < size; i++) {
searchTypes[i] = in.readUTF();
}
}
if (in.readBoolean()) {
searchScroll = readScroll(in);
}
size = in.readInt();
if (size == 0) {
searchSource = Bytes.EMPTY_ARRAY;
} else {
searchSource = new byte[in.readInt()];
in.readFully(searchSource);
}
} }
@Override public void writeTo(DataOutput out) throws IOException { @Override public void writeTo(DataOutput out) throws IOException {
@ -270,5 +394,41 @@ public class MoreLikeThisRequest implements ActionRequest {
out.writeBoolean(boostTerms); out.writeBoolean(boostTerms);
} }
out.writeFloat(boostTermsFactor); out.writeFloat(boostTermsFactor);
out.writeByte(searchType.id());
if (searchQueryHint == null) {
out.writeBoolean(false);
} else {
out.writeBoolean(true);
out.writeUTF(searchQueryHint);
}
if (searchIndices == null) {
out.writeInt(-1);
} else {
out.writeInt(searchIndices.length);
for (String index : searchIndices) {
out.writeUTF(index);
}
}
if (searchTypes == null) {
out.writeInt(-1);
} else {
out.writeInt(searchTypes.length);
for (String type : searchTypes) {
out.writeUTF(type);
}
}
if (searchScroll == null) {
out.writeBoolean(false);
} else {
out.writeBoolean(true);
searchScroll.writeTo(out);
}
if (searchSource == null) {
out.writeInt(0);
} else {
out.writeInt(searchSource.length);
out.write(searchSource);
}
} }
} }

View File

@ -121,11 +121,24 @@ public class TransportMoreLikeThisAction extends BaseAction<MoreLikeThisRequest,
listener.onFailure(e); listener.onFailure(e);
} }
SearchRequest searchRequest = searchRequest(request.index()).types(request.type()) String[] searchIndices = request.searchIndices();
.listenerThreaded(request.listenerThreaded()) if (searchIndices == null) {
.source(searchSource() searchIndices = new String[]{request.index()};
}
String[] searchTypes = request.searchTypes();
if (searchTypes == null) {
searchTypes = new String[]{request.type()};
}
SearchRequest searchRequest = searchRequest(searchIndices)
.types(searchTypes)
.searchType(request.searchType())
.source(request.searchSource())
.scroll(request.searchScroll())
.extraSource(searchSource()
.query(boolBuilder) .query(boolBuilder)
); )
.listenerThreaded(request.listenerThreaded());
searchAction.execute(searchRequest, new ActionListener<SearchResponse>() { searchAction.execute(searchRequest, new ActionListener<SearchResponse>() {
@Override public void onResponse(SearchResponse response) { @Override public void onResponse(SearchResponse response) {
listener.onResponse(response); listener.onResponse(response);

View File

@ -23,7 +23,7 @@ import org.elasticsearch.action.ActionRequest;
import org.elasticsearch.action.ActionRequestValidationException; import org.elasticsearch.action.ActionRequestValidationException;
import org.elasticsearch.search.Scroll; import org.elasticsearch.search.Scroll;
import org.elasticsearch.search.builder.SearchSourceBuilder; import org.elasticsearch.search.builder.SearchSourceBuilder;
import org.elasticsearch.util.Required; import org.elasticsearch.util.Bytes;
import org.elasticsearch.util.Strings; import org.elasticsearch.util.Strings;
import org.elasticsearch.util.TimeValue; import org.elasticsearch.util.TimeValue;
@ -36,11 +36,11 @@ import static org.elasticsearch.search.Scroll.*;
import static org.elasticsearch.util.TimeValue.*; import static org.elasticsearch.util.TimeValue.*;
/** /**
* @author kimchy (Shay Banon) * @author kimchy (shay.banon)
*/ */
public class SearchRequest implements ActionRequest { public class SearchRequest implements ActionRequest {
private SearchType searchType = SearchType.QUERY_THEN_FETCH; private SearchType searchType = SearchType.DEFAULT;
private String[] indices; private String[] indices;
@ -48,6 +48,8 @@ public class SearchRequest implements ActionRequest {
private byte[] source; private byte[] source;
private byte[] extraSource;
private Scroll scroll; private Scroll scroll;
private int from = -1; private int from = -1;
@ -116,11 +118,29 @@ public class SearchRequest implements ActionRequest {
return this; return this;
} }
@Required public SearchRequest source(SearchSourceBuilder sourceBuilder) { /**
* The source of the search request.
*/
public SearchRequest source(SearchSourceBuilder sourceBuilder) {
return source(sourceBuilder.build()); return source(sourceBuilder.build());
} }
@Required public SearchRequest source(byte[] source) { public SearchRequest source(byte[] source) {
this.source = source;
return this;
}
/**
* Allows to provide an additional source that will be used as well.
*/
public SearchRequest extraSource(SearchSourceBuilder sourceBuilder) {
return extraSource(sourceBuilder.build());
}
/**
* Allows to provide an additional source that will be used as well.
*/
public SearchRequest extraSource(byte[] source) {
this.source = source; this.source = source;
return this; return this;
} }
@ -146,6 +166,10 @@ public class SearchRequest implements ActionRequest {
return source; return source;
} }
public byte[] extraSource() {
return this.extraSource;
}
public Scroll scroll() { public Scroll scroll() {
return scroll; return scroll;
} }
@ -212,8 +236,20 @@ public class SearchRequest implements ActionRequest {
if (in.readBoolean()) { if (in.readBoolean()) {
timeout = readTimeValue(in); timeout = readTimeValue(in);
} }
source = new byte[in.readInt()]; int size = in.readInt();
in.readFully(source); if (size == 0) {
source = Bytes.EMPTY_ARRAY;
} else {
source = new byte[size];
in.readFully(source);
}
size = in.readInt();
if (size == 0) {
extraSource = Bytes.EMPTY_ARRAY;
} else {
extraSource = new byte[size];
in.readFully(extraSource);
}
int typesSize = in.readInt(); int typesSize = in.readInt();
if (typesSize > 0) { if (typesSize > 0) {
@ -254,8 +290,18 @@ public class SearchRequest implements ActionRequest {
out.writeBoolean(true); out.writeBoolean(true);
timeout.writeTo(out); timeout.writeTo(out);
} }
out.writeInt(source.length); if (source == null) {
out.write(source); out.writeInt(0);
} else {
out.writeInt(source.length);
out.write(source);
}
if (extraSource == null) {
out.writeInt(0);
} else {
out.writeInt(extraSource.length);
out.write(extraSource);
}
out.writeInt(types.length); out.writeInt(types.length);
for (String type : types) { for (String type : types) {
out.writeUTF(type); out.writeUTF(type);

View File

@ -30,6 +30,8 @@ public enum SearchType {
DFS_QUERY_AND_FETCH((byte) 2), DFS_QUERY_AND_FETCH((byte) 2),
QUERY_AND_FETCH((byte) 3); QUERY_AND_FETCH((byte) 3);
public static final SearchType DEFAULT = QUERY_THEN_FETCH;
private byte id; private byte id;
SearchType(byte id) { SearchType(byte id) {

View File

@ -61,6 +61,7 @@ public abstract class TransportSearchHelper {
public static InternalSearchRequest internalSearchRequest(ShardRouting shardRouting, SearchRequest request) { public static InternalSearchRequest internalSearchRequest(ShardRouting shardRouting, SearchRequest request) {
InternalSearchRequest internalRequest = new InternalSearchRequest(shardRouting, request.source()); InternalSearchRequest internalRequest = new InternalSearchRequest(shardRouting, request.source());
internalRequest.extraSource(request.extraSource());
internalRequest.from(request.from()).size(request.size()); internalRequest.from(request.from()).size(request.size());
internalRequest.scroll(request.scroll()); internalRequest.scroll(request.scroll());
internalRequest.timeout(request.timeout()); internalRequest.timeout(request.timeout());

View File

@ -34,6 +34,7 @@ import java.io.InputStream;
import java.util.List; import java.util.List;
import java.util.Map; import java.util.Map;
import java.util.Set; import java.util.Set;
import java.util.regex.Pattern;
import static org.elasticsearch.util.SizeValue.*; import static org.elasticsearch.util.SizeValue.*;
import static org.elasticsearch.util.TimeValue.*; import static org.elasticsearch.util.TimeValue.*;
@ -43,6 +44,8 @@ import static org.elasticsearch.util.TimeValue.*;
*/ */
public class NettyHttpRequest implements HttpRequest { public class NettyHttpRequest implements HttpRequest {
private final Pattern commaPattern = Pattern.compile(",");
private final org.jboss.netty.handler.codec.http.HttpRequest request; private final org.jboss.netty.handler.codec.http.HttpRequest request;
private QueryStringDecoder queryStringDecoder; private QueryStringDecoder queryStringDecoder;
@ -164,6 +167,14 @@ public class NettyHttpRequest implements HttpRequest {
return parseSizeValue(param(key), defaultValue); return parseSizeValue(param(key), defaultValue);
} }
@Override public String[] paramAsStringArray(String key, String[] defaultValue) {
String value = param(key);
if (value == null) {
return defaultValue;
}
return commaPattern.split(value);
}
@Override public boolean hasParam(String key) { @Override public boolean hasParam(String key) {
return queryStringDecoder.getParameters().containsKey(key); return queryStringDecoder.getParameters().containsKey(key);
} }

View File

@ -61,6 +61,8 @@ public interface RestRequest extends ToJson.Params {
String param(String key); String param(String key);
String[] paramAsStringArray(String key, String[] defaultValue);
float paramAsFloat(String key, float defaultValue); float paramAsFloat(String key, float defaultValue);
int paramAsInt(String key, int defaultValue); int paramAsInt(String key, int defaultValue);

View File

@ -40,6 +40,7 @@ import org.elasticsearch.rest.action.deletebyquery.RestDeleteByQueryAction;
import org.elasticsearch.rest.action.get.RestGetAction; import org.elasticsearch.rest.action.get.RestGetAction;
import org.elasticsearch.rest.action.index.RestIndexAction; import org.elasticsearch.rest.action.index.RestIndexAction;
import org.elasticsearch.rest.action.main.RestMainAction; import org.elasticsearch.rest.action.main.RestMainAction;
import org.elasticsearch.rest.action.mlt.RestMoreLikeThisAction;
import org.elasticsearch.rest.action.search.RestSearchAction; import org.elasticsearch.rest.action.search.RestSearchAction;
import org.elasticsearch.rest.action.terms.RestTermsAction; import org.elasticsearch.rest.action.terms.RestTermsAction;
@ -83,5 +84,7 @@ public class RestActionModule extends AbstractModule {
bind(RestTermsAction.class).asEagerSingleton(); bind(RestTermsAction.class).asEagerSingleton();
bind(RestSearchAction.class).asEagerSingleton(); bind(RestSearchAction.class).asEagerSingleton();
bind(RestMoreLikeThisAction.class).asEagerSingleton();
} }
} }

View File

@ -0,0 +1,110 @@
/*
* Licensed to Elastic Search and Shay Banon under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. Elastic Search 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.rest.action.mlt;
import com.google.inject.Inject;
import org.elasticsearch.action.ActionListener;
import org.elasticsearch.action.mlt.MoreLikeThisRequest;
import org.elasticsearch.action.search.SearchResponse;
import org.elasticsearch.client.Client;
import org.elasticsearch.rest.*;
import org.elasticsearch.search.Scroll;
import org.elasticsearch.util.json.JsonBuilder;
import org.elasticsearch.util.settings.Settings;
import java.io.IOException;
import static org.elasticsearch.client.Requests.*;
import static org.elasticsearch.rest.RestRequest.Method.*;
import static org.elasticsearch.rest.RestResponse.Status.*;
import static org.elasticsearch.rest.action.support.RestActions.*;
import static org.elasticsearch.rest.action.support.RestJsonBuilder.*;
import static org.elasticsearch.util.TimeValue.*;
/**
* @author kimchy (shay.banon)
*/
public class RestMoreLikeThisAction extends BaseRestHandler {
@Inject public RestMoreLikeThisAction(Settings settings, Client client, RestController controller) {
super(settings, client);
controller.registerHandler(GET, "/{index}/{type}/{id}/_moreLikeThis", this);
controller.registerHandler(POST, "/{index}/{type}/{id}/_moreLikeThis", this);
}
@Override public void handleRequest(final RestRequest request, final RestChannel channel) {
MoreLikeThisRequest mltRequest = moreLikeThisRequest(request.param("index")).type(request.param("type")).id(request.param("id"));
try {
mltRequest.fields(request.paramAsStringArray("fields", null));
mltRequest.percentTermsToMatch(request.paramAsFloat("percentTermsToMatch", -1));
mltRequest.minTermFrequency(request.paramAsInt("minTermFrequency", -1));
mltRequest.maxQueryTerms(request.paramAsInt("maxQueryTerms", -1));
mltRequest.stopWords(request.paramAsStringArray("stopWords", null));
mltRequest.minDocFreq(request.paramAsInt("minDocFreq", -1));
mltRequest.maxDocFreq(request.paramAsInt("maxDocFreq", -1));
mltRequest.minWordLen(request.paramAsInt("minWordLen", -1));
mltRequest.maxWordLen(request.paramAsInt("maxWordLen", -1));
mltRequest.boostTerms(request.paramAsBoolean("boostTerms", null));
mltRequest.boostTermsFactor(request.paramAsFloat("boostTermsFactor", -1));
mltRequest.searchType(parseSearchType(request.param("searchType")));
mltRequest.searchIndices(request.paramAsStringArray("searchIndices", null));
mltRequest.searchTypes(request.paramAsStringArray("searchTypes", null));
mltRequest.searchQueryHint(request.param("searchQueryHint"));
String searchScroll = request.param("searchScroll");
if (searchScroll != null) {
mltRequest.searchScroll(new Scroll(parseTimeValue(searchScroll, null)));
}
if (request.hasContent()) {
mltRequest.searchSource(request.contentAsBytes());
}
} catch (Exception e) {
try {
JsonBuilder builder = restJsonBuilder(request);
channel.sendResponse(new JsonRestResponse(request, BAD_REQUEST, builder.startObject().field("error", e.getMessage()).endObject()));
} catch (IOException e1) {
logger.error("Failed to send failure response", e1);
}
return;
}
client.moreLikeThis(mltRequest, new ActionListener<SearchResponse>() {
@Override public void onResponse(SearchResponse response) {
try {
JsonBuilder builder = restJsonBuilder(request);
builder.startObject();
response.toJson(builder, request);
builder.endObject();
channel.sendResponse(new JsonRestResponse(request, OK, builder));
} catch (Exception e) {
onFailure(e);
}
}
@Override public void onFailure(Throwable e) {
try {
channel.sendResponse(new JsonThrowableRestResponse(request, e));
} catch (IOException e1) {
logger.error("Failed to send failure response", e1);
}
}
});
}
}

View File

@ -25,13 +25,11 @@ import org.elasticsearch.action.ActionListener;
import org.elasticsearch.action.search.SearchOperationThreading; import org.elasticsearch.action.search.SearchOperationThreading;
import org.elasticsearch.action.search.SearchRequest; import org.elasticsearch.action.search.SearchRequest;
import org.elasticsearch.action.search.SearchResponse; import org.elasticsearch.action.search.SearchResponse;
import org.elasticsearch.action.search.SearchType;
import org.elasticsearch.client.Client; import org.elasticsearch.client.Client;
import org.elasticsearch.index.query.json.JsonQueryBuilders; import org.elasticsearch.index.query.json.JsonQueryBuilders;
import org.elasticsearch.index.query.json.QueryStringJsonQueryBuilder; import org.elasticsearch.index.query.json.QueryStringJsonQueryBuilder;
import org.elasticsearch.rest.*; import org.elasticsearch.rest.*;
import org.elasticsearch.rest.action.support.RestActions; import org.elasticsearch.rest.action.support.RestActions;
import org.elasticsearch.rest.action.support.RestJsonBuilder;
import org.elasticsearch.search.Scroll; import org.elasticsearch.search.Scroll;
import org.elasticsearch.search.builder.SearchSourceBuilder; import org.elasticsearch.search.builder.SearchSourceBuilder;
import org.elasticsearch.util.json.JsonBuilder; import org.elasticsearch.util.json.JsonBuilder;
@ -43,6 +41,8 @@ import java.util.regex.Pattern;
import static org.elasticsearch.rest.RestRequest.Method.*; import static org.elasticsearch.rest.RestRequest.Method.*;
import static org.elasticsearch.rest.RestResponse.Status.*; import static org.elasticsearch.rest.RestResponse.Status.*;
import static org.elasticsearch.rest.action.support.RestActions.*;
import static org.elasticsearch.rest.action.support.RestJsonBuilder.*;
import static org.elasticsearch.util.TimeValue.*; import static org.elasticsearch.util.TimeValue.*;
/** /**
@ -82,7 +82,7 @@ public class RestSearchAction extends BaseRestHandler {
searchRequest.operationThreading(operationThreading); searchRequest.operationThreading(operationThreading);
} catch (Exception e) { } catch (Exception e) {
try { try {
JsonBuilder builder = RestJsonBuilder.restJsonBuilder(request); JsonBuilder builder = restJsonBuilder(request);
channel.sendResponse(new JsonRestResponse(request, BAD_REQUEST, builder.startObject().field("error", e.getMessage()).endObject())); channel.sendResponse(new JsonRestResponse(request, BAD_REQUEST, builder.startObject().field("error", e.getMessage()).endObject()));
} catch (IOException e1) { } catch (IOException e1) {
logger.error("Failed to send failure response", e1); logger.error("Failed to send failure response", e1);
@ -90,11 +90,11 @@ public class RestSearchAction extends BaseRestHandler {
return; return;
} }
client.execSearch(searchRequest, new ActionListener<SearchResponse>() { client.execSearch(searchRequest, new ActionListener<SearchResponse>() {
@Override public void onResponse(SearchResponse result) { @Override public void onResponse(SearchResponse response) {
try { try {
JsonBuilder builder = RestJsonBuilder.restJsonBuilder(request); JsonBuilder builder = restJsonBuilder(request);
builder.startObject(); builder.startObject();
result.toJson(builder, request); response.toJson(builder, request);
builder.endObject(); builder.endObject();
channel.sendResponse(new JsonRestResponse(request, OK, builder)); channel.sendResponse(new JsonRestResponse(request, OK, builder));
} catch (Exception e) { } catch (Exception e) {
@ -116,42 +116,16 @@ public class RestSearchAction extends BaseRestHandler {
String[] indices = RestActions.splitIndices(request.param("index")); String[] indices = RestActions.splitIndices(request.param("index"));
SearchRequest searchRequest = new SearchRequest(indices, parseSearchSource(request)); SearchRequest searchRequest = new SearchRequest(indices, parseSearchSource(request));
String searchType = request.param("searchType"); searchRequest.searchType(parseSearchType(request.param("searchType")));
if (searchType != null) { searchRequest.from(request.paramAsInt("from", -1));
if ("dfs_query_then_fetch".equals(searchType)) { searchRequest.size(request.paramAsInt("size", -1));
searchRequest.searchType(SearchType.DFS_QUERY_THEN_FETCH);
} else if ("dfs_query_and_fetch".equals(searchType)) {
searchRequest.searchType(SearchType.DFS_QUERY_AND_FETCH);
} else if ("query_then_fetch".equals(searchType)) {
searchRequest.searchType(SearchType.QUERY_THEN_FETCH);
} else if ("query_and_fetch".equals(searchType)) {
searchRequest.searchType(SearchType.QUERY_AND_FETCH);
} else {
throw new ElasticSearchIllegalArgumentException("No search type for [" + searchType + "]");
}
} else {
searchRequest.searchType(SearchType.QUERY_THEN_FETCH);
}
String from = request.param("from");
if (from != null) {
searchRequest.from(Integer.parseInt(from));
}
String size = request.param("size");
if (size != null) {
searchRequest.size(Integer.parseInt(size));
}
String scroll = request.param("scroll"); String scroll = request.param("scroll");
if (scroll != null) { if (scroll != null) {
searchRequest.scroll(new Scroll(parseTimeValue(scroll, null))); searchRequest.scroll(new Scroll(parseTimeValue(scroll, null)));
} }
String timeout = request.param("timeout"); searchRequest.timeout(request.paramAsTime("timeout", null));
if (timeout != null) {
searchRequest.timeout(parseTimeValue(timeout, null));
}
String typesParam = request.param("type"); String typesParam = request.param("type");
if (typesParam != null) { if (typesParam != null) {

View File

@ -21,6 +21,7 @@ package org.elasticsearch.rest.action.support;
import org.elasticsearch.ElasticSearchIllegalArgumentException; import org.elasticsearch.ElasticSearchIllegalArgumentException;
import org.elasticsearch.action.ShardOperationFailedException; import org.elasticsearch.action.ShardOperationFailedException;
import org.elasticsearch.action.search.SearchType;
import org.elasticsearch.action.support.broadcast.BroadcastOperationResponse; import org.elasticsearch.action.support.broadcast.BroadcastOperationResponse;
import org.elasticsearch.index.query.json.JsonQueryBuilders; import org.elasticsearch.index.query.json.JsonQueryBuilders;
import org.elasticsearch.index.query.json.QueryStringJsonQueryBuilder; import org.elasticsearch.index.query.json.QueryStringJsonQueryBuilder;
@ -36,16 +37,9 @@ import java.util.regex.Pattern;
*/ */
public class RestActions { public class RestActions {
public final static Pattern indicesPattern; public final static Pattern indicesPattern = Pattern.compile(",");
public final static Pattern typesPattern; public final static Pattern typesPattern = Pattern.compile(",");
private final static Pattern nodesIdsPattern; public final static Pattern nodesIdsPattern = Pattern.compile(",");
static {
indicesPattern = Pattern.compile(",");
typesPattern = Pattern.compile(",");
nodesIdsPattern = Pattern.compile(",");
}
public static void buildBroadcastShardsHeader(JsonBuilder builder, BroadcastOperationResponse response) throws IOException { public static void buildBroadcastShardsHeader(JsonBuilder builder, BroadcastOperationResponse response) throws IOException {
builder.startObject("_shards"); builder.startObject("_shards");
@ -70,6 +64,23 @@ public class RestActions {
builder.endObject(); builder.endObject();
} }
public static SearchType parseSearchType(String searchType) {
if (searchType == null) {
return SearchType.DEFAULT;
}
if ("dfs_query_then_fetch".equals(searchType)) {
return SearchType.DFS_QUERY_THEN_FETCH;
} else if ("dfs_query_and_fetch".equals(searchType)) {
return SearchType.DFS_QUERY_AND_FETCH;
} else if ("query_then_fetch".equals(searchType)) {
return SearchType.QUERY_THEN_FETCH;
} else if ("query_and_fetch".equals(searchType)) {
return SearchType.QUERY_AND_FETCH;
} else {
throw new ElasticSearchIllegalArgumentException("No search type for [" + searchType + "]");
}
}
public static byte[] parseQuerySource(RestRequest request) { public static byte[] parseQuerySource(RestRequest request) {
if (request.hasContent()) { if (request.hasContent()) {
return request.contentAsBytes(); return request.contentAsBytes();

View File

@ -245,8 +245,7 @@ public class SearchService extends AbstractComponent implements LifecycleCompone
SearchShardTarget shardTarget = new SearchShardTarget(clusterService.state().nodes().localNodeId(), request.index(), request.shardId()); SearchShardTarget shardTarget = new SearchShardTarget(clusterService.state().nodes().localNodeId(), request.index(), request.shardId());
SearchContext context = new SearchContext(idGenerator.incrementAndGet(), shardTarget, request.timeout(), SearchContext context = new SearchContext(idGenerator.incrementAndGet(), shardTarget, request.timeout(), request.types(), engineSearcher, indexService);
request.source(), request.types(), engineSearcher, indexService);
// init the from and size // init the from and size
context.from(request.from()); context.from(request.from());
@ -254,7 +253,8 @@ public class SearchService extends AbstractComponent implements LifecycleCompone
context.scroll(request.scroll()); context.scroll(request.scroll());
parseSource(context); parseSource(context, request.source());
parseSource(context, request.extraSource());
// if the from and size are still not set, default them // if the from and size are still not set, default them
if (context.from() == -1) { if (context.from() == -1) {
@ -293,9 +293,13 @@ public class SearchService extends AbstractComponent implements LifecycleCompone
context.release(); context.release();
} }
private void parseSource(SearchContext context) throws SearchParseException { private void parseSource(SearchContext context, byte[] source) throws SearchParseException {
// nothing to parse...
if (source == null || source.length == 0) {
return;
}
try { try {
JsonParser jp = jsonFactory.createJsonParser(context.source()); JsonParser jp = jsonFactory.createJsonParser(source);
JsonToken token; JsonToken token;
while ((token = jp.nextToken()) != JsonToken.END_OBJECT) { while ((token = jp.nextToken()) != JsonToken.END_OBJECT) {
if (token == JsonToken.FIELD_NAME) { if (token == JsonToken.FIELD_NAME) {
@ -311,7 +315,7 @@ public class SearchService extends AbstractComponent implements LifecycleCompone
} }
} }
} catch (Exception e) { } catch (Exception e) {
throw new SearchParseException(context, "Failed to parse [" + Unicode.fromBytes(context.source()) + "]", e); throw new SearchParseException(context, "Failed to parse [" + Unicode.fromBytes(source) + "]", e);
} }
} }

View File

@ -154,8 +154,10 @@ public class SearchSourceBuilder {
builder.field("queryParserName", queryParserName); builder.field("queryParserName", queryParserName);
} }
builder.field("query"); if (queryBuilder != null) {
queryBuilder.toJson(builder, ToJson.EMPTY_PARAMS); builder.field("query");
queryBuilder.toJson(builder, ToJson.EMPTY_PARAMS);
}
if (explain != null) { if (explain != null) {
builder.field("explain", explain); builder.field("explain", explain);

View File

@ -21,6 +21,7 @@ package org.elasticsearch.search.internal;
import org.elasticsearch.cluster.routing.ShardRouting; import org.elasticsearch.cluster.routing.ShardRouting;
import org.elasticsearch.search.Scroll; import org.elasticsearch.search.Scroll;
import org.elasticsearch.util.Bytes;
import org.elasticsearch.util.Strings; import org.elasticsearch.util.Strings;
import org.elasticsearch.util.TimeValue; import org.elasticsearch.util.TimeValue;
import org.elasticsearch.util.io.Streamable; import org.elasticsearch.util.io.Streamable;
@ -70,6 +71,8 @@ public class InternalSearchRequest implements Streamable {
private byte[] source; private byte[] source;
private byte[] extraSource;
public InternalSearchRequest() { public InternalSearchRequest() {
} }
@ -95,6 +98,15 @@ public class InternalSearchRequest implements Streamable {
return this.source; return this.source;
} }
public byte[] extraSource() {
return this.extraSource;
}
public InternalSearchRequest extraSource(byte[] extraSource) {
this.extraSource = extraSource;
return this;
}
public Scroll scroll() { public Scroll scroll() {
return scroll; return scroll;
} }
@ -150,8 +162,20 @@ public class InternalSearchRequest implements Streamable {
if (in.readBoolean()) { if (in.readBoolean()) {
timeout = readTimeValue(in); timeout = readTimeValue(in);
} }
source = new byte[in.readInt()]; int size = in.readInt();
in.readFully(source); if (size == 0) {
source = Bytes.EMPTY_ARRAY;
} else {
source = new byte[size];
in.readFully(source);
}
size = in.readInt();
if (size == 0) {
extraSource = Bytes.EMPTY_ARRAY;
} else {
extraSource = new byte[size];
in.readFully(extraSource);
}
int typesSize = in.readInt(); int typesSize = in.readInt();
if (typesSize > 0) { if (typesSize > 0) {
types = new String[typesSize]; types = new String[typesSize];
@ -178,8 +202,18 @@ public class InternalSearchRequest implements Streamable {
out.writeBoolean(true); out.writeBoolean(true);
timeout.writeTo(out); timeout.writeTo(out);
} }
out.writeInt(source.length); if (source == null) {
out.write(source); out.writeInt(0);
} else {
out.writeInt(source.length);
out.write(source);
}
if (extraSource == null) {
out.writeInt(0);
} else {
out.writeInt(extraSource.length);
out.write(extraSource);
}
out.writeInt(types.length); out.writeInt(types.length);
for (String type : types) { for (String type : types) {
out.writeUTF(type); out.writeUTF(type);

View File

@ -50,8 +50,6 @@ public class SearchContext implements Releasable {
private final SearchShardTarget shardTarget; private final SearchShardTarget shardTarget;
private final byte[] source;
private final Engine.Searcher engineSearcher; private final Engine.Searcher engineSearcher;
private final IndexService indexService; private final IndexService indexService;
@ -100,12 +98,11 @@ public class SearchContext implements Releasable {
private volatile Timeout keepAliveTimeout; private volatile Timeout keepAliveTimeout;
public SearchContext(long id, SearchShardTarget shardTarget, TimeValue timeout, byte[] source, public SearchContext(long id, SearchShardTarget shardTarget, TimeValue timeout,
String[] types, Engine.Searcher engineSearcher, IndexService indexService) { String[] types, Engine.Searcher engineSearcher, IndexService indexService) {
this.id = id; this.id = id;
this.shardTarget = shardTarget; this.shardTarget = shardTarget;
this.timeout = timeout; this.timeout = timeout;
this.source = source;
this.types = types; this.types = types;
this.engineSearcher = engineSearcher; this.engineSearcher = engineSearcher;
this.dfsResult = new DfsSearchResult(id, shardTarget); this.dfsResult = new DfsSearchResult(id, shardTarget);
@ -137,10 +134,6 @@ public class SearchContext implements Releasable {
return this.shardTarget; return this.shardTarget;
} }
public byte[] source() {
return source;
}
public String[] types() { public String[] types() {
return types; return types;
} }

View File

@ -0,0 +1,28 @@
/*
* Licensed to Elastic Search and Shay Banon under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. Elastic Search 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.util;
/**
* @author kimchy (shay.banon)
*/
public class Bytes {
public static final byte[] EMPTY_ARRAY = new byte[0];
}

View File

@ -0,0 +1,49 @@
/*
* Licensed to Elastic Search and Shay Banon under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. Elastic Search 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.test.integration.client.transport;
import org.elasticsearch.client.Client;
import org.elasticsearch.client.transport.TransportClient;
import org.elasticsearch.server.internal.InternalServer;
import org.elasticsearch.test.integration.document.MoreLikeThisActionTests;
import org.elasticsearch.transport.TransportService;
import org.elasticsearch.util.transport.TransportAddress;
import static org.elasticsearch.util.settings.ImmutableSettings.*;
/**
* @author kimchy (shay.banon)
*/
public class TransportClientMoreLikeThisActionTests extends MoreLikeThisActionTests {
@Override protected Client getClient1() {
TransportAddress server1Address = ((InternalServer) server("server1")).injector().getInstance(TransportService.class).boundAddress().publishAddress();
TransportClient client = new TransportClient(settingsBuilder().putBoolean("discovery.enabled", false).build());
client.addTransportAddress(server1Address);
return client;
}
@Override protected Client getClient2() {
TransportAddress server1Address = ((InternalServer) server("server2")).injector().getInstance(TransportService.class).boundAddress().publishAddress();
TransportClient client = new TransportClient(settingsBuilder().putBoolean("discovery.enabled", false).build());
client.addTransportAddress(server1Address);
return client;
}
}

View File

@ -0,0 +1,6 @@
cluster:
routing:
schedule: 100ms
index:
numberOfShards: 5
numberOfReplicas: 1