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

net.kindleit.gae.runner.BackgroundKickStartRunner Maven / Gradle / Ivy

There is a newer version: 0.9.6
Show newest version
/* 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 [email protected]
 * @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 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 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 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("The server is running")) {
      notify();
    }
  }

  private synchronized void consumeErrorLine(final String line) {
    System.err.println(line);
    if (line.contains("The server is running")) {
      notify();
    }
  }

  private synchronized void setThrown(final Exception thrown) {
    this.thrown = thrown;
    notify();
  }

  private String getPluginPath(final Set 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");
  }

}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy