build.xml contains <scp>
and <sshexec>
tasks, so I provide jsch.jar and
other libraries in the same directory together with build.xml.
The following taskdef:
<taskdef name="scp"
classname="org.apache.tools.ant.taskdefs.optional.ssh.Scp"
classpath="WebContent/WEB-INF/lib/jsch-0.1.43.jar" />
throws an error
A class needed by class org.apache.tools.ant.taskdefs.optional.ssh.Scp
cannot be found: com/jcraft/jsch/UserInfo
I cannot modify the standard Ant installation (e.g. put jsch.jar in ant lib directory, or remove ant-js开发者_C百科ch.jar), or add command-line flags, or modify system environment variables, etc.: the script has to run with default Ant on different systems.
I'm actually reposting the question originally asked here: http://ant.1045680.n5.nabble.com/specifying-location-of-an-external-library-within-build-xml-td1344969.html
but could not get the answer about classloader to work.
Finally I found a working solution (for Ant 1.7.1 at least). First you have to remove ant-jsch.jar
from ANT_HOME/lib
as Ant complains about it and gets confused. Then load libraries from the project itself:
<available property="ant-jsch.present" file="${ant.home}/lib/ant-jsch.jar"/>
<fail if="ant-jsch.present" message="Please remove ant-jsch.jar from ANT_HOME/lib see [http://ant.apache.org/faq.html#delegating-classloader]"/>
<path id="jsch.path">
<pathelement location="lib/ant-jsch.jar" />
<pathelement location="lib/jsch-0.1.44.jar" />
</path>
<taskdef name="scp" classname="org.apache.tools.ant.taskdefs.optional.ssh.Scp" classpathref="jsch.path" />
<taskdef name="sshexec" classname="org.apache.tools.ant.taskdefs.optional.ssh.SSHExec" classpathref="jsch.path" />
So, this question is old, but I devised another approach which may help others. We can spawn Ant from a <java>
task with the proper classpath to run <scp>
. This avoid the classpath leaking problem, and doesn't requires changing Ant install in any way:
<target name="sendfile">
<!-- file: local file to send -->
<!-- todir: remote directory -->
<java classname="org.apache.tools.ant.launch.Launcher"
fork="true" dir="${basedir}" taskname="ant+scp">
<classpath>
<pathelement location="/where/is/jsch-0.1.49.jar"/>
<pathelement location="${ant.home}/lib/ant-launcher.jar"/>
</classpath>
<arg value="-buildfile"/>
<arg file="${ant.file}"/>
<arg value="-Dfile=${file}"/>
<arg value="-Dtodir=${todir}"/>
<arg value="sendfile.scp"/>
</java>
</target>
<target name="sendfile.scp">
<echo message="Sending ${file} to ${todir}"/>
<property file="/tmp/passwordfile"/>
<scp file="${file}" todir="username@11.22.33.44:${todir}"
trust="true" port="22" password="${PASSWORD}"/>
</target>
The port
parameter isn't needed, but it's here as a reminder for custom SSH ports. The password is a property stored on /tmp/passwordfile
, like PASSWORD=mysecretpassword
. Change these to suit your needs. Here follows an usage example:
<ant target="sendfile">
<!-- Example: send /etc/os-release file to remote dir /home/myself -->
<property name="file" value="/etc/os-release"/>
<property name="todir" value="/home/myself"/>
</ant>
For reference, an approach that I find useful is to repackage the jars, so they don't conflict - you can do this in Ant using JarJar like this:
<taskdef name="jarjar" classname="com.tonicsystems.jarjar.JarJarTask" classpath="${basedir}/lib/build/jar/jarjar-1.4.jar"/>
<taskdef name="scp" classname="repackaged.scp.org.apache.tools.ant.taskdefs.optional.ssh.Scp" classpath="${basedir}/lib/build/jar/repackaged-scp.jar"/>
<target name="repackage.scp" description="Repackages Ant's optional SCP task and the JSch implementation to avoid conflicting with one on Ant's classpath">
<delete file="${basedir}/lib/build/jar/repackaged-scp.jar" failonerror="false"/>
<jarjar basedir="." jarfile="${basedir}/lib/build/jar/repackaged-scp.jar" includes="nothing">
<zipfileset src="${basedir}/lib/build/jar/ant-jsch-1.9.1.jar"/>
<zipfileset src="${basedir}/lib/build/jar/jsch-0.1.50.jar"/>
<rule pattern="com.jcraft.jsch.**" result="repackaged.scp.com.jcraft.jsch.@1"/>
<rule pattern="org.apache.tools.ant.taskdefs.optional.ssh.**" result="repackaged.scp.org.apache.tools.ant.taskdefs.optional.ssh.@1"/>
</jarjar>
</target>
I was able to solve this issue following post from here https://stackoverflow.com/a/858744/3499805 and then
<taskdef resource="net/jtools/classloadertask/antlib.xml" classpath="${basedir}/ant-lib/ant-classloadertask.jar" />
<classloader loader="system" classpath="${basedir}/ant-lib/jsch-0.1.54.jar"/>
Create a path reference and then use it in your task definition:
<path id="ssh.path">
<pathelement location="${lib1.dir}/helloworld.jar"/>
<fileset dir="${lib2.dir}">
<include name="*.jar"/>
</fileset>
</path>
<taskdef name="mytask" classname="org.mytask" classpathref="ssh.path" />
Create ~/.ant/lib
and copy jsch.jar
in there as part of the build initialisation.
<target name="init">
<property name="user.ant.lib" location="${user.home}/.ant/lib"/>
<mkdir dir="${user.ant.lib}"/>
<copy todir="${user.ant.lib}">
<fileset dir="${basedir}/build/tools" includes="jsch-*.jar"/>
</copy>
</target>
There's a well-known trick with URLClassLoader
. By using it we can make jsch
accessible to ant-jsch
.
I wonder how classloadertask
from the answer by @user3499805 works.
<target name="injectJsch" description="inject jsch jar">
<makeurl file="${acdc.java.tools}/lib/jsch-0.1.50.jar" property="jsch.jar.url"/>
<taskdef name="injectJsch"
classname="tools.deployments.ant.InjectJsch"
classpath="${basedir}/jars/ajwf_deploytools.jar"
/>
<injectJsch jarLocation="${jsch.jar.url}"/>
</target>
_
package tools.deployments.ant;
import java.lang.reflect.Method;
import java.net.URL;
import java.net.URLClassLoader;
import org.apache.tools.ant.BuildException;
import org.apache.tools.ant.Task;
import org.apache.tools.ant.taskdefs.optional.ssh.LogListener;
public class InjectJsch extends Task {
public void setJarLocation(final String jarLocation) {
this.jarLocation = jarLocation;
}
@Override
public void execute() throws BuildException {
try {
injectJsch(new URL(jarLocation));
} catch (final Exception e) {
throw new BuildException(e);
}
}
public static void injectJsch(final URL jarLocation) throws Exception {
ClassLoader parent = LogListener.class.getClassLoader();
try {
parent.loadClass(TESTCLASS);
} catch (final ClassNotFoundException e) {
final Method addURLmethod = URLClassLoader.class.getDeclaredMethod("addURL", URL.class);
addURLmethod.setAccessible(true);
ClassLoader cl;
do {
cl = parent;
if (cl instanceof URLClassLoader) {
addURLmethod.invoke(cl, jarLocation);
break;
}
parent = cl.getParent();
} while (parent != cl && parent != null);
LogListener.class.getClassLoader().loadClass(TESTCLASS);
}
}
private String jarLocation;
private static final String TESTCLASS = "com.jcraft.jsch.UserInfo";
}
精彩评论