mirror of
https://github.com/honeymoose/OpenSearch.git
synced 2025-04-18 05:10:33 +00:00
This is related to #27933. It renames the core module to server. This is the first step towards introducing an elasticsearch-core jar.
255 lines
8.4 KiB
Java
255 lines
8.4 KiB
Java
/*
|
|
* Licensed to Elasticsearch under one or more contributor
|
|
* license agreements. See the NOTICE file distributed with
|
|
* this work for additional information regarding copyright
|
|
* ownership. Elasticsearch licenses this file to you under
|
|
* the Apache License, Version 2.0 (the "License"); you may
|
|
* not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing,
|
|
* software distributed under the License is distributed on an
|
|
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
|
|
* KIND, either express or implied. See the License for the
|
|
* specific language governing permissions and limitations
|
|
* under the License.
|
|
*/
|
|
|
|
package org.elasticsearch;
|
|
|
|
import org.apache.logging.log4j.Logger;
|
|
import org.apache.lucene.index.CorruptIndexException;
|
|
import org.apache.lucene.index.IndexFormatTooNewException;
|
|
import org.apache.lucene.index.IndexFormatTooOldException;
|
|
import org.elasticsearch.action.ShardOperationFailedException;
|
|
import org.elasticsearch.common.Nullable;
|
|
import org.elasticsearch.common.logging.Loggers;
|
|
import org.elasticsearch.index.Index;
|
|
import org.elasticsearch.rest.RestStatus;
|
|
|
|
import java.io.IOException;
|
|
import java.io.PrintWriter;
|
|
import java.io.StringWriter;
|
|
import java.util.ArrayList;
|
|
import java.util.HashSet;
|
|
import java.util.List;
|
|
import java.util.Set;
|
|
|
|
public final class ExceptionsHelper {
|
|
|
|
private static final Logger logger = Loggers.getLogger(ExceptionsHelper.class);
|
|
|
|
public static RuntimeException convertToRuntime(Exception e) {
|
|
if (e instanceof RuntimeException) {
|
|
return (RuntimeException) e;
|
|
}
|
|
return new ElasticsearchException(e);
|
|
}
|
|
|
|
public static ElasticsearchException convertToElastic(Exception e) {
|
|
if (e instanceof ElasticsearchException) {
|
|
return (ElasticsearchException) e;
|
|
}
|
|
return new ElasticsearchException(e);
|
|
}
|
|
|
|
public static RestStatus status(Throwable t) {
|
|
if (t != null) {
|
|
if (t instanceof ElasticsearchException) {
|
|
return ((ElasticsearchException) t).status();
|
|
} else if (t instanceof IllegalArgumentException) {
|
|
return RestStatus.BAD_REQUEST;
|
|
}
|
|
}
|
|
return RestStatus.INTERNAL_SERVER_ERROR;
|
|
}
|
|
|
|
public static Throwable unwrapCause(Throwable t) {
|
|
int counter = 0;
|
|
Throwable result = t;
|
|
while (result instanceof ElasticsearchWrapperException) {
|
|
if (result.getCause() == null) {
|
|
return result;
|
|
}
|
|
if (result.getCause() == result) {
|
|
return result;
|
|
}
|
|
if (counter++ > 10) {
|
|
// dear god, if we got more than 10 levels down, WTF? just bail
|
|
logger.warn("Exception cause unwrapping ran for 10 levels...", t);
|
|
return result;
|
|
}
|
|
result = result.getCause();
|
|
}
|
|
return result;
|
|
}
|
|
|
|
/**
|
|
* @deprecated Don't swallow exceptions, allow them to propagate.
|
|
*/
|
|
@Deprecated
|
|
public static String detailedMessage(Throwable t) {
|
|
if (t == null) {
|
|
return "Unknown";
|
|
}
|
|
if (t.getCause() != null) {
|
|
StringBuilder sb = new StringBuilder();
|
|
while (t != null) {
|
|
sb.append(t.getClass().getSimpleName());
|
|
if (t.getMessage() != null) {
|
|
sb.append("[");
|
|
sb.append(t.getMessage());
|
|
sb.append("]");
|
|
}
|
|
sb.append("; ");
|
|
t = t.getCause();
|
|
if (t != null) {
|
|
sb.append("nested: ");
|
|
}
|
|
}
|
|
return sb.toString();
|
|
} else {
|
|
return t.getClass().getSimpleName() + "[" + t.getMessage() + "]";
|
|
}
|
|
}
|
|
|
|
public static String stackTrace(Throwable e) {
|
|
StringWriter stackTraceStringWriter = new StringWriter();
|
|
PrintWriter printWriter = new PrintWriter(stackTraceStringWriter);
|
|
e.printStackTrace(printWriter);
|
|
return stackTraceStringWriter.toString();
|
|
}
|
|
|
|
/**
|
|
* Rethrows the first exception in the list and adds all remaining to the suppressed list.
|
|
* If the given list is empty no exception is thrown
|
|
*
|
|
*/
|
|
public static <T extends Throwable> void rethrowAndSuppress(List<T> exceptions) throws T {
|
|
T main = null;
|
|
for (T ex : exceptions) {
|
|
main = useOrSuppress(main, ex);
|
|
}
|
|
if (main != null) {
|
|
throw main;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Throws a runtime exception with all given exceptions added as suppressed.
|
|
* If the given list is empty no exception is thrown
|
|
*/
|
|
public static <T extends Throwable> void maybeThrowRuntimeAndSuppress(List<T> exceptions) {
|
|
T main = null;
|
|
for (T ex : exceptions) {
|
|
main = useOrSuppress(main, ex);
|
|
}
|
|
if (main != null) {
|
|
throw new ElasticsearchException(main);
|
|
}
|
|
}
|
|
|
|
public static <T extends Throwable> T useOrSuppress(T first, T second) {
|
|
if (first == null) {
|
|
return second;
|
|
} else {
|
|
first.addSuppressed(second);
|
|
}
|
|
return first;
|
|
}
|
|
|
|
public static IOException unwrapCorruption(Throwable t) {
|
|
return (IOException) unwrap(t, CorruptIndexException.class,
|
|
IndexFormatTooOldException.class,
|
|
IndexFormatTooNewException.class);
|
|
}
|
|
|
|
public static Throwable unwrap(Throwable t, Class<?>... clazzes) {
|
|
if (t != null) {
|
|
do {
|
|
for (Class<?> clazz : clazzes) {
|
|
if (clazz.isInstance(t)) {
|
|
return t;
|
|
}
|
|
}
|
|
} while ((t = t.getCause()) != null);
|
|
}
|
|
return null;
|
|
}
|
|
|
|
/**
|
|
* Throws the specified exception. If null if specified then <code>true</code> is returned.
|
|
*/
|
|
public static boolean reThrowIfNotNull(@Nullable Throwable e) {
|
|
if (e != null) {
|
|
if (e instanceof RuntimeException) {
|
|
throw (RuntimeException) e;
|
|
} else {
|
|
throw new RuntimeException(e);
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* Deduplicate the failures by exception message and index.
|
|
*/
|
|
public static ShardOperationFailedException[] groupBy(ShardOperationFailedException[] failures) {
|
|
List<ShardOperationFailedException> uniqueFailures = new ArrayList<>();
|
|
Set<GroupBy> reasons = new HashSet<>();
|
|
for (ShardOperationFailedException failure : failures) {
|
|
GroupBy reason = new GroupBy(failure.getCause());
|
|
if (reasons.contains(reason) == false) {
|
|
reasons.add(reason);
|
|
uniqueFailures.add(failure);
|
|
}
|
|
}
|
|
return uniqueFailures.toArray(new ShardOperationFailedException[0]);
|
|
}
|
|
|
|
static class GroupBy {
|
|
final String reason;
|
|
final String index;
|
|
final Class<? extends Throwable> causeType;
|
|
|
|
GroupBy(Throwable t) {
|
|
if (t instanceof ElasticsearchException) {
|
|
final Index index = ((ElasticsearchException) t).getIndex();
|
|
if (index != null) {
|
|
this.index = index.getName();
|
|
} else {
|
|
this.index = null;
|
|
}
|
|
} else {
|
|
index = null;
|
|
}
|
|
reason = t.getMessage();
|
|
causeType = t.getClass();
|
|
}
|
|
|
|
@Override
|
|
public boolean equals(Object o) {
|
|
if (this == o) return true;
|
|
if (o == null || getClass() != o.getClass()) return false;
|
|
|
|
GroupBy groupBy = (GroupBy) o;
|
|
|
|
if (!causeType.equals(groupBy.causeType)) return false;
|
|
if (index != null ? !index.equals(groupBy.index) : groupBy.index != null) return false;
|
|
if (reason != null ? !reason.equals(groupBy.reason) : groupBy.reason != null) return false;
|
|
|
|
return true;
|
|
}
|
|
|
|
@Override
|
|
public int hashCode() {
|
|
int result = reason != null ? reason.hashCode() : 0;
|
|
result = 31 * result + (index != null ? index.hashCode() : 0);
|
|
result = 31 * result + causeType.hashCode();
|
|
return result;
|
|
}
|
|
}
|
|
}
|