com.oracle.bedrock.runtime.java.LocalJavaApplicationLauncher Maven / Gradle / Ivy
Show all versions of bedrock-runtime Show documentation
/*
* File: LocalJavaApplicationLauncher.java
*
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER.
*
* The contents of this file are subject to the terms and conditions of
* the Common Development and Distribution License 1.0 (the "License").
*
* You may not use this file except in compliance with the License.
*
* You can obtain a copy of the License by consulting the LICENSE.txt file
* distributed with this file, or by consulting https://oss.oracle.com/licenses/CDDL
*
* See the License for the specific language governing permissions
* and limitations under the License.
*
* When distributing the software, include this License Header Notice in each
* file and include the License file LICENSE.txt.
*
* MODIFICATIONS:
* If applicable, add the following below the License Header, with the fields
* enclosed by brackets [] replaced by your own identifying information:
* "Portions Copyright [year] [name of copyright owner]"
*/
package com.oracle.bedrock.runtime.java;
import com.oracle.bedrock.Option;
import com.oracle.bedrock.OptionsByType;
import com.oracle.bedrock.Bedrock;
import com.oracle.bedrock.annotations.Internal;
import com.oracle.bedrock.deferred.AbstractDeferred;
import com.oracle.bedrock.deferred.PermanentlyUnavailableException;
import com.oracle.bedrock.deferred.TemporarilyUnavailableException;
import com.oracle.bedrock.lang.ExpressionEvaluator;
import com.oracle.bedrock.lang.StringHelper;
import com.oracle.bedrock.options.LaunchLogging;
import com.oracle.bedrock.options.Timeout;
import com.oracle.bedrock.options.Variable;
import com.oracle.bedrock.options.Variables;
import com.oracle.bedrock.runtime.ApplicationListener;
import com.oracle.bedrock.runtime.LocalApplicationProcess;
import com.oracle.bedrock.runtime.MetaClass;
import com.oracle.bedrock.runtime.Platform;
import com.oracle.bedrock.runtime.Profile;
import com.oracle.bedrock.runtime.Profiles;
import com.oracle.bedrock.runtime.Settings;
import com.oracle.bedrock.runtime.concurrent.ControllableRemoteChannel;
import com.oracle.bedrock.runtime.concurrent.RemoteCallable;
import com.oracle.bedrock.runtime.concurrent.RemoteChannel;
import com.oracle.bedrock.runtime.concurrent.RemoteEvent;
import com.oracle.bedrock.runtime.concurrent.RemoteEventListener;
import com.oracle.bedrock.runtime.concurrent.RemoteRunnable;
import com.oracle.bedrock.runtime.concurrent.socket.SocketBasedRemoteChannelServer;
import com.oracle.bedrock.runtime.java.features.JmxFeature;
import com.oracle.bedrock.runtime.java.options.BedrockRunner;
import com.oracle.bedrock.runtime.java.options.ClassName;
import com.oracle.bedrock.runtime.java.options.IPv4Preferred;
import com.oracle.bedrock.runtime.java.options.JavaHome;
import com.oracle.bedrock.runtime.java.options.JvmOption;
import com.oracle.bedrock.runtime.java.options.RemoteEvents;
import com.oracle.bedrock.runtime.java.options.SystemProperties;
import com.oracle.bedrock.runtime.java.options.JavaModules;
import com.oracle.bedrock.runtime.java.options.WaitToStart;
import com.oracle.bedrock.runtime.java.profiles.CommercialFeatures;
import com.oracle.bedrock.runtime.java.profiles.RemoteDebugging;
import com.oracle.bedrock.runtime.options.Arguments;
import com.oracle.bedrock.runtime.options.DisplayName;
import com.oracle.bedrock.runtime.options.EnvironmentVariables;
import com.oracle.bedrock.runtime.options.ErrorStreamRedirection;
import com.oracle.bedrock.runtime.options.Executable;
import com.oracle.bedrock.runtime.options.Orphanable;
import com.oracle.bedrock.runtime.options.WorkingDirectory;
import com.oracle.bedrock.table.Cell;
import com.oracle.bedrock.table.Table;
import com.oracle.bedrock.table.Tabularize;
import com.oracle.bedrock.util.ReflectionHelper;
import java.io.File;
import java.io.IOException;
import java.lang.reflect.Constructor;
import java.net.InetAddress;
import java.net.UnknownHostException;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.List;
import java.util.Map;
import java.util.Properties;
import java.util.UUID;
import java.util.concurrent.CompletableFuture;
import java.util.logging.Level;
import java.util.logging.Logger;
import java.util.stream.Collectors;
import static com.oracle.bedrock.deferred.DeferredHelper.ensure;
import static com.oracle.bedrock.deferred.DeferredHelper.within;
/**
* A {@link JavaApplicationLauncher} that launches a {@link JavaApplication}s as
* external, non-child local operating system processes, by default using the
* environment variables of the current system process.
*
* Copyright (c) 2014. All Rights Reserved. Oracle Corporation.
* Oracle is a registered trademark of Oracle Corporation and/or its affiliates.
*
* @author Brian Oliver
*/
@Internal
public class LocalJavaApplicationLauncher implements JavaApplicationLauncher
{
/**
* The {@link Logger} for this class.
*/
private static Logger LOGGER = Logger.getLogger(LocalJavaApplicationLauncher.class.getName());
/**
* Constructs a {@link LocalJavaApplicationLauncher}.
*
*/
public LocalJavaApplicationLauncher()
{
}
@Override
public A launch(Platform platform,
MetaClass metaClass,
OptionsByType optionsByType)
{
// establish the diagnostics output table
Table diagnosticsTable = new Table();
diagnosticsTable.getOptions().add(Table.orderByColumn(0));
if (platform != null)
{
diagnosticsTable.addRow("Target Platform", platform.getName());
}
// ----- establish the launch Options for the Application -----
// add the platform options
OptionsByType launchOptions = OptionsByType.of(platform.getOptions().asArray());
// add the meta-class options
metaClass.onLaunching(platform, launchOptions);
// add the launch specific options
launchOptions.addAll(optionsByType);
// ----- establish an identity for the application -----
// add a unique runtime id for expression support
launchOptions.add(Variable.with("bedrock.runtime.id", UUID.randomUUID()));
// ----- establish default Profiles for this Platform (and Builder) -----
// java applications can automatically detect the following profiles
launchOptions.get(RemoteDebugging.class);
launchOptions.get(CommercialFeatures.class);
// auto-detect and add externally defined profiles
launchOptions.addAll(Profiles.getProfiles());
// ----- notify the Profiles that the application is about to be launched -----
for (Profile profile : launchOptions.getInstancesOf(Profile.class))
{
profile.onLaunching(platform, metaClass, launchOptions);
}
// ----- give the MetaClass a last chance to manipulate any options -----
metaClass.onLaunch(platform, launchOptions);
// ----- determine the display name for the application -----
DisplayName displayName = getDisplayName(launchOptions);
// ----- establish the underlying ProcessBuilder -----
// determine the Executable, defaulting to "java" if not defined
Executable executable = launchOptions.getOrSetDefault(Executable.class, Executable.named("java"));
// we'll use the native operating system process builder to create
// and manage the local application process
ProcessBuilder processBuilder = new ProcessBuilder(executable.getName());
// ----- establish the working directory -----
// set the working directory for the Process
WorkingDirectory workingDirectory = launchOptions.getOrSetDefault(WorkingDirectory.class,
WorkingDirectory.currentDirectory());
File directory = workingDirectory.resolve(platform, launchOptions);
// set the resolved working directory back into the options
launchOptions.add(WorkingDirectory.at(directory));
if (directory != null)
{
processBuilder.directory(directory);
diagnosticsTable.addRow("Working Directory", directory.toString());
}
// ----- establish environment variables -----
EnvironmentVariables environmentVariables = launchOptions.get(EnvironmentVariables.class);
switch (environmentVariables.getSource())
{
case Custom :
processBuilder.environment().clear();
diagnosticsTable.addRow("Environment Variables", "(cleared)");
break;
case ThisApplication :
Map map = System.getenv();
processBuilder.environment().clear();
processBuilder.environment().putAll(map);
diagnosticsTable.addRow("Environment Variables", "(based on parent process)");
break;
case TargetPlatform :
diagnosticsTable.addRow("Environment Variables", "(based on platform defaults)");
break;
}
// add the optionally defined environment variables
Properties variables = environmentVariables.realize(platform, launchOptions.asArray());
for (String variableName : variables.stringPropertyNames())
{
processBuilder.environment().put(variableName, variables.getProperty(variableName));
}
if (variables.size() > 0)
{
Table table = Tabularize.tabularize(variables);
diagnosticsTable.addRow("", table.toString());
}
// ----- establish java specific environment variables -----
// by default we use the java home defined by the schema. if that's not
// defined we'll attempt to use the java home defined by this builder.
JavaHome javaHome = launchOptions.get(JavaHome.class);
// when we still don't have a java home we use what this process defines
// (using the system property)
if (javaHome == null)
{
javaHome = JavaHome.at(System.getProperty("java.home", null));
}
if (javaHome != null)
{
processBuilder.environment().put("JAVA_HOME", StringHelper.doubleQuoteIfNecessary(javaHome.get()));
}
// ----- establish the command to start java -----
String javaExecutable;
if (javaHome == null)
{
// when we don't have a java home we just use the defined executable
javaExecutable = StringHelper.doubleQuoteIfNecessary(executable.getName());
}
else
{
// when we have a java home, we prefix the executable name with the java.home/bin/
String javaHomePath = javaHome.get();
javaHomePath = javaHomePath.trim();
diagnosticsTable.addRow("Java Home", javaHomePath);
if (!javaHomePath.endsWith(File.separator))
{
javaHomePath = javaHomePath + File.separator;
}
javaExecutable = StringHelper.doubleQuoteIfNecessary(javaHomePath + "bin" + File.separator
+ executable.getName());
}
processBuilder.command(javaExecutable);
diagnosticsTable.addRow("Java Executable", javaExecutable);
// ----- establish the class path -----
JavaModules modular = launchOptions.get(JavaModules.class);
boolean useModules = modular.isEnabled();
// determine the predefined class path based on the launch options
ClassPath classPath = launchOptions.get(ClassPath.class);
try
{
// include the ClassPath of the Platform
classPath = new ClassPath(classPath, ClassPath.ofClass(platform.getClass()));
// include the ClassPath of each of the Options
for (Option option : launchOptions.getInstancesOf(Option.class))
{
classPath = new ClassPath(classPath, ClassPath.ofClass(option.getClass()));
}
// include the application runner (if defined)
BedrockRunner bedrockRunner = optionsByType.get(BedrockRunner.class);
if (bedrockRunner != null && bedrockRunner.isEnabled())
{
// include the JavaApplicationLauncher
classPath = new ClassPath(classPath, ClassPath.ofClass(bedrockRunner.getClassOfRunner()));
}
// add the updated ClassPath back into the launch options
launchOptions.add(classPath);
}
catch (IOException e)
{
throw new RuntimeException("Failed to locate required classes for the class path", e);
}
processBuilder.command().add(useModules ? "--module-path" : "-cp");
processBuilder.command().add(classPath.toString(launchOptions.asArray()));
if (useModules)
{
Table modulePathTable = classPath.getTable();
modulePathTable.getOptions().add(Cell.Separator.of(""));
diagnosticsTable.addRow("Module Path", modulePathTable.toString());
ClassPath path = modular.getClassPath();
if (path != null && !path.isEmpty())
{
processBuilder.command().add("-cp");
processBuilder.command().add(path.toString(launchOptions.asArray()));
Table classPathTable = path.getTable();
classPathTable.getOptions().add(Cell.Separator.of(""));
diagnosticsTable.addRow("Class Path", classPathTable.toString());
}
}
else
{
Table classPathTable = classPath.getTable();
classPathTable.getOptions().add(Cell.Separator.of(""));
diagnosticsTable.addRow("Class Path", classPathTable.toString());
}
// ----- establish Bedrock specific system properties -----
// configure a server channel to communicate with the native process
final SocketBasedRemoteChannelServer server = new SocketBasedRemoteChannelServer();
// register the defined RemoteEventListeners with the server so that when the application starts
// the listeners can immediately start receiving RemoteEvents
RemoteEvents remoteEvents = launchOptions.get(RemoteEvents.class);
remoteEvents.forEach((remoteEventListener, listenerOptions) -> server.addListener(remoteEventListener,
listenerOptions));
try
{
// NOTE: this listens on the wildcard address on an ephemeral port
server.open();
}
catch (IOException e)
{
throw new RuntimeException("Failed to create remote execution server for the application", e);
}
// add Bedrock specific System Properties
// NOTE: the Bedrock parent address for locally created applications is always "loopback" as
// i). they are always running locally,
// ii). they only need to connect locally, and
// iii). the "loopback" interface should work regardless of the network we're on.
InetAddress parentAddress;
IPv4Preferred iPv4Preferred = launchOptions.get(IPv4Preferred.class);
if (iPv4Preferred.isPreferred())
{
// we have to provide the schema with an IPv4 address!
try
{
parentAddress = InetAddress.getByName("127.0.0.1");
}
catch (UnknownHostException e)
{
// TODO: log that we couldn't determine the loopback address!
parentAddress = InetAddress.getLoopbackAddress();
}
}
else
{
// when the schema doesn't care, we can default to what this platform chooses
parentAddress = InetAddress.getLoopbackAddress();
}
Table systemPropertiesTable = new Table();
systemPropertiesTable.getOptions().add(Table.orderByColumn(0));
systemPropertiesTable.getOptions().add(Cell.Separator.of(""));
// establish the URI for this (parent) process
String parentURI = "//" + parentAddress.getHostAddress() + ":" + server.getPort();
systemPropertiesTable.addRow(Settings.PARENT_URI, parentURI.toString());
processBuilder.command().add("-D" + Settings.PARENT_URI + "=" + parentURI);
// add Orphanable configuration
Orphanable orphanable = launchOptions.get(Orphanable.class);
processBuilder.command().add("-D" + Settings.ORPHANABLE + "=" + orphanable.isOrphanable());
systemPropertiesTable.addRow(Settings.ORPHANABLE, Boolean.toString(orphanable.isOrphanable()));
// ----- establish the system properties for the java application -----
// define the system properties based on those defined by the launch options
Properties systemProperties = launchOptions.get(SystemProperties.class).resolve(platform, launchOptions);
for (String propertyName : systemProperties.stringPropertyNames())
{
String propertyValue = systemProperties.getProperty(propertyName);
// filter out (don't set) system properties that start with "bedrock", unless it's a profile
// (we don't want to have "parents" applications effect child applications
if (propertyName.startsWith("bedrock.profile.") ||!propertyName.startsWith("bedrock"))
{
processBuilder.command().add("-D" + propertyName
+ (propertyValue.isEmpty() ? "" : "=" + propertyValue));
systemPropertiesTable.addRow(propertyName, propertyValue);
}
}
diagnosticsTable.addRow("System Properties", systemPropertiesTable.toString());
// ----- establish Java Virtual Machine options -----
StringBuilder jvmOptions = new StringBuilder();
for (JvmOption jvmOption : launchOptions.getInstancesOf(JvmOption.class))
{
for (String value : jvmOption.resolve(launchOptions))
{
processBuilder.command().add(value);
if (jvmOptions.length() > 0)
{
jvmOptions.append(" ");
}
jvmOptions.append(value);
}
}
if (jvmOptions.length() > 0)
{
diagnosticsTable.addRow("Java Options", jvmOptions.toString());
}
// ----- add bedrock.runtime.inherit.xxx values to the command -----
for (String propertyName : System.getProperties().stringPropertyNames())
{
if (propertyName.startsWith("bedrock.runtime.inherit."))
{
// resolve the property value
String propertyValue = System.getProperty(propertyName);
// evaluate any expressions in the property value
ExpressionEvaluator evaluator = new ExpressionEvaluator(launchOptions.get(Variables.class));
propertyValue = evaluator.evaluate(propertyValue, String.class);
processBuilder.command().add(propertyValue);
}
}
// ----- establish the application command line to execute -----
// use the launcher to launch the application
// (we don't start the application directly itself)
String applicationLauncherClassName = JavaApplicationRunner.class.getName();
if (useModules)
{
applicationLauncherClassName = "com.oracle.bedrock.runtime/" + applicationLauncherClassName;
processBuilder.command().add("-m");
}
processBuilder.command().add(applicationLauncherClassName);
// set the Java application class name we need to launch
ClassName className = launchOptions.get(ClassName.class);
if (className == null)
{
throw new IllegalArgumentException("Java Application ClassName not specified");
}
String applicationClassName = className.getName();
processBuilder.command().add(applicationClassName);
diagnosticsTable.addRow("Application Launcher", applicationLauncherClassName);
diagnosticsTable.addRow("Application Class", applicationClassName);
diagnosticsTable.addRow("Application", displayName.resolve(launchOptions));
// ----- included the java arguments to the command -----
List argList = launchOptions.get(Arguments.class).resolve(platform, launchOptions);
// Set the actual arguments used back into the options
launchOptions.add(Arguments.of(argList));
String arguments = "";
for (String argument : argList)
{
processBuilder.command().add(argument);
arguments += argument + " ";
}
if (arguments.length() > 0)
{
diagnosticsTable.addRow("Application Arguments", arguments);
}
// should the standard error be redirected to the standard out?
ErrorStreamRedirection redirection = launchOptions.get(ErrorStreamRedirection.class);
processBuilder.redirectErrorStream(redirection.isEnabled());
diagnosticsTable.addRow("Standard Error Device",
redirection.isEnabled() ? "stdout" : "stderr");
diagnosticsTable.addRow("Application Launch Time",
new SimpleDateFormat("yyyy-MM-dd HH:mm:ss").format(new Date()));
// TODO: add all of the Profile options to the table
// ----- start the local process -----
boolean launchLogging = optionsByType.get(LaunchLogging.class).isEnabled();
if (launchLogging && LOGGER.isLoggable(Level.INFO))
{
LOGGER.log(Level.INFO,
"Oracle Bedrock " + Bedrock.getVersion() + ": Starting Application...\n"
+ "------------------------------------------------------------------------\n"
+ diagnosticsTable.toString() + "\n"
+ "------------------------------------------------------------------------\n");
}
// create and start the native process
Process process;
try
{
process = processBuilder.start();
}
catch (IOException e)
{
throw new RuntimeException("Failed to build the underlying native process for the application", e);
}
// ----- create the local process and application -----
// establish a LocalJavaProcess to represent the underlying Process
LocalJavaApplicationProcess localJavaProcess = new LocalJavaApplicationProcess(process,
server,
systemProperties);
// determine the application class that will represent the running application
Class extends A> applicationClass = metaClass.getImplementationClass(platform, launchOptions);
A application;
try
{
// attempt to find a constructor(Platform, JavaApplicationProcess, Options)
Constructor extends A> constructor = ReflectionHelper.getCompatibleConstructor(applicationClass,
platform.getClass(),
localJavaProcess.getClass(),
launchOptions.getClass());
// create the application
application = constructor.newInstance(platform, localJavaProcess, launchOptions);
}
catch (Exception e)
{
throw new RuntimeException("Failed to instantiate the Application class specified by the MetaClass:"
+ metaClass,
e);
}
// ----- enhance the application with java-specific features -----
if (JmxFeature.isSupportedBy(application))
{
application.add(new JmxFeature());
}
// ----- wait for the application to start -----
// ensure that the launcher process connects back to the server to
// know that the application has started
WaitToStart waitToStart = launchOptions.get(WaitToStart.class);
if (waitToStart.isEnabled())
{
Timeout timeout = launchOptions.get(Timeout.class);
ensure(new AbstractDeferred()
{
@Override
public Boolean get() throws TemporarilyUnavailableException, PermanentlyUnavailableException
{
if (!server.getRemoteChannels().iterator().hasNext())
{
throw new TemporarilyUnavailableException(this);
}
else
{
return true;
}
}
},within(timeout));
}
// ----- notify the MetaClass that the application has been launched -----
metaClass.onLaunched(platform, application, launchOptions);
// ----- notify the Profiles that the application has been launched -----
for (Profile profile : launchOptions.getInstancesOf(Profile.class))
{
profile.onLaunched(platform, application, launchOptions);
}
// ----- notify all of the application listeners -----
// notify the ApplicationListener-based Options that the application has been launched
for (ApplicationListener listener : launchOptions.getInstancesOf(ApplicationListener.class))
{
listener.onLaunched(application);
}
return application;
}
/**
* A {@link LocalApplicationProcess} specifically for Java-based applications.
*/
public static class LocalJavaApplicationProcess extends LocalApplicationProcess implements JavaApplicationProcess
{
/**
* The {@link RemoteChannel} for the {@link LocalJavaApplicationProcess}.
*/
private ControllableRemoteChannel remoteExecutor;
/**
* The resolved System {@link Properties} provided to the {@link JavaApplicationProcess} when it was launched.
*/
private Properties systemProperties;
/**
* Constructs a {@link LocalJavaApplicationProcess}.
*
* @param process the underlying operating system {@link Process}
* @param remoteExecutor the {@link ControllableRemoteChannel} that may be used
* to submit and control the process remotely
* @param systemProperties the resolved System {@link Properties} provided to the {}
*/
public LocalJavaApplicationProcess(Process process,
ControllableRemoteChannel remoteExecutor,
Properties systemProperties)
{
super(process);
this.remoteExecutor = remoteExecutor;
this.systemProperties = systemProperties;
}
@Override
public Properties getSystemProperties()
{
return systemProperties;
}
@Override
public CompletableFuture submit(RemoteCallable callable,
Option... options) throws IllegalStateException
{
return remoteExecutor.submit(callable, options);
}
@Override
public CompletableFuture submit(RemoteRunnable runnable,
Option... options) throws IllegalStateException
{
return remoteExecutor.submit(runnable, options);
}
@Override
public void addListener(RemoteEventListener listener,
Option... options)
{
remoteExecutor.addListener(listener, options);
}
@Override
public void removeListener(RemoteEventListener listener,
Option... options)
{
remoteExecutor.removeListener(listener, options);
}
@Override
public CompletableFuture raise(RemoteEvent event,
Option... options)
{
return remoteExecutor.raise(event, options);
}
@Override
public void close()
{
super.close();
remoteExecutor.close();
}
}
}