利用Ant与Proguard混淆引用的子工程项目jar包及打war包

知其然知其所以然 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>

相关推荐