web-dev-qa-db-ja.com

Antは外部で定義されたtaskdefが必要とするクラスを見つけることができません

Axis-Java2wsdl Antタスクを使用して、Javaクラスの1つからwsdlを作成しようとしていますが、クラスパスを正しく取得できません。

私は$ ANT_HOME/libにaxis-ant.jarを、/ usr/share/Javaにaxis.jarをインストールするUbuntuのlibaxis-Javaパッケージを使用しています。私のbuild.xmlの興味深い部分は次のようになります。

<property name="library.dir" value="lib"/>
<property name="system.library.dir" value="/usr/share/Java"/>
<path id="libraries">
    <fileset dir="${library.dir}">
        <include name="*.jar"/>
    </fileset>
    <fileset dir="${system.library.dir}">
        <include name="*.jar"/>
    </fileset>
</path>

<target name="genwsdl" depends="compile">
    <taskdef resource="axis-tasks.properties" classpathref="libraries"/>
    <axis-Java2wsdl>
            details omitted
    </axis-Java2wsdl>
</target>

ランニング ant genwsdl結果:

/build.xml:50: taskdef A class needed by class
org.Apache.axis.tools.ant.wsdl.Wsdl2javaAntTask
cannot be found: org/Apache/axis/utils/DefaultAuthenticator

Antはaxis-Java2wsdlタスクの定義を見つけることができます。これは、axis-ant.jarが$ ANT_HOME/libにあるためです。ただし、jarが「ライブラリ」によって定義されたパス上にあっても、axis.jarでクラスを見つけることができません。

Axis.jarを$ ANT_HOME/libにシンボリックリンクすることで、DefaultAuthenticatorを通過して他のクラスが見つからないため、クラスパスの問題であることはわかっています。すべてを$ ANT_HOME/libにシンボリックリンクせずに、/ usr/share/libまたはプロジェクトのローカルlibディレクトリにあるjarファイルをtaskdefに認識させるにはどうすればよいですか?

編集:

私は最終的に次の行でwsdlを正常に生成することができました:

ant -lib /usr/share/Java/axis.jar -lib /usr/share/Java/jaxrpc.jar -lib /usr/share/Java/wsdl4j.jar -lib /usr/share/Java/commons-logging.jar -lib /usr/share/Java/commons-discovery.jar -lib build genwsdl

Build.xmlでこれらのライブラリを定義できないことで私が間違っていることを誰かが教えてもらえれば幸いです

15
Ryan Ahearn

一般的に、これは機能します。ただし、どのクラスがどこにあるかを注意深く確認する必要があります。

タスクローダーをクラスローダー階層の上位のクラスローダーにロードできる場合(CLASSPATHやANT_HOME/libなど)、classpathrefは単に無視されます

詳細については、 FAQエントリ を参照してください。

Antのクラスローダー実装は Javaの委任モデルを使用します

ClassLoaderクラスは、委任モデルを使用してクラスとリソースを検索します。 ClassLoaderの各インスタンスには、関連付けられた親クラスローダーがあります。クラスまたはリソースを見つけるために呼び出されると、ClassLoaderインスタンスは、クラスまたはリソース自体を見つける前に、クラスまたはリソースの検索を親クラスローダーに委任します。 bootstrapクラスローダーと呼ばれる仮想マシンの組み込みクラスローダーは、それ自体には親がありませんが、ClassLoaderインスタンスの親として機能する場合があります。

注:実行中のant -diagnosticsも役立ちます

16
VonC

Ant mechanism はライブラリを追加するためのものです:

  • コマンドライン引数-lib経由
  • $ {user.home} /。ant/libに追加
  • $ {ant.home}/libに追加

のみ。 manual は、system.library.dirプロパティの使用については何も言及していません。おそらく、この目的ではほとんど無視されます。

また、詳細モード(および-verbose)でantを実行して、内部で何が行われているかを確認します。

5
OscarRyz

