All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.ops4j.pax.runner.daemon.Daemon Maven / Gradle / Ivy

package org.ops4j.pax.runner.daemon;

import java.io.BufferedReader;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.FileReader;
import java.io.FileWriter;
import java.io.IOException;
import java.io.InputStream;
import java.io.PrintWriter;
import java.net.ServerSocket;
import java.net.Socket;
import java.security.AccessControlException;
import java.security.NoSuchAlgorithmException;
import java.util.Properties;

import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.ops4j.net.Base64Encoder;
import org.ops4j.pax.runner.CommandLine;
import org.ops4j.pax.runner.CommandLineImpl;
import org.ops4j.pax.runner.Run;
import org.ops4j.pax.runner.platform.DefaultJavaRunner;
import org.ops4j.pax.runner.platform.StoppableJavaRunner;

/**
 * The class that runs as Daemon for the Pax Runner. The {@link DaemonLauncher}
 * launches the instance of this class either as an attached or a detached process.
 * In either case, the instance of this class launches the Pax-Runner and waits
 * for appropriate shutdown command to be issued on its shutdown port.
 * 
* The Daemon process terminates either when the Pax-Runner completes execution, * or when a shutdown command is issued via shutdown port. * * @author Thomas Joseph * @since 0.20.0 (29 April 2009) * */ public class Daemon { // Constants ----------------------------------------------------- /** * Configuration option to specify password file, if not specified, the name * of the default password file. */ public static final String PASSWORD_FILE = "org.ops4j.pax.runner.daemon.password.file"; private static final String INFO_FILE = "org.ops4j.pax.runner.daemon.info"; private static final String LOCK_FILE = "org.ops4j.pax.runner.daemon.lock"; private static final String NEWLINE= "\r\n"; private static final String OPT_NETWORK_TIMEOUT="org.ops4j.pax.runner.daemon.network.timeout"; private static final String OPT_SHUTDOWN_TIMEOUT = "org.ops4j.pax.runner.daemon.shutdown.timeout"; private static final String OPT_SHUTDOWN_CMD = "org.ops4j.pax.runner.daemon.shutdown.cmd"; private static final String OPT_SHUTDOWN_PORT = "org.ops4j.pax.runner.daemon.shutdown.port"; // Attributes ---------------------------------------------------- private CommandLine commandLine = null; private String[] cmdArgs = null; private StoppableJavaRunner runner = null; private Thread shutdownHook = null; private int networkTimeout = 1000*60; private long shutdownTimeout = 0; private ServerSocket serverSocket = null; private boolean continueAwait = true; private static Daemon instance = null; private static String shutdown = "shutdown"; private static int shutdownPort = 8008; // Static -------------------------------------------------------- /** logger. */ private static final Log LOG = LogFactory.getLog(Daemon.class); /** * Launches a new Daemon instance, that will bring up the Pax Runner. */ public static void main(String[] args) { getInstance(); instance.load(args); instance.start(); } /** * @return The singleton instance of the Daemon. */ public static Daemon getInstance() { if (instance == null) instance = new Daemon(); return instance; } /** * Determines if any instance of the Daemon is already started. * * @return True if any instance of this class is already started, false otherwise. */ public static boolean isDaemonStarted() { File lock = new File(getRunnerHomeDir(false), LOCK_FILE); if (lock.exists() && lock.isFile()) { return true; } return false; } // Constructors -------------------------------------------------- // Public -------------------------------------------------------- /** * Prepares the instance of this class with configurations, before launching * the Pax-Runner. * * @param args The arguments passed to start this instance. */ public void load(String... args) { cmdArgs = args; commandLine = new CommandLineImpl(args); setNetworkTimeout(commandLine.getOption(OPT_NETWORK_TIMEOUT)); setShutdown(commandLine.getOption(OPT_SHUTDOWN_CMD)); setShutdownPort(commandLine.getOption(OPT_SHUTDOWN_PORT)); setShutdownTimeout(commandLine.getOption(OPT_SHUTDOWN_TIMEOUT)); } /** * Starts the Daemon - Launches Pax Runner, Opens up shutdown port on which * it will listen to shutdown command. * */ public void start() { new RunnerLauncher().start(); await(); } /** * Stops the running instance of the Daemon and Pax Runner if any. */ public void stop() { if (shutdownHook != null) { Runtime.getRuntime().removeShutdownHook(shutdownHook); LOG.trace("Removed Shutdown Hook."); shutdownHook.run(); shutdownHook = null; } else { shutdown(); } } // Package protected --------------------------------------------- /** * Retrieves the file that stores password information for the current * configuration. * * @return The password information file. */ static File getPasswordFile(String passwordFilePath) { if (passwordFilePath == null || passwordFilePath.length() == 0) { return new File(getRunnerHomeDir(true),PASSWORD_FILE); } return new File(passwordFilePath); } /** * Encrypts and returns the encrypted string for the given message. * * @param message the pass phrase that is to be encrypted. * @return the encrypted string for the given message. */ static String encrypt(String message) { java.security.MessageDigest d =null; try { d = java.security.MessageDigest.getInstance("SHA-1"); } catch (NoSuchAlgorithmException e) { return null; } d.reset(); d.update(message.getBytes()); return new String(Base64Encoder.encode(d.digest())); } /** * Creates a file on the file system with the given content for the file. * * @param file The file that should be written to the filesystem. * @param content The content for the file. */ static void writeToFile(File file, String content) { FileWriter fw = null; try { fw = new FileWriter(file); fw.write(content); fw.flush(); } catch (IOException e) { throw new RuntimeException ("Error creating file.",e); } finally { try { if(fw != null) fw.close(); fw=null; } catch (IOException e) {/* ignore */} } } /** * * @return The shutdown port that was configured (if any) to launch the * Daemon (if it was launched), the default port otherwise * * @see #OPT_SHUTDOWN_PORT */ static int getShutdownPort() { int port = shutdownPort; String sPort = readDaemonProperty(OPT_SHUTDOWN_PORT); if ((port = parseSafeInt( sPort)) != -1) return port; return shutdownPort; } /** * Returns the command that must be issued to the shutdown port to stop the * launched Daemon instance. * * @return The shutdown command that was configured (if any) when launching the * Daemon instance (if it was launched), the default command otherwise * * @see #OPT_SHUTDOWN_CMD */ static String getShutdown() { String read = readDaemonProperty(OPT_SHUTDOWN_CMD); if (read!=null && read.length() > 0) return read; return shutdown; } // Protected ----------------------------------------------------- // Private ------------------------------------------------------- private void await() { createInfoFile(); createLockFile(); shutdownHook = createShutdownHook(); Runtime.getRuntime().addShutdownHook(shutdownHook); // Set up a server socket to wait on try { LOG.debug("Setting up shutdown port on " + getShutdownPort()); serverSocket = new ServerSocket(getShutdownPort()); } catch (IOException e) { throw new RuntimeException("Unable to set up shutdown port [" + getShutdownPort() + "].", e); } // Loop waiting for a connection and a valid command while (continueAwait) { // Wait for the next connection try { // handle each new connection in a separate thread new ClientHandler().handle(serverSocket.accept()); } catch (IOException e) { LOG.debug("Stopped accepting connections." + e.getMessage()); } } // Close the server socket and return try { if (serverSocket != null) { serverSocket.close(); } } catch (IOException e) { ; } LOG.trace("Finished awaiting..."); } private void stopAwait() { continueAwait = false; try { if (serverSocket != null) { serverSocket.close(); serverSocket = null; LOG.info("Stopped shutdown port."); } } catch (IOException e) { ; } } private StoppableJavaRunner createJavaRunner() { return new DefaultJavaRunner(); } /** * Creates a simple shutdown hook. * * @return Unstarted thread that can be registered as a shutdown hook */ private Thread createShutdownHook() { return new Thread( new Runnable() { public void run() { LOG.trace("Executing shutdown hook..."); shutdown(); } }, "Pax Runner Daemon Shutdown Hook" ); } /** * Creates a "lock" file. An empty file that is created as the Daemon is * started (attached/detached) and removed when the it stops. The file * will be used to determine if the Daemon is already running. */ private void createLockFile() { File lock = new File(getRunnerHomeDir(true), LOCK_FILE); if (lock.exists()) { throw new RuntimeException(LOCK_FILE + " exists. Please make sure" + " that the Pax Runner daemon is not already running."); } try { lock.createNewFile(); lock.deleteOnExit(); } catch (IOException e) { e.printStackTrace(); } } /** * Creates a file that stores information about the running instance of the * Daemon. */ private void createInfoFile() { File info = new File(getRunnerHomeDir(true), INFO_FILE); int count = 10; while ( info.exists() && count > 0 ) { info.delete(); count --; } try { info.createNewFile(); String content = OPT_SHUTDOWN_CMD + "=" + shutdown + NEWLINE + OPT_SHUTDOWN_PORT + "=" + shutdownPort; Daemon.writeToFile(info, content); } catch (IOException e) { e.printStackTrace(); } } private String readEncryptedPassword() { return readFile(getPasswordFile(commandLine.getOption(Daemon.PASSWORD_FILE))); } private String readFile(File file) { try { LOG.debug("Read file:" + file.getAbsolutePath()); BufferedReader br = new BufferedReader(new FileReader(file)); return br.readLine(); } catch (FileNotFoundException e) { throw new RuntimeException(e); } catch (IOException e) { throw new RuntimeException(e); } } private void shutdown() { stopAwait(); if (runner != null) { LOG.debug("Bringing down Runner..."); runner.shutdown(); runner = null; LOG.info("Pax Runner daemon stopped."); } } private void setShutdownPort(String shutdownPort) { if (parseSafeInt(shutdownPort) != -1) { Daemon.shutdownPort = parseSafeInt(shutdownPort); } } private void setNetworkTimeout(String networkTimeout) { if (parseSafeInt(networkTimeout) != -1) { this.networkTimeout = parseSafeInt(networkTimeout); } } private void setShutdownTimeout(String shutdownTimeout) { if (parseSafeInt(shutdownTimeout) != -1) { this.shutdownTimeout = parseSafeInt(shutdownTimeout); } } /** * Returns the file reference of the Runner's home directory. Creates one if * it doesn't exist and if the create flag is set to true. * * @return */ private static File getRunnerHomeDir(boolean create) { String homeDirPath = System.getProperty("user.home") + File.separator + ".pax"+ File.separator + "runner"; final File homeDir = new File(homeDirPath); if ( !homeDir.exists() && create ) { if( homeDir.mkdirs() ) { LOG.debug("Created Pax Runner Home."); } } return homeDir; } private static int parseSafeInt(String arg) { if (arg != null && arg.length() > 0) { try { return Integer.parseInt(arg); } catch(Exception e) {} } return -1; } private static String readDaemonProperty(String key) { File info = new File(getRunnerHomeDir(false), INFO_FILE); if (info.exists()) { Properties props = new Properties(); try { props.load(new FileInputStream(info)); return props.getProperty(key); } catch (FileNotFoundException e) {;} catch (IOException e) {;} } return null; } // Inner classes ------------------------------------------------- /** * Launches the Pax Runner in a separate thread within the same process. */ class RunnerLauncher extends Thread { RunnerLauncher () { super("RunnerLauncher"); this.setDaemon(true); } public void run() { if (runner == null) { runner = createJavaRunner(); LOG.trace("Created Runner."); } try { Run.main(runner, cmdArgs); } finally { instance.shutdown(); } } } /** * Handles a individual client on the given socket as a separate thread. If * the client is connecting from a loopback adapter (localhost), no password is required. * If the client connects via telnet, password is required to issue commands. * The shutdown sequence begins if the client enters the correct shutdown command. * * @author Thomas Joseph. * */ class ClientHandler { public void handle(final Socket socket) { new Thread( new Runnable () { public void run() { boolean isLocalConnect = false; InputStream stream = null; PrintWriter out = null; try { isLocalConnect = socket.getInetAddress().isLoopbackAddress(); socket.setSoTimeout(networkTimeout); LOG.trace("Connected."); stream = socket.getInputStream(); out = new PrintWriter(socket.getOutputStream(), true); } catch (AccessControlException ace) { LOG.warn("StandardServer.accept security exception: " + ace.getMessage(), ace); return; } catch (IOException e) { //LOG.error("StandardServer.await: accept: ", e); return; } // ASK FOR PASSWORD for remote system if (!isLocalConnect) { // Write a welcome note String welcome = "Welcome to Pax-Runner Remote Console"; String underline = "==============================================="; out.write(underline.substring(0, welcome.length()) + NEWLINE); out.write(welcome + NEWLINE); out.write(underline.substring(0, welcome.length()) + NEWLINE); out.flush(); String passwordHash = new String(readEncryptedPassword()); if (passwordHash != null && passwordHash.trim().length() > 0) { int count = 3; boolean matched = false; while (count > 0) { --count; out.write("Please enter password: "); out.flush(); String clientPassword = readStream(stream); readStream(stream); // skipping control character clientPassword = new String(encrypt(clientPassword)); if (passwordHash.trim().equals(clientPassword.trim())) { matched = true; break; } } if (!matched) { out.write("Invalid password attempts exceeded limits."+NEWLINE); out.write("Try connecting again."+NEWLINE); out.flush(); try { socket.close(); } catch (IOException e) { ; } return; } } } LOG.trace("Going to ask the client to supply command."); out.write("Enter the command to shutdown the Pax Runner:"+NEWLINE); out.write("> "); out.flush(); // Read a set of characters from the socket String command = readStream(stream); LOG.trace("Recieved Command [ " + command + "] from [" +socket.getRemoteSocketAddress() + "]."); try { String[] commands = command.split("\\\\s+"); if (commands.length > 1) { command = commands[0]; setShutdownTimeout(command); } } catch (Exception e) { ;// ignore } // Match against our command string boolean match = command.toString().equals(shutdown); if (match) { LOG.trace("Shutdown command recieved via Telnet."); LOG.trace("Stop after " + shutdownTimeout + "ms."); try { Thread.sleep(shutdownTimeout); } catch (InterruptedException e) { LOG.warn("Problems in shutdown timeout."); } stop(); return; } else { LOG.warn("Pax Runner: Invalid command."); out.write("Invalid Command!"+NEWLINE); out.flush(); } // Close the socket now that we are done with it try { LOG.trace("Closing the socket now that we are done with it..."); socket.close(); } catch (IOException e) { LOG.warn("Exception in closing socket..", e); } } }, "ClientHandler-" + socket.getRemoteSocketAddress() ).start(); } /** * Reads the given stream for string commands. * * @param stream * @return the command that was issued in the input stream. */ private String readStream(final InputStream stream) { StringBuffer command = new StringBuffer(); int expected = 1024; // Cut off to avoid DoS attack while (stream!=null && expected > 0) { int ch = -1; try { ch = stream.read(); } catch (IOException e) { ch = -1; } if (ch < 32) { // Control character or EOF terminates loop break; } command.append((char) ch); expected--; } return command.toString(); } } // Getters and Setters ----------------------------------------------------- private void setShutdown(String shutdown) { if (shutdown != null && shutdown.length() > 0) Daemon.shutdown = shutdown; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy