start work on cluster health api, still needs some polising, relates to #42.
This commit is contained in:
parent
e3af8716ed
commit
da510f28ab
|
@ -20,6 +20,7 @@
|
||||||
package org.elasticsearch.action;
|
package org.elasticsearch.action;
|
||||||
|
|
||||||
import com.google.inject.AbstractModule;
|
import com.google.inject.AbstractModule;
|
||||||
|
import org.elasticsearch.action.admin.cluster.health.TransportClusterHealthAction;
|
||||||
import org.elasticsearch.action.admin.cluster.node.info.TransportNodesInfo;
|
import org.elasticsearch.action.admin.cluster.node.info.TransportNodesInfo;
|
||||||
import org.elasticsearch.action.admin.cluster.ping.broadcast.TransportBroadcastPingAction;
|
import org.elasticsearch.action.admin.cluster.ping.broadcast.TransportBroadcastPingAction;
|
||||||
import org.elasticsearch.action.admin.cluster.ping.replication.TransportIndexReplicationPingAction;
|
import org.elasticsearch.action.admin.cluster.ping.replication.TransportIndexReplicationPingAction;
|
||||||
|
@ -58,6 +59,7 @@ public class TransportActionModule extends AbstractModule {
|
||||||
|
|
||||||
bind(TransportNodesInfo.class).asEagerSingleton();
|
bind(TransportNodesInfo.class).asEagerSingleton();
|
||||||
bind(TransportClusterStateAction.class).asEagerSingleton();
|
bind(TransportClusterStateAction.class).asEagerSingleton();
|
||||||
|
bind(TransportClusterHealthAction.class).asEagerSingleton();
|
||||||
|
|
||||||
bind(TransportSinglePingAction.class).asEagerSingleton();
|
bind(TransportSinglePingAction.class).asEagerSingleton();
|
||||||
bind(TransportBroadcastPingAction.class).asEagerSingleton();
|
bind(TransportBroadcastPingAction.class).asEagerSingleton();
|
||||||
|
|
|
@ -62,6 +62,7 @@ public class TransportActions {
|
||||||
public static class Cluster {
|
public static class Cluster {
|
||||||
|
|
||||||
public static final String STATE = "/cluster/state";
|
public static final String STATE = "/cluster/state";
|
||||||
|
public static final String HEALTH = "/cluster/health";
|
||||||
|
|
||||||
public static class Node {
|
public static class Node {
|
||||||
public static final String INFO = "/cluster/nodes/info";
|
public static final String INFO = "/cluster/nodes/info";
|
||||||
|
|
|
@ -0,0 +1,134 @@
|
||||||
|
/*
|
||||||
|
* Licensed to Elastic Search and Shay Banon under one
|
||||||
|
* or more contributor license agreements. See the NOTICE file
|
||||||
|
* distributed with this work for additional information
|
||||||
|
* regarding copyright ownership. Elastic Search 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.action.admin.cluster.health;
|
||||||
|
|
||||||
|
import org.elasticsearch.action.ActionRequestValidationException;
|
||||||
|
import org.elasticsearch.action.support.master.MasterNodeOperationRequest;
|
||||||
|
import org.elasticsearch.util.Strings;
|
||||||
|
import org.elasticsearch.util.TimeValue;
|
||||||
|
|
||||||
|
import java.io.DataInput;
|
||||||
|
import java.io.DataOutput;
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.util.concurrent.TimeUnit;
|
||||||
|
|
||||||
|
import static org.elasticsearch.util.TimeValue.*;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author kimchy (shay.banon)
|
||||||
|
*/
|
||||||
|
public class ClusterHealthRequest extends MasterNodeOperationRequest {
|
||||||
|
|
||||||
|
private String[] indices;
|
||||||
|
|
||||||
|
private TimeValue timeout = new TimeValue(30, TimeUnit.SECONDS);
|
||||||
|
|
||||||
|
private ClusterHealthStatus waitForStatus;
|
||||||
|
|
||||||
|
private int waitForRelocatingShards = -1;
|
||||||
|
|
||||||
|
ClusterHealthRequest() {
|
||||||
|
}
|
||||||
|
|
||||||
|
public ClusterHealthRequest(String... indices) {
|
||||||
|
this.indices = indices;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String[] indices() {
|
||||||
|
return indices;
|
||||||
|
}
|
||||||
|
|
||||||
|
public TimeValue timeout() {
|
||||||
|
return timeout;
|
||||||
|
}
|
||||||
|
|
||||||
|
public ClusterHealthRequest timeout(TimeValue timeout) {
|
||||||
|
this.timeout = timeout;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public ClusterHealthStatus waitForStatus() {
|
||||||
|
return waitForStatus;
|
||||||
|
}
|
||||||
|
|
||||||
|
public ClusterHealthRequest waitForStatus(ClusterHealthStatus waitForStatus) {
|
||||||
|
this.waitForStatus = waitForStatus;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public ClusterHealthRequest waitForGreenStatus() {
|
||||||
|
return waitForStatus(ClusterHealthStatus.GREEN);
|
||||||
|
}
|
||||||
|
|
||||||
|
public ClusterHealthRequest waitForYellowStatus() {
|
||||||
|
return waitForStatus(ClusterHealthStatus.YELLOW);
|
||||||
|
}
|
||||||
|
|
||||||
|
public int waitForRelocatingShards() {
|
||||||
|
return waitForRelocatingShards;
|
||||||
|
}
|
||||||
|
|
||||||
|
public ClusterHealthRequest waitForRelocatingShards(int waitForRelocatingShards) {
|
||||||
|
this.waitForRelocatingShards = waitForRelocatingShards;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override public ActionRequestValidationException validate() {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override public void readFrom(DataInput in) throws IOException, ClassNotFoundException {
|
||||||
|
super.readFrom(in);
|
||||||
|
int size = in.readInt();
|
||||||
|
if (size == 0) {
|
||||||
|
indices = Strings.EMPTY_ARRAY;
|
||||||
|
} else {
|
||||||
|
indices = new String[size];
|
||||||
|
for (int i = 0; i < indices.length; i++) {
|
||||||
|
indices[i] = in.readUTF();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
timeout = readTimeValue(in);
|
||||||
|
if (in.readBoolean()) {
|
||||||
|
waitForStatus = ClusterHealthStatus.fromValue(in.readByte());
|
||||||
|
}
|
||||||
|
waitForRelocatingShards = in.readInt();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override public void writeTo(DataOutput out) throws IOException {
|
||||||
|
super.writeTo(out);
|
||||||
|
if (indices == null) {
|
||||||
|
out.writeInt(0);
|
||||||
|
} else {
|
||||||
|
out.writeInt(indices.length);
|
||||||
|
for (String index : indices) {
|
||||||
|
out.writeUTF(index);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
timeout.writeTo(out);
|
||||||
|
if (waitForStatus == null) {
|
||||||
|
out.writeBoolean(false);
|
||||||
|
} else {
|
||||||
|
out.writeBoolean(true);
|
||||||
|
out.writeByte(waitForStatus.value());
|
||||||
|
}
|
||||||
|
out.writeInt(waitForRelocatingShards);
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,121 @@
|
||||||
|
/*
|
||||||
|
* Licensed to Elastic Search and Shay Banon under one
|
||||||
|
* or more contributor license agreements. See the NOTICE file
|
||||||
|
* distributed with this work for additional information
|
||||||
|
* regarding copyright ownership. Elastic Search 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.action.admin.cluster.health;
|
||||||
|
|
||||||
|
import com.google.common.collect.Maps;
|
||||||
|
import org.elasticsearch.action.ActionResponse;
|
||||||
|
|
||||||
|
import java.io.DataInput;
|
||||||
|
import java.io.DataOutput;
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.util.Iterator;
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
|
import static org.elasticsearch.action.admin.cluster.health.ClusterIndexHealth.*;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author kimchy (shay.banon)
|
||||||
|
*/
|
||||||
|
public class ClusterHealthResponse implements ActionResponse, Iterable<ClusterIndexHealth> {
|
||||||
|
|
||||||
|
private String clusterName;
|
||||||
|
|
||||||
|
int activeShards = 0;
|
||||||
|
|
||||||
|
int relocatingShards = 0;
|
||||||
|
|
||||||
|
int activePrimaryShards = 0;
|
||||||
|
|
||||||
|
boolean timedOut = false;
|
||||||
|
|
||||||
|
ClusterHealthStatus status = ClusterHealthStatus.RED;
|
||||||
|
|
||||||
|
Map<String, ClusterIndexHealth> indices = Maps.newHashMap();
|
||||||
|
|
||||||
|
ClusterHealthResponse() {
|
||||||
|
}
|
||||||
|
|
||||||
|
public ClusterHealthResponse(String clusterName) {
|
||||||
|
this.clusterName = clusterName;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String clusterName() {
|
||||||
|
return clusterName;
|
||||||
|
}
|
||||||
|
|
||||||
|
public int activeShards() {
|
||||||
|
return activeShards;
|
||||||
|
}
|
||||||
|
|
||||||
|
public int relocatingShards() {
|
||||||
|
return relocatingShards;
|
||||||
|
}
|
||||||
|
|
||||||
|
public int activePrimaryShards() {
|
||||||
|
return activePrimaryShards;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* <tt>true</tt> if the waitForXXX has timeout out and did not match.
|
||||||
|
*/
|
||||||
|
public boolean timedOut() {
|
||||||
|
return this.timedOut;
|
||||||
|
}
|
||||||
|
|
||||||
|
public ClusterHealthStatus status() {
|
||||||
|
return status;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Map<String, ClusterIndexHealth> indices() {
|
||||||
|
return indices;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override public Iterator<ClusterIndexHealth> iterator() {
|
||||||
|
return indices.values().iterator();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override public void readFrom(DataInput in) throws IOException, ClassNotFoundException {
|
||||||
|
clusterName = in.readUTF();
|
||||||
|
activePrimaryShards = in.readInt();
|
||||||
|
activeShards = in.readInt();
|
||||||
|
relocatingShards = in.readInt();
|
||||||
|
status = ClusterHealthStatus.fromValue(in.readByte());
|
||||||
|
int size = in.readInt();
|
||||||
|
for (int i = 0; i < size; i++) {
|
||||||
|
ClusterIndexHealth indexHealth = readClusterIndexHealth(in);
|
||||||
|
indices.put(indexHealth.index(), indexHealth);
|
||||||
|
}
|
||||||
|
timedOut = in.readBoolean();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override public void writeTo(DataOutput out) throws IOException {
|
||||||
|
out.writeUTF(clusterName);
|
||||||
|
out.writeInt(activePrimaryShards);
|
||||||
|
out.writeInt(activeShards);
|
||||||
|
out.writeInt(relocatingShards);
|
||||||
|
out.writeByte(status.value());
|
||||||
|
out.writeInt(indices.size());
|
||||||
|
for (ClusterIndexHealth indexHealth : this) {
|
||||||
|
indexHealth.writeTo(out);
|
||||||
|
}
|
||||||
|
out.writeBoolean(timedOut);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,54 @@
|
||||||
|
/*
|
||||||
|
* Licensed to Elastic Search and Shay Banon under one
|
||||||
|
* or more contributor license agreements. See the NOTICE file
|
||||||
|
* distributed with this work for additional information
|
||||||
|
* regarding copyright ownership. Elastic Search 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.action.admin.cluster.health;
|
||||||
|
|
||||||
|
import org.elasticsearch.ElasticSearchIllegalArgumentException;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author kimchy (shay.banon)
|
||||||
|
*/
|
||||||
|
public enum ClusterHealthStatus {
|
||||||
|
GREEN((byte) 0),
|
||||||
|
YELLOW((byte) 1),
|
||||||
|
RED((byte) 2);
|
||||||
|
|
||||||
|
private byte value;
|
||||||
|
|
||||||
|
ClusterHealthStatus(byte value) {
|
||||||
|
this.value = value;
|
||||||
|
}
|
||||||
|
|
||||||
|
public byte value() {
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static ClusterHealthStatus fromValue(byte value) {
|
||||||
|
switch (value) {
|
||||||
|
case 0:
|
||||||
|
return GREEN;
|
||||||
|
case 1:
|
||||||
|
return YELLOW;
|
||||||
|
case 2:
|
||||||
|
return RED;
|
||||||
|
default:
|
||||||
|
throw new ElasticSearchIllegalArgumentException("No cluster health status for value [" + value + "]");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,135 @@
|
||||||
|
/*
|
||||||
|
* Licensed to Elastic Search and Shay Banon under one
|
||||||
|
* or more contributor license agreements. See the NOTICE file
|
||||||
|
* distributed with this work for additional information
|
||||||
|
* regarding copyright ownership. Elastic Search 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.action.admin.cluster.health;
|
||||||
|
|
||||||
|
import com.google.common.collect.Maps;
|
||||||
|
import org.elasticsearch.util.io.Streamable;
|
||||||
|
|
||||||
|
import java.io.DataInput;
|
||||||
|
import java.io.DataOutput;
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.util.Iterator;
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
|
import static org.elasticsearch.action.admin.cluster.health.ClusterShardHealth.*;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author kimchy (shay.banon)
|
||||||
|
*/
|
||||||
|
public class ClusterIndexHealth implements Iterable<ClusterShardHealth>, Streamable {
|
||||||
|
|
||||||
|
private String index;
|
||||||
|
|
||||||
|
private int numberOfShards;
|
||||||
|
|
||||||
|
private int numberOfReplicas;
|
||||||
|
|
||||||
|
int activeShards = 0;
|
||||||
|
|
||||||
|
int relocatingShards = 0;
|
||||||
|
|
||||||
|
int activePrimaryShards = 0;
|
||||||
|
|
||||||
|
ClusterHealthStatus status = ClusterHealthStatus.RED;
|
||||||
|
|
||||||
|
final Map<Integer, ClusterShardHealth> shards = Maps.newHashMap();
|
||||||
|
|
||||||
|
private ClusterIndexHealth() {
|
||||||
|
}
|
||||||
|
|
||||||
|
public ClusterIndexHealth(String index, int numberOfShards, int numberOfReplicas) {
|
||||||
|
this.index = index;
|
||||||
|
this.numberOfShards = numberOfShards;
|
||||||
|
this.numberOfReplicas = numberOfReplicas;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String index() {
|
||||||
|
return index;
|
||||||
|
}
|
||||||
|
|
||||||
|
public int numberOfShards() {
|
||||||
|
return numberOfShards;
|
||||||
|
}
|
||||||
|
|
||||||
|
public int numberOfReplicas() {
|
||||||
|
return numberOfReplicas;
|
||||||
|
}
|
||||||
|
|
||||||
|
public int activeShards() {
|
||||||
|
return activeShards;
|
||||||
|
}
|
||||||
|
|
||||||
|
public int relocatingShards() {
|
||||||
|
return relocatingShards;
|
||||||
|
}
|
||||||
|
|
||||||
|
public int activePrimaryShards() {
|
||||||
|
return activePrimaryShards;
|
||||||
|
}
|
||||||
|
|
||||||
|
public ClusterHealthStatus status() {
|
||||||
|
return status;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Map<Integer, ClusterShardHealth> shards() {
|
||||||
|
return this.shards;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override public Iterator<ClusterShardHealth> iterator() {
|
||||||
|
return shards.values().iterator();
|
||||||
|
}
|
||||||
|
|
||||||
|
public static ClusterIndexHealth readClusterIndexHealth(DataInput in) throws IOException, ClassNotFoundException {
|
||||||
|
ClusterIndexHealth indexHealth = new ClusterIndexHealth();
|
||||||
|
indexHealth.readFrom(in);
|
||||||
|
return indexHealth;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override public void readFrom(DataInput in) throws IOException, ClassNotFoundException {
|
||||||
|
index = in.readUTF();
|
||||||
|
numberOfShards = in.readInt();
|
||||||
|
numberOfReplicas = in.readInt();
|
||||||
|
activePrimaryShards = in.readInt();
|
||||||
|
activeShards = in.readInt();
|
||||||
|
relocatingShards = in.readInt();
|
||||||
|
status = ClusterHealthStatus.fromValue(in.readByte());
|
||||||
|
|
||||||
|
int size = in.readInt();
|
||||||
|
for (int i = 0; i < size; i++) {
|
||||||
|
ClusterShardHealth shardHealth = readClusterShardHealth(in);
|
||||||
|
shards.put(shardHealth.id(), shardHealth);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override public void writeTo(DataOutput out) throws IOException {
|
||||||
|
out.writeUTF(index);
|
||||||
|
out.writeInt(numberOfShards);
|
||||||
|
out.writeInt(numberOfReplicas);
|
||||||
|
out.writeInt(activePrimaryShards);
|
||||||
|
out.writeInt(activeShards);
|
||||||
|
out.writeInt(relocatingShards);
|
||||||
|
out.writeByte(status.value());
|
||||||
|
|
||||||
|
out.writeInt(shards.size());
|
||||||
|
for (ClusterShardHealth shardHealth : this) {
|
||||||
|
shardHealth.writeTo(out);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,92 @@
|
||||||
|
/*
|
||||||
|
* Licensed to Elastic Search and Shay Banon under one
|
||||||
|
* or more contributor license agreements. See the NOTICE file
|
||||||
|
* distributed with this work for additional information
|
||||||
|
* regarding copyright ownership. Elastic Search 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.action.admin.cluster.health;
|
||||||
|
|
||||||
|
import org.elasticsearch.util.io.Streamable;
|
||||||
|
|
||||||
|
import java.io.DataInput;
|
||||||
|
import java.io.DataOutput;
|
||||||
|
import java.io.IOException;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author kimchy (shay.banon)
|
||||||
|
*/
|
||||||
|
public class ClusterShardHealth implements Streamable {
|
||||||
|
|
||||||
|
private int shardId;
|
||||||
|
|
||||||
|
ClusterHealthStatus status = ClusterHealthStatus.RED;
|
||||||
|
|
||||||
|
int activeShards = 0;
|
||||||
|
|
||||||
|
int relocatingShards = 0;
|
||||||
|
|
||||||
|
boolean primaryActive = false;
|
||||||
|
|
||||||
|
private ClusterShardHealth() {
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
ClusterShardHealth(int shardId) {
|
||||||
|
this.shardId = shardId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public int id() {
|
||||||
|
return shardId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public ClusterHealthStatus status() {
|
||||||
|
return status;
|
||||||
|
}
|
||||||
|
|
||||||
|
public int relocatingShards() {
|
||||||
|
return relocatingShards;
|
||||||
|
}
|
||||||
|
|
||||||
|
public int activeShards() {
|
||||||
|
return activeShards;
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean primaryActive() {
|
||||||
|
return primaryActive;
|
||||||
|
}
|
||||||
|
|
||||||
|
static ClusterShardHealth readClusterShardHealth(DataInput in) throws IOException, ClassNotFoundException {
|
||||||
|
ClusterShardHealth ret = new ClusterShardHealth();
|
||||||
|
ret.readFrom(in);
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override public void readFrom(DataInput in) throws IOException, ClassNotFoundException {
|
||||||
|
shardId = in.readInt();
|
||||||
|
status = ClusterHealthStatus.fromValue(in.readByte());
|
||||||
|
activeShards = in.readInt();
|
||||||
|
relocatingShards = in.readInt();
|
||||||
|
primaryActive = in.readBoolean();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override public void writeTo(DataOutput out) throws IOException {
|
||||||
|
out.writeInt(shardId);
|
||||||
|
out.writeByte(status.value());
|
||||||
|
out.writeInt(activeShards);
|
||||||
|
out.writeInt(relocatingShards);
|
||||||
|
out.writeBoolean(primaryActive);
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,187 @@
|
||||||
|
/*
|
||||||
|
* Licensed to Elastic Search and Shay Banon under one
|
||||||
|
* or more contributor license agreements. See the NOTICE file
|
||||||
|
* distributed with this work for additional information
|
||||||
|
* regarding copyright ownership. Elastic Search 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.action.admin.cluster.health;
|
||||||
|
|
||||||
|
import com.google.inject.Inject;
|
||||||
|
import org.elasticsearch.ElasticSearchException;
|
||||||
|
import org.elasticsearch.action.TransportActions;
|
||||||
|
import org.elasticsearch.action.support.master.TransportMasterNodeOperationAction;
|
||||||
|
import org.elasticsearch.cluster.ClusterName;
|
||||||
|
import org.elasticsearch.cluster.ClusterService;
|
||||||
|
import org.elasticsearch.cluster.ClusterState;
|
||||||
|
import org.elasticsearch.cluster.metadata.IndexMetaData;
|
||||||
|
import org.elasticsearch.cluster.routing.IndexRoutingTable;
|
||||||
|
import org.elasticsearch.cluster.routing.IndexShardRoutingTable;
|
||||||
|
import org.elasticsearch.cluster.routing.ShardRouting;
|
||||||
|
import org.elasticsearch.threadpool.ThreadPool;
|
||||||
|
import org.elasticsearch.timer.TimerService;
|
||||||
|
import org.elasticsearch.transport.TransportService;
|
||||||
|
import org.elasticsearch.util.settings.Settings;
|
||||||
|
|
||||||
|
import static org.elasticsearch.action.Actions.*;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author kimchy (shay.banon)
|
||||||
|
*/
|
||||||
|
public class TransportClusterHealthAction extends TransportMasterNodeOperationAction<ClusterHealthRequest, ClusterHealthResponse> {
|
||||||
|
|
||||||
|
private final ClusterName clusterName;
|
||||||
|
|
||||||
|
private final TimerService timerService;
|
||||||
|
|
||||||
|
@Inject public TransportClusterHealthAction(Settings settings, TransportService transportService, ClusterService clusterService, ThreadPool threadPool,
|
||||||
|
TimerService timerService, ClusterName clusterName) {
|
||||||
|
super(settings, transportService, clusterService, threadPool);
|
||||||
|
this.clusterName = clusterName;
|
||||||
|
this.timerService = timerService;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override protected String transportAction() {
|
||||||
|
return TransportActions.Admin.Cluster.HEALTH;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override protected ClusterHealthRequest newRequest() {
|
||||||
|
return new ClusterHealthRequest();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override protected ClusterHealthResponse newResponse() {
|
||||||
|
return new ClusterHealthResponse();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override protected ClusterHealthResponse masterOperation(ClusterHealthRequest request) throws ElasticSearchException {
|
||||||
|
int waitFor = 2;
|
||||||
|
if (request.waitForStatus() == null) {
|
||||||
|
waitFor--;
|
||||||
|
}
|
||||||
|
if (request.waitForRelocatingShards() == -1) {
|
||||||
|
waitFor--;
|
||||||
|
}
|
||||||
|
if (waitFor == 0) {
|
||||||
|
// no need to wait for anything
|
||||||
|
return clusterHealth(request);
|
||||||
|
}
|
||||||
|
long endTime = System.currentTimeMillis() + request.timeout().millis();
|
||||||
|
while (true) {
|
||||||
|
int waitForCounter = 0;
|
||||||
|
ClusterHealthResponse response = clusterHealth(request);
|
||||||
|
if (request.waitForStatus() != null && response.status() == request.waitForStatus()) {
|
||||||
|
waitForCounter++;
|
||||||
|
}
|
||||||
|
if (request.waitForRelocatingShards() != -1 && response.relocatingShards() <= request.waitForRelocatingShards()) {
|
||||||
|
waitForCounter++;
|
||||||
|
}
|
||||||
|
if (waitForCounter == waitFor) {
|
||||||
|
return response;
|
||||||
|
}
|
||||||
|
if (timerService.estimatedTimeInMillis() > endTime) {
|
||||||
|
response.timedOut = true;
|
||||||
|
return response;
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
Thread.sleep(200);
|
||||||
|
} catch (InterruptedException e) {
|
||||||
|
response.timedOut = true;
|
||||||
|
// we got interrupted, bail
|
||||||
|
return response;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private ClusterHealthResponse clusterHealth(ClusterHealthRequest request) {
|
||||||
|
ClusterHealthResponse response = new ClusterHealthResponse(clusterName.value());
|
||||||
|
ClusterState clusterState = clusterService.state();
|
||||||
|
String[] indices = processIndices(clusterState, request.indices());
|
||||||
|
for (String index : indices) {
|
||||||
|
IndexRoutingTable indexRoutingTable = clusterState.routingTable().index(index);
|
||||||
|
IndexMetaData indexMetaData = clusterState.metaData().index(index);
|
||||||
|
if (indexRoutingTable == null) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
ClusterIndexHealth indexHealth = new ClusterIndexHealth(index, indexMetaData.numberOfShards(), indexMetaData.numberOfReplicas());
|
||||||
|
|
||||||
|
for (IndexShardRoutingTable shardRoutingTable : indexRoutingTable) {
|
||||||
|
ClusterShardHealth shardHealth = new ClusterShardHealth(shardRoutingTable.shardId().id());
|
||||||
|
for (ShardRouting shardRouting : shardRoutingTable) {
|
||||||
|
if (shardRouting.active()) {
|
||||||
|
shardHealth.activeShards++;
|
||||||
|
if (shardRouting.relocating()) {
|
||||||
|
// the shard is relocating, the one he is relocating to will be in initializing state, so we don't count it
|
||||||
|
shardHealth.relocatingShards++;
|
||||||
|
}
|
||||||
|
if (shardRouting.primary()) {
|
||||||
|
shardHealth.primaryActive = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (shardHealth.primaryActive) {
|
||||||
|
if (shardHealth.activeShards == shardRoutingTable.size()) {
|
||||||
|
shardHealth.status = ClusterHealthStatus.GREEN;
|
||||||
|
} else {
|
||||||
|
shardHealth.status = ClusterHealthStatus.YELLOW;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
shardHealth.status = ClusterHealthStatus.RED;
|
||||||
|
}
|
||||||
|
indexHealth.shards.put(shardHealth.id(), shardHealth);
|
||||||
|
}
|
||||||
|
|
||||||
|
for (ClusterShardHealth shardHealth : indexHealth) {
|
||||||
|
if (shardHealth.primaryActive()) {
|
||||||
|
indexHealth.activePrimaryShards++;
|
||||||
|
}
|
||||||
|
indexHealth.activeShards += shardHealth.activeShards;
|
||||||
|
indexHealth.relocatingShards += shardHealth.relocatingShards;
|
||||||
|
}
|
||||||
|
// update the index status
|
||||||
|
indexHealth.status = ClusterHealthStatus.GREEN;
|
||||||
|
for (ClusterShardHealth shardHealth : indexHealth) {
|
||||||
|
if (shardHealth.status() == ClusterHealthStatus.RED) {
|
||||||
|
indexHealth.status = ClusterHealthStatus.RED;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (shardHealth.status() == ClusterHealthStatus.YELLOW) {
|
||||||
|
indexHealth.status = ClusterHealthStatus.YELLOW;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
response.indices.put(indexHealth.index(), indexHealth);
|
||||||
|
}
|
||||||
|
|
||||||
|
for (ClusterIndexHealth indexHealth : response) {
|
||||||
|
response.activePrimaryShards += indexHealth.activePrimaryShards;
|
||||||
|
response.activeShards += indexHealth.activeShards;
|
||||||
|
response.relocatingShards += indexHealth.relocatingShards;
|
||||||
|
}
|
||||||
|
|
||||||
|
response.status = ClusterHealthStatus.GREEN;
|
||||||
|
for (ClusterIndexHealth indexHealth : response) {
|
||||||
|
if (indexHealth.status() == ClusterHealthStatus.RED) {
|
||||||
|
response.status = ClusterHealthStatus.RED;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (indexHealth.status() == ClusterHealthStatus.YELLOW) {
|
||||||
|
response.status = ClusterHealthStatus.YELLOW;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
return response;
|
||||||
|
}
|
||||||
|
}
|
|
@ -22,6 +22,7 @@ package org.elasticsearch.action.support.broadcast;
|
||||||
import org.elasticsearch.action.ActionRequest;
|
import org.elasticsearch.action.ActionRequest;
|
||||||
import org.elasticsearch.action.ActionRequestValidationException;
|
import org.elasticsearch.action.ActionRequestValidationException;
|
||||||
import org.elasticsearch.util.Nullable;
|
import org.elasticsearch.util.Nullable;
|
||||||
|
import org.elasticsearch.util.Strings;
|
||||||
|
|
||||||
import java.io.DataInput;
|
import java.io.DataInput;
|
||||||
import java.io.DataOutput;
|
import java.io.DataOutput;
|
||||||
|
@ -78,10 +79,14 @@ public abstract class BroadcastOperationRequest implements ActionRequest {
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override public void writeTo(DataOutput out) throws IOException {
|
@Override public void writeTo(DataOutput out) throws IOException {
|
||||||
|
if (indices == null) {
|
||||||
|
out.writeInt(0);
|
||||||
|
} else {
|
||||||
out.writeInt(indices.length);
|
out.writeInt(indices.length);
|
||||||
for (String index : indices) {
|
for (String index : indices) {
|
||||||
out.writeUTF(index);
|
out.writeUTF(index);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
if (queryHint == null) {
|
if (queryHint == null) {
|
||||||
out.writeBoolean(false);
|
out.writeBoolean(false);
|
||||||
} else {
|
} else {
|
||||||
|
@ -92,10 +97,15 @@ public abstract class BroadcastOperationRequest implements ActionRequest {
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override public void readFrom(DataInput in) throws IOException, ClassNotFoundException {
|
@Override public void readFrom(DataInput in) throws IOException, ClassNotFoundException {
|
||||||
indices = new String[in.readInt()];
|
int size = in.readInt();
|
||||||
|
if (size == 0) {
|
||||||
|
indices = Strings.EMPTY_ARRAY;
|
||||||
|
} else {
|
||||||
|
indices = new String[size];
|
||||||
for (int i = 0; i < indices.length; i++) {
|
for (int i = 0; i < indices.length; i++) {
|
||||||
indices[i] = in.readUTF();
|
indices[i] = in.readUTF();
|
||||||
}
|
}
|
||||||
|
}
|
||||||
if (in.readBoolean()) {
|
if (in.readBoolean()) {
|
||||||
queryHint = in.readUTF();
|
queryHint = in.readUTF();
|
||||||
}
|
}
|
||||||
|
|
|
@ -21,6 +21,8 @@ package org.elasticsearch.client;
|
||||||
|
|
||||||
import org.elasticsearch.action.ActionFuture;
|
import org.elasticsearch.action.ActionFuture;
|
||||||
import org.elasticsearch.action.ActionListener;
|
import org.elasticsearch.action.ActionListener;
|
||||||
|
import org.elasticsearch.action.admin.cluster.health.ClusterHealthRequest;
|
||||||
|
import org.elasticsearch.action.admin.cluster.health.ClusterHealthResponse;
|
||||||
import org.elasticsearch.action.admin.cluster.node.info.NodesInfoRequest;
|
import org.elasticsearch.action.admin.cluster.node.info.NodesInfoRequest;
|
||||||
import org.elasticsearch.action.admin.cluster.node.info.NodesInfoResponse;
|
import org.elasticsearch.action.admin.cluster.node.info.NodesInfoResponse;
|
||||||
import org.elasticsearch.action.admin.cluster.ping.broadcast.BroadcastPingRequest;
|
import org.elasticsearch.action.admin.cluster.ping.broadcast.BroadcastPingRequest;
|
||||||
|
@ -40,6 +42,34 @@ import org.elasticsearch.action.admin.cluster.state.ClusterStateResponse;
|
||||||
*/
|
*/
|
||||||
public interface ClusterAdminClient {
|
public interface ClusterAdminClient {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The health of the cluster.
|
||||||
|
*
|
||||||
|
* @param request The cluster state request
|
||||||
|
* @return The result future
|
||||||
|
* @see Requests#clusterHealth(String...)
|
||||||
|
*/
|
||||||
|
ActionFuture<ClusterHealthResponse> health(ClusterHealthRequest request);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The health of the cluster.
|
||||||
|
*
|
||||||
|
* @param request The cluster state request
|
||||||
|
* @param listener A listener to be notified with a result
|
||||||
|
* @return The result future
|
||||||
|
* @see Requests#clusterHealth(String...)
|
||||||
|
*/
|
||||||
|
ActionFuture<ClusterHealthResponse> health(ClusterHealthRequest request, ActionListener<ClusterHealthResponse> listener);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The health of the cluster.
|
||||||
|
*
|
||||||
|
* @param request The cluster state request
|
||||||
|
* @param listener A listener to be notified with a result
|
||||||
|
* @see Requests#clusterHealth(String...)
|
||||||
|
*/
|
||||||
|
void execHealth(ClusterHealthRequest request, ActionListener<ClusterHealthResponse> listener);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The state of the cluster.
|
* The state of the cluster.
|
||||||
*
|
*
|
||||||
|
|
|
@ -19,6 +19,7 @@
|
||||||
|
|
||||||
package org.elasticsearch.client;
|
package org.elasticsearch.client;
|
||||||
|
|
||||||
|
import org.elasticsearch.action.admin.cluster.health.ClusterHealthRequest;
|
||||||
import org.elasticsearch.action.admin.cluster.node.info.NodesInfoRequest;
|
import org.elasticsearch.action.admin.cluster.node.info.NodesInfoRequest;
|
||||||
import org.elasticsearch.action.admin.cluster.ping.broadcast.BroadcastPingRequest;
|
import org.elasticsearch.action.admin.cluster.ping.broadcast.BroadcastPingRequest;
|
||||||
import org.elasticsearch.action.admin.cluster.ping.replication.ReplicationPingRequest;
|
import org.elasticsearch.action.admin.cluster.ping.replication.ReplicationPingRequest;
|
||||||
|
@ -233,6 +234,17 @@ public class Requests {
|
||||||
return new ClusterStateRequest();
|
return new ClusterStateRequest();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates a cluster health request.
|
||||||
|
*
|
||||||
|
* @param indices The indices to optimize. Use <tt>null</tt> or <tt>_all</tt> to execute against all indices
|
||||||
|
* @return The cluster health request
|
||||||
|
* @see org.elasticsearch.client.ClusterAdminClient#health(org.elasticsearch.action.admin.cluster.health.ClusterHealthRequest)
|
||||||
|
*/
|
||||||
|
public static ClusterHealthRequest clusterHealth(String... indices) {
|
||||||
|
return new ClusterHealthRequest(indices);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Creates a nodes info request against all the nodes.
|
* Creates a nodes info request against all the nodes.
|
||||||
*
|
*
|
||||||
|
|
|
@ -22,6 +22,9 @@ package org.elasticsearch.client.server;
|
||||||
import com.google.inject.Inject;
|
import com.google.inject.Inject;
|
||||||
import org.elasticsearch.action.ActionFuture;
|
import org.elasticsearch.action.ActionFuture;
|
||||||
import org.elasticsearch.action.ActionListener;
|
import org.elasticsearch.action.ActionListener;
|
||||||
|
import org.elasticsearch.action.admin.cluster.health.ClusterHealthRequest;
|
||||||
|
import org.elasticsearch.action.admin.cluster.health.ClusterHealthResponse;
|
||||||
|
import org.elasticsearch.action.admin.cluster.health.TransportClusterHealthAction;
|
||||||
import org.elasticsearch.action.admin.cluster.node.info.NodesInfoRequest;
|
import org.elasticsearch.action.admin.cluster.node.info.NodesInfoRequest;
|
||||||
import org.elasticsearch.action.admin.cluster.node.info.NodesInfoResponse;
|
import org.elasticsearch.action.admin.cluster.node.info.NodesInfoResponse;
|
||||||
import org.elasticsearch.action.admin.cluster.node.info.TransportNodesInfo;
|
import org.elasticsearch.action.admin.cluster.node.info.TransportNodesInfo;
|
||||||
|
@ -46,6 +49,8 @@ import org.elasticsearch.util.settings.Settings;
|
||||||
*/
|
*/
|
||||||
public class ServerClusterAdminClient extends AbstractComponent implements ClusterAdminClient {
|
public class ServerClusterAdminClient extends AbstractComponent implements ClusterAdminClient {
|
||||||
|
|
||||||
|
private final TransportClusterHealthAction clusterHealthAction;
|
||||||
|
|
||||||
private final TransportClusterStateAction clusterStateAction;
|
private final TransportClusterStateAction clusterStateAction;
|
||||||
|
|
||||||
private final TransportSinglePingAction singlePingAction;
|
private final TransportSinglePingAction singlePingAction;
|
||||||
|
@ -57,10 +62,11 @@ public class ServerClusterAdminClient extends AbstractComponent implements Clust
|
||||||
private final TransportNodesInfo nodesInfo;
|
private final TransportNodesInfo nodesInfo;
|
||||||
|
|
||||||
@Inject public ServerClusterAdminClient(Settings settings,
|
@Inject public ServerClusterAdminClient(Settings settings,
|
||||||
TransportClusterStateAction clusterStateAction,
|
TransportClusterHealthAction clusterHealthAction, TransportClusterStateAction clusterStateAction,
|
||||||
TransportSinglePingAction singlePingAction, TransportBroadcastPingAction broadcastPingAction, TransportReplicationPingAction replicationPingAction,
|
TransportSinglePingAction singlePingAction, TransportBroadcastPingAction broadcastPingAction, TransportReplicationPingAction replicationPingAction,
|
||||||
TransportNodesInfo nodesInfo) {
|
TransportNodesInfo nodesInfo) {
|
||||||
super(settings);
|
super(settings);
|
||||||
|
this.clusterHealthAction = clusterHealthAction;
|
||||||
this.clusterStateAction = clusterStateAction;
|
this.clusterStateAction = clusterStateAction;
|
||||||
this.nodesInfo = nodesInfo;
|
this.nodesInfo = nodesInfo;
|
||||||
this.singlePingAction = singlePingAction;
|
this.singlePingAction = singlePingAction;
|
||||||
|
@ -68,6 +74,18 @@ public class ServerClusterAdminClient extends AbstractComponent implements Clust
|
||||||
this.replicationPingAction = replicationPingAction;
|
this.replicationPingAction = replicationPingAction;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override public ActionFuture<ClusterHealthResponse> health(ClusterHealthRequest request) {
|
||||||
|
return clusterHealthAction.submit(request);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override public ActionFuture<ClusterHealthResponse> health(ClusterHealthRequest request, ActionListener<ClusterHealthResponse> listener) {
|
||||||
|
return clusterHealthAction.submit(request, listener);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override public void execHealth(ClusterHealthRequest request, ActionListener<ClusterHealthResponse> listener) {
|
||||||
|
clusterHealthAction.execute(request, listener);
|
||||||
|
}
|
||||||
|
|
||||||
@Override public ActionFuture<ClusterStateResponse> state(ClusterStateRequest request) {
|
@Override public ActionFuture<ClusterStateResponse> state(ClusterStateRequest request) {
|
||||||
return clusterStateAction.submit(request);
|
return clusterStateAction.submit(request);
|
||||||
}
|
}
|
||||||
|
|
|
@ -20,6 +20,7 @@
|
||||||
package org.elasticsearch.client.transport.action;
|
package org.elasticsearch.client.transport.action;
|
||||||
|
|
||||||
import com.google.inject.AbstractModule;
|
import com.google.inject.AbstractModule;
|
||||||
|
import org.elasticsearch.client.transport.action.admin.cluster.health.ClientTransportClusterHealthAction;
|
||||||
import org.elasticsearch.client.transport.action.admin.cluster.node.info.ClientTransportNodesInfoAction;
|
import org.elasticsearch.client.transport.action.admin.cluster.node.info.ClientTransportNodesInfoAction;
|
||||||
import org.elasticsearch.client.transport.action.admin.cluster.ping.broadcast.ClientTransportBroadcastPingAction;
|
import org.elasticsearch.client.transport.action.admin.cluster.ping.broadcast.ClientTransportBroadcastPingAction;
|
||||||
import org.elasticsearch.client.transport.action.admin.cluster.ping.replication.ClientTransportReplicationPingAction;
|
import org.elasticsearch.client.transport.action.admin.cluster.ping.replication.ClientTransportReplicationPingAction;
|
||||||
|
@ -43,7 +44,7 @@ import org.elasticsearch.client.transport.action.search.ClientTransportSearchScr
|
||||||
import org.elasticsearch.client.transport.action.terms.ClientTransportTermsAction;
|
import org.elasticsearch.client.transport.action.terms.ClientTransportTermsAction;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @author kimchy (Shay Banon)
|
* @author kimchy (shay.banon)
|
||||||
*/
|
*/
|
||||||
public class ClientTransportActionModule extends AbstractModule {
|
public class ClientTransportActionModule extends AbstractModule {
|
||||||
|
|
||||||
|
@ -70,5 +71,6 @@ public class ClientTransportActionModule extends AbstractModule {
|
||||||
bind(ClientTransportReplicationPingAction.class).asEagerSingleton();
|
bind(ClientTransportReplicationPingAction.class).asEagerSingleton();
|
||||||
bind(ClientTransportBroadcastPingAction.class).asEagerSingleton();
|
bind(ClientTransportBroadcastPingAction.class).asEagerSingleton();
|
||||||
bind(ClientTransportClusterStateAction.class).asEagerSingleton();
|
bind(ClientTransportClusterStateAction.class).asEagerSingleton();
|
||||||
|
bind(ClientTransportClusterHealthAction.class).asEagerSingleton();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,42 @@
|
||||||
|
/*
|
||||||
|
* Licensed to Elastic Search and Shay Banon under one
|
||||||
|
* or more contributor license agreements. See the NOTICE file
|
||||||
|
* distributed with this work for additional information
|
||||||
|
* regarding copyright ownership. Elastic Search 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.client.transport.action.admin.cluster.health;
|
||||||
|
|
||||||
|
import com.google.inject.Inject;
|
||||||
|
import org.elasticsearch.action.TransportActions;
|
||||||
|
import org.elasticsearch.action.admin.cluster.health.ClusterHealthRequest;
|
||||||
|
import org.elasticsearch.action.admin.cluster.health.ClusterHealthResponse;
|
||||||
|
import org.elasticsearch.client.transport.action.support.BaseClientTransportAction;
|
||||||
|
import org.elasticsearch.transport.TransportService;
|
||||||
|
import org.elasticsearch.util.settings.Settings;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author kimchy (Shay Banon)
|
||||||
|
*/
|
||||||
|
public class ClientTransportClusterHealthAction extends BaseClientTransportAction<ClusterHealthRequest, ClusterHealthResponse> {
|
||||||
|
|
||||||
|
@Inject public ClientTransportClusterHealthAction(Settings settings, TransportService transportService) {
|
||||||
|
super(settings, transportService, ClusterHealthResponse.class);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override protected String action() {
|
||||||
|
return TransportActions.Admin.Cluster.HEALTH;
|
||||||
|
}
|
||||||
|
}
|
|
@ -23,6 +23,8 @@ import com.google.inject.Inject;
|
||||||
import org.elasticsearch.ElasticSearchException;
|
import org.elasticsearch.ElasticSearchException;
|
||||||
import org.elasticsearch.action.ActionFuture;
|
import org.elasticsearch.action.ActionFuture;
|
||||||
import org.elasticsearch.action.ActionListener;
|
import org.elasticsearch.action.ActionListener;
|
||||||
|
import org.elasticsearch.action.admin.cluster.health.ClusterHealthRequest;
|
||||||
|
import org.elasticsearch.action.admin.cluster.health.ClusterHealthResponse;
|
||||||
import org.elasticsearch.action.admin.cluster.node.info.NodesInfoRequest;
|
import org.elasticsearch.action.admin.cluster.node.info.NodesInfoRequest;
|
||||||
import org.elasticsearch.action.admin.cluster.node.info.NodesInfoResponse;
|
import org.elasticsearch.action.admin.cluster.node.info.NodesInfoResponse;
|
||||||
import org.elasticsearch.action.admin.cluster.ping.broadcast.BroadcastPingRequest;
|
import org.elasticsearch.action.admin.cluster.ping.broadcast.BroadcastPingRequest;
|
||||||
|
@ -35,6 +37,7 @@ import org.elasticsearch.action.admin.cluster.state.ClusterStateRequest;
|
||||||
import org.elasticsearch.action.admin.cluster.state.ClusterStateResponse;
|
import org.elasticsearch.action.admin.cluster.state.ClusterStateResponse;
|
||||||
import org.elasticsearch.client.ClusterAdminClient;
|
import org.elasticsearch.client.ClusterAdminClient;
|
||||||
import org.elasticsearch.client.transport.TransportClientNodesService;
|
import org.elasticsearch.client.transport.TransportClientNodesService;
|
||||||
|
import org.elasticsearch.client.transport.action.admin.cluster.health.ClientTransportClusterHealthAction;
|
||||||
import org.elasticsearch.client.transport.action.admin.cluster.node.info.ClientTransportNodesInfoAction;
|
import org.elasticsearch.client.transport.action.admin.cluster.node.info.ClientTransportNodesInfoAction;
|
||||||
import org.elasticsearch.client.transport.action.admin.cluster.ping.broadcast.ClientTransportBroadcastPingAction;
|
import org.elasticsearch.client.transport.action.admin.cluster.ping.broadcast.ClientTransportBroadcastPingAction;
|
||||||
import org.elasticsearch.client.transport.action.admin.cluster.ping.replication.ClientTransportReplicationPingAction;
|
import org.elasticsearch.client.transport.action.admin.cluster.ping.replication.ClientTransportReplicationPingAction;
|
||||||
|
@ -51,6 +54,8 @@ public class InternalTransportClusterAdminClient extends AbstractComponent imple
|
||||||
|
|
||||||
private final TransportClientNodesService nodesService;
|
private final TransportClientNodesService nodesService;
|
||||||
|
|
||||||
|
private final ClientTransportClusterHealthAction clusterHealthAction;
|
||||||
|
|
||||||
private final ClientTransportClusterStateAction clusterStateAction;
|
private final ClientTransportClusterStateAction clusterStateAction;
|
||||||
|
|
||||||
private final ClientTransportSinglePingAction singlePingAction;
|
private final ClientTransportSinglePingAction singlePingAction;
|
||||||
|
@ -62,11 +67,12 @@ public class InternalTransportClusterAdminClient extends AbstractComponent imple
|
||||||
private final ClientTransportNodesInfoAction nodesInfoAction;
|
private final ClientTransportNodesInfoAction nodesInfoAction;
|
||||||
|
|
||||||
@Inject public InternalTransportClusterAdminClient(Settings settings, TransportClientNodesService nodesService,
|
@Inject public InternalTransportClusterAdminClient(Settings settings, TransportClientNodesService nodesService,
|
||||||
ClientTransportClusterStateAction clusterStateAction,
|
ClientTransportClusterHealthAction clusterHealthAction, ClientTransportClusterStateAction clusterStateAction,
|
||||||
ClientTransportSinglePingAction singlePingAction, ClientTransportReplicationPingAction replicationPingAction, ClientTransportBroadcastPingAction broadcastPingAction,
|
ClientTransportSinglePingAction singlePingAction, ClientTransportReplicationPingAction replicationPingAction, ClientTransportBroadcastPingAction broadcastPingAction,
|
||||||
ClientTransportNodesInfoAction nodesInfoAction) {
|
ClientTransportNodesInfoAction nodesInfoAction) {
|
||||||
super(settings);
|
super(settings);
|
||||||
this.nodesService = nodesService;
|
this.nodesService = nodesService;
|
||||||
|
this.clusterHealthAction = clusterHealthAction;
|
||||||
this.clusterStateAction = clusterStateAction;
|
this.clusterStateAction = clusterStateAction;
|
||||||
this.nodesInfoAction = nodesInfoAction;
|
this.nodesInfoAction = nodesInfoAction;
|
||||||
this.singlePingAction = singlePingAction;
|
this.singlePingAction = singlePingAction;
|
||||||
|
@ -74,6 +80,31 @@ public class InternalTransportClusterAdminClient extends AbstractComponent imple
|
||||||
this.broadcastPingAction = broadcastPingAction;
|
this.broadcastPingAction = broadcastPingAction;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override public ActionFuture<ClusterHealthResponse> health(final ClusterHealthRequest request) {
|
||||||
|
return nodesService.execute(new TransportClientNodesService.NodeCallback<ActionFuture<ClusterHealthResponse>>() {
|
||||||
|
@Override public ActionFuture<ClusterHealthResponse> doWithNode(Node node) throws ElasticSearchException {
|
||||||
|
return clusterHealthAction.submit(node, request);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override public ActionFuture<ClusterHealthResponse> health(final ClusterHealthRequest request, final ActionListener<ClusterHealthResponse> listener) {
|
||||||
|
return nodesService.execute(new TransportClientNodesService.NodeCallback<ActionFuture<ClusterHealthResponse>>() {
|
||||||
|
@Override public ActionFuture<ClusterHealthResponse> doWithNode(Node node) throws ElasticSearchException {
|
||||||
|
return clusterHealthAction.submit(node, request, listener);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override public void execHealth(final ClusterHealthRequest request, final ActionListener<ClusterHealthResponse> listener) {
|
||||||
|
nodesService.execute(new TransportClientNodesService.NodeCallback<Void>() {
|
||||||
|
@Override public Void doWithNode(Node node) throws ElasticSearchException {
|
||||||
|
clusterHealthAction.execute(node, request, listener);
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
@Override public ActionFuture<ClusterStateResponse> state(final ClusterStateRequest request) {
|
@Override public ActionFuture<ClusterStateResponse> state(final ClusterStateRequest request) {
|
||||||
return nodesService.execute(new TransportClientNodesService.NodeCallback<ActionFuture<ClusterStateResponse>>() {
|
return nodesService.execute(new TransportClientNodesService.NodeCallback<ActionFuture<ClusterStateResponse>>() {
|
||||||
@Override public ActionFuture<ClusterStateResponse> doWithNode(Node node) throws ElasticSearchException {
|
@Override public ActionFuture<ClusterStateResponse> doWithNode(Node node) throws ElasticSearchException {
|
||||||
|
|
|
@ -357,7 +357,6 @@ public class DefaultShardsRoutingStrategy implements ShardsRoutingStrategy {
|
||||||
shard.deassignNode();
|
shard.deassignNode();
|
||||||
shards.remove();
|
shards.remove();
|
||||||
} else {
|
} else {
|
||||||
assert shard.state() == ShardRoutingState.RELOCATING;
|
|
||||||
shard.cancelRelocation();
|
shard.cancelRelocation();
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
|
@ -145,6 +145,7 @@ public class IndexShardGatewayService extends AbstractIndexShardComponent {
|
||||||
// do not snapshot when in the process of relocation of primaries so we won't get conflicts
|
// do not snapshot when in the process of relocation of primaries so we won't get conflicts
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
try {
|
||||||
indexShard.snapshot(new Engine.SnapshotHandler() {
|
indexShard.snapshot(new Engine.SnapshotHandler() {
|
||||||
@Override public void snapshot(SnapshotIndexCommit snapshotIndexCommit, Translog.Snapshot translogSnapshot) throws EngineException {
|
@Override public void snapshot(SnapshotIndexCommit snapshotIndexCommit, Translog.Snapshot translogSnapshot) throws EngineException {
|
||||||
if (lastIndexVersion != snapshotIndexCommit.getVersion() || lastTranslogId != translogSnapshot.translogId() || lastTranslogSize != translogSnapshot.size()) {
|
if (lastIndexVersion != snapshotIndexCommit.getVersion() || lastTranslogId != translogSnapshot.translogId() || lastTranslogSize != translogSnapshot.size()) {
|
||||||
|
@ -157,6 +158,11 @@ public class IndexShardGatewayService extends AbstractIndexShardComponent {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
} catch (IllegalIndexShardStateException e) {
|
||||||
|
// ignore, that's fine
|
||||||
|
} catch (Exception e) {
|
||||||
|
logger.warn("Failed to snapshot on close", e);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public void close() {
|
public void close() {
|
||||||
|
|
|
@ -48,7 +48,9 @@ import java.io.IOException;
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.concurrent.ConcurrentMap;
|
import java.util.concurrent.ConcurrentMap;
|
||||||
|
import java.util.concurrent.CopyOnWriteArrayList;
|
||||||
import java.util.concurrent.CountDownLatch;
|
import java.util.concurrent.CountDownLatch;
|
||||||
|
import java.util.concurrent.Future;
|
||||||
import java.util.concurrent.atomic.AtomicReference;
|
import java.util.concurrent.atomic.AtomicReference;
|
||||||
|
|
||||||
import static java.util.concurrent.TimeUnit.*;
|
import static java.util.concurrent.TimeUnit.*;
|
||||||
|
@ -77,6 +79,16 @@ public class RecoveryAction extends AbstractIndexShardComponent {
|
||||||
|
|
||||||
private final String snapshotTransportAction;
|
private final String snapshotTransportAction;
|
||||||
|
|
||||||
|
private volatile boolean closed = false;
|
||||||
|
|
||||||
|
private volatile Thread sendStartRecoveryThread;
|
||||||
|
|
||||||
|
private volatile Thread receiveSnapshotRecoveryThread;
|
||||||
|
|
||||||
|
private volatile Thread sendSnapshotRecoveryThread;
|
||||||
|
|
||||||
|
private final CopyOnWriteArrayList<Future> sendFileChunksRecoveryFutures = new CopyOnWriteArrayList<Future>();
|
||||||
|
|
||||||
@Inject public RecoveryAction(ShardId shardId, @IndexSettings Settings indexSettings, ThreadPool threadPool, TransportService transportService, IndexShard indexShard, Store store) {
|
@Inject public RecoveryAction(ShardId shardId, @IndexSettings Settings indexSettings, ThreadPool threadPool, TransportService transportService, IndexShard indexShard, Store store) {
|
||||||
super(shardId, indexSettings);
|
super(shardId, indexSettings);
|
||||||
this.threadPool = threadPool;
|
this.threadPool = threadPool;
|
||||||
|
@ -96,14 +108,36 @@ public class RecoveryAction extends AbstractIndexShardComponent {
|
||||||
}
|
}
|
||||||
|
|
||||||
public void close() {
|
public void close() {
|
||||||
|
closed = true;
|
||||||
transportService.removeHandler(startTransportAction);
|
transportService.removeHandler(startTransportAction);
|
||||||
transportService.removeHandler(fileChunkTransportAction);
|
transportService.removeHandler(fileChunkTransportAction);
|
||||||
transportService.removeHandler(snapshotTransportAction);
|
transportService.removeHandler(snapshotTransportAction);
|
||||||
|
|
||||||
cleanOpenIndex();
|
cleanOpenIndex();
|
||||||
|
|
||||||
|
if (true) {
|
||||||
|
// disable the interruptions for now
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// interrupt the startRecovery thread if its performing recovery
|
||||||
|
if (sendStartRecoveryThread != null) {
|
||||||
|
sendStartRecoveryThread.interrupt();
|
||||||
|
}
|
||||||
|
if (receiveSnapshotRecoveryThread != null) {
|
||||||
|
receiveSnapshotRecoveryThread.interrupt();
|
||||||
|
}
|
||||||
|
if (sendSnapshotRecoveryThread != null) {
|
||||||
|
sendSnapshotRecoveryThread.interrupt();
|
||||||
|
}
|
||||||
|
for (Future future : sendFileChunksRecoveryFutures) {
|
||||||
|
future.cancel(true);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public synchronized void startRecovery(Node node, Node targetNode, boolean markAsRelocated) throws ElasticSearchException {
|
public synchronized void startRecovery(Node node, Node targetNode, boolean markAsRelocated) throws ElasticSearchException {
|
||||||
|
sendStartRecoveryThread = Thread.currentThread();
|
||||||
|
try {
|
||||||
// mark the shard as recovering
|
// mark the shard as recovering
|
||||||
IndexShardState preRecoveringState;
|
IndexShardState preRecoveringState;
|
||||||
try {
|
try {
|
||||||
|
@ -118,11 +152,14 @@ public class RecoveryAction extends AbstractIndexShardComponent {
|
||||||
// that's fine, since we might be called concurrently, just ignore this, we are already relocated
|
// that's fine, since we might be called concurrently, just ignore this, we are already relocated
|
||||||
throw new IgnoreRecoveryException("Already in recovering process", e);
|
throw new IgnoreRecoveryException("Already in recovering process", e);
|
||||||
} catch (IndexShardClosedException e) {
|
} catch (IndexShardClosedException e) {
|
||||||
throw new IgnoreRecoveryException("can't recover a closed shard.", e);
|
throw new IgnoreRecoveryException("Can't recover a closed shard.", e);
|
||||||
}
|
}
|
||||||
logger.debug("Starting recovery from {}", targetNode);
|
logger.debug("Starting recovery from {}", targetNode);
|
||||||
StopWatch stopWatch = new StopWatch().start();
|
StopWatch stopWatch = new StopWatch().start();
|
||||||
try {
|
try {
|
||||||
|
if (closed) {
|
||||||
|
throw new IgnoreRecoveryException("Recovery closed");
|
||||||
|
}
|
||||||
RecoveryStatus recoveryStatus = transportService.submitRequest(targetNode, startTransportAction, new StartRecoveryRequest(node, markAsRelocated), new FutureTransportResponseHandler<RecoveryStatus>() {
|
RecoveryStatus recoveryStatus = transportService.submitRequest(targetNode, startTransportAction, new StartRecoveryRequest(node, markAsRelocated), new FutureTransportResponseHandler<RecoveryStatus>() {
|
||||||
@Override public RecoveryStatus newInstance() {
|
@Override public RecoveryStatus newInstance() {
|
||||||
return new RecoveryStatus();
|
return new RecoveryStatus();
|
||||||
|
@ -144,17 +181,25 @@ public class RecoveryAction extends AbstractIndexShardComponent {
|
||||||
logger.debug(sb.toString());
|
logger.debug(sb.toString());
|
||||||
}
|
}
|
||||||
} catch (RemoteTransportException e) {
|
} catch (RemoteTransportException e) {
|
||||||
|
if (closed) {
|
||||||
|
throw new IgnoreRecoveryException("Recovery closed", e);
|
||||||
|
}
|
||||||
Throwable cause = ExceptionsHelper.unwrapCause(e);
|
Throwable cause = ExceptionsHelper.unwrapCause(e);
|
||||||
if (cause instanceof ActionNotFoundTransportException ||
|
if (cause instanceof ActionNotFoundTransportException || cause instanceof IndexShardNotStartedException) {
|
||||||
cause instanceof IndexShardNotStartedException) {
|
|
||||||
// the remote shard has not yet registered the action or not started yet, we need to ignore this recovery attempt, and restore the state previous to recovering
|
// the remote shard has not yet registered the action or not started yet, we need to ignore this recovery attempt, and restore the state previous to recovering
|
||||||
indexShard.restoreRecoveryState(preRecoveringState);
|
indexShard.restoreRecoveryState(preRecoveringState);
|
||||||
throw new IgnoreRecoveryException("Ignoring recovery attempt, remote shard not started", e);
|
throw new IgnoreRecoveryException("Ignoring recovery attempt, remote shard not started", e);
|
||||||
}
|
}
|
||||||
throw new RecoveryFailedException(shardId, node, targetNode, e);
|
throw new RecoveryFailedException(shardId, node, targetNode, e);
|
||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
|
if (closed) {
|
||||||
|
throw new IgnoreRecoveryException("Recovery closed", e);
|
||||||
|
}
|
||||||
throw new RecoveryFailedException(shardId, node, targetNode, e);
|
throw new RecoveryFailedException(shardId, node, targetNode, e);
|
||||||
}
|
}
|
||||||
|
} finally {
|
||||||
|
sendStartRecoveryThread = null;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void cleanOpenIndex() {
|
private void cleanOpenIndex() {
|
||||||
|
@ -226,7 +271,7 @@ public class RecoveryAction extends AbstractIndexShardComponent {
|
||||||
final CountDownLatch latch = new CountDownLatch(snapshot.getFiles().length);
|
final CountDownLatch latch = new CountDownLatch(snapshot.getFiles().length);
|
||||||
final AtomicReference<Exception> lastException = new AtomicReference<Exception>();
|
final AtomicReference<Exception> lastException = new AtomicReference<Exception>();
|
||||||
for (final String name : snapshot.getFiles()) {
|
for (final String name : snapshot.getFiles()) {
|
||||||
threadPool.execute(new Runnable() {
|
sendFileChunksRecoveryFutures.add(threadPool.submit(new Runnable() {
|
||||||
@Override public void run() {
|
@Override public void run() {
|
||||||
IndexInput indexInput = null;
|
IndexInput indexInput = null;
|
||||||
try {
|
try {
|
||||||
|
@ -256,7 +301,7 @@ public class RecoveryAction extends AbstractIndexShardComponent {
|
||||||
latch.countDown();
|
latch.countDown();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
}));
|
||||||
}
|
}
|
||||||
|
|
||||||
latch.await();
|
latch.await();
|
||||||
|
@ -270,10 +315,17 @@ public class RecoveryAction extends AbstractIndexShardComponent {
|
||||||
recoveryStatus.phase1Time = stopWatch.totalTime().millis();
|
recoveryStatus.phase1Time = stopWatch.totalTime().millis();
|
||||||
} catch (Throwable e) {
|
} catch (Throwable e) {
|
||||||
throw new RecoverFilesRecoveryException(shardId, snapshot.getFiles().length, new SizeValue(totalSize), e);
|
throw new RecoverFilesRecoveryException(shardId, snapshot.getFiles().length, new SizeValue(totalSize), e);
|
||||||
|
} finally {
|
||||||
|
sendFileChunksRecoveryFutures.clear();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override public void phase2(Translog.Snapshot snapshot) throws ElasticSearchException {
|
@Override public void phase2(Translog.Snapshot snapshot) throws ElasticSearchException {
|
||||||
|
sendSnapshotRecoveryThread = Thread.currentThread();
|
||||||
|
try {
|
||||||
|
if (closed) {
|
||||||
|
throw new IndexShardClosedException(shardId);
|
||||||
|
}
|
||||||
logger.trace("Recovery [phase2] to {}: sending [{}] transaction log operations", node, snapshot.size());
|
logger.trace("Recovery [phase2] to {}: sending [{}] transaction log operations", node, snapshot.size());
|
||||||
StopWatch stopWatch = new StopWatch().start();
|
StopWatch stopWatch = new StopWatch().start();
|
||||||
sendSnapshot(snapshot, false);
|
sendSnapshot(snapshot, false);
|
||||||
|
@ -281,9 +333,17 @@ public class RecoveryAction extends AbstractIndexShardComponent {
|
||||||
logger.trace("Recovery [phase2] to {}: took [{}]", node, stopWatch.totalTime());
|
logger.trace("Recovery [phase2] to {}: took [{}]", node, stopWatch.totalTime());
|
||||||
recoveryStatus.phase2Time = stopWatch.totalTime().millis();
|
recoveryStatus.phase2Time = stopWatch.totalTime().millis();
|
||||||
recoveryStatus.phase2Operations = snapshot.size();
|
recoveryStatus.phase2Operations = snapshot.size();
|
||||||
|
} finally {
|
||||||
|
sendSnapshotRecoveryThread = null;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override public void phase3(Translog.Snapshot snapshot) throws ElasticSearchException {
|
@Override public void phase3(Translog.Snapshot snapshot) throws ElasticSearchException {
|
||||||
|
sendSnapshotRecoveryThread = Thread.currentThread();
|
||||||
|
try {
|
||||||
|
if (closed) {
|
||||||
|
throw new IndexShardClosedException(shardId);
|
||||||
|
}
|
||||||
logger.trace("Recovery [phase3] to {}: sending [{}] transaction log operations", node, snapshot.size());
|
logger.trace("Recovery [phase3] to {}: sending [{}] transaction log operations", node, snapshot.size());
|
||||||
StopWatch stopWatch = new StopWatch().start();
|
StopWatch stopWatch = new StopWatch().start();
|
||||||
sendSnapshot(snapshot, true);
|
sendSnapshot(snapshot, true);
|
||||||
|
@ -295,6 +355,9 @@ public class RecoveryAction extends AbstractIndexShardComponent {
|
||||||
logger.trace("Recovery [phase3] to {}: took [{}]", node, stopWatch.totalTime());
|
logger.trace("Recovery [phase3] to {}: took [{}]", node, stopWatch.totalTime());
|
||||||
recoveryStatus.phase3Time = stopWatch.totalTime().millis();
|
recoveryStatus.phase3Time = stopWatch.totalTime().millis();
|
||||||
recoveryStatus.phase3Operations = snapshot.size();
|
recoveryStatus.phase3Operations = snapshot.size();
|
||||||
|
} finally {
|
||||||
|
sendSnapshotRecoveryThread = null;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void sendSnapshot(Translog.Snapshot snapshot, boolean phase3) throws ElasticSearchException {
|
private void sendSnapshot(Translog.Snapshot snapshot, boolean phase3) throws ElasticSearchException {
|
||||||
|
@ -371,6 +434,11 @@ public class RecoveryAction extends AbstractIndexShardComponent {
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override public void messageReceived(SnapshotWrapper snapshot, TransportChannel channel) throws Exception {
|
@Override public void messageReceived(SnapshotWrapper snapshot, TransportChannel channel) throws Exception {
|
||||||
|
receiveSnapshotRecoveryThread = Thread.currentThread();
|
||||||
|
try {
|
||||||
|
if (closed) {
|
||||||
|
throw new IndexShardClosedException(shardId);
|
||||||
|
}
|
||||||
if (!snapshot.phase3) {
|
if (!snapshot.phase3) {
|
||||||
// clean open index outputs in any case (there should not be any open, we close then in the chunk)
|
// clean open index outputs in any case (there should not be any open, we close then in the chunk)
|
||||||
cleanOpenIndex();
|
cleanOpenIndex();
|
||||||
|
@ -381,6 +449,9 @@ public class RecoveryAction extends AbstractIndexShardComponent {
|
||||||
// probably need to do more here...
|
// probably need to do more here...
|
||||||
}
|
}
|
||||||
channel.sendResponse(VoidStreamable.INSTANCE);
|
channel.sendResponse(VoidStreamable.INSTANCE);
|
||||||
|
} finally {
|
||||||
|
receiveSnapshotRecoveryThread = null;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -417,6 +488,9 @@ public class RecoveryAction extends AbstractIndexShardComponent {
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override public void messageReceived(FileChunk request, TransportChannel channel) throws Exception {
|
@Override public void messageReceived(FileChunk request, TransportChannel channel) throws Exception {
|
||||||
|
if (closed) {
|
||||||
|
throw new IndexShardClosedException(shardId);
|
||||||
|
}
|
||||||
IndexOutput indexOutput;
|
IndexOutput indexOutput;
|
||||||
if (request.position == 0) {
|
if (request.position == 0) {
|
||||||
// first request
|
// first request
|
||||||
|
|
|
@ -20,6 +20,7 @@
|
||||||
package org.elasticsearch.rest.action;
|
package org.elasticsearch.rest.action;
|
||||||
|
|
||||||
import com.google.inject.AbstractModule;
|
import com.google.inject.AbstractModule;
|
||||||
|
import org.elasticsearch.rest.action.admin.cluster.health.RestClusterHealthAction;
|
||||||
import org.elasticsearch.rest.action.admin.cluster.node.info.RestNodesInfoAction;
|
import org.elasticsearch.rest.action.admin.cluster.node.info.RestNodesInfoAction;
|
||||||
import org.elasticsearch.rest.action.admin.cluster.ping.broadcast.RestBroadcastPingAction;
|
import org.elasticsearch.rest.action.admin.cluster.ping.broadcast.RestBroadcastPingAction;
|
||||||
import org.elasticsearch.rest.action.admin.cluster.ping.replication.RestReplicationPingAction;
|
import org.elasticsearch.rest.action.admin.cluster.ping.replication.RestReplicationPingAction;
|
||||||
|
@ -52,6 +53,7 @@ public class RestActionModule extends AbstractModule {
|
||||||
|
|
||||||
bind(RestNodesInfoAction.class).asEagerSingleton();
|
bind(RestNodesInfoAction.class).asEagerSingleton();
|
||||||
bind(RestClusterStateAction.class).asEagerSingleton();
|
bind(RestClusterStateAction.class).asEagerSingleton();
|
||||||
|
bind(RestClusterHealthAction.class).asEagerSingleton();
|
||||||
|
|
||||||
bind(RestSinglePingAction.class).asEagerSingleton();
|
bind(RestSinglePingAction.class).asEagerSingleton();
|
||||||
bind(RestBroadcastPingAction.class).asEagerSingleton();
|
bind(RestBroadcastPingAction.class).asEagerSingleton();
|
||||||
|
|
|
@ -0,0 +1,142 @@
|
||||||
|
/*
|
||||||
|
* Licensed to Elastic Search and Shay Banon under one
|
||||||
|
* or more contributor license agreements. See the NOTICE file
|
||||||
|
* distributed with this work for additional information
|
||||||
|
* regarding copyright ownership. Elastic Search 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.rest.action.admin.cluster.health;
|
||||||
|
|
||||||
|
import com.google.inject.Inject;
|
||||||
|
import org.elasticsearch.action.ActionListener;
|
||||||
|
import org.elasticsearch.action.admin.cluster.health.*;
|
||||||
|
import org.elasticsearch.client.Client;
|
||||||
|
import org.elasticsearch.rest.*;
|
||||||
|
import org.elasticsearch.rest.action.support.RestActions;
|
||||||
|
import org.elasticsearch.rest.action.support.RestJsonBuilder;
|
||||||
|
import org.elasticsearch.util.json.JsonBuilder;
|
||||||
|
import org.elasticsearch.util.settings.Settings;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
|
||||||
|
import static org.elasticsearch.client.Requests.*;
|
||||||
|
import static org.elasticsearch.rest.RestResponse.Status.*;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author kimchy (shay.banon)
|
||||||
|
*/
|
||||||
|
public class RestClusterHealthAction extends BaseRestHandler {
|
||||||
|
|
||||||
|
@Inject public RestClusterHealthAction(Settings settings, Client client, RestController controller) {
|
||||||
|
super(settings, client);
|
||||||
|
|
||||||
|
controller.registerHandler(RestRequest.Method.GET, "/_cluster/health", this);
|
||||||
|
controller.registerHandler(RestRequest.Method.GET, "/_cluster/health/{index}", this);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override public void handleRequest(final RestRequest request, final RestChannel channel) {
|
||||||
|
ClusterHealthRequest clusterHealthRequest = clusterHealth(RestActions.splitIndices(request.param("index")));
|
||||||
|
int level = 0;
|
||||||
|
try {
|
||||||
|
clusterHealthRequest.timeout(request.paramAsTime("timeout", clusterHealthRequest.timeout()));
|
||||||
|
String waitForStatus = request.param("waitForStatus");
|
||||||
|
if (waitForStatus != null) {
|
||||||
|
clusterHealthRequest.waitForStatus(ClusterHealthStatus.valueOf(waitForStatus.toUpperCase()));
|
||||||
|
}
|
||||||
|
clusterHealthRequest.waitForRelocatingShards(request.paramAsInt("waitForRelocatingShards", clusterHealthRequest.waitForRelocatingShards()));
|
||||||
|
String sLevel = request.param("level");
|
||||||
|
if (sLevel != null) {
|
||||||
|
if ("cluster".equals("sLevel")) {
|
||||||
|
level = 0;
|
||||||
|
} else if ("indices".equals(sLevel)) {
|
||||||
|
level = 1;
|
||||||
|
} else if ("shards".equals(sLevel)) {
|
||||||
|
level = 2;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch (Exception e) {
|
||||||
|
try {
|
||||||
|
JsonBuilder builder = RestJsonBuilder.restJsonBuilder(request);
|
||||||
|
channel.sendResponse(new JsonRestResponse(request, PRECONDITION_FAILED, builder.startObject().field("error", e.getMessage()).endObject()));
|
||||||
|
} catch (IOException e1) {
|
||||||
|
logger.error("Failed to send failure response", e1);
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
final int fLevel = level;
|
||||||
|
client.admin().cluster().execHealth(clusterHealthRequest, new ActionListener<ClusterHealthResponse>() {
|
||||||
|
@Override public void onResponse(ClusterHealthResponse response) {
|
||||||
|
try {
|
||||||
|
JsonBuilder builder = RestJsonBuilder.restJsonBuilder(request);
|
||||||
|
builder.startObject();
|
||||||
|
|
||||||
|
builder.field("status", response.status().name().toLowerCase());
|
||||||
|
builder.field("timedOut", response.timedOut());
|
||||||
|
builder.field("activePrimaryShards", response.activePrimaryShards());
|
||||||
|
builder.field("activeShards", response.activeShards());
|
||||||
|
builder.field("relocatingShards", response.relocatingShards());
|
||||||
|
|
||||||
|
if (fLevel > 0) {
|
||||||
|
builder.startObject("indices");
|
||||||
|
for (ClusterIndexHealth indexHealth : response) {
|
||||||
|
builder.startObject(indexHealth.index());
|
||||||
|
|
||||||
|
builder.field("status", indexHealth.status().name().toLowerCase());
|
||||||
|
builder.field("numberOfShards", indexHealth.numberOfShards());
|
||||||
|
builder.field("numberOfReplicas", indexHealth.numberOfReplicas());
|
||||||
|
builder.field("activePrimaryShards", indexHealth.activePrimaryShards());
|
||||||
|
builder.field("activeShards", indexHealth.activeShards());
|
||||||
|
builder.field("relocatingShards", indexHealth.relocatingShards());
|
||||||
|
|
||||||
|
if (fLevel > 1) {
|
||||||
|
builder.startObject("shards");
|
||||||
|
|
||||||
|
for (ClusterShardHealth shardHealth : indexHealth) {
|
||||||
|
builder.startObject(Integer.toString(shardHealth.id()));
|
||||||
|
|
||||||
|
builder.field("status", shardHealth.status().name().toLowerCase());
|
||||||
|
builder.field("primaryActive", shardHealth.primaryActive());
|
||||||
|
builder.field("activeShards", shardHealth.activeShards());
|
||||||
|
builder.field("relocatingShards", shardHealth.relocatingShards());
|
||||||
|
|
||||||
|
builder.endObject();
|
||||||
|
}
|
||||||
|
|
||||||
|
builder.endObject();
|
||||||
|
}
|
||||||
|
|
||||||
|
builder.endObject();
|
||||||
|
}
|
||||||
|
builder.endObject();
|
||||||
|
}
|
||||||
|
|
||||||
|
builder.endObject();
|
||||||
|
|
||||||
|
channel.sendResponse(new JsonRestResponse(request, RestResponse.Status.OK, builder));
|
||||||
|
} catch (Exception e) {
|
||||||
|
onFailure(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override public void onFailure(Throwable e) {
|
||||||
|
try {
|
||||||
|
channel.sendResponse(new JsonThrowableRestResponse(request, e));
|
||||||
|
} catch (IOException e1) {
|
||||||
|
logger.error("Failed to send failure response", e1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
|
@ -19,6 +19,10 @@
|
||||||
|
|
||||||
package org.elasticsearch.test.integration.indexlifecycle;
|
package org.elasticsearch.test.integration.indexlifecycle;
|
||||||
|
|
||||||
|
import org.elasticsearch.action.admin.cluster.health.ClusterHealthResponse;
|
||||||
|
import org.elasticsearch.action.admin.cluster.health.ClusterHealthStatus;
|
||||||
|
import org.elasticsearch.action.admin.indices.create.CreateIndexResponse;
|
||||||
|
import org.elasticsearch.action.admin.indices.delete.DeleteIndexResponse;
|
||||||
import org.elasticsearch.cluster.ClusterService;
|
import org.elasticsearch.cluster.ClusterService;
|
||||||
import org.elasticsearch.cluster.ClusterState;
|
import org.elasticsearch.cluster.ClusterState;
|
||||||
import org.elasticsearch.cluster.routing.RoutingNode;
|
import org.elasticsearch.cluster.routing.RoutingNode;
|
||||||
|
@ -61,9 +65,18 @@ public class IndexLifecycleActionTests extends AbstractServersTests {
|
||||||
ClusterService clusterService1 = ((InternalServer) server("server1")).injector().getInstance(ClusterService.class);
|
ClusterService clusterService1 = ((InternalServer) server("server1")).injector().getInstance(ClusterService.class);
|
||||||
|
|
||||||
logger.info("Creating index [test]");
|
logger.info("Creating index [test]");
|
||||||
client("server1").admin().indices().create(createIndexRequest("test")).actionGet();
|
CreateIndexResponse createIndexResponse = client("server1").admin().indices().create(createIndexRequest("test")).actionGet();
|
||||||
|
assertThat(createIndexResponse.acknowledged(), equalTo(true));
|
||||||
|
|
||||||
|
logger.info("Running Cluster Health");
|
||||||
|
ClusterHealthResponse clusterHealth = client("server1").admin().cluster().health(clusterHealth().waitForYellowStatus()).actionGet();
|
||||||
|
logger.info("Done Cluster Health, status " + clusterHealth.status());
|
||||||
|
assertThat(clusterHealth.timedOut(), equalTo(false));
|
||||||
|
assertThat(clusterHealth.status(), equalTo(ClusterHealthStatus.YELLOW));
|
||||||
|
|
||||||
|
// sleep till the cluster state gets published, since we check the master
|
||||||
|
Thread.sleep(200);
|
||||||
|
|
||||||
Thread.sleep(1000);
|
|
||||||
|
|
||||||
ClusterState clusterState1 = clusterService1.state();
|
ClusterState clusterState1 = clusterService1.state();
|
||||||
RoutingNode routingNodeEntry1 = clusterState1.routingNodes().nodesToShards().get(clusterState1.nodes().localNodeId());
|
RoutingNode routingNodeEntry1 = clusterState1.routingNodes().nodesToShards().get(clusterState1.nodes().localNodeId());
|
||||||
|
@ -79,7 +92,14 @@ public class IndexLifecycleActionTests extends AbstractServersTests {
|
||||||
|
|
||||||
ClusterService clusterService2 = ((InternalServer) server("server2")).injector().getInstance(ClusterService.class);
|
ClusterService clusterService2 = ((InternalServer) server("server2")).injector().getInstance(ClusterService.class);
|
||||||
|
|
||||||
Thread.sleep(1500);
|
logger.info("Running Cluster Health");
|
||||||
|
clusterHealth = client("server1").admin().cluster().health(clusterHealth().waitForGreenStatus()).actionGet();
|
||||||
|
logger.info("Done Cluster Health, status " + clusterHealth.status());
|
||||||
|
assertThat(clusterHealth.timedOut(), equalTo(false));
|
||||||
|
assertThat(clusterHealth.status(), equalTo(ClusterHealthStatus.GREEN));
|
||||||
|
|
||||||
|
// sleep till the cluster state gets published, since we check the master
|
||||||
|
Thread.sleep(200);
|
||||||
|
|
||||||
clusterState1 = clusterService1.state();
|
clusterState1 = clusterService1.state();
|
||||||
routingNodeEntry1 = clusterState1.routingNodes().nodesToShards().get(clusterState1.nodes().localNodeId());
|
routingNodeEntry1 = clusterState1.routingNodes().nodesToShards().get(clusterState1.nodes().localNodeId());
|
||||||
|
@ -95,7 +115,17 @@ public class IndexLifecycleActionTests extends AbstractServersTests {
|
||||||
|
|
||||||
ClusterService clusterService3 = ((InternalServer) server("server3")).injector().getInstance(ClusterService.class);
|
ClusterService clusterService3 = ((InternalServer) server("server3")).injector().getInstance(ClusterService.class);
|
||||||
|
|
||||||
Thread.sleep(1500);
|
logger.info("Running Cluster Health");
|
||||||
|
clusterHealth = client("server1").admin().cluster().health(clusterHealth().waitForGreenStatus().waitForRelocatingShards(0)).actionGet();
|
||||||
|
logger.info("Done Cluster Health, status " + clusterHealth.status());
|
||||||
|
assertThat(clusterHealth.timedOut(), equalTo(false));
|
||||||
|
assertThat(clusterHealth.status(), equalTo(ClusterHealthStatus.GREEN));
|
||||||
|
assertThat(clusterHealth.relocatingShards(), equalTo(0));
|
||||||
|
assertThat(clusterHealth.activeShards(), equalTo(22));
|
||||||
|
assertThat(clusterHealth.activePrimaryShards(), equalTo(11));
|
||||||
|
|
||||||
|
// sleep till the cluster state gets published, since we check the master
|
||||||
|
Thread.sleep(200);
|
||||||
|
|
||||||
clusterState1 = clusterService1.state();
|
clusterState1 = clusterService1.state();
|
||||||
routingNodeEntry1 = clusterState1.routingNodes().nodesToShards().get(clusterState1.nodes().localNodeId());
|
routingNodeEntry1 = clusterState1.routingNodes().nodesToShards().get(clusterState1.nodes().localNodeId());
|
||||||
|
@ -114,8 +144,20 @@ public class IndexLifecycleActionTests extends AbstractServersTests {
|
||||||
logger.info("Closing server1");
|
logger.info("Closing server1");
|
||||||
// kill the first server
|
// kill the first server
|
||||||
closeServer("server1");
|
closeServer("server1");
|
||||||
|
// wait a bit so it will be discovered as removed
|
||||||
|
Thread.sleep(200);
|
||||||
|
// verify health
|
||||||
|
logger.info("Running Cluster Health");
|
||||||
|
clusterHealth = client("server2").admin().cluster().health(clusterHealth().waitForGreenStatus().waitForRelocatingShards(0)).actionGet();
|
||||||
|
logger.info("Done Cluster Health, status " + clusterHealth.status());
|
||||||
|
assertThat(clusterHealth.timedOut(), equalTo(false));
|
||||||
|
assertThat(clusterHealth.status(), equalTo(ClusterHealthStatus.GREEN));
|
||||||
|
assertThat(clusterHealth.relocatingShards(), equalTo(0));
|
||||||
|
assertThat(clusterHealth.activeShards(), equalTo(22));
|
||||||
|
assertThat(clusterHealth.activePrimaryShards(), equalTo(11));
|
||||||
|
|
||||||
Thread.sleep(1500);
|
// sleep till the cluster state gets published, since we check the master
|
||||||
|
Thread.sleep(200);
|
||||||
|
|
||||||
clusterState2 = clusterService2.state();
|
clusterState2 = clusterService2.state();
|
||||||
routingNodeEntry2 = clusterState2.routingNodes().nodesToShards().get(clusterState2.nodes().localNodeId());
|
routingNodeEntry2 = clusterState2.routingNodes().nodesToShards().get(clusterState2.nodes().localNodeId());
|
||||||
|
@ -129,9 +171,10 @@ public class IndexLifecycleActionTests extends AbstractServersTests {
|
||||||
|
|
||||||
logger.info("Deleting index [test]");
|
logger.info("Deleting index [test]");
|
||||||
// last, lets delete the index
|
// last, lets delete the index
|
||||||
client("server2").admin().indices().delete(deleteIndexRequest("test")).actionGet();
|
DeleteIndexResponse deleteIndexResponse = client("server2").admin().indices().delete(deleteIndexRequest("test")).actionGet();
|
||||||
|
assertThat(deleteIndexResponse.acknowledged(), equalTo(true));
|
||||||
|
|
||||||
Thread.sleep(1500);
|
Thread.sleep(200);
|
||||||
clusterState2 = clusterService2.state();
|
clusterState2 = clusterService2.state();
|
||||||
routingNodeEntry2 = clusterState2.routingNodes().nodesToShards().get(clusterState2.nodes().localNodeId());
|
routingNodeEntry2 = clusterState2.routingNodes().nodesToShards().get(clusterState2.nodes().localNodeId());
|
||||||
assertThat(routingNodeEntry2, nullValue());
|
assertThat(routingNodeEntry2, nullValue());
|
||||||
|
@ -155,9 +198,17 @@ public class IndexLifecycleActionTests extends AbstractServersTests {
|
||||||
ClusterService clusterService1 = ((InternalServer) server("server1")).injector().getInstance(ClusterService.class);
|
ClusterService clusterService1 = ((InternalServer) server("server1")).injector().getInstance(ClusterService.class);
|
||||||
|
|
||||||
logger.info("Creating index [test]");
|
logger.info("Creating index [test]");
|
||||||
client("server1").admin().indices().create(createIndexRequest("test")).actionGet();
|
CreateIndexResponse createIndexResponse = client("server1").admin().indices().create(createIndexRequest("test")).actionGet();
|
||||||
|
assertThat(createIndexResponse.acknowledged(), equalTo(true));
|
||||||
|
|
||||||
Thread.sleep(1000);
|
logger.info("Running Cluster Health");
|
||||||
|
ClusterHealthResponse clusterHealth = client("server1").admin().cluster().health(clusterHealth().waitForGreenStatus()).actionGet();
|
||||||
|
logger.info("Done Cluster Health, status " + clusterHealth.status());
|
||||||
|
assertThat(clusterHealth.timedOut(), equalTo(false));
|
||||||
|
assertThat(clusterHealth.status(), equalTo(ClusterHealthStatus.GREEN));
|
||||||
|
assertThat(clusterHealth.relocatingShards(), equalTo(0));
|
||||||
|
assertThat(clusterHealth.activeShards(), equalTo(11));
|
||||||
|
assertThat(clusterHealth.activePrimaryShards(), equalTo(11));
|
||||||
|
|
||||||
ClusterState clusterState1 = clusterService1.state();
|
ClusterState clusterState1 = clusterService1.state();
|
||||||
RoutingNode routingNodeEntry1 = clusterState1.routingNodes().nodesToShards().get(clusterState1.nodes().localNodeId());
|
RoutingNode routingNodeEntry1 = clusterState1.routingNodes().nodesToShards().get(clusterState1.nodes().localNodeId());
|
||||||
|
@ -166,11 +217,23 @@ public class IndexLifecycleActionTests extends AbstractServersTests {
|
||||||
// start another server
|
// start another server
|
||||||
logger.info("Starting server2");
|
logger.info("Starting server2");
|
||||||
startServer("server2", settings);
|
startServer("server2", settings);
|
||||||
|
// wait a bit
|
||||||
|
Thread.sleep(200);
|
||||||
|
|
||||||
|
logger.info("Running Cluster Health");
|
||||||
|
clusterHealth = client("server1").admin().cluster().health(clusterHealth().waitForGreenStatus().waitForRelocatingShards(0)).actionGet();
|
||||||
|
logger.info("Done Cluster Health, status " + clusterHealth.status());
|
||||||
|
assertThat(clusterHealth.timedOut(), equalTo(false));
|
||||||
|
assertThat(clusterHealth.status(), equalTo(ClusterHealthStatus.GREEN));
|
||||||
|
assertThat(clusterHealth.relocatingShards(), equalTo(0));
|
||||||
|
assertThat(clusterHealth.activeShards(), equalTo(11));
|
||||||
|
assertThat(clusterHealth.activePrimaryShards(), equalTo(11));
|
||||||
|
|
||||||
|
// sleep till the cluster state gets published, since we check the master
|
||||||
|
Thread.sleep(200);
|
||||||
|
|
||||||
ClusterService clusterService2 = ((InternalServer) server("server2")).injector().getInstance(ClusterService.class);
|
ClusterService clusterService2 = ((InternalServer) server("server2")).injector().getInstance(ClusterService.class);
|
||||||
|
|
||||||
Thread.sleep(2000);
|
|
||||||
|
|
||||||
clusterState1 = clusterService1.state();
|
clusterState1 = clusterService1.state();
|
||||||
routingNodeEntry1 = clusterState1.routingNodes().nodesToShards().get(clusterState1.nodes().localNodeId());
|
routingNodeEntry1 = clusterState1.routingNodes().nodesToShards().get(clusterState1.nodes().localNodeId());
|
||||||
assertThat(routingNodeEntry1.numberOfShardsWithState(STARTED), anyOf(equalTo(6), equalTo(5)));
|
assertThat(routingNodeEntry1.numberOfShardsWithState(STARTED), anyOf(equalTo(6), equalTo(5)));
|
||||||
|
@ -182,10 +245,22 @@ public class IndexLifecycleActionTests extends AbstractServersTests {
|
||||||
// start another server
|
// start another server
|
||||||
logger.info("Starting server3");
|
logger.info("Starting server3");
|
||||||
startServer("server3");
|
startServer("server3");
|
||||||
|
// wait a bit so assignment will start
|
||||||
|
Thread.sleep(200);
|
||||||
|
|
||||||
ClusterService clusterService3 = ((InternalServer) server("server3")).injector().getInstance(ClusterService.class);
|
ClusterService clusterService3 = ((InternalServer) server("server3")).injector().getInstance(ClusterService.class);
|
||||||
|
|
||||||
Thread.sleep(1500);
|
logger.info("Running Cluster Health");
|
||||||
|
clusterHealth = client("server1").admin().cluster().health(clusterHealth().waitForGreenStatus().waitForRelocatingShards(0)).actionGet();
|
||||||
|
logger.info("Done Cluster Health, status " + clusterHealth.status());
|
||||||
|
assertThat(clusterHealth.timedOut(), equalTo(false));
|
||||||
|
assertThat(clusterHealth.status(), equalTo(ClusterHealthStatus.GREEN));
|
||||||
|
assertThat(clusterHealth.relocatingShards(), equalTo(0));
|
||||||
|
assertThat(clusterHealth.activeShards(), equalTo(11));
|
||||||
|
assertThat(clusterHealth.activePrimaryShards(), equalTo(11));
|
||||||
|
|
||||||
|
// sleep till the cluster state gets published, since we check the master
|
||||||
|
Thread.sleep(200);
|
||||||
|
|
||||||
clusterState1 = clusterService1.state();
|
clusterState1 = clusterService1.state();
|
||||||
routingNodeEntry1 = clusterState1.routingNodes().nodesToShards().get(clusterState1.nodes().localNodeId());
|
routingNodeEntry1 = clusterState1.routingNodes().nodesToShards().get(clusterState1.nodes().localNodeId());
|
||||||
|
@ -204,8 +279,20 @@ public class IndexLifecycleActionTests extends AbstractServersTests {
|
||||||
logger.info("Closing server1");
|
logger.info("Closing server1");
|
||||||
// kill the first server
|
// kill the first server
|
||||||
closeServer("server1");
|
closeServer("server1");
|
||||||
|
// wait a bit so it will be discovered as removed
|
||||||
|
Thread.sleep(200);
|
||||||
|
|
||||||
Thread.sleep(2000);
|
logger.info("Running Cluster Health");
|
||||||
|
clusterHealth = client("server3").admin().cluster().health(clusterHealth().waitForGreenStatus().waitForRelocatingShards(0)).actionGet();
|
||||||
|
logger.info("Done Cluster Health, status " + clusterHealth.status());
|
||||||
|
assertThat(clusterHealth.timedOut(), equalTo(false));
|
||||||
|
assertThat(clusterHealth.status(), equalTo(ClusterHealthStatus.GREEN));
|
||||||
|
assertThat(clusterHealth.relocatingShards(), equalTo(0));
|
||||||
|
assertThat(clusterHealth.activeShards(), equalTo(11));
|
||||||
|
assertThat(clusterHealth.activePrimaryShards(), equalTo(11));
|
||||||
|
|
||||||
|
// sleep till the cluster state gets published, since we check the master
|
||||||
|
Thread.sleep(200);
|
||||||
|
|
||||||
clusterState2 = clusterService2.state();
|
clusterState2 = clusterService2.state();
|
||||||
routingNodeEntry2 = clusterState2.routingNodes().nodesToShards().get(clusterState2.nodes().localNodeId());
|
routingNodeEntry2 = clusterState2.routingNodes().nodesToShards().get(clusterState2.nodes().localNodeId());
|
||||||
|
@ -219,9 +306,9 @@ public class IndexLifecycleActionTests extends AbstractServersTests {
|
||||||
|
|
||||||
logger.info("Deleting index [test]");
|
logger.info("Deleting index [test]");
|
||||||
// last, lets delete the index
|
// last, lets delete the index
|
||||||
client("server2").admin().indices().delete(deleteIndexRequest("test")).actionGet();
|
DeleteIndexResponse deleteIndexResponse = client("server2").admin().indices().delete(deleteIndexRequest("test")).actionGet();
|
||||||
|
assertThat(deleteIndexResponse.acknowledged(), equalTo(true));
|
||||||
|
|
||||||
Thread.sleep(2000);
|
|
||||||
clusterState2 = clusterService2.state();
|
clusterState2 = clusterService2.state();
|
||||||
routingNodeEntry2 = clusterState2.routingNodes().nodesToShards().get(clusterState2.nodes().localNodeId());
|
routingNodeEntry2 = clusterState2.routingNodes().nodesToShards().get(clusterState2.nodes().localNodeId());
|
||||||
assertThat(routingNodeEntry2, nullValue());
|
assertThat(routingNodeEntry2, nullValue());
|
||||||
|
|
Loading…
Reference in New Issue