Change jetty:run-forked to use presence of file as signal child process has started.
This commit is contained in:
Jan Bartel 2017-09-21 08:57:17 +10:00
parent 82bfdf74f0
commit c124803e6d
3 changed files with 78 additions and 114 deletions

View File

@ -109,14 +109,14 @@ public class JettyRunDistro extends JettyRunMojo
/** /**
* * Optional jetty.home dir
* @parameter * @parameter
*/ */
private File jettyHome; private File jettyHome;
/** /**
* * Optional jetty.base dir
* @parameter * @parameter
*/ */
private File jettyBase; private File jettyBase;

View File

@ -24,6 +24,8 @@ import java.io.IOException;
import java.io.InputStream; import java.io.InputStream;
import java.io.InputStreamReader; import java.io.InputStreamReader;
import java.io.LineNumberReader; import java.io.LineNumberReader;
import java.nio.file.Files;
import java.nio.file.Path;
import java.util.ArrayList; import java.util.ArrayList;
import java.util.Arrays; import java.util.Arrays;
import java.util.HashMap; import java.util.HashMap;
@ -36,6 +38,7 @@ import java.util.Random;
import java.util.Set; import java.util.Set;
import org.apache.maven.artifact.Artifact; import org.apache.maven.artifact.Artifact;
import org.apache.maven.model.Dependency;
import org.apache.maven.plugin.MojoExecutionException; import org.apache.maven.plugin.MojoExecutionException;
import org.apache.maven.plugin.MojoFailureException; import org.apache.maven.plugin.MojoFailureException;
import org.apache.maven.plugin.descriptor.PluginDescriptor; import org.apache.maven.plugin.descriptor.PluginDescriptor;
@ -120,11 +123,22 @@ public class JettyRunForkedMojo extends JettyRunMojo
*/ */
private boolean waitForChild; private boolean waitForChild;
/**
* @parameter default-value="50"
*/
private int maxStartupLines;
/**
* Max number of times to try checking if the
* child has started successfully.
*
* @parameter alias="maxStartupLines" default-value="50"
*/
private int maxChildChecks;
/**
* Millisecs to wait between each
* check to see if the child started successfully.
*
* @parameter default-value="100"
*/
private long maxChildCheckInterval;
/** /**
* Extra environment variables to be passed to the forked process * Extra environment variables to be passed to the forked process
@ -145,9 +159,15 @@ public class JettyRunForkedMojo extends JettyRunMojo
private Random random; private Random random;
private boolean originalPersistTemp; /**
* Whether or not the plugin has explicit slf4j dependencies.
* The maven environment will always have slf4j on the classpath,
* which we don't want to put onto the forked process unless the
* pom has an explicit dependency on it.
*/
private boolean hasSlf4jDeps;
/** /**
* ShutdownThread * ShutdownThread
* *
@ -168,54 +188,6 @@ public class JettyRunForkedMojo extends JettyRunMojo
} }
} }
} }
/**
* we o
*/
// protected MavenProject getProjectReferences( Artifact artifact, MavenProject project )
// {
//
// return null;
// }
/**
* ConsoleStreamer
*
* Simple streamer for the console output from a Process
*/
private static class ConsoleStreamer implements Runnable
{
private String mode;
private BufferedReader reader;
public ConsoleStreamer(String mode, InputStream is)
{
this.mode = mode;
this.reader = new BufferedReader(new InputStreamReader(is));
}
public void run()
{
String line;
try
{
while ((line = reader.readLine()) != (null))
{
System.out.println("[" + mode + "] " + line);
}
}
catch (IOException ignore)
{
/* ignore */
}
finally
{
IO.close(reader);
}
}
}
@ -228,6 +200,17 @@ public class JettyRunForkedMojo extends JettyRunMojo
{ {
Runtime.getRuntime().addShutdownHook(new ShutdownThread()); Runtime.getRuntime().addShutdownHook(new ShutdownThread());
random = new Random(); random = new Random();
List<Dependency> deps = plugin.getPlugin().getDependencies();
for (Dependency d:deps)
{
if (d.getGroupId().contains("slf4j"))
{
hasSlf4jDeps = true;
break;
}
}
super.execute(); super.execute();
} }
@ -256,9 +239,6 @@ public class JettyRunForkedMojo extends JettyRunMojo
//ensure config of the webapp based on settings in plugin //ensure config of the webapp based on settings in plugin
configureWebApplication(); configureWebApplication();
//get the original persistance setting
originalPersistTemp = webApp.isPersistTempDirectory();
//set the webapp up to do very little other than generate the quickstart-web.xml //set the webapp up to do very little other than generate the quickstart-web.xml
webApp.setCopyWebDir(false); webApp.setCopyWebDir(false);
webApp.setCopyWebInf(false); webApp.setCopyWebInf(false);
@ -341,8 +321,9 @@ public class JettyRunForkedMojo extends JettyRunMojo
cmd.add(props.getAbsolutePath()); cmd.add(props.getAbsolutePath());
String token = createToken(); String token = createToken();
Path tokenFile = target.toPath().resolve(createToken()+".txt");
cmd.add("--token"); cmd.add("--token");
cmd.add(token); cmd.add(tokenFile.toAbsolutePath().toString());
if (jettyProperties != null) if (jettyProperties != null)
{ {
@ -368,49 +349,32 @@ public class JettyRunForkedMojo extends JettyRunMojo
if (waitForChild) if (waitForChild)
{ {
forkedProcess = builder.start(); builder.inheritIO();
startPump("STDOUT",forkedProcess.getInputStream()); }
startPump("STDERR",forkedProcess.getErrorStream()); else
{
builder.redirectOutput(new File(target, "jetty.out"));
builder.redirectErrorStream(true);
}
forkedProcess = builder.start();
if (waitForChild)
{
int exitcode = forkedProcess.waitFor(); int exitcode = forkedProcess.waitFor();
PluginLog.getLog().info("Forked execution exit: "+exitcode); PluginLog.getLog().info("Forked execution exit: "+exitcode);
} }
else else
{ //merge stderr and stdout from child {
builder.redirectErrorStream(true); //just wait until the child has started successfully
forkedProcess = builder.start(); int attempts = maxChildChecks;
while (!Files.exists(tokenFile) && attempts > 0)
//wait for the child to be ready before terminating.
//child indicates it has finished starting by printing on stdout the token passed to it
try
{ {
String line = ""; Thread.currentThread().sleep(maxChildCheckInterval);
try (InputStream is = forkedProcess.getInputStream(); --attempts;
LineNumberReader reader = new LineNumberReader(new InputStreamReader(is)))
{
int attempts = maxStartupLines; //max lines we'll read trying to get token
while (attempts>0 && line != null)
{
--attempts;
line = reader.readLine();
if (line != null && line.startsWith(token))
break;
}
}
if (line != null && line.trim().equals(token))
PluginLog.getLog().info("Forked process started.");
else
{
String err = (line == null?"":(line.startsWith(token)?line.substring(token.length()):line));
PluginLog.getLog().info("Forked process startup errors"+(!"".equals(err)?", received: "+err:""));
}
} }
catch (Exception e) if (attempts <=0 )
{ getLog().info("Couldn't verify success of child startup");
throw new MojoExecutionException ("Problem determining if forked process is ready: "+e.getMessage());
}
} }
} }
catch (InterruptedException ex) catch (InterruptedException ex)
@ -522,6 +486,9 @@ public class JettyRunForkedMojo extends JettyRunMojo
Artifact artifact = (Artifact) obj; Artifact artifact = (Artifact) obj;
if ("jar".equals(artifact.getType())) if ("jar".equals(artifact.getType()))
{ {
//ignore slf4j from inside maven
if (artifact.getGroupId().contains("slf4j") && !hasSlf4jDeps)
continue;
if (classPath.length() > 0) if (classPath.length() > 0)
{ {
classPath.append(File.pathSeparator); classPath.append(File.pathSeparator);
@ -530,6 +497,7 @@ public class JettyRunForkedMojo extends JettyRunMojo
} }
} }
//Any jars that we need from the plugin environment (like the ones containing Starter class) //Any jars that we need from the plugin environment (like the ones containing Starter class)
Set<Artifact> extraJars = getExtraJars(); Set<Artifact> extraJars = getExtraJars();
@ -617,12 +585,4 @@ public class JettyRunForkedMojo extends JettyRunMojo
{ {
return Long.toString(random.nextLong()^System.currentTimeMillis(), 36).toUpperCase(Locale.ENGLISH); return Long.toString(random.nextLong()^System.currentTimeMillis(), 36).toUpperCase(Locale.ENGLISH);
} }
private void startPump(String mode, InputStream inputStream)
{
ConsoleStreamer pump = new ConsoleStreamer(mode,inputStream);
Thread thread = new Thread(pump,"ConsoleStreamer/" + mode);
thread.setDaemon(true);
thread.start();
}
} }

View File

@ -175,16 +175,21 @@ public class Starter
{ {
server.join(); server.join();
} }
public void communicateStartupResult (Exception e) public void communicateStartupResult ()
{ {
if (token != null) if (token != null)
{ {
if (e==null) try
System.out.println(token); {
else Resource r = Resource.newResource(token);
System.out.println(token+"\t"+e.getMessage()); r.getFile().createNewFile();
}
catch (Exception x)
{
throw new IllegalStateException (x);
}
} }
} }
@ -250,12 +255,11 @@ public class Starter
starter.getConfiguration(args); starter.getConfiguration(args);
starter.configureJetty(); starter.configureJetty();
starter.run(); starter.run();
starter.communicateStartupResult(null); starter.communicateStartupResult();
starter.join(); starter.join();
} }
catch (Exception e) catch (Exception e)
{ {
starter.communicateStartupResult(e);
e.printStackTrace(); e.printStackTrace();
System.exit(1); System.exit(1);
} }