mirror of
https://github.com/spring-projects/spring-data-elasticsearch.git
synced 2025-06-29 15:22:11 +00:00
DATAES-64 - Add dynamic settings using @Setting annotation
This commit is contained in:
parent
403930d448
commit
1e3ec3eaf0
@ -0,0 +1,36 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2014 the original author or authors.
|
||||||
|
*
|
||||||
|
* Licensed 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.springframework.data.elasticsearch.annotations;
|
||||||
|
|
||||||
|
import java.lang.annotation.*;
|
||||||
|
|
||||||
|
import org.springframework.data.annotation.Persistent;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Elasticsearch Setting
|
||||||
|
*
|
||||||
|
* @author Mohsin Husen
|
||||||
|
*/
|
||||||
|
|
||||||
|
@Persistent
|
||||||
|
@Inherited
|
||||||
|
@Retention(RetentionPolicy.RUNTIME)
|
||||||
|
@Target({ElementType.TYPE})
|
||||||
|
public @interface Setting {
|
||||||
|
|
||||||
|
String settingPath() default "";
|
||||||
|
|
||||||
|
}
|
@ -23,7 +23,9 @@ import static org.elasticsearch.common.collect.Sets.*;
|
|||||||
import static org.elasticsearch.index.VersionType.*;
|
import static org.elasticsearch.index.VersionType.*;
|
||||||
import static org.springframework.data.elasticsearch.core.MappingBuilder.*;
|
import static org.springframework.data.elasticsearch.core.MappingBuilder.*;
|
||||||
|
|
||||||
|
import java.io.BufferedReader;
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
|
import java.io.InputStreamReader;
|
||||||
import java.lang.reflect.Method;
|
import java.lang.reflect.Method;
|
||||||
import java.util.*;
|
import java.util.*;
|
||||||
|
|
||||||
@ -64,10 +66,14 @@ import org.elasticsearch.search.facet.FacetBuilder;
|
|||||||
import org.elasticsearch.search.highlight.HighlightBuilder;
|
import org.elasticsearch.search.highlight.HighlightBuilder;
|
||||||
import org.elasticsearch.search.sort.SortBuilder;
|
import org.elasticsearch.search.sort.SortBuilder;
|
||||||
import org.elasticsearch.search.sort.SortOrder;
|
import org.elasticsearch.search.sort.SortOrder;
|
||||||
|
import org.slf4j.Logger;
|
||||||
|
import org.slf4j.LoggerFactory;
|
||||||
|
import org.springframework.core.io.ClassPathResource;
|
||||||
import org.springframework.data.domain.Page;
|
import org.springframework.data.domain.Page;
|
||||||
import org.springframework.data.domain.Sort;
|
import org.springframework.data.domain.Sort;
|
||||||
import org.springframework.data.elasticsearch.ElasticsearchException;
|
import org.springframework.data.elasticsearch.ElasticsearchException;
|
||||||
import org.springframework.data.elasticsearch.annotations.Document;
|
import org.springframework.data.elasticsearch.annotations.Document;
|
||||||
|
import org.springframework.data.elasticsearch.annotations.Setting;
|
||||||
import org.springframework.data.elasticsearch.core.convert.ElasticsearchConverter;
|
import org.springframework.data.elasticsearch.core.convert.ElasticsearchConverter;
|
||||||
import org.springframework.data.elasticsearch.core.convert.MappingElasticsearchConverter;
|
import org.springframework.data.elasticsearch.core.convert.MappingElasticsearchConverter;
|
||||||
import org.springframework.data.elasticsearch.core.facet.FacetRequest;
|
import org.springframework.data.elasticsearch.core.facet.FacetRequest;
|
||||||
@ -87,6 +93,7 @@ import org.springframework.util.Assert;
|
|||||||
|
|
||||||
public class ElasticsearchTemplate implements ElasticsearchOperations {
|
public class ElasticsearchTemplate implements ElasticsearchOperations {
|
||||||
|
|
||||||
|
private static final Logger logger = LoggerFactory.getLogger(ElasticsearchTemplate.class);
|
||||||
private Client client;
|
private Client client;
|
||||||
private ElasticsearchConverter elasticsearchConverter;
|
private ElasticsearchConverter elasticsearchConverter;
|
||||||
private ResultsMapper resultsMapper;
|
private ResultsMapper resultsMapper;
|
||||||
@ -554,6 +561,17 @@ public class ElasticsearchTemplate implements ElasticsearchOperations {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private <T> boolean createIndexWithSettings(Class<T> clazz) {
|
private <T> boolean createIndexWithSettings(Class<T> clazz) {
|
||||||
|
if(clazz.isAnnotationPresent(Setting.class)) {
|
||||||
|
String settingPath = clazz.getAnnotation(Setting.class).settingPath();
|
||||||
|
if(isNotBlank(settingPath)) {
|
||||||
|
String settings = readFileFromClasspath(settingPath);
|
||||||
|
if(isNotBlank(settings)) {
|
||||||
|
return createIndex(getPersistentEntityFor(clazz).getIndexName(), settings);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
logger.info("settingPath in @Setting has to be defined. Using default instead.");
|
||||||
|
}
|
||||||
|
}
|
||||||
return createIndex(getPersistentEntityFor(clazz).getIndexName(), getDefaultSettings(getPersistentEntityFor(clazz)));
|
return createIndex(getPersistentEntityFor(clazz).getIndexName(), getDefaultSettings(getPersistentEntityFor(clazz)));
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -790,4 +808,33 @@ public class ElasticsearchTemplate implements ElasticsearchOperations {
|
|||||||
private boolean isDocument(Class clazz) {
|
private boolean isDocument(Class clazz) {
|
||||||
return clazz.isAnnotationPresent(Document.class);
|
return clazz.isAnnotationPresent(Document.class);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static String readFileFromClasspath(String url) {
|
||||||
|
StringBuilder stringBuilder = new StringBuilder();
|
||||||
|
|
||||||
|
BufferedReader bufferedReader = null;
|
||||||
|
|
||||||
|
try {
|
||||||
|
ClassPathResource classPathResource = new ClassPathResource(url);
|
||||||
|
InputStreamReader inputStreamReader = new InputStreamReader(classPathResource.getInputStream());
|
||||||
|
bufferedReader = new BufferedReader(inputStreamReader);
|
||||||
|
String line;
|
||||||
|
|
||||||
|
while ((line = bufferedReader.readLine()) != null) {
|
||||||
|
stringBuilder.append(line);
|
||||||
|
}
|
||||||
|
} catch (Exception e) {
|
||||||
|
logger.debug(String.format("Failed to load file from url: %s: %s", url, e.getMessage()));
|
||||||
|
return null;
|
||||||
|
} finally {
|
||||||
|
if (bufferedReader != null)
|
||||||
|
try {
|
||||||
|
bufferedReader.close();
|
||||||
|
} catch (IOException e) {
|
||||||
|
logger.debug(String.format("Unable to close buffered reader.. %s", e.getMessage()));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return stringBuilder.toString();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright 2013 the original author or authors.
|
* Copyright 2013-2014 the original author or authors.
|
||||||
*
|
*
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
* you may not use this file except in compliance with the License.
|
* you may not use this file except in compliance with the License.
|
||||||
@ -43,4 +43,6 @@ public interface ElasticsearchPersistentEntity<T> extends PersistentEntity<T, El
|
|||||||
String getParentType();
|
String getParentType();
|
||||||
|
|
||||||
ElasticsearchPersistentProperty getParentIdProperty();
|
ElasticsearchPersistentProperty getParentIdProperty();
|
||||||
|
|
||||||
|
String settingPath();
|
||||||
}
|
}
|
||||||
|
@ -26,6 +26,7 @@ import org.springframework.context.expression.BeanFactoryAccessor;
|
|||||||
import org.springframework.context.expression.BeanFactoryResolver;
|
import org.springframework.context.expression.BeanFactoryResolver;
|
||||||
import org.springframework.data.elasticsearch.annotations.Document;
|
import org.springframework.data.elasticsearch.annotations.Document;
|
||||||
import org.springframework.data.elasticsearch.annotations.Parent;
|
import org.springframework.data.elasticsearch.annotations.Parent;
|
||||||
|
import org.springframework.data.elasticsearch.annotations.Setting;
|
||||||
import org.springframework.data.mapping.model.BasicPersistentEntity;
|
import org.springframework.data.mapping.model.BasicPersistentEntity;
|
||||||
import org.springframework.data.util.TypeInformation;
|
import org.springframework.data.util.TypeInformation;
|
||||||
import org.springframework.expression.spel.support.StandardEvaluationContext;
|
import org.springframework.expression.spel.support.StandardEvaluationContext;
|
||||||
@ -50,6 +51,7 @@ public class SimpleElasticsearchPersistentEntity<T> extends BasicPersistentEntit
|
|||||||
private String indexStoreType;
|
private String indexStoreType;
|
||||||
private String parentType;
|
private String parentType;
|
||||||
private ElasticsearchPersistentProperty parentIdProperty;
|
private ElasticsearchPersistentProperty parentIdProperty;
|
||||||
|
private String settingPath;
|
||||||
|
|
||||||
public SimpleElasticsearchPersistentEntity(TypeInformation<T> typeInformation) {
|
public SimpleElasticsearchPersistentEntity(TypeInformation<T> typeInformation) {
|
||||||
super(typeInformation);
|
super(typeInformation);
|
||||||
@ -66,6 +68,9 @@ public class SimpleElasticsearchPersistentEntity<T> extends BasicPersistentEntit
|
|||||||
this.refreshInterval = typeInformation.getType().getAnnotation(Document.class).refreshInterval();
|
this.refreshInterval = typeInformation.getType().getAnnotation(Document.class).refreshInterval();
|
||||||
this.indexStoreType = typeInformation.getType().getAnnotation(Document.class).indexStoreType();
|
this.indexStoreType = typeInformation.getType().getAnnotation(Document.class).indexStoreType();
|
||||||
}
|
}
|
||||||
|
if(clazz.isAnnotationPresent(Setting.class)) {
|
||||||
|
this.settingPath = typeInformation.getType().getAnnotation(Setting.class).settingPath();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
@ -115,6 +120,11 @@ public class SimpleElasticsearchPersistentEntity<T> extends BasicPersistentEntit
|
|||||||
return parentIdProperty;
|
return parentIdProperty;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String settingPath() {
|
||||||
|
return settingPath;
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void addPersistentProperty(ElasticsearchPersistentProperty property) {
|
public void addPersistentProperty(ElasticsearchPersistentProperty property) {
|
||||||
super.addPersistentProperty(property);
|
super.addPersistentProperty(property);
|
||||||
|
@ -0,0 +1,62 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2014 the original author or authors.
|
||||||
|
*
|
||||||
|
* Licensed 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.springframework.data.elasticsearch.entities;
|
||||||
|
|
||||||
|
import org.springframework.data.annotation.Id;
|
||||||
|
import org.springframework.data.elasticsearch.annotations.Document;
|
||||||
|
import org.springframework.data.elasticsearch.annotations.Field;
|
||||||
|
import org.springframework.data.elasticsearch.annotations.FieldType;
|
||||||
|
import org.springframework.data.elasticsearch.annotations.Setting;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sample SettingEntity for test out dynamic setting using @Setting Annotation
|
||||||
|
*
|
||||||
|
* @author Mohsin Husen
|
||||||
|
*/
|
||||||
|
@Setting(settingPath = "/settings/test-settings.json")
|
||||||
|
@Document(indexName = "test-setting-index", type = "test-setting-type")
|
||||||
|
public class SettingEntity {
|
||||||
|
|
||||||
|
@Id
|
||||||
|
private String id;
|
||||||
|
private String name;
|
||||||
|
@Field(type = FieldType.String, searchAnalyzer = "emailAnalyzer", indexAnalyzer = "emailAnalyzer")
|
||||||
|
private String email;
|
||||||
|
|
||||||
|
public String getId() {
|
||||||
|
return id;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setId(String id) {
|
||||||
|
this.id = id;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getName() {
|
||||||
|
return name;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setName(String name) {
|
||||||
|
this.name = name;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getEmail() {
|
||||||
|
return email;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setEmail(String email) {
|
||||||
|
this.email = email;
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,28 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2014 the original author or authors.
|
||||||
|
*
|
||||||
|
* Licensed 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.springframework.data.elasticsearch.repositories.setting;
|
||||||
|
|
||||||
|
import org.springframework.data.elasticsearch.entities.SettingEntity;
|
||||||
|
import org.springframework.data.elasticsearch.repository.ElasticsearchCrudRepository;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* SettingEntityRepository
|
||||||
|
*
|
||||||
|
* @author Mohsin Husen
|
||||||
|
*/
|
||||||
|
public interface SettingEntityRepository extends ElasticsearchCrudRepository<SettingEntity, String>{
|
||||||
|
|
||||||
|
}
|
@ -0,0 +1,112 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2014 the original author or authors.
|
||||||
|
*
|
||||||
|
* Licensed 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.springframework.data.elasticsearch.repositories.setting;
|
||||||
|
|
||||||
|
import static org.hamcrest.Matchers.*;
|
||||||
|
import static org.junit.Assert.assertThat;
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
|
import org.apache.commons.lang.RandomStringUtils;
|
||||||
|
import org.elasticsearch.index.query.QueryBuilders;
|
||||||
|
import org.junit.Before;
|
||||||
|
import org.junit.Test;
|
||||||
|
import org.junit.runner.RunWith;
|
||||||
|
import org.springframework.beans.factory.annotation.Autowired;
|
||||||
|
import org.springframework.data.elasticsearch.core.ElasticsearchTemplate;
|
||||||
|
import org.springframework.data.elasticsearch.core.query.NativeSearchQueryBuilder;
|
||||||
|
import org.springframework.data.elasticsearch.core.query.SearchQuery;
|
||||||
|
import org.springframework.data.elasticsearch.entities.SettingEntity;
|
||||||
|
import org.springframework.test.context.ContextConfiguration;
|
||||||
|
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* SettingEntityRepositoryTest
|
||||||
|
*
|
||||||
|
* @author Mohsin Husen
|
||||||
|
*/
|
||||||
|
@RunWith(SpringJUnit4ClassRunner.class)
|
||||||
|
@ContextConfiguration("classpath:dynamic-settings-test.xml")
|
||||||
|
public class SettingEntityRepositoryTest {
|
||||||
|
|
||||||
|
@Autowired
|
||||||
|
private SettingEntityRepository repository;
|
||||||
|
|
||||||
|
@Autowired
|
||||||
|
private ElasticsearchTemplate elasticsearchTemplate;
|
||||||
|
|
||||||
|
@Before
|
||||||
|
public void before() {
|
||||||
|
elasticsearchTemplate.deleteIndex(SettingEntity.class);
|
||||||
|
elasticsearchTemplate.createIndex(SettingEntity.class);
|
||||||
|
elasticsearchTemplate.putMapping(SettingEntity.class);
|
||||||
|
elasticsearchTemplate.refresh(SettingEntity.class, true);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
DATAES-64
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void shouldCreateGivenDynamicSettingsForGivenIndex() {
|
||||||
|
//given
|
||||||
|
//delete , create and apply mapping in before method
|
||||||
|
|
||||||
|
// then
|
||||||
|
assertThat(elasticsearchTemplate.indexExists(SettingEntity.class), is(true));
|
||||||
|
Map map = elasticsearchTemplate.getSetting(SettingEntity.class);
|
||||||
|
assertThat(map.containsKey("index.number_of_replicas"), is(true));
|
||||||
|
assertThat(map.containsKey("index.number_of_shards"), is(true));
|
||||||
|
assertThat(map.containsKey("index.analysis.analyzer.emailAnalyzer.tokenizer"), is(true));
|
||||||
|
assertThat((String) map.get("index.number_of_replicas"), is("0"));
|
||||||
|
assertThat((String) map.get("index.number_of_shards"), is("1"));
|
||||||
|
assertThat((String) map.get("index.analysis.analyzer.emailAnalyzer.tokenizer"), is("uax_url_email"));
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
DATAES-64
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void shouldSearchOnGivenTokenizerUsingGivenDynamicSettingsForGivenIndex() {
|
||||||
|
//given
|
||||||
|
SettingEntity settingEntity1 = new SettingEntity();
|
||||||
|
settingEntity1.setId(RandomStringUtils.randomNumeric(5));
|
||||||
|
settingEntity1.setName("test-setting1");
|
||||||
|
settingEntity1.setEmail("test_setting1@test.com");
|
||||||
|
|
||||||
|
repository.save(settingEntity1);
|
||||||
|
|
||||||
|
SettingEntity settingEntity2 = new SettingEntity();
|
||||||
|
settingEntity2.setId(RandomStringUtils.randomNumeric(5));
|
||||||
|
settingEntity2.setName("test-setting2");
|
||||||
|
settingEntity2.setEmail("test_setting2@test.com");
|
||||||
|
|
||||||
|
repository.save(settingEntity2);
|
||||||
|
|
||||||
|
//when
|
||||||
|
SearchQuery searchQuery = new NativeSearchQueryBuilder()
|
||||||
|
.withQuery(QueryBuilders.termQuery("email", settingEntity1.getEmail())).build();
|
||||||
|
|
||||||
|
long count = elasticsearchTemplate.count(searchQuery, SettingEntity.class);
|
||||||
|
List<SettingEntity> entityList = elasticsearchTemplate.queryForList(searchQuery, SettingEntity.class);
|
||||||
|
|
||||||
|
//then
|
||||||
|
assertThat(count, is(1L));
|
||||||
|
assertThat(entityList, is(notNullValue()));
|
||||||
|
assertThat(entityList.size(), is(1));
|
||||||
|
assertThat(entityList.get(0).getEmail(), is(settingEntity1.getEmail()));
|
||||||
|
}
|
||||||
|
}
|
18
src/test/resources/dynamic-settings-test.xml
Normal file
18
src/test/resources/dynamic-settings-test.xml
Normal file
@ -0,0 +1,18 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<beans xmlns="http://www.springframework.org/schema/beans"
|
||||||
|
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
|
||||||
|
xmlns:elasticsearch="http://www.springframework.org/schema/data/elasticsearch"
|
||||||
|
xsi:schemaLocation="http://www.springframework.org/schema/data/elasticsearch http://www.springframework.org/schema/data/elasticsearch/spring-elasticsearch-1.0.xsd
|
||||||
|
http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.1.xsd">
|
||||||
|
|
||||||
|
<import resource="infrastructure.xml"/>
|
||||||
|
|
||||||
|
<bean name="elasticsearchTemplate"
|
||||||
|
class="org.springframework.data.elasticsearch.core.ElasticsearchTemplate">
|
||||||
|
<constructor-arg name="client" ref="client"/>
|
||||||
|
</bean>
|
||||||
|
|
||||||
|
<elasticsearch:repositories
|
||||||
|
base-package="org.springframework.data.elasticsearch.repositories.setting"/>
|
||||||
|
|
||||||
|
</beans>
|
14
src/test/resources/settings/test-settings.json
Normal file
14
src/test/resources/settings/test-settings.json
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
{
|
||||||
|
"index": {
|
||||||
|
"number_of_shards" : "1",
|
||||||
|
"number_of_replicas" : "0",
|
||||||
|
"analysis" :{
|
||||||
|
"analyzer": {
|
||||||
|
"emailAnalyzer": {
|
||||||
|
"type" : "custom",
|
||||||
|
"tokenizer" : "uax_url_email"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user