321 lines
12 KiB
Java
Raw Normal View History

2010-02-08 15:30:06 +02: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
2010-02-08 15:30:06 +02:00
*
* 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.apache.lucene.util.StringHelper;
2010-02-08 15:30:06 +02:00
import org.elasticsearch.ExceptionsHelper;
import org.elasticsearch.Version;
import org.elasticsearch.common.PidFile;
import org.elasticsearch.common.SuppressForbidden;
import org.elasticsearch.common.collect.Tuple;
import org.elasticsearch.common.inject.CreationException;
import org.elasticsearch.common.inject.spi.Message;
2015-04-15 18:23:30 -04:00
import org.elasticsearch.common.io.PathUtils;
import org.elasticsearch.common.jna.Kernel32Library;
import org.elasticsearch.common.jna.Natives;
2015-05-04 14:55:10 -04:00
import org.elasticsearch.common.lease.Releasables;
import org.elasticsearch.common.logging.ESLogger;
import org.elasticsearch.common.logging.Loggers;
import org.elasticsearch.common.logging.log4j.LogConfigurator;
import org.elasticsearch.common.settings.Settings;
2010-02-08 15:30:06 +02:00
import org.elasticsearch.env.Environment;
import org.elasticsearch.monitor.jvm.JvmInfo;
import org.elasticsearch.monitor.process.JmxProcessProbe;
import org.elasticsearch.node.Node;
import org.elasticsearch.node.NodeBuilder;
2013-09-11 13:49:33 -05:00
import org.elasticsearch.node.internal.InternalSettingsPreparer;
import org.hyperic.sigar.Sigar;
2010-02-08 15:30:06 +02:00
import java.util.Locale;
2010-02-08 15:30:06 +02:00
import java.util.Set;
import java.util.concurrent.CountDownLatch;
2010-02-08 15:30:06 +02:00
2011-12-06 02:42:25 +02:00
import static com.google.common.collect.Sets.newHashSet;
import static org.elasticsearch.common.jna.Kernel32Library.ConsoleCtrlHandler;
2011-12-06 02:42:25 +02:00
import static org.elasticsearch.common.settings.ImmutableSettings.Builder.EMPTY_SETTINGS;
2010-02-08 15:30:06 +02:00
/**
* A main entry point when starting from the command line.
2010-02-08 15:30:06 +02:00
*/
public class Bootstrap {
2015-05-04 14:06:32 -04:00
private static volatile Bootstrap INSTANCE;
2010-02-08 15:30:06 +02:00
2015-05-04 14:06:32 -04:00
private Node node;
private final CountDownLatch keepAliveLatch = new CountDownLatch(1);
private final Thread keepAliveThread;
/** creates a new instance */
Bootstrap() {
keepAliveThread = new Thread(new Runnable() {
@Override
public void run() {
try {
keepAliveLatch.await();
} catch (InterruptedException e) {
// bail out
}
}
}, "elasticsearch[keepAlive/" + Version.CURRENT + "]");
keepAliveThread.setDaemon(false);
// keep this thread alive (non daemon thread) until we shutdown
Runtime.getRuntime().addShutdownHook(new Thread() {
@Override
public void run() {
keepAliveLatch.countDown();
}
});
}
/** initialize native resources */
public static void initializeNatives(boolean mlockAll, boolean ctrlHandler) {
// mlockall if requested
if (mlockAll) {
Natives.tryMlockall();
}
2010-02-08 15:30:06 +02:00
// listener for windows close event
if (ctrlHandler) {
Natives.addConsoleCtrlHandler(new ConsoleCtrlHandler() {
@Override
public boolean handle(int code) {
if (CTRL_CLOSE_EVENT == code) {
ESLogger logger = Loggers.getLogger(Bootstrap.class);
logger.info("running graceful exit on windows");
2015-05-04 14:06:32 -04:00
Bootstrap.INSTANCE.stop();
return true;
}
return false;
}
});
}
// force remainder of JNA to be loaded (if available).
try {
Kernel32Library.getInstance();
} catch (Throwable ignored) {
// we've already logged this.
}
// initialize sigar explicitly
try {
Sigar.load();
Loggers.getLogger(Bootstrap.class).trace("sigar libraries loaded successfully");
} catch (Throwable t) {
Loggers.getLogger(Bootstrap.class).trace("failed to load sigar libraries", t);
}
// init lucene random seed. it will use /dev/urandom where available:
StringHelper.randomId();
}
private void setup(boolean addShutdownHook, Settings settings, Environment environment) throws Exception {
initializeNatives(settings.getAsBoolean("bootstrap.mlockall", false),
settings.getAsBoolean("bootstrap.ctrlhandler", true));
NodeBuilder nodeBuilder = NodeBuilder.nodeBuilder().settings(settings).loadConfigSettings(false);
node = nodeBuilder.build();
if (addShutdownHook) {
Runtime.getRuntime().addShutdownHook(new Thread() {
@Override
public void run() {
node.close();
}
});
}
// install SM after natives, shutdown hooks, etc.
setupSecurity(settings, environment);
}
2015-04-22 03:04:50 -04:00
/**
* option for elasticsearch.yml etc to turn off our security manager completely,
* for example if you want to have your own configuration or just disable.
*/
static final String SECURITY_SETTING = "security.manager.enabled";
2015-04-22 03:04:50 -04:00
private void setupSecurity(Settings settings, Environment environment) throws Exception {
if (settings.getAsBoolean(SECURITY_SETTING, true)) {
2015-04-22 03:04:50 -04:00
Security.configure(environment);
}
}
@SuppressForbidden(reason = "Exception#printStackTrace()")
2015-04-24 10:09:30 -04:00
private static void setupLogging(Settings settings, Environment environment) {
2010-02-08 15:30:06 +02:00
try {
2015-04-24 10:09:30 -04:00
settings.getClassLoader().loadClass("org.apache.log4j.Logger");
LogConfigurator.configure(settings);
2010-02-08 15:30:06 +02:00
} catch (ClassNotFoundException e) {
// no log4j
} catch (NoClassDefFoundError e) {
// no log4j
} catch (Exception e) {
sysError("Failed to configure logging...", false);
2010-02-08 15:30:06 +02:00
e.printStackTrace();
}
}
2010-02-08 15:30:06 +02:00
private static Tuple<Settings, Environment> initialSettings() {
2013-09-11 13:49:33 -05:00
return InternalSettingsPreparer.prepareSettings(EMPTY_SETTINGS, true);
2010-02-08 15:30:06 +02:00
}
2015-05-04 14:06:32 -04:00
private void start() {
node.start();
2015-05-04 14:06:32 -04:00
keepAliveThread.start();
2010-02-08 15:30:06 +02:00
}
2015-05-04 14:06:32 -04:00
private void stop() {
try {
2015-05-04 14:55:10 -04:00
Releasables.close(node);
2015-05-04 14:06:32 -04:00
} finally {
keepAliveLatch.countDown();
}
}
2010-02-08 15:30:06 +02:00
public static void main(String[] args) {
System.setProperty("es.logger.prefix", "");
2015-05-04 14:06:32 -04:00
INSTANCE = new Bootstrap();
final String pidFile = System.getProperty("es.pidfile", System.getProperty("es-pidfile"));
2010-02-08 15:30:06 +02:00
if (pidFile != null) {
try {
2015-04-15 18:23:30 -04:00
PidFile.create(PathUtils.get(pidFile), true);
} catch (Exception e) {
String errorMessage = buildErrorMessage("pid", e);
sysError(errorMessage, true);
System.exit(3);
}
}
boolean foreground = System.getProperty("es.foreground", System.getProperty("es-foreground")) != null;
// handle the wrapper system property, if its a service, don't run as a service
if (System.getProperty("wrapper.service", "XXX").equalsIgnoreCase("true")) {
foreground = false;
}
2010-02-08 15:30:06 +02:00
2015-04-24 10:09:30 -04:00
Settings settings = null;
Environment environment = null;
try {
2015-04-24 10:09:30 -04:00
Tuple<Settings, Environment> tuple = initialSettings();
settings = tuple.v1();
environment = tuple.v2();
setupLogging(settings, environment);
} catch (Exception e) {
String errorMessage = buildErrorMessage("Setup", e);
sysError(errorMessage, true);
System.exit(3);
}
if (System.getProperty("es.max-open-files", "false").equals("true")) {
ESLogger logger = Loggers.getLogger(Bootstrap.class);
logger.info("max_open_files [{}]", JmxProcessProbe.getMaxFileDescriptorCount());
}
// warn if running using the client VM
if (JvmInfo.jvmInfo().getVmName().toLowerCase(Locale.ROOT).contains("client")) {
ESLogger logger = Loggers.getLogger(Bootstrap.class);
logger.warn("jvm uses the client vm, make sure to run `java` with the server vm for best performance by adding `-server` to the command line");
}
2010-02-08 15:30:06 +02:00
String stage = "Initialization";
try {
if (!foreground) {
Loggers.disableConsoleLogging();
closeSystOut();
2010-02-08 15:30:06 +02:00
}
// fail if using broken version
JVMCheck.check();
2015-05-04 14:06:32 -04:00
INSTANCE.setup(true, settings, environment);
2015-05-04 10:18:09 -04:00
stage = "Startup";
2015-05-04 14:06:32 -04:00
INSTANCE.start();
2015-05-04 10:18:09 -04:00
if (!foreground) {
closeSysError();
}
2010-02-08 15:30:06 +02:00
} catch (Throwable e) {
ESLogger logger = Loggers.getLogger(Bootstrap.class);
2015-05-04 14:06:32 -04:00
if (INSTANCE.node != null) {
logger = Loggers.getLogger(Bootstrap.class, INSTANCE.node.settings().get("name"));
2010-02-08 15:30:06 +02:00
}
String errorMessage = buildErrorMessage(stage, e);
2010-02-08 15:30:06 +02:00
if (foreground) {
sysError(errorMessage, true);
Loggers.disableConsoleLogging();
2010-02-08 15:30:06 +02:00
}
logger.error("Exception", e);
2010-02-08 15:30:06 +02:00
System.exit(3);
}
}
@SuppressForbidden(reason = "System#out")
private static void closeSystOut() {
System.out.close();
}
@SuppressForbidden(reason = "System#err")
private static void closeSysError() {
System.err.close();
}
@SuppressForbidden(reason = "System#err")
private static void sysError(String line, boolean flush) {
System.err.println(line);
if (flush) {
System.err.flush();
}
}
private static String buildErrorMessage(String stage, Throwable e) {
StringBuilder errorMessage = new StringBuilder("{").append(Version.CURRENT).append("}: ");
errorMessage.append(stage).append(" Failed ...\n");
if (e instanceof CreationException) {
CreationException createException = (CreationException) e;
Set<String> seenMessages = newHashSet();
int counter = 1;
for (Message message : createException.getErrorMessages()) {
String detailedMessage;
if (message.getCause() == null) {
detailedMessage = message.getMessage();
} else {
detailedMessage = ExceptionsHelper.detailedMessage(message.getCause(), true, 0);
}
if (detailedMessage == null) {
detailedMessage = message.getMessage();
}
if (seenMessages.contains(detailedMessage)) {
continue;
}
seenMessages.add(detailedMessage);
errorMessage.append("").append(counter++).append(") ").append(detailedMessage);
}
} else {
errorMessage.append("- ").append(ExceptionsHelper.detailedMessage(e, true, 0));
}
if (Loggers.getLogger(Bootstrap.class).isDebugEnabled()) {
errorMessage.append("\n").append(ExceptionsHelper.stackTrace(e));
}
return errorMessage.toString();
}
2010-02-08 15:30:06 +02:00
}