BlackBerry:从 Ant 脚本中的 JAR 源文件创建 COD



如何使用 Ant (bb-ant-tools( 将 JAR 文件编译成 COD? 即没有其他源文件


原因

最终,我需要一个脚本,该脚本将在构建期间运行 Jar Jar 链接 (jarjar(,以解决在多个应用程序中使用我们的 sdk 的命名空间问题。 Jarjar 将 JAR 文件作为输入,我认为它输出一个 JAR 文件。

因此,要在bb-ant-tools构建中运行jarjar,我需要知道如何将该输出JAR编译为COD

我只关注问题的这一部分,以尽量减少其他可能的错误来源。一旦我能够掌握将JAR构建为COD,我将尝试jarjar步骤。

进展

1(我可以将我的项目构建为工作COD,签名并在设备上运行。

2( 为了包含 sdk 代码,我目前将源文件夹添加到主项目构建中。(将来我想知道如何将库作为 JAR - 黑莓 - Ant 脚本包含在构建中,以便在没有外部依赖项的情况下将 JAR 包含在项目中(

3( 此步骤的输出包括常用文件:

  • MyApp.cod(如果我签名,它会在设备上完美运行(
  • MyApp.csl
  • 我的应用网
  • MyApp.debug
  • MyApp.jad
  • MyApp.jar(我想在这个上运行jarjar(
  • MyApp.rapc

4(我尝试运行第二个构建,采用上述JAR文件并将其用作rapc调用中的唯一源文件。为此,我将rapcsrc标签指向仅包含我的 JAR 文件的文件夹。

在构建的最后一部分,当 rapc adding文件时,我收到错误:

 java.util.zip.ZipException: duplicate entry: MyApp-1.cod

(下面编辑的构建输出中错误的完整详细信息(

我可以看到这是 rapc 第二次尝试添加此文件。我不明白为什么,因为 JAR 中只有该文件的一个副本。


构建输出(使用 ...etc 编辑以使其可读(

build:
    [mkdir] Skipping C:developmentantnew_testMyAppbuild because it already exists.
     [copy] Copying 1 file to C:developmentantnew_testMyAppbuild
     [copy] Copying C:developmentantnew_testMyAppicon.png to C:developmentantnew_testMyAppbuildicon.png
     [rapc] Compiling 1 source files to MyApp.cod
     [rapc] Executing 'C:Javajdk1.6.0_24jrebinjava.exe' with arguments:
     [rapc] '-classpath'
     [rapc] 'C:Javajdk1.6.0_24libtools.jar;C:developmenttoolsbb-jdejde5.0componentsbinrapc.jar'
     [rapc] 'net.rim.tools.compiler.Compiler'
     [rapc] '-verbose'
     [rapc] 'import=C:developmenttoolsbb-jdejde5.0componentslibnet_rim_api.jar'
     [rapc] 'codename=MyApp'
     [rapc] 'MyApp.rapc'
     [rapc] '@sources.txt'
     [rapc]
     [rapc] The ' characters around the executable and arguments are
     [rapc] not part of the command.
     [rapc] Setting environment variable: PATH=........etc
     [rapc] Reading resource: MyApp.cod
...etc
     [rapc] Parsing classfile: com/MyApp/ui/views/WelcomeBar.class
...etc
     [rapc] Parsing import: C:developmenttoolsbb-jdejde5.0componentslibnet_rim_api.jar(net_rim_amms.cod)
...etc
     [rapc] Resolving
...etc
     [rapc] Optimizing
     [rapc] Utilities.java:449: Warning!: local variable(s) { finished } initialized but not used in: com.cobi.library.Utilities.split(String,String)
...etc
     [rapc] Populating
     [rapc] Invoking: jar -cfmv C:developmentantnew_testMyAppbuildMyApp.jar C:UsersRichardAppDataLocalTemprapc_598c0c5a.dirMETA-INFMANIFEST.MF MyApp.cod MyApp-1.cod MyApp-2.cod MyApp.csl MyApp.cso -C C:UsersRichardAppDataLocalTemprapc_598c2ad7.dir .
     [rapc] added manifest
     [rapc] adding: MyApp.cod(in = 63208) (out= 41042)(deflated 35%)
     [rapc] adding: MyApp-1.cod(in = 75448) (out= 42559)(deflated 43%)
     [rapc] adding: MyApp.csl(in = 91) (out= 69)(deflated 24%)
     [rapc] adding: MyApp.cso(in = 157) (out= 93)(deflated 40%)
...etc - adding all files I can see in the JAR...
     [rapc] adding: MyApp-1.cod
     java.util.zip.ZipException: duplicate entry: MyApp-1.cod
     [rapc]     at java.util.zip.ZipOutputStream.putNextEntry(ZipOutputStream.java:175)
     [rapc]     at java.util.jar.JarOutputStream.putNextEntry(JarOutputStream.java:92)
     [rapc]     at sun.tools.jar.Main.addFile(Main.java:713)
     [rapc]     at sun.tools.jar.Main.create(Main.java:466)
     [rapc]     at sun.tools.jar.Main.run(Main.java:180)
     [rapc]     at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
     [rapc]     at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:39)
     [rapc]     at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:25)
     [rapc]     at java.lang.reflect.Method.invoke(Method.java:597)
     [rapc]     at net.rim.tools.compiler.c.e.if(Unknown Source)
     [rapc]     at net.rim.tools.compiler.c.e.a(Unknown Source)
     [rapc]     at net.rim.tools.compiler.Compiler.a(Unknown Source)
     [rapc]     at net.rim.tools.compiler.Compiler.a(Unknown Source)
     [rapc]     at net.rim.tools.compiler.Compiler.compile(Unknown Source)
     [rapc]     at net.rim.tools.compiler.Compiler.main(Unknown Source)
     [rapc] java.io.IOException: jar command failed: jar -cfmv C:developmentantnew_testMyAppbuildMyApp.jar C:UsersRichardAppDataLocalTemprapc_598c0c5a.dirMETA-INFMANIFEST.MF MyApp.cod MyApp-1.cod MyApp-2.cod MyApp.csl MyApp.cso -C C:UsersRichardAppDataLocalTemprapc_598c2ad7.dir .
     [rapc]     at net.rim.tools.compiler.Compiler.a(Unknown Source)
     [rapc]     at net.rim.tools.compiler.Compiler.a(Unknown Source)
     [rapc]     at net.rim.tools.I/O Error: jar command failed: jar -cfmv C:developmentantnew_testMyAppbuildMyApp.jar C:UsersRichardAppDataLocalTemprapc_598c0c5a.dirMETA-INFMANIFEST.MF MyApp.cod MyApp-1.cod MyApp-2.cocompiler.Compiler.compile(Unknown Source)
     [rapc]     at net.rim.tools.compiler.Compiler.main(Unknown Source)
     [rapc] d MyApp.csl MyApp.cso -C C:UsersRichardAppDataLocalTemprapc_598c2ad7.dir .
BUILD FAILED
C:developmentantnew_testMyAppbuild.xml:65: Java returned: -1
        at org.apache.tools.ant.taskdefs.Java.execute(Java.java:111)
        at ca.slashdev.bb.tasks.RapcTask.executeRapc(RapcTask.java:583)
        at ca.slashdev.bb.tasks.RapcTask.execute(RapcTask.java:401)
        at org.apache.tools.ant.UnknownElement.execute(UnknownElement.java:291)
        at sun.reflect.GeneratedMethodAccessor4.invoke(Unknown Source)
        at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:25)
        at java.lang.reflect.Method.invoke(Method.java:597)
        at org.apache.tools.ant.dispatch.DispatchUtils.execute(DispatchUtils.java:106)
        at org.apache.tools.ant.Task.perform(Task.java:348)
        at org.apache.tools.ant.Target.execute(Target.java:390)
        at org.apache.tools.ant.Target.performTasks(Target.java:411)
        at org.apache.tools.ant.Project.executeSortedTargets(Project.java:1399)
        at org.apache.tools.ant.Project.executeTarget(Project.java:1368)
        at org.apache.tools.ant.helper.DefaultExecutor.executeTargets(DefaultExecutor.java:41)
        at org.apache.tools.ant.Project.executeTargets(Project.java:1251)
        at org.apache.tools.ant.Main.runBuild(Main.java:809)
        at org.apache.tools.ant.Main.startAnt(Main.java:217)
        at org.apache.tools.ant.launch.Launcher.run(Launcher.java:280)
        at org.apache.tools.ant.launch.Launcher.main(Launcher.java:109)
Total time: 9 seconds
C:developmentantnew_testMyApp>

解决方法

Esaj在下面有一个很好的答案,在编译之前在源代码上运行Ant replace任务。这意味着无需对任何生成的 JAR 文件运行 jarjar 来解决命名空间问题。

这将对我有用,因为我有 SDK 源代码。它不适用于我的客户端,因为我以 JAR 格式分发我的 SDK。所以我仍然希望这个问题的答案。


相关

这是[如何使用Ant Build将JAR文件转换为COD文件]的副本,但这个问题尚未得到解答,我添加了更多详细信息。

这不是以下内容的副本:

  • [ 使用 bb-ant 将.jar文件转换为 .cod 文件 ] - 我使用了公认的答案,我的错误不同。
  • [ 黑莓 - Ant 脚本将 JAR 包含在项目中而没有外部依赖项 ] - 这个问题是关于将库添加为 JAR 而不是像我上面那样使用源代码。在某种程度上,它是一个前体步骤,因为它的输出仍然需要通过jarjar运行。

根据 @RichardLeMesurier 的请求,从 BlackBerry - Ant 脚本复制此答案以将 JAR 包含在没有外部依赖项的项目中:

去年我遇到了类似的问题,我创建了一个"框架",用作多个 BB 应用程序的基础,但遇到了多个 COD 的问题(我不记得具体是什么了,比如设备拒绝安装多个具有相同外部编码的应用程序,如果外部 COD 没有先单独安装,然后再安装应用程序(。由于应用程序可以单独安装(一个人可能只安装应用程序 A,另一个人可能只安装应用程序 B,还有一个人可能同时安装 A 和 B(,因此整个框架需要包含在使用它的所有应用程序中。我使用 bb-ant 工具制作了这个 Ant 脚本(希望我没有破坏任何东西,删除一些特定于我们应用程序的东西并混淆包名称等(:

<?xml version="1.0" encoding="UTF-8"?>
<project name="${description}" default="build" basedir=".">
    <taskdef resource="bb-ant-defs.xml" classpath="lib/bb-ant-tools.jar" />
    <!-- rapc and sigtool require the jde.home property to be set -->
    <!-- NOTE: You may need to copy the signature files from Eclipsepluginsnet.rim.ejdevmTools to the componentsbin -dir 
        if the keys were installed using the Eclipse-plugin     -->
    <property name="jdehome" value="C:BBEclipsepluginsnet.rim.ejde.componentpack5.0.0_5.0.0.25components" />
    <!-- Framework source locations, these must be set correctly -->
    <property name="frameworkRes.dir" value="C:BBworkspaceBB_Frameworkres" />
    <property name="frameworkSrc.dir" value="C:BBworkspaceBB_Frameworksrccomwhateverframe" />
    <!-- Locations for simulator, binaries, jde home, don't touch these -->
    <property name="simulator" value="${jdehome}simulator" />
    <property name="bin" value="${jdehome}bin" />
    <property name="jde.home" location="${jdehome}" />
    <!-- directory of simulator to copy files to -->
    <property name="simulator.home" location="${simulator}" />
    <property name="src.dir" location="src" />
    <property name="build.dir" location="build" />
    <property name="temp.dir" location="C:tempsrc" />
    <!-- Project specific -->
    <!-- Application title -->
    <property name="app.title" value="Application Name" />
    <property name="app.version" value="1.0.0" />
    <!-- Value to prepend before frame-class packages -->
    <property name="frame.prefix" value="appname" />
    <!-- Name of the COD to produce --> 
    <property name="cod.name" value="Appname" />
    <target name="build">
        <mkdir dir="${build.dir}" />
        <delete dir="${temp.dir}" />
        <mkdir dir="${temp.dir}" />
        <mkdir dir="${temp.dir}${frame.prefix}" />
        <copy toDir="${temp.dir}">
            <fileset dir="${src.dir}">
                <include name="**/*.java" />
            </fileset>
        </copy>
        <copy toDir="${temp.dir}${frame.prefix}">
            <fileset dir="${frameworkSrc.dir}">
                <include name="**/*.java" />
            </fileset>
        </copy>
        <copy toDir="${temp.dir}res">
            <fileset dir="${frameworkRes.dir}">
                <include name="**/*" />
            </fileset>
        </copy>
        <copy toDir="${temp.dir}res">
            <fileset dir="res">
                <include name="**/*" />
            </fileset>
        </copy>
        <!-- This replaces the package names for classes copied from under 
            framework-directory to ${frame.prefix} -directory as well as changing any 
            imports using the classes in framework-package -->  
        <replace dir="${temp.dir}" value="${frame.prefix}">
            <include name="**/*.java"/>
            <replacetoken>com.whatever.frame</replacetoken>
        </replace>
        <rapc output="${cod.name}" srcdir="${temp.dir}" destdir="${build.dir}">
            <jdp title="${app.title}" 
                version="${app.version}" 
                vendor="Your Company"
                icon="../res/img/icon.png"
            />
        </rapc>
    </target>
    <target name="sign">
        <sigtool codfile="${build.dir}/${cod.name}.cod" />
    </target>
    <target name="clean">
        <delete dir="${build.dir}" />
    </target>
    <target name="load-simulator" depends="build">
        <copy todir="${simulator.home}">
            <fileset dir="${build.dir}" includes="*.cod,*.cso,*.debug,*.jad,*.jar" />
        </copy>
    </target>
</project>

这样做的目的是将所有 java 文件和资源从当前项目复制,然后从框架项目复制到临时目录,在框架工作文件的过程中替换包名称(因为它们被放入一个单独命名的目录中(,这与以下事实有关:设备还拒绝安装多个在同一包下具有相同类的应用程序(即 框架类,对于您的情况,这可能不是必需的(。复制和替换完成后,将使用 rapc 将应用程序构建为目标构建目录。签名、清理应用和将应用加载到模拟器有单独的任务。希望这有帮助。

从控制台输出中,失败的命令:

jar -cfmv C:\development\antew_test\MyApp\build\MyApp.jar C:\Users\Richard\AppData\Local\Temp\rapc_598c0c5a.dir\META-INF\MANIFEST.MF MyApp.cod MyApp-1.cod MyApp-2.cod MyApp.csl MyApp.cso -C C:\Users\Richard\AppData\Local\Temp\rapc_598c2ad7.dir .

在 jar 工具文档中查找这些选项:

-C 目录
在处理以下输入文件时,在执行 jar 命令期间临时更改目录 (cd 目录(论点。它的操作旨在类似于 -C 选项UNIX tar 实用程序。

基于此,我认为rapc将解压缩的cod文件放在C:\Users\Richard\AppData\Local\Temp\rapc_598c2ad7.dir中,这会导致与命令行上指定的cod文件发生冲突。 那个目录还在吗? 看看里面有什么。

用一些细节来回答我自己的问题......

不能多次调用rapc - 它会创建太多的 COD 文件。这就是我收到该错误的原因。

根据Michael的回答,正确的方法是使用普通的java工具(javac和jar(以及RIM的preverify命令构建最终的JAR文件。

仅在最后一步中使用rapc - 将该 JAR 文件转换为 COD。

一个完整的 ANT 构建框架来处理这个问题太大了,无法放在这里,但下面列出了创建它所需的步骤。每个步骤都可以在这个网站上(或使用一些谷歌(轻松研究。每个步骤都非常简单,可以单独调试。

步骤

  1. javac开发工具包以创建类文件
  2. preverify类文件
  3. jar开发工具包
  4. 将 SDK JAR 文件复制到项目中
  5. javac项目 - 使用 SDK JAR 作为类路径
  6. preverify项目 CLASS 文件(同样,在类路径中使用 SDK JAR(
  7. jar项目 - 将 SDK JAR 添加为 zip 文件集
  8. jarjar此项目 JAR 以根据需要重构包名称
  9. 最后,在此 JAR 上运行rapc - 它不会找到重复的 COD 文件,并且应该运行良好。

注意:只需在 SDK 上使用 rapc即可组合步骤 1-3(如果您需要在 SDK 代码上运行预处理器标记,则需要这样做(。

通过将其分解为这样的简单步骤,我了解了普通的Java工具如何链接到RIM的工具链(通常,当您简单地在源文件夹上调用rapc时,这一切都是隐藏的(。

当然,您仍然需要与sigtool签署COD。

我在ANT中完成所有这些工作。我使用不同的文件夹来存储每个步骤的输出。这样,我最终会在最后得到 5 个临时文件夹,但它使调试步骤变得容易。


我现在终于明白为什么很少有人能够为我的各种BB ANT构建脚本问题提供结论性的答案。这个过程很费力,而且很长,而且很难解释。

一个完整的 ANT 构建框架来完成这一点可以扩展到许多不同的文件(就我而言,我想我现在使用 8 个包括属性文件(。它需要对 ANT、普通的 Java 构建工具和 RIM 的rapc命令有很好的工作知识。

我想我已经很好地记录了这个过程的每一步,并在此过程中获得了一些很好的答案。有关更多详细信息,请查看其他问题和答案。每个都包含有用的链接,以及来自该社区中其他开发人员的一些良好见解

最新更新