diff --git a/hbase-client/src/main/java/org/apache/hadoop/hbase/client/AsyncRequestFutureImpl.java b/hbase-client/src/main/java/org/apache/hadoop/hbase/client/AsyncRequestFutureImpl.java index b34ef863d565..f8b6096c771f 100644 --- a/hbase-client/src/main/java/org/apache/hadoop/hbase/client/AsyncRequestFutureImpl.java +++ b/hbase-client/src/main/java/org/apache/hadoop/hbase/client/AsyncRequestFutureImpl.java @@ -731,7 +731,7 @@ Retry manageError(int originalIndex, Row row, Retry canRetry, Throwable throwabl if (canRetry != Retry.YES) { // Batch.Callback was not called on failure in 0.94. We keep this. setError(originalIndex, row, throwable, server); - } else if (isActionComplete(originalIndex, row)) { + } else if (AsyncProcess.isReplicaGet(row) && isActionComplete(originalIndex)) { canRetry = Retry.NO_OTHER_SUCCEEDED; } return canRetry; @@ -1164,10 +1164,15 @@ private void setError(int index, Row row, Throwable throwable, ServerName server * synchronize, assuming element index/field accesses are atomic. This is an opportunistic * optimization check, doesn't have to be strict. * @param index Original action index. - * @param row Original request. + * @return If results are non-null, returns whether a result is currently set for action index. + * @throws IllegalStateException If results are null. We cannot readily tell here whether an + * action that is part of an AsyncRequestFuture that doesn't track results is complete */ - private boolean isActionComplete(int index, Row row) { - if (!AsyncProcess.isReplicaGet(row)) return false; + private boolean isActionComplete(int index) { + if (results == null) { + throw new IllegalStateException( + String.format("Cannot check action %d completion when results are null", index)); + } Object resObj = results[index]; return (resObj != null) && (!(resObj instanceof ReplicaResultState) || ((ReplicaResultState) resObj).callCount == 0);