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

org.eclipse.jetty.start.Main Maven / Gradle / Ivy

//
// ========================================================================
// Copyright (c) 1995 Mort Bay Consulting Pty Ltd and others.
//
// This program and the accompanying materials are made available under the
// terms of the Eclipse Public License v. 2.0 which is available at
// https://www.eclipse.org/legal/epl-2.0, or the Apache License, Version 2.0
// which is available at https://www.apache.org/licenses/LICENSE-2.0.
//
// SPDX-License-Identifier: EPL-2.0 OR Apache-2.0
// ========================================================================
//

package org.eclipse.jetty.start;

import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.LineNumberReader;
import java.io.OutputStream;
import java.io.PrintStream;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.net.ConnectException;
import java.net.InetAddress;
import java.net.Socket;
import java.net.SocketTimeoutException;
import java.nio.file.Files;
import java.nio.file.Path;
import java.util.ArrayList;
import java.util.Comparator;
import java.util.List;
import java.util.Objects;
import java.util.Optional;
import java.util.Set;
import java.util.jar.JarEntry;
import java.util.jar.JarFile;
import java.util.jar.Manifest;
import java.util.stream.Collectors;

import org.eclipse.jetty.start.Props.Prop;
import org.eclipse.jetty.start.config.CommandLineConfigSource;
import org.eclipse.jetty.start.config.ConfigSource;
import org.eclipse.jetty.util.FileID;

import static org.eclipse.jetty.start.UsageException.ERR_BAD_STOP_PROPS;
import static org.eclipse.jetty.start.UsageException.ERR_INVOKE_MAIN;
import static org.eclipse.jetty.start.UsageException.ERR_NOT_STOPPED;
import static org.eclipse.jetty.start.UsageException.ERR_UNKNOWN;

