|  | @@ -0,0 +1,350 @@
 | 
	
		
			
				|  |  | +/*
 | 
	
		
			
				|  |  | + * Licensed to Elasticsearch under one or more contributor
 | 
	
		
			
				|  |  | + * license agreements. See the NOTICE file distributed with
 | 
	
		
			
				|  |  | + * this work for additional information regarding copyright
 | 
	
		
			
				|  |  | + * ownership. Elasticsearch 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.snapshots;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +import org.elasticsearch.Version;
 | 
	
		
			
				|  |  | +import org.elasticsearch.action.ActionListener;
 | 
	
		
			
				|  |  | +import org.elasticsearch.action.support.PlainActionFuture;
 | 
	
		
			
				|  |  | +import org.elasticsearch.cluster.ClusterState;
 | 
	
		
			
				|  |  | +import org.elasticsearch.cluster.metadata.IndexMetadata;
 | 
	
		
			
				|  |  | +import org.elasticsearch.cluster.metadata.Metadata;
 | 
	
		
			
				|  |  | +import org.elasticsearch.cluster.node.DiscoveryNode;
 | 
	
		
			
				|  |  | +import org.elasticsearch.cluster.node.DiscoveryNodeRole;
 | 
	
		
			
				|  |  | +import org.elasticsearch.cluster.node.DiscoveryNodes;
 | 
	
		
			
				|  |  | +import org.elasticsearch.cluster.routing.IndexRoutingTable;
 | 
	
		
			
				|  |  | +import org.elasticsearch.cluster.routing.IndexShardRoutingTable;
 | 
	
		
			
				|  |  | +import org.elasticsearch.cluster.routing.RecoverySource;
 | 
	
		
			
				|  |  | +import org.elasticsearch.cluster.routing.RerouteService;
 | 
	
		
			
				|  |  | +import org.elasticsearch.cluster.routing.RoutingTable;
 | 
	
		
			
				|  |  | +import org.elasticsearch.cluster.routing.ShardRouting;
 | 
	
		
			
				|  |  | +import org.elasticsearch.cluster.routing.ShardRoutingState;
 | 
	
		
			
				|  |  | +import org.elasticsearch.cluster.routing.TestShardRouting;
 | 
	
		
			
				|  |  | +import org.elasticsearch.cluster.service.ClusterApplier;
 | 
	
		
			
				|  |  | +import org.elasticsearch.cluster.service.ClusterService;
 | 
	
		
			
				|  |  | +import org.elasticsearch.common.Priority;
 | 
	
		
			
				|  |  | +import org.elasticsearch.common.UUIDs;
 | 
	
		
			
				|  |  | +import org.elasticsearch.common.settings.Settings;
 | 
	
		
			
				|  |  | +import org.elasticsearch.index.Index;
 | 
	
		
			
				|  |  | +import org.elasticsearch.index.shard.ShardId;
 | 
	
		
			
				|  |  | +import org.elasticsearch.index.snapshots.IndexShardSnapshotStatus;
 | 
	
		
			
				|  |  | +import org.elasticsearch.repositories.FilterRepository;
 | 
	
		
			
				|  |  | +import org.elasticsearch.repositories.IndexId;
 | 
	
		
			
				|  |  | +import org.elasticsearch.repositories.RepositoriesService;
 | 
	
		
			
				|  |  | +import org.elasticsearch.repositories.Repository;
 | 
	
		
			
				|  |  | +import org.elasticsearch.test.ClusterServiceUtils;
 | 
	
		
			
				|  |  | +import org.elasticsearch.test.ESTestCase;
 | 
	
		
			
				|  |  | +import org.elasticsearch.threadpool.TestThreadPool;
 | 
	
		
			
				|  |  | +import org.elasticsearch.threadpool.ThreadPool;
 | 
	
		
			
				|  |  | +import org.elasticsearch.threadpool.ThreadPoolStats;
 | 
	
		
			
				|  |  | +import org.junit.After;
 | 
	
		
			
				|  |  | +import org.junit.Before;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +import java.util.Collections;
 | 
	
		
			
				|  |  | +import java.util.Locale;
 | 
	
		
			
				|  |  | +import java.util.Map;
 | 
	
		
			
				|  |  | +import java.util.concurrent.ConcurrentHashMap;
 | 
	
		
			
				|  |  | +import java.util.concurrent.CountDownLatch;
 | 
	
		
			
				|  |  | +import java.util.concurrent.TimeUnit;
 | 
	
		
			
				|  |  | +import java.util.concurrent.atomic.AtomicInteger;
 | 
	
		
			
				|  |  | +import java.util.function.Function;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +import static org.elasticsearch.cluster.metadata.IndexMetadata.SETTING_CREATION_DATE;
 | 
	
		
			
				|  |  | +import static org.elasticsearch.cluster.metadata.IndexMetadata.SETTING_NUMBER_OF_REPLICAS;
 | 
	
		
			
				|  |  | +import static org.elasticsearch.cluster.metadata.IndexMetadata.SETTING_NUMBER_OF_SHARDS;
 | 
	
		
			
				|  |  | +import static org.elasticsearch.cluster.metadata.IndexMetadata.SETTING_VERSION_CREATED;
 | 
	
		
			
				|  |  | +import static org.elasticsearch.snapshots.InternalSnapshotsInfoService.INTERNAL_SNAPSHOT_INFO_MAX_CONCURRENT_FETCHES_SETTING;
 | 
	
		
			
				|  |  | +import static org.hamcrest.Matchers.allOf;
 | 
	
		
			
				|  |  | +import static org.hamcrest.Matchers.equalTo;
 | 
	
		
			
				|  |  | +import static org.hamcrest.Matchers.greaterThanOrEqualTo;
 | 
	
		
			
				|  |  | +import static org.hamcrest.Matchers.is;
 | 
	
		
			
				|  |  | +import static org.hamcrest.Matchers.lessThan;
 | 
	
		
			
				|  |  | +import static org.hamcrest.Matchers.notNullValue;
 | 
	
		
			
				|  |  | +import static org.hamcrest.Matchers.nullValue;
 | 
	
		
			
				|  |  | +import static org.mockito.Matchers.any;
 | 
	
		
			
				|  |  | +import static org.mockito.Matchers.anyString;
 | 
	
		
			
				|  |  | +import static org.mockito.Mockito.doAnswer;
 | 
	
		
			
				|  |  | +import static org.mockito.Mockito.mock;
 | 
	
		
			
				|  |  | +import static org.mockito.Mockito.times;
 | 
	
		
			
				|  |  | +import static org.mockito.Mockito.verify;
 | 
	
		
			
				|  |  | +import static org.mockito.Mockito.when;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +public class InternalSnapshotsInfoServiceTests extends ESTestCase {
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +    private TestThreadPool threadPool;
 | 
	
		
			
				|  |  | +    private ClusterService clusterService;
 | 
	
		
			
				|  |  | +    private RepositoriesService repositoriesService;
 | 
	
		
			
				|  |  | +    private RerouteService rerouteService;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +    @Before
 | 
	
		
			
				|  |  | +    @Override
 | 
	
		
			
				|  |  | +    public void setUp() throws Exception {
 | 
	
		
			
				|  |  | +        super.setUp();
 | 
	
		
			
				|  |  | +        threadPool = new TestThreadPool(getTestName());
 | 
	
		
			
				|  |  | +        clusterService = ClusterServiceUtils.createClusterService(threadPool);
 | 
	
		
			
				|  |  | +        repositoriesService = mock(RepositoriesService.class);
 | 
	
		
			
				|  |  | +        rerouteService = mock(RerouteService.class);
 | 
	
		
			
				|  |  | +        doAnswer(invocation -> {
 | 
	
		
			
				|  |  | +            @SuppressWarnings("unchecked")
 | 
	
		
			
				|  |  | +            final ActionListener<ClusterState> listener = (ActionListener<ClusterState>) invocation.getArguments()[2];
 | 
	
		
			
				|  |  | +            listener.onResponse(clusterService.state());
 | 
	
		
			
				|  |  | +            return null;
 | 
	
		
			
				|  |  | +        }).when(rerouteService).reroute(anyString(), any(Priority.class), any());
 | 
	
		
			
				|  |  | +    }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +    @After
 | 
	
		
			
				|  |  | +    @Override
 | 
	
		
			
				|  |  | +    public void tearDown() throws Exception {
 | 
	
		
			
				|  |  | +        super.tearDown();
 | 
	
		
			
				|  |  | +        final boolean terminated = terminate(threadPool);
 | 
	
		
			
				|  |  | +        assert terminated;
 | 
	
		
			
				|  |  | +        clusterService.close();
 | 
	
		
			
				|  |  | +    }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +    public void testSnapshotShardSizes() throws Exception {
 | 
	
		
			
				|  |  | +        final int maxConcurrentFetches = randomIntBetween(1, 10);
 | 
	
		
			
				|  |  | +        final InternalSnapshotsInfoService snapshotsInfoService =
 | 
	
		
			
				|  |  | +            new InternalSnapshotsInfoService(Settings.builder()
 | 
	
		
			
				|  |  | +                .put(INTERNAL_SNAPSHOT_INFO_MAX_CONCURRENT_FETCHES_SETTING.getKey(), maxConcurrentFetches)
 | 
	
		
			
				|  |  | +                .build(), clusterService, () -> repositoriesService, () -> rerouteService);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +        final int numberOfShards = randomIntBetween(1, 50);
 | 
	
		
			
				|  |  | +        final String indexName = randomAlphaOfLength(10).toLowerCase(Locale.ROOT);
 | 
	
		
			
				|  |  | +        final long[] expectedShardSizes = new long[numberOfShards];
 | 
	
		
			
				|  |  | +        for (int i = 0; i < expectedShardSizes.length; i++) {
 | 
	
		
			
				|  |  | +            expectedShardSizes[i] = randomNonNegativeLong();
 | 
	
		
			
				|  |  | +        }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +        final AtomicInteger getShardSnapshotStatusCount = new AtomicInteger(0);
 | 
	
		
			
				|  |  | +        final CountDownLatch latch = new CountDownLatch(1);
 | 
	
		
			
				|  |  | +        final Repository mockRepository = new FilterRepository(mock(Repository.class)) {
 | 
	
		
			
				|  |  | +            @Override
 | 
	
		
			
				|  |  | +            public IndexShardSnapshotStatus getShardSnapshotStatus(SnapshotId snapshotId, IndexId indexId, ShardId shardId) {
 | 
	
		
			
				|  |  | +                try {
 | 
	
		
			
				|  |  | +                    assertThat(indexId.getName(), equalTo(indexName));
 | 
	
		
			
				|  |  | +                    assertThat(shardId.id(), allOf(greaterThanOrEqualTo(0), lessThan(numberOfShards)));
 | 
	
		
			
				|  |  | +                    latch.await();
 | 
	
		
			
				|  |  | +                    getShardSnapshotStatusCount.incrementAndGet();
 | 
	
		
			
				|  |  | +                    return IndexShardSnapshotStatus.newDone(0L, 0L, 0, 0, 0L, expectedShardSizes[shardId.id()], null);
 | 
	
		
			
				|  |  | +                } catch (InterruptedException e) {
 | 
	
		
			
				|  |  | +                    throw new AssertionError(e);
 | 
	
		
			
				|  |  | +                }
 | 
	
		
			
				|  |  | +            }
 | 
	
		
			
				|  |  | +        };
 | 
	
		
			
				|  |  | +        when(repositoriesService.repository("_repo")).thenReturn(mockRepository);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +        applyClusterState("add-unassigned-shards", clusterState -> addUnassignedShards(clusterState, indexName, numberOfShards));
 | 
	
		
			
				|  |  | +        waitForMaxActiveGenericThreads(Math.min(numberOfShards, maxConcurrentFetches));
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +        if (randomBoolean()) {
 | 
	
		
			
				|  |  | +            applyClusterState("reapply-last-cluster-state-to-check-deduplication-works",
 | 
	
		
			
				|  |  | +                state -> ClusterState.builder(state).incrementVersion().build());
 | 
	
		
			
				|  |  | +        }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +        assertThat(snapshotsInfoService.numberOfUnknownSnapshotShardSizes(), equalTo(numberOfShards));
 | 
	
		
			
				|  |  | +        assertThat(snapshotsInfoService.numberOfKnownSnapshotShardSizes(), equalTo(0));
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +        latch.countDown();
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +        assertBusy(() -> {
 | 
	
		
			
				|  |  | +            assertThat(snapshotsInfoService.numberOfKnownSnapshotShardSizes(), equalTo(numberOfShards));
 | 
	
		
			
				|  |  | +            assertThat(snapshotsInfoService.numberOfUnknownSnapshotShardSizes(), equalTo(0));
 | 
	
		
			
				|  |  | +            assertThat(snapshotsInfoService.numberOfFailedSnapshotShardSizes(), equalTo(0));
 | 
	
		
			
				|  |  | +        });
 | 
	
		
			
				|  |  | +        verify(rerouteService, times(numberOfShards)).reroute(anyString(), any(Priority.class), any());
 | 
	
		
			
				|  |  | +        assertThat(getShardSnapshotStatusCount.get(), equalTo(numberOfShards));
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +        for (int i = 0; i < numberOfShards; i++) {
 | 
	
		
			
				|  |  | +            final ShardRouting shardRouting = clusterService.state().routingTable().index(indexName).shard(i).primaryShard();
 | 
	
		
			
				|  |  | +            assertThat(snapshotsInfoService.snapshotShardSizes().getShardSize(shardRouting), equalTo(expectedShardSizes[i]));
 | 
	
		
			
				|  |  | +        }
 | 
	
		
			
				|  |  | +    }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +    public void testErroneousSnapshotShardSizes() throws Exception {
 | 
	
		
			
				|  |  | +        final InternalSnapshotsInfoService snapshotsInfoService =
 | 
	
		
			
				|  |  | +            new InternalSnapshotsInfoService(Settings.builder()
 | 
	
		
			
				|  |  | +                .put(INTERNAL_SNAPSHOT_INFO_MAX_CONCURRENT_FETCHES_SETTING.getKey(), randomIntBetween(1, 10))
 | 
	
		
			
				|  |  | +                .build(), clusterService, () -> repositoriesService, () -> rerouteService);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +        final Map<InternalSnapshotsInfoService.SnapshotShard, Boolean> results = new ConcurrentHashMap<>();
 | 
	
		
			
				|  |  | +        final Repository mockRepository = new FilterRepository(mock(Repository.class)) {
 | 
	
		
			
				|  |  | +            @Override
 | 
	
		
			
				|  |  | +            public IndexShardSnapshotStatus getShardSnapshotStatus(SnapshotId snapshotId, IndexId indexId, ShardId shardId) {
 | 
	
		
			
				|  |  | +                final InternalSnapshotsInfoService.SnapshotShard snapshotShard =
 | 
	
		
			
				|  |  | +                    new InternalSnapshotsInfoService.SnapshotShard(new Snapshot("_repo", snapshotId), indexId, shardId);
 | 
	
		
			
				|  |  | +                if (randomBoolean()) {
 | 
	
		
			
				|  |  | +                    results.put(snapshotShard, Boolean.FALSE);
 | 
	
		
			
				|  |  | +                    throw new SnapshotException(snapshotShard.snapshot(), "simulated");
 | 
	
		
			
				|  |  | +                } else {
 | 
	
		
			
				|  |  | +                    results.put(snapshotShard, Boolean.TRUE);
 | 
	
		
			
				|  |  | +                    return IndexShardSnapshotStatus.newDone(0L, 0L, 0, 0, 0L, randomNonNegativeLong(), null);
 | 
	
		
			
				|  |  | +                }
 | 
	
		
			
				|  |  | +            }
 | 
	
		
			
				|  |  | +        };
 | 
	
		
			
				|  |  | +        when(repositoriesService.repository("_repo")).thenReturn(mockRepository);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +        final int maxShardsToCreate = scaledRandomIntBetween(10, 500);
 | 
	
		
			
				|  |  | +        final Thread addSnapshotRestoreIndicesThread = new Thread(() -> {
 | 
	
		
			
				|  |  | +            int remainingShards = maxShardsToCreate;
 | 
	
		
			
				|  |  | +            while (remainingShards > 0) {
 | 
	
		
			
				|  |  | +                final String indexName = randomAlphaOfLength(10).toLowerCase(Locale.ROOT);
 | 
	
		
			
				|  |  | +                final int numberOfShards = randomIntBetween(1, remainingShards);
 | 
	
		
			
				|  |  | +                try {
 | 
	
		
			
				|  |  | +                    applyClusterState("add-more-unassigned-shards-for-" + indexName,
 | 
	
		
			
				|  |  | +                        clusterState -> addUnassignedShards(clusterState, indexName, numberOfShards));
 | 
	
		
			
				|  |  | +                } catch (Exception e) {
 | 
	
		
			
				|  |  | +                    throw new AssertionError(e);
 | 
	
		
			
				|  |  | +                } finally {
 | 
	
		
			
				|  |  | +                    remainingShards -= numberOfShards;
 | 
	
		
			
				|  |  | +                }
 | 
	
		
			
				|  |  | +            }
 | 
	
		
			
				|  |  | +        });
 | 
	
		
			
				|  |  | +        addSnapshotRestoreIndicesThread.start();
 | 
	
		
			
				|  |  | +        addSnapshotRestoreIndicesThread.join();
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +        assertBusy(() -> {
 | 
	
		
			
				|  |  | +            assertThat(snapshotsInfoService.numberOfKnownSnapshotShardSizes(),
 | 
	
		
			
				|  |  | +                equalTo((int) results.values().stream().filter(result -> result.equals(Boolean.TRUE)).count()));
 | 
	
		
			
				|  |  | +            assertThat(snapshotsInfoService.numberOfFailedSnapshotShardSizes(),
 | 
	
		
			
				|  |  | +                equalTo((int) results.values().stream().filter(result -> result.equals(Boolean.FALSE)).count()));
 | 
	
		
			
				|  |  | +            assertThat(snapshotsInfoService.numberOfUnknownSnapshotShardSizes(), equalTo(0));
 | 
	
		
			
				|  |  | +        });
 | 
	
		
			
				|  |  | +    }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +    public void testNoLongerMaster() throws Exception {
 | 
	
		
			
				|  |  | +        final InternalSnapshotsInfoService snapshotsInfoService =
 | 
	
		
			
				|  |  | +            new InternalSnapshotsInfoService(Settings.EMPTY, clusterService, () -> repositoriesService, () -> rerouteService);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +        final Repository mockRepository = new FilterRepository(mock(Repository.class)) {
 | 
	
		
			
				|  |  | +            @Override
 | 
	
		
			
				|  |  | +            public IndexShardSnapshotStatus getShardSnapshotStatus(SnapshotId snapshotId, IndexId indexId, ShardId shardId) {
 | 
	
		
			
				|  |  | +                return IndexShardSnapshotStatus.newDone(0L, 0L, 0, 0, 0L, randomNonNegativeLong(), null);
 | 
	
		
			
				|  |  | +            }
 | 
	
		
			
				|  |  | +        };
 | 
	
		
			
				|  |  | +        when(repositoriesService.repository("_repo")).thenReturn(mockRepository);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +        for (int i = 0; i < randomIntBetween(1, 10); i++) {
 | 
	
		
			
				|  |  | +            final String indexName = randomAlphaOfLength(10).toLowerCase(Locale.ROOT);
 | 
	
		
			
				|  |  | +            final int nbShards =  randomIntBetween(1, 5);
 | 
	
		
			
				|  |  | +            applyClusterState("restore-indices-when-master-" + indexName,
 | 
	
		
			
				|  |  | +                clusterState -> addUnassignedShards(clusterState, indexName, nbShards));
 | 
	
		
			
				|  |  | +        }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +        applyClusterState("demote-current-master", this::demoteMasterNode);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +        for (int i = 0; i < randomIntBetween(1, 10); i++) {
 | 
	
		
			
				|  |  | +            final String indexName = randomAlphaOfLength(10).toLowerCase(Locale.ROOT);
 | 
	
		
			
				|  |  | +            final int nbShards =  randomIntBetween(1, 5);
 | 
	
		
			
				|  |  | +            applyClusterState("restore-indices-when-no-longer-master-" + indexName,
 | 
	
		
			
				|  |  | +                clusterState -> addUnassignedShards(clusterState, indexName, nbShards));
 | 
	
		
			
				|  |  | +        }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +        assertBusy(() -> {
 | 
	
		
			
				|  |  | +            assertThat(snapshotsInfoService.numberOfKnownSnapshotShardSizes(), equalTo(0));
 | 
	
		
			
				|  |  | +            assertThat(snapshotsInfoService.numberOfUnknownSnapshotShardSizes(), equalTo(0));
 | 
	
		
			
				|  |  | +            assertThat(snapshotsInfoService.numberOfFailedSnapshotShardSizes(), equalTo(0));
 | 
	
		
			
				|  |  | +        });
 | 
	
		
			
				|  |  | +    }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +    private void applyClusterState(final String reason, final Function<ClusterState, ClusterState> applier) {
 | 
	
		
			
				|  |  | +        PlainActionFuture.get(future -> clusterService.getClusterApplierService().onNewClusterState(reason,
 | 
	
		
			
				|  |  | +            () -> applier.apply(clusterService.state()),
 | 
	
		
			
				|  |  | +            new ClusterApplier.ClusterApplyListener() {
 | 
	
		
			
				|  |  | +                @Override
 | 
	
		
			
				|  |  | +                public void onSuccess(String source) {
 | 
	
		
			
				|  |  | +                    future.onResponse(source);
 | 
	
		
			
				|  |  | +                }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +                @Override
 | 
	
		
			
				|  |  | +                public void onFailure(String source, Exception e) {
 | 
	
		
			
				|  |  | +                    future.onFailure(e);
 | 
	
		
			
				|  |  | +                }
 | 
	
		
			
				|  |  | +            })
 | 
	
		
			
				|  |  | +        );
 | 
	
		
			
				|  |  | +    }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +    private void waitForMaxActiveGenericThreads(final int nbActive) throws Exception {
 | 
	
		
			
				|  |  | +        assertBusy(() -> {
 | 
	
		
			
				|  |  | +            final ThreadPoolStats threadPoolStats = clusterService.getClusterApplierService().threadPool().stats();
 | 
	
		
			
				|  |  | +            ThreadPoolStats.Stats generic = null;
 | 
	
		
			
				|  |  | +            for (ThreadPoolStats.Stats threadPoolStat : threadPoolStats) {
 | 
	
		
			
				|  |  | +                if (ThreadPool.Names.GENERIC.equals(threadPoolStat.getName())) {
 | 
	
		
			
				|  |  | +                    generic = threadPoolStat;
 | 
	
		
			
				|  |  | +                }
 | 
	
		
			
				|  |  | +            }
 | 
	
		
			
				|  |  | +            assertThat(generic, notNullValue());
 | 
	
		
			
				|  |  | +            assertThat(generic.getActive(), equalTo(nbActive));
 | 
	
		
			
				|  |  | +        }, 30L, TimeUnit.SECONDS);
 | 
	
		
			
				|  |  | +    }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +    private ClusterState addUnassignedShards(final ClusterState currentState, String indexName, int numberOfShards) {
 | 
	
		
			
				|  |  | +        assertThat(currentState.metadata().hasIndex(indexName), is(false));
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +        final Metadata.Builder metadata = Metadata.builder(currentState.metadata())
 | 
	
		
			
				|  |  | +            .put(IndexMetadata.builder(indexName)
 | 
	
		
			
				|  |  | +                .settings(Settings.builder()
 | 
	
		
			
				|  |  | +                    .put(SETTING_VERSION_CREATED, Version.CURRENT)
 | 
	
		
			
				|  |  | +                    .put(SETTING_NUMBER_OF_SHARDS, numberOfShards)
 | 
	
		
			
				|  |  | +                    .put(SETTING_NUMBER_OF_REPLICAS, randomIntBetween(0, 1))
 | 
	
		
			
				|  |  | +                    .put(SETTING_CREATION_DATE, System.currentTimeMillis()))
 | 
	
		
			
				|  |  | +                .build(), true)
 | 
	
		
			
				|  |  | +            .generateClusterUuidIfNeeded();
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +        final RecoverySource.SnapshotRecoverySource recoverySource = new RecoverySource.SnapshotRecoverySource(
 | 
	
		
			
				|  |  | +            UUIDs.randomBase64UUID(random()),
 | 
	
		
			
				|  |  | +            new Snapshot("_repo", new SnapshotId(randomAlphaOfLength(5), UUIDs.randomBase64UUID(random()))),
 | 
	
		
			
				|  |  | +            Version.CURRENT,
 | 
	
		
			
				|  |  | +            new IndexId(indexName, UUIDs.randomBase64UUID(random()))
 | 
	
		
			
				|  |  | +        );
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +        final Index index = metadata.get(indexName).getIndex();
 | 
	
		
			
				|  |  | +        final IndexRoutingTable.Builder indexRoutingTable = IndexRoutingTable.builder(index);
 | 
	
		
			
				|  |  | +        for (int primary = 0; primary < numberOfShards; primary++) {
 | 
	
		
			
				|  |  | +            final ShardId shardId = new ShardId(index, primary);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +            final IndexShardRoutingTable.Builder indexShards = new IndexShardRoutingTable.Builder(shardId);
 | 
	
		
			
				|  |  | +            indexShards.addShard(TestShardRouting.newShardRouting(shardId, null, true, ShardRoutingState.UNASSIGNED, recoverySource));
 | 
	
		
			
				|  |  | +            for (int replica = 0; replica < metadata.get(indexName).getNumberOfReplicas(); replica++) {
 | 
	
		
			
				|  |  | +                indexShards.addShard(TestShardRouting.newShardRouting(shardId, null, false, ShardRoutingState.UNASSIGNED,
 | 
	
		
			
				|  |  | +                    RecoverySource.PeerRecoverySource.INSTANCE));
 | 
	
		
			
				|  |  | +            }
 | 
	
		
			
				|  |  | +            indexRoutingTable.addIndexShard(indexShards.build());
 | 
	
		
			
				|  |  | +        }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +        final RoutingTable.Builder routingTable = RoutingTable.builder(currentState.routingTable());
 | 
	
		
			
				|  |  | +        routingTable.add(indexRoutingTable.build());
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +        return ClusterState.builder(currentState)
 | 
	
		
			
				|  |  | +            .routingTable(routingTable.build())
 | 
	
		
			
				|  |  | +            .metadata(metadata)
 | 
	
		
			
				|  |  | +            .build();
 | 
	
		
			
				|  |  | +    }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +    private ClusterState demoteMasterNode(final ClusterState currentState) {
 | 
	
		
			
				|  |  | +        final DiscoveryNode node = new DiscoveryNode("other", ESTestCase.buildNewFakeTransportAddress(), Collections.emptyMap(),
 | 
	
		
			
				|  |  | +            DiscoveryNodeRole.BUILT_IN_ROLES, Version.CURRENT);
 | 
	
		
			
				|  |  | +        assertThat(currentState.nodes().get(node.getId()), nullValue());
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +        return ClusterState.builder(currentState)
 | 
	
		
			
				|  |  | +            .nodes(DiscoveryNodes.builder(currentState.nodes())
 | 
	
		
			
				|  |  | +                .add(node)
 | 
	
		
			
				|  |  | +                .masterNodeId(node.getId()))
 | 
	
		
			
				|  |  | +            .build();
 | 
	
		
			
				|  |  | +    }
 | 
	
		
			
				|  |  | +}
 |