|  | @@ -22,11 +22,15 @@ import org.elasticsearch.cluster.metadata.Metadata;
 | 
											
												
													
														|  |  import org.elasticsearch.cluster.node.DiscoveryNode;
 |  |  import org.elasticsearch.cluster.node.DiscoveryNode;
 | 
											
												
													
														|  |  import org.elasticsearch.cluster.node.DiscoveryNodeRole;
 |  |  import org.elasticsearch.cluster.node.DiscoveryNodeRole;
 | 
											
												
													
														|  |  import org.elasticsearch.cluster.node.DiscoveryNodes;
 |  |  import org.elasticsearch.cluster.node.DiscoveryNodes;
 | 
											
												
													
														|  | 
 |  | +import org.elasticsearch.cluster.routing.AllocationId;
 | 
											
												
													
														|  | 
 |  | +import org.elasticsearch.cluster.routing.IndexRoutingTable;
 | 
											
												
													
														|  |  import org.elasticsearch.cluster.routing.RecoverySource;
 |  |  import org.elasticsearch.cluster.routing.RecoverySource;
 | 
											
												
													
														|  |  import org.elasticsearch.cluster.routing.RoutingNode;
 |  |  import org.elasticsearch.cluster.routing.RoutingNode;
 | 
											
												
													
														|  |  import org.elasticsearch.cluster.routing.RoutingTable;
 |  |  import org.elasticsearch.cluster.routing.RoutingTable;
 | 
											
												
													
														|  |  import org.elasticsearch.cluster.routing.ShardRouting;
 |  |  import org.elasticsearch.cluster.routing.ShardRouting;
 | 
											
												
													
														|  |  import org.elasticsearch.cluster.routing.ShardRoutingState;
 |  |  import org.elasticsearch.cluster.routing.ShardRoutingState;
 | 
											
												
													
														|  | 
 |  | +import org.elasticsearch.cluster.routing.allocation.allocator.DesiredBalanceShardsAllocator;
 | 
											
												
													
														|  | 
 |  | +import org.elasticsearch.cluster.routing.allocation.allocator.ShardAssignment;
 | 
											
												
													
														|  |  import org.elasticsearch.cluster.routing.allocation.command.AbstractAllocateAllocationCommand;
 |  |  import org.elasticsearch.cluster.routing.allocation.command.AbstractAllocateAllocationCommand;
 | 
											
												
													
														|  |  import org.elasticsearch.cluster.routing.allocation.command.AllocateEmptyPrimaryAllocationCommand;
 |  |  import org.elasticsearch.cluster.routing.allocation.command.AllocateEmptyPrimaryAllocationCommand;
 | 
											
												
													
														|  |  import org.elasticsearch.cluster.routing.allocation.command.AllocateReplicaAllocationCommand;
 |  |  import org.elasticsearch.cluster.routing.allocation.command.AllocateReplicaAllocationCommand;
 | 
											
										
											
												
													
														|  | @@ -36,6 +40,7 @@ import org.elasticsearch.cluster.routing.allocation.command.CancelAllocationComm
 | 
											
												
													
														|  |  import org.elasticsearch.cluster.routing.allocation.command.MoveAllocationCommand;
 |  |  import org.elasticsearch.cluster.routing.allocation.command.MoveAllocationCommand;
 | 
											
												
													
														|  |  import org.elasticsearch.cluster.routing.allocation.decider.AllocationDeciders;
 |  |  import org.elasticsearch.cluster.routing.allocation.decider.AllocationDeciders;
 | 
											
												
													
														|  |  import org.elasticsearch.cluster.routing.allocation.decider.EnableAllocationDecider;
 |  |  import org.elasticsearch.cluster.routing.allocation.decider.EnableAllocationDecider;
 | 
											
												
													
														|  | 
 |  | +import org.elasticsearch.common.UUIDs;
 | 
											
												
													
														|  |  import org.elasticsearch.common.io.stream.BytesStreamOutput;
 |  |  import org.elasticsearch.common.io.stream.BytesStreamOutput;
 | 
											
												
													
														|  |  import org.elasticsearch.common.io.stream.NamedWriteableAwareStreamInput;
 |  |  import org.elasticsearch.common.io.stream.NamedWriteableAwareStreamInput;
 | 
											
												
													
														|  |  import org.elasticsearch.common.io.stream.NamedWriteableRegistry;
 |  |  import org.elasticsearch.common.io.stream.NamedWriteableRegistry;
 | 
											
										
											
												
													
														|  | @@ -44,6 +49,7 @@ import org.elasticsearch.common.network.NetworkModule;
 | 
											
												
													
														|  |  import org.elasticsearch.common.settings.Settings;
 |  |  import org.elasticsearch.common.settings.Settings;
 | 
											
												
													
														|  |  import org.elasticsearch.index.Index;
 |  |  import org.elasticsearch.index.Index;
 | 
											
												
													
														|  |  import org.elasticsearch.index.IndexNotFoundException;
 |  |  import org.elasticsearch.index.IndexNotFoundException;
 | 
											
												
													
														|  | 
 |  | +import org.elasticsearch.index.IndexVersion;
 | 
											
												
													
														|  |  import org.elasticsearch.index.shard.ShardId;
 |  |  import org.elasticsearch.index.shard.ShardId;
 | 
											
												
													
														|  |  import org.elasticsearch.index.shard.ShardNotFoundException;
 |  |  import org.elasticsearch.index.shard.ShardNotFoundException;
 | 
											
												
													
														|  |  import org.elasticsearch.snapshots.SnapshotShardSizeInfo;
 |  |  import org.elasticsearch.snapshots.SnapshotShardSizeInfo;
 | 
											
										
											
												
													
														|  | @@ -63,6 +69,7 @@ import static org.elasticsearch.cluster.routing.ShardRoutingState.INITIALIZING;
 | 
											
												
													
														|  |  import static org.elasticsearch.cluster.routing.ShardRoutingState.RELOCATING;
 |  |  import static org.elasticsearch.cluster.routing.ShardRoutingState.RELOCATING;
 | 
											
												
													
														|  |  import static org.elasticsearch.cluster.routing.ShardRoutingState.STARTED;
 |  |  import static org.elasticsearch.cluster.routing.ShardRoutingState.STARTED;
 | 
											
												
													
														|  |  import static org.elasticsearch.cluster.routing.ShardRoutingState.UNASSIGNED;
 |  |  import static org.elasticsearch.cluster.routing.ShardRoutingState.UNASSIGNED;
 | 
											
												
													
														|  | 
 |  | +import static org.elasticsearch.cluster.routing.TestShardRouting.newShardRouting;
 | 
											
												
													
														|  |  import static org.hamcrest.Matchers.containsString;
 |  |  import static org.hamcrest.Matchers.containsString;
 | 
											
												
													
														|  |  import static org.hamcrest.Matchers.equalTo;
 |  |  import static org.hamcrest.Matchers.equalTo;
 | 
											
												
													
														|  |  import static org.hamcrest.Matchers.hasSize;
 |  |  import static org.hamcrest.Matchers.hasSize;
 | 
											
										
											
												
													
														|  | @@ -689,6 +696,57 @@ public class AllocationCommandsTests extends ESAllocationTestCase {
 | 
											
												
													
														|  |          }
 |  |          }
 | 
											
												
													
														|  |      }
 |  |      }
 | 
											
												
													
														|  |  
 |  |  
 | 
											
												
													
														|  | 
 |  | +    public void testCanceledShardIsInitializedRespectingAllocationDeciders() {
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +        var allocationId1 = AllocationId.newInitializing(UUIDs.randomBase64UUID());
 | 
											
												
													
														|  | 
 |  | +        var allocationId2 = AllocationId.newInitializing(UUIDs.randomBase64UUID());
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +        var indexMetadata = IndexMetadata.builder("test")
 | 
											
												
													
														|  | 
 |  | +            .settings(indexSettings(IndexVersion.current(), 1, 1).put("index.routing.allocation.exclude._id", "node-0"))
 | 
											
												
													
														|  | 
 |  | +            .putInSyncAllocationIds(0, Set.of(allocationId1.getId(), allocationId2.getId()))
 | 
											
												
													
														|  | 
 |  | +            .build();
 | 
											
												
													
														|  | 
 |  | +        var shardId = new ShardId(indexMetadata.getIndex(), 0);
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +        ShardRouting primary = newShardRouting(shardId, "node-0", null, true, STARTED, allocationId1);
 | 
											
												
													
														|  | 
 |  | +        ShardRouting replica = newShardRouting(shardId, "node-1", null, false, STARTED, allocationId2);
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +        ClusterState clusterState = ClusterState.builder(ClusterName.DEFAULT)
 | 
											
												
													
														|  | 
 |  | +            .nodes(
 | 
											
												
													
														|  | 
 |  | +                DiscoveryNodes.builder()
 | 
											
												
													
														|  | 
 |  | +                    .add(newNode("node-0", Version.V_8_10_0))
 | 
											
												
													
														|  | 
 |  | +                    .add(newNode("node-1", Version.V_8_9_0))
 | 
											
												
													
														|  | 
 |  | +                    .add(newNode("node-2", Version.V_8_9_0))
 | 
											
												
													
														|  | 
 |  | +            )
 | 
											
												
													
														|  | 
 |  | +            .metadata(Metadata.builder().put(indexMetadata, false))
 | 
											
												
													
														|  | 
 |  | +            .routingTable(RoutingTable.builder().add(IndexRoutingTable.builder(shardId.getIndex()).addShard(primary).addShard(replica)))
 | 
											
												
													
														|  | 
 |  | +            .build();
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +        var allocation = createAllocationService();
 | 
											
												
													
														|  | 
 |  | +        clusterState = startInitializingShardsAndReroute(allocation, clusterState);
 | 
											
												
													
														|  | 
 |  | +        if (allocation.shardsAllocator instanceof DesiredBalanceShardsAllocator dbsa) {
 | 
											
												
													
														|  | 
 |  | +            // ShardAssignment still contains `node-0` even though `can_remain_decision=no` for it
 | 
											
												
													
														|  | 
 |  | +            assertThat(dbsa.getDesiredBalance().getAssignment(shardId), equalTo(new ShardAssignment(Set.of("node-0", "node-1"), 2, 0, 0)));
 | 
											
												
													
														|  | 
 |  | +        }
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +        clusterState = allocation.reroute(
 | 
											
												
													
														|  | 
 |  | +            clusterState,
 | 
											
												
													
														|  | 
 |  | +            new AllocationCommands(new CancelAllocationCommand(shardId.getIndexName(), 0, "node-0", true)),
 | 
											
												
													
														|  | 
 |  | +            false,
 | 
											
												
													
														|  | 
 |  | +            false,
 | 
											
												
													
														|  | 
 |  | +            false,
 | 
											
												
													
														|  | 
 |  | +            ActionListener.noop()
 | 
											
												
													
														|  | 
 |  | +        ).clusterState();
 | 
											
												
													
														|  | 
 |  | +        clusterState = startInitializingShardsAndReroute(allocation, clusterState);
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +        assertThat(clusterState.getRoutingNodes().node("node-0").size(), equalTo(0));
 | 
											
												
													
														|  | 
 |  | +        assertThat(clusterState.getRoutingNodes().node("node-1").numberOfShardsWithState(STARTED), equalTo(1));
 | 
											
												
													
														|  | 
 |  | +        assertThat(clusterState.getRoutingNodes().node("node-2").numberOfShardsWithState(STARTED), equalTo(1));
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  | 
 |  | +        if (allocation.shardsAllocator instanceof DesiredBalanceShardsAllocator dbsa) {
 | 
											
												
													
														|  | 
 |  | +            assertThat(dbsa.getDesiredBalance().getAssignment(shardId), equalTo(new ShardAssignment(Set.of("node-1", "node-2"), 2, 0, 0)));
 | 
											
												
													
														|  | 
 |  | +        }
 | 
											
												
													
														|  | 
 |  | +    }
 | 
											
												
													
														|  | 
 |  | +
 | 
											
												
													
														|  |      public void testSerialization() throws Exception {
 |  |      public void testSerialization() throws Exception {
 | 
											
												
													
														|  |          AllocationCommands commands = new AllocationCommands(
 |  |          AllocationCommands commands = new AllocationCommands(
 | 
											
												
													
														|  |              new AllocateEmptyPrimaryAllocationCommand("test", 1, "node1", true),
 |  |              new AllocateEmptyPrimaryAllocationCommand("test", 1, "node1", true),
 |