/**
 * Main start class.
 * 

* This class is intended to be the main class listed in the MANIFEST.MF of the start.jar archive. It allows the Jetty Application server to be started with the * command "java -jar start.jar". *

* Argument processing steps: *

    *
  1. Directory Location: jetty.home=[directory] (the jetty.home location)
  2. *
  3. Directory Location: jetty.base=[directory] (the jetty.base location)
  4. *
  5. Start Logging behavior: --debug (debugging enabled)
  6. *
  7. Start Logging behavior: --start-log-file=logs/start.log (output start logs to logs/start.log location)
  8. *
  9. Module Resolution
  10. *
  11. Properties Resolution
  12. *
  13. Present Optional Informational Options
  14. *
  15. Normal Startup
  16. *
*/ public class Main { private static final int EXIT_USAGE = 1; public static void main(String[] args) { boolean test = false; try { Main main = new Main(); StartArgs startArgs = main.processCommandLine(args); test = startArgs.isTestingModeEnabled(); main.start(startArgs); } catch (UsageException e) { StartLog.error(e.getMessage()); usageExit(e.getCause(), e.getExitCode(), test); } catch (Throwable e) { usageExit(e, UsageException.ERR_UNKNOWN, test); } } static void usageExit(int exit) { usageExit(null, exit, false); } static void usageExit(Throwable t, int exit, boolean test) { if (t != null) { t.printStackTrace(System.err); } System.err.println(); System.err.println("Usage: java -jar $JETTY_HOME/start.jar [options] [properties] [configs]"); System.err.println(" java -jar $JETTY_HOME/start.jar --help # for more information"); if (test) System.err.println("EXIT: " + exit); else System.exit(exit); } private BaseHome baseHome; private StartArgs jsvcStartArgs; public Main() throws IOException { } private void copyInThread(final InputStream in, final OutputStream out) { new Thread(() -> { try { byte[] buf = new byte[1024]; int len = in.read(buf); while (len > 0) { out.write(buf, 0, len); len = in.read(buf); } } catch (IOException e) { // e.printStackTrace(); } }).start(); } private void listClasspath(String name, PrintStream out, Classpath classpath) { StartLog.endStartLog(); out.println(); out.printf("Classpath: %s%n", name); out.printf("-----------%s%n", "-".repeat(name.length())); if (classpath.count() == 0) { out.println("No classpath entries and/or version information available show."); return; } out.println("Version Information on " + classpath.count() + " entr" + ((classpath.count() > 1) ? "ies" : "y") + " in the classpath."); out.println("Note: order presented here is how they would appear on the classpath."); out.println(" changes to the --module=name command line options will be reflected here."); int i = 0; for (Path element : classpath.getElements()) { String license = getLicenceFromJar(element); if (license != null && !license.isEmpty()) out.printf("%2d: %24s | %s | %s\n", i++, getVersion(element), baseHome.toShortForm(element), license); else out.printf("%2d: %24s | %s\n", i++, getVersion(element), baseHome.toShortForm(element)); } } private String getLicenceFromJar(Path jar) { if (!Files.exists(jar) || Files.isDirectory(jar) || !Files.isReadable(jar)) return null; try { try (JarFile jarFile = new JarFile(jar.toFile())) { Manifest manifest = jarFile.getManifest(); if (manifest != null) { String spdxLicense = manifest.getMainAttributes().getValue("SPDX-License-Identifier"); if (spdxLicense != null) return spdxLicense; String bundleLicense = manifest.getMainAttributes().getValue("Bundle-License"); if (bundleLicense != null) return bundleLicense; } Optional license = jarFile.stream().filter(Main::isLicenseFile).map(e -> getLicenceFromFile(jarFile, e)).filter(Objects::nonNull).findFirst(); if (license.isPresent()) return license.get(); } } catch (Throwable ignored) { } return null; } private static boolean isLicenseFile(JarEntry entry) { String name = entry.getName(); return name.matches("(?i)^(META-INF/)?LICEN[SC]E.*") || name.matches("(?i)^LICEN[SC]E.*"); } private String getLicenceFromFile(JarFile jarFile, JarEntry entry) { try { try (BufferedReader reader = new BufferedReader(new InputStreamReader(jarFile.getInputStream(entry)))) { String line; StringBuilder licenseBuilder = new StringBuilder(); int nonEmptyLines = 0; List links = new ArrayList<>(); while ((line = reader.readLine()) != null) { line = line.trim(); if (!line.isEmpty()) { if (line.contains("SPDX-License-Identifier:")) return line.substring(line.indexOf(':') + 1).trim(); if (line.startsWith("http:") || line.startsWith("https:")) links.add(line); if (nonEmptyLines < 2) { licenseBuilder.append(line).append(" "); nonEmptyLines++; } } } if (!links.isEmpty()) return links.stream().max(Comparator.comparingInt(String::length)).get(); return nonEmptyLines > 0 ? licenseBuilder.toString().trim() : null; } } catch (Throwable ignored) { } return null; } public BaseHome getBaseHome() { return baseHome; } private String getVersion(Path element) { if (Files.isDirectory(element)) { return "(dir)"; } if (Files.isRegularFile(element) && FileID.isJavaArchive(element)) { return JarVersion.getVersion(element); } return ""; } public void invokeMain(ClassLoader classloader, StartArgs args) throws IllegalAccessException, InvocationTargetException, NoSuchMethodException, IOException { if (args.getSelectedModules().isEmpty()) { if (Files.exists(getBaseHome().getBasePath("start.jar"))) StartLog.error("Do not start with ${jetty.base} == ${jetty.home}!"); else StartLog.error("No enabled jetty modules found!"); StartLog.info("${jetty.home} = %s", getBaseHome().getHomePath()); StartLog.info("${jetty.base} = %s", getBaseHome().getBasePath()); StartLog.error("Please create and/or configure a ${jetty.base} directory."); usageExit(ERR_INVOKE_MAIN); return; } Class invokedClass; String mainclass = args.getMainClassname(); try { invokedClass = classloader.loadClass(mainclass); } catch (ClassNotFoundException e) { StartLog.error("Unable to find: %s", mainclass); StartLog.debug(e); usageExit(ERR_INVOKE_MAIN); return; } StartLog.debug("%s - %s", invokedClass, invokedClass.getPackage().getImplementationVersion()); CommandLineBuilder cmd = args.getMainArgs(StartArgs.ARG_PARTS); String[] argArray = cmd.getArgs().toArray(new String[0]); StartLog.debug("Command Line Args: %s", cmd.toString()); Class[] methodParamTypes = {argArray.getClass()}; Method main = invokedClass.getDeclaredMethod("main", methodParamTypes); Object[] methodParams = new Object[]{argArray}; StartLog.endStartLog(); main.invoke(null, methodParams); } public void listConfig(PrintStream out, StartArgs args) { StartLog.endStartLog(); Modules modules = args.getAllModules(); // Dump Enabled Modules modules.listEnabled(out); // Dump Jetty Home / Base args.dumpJavaEnvironment(out); // Dump JVM Args args.dumpJvmArgs(out); // Dump System Properties args.dumpSystemProperties(out); StartEnvironment jettyEnvironment = args.getJettyEnvironment(); // Dump Jetty Properties jettyEnvironment.dumpProperties(out); // Dump Jetty Classpath listClasspath(jettyEnvironment.getName(), out, jettyEnvironment.getClasspath()); // Dump Jetty Resolved XMLs jettyEnvironment.dumpActiveXmls(out); for (StartEnvironment environment : args.getEnvironments()) { // Dump Properties environment.dumpProperties(out); // Dump Classpath listClasspath(environment.getName(), out, environment.getClasspath()); // Dump Resolved XMLs environment.dumpActiveXmls(out); } } public void listModules(PrintStream out, StartArgs args) { final List tags = args.getListModules(); StartLog.endStartLog(); String t = tags.toString(); out.printf("%nModules %s:%n", t); out.printf("=========%s%n", "=".repeat(t.length())); args.getAllModules().listModules(out, tags); // for default module listings, also show enabled modules if ("[-internal]".equals(t) || "[*]".equals(t)) args.getAllModules().listEnabled(out); } public void showModules(PrintStream out, StartArgs args) { StartLog.endStartLog(); args.getAllModules().showModules(out, args.getShowModules()); } /** * Convenience for processCommandLine(cmdLine.toArray(new String[cmdLine.size()])) * * @param cmdLine the command line * @return the start args parsed from the command line * @throws Exception if unable to process the command line */ public StartArgs processCommandLine(List cmdLine) throws Exception { return this.processCommandLine(cmdLine.toArray(new String[0])); } public StartArgs processCommandLine(String[] cmdLine) throws Exception { // Processing Order is important! // 1) Configuration Locations CommandLineConfigSource cmdLineSource = new CommandLineConfigSource(cmdLine); baseHome = new BaseHome(cmdLineSource); StartArgs args = new StartArgs(baseHome); StartLog.debug("jetty.home=%s", baseHome.getHome()); StartLog.debug("jetty.base=%s", baseHome.getBase()); // 3) Module Registration Modules modules = new Modules(baseHome, args); StartLog.debug("Registering all modules"); modules.registerAll(); args.setAllModules(modules); // 2) Parse everything provided. // This would be the directory information + // the various start inis // and then the raw command line arguments StartLog.debug("Parsing collected arguments"); args.parse(baseHome.getConfigSources()); Props props = baseHome.getConfigSources().getProps(); Prop home = props.getProp(BaseHome.JETTY_HOME); Props argProps = args.getJettyEnvironment().getProperties(); if (!argProps.containsKey(BaseHome.JETTY_HOME)) argProps.setProperty(home); argProps.setProperty(BaseHome.JETTY_HOME + ".uri", normalizeURI(baseHome.getHomePath().toUri().toString()), home.source); Prop base = props.getProp(BaseHome.JETTY_BASE); if (!argProps.containsKey(BaseHome.JETTY_BASE)) argProps.setProperty(base); argProps.setProperty(BaseHome.JETTY_BASE + ".uri", normalizeURI(baseHome.getBasePath().toUri().toString()), base.source); // 4) Active Module Resolution Set selectedModules = args.getSelectedModules(); List sortedSelectedModules = modules.getSortedNames(selectedModules); List unknownModules = new ArrayList<>(selectedModules); unknownModules.removeAll(sortedSelectedModules); if (unknownModules.size() >= 1) { throw new UsageException(UsageException.ERR_UNKNOWN, "Unknown module%s=[%s] List available with --list-modules", unknownModules.size() > 1 ? 's' : "", String.join(", ", unknownModules)); } for (String selectedModule : sortedSelectedModules) { for (String source : args.getSources(selectedModule)) { String shortForm = baseHome.toShortForm(source); modules.enable(selectedModule, shortForm); } } List activeModules = modules.getEnabled(); final Version START_VERSION = new Version(StartArgs.VERSION); for (Module enabled : activeModules) { if (enabled.getVersion().isNewerThan(START_VERSION)) { throw new UsageException(UsageException.ERR_BAD_GRAPH, "Module [" + enabled.getName() + "] specifies jetty version [" + enabled.getVersion() + "] which is newer than this version of jetty [" + START_VERSION + "]"); } } for (String name : args.getSkipFileValidationModules()) { Module module = modules.get(name); module.setSkipFilesValidation(true); } // 5) Lib & XML Expansion / Resolution // 6) Resolve Extra XMLs // 7) JPMS Expansion // 8) Resolve Property Files args.expandEnvironments(activeModules); return args; } private String normalizeURI(String uri) { if (uri.endsWith("/")) return uri.substring(0, uri.length() - 1); return uri; } public void start(StartArgs args) throws IOException, InterruptedException { StartLog.debug("StartArgs: %s", args); // Get Desired Classpath based on user provided Active Options. Classpath classpath = args.getJettyEnvironment().getClasspath(); // Show the usage information and return if (args.isHelp()) { usage(true); } // Show the version information and return if (args.isListClasspath()) { listClasspath("Jetty", System.out, classpath); } // Show configuration if (args.isListConfig()) { listConfig(System.out, args); } // List modules if (args.getListModules() != null) { listModules(System.out, args); } // Show modules if (args.getShowModules() != null) { showModules(System.out, args); } // Generate Module Graph File if (args.getModuleGraphFilename() != null) { Path outputFile = baseHome.getBasePath(args.getModuleGraphFilename()); System.out.printf("Generating GraphViz Graph of Jetty Modules at %s%n", baseHome.toShortForm(outputFile)); ModuleGraphWriter writer = new ModuleGraphWriter(); writer.config(args.getJettyEnvironment().getProperties()); writer.write(args.getAllModules(), outputFile); } // Show Command Line to execute Jetty if (args.isDryRun()) { CommandLineBuilder cmd = args.getMainArgs(args.getDryRunParts()); cmd.debug(); System.out.println(cmd.toCommandLine()); } if (args.isStopCommand()) { doStop(args); } if (args.isUpdateIni()) { for (ConfigSource config : baseHome.getConfigSources()) { for (StartIni ini : config.getStartInis()) { ini.update(baseHome, args.getJettyEnvironment().getProperties()); } } } // Check base directory BaseBuilder baseBuilder = new BaseBuilder(baseHome, args); if (baseBuilder.build()) StartLog.info("Base directory was modified"); else if (args.isCreateFiles() || !args.getStartModules().isEmpty()) StartLog.info("Base directory was not modified"); // Check module dependencies args.getAllModules().checkEnabledModules(); // Informational command line, don't run jetty if (!args.isRun()) { return; } // execute Jetty in another JVM if (args.isExec()) { CommandLineBuilder cmd = args.getMainArgs(StartArgs.ALL_PARTS); cmd.debug(); List execModules = args.getAllModules().getEnabled().stream() // Keep only the forking modules. .filter(module -> !module.getJvmArgs().isEmpty()) .map(Module::getName) .collect(Collectors.toList()); StartLog.warn("Forking second JVM due to forking module(s): %s. Use --dry-run to generate the command line to avoid forking.", execModules); ProcessBuilder pbuilder = new ProcessBuilder(cmd.getArgs()); StartLog.endStartLog(); final Process process = pbuilder.start(); Runtime.getRuntime().addShutdownHook(new Thread(() -> { StartLog.debug("Destroying %s", process); process.destroy(); })); copyInThread(process.getErrorStream(), System.err); copyInThread(process.getInputStream(), System.out); copyInThread(System.in, process.getOutputStream()); process.waitFor(); System.exit(0); // exit JVM when child process ends. return; } if (args.hasJvmArgs() || args.hasSystemProperties()) { StartLog.warn("Unknown Arguments detected. Consider using --dry-run or --exec"); if (args.hasSystemProperties()) args.getSystemProperties().forEach((k, v) -> StartLog.warn(" Argument: -D%s=%s (interpreted as a System property, from %s)", k, System.getProperty(k), v)); if (args.hasJvmArgs()) args.getJvmArgSources().forEach((jvmArg, source) -> StartLog.warn(" Argument: %s (interpreted as a JVM argument, from %s)", jvmArg, source)); } ClassLoader cl = classpath.getClassLoader(); Thread.currentThread().setContextClassLoader(cl); // Invoke the Main Class try { invokeMain(cl, args); } catch (Throwable e) { e.printStackTrace(); usageExit(e, ERR_INVOKE_MAIN, args.isTestingModeEnabled()); } } /* implement Apache commons daemon (jsvc) lifecycle methods (init, start, stop, destroy) */ public void start() throws Exception { start(jsvcStartArgs); } private void doStop(StartArgs args) { Props argsProps = args.getJettyEnvironment().getProperties(); final Prop stopHostProp = argsProps.getProp("STOP.HOST", true); final Prop stopPortProp = argsProps.getProp("STOP.PORT", true); final Prop stopKeyProp = argsProps.getProp("STOP.KEY", true); final Prop stopWaitProp = argsProps.getProp("STOP.WAIT", true); String stopHost = "127.0.0.1"; int stopPort = -1; String stopKey = ""; if (stopHostProp != null) { stopHost = stopHostProp.value; } if (stopPortProp != null) { stopPort = Integer.parseInt(stopPortProp.value); } if (stopKeyProp != null) { stopKey = stopKeyProp.value; } if (stopWaitProp != null) { int stopWait = Integer.parseInt(stopWaitProp.value); stop(stopHost, stopPort, stopKey, stopWait); } else { stop(stopHost, stopPort, stopKey); } } /** * Stop a running jetty instance. * * @param host the host * @param port the port * @param key the key */ public void stop(String host, int port, String key) { stop(host, port, key, 0); } public void stop(String host, int port, String key, int timeout) { if (host == null || host.length() == 0) { host = "127.0.0.1"; } try { if ((port <= 0) || (port > 65535)) { System.err.println("STOP.PORT property must be specified with a valid port number"); usageExit(ERR_BAD_STOP_PROPS); } if (key == null) { key = ""; System.err.println("STOP.KEY property must be specified"); System.err.println("Using empty key"); } try (Socket s = new Socket(InetAddress.getByName(host), port)) { if (timeout > 0) { s.setSoTimeout(timeout * 1000); } try (OutputStream out = s.getOutputStream()) { out.write((key + "\r\nstop\r\n").getBytes()); out.flush(); if (timeout > 0) { StartLog.info("Waiting %,d seconds for jetty to stop%n", timeout); LineNumberReader lin = new LineNumberReader(new InputStreamReader(s.getInputStream())); String response; while ((response = lin.readLine()) != null) { StartLog.debug("Received \"%s\"", response); if ("Stopped".equals(response)) { StartLog.warn("Server reports itself as Stopped"); } } } } } } catch (SocketTimeoutException e) { StartLog.warn("Timed out waiting for stop confirmation"); System.exit(ERR_UNKNOWN); } catch (ConnectException e) { usageExit(e, ERR_NOT_STOPPED, jsvcStartArgs != null && jsvcStartArgs.isTestingModeEnabled()); } catch (Exception e) { usageExit(e, ERR_UNKNOWN, jsvcStartArgs != null && jsvcStartArgs.isTestingModeEnabled()); } } /* implement Apache commons daemon (jsvc) lifecycle methods (init, start, stop, destroy) */ public void stop() { doStop(jsvcStartArgs); } public void usage(boolean exit) { StartLog.endStartLog(); if (!printTextResource("org/eclipse/jetty/start/usage.txt")) { StartLog.warn("detailed usage resource unavailable"); } if (exit) { System.exit(EXIT_USAGE); } } public static boolean printTextResource(String resourceName) { boolean resourcePrinted = false; try (InputStream stream = Thread.currentThread().getContextClassLoader().getResourceAsStream(resourceName)) { if (stream != null) { try (InputStreamReader reader = new InputStreamReader(stream); BufferedReader buf = new BufferedReader(reader)) { resourcePrinted = true; String line; while ((line = buf.readLine()) != null) { System.out.println(line); } } } else { StartLog.warn("Unable to find resource: %s", resourceName); } } catch (IOException e) { StartLog.warn(e); } return resourcePrinted; } /* implement Apache commons daemon (jsvc) lifecycle methods (init, start, stop, destroy) */ public void init(String[] args) { try { jsvcStartArgs = processCommandLine(args); } catch (UsageException e) { StartLog.error(e.getMessage()); usageExit(e.getCause(), e.getExitCode(), false); } catch (Throwable e) { usageExit(e, UsageException.ERR_UNKNOWN, false); } } /* implement Apache commons daemon (jsvc) lifecycle methods (init, start, stop, destroy) */ public void destroy() { } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy