知其然知其所以然 2012-08-10
http://jimmy-duan.iteye.com/blog/1057967
当前的web项目有引用到子工程项目,而且多个子工程项目也有引用到其它的工程项目,现要求利用Ant自动将web项目打包成war包,其中引用到的子工程项目需打成jar包,而且必须是混淆后的jar包。其中混淆代码的工具选择了开源的Proguard(http://proguard.sourceforge.net/),可以运行proguard自带的proguardgui.jar(图形化用户界面)生成proguard配置文件。
ProGuard是一个免费的java类文件压缩,优化,混淆器.它探测并删除没有使用的类,字段,方法和属性.它删除没有用的说明并使用字节码得到最大优化.它使用无意义的名字来重命名类,字段和方法.
ProGuard的使用是为了:
1.创建紧凑的代码文档是为了更快的网络传输,快速装载和更小的内存占用.
2.创建的程序和程序库很难使用反向工程.
3.所以它能删除来自源文件中的没有调用的代码
4.充分利用java6的快速加载的优点来提前检测和返回java6中存在的类文件.
参数:
-include{filename}从给定的文件中读取配置参数
-basedirectory{directoryname}指定基础目录为以后相对的档案名称
-injars{class_path}指定要处理的应用程序jar,war,ear和目录
-outjars{class_path}指定处理完后要输出的jar,war,ear和目录的名称
-libraryjars{classpath}指定要处理的应用程序jar,war,ear和目录所需要的程序库文件
-dontskipnonpubliclibraryclasses指定不去忽略非公共的库类。
-dontskipnonpubliclibraryclassmembers指定不去忽略包可见的库类的成员。
保留选项
-keep{Modifier}{class_specification}保护指定的类文件和类的成员
-keepclassmembers{modifier}{class_specification}保护指定类的成员,如果此类受到保护他们会保护的更好
-keepclasseswithmembers{class_specification}保护指定的类和类的成员,但条件是所有指定的类和类成员是要存在。
-keepnames{class_specification}保护指定的类和类的成员的名称(如果他们不会压缩步骤中删除)
-keepclassmembernames{class_specification}保护指定的类的成员的名称(如果他们不会压缩步骤中删除)
-keepclasseswithmembernames{class_specification}保护指定的类和类的成员的名称,如果所有指定的类成员出席(在压缩步骤之后)
-printseeds{filename}列出类和类的成员-keep选项的清单,标准输出到给定的文件
压缩
-dontshrink不压缩输入的类文件
-printusage{filename}
-whyareyoukeeping{class_specification}
优化
-dontoptimize不优化输入的类文件
-assumenosideeffects{class_specification}优化时假设指定的方法,没有任何副作用
-allowaccessmodification优化时允许访问并修改有修饰符的类和类的成员
混淆
-dontobfuscate不混淆输入的类文件
-printmapping{filename}
-applymapping{filename}重用映射增加混淆
-obfuscationdictionary{filename}使用给定文件中的关键字作为要混淆方法的名称
-overloadaggressively混淆时应用侵入式重载
-useuniqueclassmembernames确定统一的混淆类的成员名称来增加混淆
-flattenpackagehierarchy{package_name}重新包装所有重命名的包并放在给定的单一包中
-repackageclass{package_name}重新包装所有重命名的类文件中放在给定的单一包中
-dontusemixedcaseclassnames混淆时不会产生形形色色的类名
-keepattributes{attribute_name,...}保护给定的可选属性,例如LineNumberTable,LocalVariableTable,SourceFile,Deprecated,Synthetic,Signature,andInnerClasses.
-renamesourcefileattribute{string}设置源文件中给定的字符串常量
因为我们开发的是struts2+spring+openjpa的架构的项目,所有需要很详细的配置,其中proguard需要的injars,outjars还有其它的一些libraryjars在Ant的build.xml文件以参数形式加入。
proguard.pro文件具体内容:
-libraryjars<java.home>/lib/rt.jar
-libraryjars<java.home>/lib/jce.jar
-libraryjars<java.home>/lib/jsse.jar
-obfuscationdictionarydictionaries/compact.txt
-classobfuscationdictionarydictionaries/shakespeare.txt
#-printmappingproguard.map
-overloadaggressively
-defaultpackage''
#-flattenpackagehierarchy''
#-dontusemixedcaseclassnames
#-keeppackagenames
-allowaccessmodification
-dontoptimize
#因为项目中使用到了jpa的Annotation注解,需要保留这个属性
-keepattributes*Annotation*
#Keepweblistener-class,自己编写的weblistener,不能混淆,不然启动服务的时候会报错
-keeppublicclasscom.test.mylistener{
publicprotected*;
}
#Keepjpadao,自身编写的一些dao接口不能混淆,而且如果是有被spring管理的dao,也应该不进行混淆
-keeppublicclasscom.test.dao.*{
publicprotected*;
}
#其中jpadao中使用到的一些重要接口也能进行混淆
-keeppublicclasscom.test.persistence.core.QueryLiteral{
publicprotected*;
}
-keeppublicclasscom.test.persistence.JpaDao{
publicprotected*;
}
#Keepbeansmanagedbyspringframework,spring管理的bean不进行混淆
-keeppublicclasscom.test.provider.*{
publicprotected*;
}
#KeepNetworkinterface
-keeppublicclasscom.test.loader.Network{
publicprotected*;
}
#Keepentityclassesextendsjava.io.Serializable
#保留jpa中使用到的所有实体类,不进行混淆
-keeppublicclass*implementsjava.io.Serializable{
publicprotectedprivate*;
}
-keeppublicclasscom.test.entity.*{
publicprotectedprivate*;
}
#Keep-Applications.Keepallapplicationclasses,alongwiththeir'main'methods.
-keepclasseswithmemberspublicclass*{
publicstaticvoidmain(java.lang.String[]);
}
#Keepnames-Nativemethodnames.Keepallnativeclass/methodnames.
-keepclasseswithmembers,allowshrinkingclass*{
native<methods>;
}
proguard和build.xml文件具体目录结构:
server工程
-src目录
-build目录
-proguard目录
-dictionaries主要存放混淆时需要用到字典文件,可以使用官网下载到的proguard.zip包中dictionaries的两个文件(dictionaries/compact.txt,dictionaries/shakespeare.txt)
-lib存放proguard.jar包,这是混淆中使用到的最主要的一个jar包(下载到的proguard.zip中)
-proguard.pro主要的混淆配置文件
-build.xml
Ant的build.xml文件编写如下:
<?xml version="1.0" encoding="UTF-8"?>
<project name="com.ibm.crl.scm.crc.server" default="dist" basedir=".">
<target name="init" description="设置初始化打war包需要的路径变量">
<!-- 你的eclipse工作空间站目录路径,以备引用到子工程项目使用-->
<property name="workspace" value="${basedir}/../"/>
<!-- 需要被打包成war包的主工程项目-->
<property name="server.name" value="server"/>
<!-- 主工程项目引用到的其它子工程项目-->
<property name="childproject1.name" value="childproject1"/>
<property name="childproject2.name" value="childproject2"/>
<property name="src" value="${basedir}/src"/>
<property name="resources" value="${basedir}/resources"/>
<property name="lib" value="${basedir}/WebContent/WEB-INF/lib"/>
<property name="webapp" value="${basedir}/WebContent"/>
<property name="proguard" value="${basedir}/proguard"/>
<property name="proguard.lib" value="${proguard}/lib"/>
<property name="build.src" value="${basedir}/build/src"/>
<property name="build.dest" value="${basedir}/build/WEB-INF/classes"/>
<property name="buildwar.dest" value="${basedir}/build/server"/>
<property name="war.dest" value="${basedir}/build/war"/>
<!-- 该目录是存储引用到的子工程混淆后的jar包-->
<property name="referenced.lib" value="${basedir}/build/referenced"/>
<path id="classpath">
<fileset dir="${referenced.lib}">
<include name="**/*.jar"/>
</fileset>
<fileset dir="${lib}">
<include name="**/*.jar"/>
</fileset>
</path>
</target>
<target name="prepare" depends="init" description="创建打包需要的路径,拷贝源文件到打包路径下">
<mkdir dir="${build.src}"/>
<mkdir dir="${build.dest}"/>
<mkdir dir="${buildwar.dest}"/>
<mkdir dir="${war.dest}"/>
<mkdir dir="${referenced.lib}"/>
<copy todir="${build.src}">
<fileset dir="${src}"/>
<fileset dir="${resources}"/>
</copy>
<copy todir="${buildwar.dest}">
<fileset dir="${webapp}"/>
</copy>
</target>
<!-- 因为当前我们项目中,各个子工程项目有互相引用的情况,打包成jar包就成问题了,所以直接将eclipse自动编译的.class文件打包成jar包,然后再进行混淆-->
<target name="references" depends="prepare" description="将引用到的子工程打包成jar文件">
<jar destfile="${referenced.lib}/${childproject1.name}.jar">
<fileset dir="${workspace}/${childproject1.name}/bin"/>
</jar>
<jar destfile="${referenced.lib}/${childproject2.name}.jar">
<fileset dir="${workspace}/${childproject2.name}/bin"/>
</jar>
</target>
<target name="proguard" depends="references" description="混淆引用到的子工程jar文件,并替换">
<taskdef resource="proguard/ant/task.properties" classpath="${proguard.lib}/proguard.jar" />
<proguard configuration="${proguard}/proguard.pro">
<injar file="${referenced.lib}/${childproject1.name}.jar"/>
<outjar file="${referenced.lib}/OB_${childproject1.name}.jar"/>
<injar file="${referenced.lib}/${childproject2.name}.jar"/>
<outjar file="${referenced.lib}/OB_${childproject2.name}.jar"/>
<libraryjar dir="${lib}"/>
</proguard>
<move file="${referenced.lib}/OB_${childproject1.name}.jar" tofile="${referenced.lib}/${childproject1.name}.jar"/>
<move file="${referenced.lib}/OB_${childproject2.name}.jar" tofile="${referenced.lib}/${childproject2.name}.jar"/>
</target>
<target name="build" depends="proguard" description="编译java文件,拷贝properties属性配置文件到编译后的路径下">
<javac srcdir="${build.src}" destdir="${build.dest}">
<classpath refid="classpath"/>
</javac>
<copy todir="${build.dest}">
<fileset dir="${build.src}">
<include name="**/*.properties"/>
<include name="**/server.keystore"/>
<include name="**/*.xml"/>
</fileset>
</copy>
</target>
<target name="dist" depends="build" description="打war包,不将java文件打入包内">
<delete dir="${build.src}"/>
<war warfile="${war.dest}/${server.name}.war" webxml="${buildwar.dest}/WEB-INF/web.xml">
<lib dir="${referenced.lib}/"/>
<classes dir="${build.dest}"/>
<fileset dir="${buildwar.dest}"/>
</war>
<antcall target="clean"></antcall>
</target>
<target name="clean" depends="init" description="清除打包用临时文件">
<delete dir="${build.src}"/>
<delete dir="${basedir}/build/WEB-INF"/>
<delete dir="${buildwar.dest}"/>
</target>
<target name="cleanAll" depends="clean" description="清除所有打包生成文件">
<delete dir="${war.dest}"/>
<delete dir="${referenced.lib}"/>
</target>
</project>