/* * 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; } } }