2010-02-08 15:30:06 +02:00
|
|
|
/*
|
2014-01-06 22:48:02 +01: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;
|
|
|
|
|
2015-05-04 12:30:03 -04:00
|
|
|
import org.apache.lucene.util.StringHelper;
|
2010-02-08 15:30:06 +02:00
|
|
|
import org.elasticsearch.ExceptionsHelper;
|
|
|
|
import org.elasticsearch.Version;
|
2014-12-04 10:20:05 +01:00
|
|
|
import org.elasticsearch.common.PidFile;
|
2015-04-13 10:03:50 +02:00
|
|
|
import org.elasticsearch.common.SuppressForbidden;
|
2010-06-15 17:28:05 +03:00
|
|
|
import org.elasticsearch.common.collect.Tuple;
|
2010-06-15 16:51:38 +03:00
|
|
|
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;
|
2015-05-04 12:30:03 -04:00
|
|
|
import org.elasticsearch.common.jna.Kernel32Library;
|
2010-10-30 19:40:24 +02:00
|
|
|
import org.elasticsearch.common.jna.Natives;
|
2015-05-04 14:55:10 -04:00
|
|
|
import org.elasticsearch.common.lease.Releasables;
|
2010-06-15 16:51:38 +03:00
|
|
|
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;
|
2011-03-23 18:06:29 +02:00
|
|
|
import org.elasticsearch.monitor.jvm.JvmInfo;
|
2014-04-14 21:39:13 +02:00
|
|
|
import org.elasticsearch.monitor.process.JmxProcessProbe;
|
2010-04-09 00:54:54 +03:00
|
|
|
import org.elasticsearch.node.Node;
|
|
|
|
import org.elasticsearch.node.NodeBuilder;
|
2013-09-11 13:49:33 -05:00
|
|
|
import org.elasticsearch.node.internal.InternalSettingsPreparer;
|
2015-05-04 12:30:03 -04:00
|
|
|
import org.hyperic.sigar.Sigar;
|
2010-02-08 15:30:06 +02:00
|
|
|
|
2013-05-18 23:49:04 +02:00
|
|
|
import java.util.Locale;
|
2010-02-08 15:30:06 +02:00
|
|
|
import java.util.Set;
|
2010-05-01 03:00:06 +03:00
|
|
|
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;
|
2014-12-17 18:00:53 +01:00
|
|
|
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
|
|
|
|
|
|
|
/**
|
2010-04-04 17:18:18 +03: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();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
2015-05-04 12:30:03 -04:00
|
|
|
|
|
|
|
/** initialize native resources */
|
|
|
|
public static void initializeNatives(boolean mlockAll, boolean ctrlHandler) {
|
|
|
|
// mlockall if requested
|
|
|
|
if (mlockAll) {
|
2010-11-14 15:54:40 +02:00
|
|
|
Natives.tryMlockall();
|
|
|
|
}
|
2010-02-08 15:30:06 +02:00
|
|
|
|
2015-05-04 12:30:03 -04:00
|
|
|
// listener for windows close event
|
|
|
|
if (ctrlHandler) {
|
2014-12-17 18:00:53 +01:00
|
|
|
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();
|
2014-12-17 18:00:53 +01:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
2015-05-05 11:49:40 -04:00
|
|
|
|
|
|
|
// force remainder of JNA to be loaded (if available).
|
|
|
|
try {
|
|
|
|
Kernel32Library.getInstance();
|
|
|
|
} catch (Throwable ignored) {
|
|
|
|
// we've already logged this.
|
|
|
|
}
|
2015-05-04 12:30:03 -04:00
|
|
|
|
|
|
|
// 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.
|
2015-04-27 20:29:57 -04:00
|
|
|
setupSecurity(settings, environment);
|
2010-03-20 00:28:15 +02:00
|
|
|
}
|
2015-04-22 03:04:50 -04:00
|
|
|
|
2015-04-24 09:51:01 -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 {
|
2015-04-24 09:51:01 -04:00
|
|
|
if (settings.getAsBoolean(SECURITY_SETTING, true)) {
|
2015-04-22 03:04:50 -04:00
|
|
|
Security.configure(environment);
|
|
|
|
}
|
|
|
|
}
|
2010-03-20 00:28:15 +02:00
|
|
|
|
2015-04-13 10:03:50 +02:00
|
|
|
@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) {
|
2015-04-13 10:03:50 +02:00
|
|
|
sysError("Failed to configure logging...", false);
|
2010-02-08 15:30:06 +02:00
|
|
|
e.printStackTrace();
|
|
|
|
}
|
2010-03-20 00:28:15 +02:00
|
|
|
}
|
2010-02-08 15:30:06 +02:00
|
|
|
|
2010-03-20 00:28:15 +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() {
|
2010-04-09 00:54:54 +03:00
|
|
|
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();
|
|
|
|
}
|
2011-10-17 20:25:41 +02:00
|
|
|
}
|
2010-02-08 15:30:06 +02:00
|
|
|
|
|
|
|
public static void main(String[] args) {
|
2010-10-28 23:39:37 +02:00
|
|
|
System.setProperty("es.logger.prefix", "");
|
2015-05-04 14:06:32 -04:00
|
|
|
INSTANCE = new Bootstrap();
|
2011-12-21 05:28:28 +02:00
|
|
|
final String pidFile = System.getProperty("es.pidfile", System.getProperty("es-pidfile"));
|
2010-02-08 15:30:06 +02:00
|
|
|
|
2011-12-21 05:28:28 +02:00
|
|
|
if (pidFile != null) {
|
|
|
|
try {
|
2015-04-15 18:23:30 -04:00
|
|
|
PidFile.create(PathUtils.get(pidFile), true);
|
2011-12-21 05:28:28 +02:00
|
|
|
} catch (Exception e) {
|
|
|
|
String errorMessage = buildErrorMessage("pid", e);
|
2015-04-13 10:03:50 +02:00
|
|
|
sysError(errorMessage, true);
|
2011-12-21 05:28:28 +02:00
|
|
|
System.exit(3);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
boolean foreground = System.getProperty("es.foreground", System.getProperty("es-foreground")) != null;
|
2010-05-01 01:42:30 +03:00
|
|
|
// 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;
|
2010-03-20 00:28:15 +02:00
|
|
|
try {
|
2015-04-24 10:09:30 -04:00
|
|
|
Tuple<Settings, Environment> tuple = initialSettings();
|
|
|
|
settings = tuple.v1();
|
|
|
|
environment = tuple.v2();
|
|
|
|
setupLogging(settings, environment);
|
2010-03-20 00:28:15 +02:00
|
|
|
} catch (Exception e) {
|
|
|
|
String errorMessage = buildErrorMessage("Setup", e);
|
2015-04-13 10:03:50 +02:00
|
|
|
sysError(errorMessage, true);
|
2010-03-20 00:28:15 +02:00
|
|
|
System.exit(3);
|
|
|
|
}
|
|
|
|
|
2010-11-06 22:47:09 +02:00
|
|
|
if (System.getProperty("es.max-open-files", "false").equals("true")) {
|
|
|
|
ESLogger logger = Loggers.getLogger(Bootstrap.class);
|
2014-04-14 21:39:13 +02:00
|
|
|
logger.info("max_open_files [{}]", JmxProcessProbe.getMaxFileDescriptorCount());
|
2010-11-06 22:47:09 +02:00
|
|
|
}
|
|
|
|
|
2011-03-23 18:06:29 +02:00
|
|
|
// warn if running using the client VM
|
2015-04-12 16:55:01 +02:00
|
|
|
if (JvmInfo.jvmInfo().getVmName().toLowerCase(Locale.ROOT).contains("client")) {
|
2011-03-23 18:06:29 +02:00
|
|
|
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();
|
2015-04-13 10:03:50 +02:00
|
|
|
closeSystOut();
|
2010-02-08 15:30:06 +02:00
|
|
|
}
|
2015-03-21 02:47:44 -04: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) {
|
2010-04-21 00:29:42 +03:00
|
|
|
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
|
|
|
}
|
2010-03-20 00:28:15 +02:00
|
|
|
String errorMessage = buildErrorMessage(stage, e);
|
2010-02-08 15:30:06 +02:00
|
|
|
if (foreground) {
|
2015-04-13 10:03:50 +02:00
|
|
|
sysError(errorMessage, true);
|
2014-06-20 11:01:01 -04:00
|
|
|
Loggers.disableConsoleLogging();
|
2010-02-08 15:30:06 +02:00
|
|
|
}
|
2014-06-20 11:01:01 -04:00
|
|
|
logger.error("Exception", e);
|
|
|
|
|
2010-02-08 15:30:06 +02:00
|
|
|
System.exit(3);
|
|
|
|
}
|
|
|
|
}
|
2010-03-20 00:28:15 +02:00
|
|
|
|
2015-04-13 10:03:50 +02:00
|
|
|
@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();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-03-20 00:28:15 +02:00
|
|
|
private static String buildErrorMessage(String stage, Throwable e) {
|
2011-10-05 12:42:27 +02:00
|
|
|
StringBuilder errorMessage = new StringBuilder("{").append(Version.CURRENT).append("}: ");
|
2011-12-16 15:36:22 +02:00
|
|
|
errorMessage.append(stage).append(" Failed ...\n");
|
2010-03-20 00:28:15 +02:00
|
|
|
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));
|
|
|
|
}
|
2014-02-12 21:53:23 +02:00
|
|
|
if (Loggers.getLogger(Bootstrap.class).isDebugEnabled()) {
|
|
|
|
errorMessage.append("\n").append(ExceptionsHelper.stackTrace(e));
|
|
|
|
}
|
2010-03-20 00:28:15 +02:00
|
|
|
return errorMessage.toString();
|
|
|
|
}
|
2010-02-08 15:30:06 +02:00
|
|
|
}
|