2010-09-16 1 views
8

build.xml enthält <scp> und <sshexec> Aufgaben, so biete ich jsch.jar und andere Bibliotheken im selben Verzeichnis zusammen mit build.xml.Gibt es eine Möglichkeit, den Speicherort eines lokalen jsch.jar aus build.xml anzugeben?

Die folgende taskdef:

<taskdef name="scp" 
    classname="org.apache.tools.ant.taskdefs.optional.ssh.Scp" 
    classpath="WebContent/WEB-INF/lib/jsch-0.1.43.jar" /> 

wirft einen Fehler

A class needed by class org.apache.tools.ant.taskdefs.optional.ssh.Scp 
cannot be found: com/jcraft/jsch/UserInfo 

ich die Standard-Ant-Installation nicht ändern können (zB setzen jsch.jar in ant lib Verzeichnis oder entfernen Ameisen jsch. jar), oder fügen Sie Befehlszeilenflags hinzu oder ändern Sie Systemumgebungsvariablen etc .: Das Skript muss mit dem Standard Ant auf verschiedenen Systemen ausgeführt werden.

Ich reposting tatsächlich die Frage ursprünglich hier gefragt: http://ant.1045680.n5.nabble.com/specifying-location-of-an-external-library-within-build-xml-td1344969.html

konnte aber die Antwort über Classloader nicht zur Arbeit bekommen.

Antwort

17

Schließlich fand ich eine Arbeitslösung (für Ant 1.7.1 zumindest). Zuerst müssen Sie ant-jsch.jar von ANT_HOME/lib entfernen, da Ant sich darüber beschwert und verwirrt wird. Dann laden Bibliotheken aus dem Projekt selbst:

+1

Es scheint, dies ist ein häufiges Problem und ich denke, das ist die beste Lösung, aber anstatt zu scheitern, ich werde einfach weitermachen und das/lib/ant löschen -jsch.jar. Falls es anderen hilft, erklärt Paulo das Kernproblem in seiner SO [Antwort auf eine verwandte SCP-Frage] (http://stackoverflow.com/questions/5796587/problems-with-ant-optional-tasks-sshexec- and-scp-classpath-issue) und [das gleiche Problem tritt bei junit auf und wird hier erklärt] (http://ant.apache.org/faq.html#delegating-classloader) – gMale

+1

Dies erfordert jedoch eine Änderung der Standard-Ant-Installation . –

+0

Ich habe mich seit Tagen mit diesem Problem herumgeschlagen und diese Lösung ist die einzige, mit der ich arbeiten konnte! –

0

einen Pfadverweis erstellen und es dann in der Task-Definition verwenden:

<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" /> 
3

Also, diese Frage ist alt, aber ich habe einen anderen Ansatz entwickelt, die anderen helfen kann. Wir können Ant von einer <java> Aufgabe mit dem richtigen Klassenpfad erzeugen, um <scp> auszuführen. Dies vermeidet die Classpath Problem undicht, und nicht erfordert Ändern Ant in irgendeiner Weise installieren:

<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="[email protected]:${todir}" 
     trust="true" port="22" password="${PASSWORD}"/> 
</target> 

Der port Parameter ist nicht erforderlich, aber es ist hier als Erinnerung für benutzerdefinierte SSH-Ports. Das Passwort ist eine Eigenschaft, die auf /tmp/passwordfile, wie PASSWORD=mysecretpassword gespeichert ist. Ändern Sie diese nach Ihren Bedürfnissen. Hier folgt ein Anwendungsbeispiel:

<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> 
2

Als Referenz ein Ansatz, der finde ich nützlich ist, die Gläser neu zu verpacken, so dass sie nicht in Konflikt stehen - Sie können dies tun, in Ant mit JarJar wie folgt aus:

<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="[email protected]"/> 
     <rule pattern="org.apache.tools.ant.taskdefs.optional.ssh.**" result="[email protected]"/> 
    </jarjar> 
</target> 
0

Erstellen Sie ~/.ant/lib und kopieren Sie jsch.jar dort als Teil der Build-Initialisierung.

<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> 
1

konnte ich von hier https://stackoverflow.com/a/858744/3499805 dieses Problem folgende Beitrag lösen und dann

<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"/> 
0

Es gibt eine bekannte trick mit URLClassLoader. Indem wir es benutzen, können wir jsch zugänglich zu ant-jsch machen.

Ich frage mich, wie classloadertask aus der Antwort von @ user3499805 funktioniert.

<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"; 
}