/* * Copyright 2010-2013 the original author or authors. * * Licensed 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.springframework.data.gemfire.fork; import java.io.File; import java.io.IOException; import java.util.Properties; import java.util.concurrent.TimeUnit; import org.apache.geode.distributed.Locator; import org.apache.geode.distributed.LocatorLauncher; import org.apache.geode.distributed.internal.DistributionConfig; import org.apache.geode.distributed.internal.InternalLocator; import org.apache.geode.distributed.internal.SharedConfiguration; import org.springframework.data.gemfire.GemfireUtils; import org.springframework.data.gemfire.process.support.ProcessUtils; import org.springframework.data.gemfire.test.support.FileSystemUtils; import org.springframework.data.gemfire.test.support.ThreadUtils; /** * The LocatorProcess class is a main Java class that is used fork and launch a GemFire Locator process using the * LocatorLauncher class. * * @author John Blum * @see org.apache.geode.distributed.LocatorLauncher * @since 1.5.0 */ public class LocatorProcess { private static final int DEFAULT_LOCATOR_PORT = 20668; private static final String GEMFIRE_NAME = "SpringDataGemFireLocator"; private static final String GEMFIRE_LOG_LEVEL = "warning"; private static final String HOSTNAME_FOR_CLIENTS = "localhost"; private static final String HTTP_SERVICE_PORT = "0"; public static void main(final String... args) throws IOException { //runLocator(); runInternalLocator(); registerShutdownHook(); waitForLocatorStart(TimeUnit.SECONDS.toMillis(30)); ProcessUtils.writePid(new File(FileSystemUtils.WORKING_DIRECTORY, getLocatorProcessControlFilename()), ProcessUtils.currentPid()); ProcessUtils.waitForStopSignal(); } public static String getLocatorProcessControlFilename() { return LocatorProcess.class.getSimpleName().toLowerCase().concat(".pid"); } @SuppressWarnings("unused") private static InternalLocator runInternalLocator() throws IOException { String hostnameForClients = System.getProperty("spring.gemfire.hostname-for-clients", HOSTNAME_FOR_CLIENTS); int locatorPort = Integer.getInteger("spring.data.gemfire.locator.port", DEFAULT_LOCATOR_PORT); boolean loadClusterConfigurationFromDirectory = Boolean.getBoolean("spring.data.gemfire.load-cluster-configuration"); Properties distributedSystemProperties = new Properties(); distributedSystemProperties.setProperty(DistributionConfig.ENABLE_CLUSTER_CONFIGURATION_NAME, String.valueOf(Boolean.getBoolean("spring.data.gemfire.enable-cluster-configuration"))); distributedSystemProperties.setProperty(DistributionConfig.HTTP_SERVICE_PORT_NAME, System.getProperty("spring.data.gemfire.http-service-port", HTTP_SERVICE_PORT)); distributedSystemProperties.setProperty(DistributionConfig.JMX_MANAGER_NAME, System.getProperty("spring.data.gemfire.jmx-manager", Boolean.TRUE.toString())); distributedSystemProperties.setProperty(DistributionConfig.JMX_MANAGER_START_NAME, System.getProperty("spring.data.gemfire.jmx-manager-start", Boolean.FALSE.toString())); distributedSystemProperties.setProperty(DistributionConfig.LOAD_CLUSTER_CONFIG_FROM_DIR_NAME, String.valueOf(loadClusterConfigurationFromDirectory)); distributedSystemProperties.setProperty(DistributionConfig.LOG_LEVEL_NAME, System.getProperty("spring.data.gemfire.log-level", GEMFIRE_LOG_LEVEL)); return InternalLocator.startLocator(locatorPort, null, null, null, null, null, distributedSystemProperties, true, true, hostnameForClients, loadClusterConfigurationFromDirectory); } @SuppressWarnings("unused") private static LocatorLauncher runLocator() { LocatorLauncher locatorLauncher = buildLocatorLauncher(); // start the GemFire Locator process... locatorLauncher.start(); return locatorLauncher; } private static LocatorLauncher buildLocatorLauncher() { return new LocatorLauncher.Builder() .setMemberName(GEMFIRE_NAME) .setHostnameForClients(getProperty("spring.data.gemfire.hostname-for-clients", HOSTNAME_FOR_CLIENTS)) .setPort(getInteger("spring.data.gemfire.locator.port", DEFAULT_LOCATOR_PORT)) .setRedirectOutput(false) .set(DistributionConfig.ENABLE_CLUSTER_CONFIGURATION_NAME, String.valueOf(getBoolean("spring.data.gemfire.enable-cluster-configuration"))) .set(DistributionConfig.HTTP_SERVICE_PORT_NAME, getProperty("spring.data.gemfire.http-service-port", HTTP_SERVICE_PORT)) .set(DistributionConfig.JMX_MANAGER_NAME, Boolean.TRUE.toString()) .set(DistributionConfig.JMX_MANAGER_START_NAME, Boolean.FALSE.toString()) .set(DistributionConfig.LOAD_CLUSTER_CONFIG_FROM_DIR_NAME, String.valueOf(getBoolean("spring.data.gemfire.load-cluster-configuration"))) .set(DistributionConfig.LOG_LEVEL_NAME, getProperty("spring.data.gemfire.log-level", GEMFIRE_LOG_LEVEL)) .build(); } private static boolean getBoolean(String name) { return Boolean.getBoolean(name); } private static int getInteger(String name, int defaultValue) { return Integer.getInteger(name, defaultValue); } private static String getProperty(String name, String defaultValue) { return System.getProperty(name, defaultValue); } private static void registerShutdownHook() { Runtime.getRuntime().addShutdownHook(new Thread(() -> { Locator locator = GemfireUtils.getLocator(); if (locator != null) { stopClusterConfigurationService(locator).stop(); } })); } private static boolean isClusterConfigurationEnabled(Locator locator) { return (locator instanceof InternalLocator && Boolean.valueOf( locator.getDistributedSystem().getProperties().getProperty( DistributionConfig.ENABLE_CLUSTER_CONFIGURATION_NAME))); } private static Locator stopClusterConfigurationService(Locator locator) { if (isClusterConfigurationEnabled(locator)) { SharedConfiguration sharedConfiguration = ((InternalLocator) locator).getSharedConfiguration(); if (sharedConfiguration != null) { if (Boolean.valueOf(getProperty("spring.data.gemfire.fork.clean", Boolean.TRUE.toString()))) { sharedConfiguration.destroySharedConfiguration(); } } } return locator; } private static void waitForLocatorStart(long milliseconds) { InternalLocator locator = InternalLocator.getLocator(); if (isClusterConfigurationEnabled(locator)) { ThreadUtils.timedWait(milliseconds, 500L, () -> !locator.isSharedConfigurationRunning()); } else { LocatorLauncher.getInstance().waitOnStatusResponse( milliseconds, Math.min(500L, milliseconds), TimeUnit.MILLISECONDS); } } }