我正在寻找处理混合项目类型之间的项目间依赖关系的最佳实践,其中一些项目是eclipse插件/ OSGI包项目(RCP应用程序),而其他项目只是普通的旧Java项目(Web服务模块) )。很少有eclipse插件依赖于Java项目。
我的问题是,至少就我看来,在Eclipse PDE环境中无法干净地表达这种依赖关系。我可以让插件项目依赖于其他插件项目(通过Import-Package
或Require-Bundle
清单标题),但不是普通的java项目。
我似乎能够让项目从工作空间中的另一个项目声明对jar的依赖,但这些jar文件既不会被导出也不会被启动配置(虽然,java代码编辑看到库很好)。
“Java项目”用于构建服务以部署在J2EE容器(目前是JBoss 4.2.2)上,并在某些情况下生成多个jar - 一个用于部署到JBoss耳朵,另一个用于客户端使用代码(RCP应用程序)。
我们现在“解决”这个问题的方式是我们还有两个外部工具启动器配置 - 一个用于构建所有jar,另一个用于将这些jar复制到插件项目中。这样做(有点),但是“整个构建”和“复制罐子”目标会产生相当大的构建步骤,绕过整个eclipse增量构建功能并通过复制jar而不是仅仅引用项目我将依赖关系信息解耦并请求相当大的工作空间刷新,这会耗费开发时间,就像糖果一样。
我希望拥有一个更加“自然”的工作空间设置,它可以管理项目之间的依赖关系并仅在需要时请求增量重建,能够使用RCP应用程序插件中服务库的客户端代码并且能够在需要的所有必要类中启动RCP应用程序。
我也可以吃蛋糕然后吃它;)
需要明确的是,目前依赖管理和模块管理与Eclipse PDE配置有关。
我很清楚[Maven],[Ivy]和[Buckminster]等产品,他们解决了一个完全不同的问题(一旦我解决了工作区配置问题,这些产品实际上可以派上用场实现工作空间并构建产品)
答案 0 :(得分:11)
Eclipse项目依赖于项目属性(依赖项目?)中的复选框,这就是Eclipse决定构建哪个项目的方式。您可以自己设置,但通常在更改Java构建路径时设置。它将数据存储在.project文件IIRC中,因此一旦您浏览了GUI并看到了哪些更改,您就可以更灵活地应用其他文件。
然而,听起来你想要混合搭配Jars和Bundles。简单的方法是将所有项目视为Java项目。在PDE项目中,您实际上可以进入并调整Java构建路径;它会抱怨并说它不是正确的方法,但它会让你有一个PDE项目依赖于Java项目而没有那么蓬松的JARing。话虽如此,如果这种方法存在运行时问题,我也不会感到惊讶 - PDE运行时可能不会这样看。
另一种方法是让你的JAR自己成为PDE / OSGi包。毕竟,一个OSGi包只不过是一个JAR,它在Manifest中有一些额外的功能,它可以让你使用自动依赖管理轻松地开发和组装你的项目。这可能是最容易实现的,即使你真的不需要清单出现在你的捆绑包中。但这样做意味着你的PDE应用程序可以采用更模块化的方法,而不是在必要时在每个插件中嵌入库。
因此,PDE可以生成OSGi包,这只是JAR + Manifest的另一个名称。您可以在其他环境中以完全相同的方式使用JAR(例如,对于您的EAR或其他客户端使用),您可以利用应用程序中的OSGi层。鉴于你所谈论的混合束的类型,没有理由不这样做。
答案 1 :(得分:4)
我从未这样做过,所以这是一种理论方法。但我会尝试像ivy或maven2这样的依赖管理系统。
由于maven2做得更多,然后只是依赖管理,我建议在这种情况下使用常春藤。
答案 2 :(得分:2)
我们的解决方案使用Ant构建器将普通Java项目的“classes”目录直接复制到插件项目的顶级目录中。我们跳过JAR构建步骤以节省时间,并且它运行良好。如果插件项目依赖于已经构建的外部JAR,我们也会复制它们。
以下是如何在Eclipse 3.5中进行设置(对于奇怪的格式,但这是我能找到保留缩进的唯一方法):
Create empty "classes" dir in plugin project Select plugin project, hit F5 to refresh resources Create new Ant build file in plugin project to copy dependencies (ours is shown below) Right-click plugin project, select Properties Select Builders Click "New..." (brings up Edit Configuration dialog) Select Ant Builder and click "OK" Name your builder (ours is called "PluginProject externals") Browse workspace for Buildfile (ours is ${workspace_loc:/PluginProject/copyDependencies.xml}) click Refresh tab check "Refresh resources upon completion", click "Specific resources" click "Specify Resources...", check box for the classes dir, click "Finish" Click "OK" (closes Edit Configuration dialog) Click "Up" to move "PluginProject externals" to top of builder list Click "OK" (closes Properties dialog) Open your plugin project's MANIFEST.MF Click "Runtime" tab Click "Add..." under "Classpath", select your the "classes" dir and JARs and click "OK"
在插件项目中手动创建空的“classes”目录,这样您就可以告诉新构建器刷新该资源(在运行新构建器之前尚不存在)。这是我们的copyDependencies.xml文件中的内容:
<project name="Copy dependencies" default="copyDependencies" basedir=".">
<!--
This copying is needed because it appears that Eclipse plugins can't
depend directly on external Eclipse projects.
-->
<description>
Copies external dependency class andd JAR files into this plugin's directory.
</description>
<target name="copyDependencies">
<copy file="../External/JDOM/jdom-1.0/build/jdom.jar" todir="." preservelastmodified="true"/>
<copy file="../External/Xalan/xalan-j_2_6_0/bin/xalan.jar" todir="." preservelastmodified="true"/>
<copy file="../External/Xalan/xalan-j_2_6_0/bin/xercesImpl.jar" todir="." preservelastmodified="true"/>
<copy file="../External/Xalan/xalan-j_2_6_0/bin/xml-apis.jar" todir="." preservelastmodified="true"/>
<copy todir="./classes/com/arm" preservelastmodified="true">
<fileset dir="../Utilities/src/com/arm" excludes="**/*.java"/>
</copy>
</target>
<target name="clean" description="Deletes local copies of external classes and JARs.">
<delete file="jdom.jar" quiet="true"/>
<delete file="xalan.jar" quiet="true"/>
<delete file="xercesImpl.jar" quiet="true"/>
<delete file="xml-apis.jar" quiet="true"/>
<delete dir="./classes/com/arm/utilities" quiet="true"/>
</target>
</project>
这个方法的唯一缺点似乎是Eclipse在需要调用外部构建器时不是100%完美,所以偶尔你必须在Eclipse中做一个“Project&gt; Clean ...”强迫它。
答案 3 :(得分:1)
你有我的同情心。我也与这个问题作斗争,来自Eclipse开发的Wall-of-Silence开发了一个如此简单明了的问题:如何从插件声明一个依赖项到普通的Java项目(这样它在运行时工作)?
我认为他们不支持它。解决问题的唯一方法是在我的插件项目中创建文件夹,这些文件夹实际上是指向java项目的bin /文件夹的链接,然后将这些文件夹包含在插件中。这至少有效,但由于需要绝对的文件系统路径,它很脆弱。
答案 4 :(得分:1)
也许你可以使用“项目属性” - &gt; Eclipse上的“部署程序集”,并将其他项目添加到主项目中。其他项目看起来像“罐子”,自动添加到您的部署文件(战争,耳朵或其他)。也许这可行。至少它对我有用。祝你好运!!
Ariesandes。
答案 5 :(得分:1)
“构建路径”属性中有一个“链接源”选项,允许您为项目定义其他源文件夹。您可以选择另一个工作区项目的“src”文件夹,并将其重命名为“src2”或任何您想要的。这样,类被编译并部署到插件项目输出文件夹中,并且可以在运行时加载。
答案 6 :(得分:0)
使用一组复杂的构建依赖项,我发现Maven2和Hudson(用于CI)是一个非常好的组合。它需要一段时间来设置基础设施并让我了解配置,但在此之后,它才起作用。
当然,您依赖于Maven2(或Hudson)对构建机制的支持。我不确定Eclipse Headless构建的支持程度如何。但是,如果您使用Eclipse无头的唯一原因是允许依赖关系在一个地方表达,请自己帮忙并切换。
答案 7 :(得分:0)
我遇到了完全相同的问题。我们有一组可以由Maven构建的多个普通java项目,并且(当前)应该共享相同的类路径以正常工作。这些项目可用于在非OSGi环境中启动服务器。然后我们还有一个eclipse RCP客户端,它将这些项目作为一个捆绑包使用。我完全能够使用Apache Felix Maven Bundle Plugin使用Maven构建这个大包,一切正常。但每当我在正常项目中更改一个类时,我必须重建整个包。增量构建和资源链接不起作用。
我尝试了将这些项目中的二进制文件/源目录链接到一个大包的Manifest类路径中的“解决方案”,它似乎可以工作,但这将是一个真正的维护噩梦,因为我甚至需要链接一些个人文件。
具有讽刺意味的是,由于在客户端使用OSGi,我实际上正在考虑将Maven模块的漂亮模块化结构折叠成只有一个插件项目的子包。这是一个比开发速度极慢的更好的选择。
我将首先调查的另一个更好的替代方案(根据OSGi人员)是将我的所有Maven模块都变成OSGi包。但这可能是一个真正的PITA,因为我依赖于类路径扫描并组合来自多个包(有时是同一个名称)的多个配置文件来形成一个配置。
(具体来说,我们使用Spring框架,并将多个persistence.xml文件合并到一个持久化上下文中。我们还合并了来自不同模块的几个spring XML上下文文件(都提供了不同的ASPECT),这些文件应该共同构成一个Spring上下文由Spring-DM使用。)