diff --git a/server/src/main/java/org/elasticsearch/common/settings/IndexScopedSettings.java b/server/src/main/java/org/elasticsearch/common/settings/IndexScopedSettings.java index 4a9ab1ad663..4d064536509 100644 --- a/server/src/main/java/org/elasticsearch/common/settings/IndexScopedSettings.java +++ b/server/src/main/java/org/elasticsearch/common/settings/IndexScopedSettings.java @@ -170,6 +170,7 @@ public final class IndexScopedSettings extends AbstractScopedSettings { IndexSettings.DEFAULT_PIPELINE, IndexSettings.FINAL_PIPELINE, MetaDataIndexStateService.VERIFIED_BEFORE_CLOSE_SETTING, + IndexSettings.ON_HEAP_ID_TERMS_INDEX, // validate that built-in similarities don't get redefined Setting.groupSetting("index.similarity.", (s) -> { diff --git a/server/src/main/java/org/elasticsearch/index/IndexSettings.java b/server/src/main/java/org/elasticsearch/index/IndexSettings.java index 46dbddec022..2137c8d02bf 100644 --- a/server/src/main/java/org/elasticsearch/index/IndexSettings.java +++ b/server/src/main/java/org/elasticsearch/index/IndexSettings.java @@ -84,6 +84,9 @@ public final class IndexSettings { "[true, false, checksum] but was: " + s); } }, Property.IndexScope); + // This setting is undocumented as it is considered as an escape hatch. + public static final Setting ON_HEAP_ID_TERMS_INDEX = + Setting.boolSetting("index.force_memory_id_terms_dictionary", false, Property.IndexScope); /** * Index setting describing the maximum value of from + size on a query. diff --git a/server/src/main/java/org/elasticsearch/index/engine/InternalEngine.java b/server/src/main/java/org/elasticsearch/index/engine/InternalEngine.java index 379642febcf..365980f7e08 100644 --- a/server/src/main/java/org/elasticsearch/index/engine/InternalEngine.java +++ b/server/src/main/java/org/elasticsearch/index/engine/InternalEngine.java @@ -2211,22 +2211,24 @@ public class InternalEngine extends Engine { } } - static Map getReaderAttributes(Directory directory) { + static Map getReaderAttributes(Directory directory, IndexSettings indexSettings) { Directory unwrap = FilterDirectory.unwrap(directory); boolean defaultOffHeap = FsDirectoryFactory.isHybridFs(unwrap) || unwrap instanceof MMapDirectory; - HashMap map = new HashMap(2); - map.put(BlockTreeTermsReader.FST_MODE_KEY, // if we are using MMAP for term dics we force all off heap unless it's the ID field - defaultOffHeap ? FSTLoadMode.OFF_HEAP.name() : FSTLoadMode.ON_HEAP.name()); - map.put(BlockTreeTermsReader.FST_MODE_KEY + "." + IdFieldMapper.NAME, // always force ID field on-heap for fast updates - FSTLoadMode.ON_HEAP.name()); - return Collections.unmodifiableMap(map); + Map attributes = new HashMap<>(); + attributes.put(BlockTreeTermsReader.FST_MODE_KEY, defaultOffHeap ? FSTLoadMode.OFF_HEAP.name() : FSTLoadMode.ON_HEAP.name()); + if (IndexSettings.ON_HEAP_ID_TERMS_INDEX.exists(indexSettings.getSettings())) { + final boolean idOffHeap = IndexSettings.ON_HEAP_ID_TERMS_INDEX.get(indexSettings.getSettings()) == false; + attributes.put(BlockTreeTermsReader.FST_MODE_KEY + "." + IdFieldMapper.NAME, + idOffHeap ? FSTLoadMode.OFF_HEAP.name() : FSTLoadMode.ON_HEAP.name()); + } + return Collections.unmodifiableMap(attributes); } private IndexWriterConfig getIndexWriterConfig() { final IndexWriterConfig iwc = new IndexWriterConfig(engineConfig.getAnalyzer()); iwc.setCommitOnClose(false); // we by default don't commit on close iwc.setOpenMode(IndexWriterConfig.OpenMode.APPEND); - iwc.setReaderAttributes(getReaderAttributes(store.directory())); + iwc.setReaderAttributes(getReaderAttributes(store.directory(), engineConfig.getIndexSettings())); iwc.setIndexDeletionPolicy(combinedDeletionPolicy); // with tests.verbose, lucene sets this up: plumb to align with filesystem stream boolean verbose = false; diff --git a/server/src/test/java/org/elasticsearch/index/engine/InternalEngineTests.java b/server/src/test/java/org/elasticsearch/index/engine/InternalEngineTests.java index 2fdd1f4c99d..33562992839 100644 --- a/server/src/test/java/org/elasticsearch/index/engine/InternalEngineTests.java +++ b/server/src/test/java/org/elasticsearch/index/engine/InternalEngineTests.java @@ -5886,32 +5886,25 @@ public class InternalEngineTests extends EngineTestCase { } public void testGetReaderAttributes() throws IOException { + Settings.Builder settingsBuilder = Settings.builder() + .put(IndexMetaData.SETTING_VERSION_CREATED, Version.CURRENT); + Settings settings = settingsBuilder.build(); + IndexSettings indexSettings = IndexSettingsModule.newIndexSettings("foo", settings); + try(BaseDirectoryWrapper dir = newFSDirectory(createTempDir())) { Directory unwrap = FilterDirectory.unwrap(dir); boolean isMMap = unwrap instanceof MMapDirectory; - Map readerAttributes = InternalEngine.getReaderAttributes(dir); - assertEquals(2, readerAttributes.size()); - assertEquals("ON_HEAP", readerAttributes.get("blocktree.terms.fst._id")); - if (isMMap) { - assertEquals("OFF_HEAP", readerAttributes.get("blocktree.terms.fst")); - } else { - assertEquals("ON_HEAP", readerAttributes.get("blocktree.terms.fst")); - } + Map readerAttributes = InternalEngine.getReaderAttributes(dir, indexSettings); + assertEquals(Collections.singletonMap("blocktree.terms.fst", isMMap ? "OFF_HEAP" : "ON_HEAP"), readerAttributes); } try(MMapDirectory dir = new MMapDirectory(createTempDir())) { Map readerAttributes = InternalEngine.getReaderAttributes(randomBoolean() ? dir : - new MockDirectoryWrapper(random(), dir)); - assertEquals(2, readerAttributes.size()); - assertEquals("ON_HEAP", readerAttributes.get("blocktree.terms.fst._id")); - assertEquals("OFF_HEAP", readerAttributes.get("blocktree.terms.fst")); + new MockDirectoryWrapper(random(), dir), indexSettings); + assertEquals(Collections.singletonMap("blocktree.terms.fst", "OFF_HEAP"), readerAttributes); } - Settings.Builder settingsBuilder = Settings.builder() - .put(IndexMetaData.SETTING_VERSION_CREATED, Version.CURRENT); - Settings settings = settingsBuilder.build(); - IndexSettings indexSettings = IndexSettingsModule.newIndexSettings("foo", settings); FsDirectoryFactory service = new FsDirectoryFactory(); Path tempDir = createTempDir().resolve(indexSettings.getUUID()).resolve("0"); ShardPath path = new ShardPath(false, tempDir, tempDir, new ShardId(indexSettings.getIndex(), 0)); @@ -5919,25 +5912,50 @@ public class InternalEngineTests extends EngineTestCase { Map readerAttributes = InternalEngine.getReaderAttributes(randomBoolean() ? directory : - new MockDirectoryWrapper(random(), directory)); - assertEquals(2, readerAttributes.size()); + new MockDirectoryWrapper(random(), directory), indexSettings); + assertEquals(1, readerAttributes.size()); switch (IndexModule.defaultStoreType(true)) { case HYBRIDFS: case MMAPFS: - assertEquals("ON_HEAP", readerAttributes.get("blocktree.terms.fst._id")); assertEquals("OFF_HEAP", readerAttributes.get("blocktree.terms.fst")); break; case NIOFS: case SIMPLEFS: case FS: - assertEquals("ON_HEAP", readerAttributes.get("blocktree.terms.fst._id")); assertEquals("ON_HEAP", readerAttributes.get("blocktree.terms.fst")); break; default: fail("unknownw type"); } } + + settingsBuilder.put(IndexSettings.ON_HEAP_ID_TERMS_INDEX.getKey(), true); + settings = settingsBuilder.build(); + indexSettings = IndexSettingsModule.newIndexSettings("foo", settings); + try (Directory directory = service.newDirectory(indexSettings, path)) { + + Map readerAttributes = + InternalEngine.getReaderAttributes(randomBoolean() ? directory : + new MockDirectoryWrapper(random(), directory), indexSettings); + assertEquals(2, readerAttributes.size()); + + switch (IndexModule.defaultStoreType(true)) { + case HYBRIDFS: + case MMAPFS: + assertEquals("OFF_HEAP", readerAttributes.get("blocktree.terms.fst")); + assertEquals("ON_HEAP", readerAttributes.get("blocktree.terms.fst._id")); + break; + case NIOFS: + case SIMPLEFS: + case FS: + assertEquals("ON_HEAP", readerAttributes.get("blocktree.terms.fst")); + assertEquals("ON_HEAP", readerAttributes.get("blocktree.terms.fst._id")); + break; + default: + fail("unknownw type"); + } + } } public void testPruneAwayDeletedButRetainedIds() throws Exception { diff --git a/test/framework/src/main/java/org/elasticsearch/test/ESIntegTestCase.java b/test/framework/src/main/java/org/elasticsearch/test/ESIntegTestCase.java index 4fdbdeba388..4523961fd37 100644 --- a/test/framework/src/main/java/org/elasticsearch/test/ESIntegTestCase.java +++ b/test/framework/src/main/java/org/elasticsearch/test/ESIntegTestCase.java @@ -457,10 +457,14 @@ public abstract class ESIntegTestCase extends ESTestCase { RandomNumbers.randomIntBetween(random, 1, 15) + "ms"); } - if (randomBoolean()) { + if (random.nextBoolean()) { builder.put(Store.FORCE_RAM_TERM_DICT.getKey(), true); } + if (random.nextBoolean()) { + builder.put(IndexSettings.ON_HEAP_ID_TERMS_INDEX.getKey(), random.nextBoolean()); + } + return builder; } diff --git a/x-pack/plugin/ccr/src/main/java/org/elasticsearch/xpack/ccr/action/TransportResumeFollowAction.java b/x-pack/plugin/ccr/src/main/java/org/elasticsearch/xpack/ccr/action/TransportResumeFollowAction.java index f0766342209..16dac60cf03 100644 --- a/x-pack/plugin/ccr/src/main/java/org/elasticsearch/xpack/ccr/action/TransportResumeFollowAction.java +++ b/x-pack/plugin/ccr/src/main/java/org/elasticsearch/xpack/ccr/action/TransportResumeFollowAction.java @@ -408,6 +408,7 @@ public class TransportResumeFollowAction extends TransportMasterNodeAction