/* Copyright 2009 Kindleit.net Software Development
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package net.kindleit.gae.runner;
import java.io.File;
import java.io.IOException;
import java.util.List;
import java.util.Properties;
import java.util.Set;
import org.apache.maven.artifact.Artifact;
import org.apache.maven.plugin.logging.Log;
import org.codehaus.plexus.util.StringUtils;
import org.codehaus.plexus.util.cli.CommandLineUtils;
import org.codehaus.plexus.util.cli.Commandline;
import org.codehaus.plexus.util.cli.StreamConsumer;
/**
* An implementation of {@code KickStartRunner} that asynchronously invokes {@link com.google.appengine.tools.KickStart}
* in a forked process.
*
* @author tmoore@incrementalism.net
* @since 0.5.8
*/
final class BackgroundKickStartRunner extends KickStartRunner {
private final String pluginPath;
private final Log log;
private Thread thread;
private volatile Exception thrown;
/**
* Creates a new {@code BackgroundKickStartRunner}.
*
* @param artifacts The Maven Project.
* @param gaeProperties Properties with the plugin's groupId, and artifactId.
* @param log The Maven plugin logger to direct output to.
* @throws KickStartExecutionException if the plugin cannot be found.
*/
public BackgroundKickStartRunner(final Set<Artifact> artifacts,
final Properties gaeProperties, final Log log)
throws KickStartExecutionException {
this.log = log;
pluginPath = getPluginPath(artifacts, gaeProperties);
}
/**
* Asynchronously starts a {@code KickStart} instance with the specified arguments.
* This method method will block until the server starts up, and then allows the current thread to continue while
* the server runs in the background.
*
* @param args the arguments to pass to {@code KickStart}
*/
@Override
public synchronized void start(final int monitorPort, final String monitorKey,
final List<String> args) throws KickStartExecutionException {
if (thread != null) {
throw new IllegalStateException("Already started");
}
thread = setupCommandLine(monitorPort, monitorKey, args);
thread.start();
try {
wait();
} catch (final InterruptedException e) {
thrown = e;
}
if (thrown != null) {
throw new KickStartExecutionException(thrown);
}
}
private Thread setupCommandLine(final int monitorPort,
final String monitorKey, final List<String> args) {
final String javaExe = System.getProperty("java.home") + File.separator
+ "bin" + File.separator + "java";
final Commandline commandline = new Commandline(StringUtils.quoteAndEscape(javaExe, '"'));
final String classPath =
System.getProperty("java.class.path") + File.pathSeparator + pluginPath;
commandline.createArg().setValue("-ea");
commandline.createArg().setValue("-cp");
commandline.createArg().setValue(classPath);
commandline.createArg().setValue("-Dmonitor.port=" + monitorPort);
commandline.createArg().setValue("-Dmonitor.key=" + monitorKey);
commandline.createArg().setValue("-Dappengine.sdk.root=" + System.getProperty("appengine.sdk.root"));
commandline.createArg().setValue(AppEnginePluginMonitor.class.getName());
commandline.addArguments(args.toArray(new String[args.size()]));
final StreamConsumer outConsumer = new StreamConsumer() {
public void consumeLine(final String line) {
consumeOutputLine(line);
}
};
final StreamConsumer errConsumer = new StreamConsumer() {
public void consumeLine(final String line) {
consumeErrorLine(line);
}
};
if (log.isDebugEnabled()) {
log.debug("Forking executable: " + commandline.getExecutable());
log.debug("Command line: " + commandline.toString());
}
return new Thread(new Runnable() {
public void run() {
try {
CommandLineUtils.executeCommandLine(commandline, outConsumer, errConsumer);
} catch (final Exception e) {
setThrown(e);
}
}
});
}
private synchronized void consumeOutputLine(final String line) {
System.out.println(line);
if (line.contains("is running")) {
notify();
}
}
private synchronized void consumeErrorLine(final String line) {
System.err.println(line);
if (line.contains("is running")) {
notify();
}
}
private synchronized void setThrown(final Exception thrown) {
this.thrown = thrown;
notify();
}
private String getPluginPath(final Set<Artifact> artifacts,
final Properties gaeProperties) throws KickStartExecutionException {
final String groupId = gaeProperties.getProperty("plugin.groupId");
final String artifactId = gaeProperties.getProperty("plugin.artifactId");
for (final Artifact a : artifacts) {
if (groupId.equals(a.getGroupId())
&& artifactId.equals(a.getArtifactId())) {
try {
return a.getFile().getCanonicalPath();
} catch (final IOException e) {
log.error(e);
throw new KickStartExecutionException(e);
}
}
}
throw new KickStartExecutionException("Plugin not found");
}
}