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

时间:2012-05-17 08:32:43

标签: blackberry ant

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


原因

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

所以要在bb-ant-tools构建中运行jarjar,我需要知道如何将输出JAR编译成COD

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

进度

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

2)要包含sdk代码,我目前将源文件夹添加到主项目构建中。 (将来我想知道如何将库作为JAR包含到构建中 - BlackBerry - Ant script to include JAR in project without external dependancies

3)此步骤的输出包括通常的文件:

  • MyApp.cod(如果我签名,它在设备上运行完美)
  • MyApp.csl
  • MyApp.cso
  • MyApp.debug
  • MyApp.jad
  • MyApp.jar (我想在此运行jarjar)
  • MyApp.rapc

4)我尝试运行第二个构建,获取上述JAR文件并将其用作rapc调用中的唯一源文件。我这样做是将src rapc标记指向仅包含我的JAR文件的文件夹。

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

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

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

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


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

build:
    [mkdir] Skipping C:\development\ant\new_test\MyApp\build because it already exists.
     [copy] Copying 1 file to C:\development\ant\new_test\MyApp\build
     [copy] Copying C:\development\ant\new_test\MyApp\icon.png to C:\development\ant\new_test\MyApp\build\icon.png
     [rapc] Compiling 1 source files to MyApp.cod
     [rapc] Executing 'C:\Java\jdk1.6.0_24\jre\bin\java.exe' with arguments:
     [rapc] '-classpath'
     [rapc] 'C:\Java\jdk1.6.0_24\lib\tools.jar;C:\development\tools\bb-jde\jde5.0\components\bin\rapc.jar'
     [rapc] 'net.rim.tools.compiler.Compiler'
     [rapc] '-verbose'
     [rapc] 'import=C:\development\tools\bb-jde\jde5.0\components\lib\net_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:\development\tools\bb-jde\jde5.0\components\lib\net_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:\development\ant\new_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 .
     [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:\development\ant\new_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 .
     [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:\development\ant\new_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.cocompiler.Compiler.compile(Unknown Source)
     [rapc]     at net.rim.tools.compiler.Compiler.main(Unknown Source)
     [rapc] d MyApp.csl MyApp.cso -C C:\Users\Richard\AppData\Local\Temp\rapc_598c2ad7.dir .

BUILD FAILED
C:\development\ant\new_test\MyApp\build.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:\development\ant\new_test\MyApp>

解决方法

在进行编译之前,

Esaj有一个good answer below在源代码上运行Ant replace任务。这意味着无需在任何生成的JAR文件上运行jarjar来解决命名空间问题。

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


相关

是[How to convert JAR FILE to COD file using Ant Build]的重复,但该问题尚未得到解答,我添加了更多详细信息。

这是重复:

3 个答案:

答案 0 :(得分:2)

根据@ RichardLeMesurier的要求,从BlackBerry - Ant script to include JAR in project without external dependancies复制此答案:

去年我遇到了类似的问题,我创建了一个“框架”,用作多个BB应用程序的基础,但遇到了多个COD的问题(我不记得究竟是什么,类似于设备拒绝安装具有相同外部鳕鱼的多个应用程序,如果外部COD没有先单独安装,然后是应用程序)。由于应用程序可以单独安装(一个人可能只安装应用程序A,另一个人可能只安装应用程序B,而另一个人可能同时安装A和B),整个框架需要包含在使用它的所有应用程序中。我使用bb-ant-tools编写了这个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 Eclipse\plugins\net.rim.ejde\vmTools to the components\bin -dir 
        if the keys were installed using the Eclipse-plugin     -->
    <property name="jdehome" value="C:\BB\Eclipse\plugins\net.rim.ejde.componentpack5.0.0_5.0.0.25\components" />

    <!-- Framework source locations, these must be set correctly -->
    <property name="frameworkRes.dir" value="C:\BB\workspace\BB_Framework\res" />
    <property name="frameworkSrc.dir" value="C:\BB\workspace\BB_Framework\src\com\whatever\frame" />

    <!-- 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定位构建目录。签名,清理和将应用程序加载到模拟器有单独的任务。希望这会有所帮助。

答案 1 :(得分:0)

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

  

jar -cfmv C:\ development \ ant \ new_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 -CC:\ Users \ Richard \ AppData \ Local \ Temp \ rapc_598c2ad7.dir。

jar tool documentation中查找这些选项:

  

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

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

答案 2 :(得分:0)

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

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

Michael's answer开始,正确的方法是使用普通的java工具(javac&amp; jar)以及RIM的preverify命令构建最终的JAR文件。

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

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

<强>步骤

  1. javac用于创建CLASS文件的SDK
  2. preverify CLASS文件
  3. jar SDK
  4. 将SDK JAR文件复制到项目中
  5. javac项目 - 使用SDK JAR作为类路径
  6. preverify项目CLASS文件(再次使用类路径中的SDK JAR
  7. jar项目 - 将SDK JAR添加为 zipfileset
  8. jarjar此项目JAR根据需要重构包名称
  9. 最后,在此JAR上运行rapc - 它将找不到重复的COD文件&amp;应该运行良好。
  10. 注意:只需在SDK上使用rapc即可组合步骤1-3(如果您需要在SDK代码上运行预处理程序代码,则需要此步骤。)

    通过将其分解为这样的简单步骤,我了解了普通的java工具如何链接到RIM的工具链(通常,只需在源文件夹上调用rapc时就会隐藏它。)

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

    我在ANT中做了所有这些。我使用不同的文件夹来存储每个步骤的输出。这样我最终会得到5个临时文件夹,但是我可以很容易地调试这些步骤。


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

    完成此任务的完整ANT构建框架可以扩展到许多不同的文件(在我的情况下,我认为我现在使用8包括属性文件)。它需要有良好的ANT工作知识,普通的java构建工具和RIM的rapc命令。

    我想我已经在我的问题中很好地记录了这个过程的每一步,并且沿着这条线找到了一些很好的答案。有关详细信息,请查看其他问题&amp;答案。每个都包含有用的链接,以及来自该社区其他开发人员的一些好的见解