Mappings: Reencode transformed result with same xcontent

When I originally wrote the transform feature I didn't think that the
XContentType of the reencoded source mattered.  It actually matters because
payloads for the completion suggester are stored and returned exactly
as encoded by this XContentType.

This revision changes the transform feature from always reencoding with smile
to always reencoding with the provided XContentType to support the completion
suggester.

Closes #8959
This commit is contained in:
Nik Everett 2014-12-16 10:05:58 -05:00 committed by Adrien Grand
parent a4133ec4a3
commit a95d75e074
2 changed files with 51 additions and 10 deletions

View File

@ -22,6 +22,7 @@ package org.elasticsearch.index.mapper;
import com.google.common.collect.ImmutableMap; import com.google.common.collect.ImmutableMap;
import com.google.common.collect.Maps; import com.google.common.collect.Maps;
import com.google.common.collect.Sets; import com.google.common.collect.Sets;
import org.apache.lucene.analysis.Analyzer; import org.apache.lucene.analysis.Analyzer;
import org.apache.lucene.document.Field; import org.apache.lucene.document.Field;
import org.apache.lucene.document.FieldType; import org.apache.lucene.document.FieldType;
@ -652,9 +653,8 @@ public class DocumentMapper implements ToXContent {
private XContentParser transform(XContentParser parser) throws IOException { private XContentParser transform(XContentParser parser) throws IOException {
Map<String, Object> transformed = transformSourceAsMap(parser.mapOrderedAndClose()); Map<String, Object> transformed = transformSourceAsMap(parser.mapOrderedAndClose());
// TODO it'd be nice to have a MapXContent or something that could spit out the parser for this map XContentBuilder builder = XContentFactory.contentBuilder(parser.contentType()).value(transformed);
XContentBuilder builder = XContentFactory.smileBuilder().value(transformed); return parser.contentType().xContent().createParser(builder.bytes());
return SmileXContent.smileXContent.createParser(builder.bytes());
} }
public void addFieldMappers(List<FieldMapper<?>> fieldMappers) { public void addFieldMappers(List<FieldMapper<?>> fieldMappers) {

View File

@ -20,10 +20,15 @@
package org.elasticsearch.index.mapper; package org.elasticsearch.index.mapper;
import com.google.common.collect.ImmutableMap; import com.google.common.collect.ImmutableMap;
import org.elasticsearch.action.get.GetResponse; import org.elasticsearch.action.get.GetResponse;
import org.elasticsearch.action.search.SearchResponse; import org.elasticsearch.action.search.SearchResponse;
import org.elasticsearch.action.suggest.SuggestResponse;
import org.elasticsearch.common.xcontent.XContentBuilder; import org.elasticsearch.common.xcontent.XContentBuilder;
import org.elasticsearch.common.xcontent.XContentFactory; import org.elasticsearch.common.xcontent.XContentFactory;
import org.elasticsearch.common.xcontent.XContentType;
import org.elasticsearch.search.suggest.SuggestBuilders;
import org.elasticsearch.search.suggest.completion.CompletionSuggestion;
import org.elasticsearch.test.ElasticsearchIntegrationTest; import org.elasticsearch.test.ElasticsearchIntegrationTest;
import org.junit.Test; import org.junit.Test;
@ -32,8 +37,15 @@ import java.util.Map;
import java.util.concurrent.ExecutionException; import java.util.concurrent.ExecutionException;
import static org.elasticsearch.index.query.QueryBuilders.termQuery; import static org.elasticsearch.index.query.QueryBuilders.termQuery;
import static org.elasticsearch.test.hamcrest.ElasticsearchAssertions.*; import static org.elasticsearch.test.hamcrest.ElasticsearchAssertions.assertAcked;
import static org.hamcrest.Matchers.*; import static org.elasticsearch.test.hamcrest.ElasticsearchAssertions.assertExists;
import static org.elasticsearch.test.hamcrest.ElasticsearchAssertions.assertHitCount;
import static org.elasticsearch.test.hamcrest.ElasticsearchAssertions.assertSearchHits;
import static org.elasticsearch.test.hamcrest.ElasticsearchAssertions.assertSuggestion;
import static org.hamcrest.Matchers.both;
import static org.hamcrest.Matchers.hasEntry;
import static org.hamcrest.Matchers.hasKey;
import static org.hamcrest.Matchers.not;
/** /**
* Tests for transforming the source document before indexing. * Tests for transforming the source document before indexing.
@ -41,7 +53,7 @@ import static org.hamcrest.Matchers.*;
public class TransformOnIndexMapperIntegrationTest extends ElasticsearchIntegrationTest { public class TransformOnIndexMapperIntegrationTest extends ElasticsearchIntegrationTest {
@Test @Test
public void searchOnTransformed() throws Exception { public void searchOnTransformed() throws Exception {
setup(false); setup(true);
// Searching by the field created in the transport finds the entry // Searching by the field created in the transport finds the entry
SearchResponse response = client().prepareSearch("test").setQuery(termQuery("destination", "findme")).get(); SearchResponse response = client().prepareSearch("test").setQuery(termQuery("destination", "findme")).get();
@ -67,6 +79,35 @@ public class TransformOnIndexMapperIntegrationTest extends ElasticsearchIntegrat
assertRightTitleSourceTransformed(response.getSource()); assertRightTitleSourceTransformed(response.getSource());
} }
// TODO: the completion suggester currently returns payloads with no reencoding so this test
// exists to make sure that _source transformation and completion work well together. If we
// ever fix the completion suggester to reencode the payloads then we can remove this test.
@Test
public void contextSuggestPayloadTransformed() throws Exception {
XContentBuilder builder = XContentFactory.jsonBuilder().startObject();
builder.startObject("properties");
builder.startObject("suggest").field("type", "completion").field("payloads", true).endObject();
builder.endObject();
builder.startObject("transform");
builder.field("script", "ctx._source.suggest = ['input': ctx._source.text];ctx._source.suggest.payload = ['display': ctx._source.text, 'display_detail': 'on the fly']");
builder.field("lang", "groovy");
builder.endObject();
assertAcked(client().admin().indices().prepareCreate("test").addMapping("test", builder));
// Payload is stored using original source format (json, smile, yaml, whatever)
XContentType type = XContentType.values()[between(0, XContentType.values().length - 1)];
XContentBuilder source = XContentFactory.contentBuilder(type);
source.startObject().field("text", "findme").endObject();
indexRandom(true, client().prepareIndex("test", "test", "findme").setSource(source));
SuggestResponse response = client().prepareSuggest("test").addSuggestion(
SuggestBuilders.completionSuggestion("test").field("suggest").text("findme")).get();
assertSuggestion(response.getSuggest(), 0, 0, "test", "findme");
CompletionSuggestion.Entry.Option option = (CompletionSuggestion.Entry.Option)response.getSuggest().getSuggestion("test").getEntries().get(0).getOptions().get(0);
// And it comes back in exactly that way.
XContentBuilder expected = XContentFactory.contentBuilder(type);
expected.startObject().field("display", "findme").field("display_detail", "on the fly").endObject();
assertEquals(expected.string(), option.getPayloadAsString());
}
/** /**
* Setup an index with some source transforms. Randomly picks the number of * Setup an index with some source transforms. Randomly picks the number of
* transforms but all but one of the transforms is a noop. The other is a * transforms but all but one of the transforms is a noop. The other is a
@ -74,12 +115,12 @@ public class TransformOnIndexMapperIntegrationTest extends ElasticsearchIntegrat
* if the 'title' field starts with 't' and then always removes the * if the 'title' field starts with 't' and then always removes the
* 'content' field regarless of the contents of 't'. The actual script * 'content' field regarless of the contents of 't'. The actual script
* randomly uses parameters or not. * randomly uses parameters or not.
* *
* @param flush * @param forceRefresh
* should the data be flushed to disk? Set to false to test real * should the data be flushed to disk? Set to false to test real
* time fetching * time fetching
*/ */
private void setup(boolean flush) throws IOException, InterruptedException, ExecutionException { private void setup(boolean forceRefresh) throws IOException, InterruptedException, ExecutionException {
XContentBuilder builder = XContentFactory.jsonBuilder().startObject(); XContentBuilder builder = XContentFactory.jsonBuilder().startObject();
builder.field("transform"); builder.field("transform");
if (getRandom().nextBoolean()) { if (getRandom().nextBoolean()) {
@ -107,7 +148,7 @@ public class TransformOnIndexMapperIntegrationTest extends ElasticsearchIntegrat
} }
assertAcked(client().admin().indices().prepareCreate("test").addMapping("test", builder)); assertAcked(client().admin().indices().prepareCreate("test").addMapping("test", builder));
indexRandom(!flush, client().prepareIndex("test", "test", "notitle").setSource("content", "findme"), indexRandom(forceRefresh, client().prepareIndex("test", "test", "notitle").setSource("content", "findme"),
client().prepareIndex("test", "test", "badtitle").setSource("content", "findme", "title", "cat"), client().prepareIndex("test", "test", "badtitle").setSource("content", "findme", "title", "cat"),
client().prepareIndex("test", "test", "righttitle").setSource("content", "findme", "title", "table")); client().prepareIndex("test", "test", "righttitle").setSource("content", "findme", "title", "table"));
} }