/*
* $Id$
*
* Copyright (C) 2003-2015 JNode.org
*
* This library is free software; you can redistribute it and/or modify it
* under the terms of the GNU Lesser General Public License as published
* by the Free Software Foundation; either version 2.1 of the License, or
* (at your option) any later version.
*
* This library 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 Lesser General Public
* License for more details.
*
* You should have received a copy of the GNU Lesser General Public License
* along with this library; If not, write to the Free Software Foundation, Inc.,
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
*/
package org.jnode.ant.taskdefs;
import com.sun.jdi.Bootstrap;
import com.sun.jdi.ReferenceType;
import com.sun.jdi.VirtualMachine;
import com.sun.jdi.VirtualMachineManager;
import com.sun.jdi.connect.AttachingConnector;
import com.sun.jdi.connect.Connector;
import com.sun.jdi.connect.Connector.Argument;
import java.io.DataInputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.IOException;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Vector;
import org.apache.tools.ant.BuildException;
import org.apache.tools.ant.DirectoryScanner;
import org.apache.tools.ant.taskdefs.MatchingTask;
import org.apache.tools.ant.types.FileSet;
/*
This code is based on http://hotswap.dev.java.net/.
Adapted to JNode by Levente S\u00e1ntha.
*/
/**
* This task replaces class on a running JVM. This task can take the following
* arguments:
* <ul>
* <li/>verbose
* <li/>failonerror
* <li/>host
* <li/>port
* <li/>name
* </ul>
* Of these arguments, the <b>host</b> and <b>port</b> are required. Or,
* the <b>name</b> can be used instead to indicate a shared mem connection.
* <p/>
* See the JPDA documentation for details on the JVM runtime options.
* <a href="http://java.sun.com/j2se/1.4.2/docs/guide/jpda/conninv.html#Invocation">
* http://java.sun.com/j2se/1.4.2/docs/guide/jpda/conninv.html#Invocation</a>
* <p/>
* Add this line to your build.xml<br/>
* <code>
* <taskdef name="hotswap" classname="org.apache.tools.ant.taskdefs.Hotswap"/>
* </code>
* <p/>
* This is an example of how to hotswap with a JVM on port 9000 on your local machine
* <br/>
* <code>
* <!-- note, replace the <star> tags below with "*". This kept the example from breaking the javadoc -->
* <hotswap verbose="true" port="9000">
* <!-- This line matches 3 classes in the ant build/classes directory -->
* <fileset dir="build/classes" includes="<star><star>/Hot*.class"/>
* <!-- This line matches all classes in the taskefs package (and below) -->
* <fileset dir="build/classes" includes="<star><star>/taskdefs"/>
* </code>
* <br/>
* The preferred way to build the <fileset> would be based on modification time.
* At present, the tstamp isn't fine grained enough. The <outofdate> task from ant-contrib
* provides absolute paths to all of the class files, which isn't compatible with the
* way <hotswap> needs the paths.
*
* @author David A. Kavanagh <a href="mailto:dak@dotech.com">dak@dotech.com</a>
*/
public class Hotswap extends MatchingTask {
private static final String FAIL_MSG
= "Hotswap failed; changes to class(es) might not be compatible with replacement on your VM.";
private boolean verbose = false;
private boolean failonerror = true;
protected String host;
protected String port;
protected String name;
protected Vector<FileSet> filesets = new Vector<FileSet>();
private boolean useSocket = true;
/**
* Hotswap task for compilation of Java files.
*/
public Hotswap() {
}
/**
* If true, asks the compiler for verbose output.
*
* @param verbose if true, asks the compiler for verbose output
*/
public void setVerbose(boolean verbose) {
this.verbose = verbose;
}
/**
* Gets the verbose flag.
*
* @return the verbose flag
*/
public boolean getVerbose() {
return verbose;
}
/**
* Gets the name of the host with the running VM.
*
* @return the hotswap host name
*/
public String getHost() {
return host;
}
/**
* Sets the name of the host with the running VM.
*
* @param host the host to be used when connecting to a running VM
*/
public void setHost(String host) {
this.host = host;
}
/**
* Gets the socket address of the host with the running VM.
*
* @return the hotswap socket address
*/
public String getPort() {
return port;
}
/**
* Sets the socket address of the host with the running VM.
*
* @param port the socket address to be used when connecting to a running VM
*/
public void setPort(String port) {
this.port = port;
}
/**
* Gets the shared mem name to use when connecting to the running VM.
*
* @return the hotswap socket address
*/
public String getName() {
return name;
}
/**
* Sets the shared mem name to use when connecting to the running VM.
*
* @param name the shared memory name
*/
public void setName(String name) {
this.name = name;
}
/**
* If false, note errors but continue.
*
* @param failonerror true or false
*/
public void setFailOnError(boolean failonerror) {
this.failonerror = failonerror;
}
/**
* Adds a set of files to be deployed.
*
* @param set the set of files to be deployed
*/
public void addFileset(FileSet set) {
filesets.addElement(set);
}
/**
* Executes the task.
*
* @throws BuildException if an error occurs
*/
public void execute() throws BuildException {
checkParameters();
try {
HotSwapHelper hsh = new HotSwapHelper();
// attach
if (useSocket) {
hsh.connect(host, port);
} else {
hsh.connect(name);
}
// load classes and replace them on target VM
for (int i = 0; i < filesets.size(); i++) {
FileSet fs = filesets.elementAt(i);
try {
DirectoryScanner ds = fs.getDirectoryScanner(getProject());
String[] files = ds.getIncludedFiles();
String[] dirs = ds.getIncludedDirectories();
hotswapFiles(hsh, fs.getDir(getProject()), files, dirs);
} catch (BuildException be) {
// directory doesn't exist or is not readable
if (failonerror) {
throw be;
} else {
log(FAIL_MSG);
log(be.getMessage());
}
}
}
hsh.disconnect();
} catch (Exception ex) {
if (failonerror) {
throw new BuildException(ex);
} else {
log(FAIL_MSG);
log(ex.getMessage());
}
}
}
/**
* Check that all required attributes have been set and nothing
* silly has been entered.
*
* @throws BuildException if an error occurs
* @since Ant 1.5
*/
protected void checkParameters() throws BuildException {
if (filesets.size() == 0) {
throw new BuildException("At least one of the file or dir "
+ "attributes, or a fileset element, "
+ "must be set.");
}
if ((port == null) && (name == null)) {
throw new BuildException("port is null or name is null");
}
if (port != null)
useSocket = true;
else
useSocket = false;
}
/**
* remove an array of files in a directory, and a list of subdirectories
* which will only be deleted if 'includeEmpty' is true.
*
* @param hsh the hotswap helper class
* @param d directory to work from
* @param files array of files to delete; can be of zero length
* @param dirs array of directories to delete; can of zero length
*/
protected void hotswapFiles(HotSwapHelper hsh, File d, String[] files, String[] dirs) throws Exception {
if (files.length > 0) {
log("hotswapping " + files.length + " files from "
+ d.getAbsolutePath());
for (int j = 0; j < files.length; j++) {
processHotswap(hsh, d, files[j]);
}
}
if (dirs.length > 0) {
int dirCount = 0;
for (int j = dirs.length - 1; j >= 0; j--) {
log("swapping dir " + d.getAbsolutePath() + ", " + dirs[j]);
processDirectory(hsh, d, dirs[j]);
// dirCount++;
}
// TODO: need an accurate count?
if (dirCount > 0) {
log("Hotswapped " + dirCount + " director"
+ (dirCount == 1 ? "y" : "ies")
+ " from " + d.getAbsolutePath());
}
}
}
private void processDirectory(HotSwapHelper hsh, File d, String subdir) throws Exception {
File[] files = new File(d, subdir).listFiles();
for (int i = 0; i < files.length; i++) {
if (files[i].isDirectory()) {
processDirectory(hsh, d, getClassOrPackage(d, files[i]));
} else {
processHotswap(hsh, d, getClassOrPackage(d, files[i]));
}
}
}
private String getClassOrPackage(File baseDir, File fileOrDir) {
return fileOrDir.getAbsolutePath().substring(baseDir.getAbsolutePath().length() + 1);
}
private void processHotswap(HotSwapHelper hsh, File d, String file) throws Exception {
File f = new File(d, file);
String className = file;
className = className.substring(0, className.length() - 6); // chop off ".class"
className = className.replace('/', '.');
className = className.replace('\\', '.');
if (verbose)
log("hotswapping " + className);
hsh.replace(f, className);
}
}
class HotSwapHelper {
private VirtualMachine vm;
public HotSwapHelper() {
}
public void connect(String name) throws Exception {
connect(null, null, name);
}
public void connect(String host, String port) throws Exception {
connect(host, port, null);
}
// either host,port will be set, or name
private void connect(String host, String port, String name) throws Exception {
// connect to JVM
boolean useSocket = (port != null);
VirtualMachineManager manager = Bootstrap.virtualMachineManager();
List<AttachingConnector> connectors = manager.attachingConnectors();
AttachingConnector connector = null;
// System.err.println("Connectors available");
for (int i = 0; i < connectors.size(); i++) {
AttachingConnector tmp = connectors.get(i);
// System.err.println("conn "+i+" name="+tmp.name()+" transport="+tmp.transport().name()+
// " description="+tmp.description());
if (!useSocket && tmp.transport().name().equals("dt_shmem")) {
connector = tmp;
break;
}
if (useSocket && tmp.transport().name().equals("dt_socket")) {
connector = tmp;
break;
}
}
if (connector == null) {
throw new IllegalStateException("Cannot find shared memory connector");
}
Map<String, Argument> args = connector.defaultArguments();
// Iterator iter = args.keySet().iterator();
// while (iter.hasNext()) {
// Object key = iter.next();
// Object val = args.get(key);
// System.err.println("key:"+key.toString()+" = "+val.toString());
// }
Connector.Argument arg;
// use name if using dt_shmem
if (!useSocket) {
arg = (Connector.Argument) args.get("name");
arg.setValue(name);
} else {
// use port if using dt_socket
arg = (Connector.Argument) args.get("port");
arg.setValue(port);
if (host != null) {
arg = (Connector.Argument) args.get("hostname");
arg.setValue(host);
}
}
vm = connector.attach(args);
// query capabilities
if (!vm.canRedefineClasses()) {
throw new Exception("JVM doesn't support class replacement");
}
// if (!vm.canAddMethod()) {
// throw new Exception("JVM doesn't support adding method");
// }
// System.err.println("attached!");
}
public void replace(File classFile, String className) throws Exception {
// load class(es)
byte[] classBytes = loadClassFile(classFile);
// redefine in JVM
List<ReferenceType> classes = vm.classesByName(className);
// if the class isn't loaded on the VM, can't do the replace.
if (classes == null || classes.size() == 0)
return;
// for now, just grab the first ref.
ReferenceType refType = classes.get(0);
HashMap<ReferenceType, byte[]> map = new HashMap<ReferenceType, byte[]>();
map.put(refType, classBytes);
vm.redefineClasses(map);
// System.err.println("class replaced!");
}
public void disconnect() throws Exception {
vm.dispose();
}
private byte[] loadClassFile(File classFile) throws IOException {
DataInputStream in = new DataInputStream(new FileInputStream(classFile));
byte[] ret = new byte[(int) classFile.length()];
in.readFully(ret);
in.close();
// System.err.println("class file loaded.");
return ret;
}
}