/* * Copyright (c) 2011, 2022, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it * under the terms of the GNU General Public License version 2 only, as * published by the Free Software Foundation. Oracle designates this * particular file as subject to the "Classpath" exception as provided * by Oracle in the LICENSE file that accompanied this code. * * This code is distributed in the hope that it will be useful, but WITHOUT * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License * version 2 for more details (a copy is included in the LICENSE file that * accompanied this code). * * You should have received a copy of the GNU General Public License version * 2 along with this work; if not, write to the Free Software Foundation, * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. * * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA * or visit www.oracle.com if you need additional information or have any * questions. */ package javacserver.server; import java.io.BufferedReader; import java.io.FileNotFoundException; import java.io.IOException; import java.io.InputStreamReader; import java.io.PrintStream; import java.io.PrintWriter; import java.io.StringWriter; import java.net.InetAddress; import java.net.InetSocketAddress; import java.net.ServerSocket; import java.net.Socket; import java.net.SocketException; import java.nio.file.Path; import java.util.Optional; import java.util.Random; import java.util.concurrent.atomic.AtomicBoolean; import java.util.spi.ToolProvider; import javacserver.shared.PortFile; import javacserver.shared.Protocol; import javacserver.shared.Result; import javacserver.util.LazyInitFileLog; import javacserver.util.Log; import javacserver.util.LoggingOutputStream; import javacserver.util.Util; /** * Start a new server main thread, that will listen to incoming connection requests from the client, * and dispatch these on to worker threads in a thread pool, running javac. */ public class Server { private ServerSocket serverSocket; private PortFile portFile; private PortFileMonitor portFileMonitor; private IdleMonitor idleMonitor; private CompilerThreadPool compilerThreadPool; // Set to false break accept loop final AtomicBoolean keepAcceptingRequests = new AtomicBoolean(); // For logging server internal (non request specific) errors. private static LazyInitFileLog errorLog; public static void main(String... args) { initLogging(); try { PortFile portFile = getPortFileFromArguments(args); if (portFile == null) { System.exit(Result.CMDERR.exitCode); return; } Server server = new Server(portFile); if (!server.start()) { System.exit(Result.ERROR.exitCode); } else { System.exit(Result.OK.exitCode); } } catch (IOException | InterruptedException ex) { ex.printStackTrace(); System.exit(Result.ERROR.exitCode); } } private static void initLogging() { // Under normal operation, all logging messages generated server-side // are due to compilation requests. These logging messages should // be relayed back to the requesting client rather than written to the // server log. The only messages that should be written to the server // log (in production mode) should be errors, errorLog = new LazyInitFileLog("server.log"); Log.setLogForCurrentThread(errorLog); Log.setLogLevel(Log.Level.ERROR); // should be set to ERROR. // Make sure no exceptions go under the radar Thread.setDefaultUncaughtExceptionHandler((t, e) -> { restoreServerErrorLog(); Log.error(e); }); // Inevitably someone will try to print messages using System.{out,err}. // Make sure this output also ends up in the log. System.setOut(new PrintStream(new LoggingOutputStream(System.out, Log.Level.INFO, "[stdout] "))); System.setErr(new PrintStream(new LoggingOutputStream(System.err, Log.Level.ERROR, "[stderr] "))); } private static PortFile getPortFileFromArguments(String[] args) { if (args.length != 1) { Log.error("javacserver daemon incorrectly called"); return null; } String portfilename = args[0]; PortFile portFile = new PortFile(portfilename); return portFile; } public Server(PortFile portFile) throws FileNotFoundException { this.portFile = portFile; } /** * Start the daemon, unless another one is already running, in which it returns * false and exits immediately. */ private boolean start() throws IOException, InterruptedException { // The port file is locked and the server port and cookie is written into it. portFile.lock(); portFile.getValues(); if (portFile.containsPortInfo()) { Log.debug("javacserver daemon not started because portfile exists!"); portFile.unlock(); return false; } serverSocket = new ServerSocket(); InetAddress localhost = InetAddress.getByName(null); serverSocket.bind(new InetSocketAddress(localhost, 0)); // At this point the server accepts connections, so it is now safe // to publish the port / cookie information // The secret cookie shared between server and client through the port file. // Used to prevent clients from believing that they are communicating with // an old server when a new server has started and reused the same port as // an old server. long myCookie = new Random().nextLong(); portFile.setValues(serverSocket.getLocalPort(), myCookie); portFile.unlock(); portFileMonitor = new PortFileMonitor(portFile, this::shutdownServer); portFileMonitor.start(); compilerThreadPool = new CompilerThreadPool(); idleMonitor = new IdleMonitor(this::shutdownServer); Log.debug("javacserver daemon started. Accepting connections..."); Log.debug(" port: " + serverSocket.getLocalPort()); Log.debug(" time: " + new java.util.Date()); Log.debug(" poolsize: " + compilerThreadPool.poolSize()); keepAcceptingRequests.set(true); do { try { Socket socket = serverSocket.accept(); // Handle each incoming request in a separate thread. This is just for socket communication, // the actual compilation will be done by the threadpool. Thread requestHandler = new Thread(() -> handleRequest(socket)); requestHandler.start(); } catch (SocketException se) { // Caused by serverSocket.close() and indicates shutdown } } while (keepAcceptingRequests.get()); Log.debug("Shutting down."); // No more connections accepted. If any client managed to connect after // the accept() was interrupted but before the server socket is closed // here, any attempt to read or write to the socket will result in an // IOException on the client side. // Shut down idleMonitor.shutdown(); compilerThreadPool.shutdown(); return true; } private void handleRequest(Socket socket) { try (BufferedReader in = new BufferedReader(new InputStreamReader(socket.getInputStream())); PrintWriter out = new PrintWriter(socket.getOutputStream(), true)) { try { idleMonitor.startCall(); // Set up logging for this thread. Stream back logging messages to // client on the format "level:msg". Log.setLogForCurrentThread(new Protocol.ProtocolLog(out)); String[] args = Protocol.readCommand(in); // If there has been any internal errors, notify client checkInternalErrorLog(); // Perform compilation. This will call runCompiler() on a // thread in the thread pool int exitCode = compilerThreadPool.dispatchCompilation(args); Protocol.sendExitCode(out, exitCode); // Check for internal errors again. checkInternalErrorLog(); } finally { idleMonitor.endCall(); } } catch (Exception ex) { // Not much to be done at this point. The client side request // code will most likely throw an IOException and the // compilation will fail. Log.error(ex); } finally { Log.setLogForCurrentThread(null); } } public static int runCompiler(Log log, String[] args) { Log.setLogForCurrentThread(log); // Direct logging to our byte array stream. StringWriter strWriter = new StringWriter(); PrintWriter printWriter = new PrintWriter(strWriter); // Compile Optional tool = ToolProvider.findFirst("javac"); if (tool.isEmpty()) { Log.error("Can't find tool javac"); return Result.ERROR.exitCode; } int exitcode = tool.get().run(printWriter, printWriter, args); // Process compiler output (which is always errors) printWriter.flush(); Util.getLines(strWriter.toString()).forEach(Log::error); return exitcode; } private void checkInternalErrorLog() { Path errorLogPath = errorLog.getLogDestination(); if (errorLogPath != null) { Log.error("Server has encountered an internal error. See " + errorLogPath.toAbsolutePath() + " for details."); } } public static void restoreServerErrorLog() { Log.setLogForCurrentThread(errorLog); } public void shutdownServer(String quitMsg) { if (!keepAcceptingRequests.compareAndSet(true, false)) { // Already stopped, no need to shut down again return; } Log.debug("Quitting: " + quitMsg); portFileMonitor.shutdown(); // No longer any need to monitor port file // Unpublish port before shutting down socket to minimize the number of // failed connection attempts try { portFile.delete(); } catch (IOException | InterruptedException e) { Log.error(e); } try { serverSocket.close(); } catch (IOException e) { Log.error(e); } } }