Package org.jnode.ant.taskdefs

Source Code of org.jnode.ant.taskdefs.Hotswap

/*
* $Id$
*
* Copyright (C) 2003-2014 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;
    }
}
TOP

Related Classes of org.jnode.ant.taskdefs.Hotswap

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.