2015-08-21 01:19:29 -04:00
|
|
|
/*
|
|
|
|
* 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.bootstrap;
|
|
|
|
|
|
|
|
import org.elasticsearch.common.inject.CreationException;
|
|
|
|
import org.elasticsearch.common.inject.spi.Message;
|
|
|
|
|
|
|
|
import java.io.PrintStream;
|
2016-08-19 15:10:54 -04:00
|
|
|
import java.io.PrintWriter;
|
|
|
|
import java.util.function.Consumer;
|
2015-08-21 01:19:29 -04:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Wraps an exception in a special way that it gets formatted
|
|
|
|
* "reasonably". This means limits on stacktrace frames and
|
|
|
|
* cleanup for guice, and some guidance about consulting full
|
|
|
|
* logs for the whole exception.
|
|
|
|
*/
|
|
|
|
//TODO: remove this when guice is removed, and exceptions are cleaned up
|
|
|
|
//this is horrible, but its what we must do
|
2016-08-19 14:51:54 -04:00
|
|
|
final class StartupException extends RuntimeException {
|
|
|
|
|
2015-08-21 01:19:29 -04:00
|
|
|
/** maximum length of a stacktrace, before we truncate it */
|
|
|
|
static final int STACKTRACE_LIMIT = 30;
|
|
|
|
/** all lines from this package are RLE-compressed */
|
|
|
|
static final String GUICE_PACKAGE = "org.elasticsearch.common.inject";
|
2016-08-19 14:51:54 -04:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Create a new StartupException that will format {@code cause}
|
2015-08-21 01:19:29 -04:00
|
|
|
* to the console on failure.
|
|
|
|
*/
|
2016-08-19 14:51:54 -04:00
|
|
|
StartupException(Throwable cause) {
|
2015-08-21 01:19:29 -04:00
|
|
|
super(cause);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* This logic actually prints the exception to the console, its
|
|
|
|
* what is invoked by the JVM when we throw the exception from main()
|
|
|
|
*/
|
|
|
|
@Override
|
|
|
|
public void printStackTrace(PrintStream s) {
|
2016-08-19 15:10:54 -04:00
|
|
|
printStackTrace(s::println);
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public void printStackTrace(PrintWriter s) {
|
|
|
|
printStackTrace(s::println);
|
|
|
|
}
|
|
|
|
|
|
|
|
private void printStackTrace(Consumer<String> consumer) {
|
2015-08-21 01:19:29 -04:00
|
|
|
Throwable originalCause = getCause();
|
|
|
|
Throwable cause = originalCause;
|
|
|
|
if (cause instanceof CreationException) {
|
|
|
|
cause = getFirstGuiceCause((CreationException)cause);
|
|
|
|
}
|
2016-08-19 14:51:54 -04:00
|
|
|
|
2015-08-21 18:21:01 -04:00
|
|
|
String message = cause.toString();
|
2016-08-19 15:10:54 -04:00
|
|
|
consumer.accept(message);
|
2016-08-19 14:51:54 -04:00
|
|
|
|
2015-08-21 01:19:29 -04:00
|
|
|
if (cause != null) {
|
|
|
|
// walk to the root cause
|
|
|
|
while (cause.getCause() != null) {
|
|
|
|
cause = cause.getCause();
|
|
|
|
}
|
|
|
|
|
|
|
|
// print the root cause message, only if it differs!
|
|
|
|
if (cause != originalCause && (message.equals(cause.toString()) == false)) {
|
2016-08-19 15:10:54 -04:00
|
|
|
consumer.accept("Likely root cause: " + cause);
|
2015-08-21 01:19:29 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// print stacktrace of cause
|
|
|
|
StackTraceElement stack[] = cause.getStackTrace();
|
|
|
|
int linesWritten = 0;
|
|
|
|
for (int i = 0; i < stack.length; i++) {
|
|
|
|
if (linesWritten == STACKTRACE_LIMIT) {
|
2016-08-19 15:10:54 -04:00
|
|
|
consumer.accept("\t<<<truncated>>>");
|
2015-08-21 01:19:29 -04:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
String line = stack[i].toString();
|
2016-08-19 14:51:54 -04:00
|
|
|
|
2015-08-21 01:19:29 -04:00
|
|
|
// skip past contiguous runs of this garbage:
|
|
|
|
if (line.startsWith(GUICE_PACKAGE)) {
|
|
|
|
while (i + 1 < stack.length && stack[i + 1].toString().startsWith(GUICE_PACKAGE)) {
|
|
|
|
i++;
|
|
|
|
}
|
2016-08-19 15:10:54 -04:00
|
|
|
consumer.accept("\tat <<<guice>>>");
|
2015-08-21 01:19:29 -04:00
|
|
|
linesWritten++;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2016-12-02 23:11:41 +08:00
|
|
|
consumer.accept("\tat " + line);
|
2015-08-21 01:19:29 -04:00
|
|
|
linesWritten++;
|
|
|
|
}
|
|
|
|
}
|
2015-09-25 14:46:12 -04:00
|
|
|
// if its a guice exception, the whole thing really will not be in the log, its megabytes.
|
|
|
|
// refer to the hack in bootstrap, where we don't log it
|
|
|
|
if (originalCause instanceof CreationException == false) {
|
2019-11-22 14:58:17 +00:00
|
|
|
final String basePath = System.getProperty("es.logs.base_path");
|
|
|
|
// It's possible to fail before logging has been configured, in which case there's no point
|
|
|
|
// suggested that the user look in the log file.
|
|
|
|
if (basePath != null) {
|
|
|
|
final String logPath = System.getProperty("es.logs.base_path")
|
|
|
|
+ System.getProperty("file.separator")
|
|
|
|
+ System.getProperty("es.logs.cluster_name")
|
|
|
|
+ ".log";
|
|
|
|
|
|
|
|
consumer.accept("For complete error details, refer to the log at " + logPath);
|
|
|
|
}
|
2015-09-25 14:46:12 -04:00
|
|
|
}
|
2015-08-21 01:19:29 -04:00
|
|
|
}
|
2016-08-19 14:51:54 -04:00
|
|
|
|
|
|
|
/**
|
2015-08-21 01:19:29 -04:00
|
|
|
* Returns first cause from a guice error (it can have multiple).
|
|
|
|
*/
|
|
|
|
static Throwable getFirstGuiceCause(CreationException guice) {
|
|
|
|
for (Message message : guice.getErrorMessages()) {
|
|
|
|
Throwable cause = message.getCause();
|
|
|
|
if (cause != null) {
|
|
|
|
return cause;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return guice; // we tried
|
|
|
|
}
|
2016-08-19 15:10:54 -04:00
|
|
|
|
2015-08-21 01:19:29 -04:00
|
|
|
}
|