ここからの回答とすべての部分的な情報を使用して、解決策を考え出しました。プロジェクトのlibフォルダー(特にmysql jdbcドライバー)に必要なantファイルをjarに追加しました。次に、antでセットアップタスクを実行し、ユーザーのホームの.ant/libフォルダーにコピーします。その後、antが失敗し、再起動するメッセージが表示されます。ユーザーに対して1回だけ失敗し、その後は常に機能します。 jarを新しいバージョンに更新するのは難しいかもしれません...

Ant build.xmlは次のとおりです。

<project name="data_source" default="build">

  <!-- BUILD PROPERTIES -->
  <property file="build.properties"/>

  <!-- SQL Server Properties -->
  <property name="sql.driver" value="org.gjt.mm.mysql.Driver"/>
  <property name="sql.url" value="jdbc:mysql://127.0.0.1/datastore"/>
  <property name="sql.user" value="user"/>
  <property name="sql.pass" value="password"/>

  <!-- FILE LOCATIONS  -->
  <property name="sql.dir" location="sql"/>

  <!-- INITIALIZE PROJECT -->
  <target name="init" description="Initialize the project">
      <available property="no.ant.lib.dir" file="${user.home}/.ant/lib/" type="dir" />
  </target>

  <!-- SETUP MYSQL -->
  <target name="setup_mysql" description="Copy the lib so ant can see it" unless="no.ant.lib.dir">
      <mkdir dir="${user.home}/.ant"/>
      <mkdir dir="${user.home}/.ant/lib"/>
      <copy file="lib/mysql-connector-Java-5.1.13-bin.jar" todir="${user.home}/.ant/lib"/>

      <!-- ant already missed picking up the jar - we have to restart -->
      <fail message="JDBC mysql class copied to ${user.dir}/.ant/lib/ - please restart ant" />
  </target>

  <!-- BUILD DATA SOURCES -->
  <target name="build" depends="init,setup_mysql,clean_data" description="Create and populate tables">

    <sql driver="${sql.driver}" url="${sql.url}" userid="${sql.user}" password="${sql.pass}" >
      <transaction src="${sql.dir}/create_tables.sql"/>
      <transaction src="${sql.dir}/insert_data.sql"/>
    </sql>
  </target>

   <!-- CLEAN PROJECT -->
  <target name="clean" description="Cleans up project build">
    <!-- Don't clean data sources here - may get called by accident -->
  </target>

  <!-- Delete all tables and data -->
  <target name="clean_data" description="Deletes all data and tables">
    <echo>Dropping all database tables in ${sql.schema}...</echo>
    <property name="exec.command" value="mysqldump -u${sql.user} -p${sql.pass} --add-drop-table --no-data ${sql.schema} | grep ^DROP | mysql -u${sql.user} -p${sql.pass} ${sql.schema}" />
    <exec executable="sh">
      <arg value = "-c"/>
      <arg value="${exec.command}"/>
    </exec>

  </target>

</project>

お役に立てれば

1
Kieveli

最も単純なオプションを取り、<taskdef>にクラスパスを指定しないのはなぜですか?

<taskdef resource="axis-tasks.properties">
    <classpath>
        <fileset file="/path/to/axis/jars"/>
    </classpath>
</taskdef>

または、<classpath>をサブセット化する2番目のlibrary.dirエントリを作成しますか?

<path id="axis-tools-classpath">
    <fileset dir="/path/to/axis/home">
        <include name="*.jar"/>
    </fileset>
    <path refid="library.dir"/>
</path>

${ant.home}/libをいじることはあまり良い考えではなく、ほとんど常に回避できます。

1
matt b

matt bで言及されているように、私はtaskdefタスクで直接クラスパスを指定することができます。私のプロジェクトでは、プロジェクトフォルダーにtaskdefライブラリーを含め、antビルドファイルでクラスパスを指定して、他の開発用PCで単純にセットアップすると便利です。次のtaskdefを使用します。

<taskdef resource="antenna.properties" classpath="${myprojectroot}/lib/antenna-bin-1.2.1-beta.jar"/>

これは1.7.0より前のバージョンのantでは機能しない可能性があることに注意してください。

1
Ralf

* .jarを{ant.home}/libsにコピーすると、同じ問題が発生し、-libを使用して* .jarを見つけました。新しいjarをロードできないので、それをテストします。

0
Andrew