proguard代码混淆

springboot项目混淆方案

proguard简单来说是为了防止反编译,更准确的说,进行业务代码的混淆,是使得代码易读性变差。
引入proguard-maven-plugin
一:

<plugin>
        <groupId>com.github.wvengen</groupId>
        <artifactId>proguard-maven-plugin</artifactId>
        <version>2.0.14</version>
        <executions>
          <execution>
            <phase>package</phase>
            <goals>
              <goal>proguard</goal>
            </goals>
          </execution>
        </executions>
        <configuration>
          <proguardVersion>6.0.3</proguardVersion>
          <injar>${project.build.finalName}.jar</injar>
          
          <outjar>${project.build.finalName}.jar</outjar>
          
                    <outjar>${project.build.finalName}.jar</outjar>
          -->
          <obfuscate>true</obfuscate>
          <options>
            <!--# JDK目标版本1.8-->
            <option>-target 1.8</option>
            <!-- 不做收缩(删除注释、未被引用代码)-->
            <option>-dontshrink</option>
            <!-- 不做优化(变更代码实现逻辑)-->
            <option>-dontoptimize</option>
            <!--  ##对于类成员的命名的混淆采取唯一策略-->
            <option>-useuniqueclassmembernames</option>
            <!--## 混淆类名之后,对使用Class.forName('className')之类的地方进行相应替代-->
            <option>-adaptclassstrings</option>
            <!--混淆时不生成大小写混合的类名,默认是可以大小写混合-->
            <option>-dontusemixedcaseclassnames</option>
            <!--忽略警告-->
            <option>-ignorewarnings</option>
            <!-- This option will replace all strings in reflections method invocations with new class names.
                 For example, invokes Class.forName('className')-->
            <!-- <option>-adaptclassstrings</option> -->
            <!-- This option will save all original annotations and etc. Otherwise all we be removed from files.-->
            <!-- 不混淆所有特殊的类-->
            <option>-keepattributes Exceptions,InnerClasses,Signature,Deprecated,
              SourceFile,LineNumberTable,*Annotation*,EnclosingMethod
            </option>
            <!-- This option will save all original names in interfaces (without obfuscate).-->
            <!--
                         <option>-keepnames interface **</option>
            -->
            <!-- This option will save all original methods parameters in files defined in -keep sections,
                 otherwise all parameter names will be obfuscate.-->

            <!--保留参数名字-->
            <option>-keepparameternames</option>
            <!--保留主进程入口-->
            <!--
                        <option>-keep @org.springframework.boot.autoconfigure.SpringBootApplication class * {*;}</option>
            -->
            <!-- <option>-libraryjars **</option> -->
            <!-- This option will save all original class files (without obfuscate) but obfuscate all in domain package.-->
            <!--<option>-keep class !com.slm.proguard.example.spring.boot.domain.** { *; }</option>-->

<!--
            <option>-keep class !com.bmsoft.graph.** { *; }</option>
-->
            <option>-keep class com.bmsoft.graph.config.** { *; }</option>
            <option>-keep class com.bmsoft.graph.LinkGraphApplication { *; }</option>
            <option>-keep class com.bmsoft.graph.mapper.** { *; }</option>
<!--
            <option>-keep class com.bmsoft.graph.auth.filter.** { *; }</option>
-->
            <option>-keep class com.bmsoft.graph.aspect.** { *; }</option>
            <option>-keep class com.bmsoft.graph.domain.** { *; }</option>
            <option>-keep class com.bmsoft.graph.controller.** { *; }</option>
<!--
            <option>-keep interface * extends * { *; }</option>
-->
             <!--##保留枚举成员及方法-->
            <option> -keepclassmembers enum * { *; }</option>
            <option>-keepclassmembers class * {
              <!-- @org.springframework.beans.factory.annotation.Autowired *; -->
              @org.springframework.beans.factory.annotation.Autowired *;
              @org.springframework.beans.factory.annotation.Value *;
              }
            </option>
          </options>
          <libs>
            <!-- Include main JAVA library required.-->
            <lib>${java.home}/lib/rt.jar</lib>
            <lib>${java.home}/lib/jce.jar</lib>
            <!-- <lib>${java.home}/lib/spring-boot-starter-web-1.4.1.RELEASE.jar</lib> -->
          </libs>
        </configuration>
        <dependencies>
          <dependency>
            <groupId>net.sf.proguard</groupId>
            <artifactId>proguard-base</artifactId>
            <version>6.0.3</version>
          </dependency>
        </dependencies>
      </plugin>

这里引用了com.github.wvengen的proguard-maven-plugin插件,使用的proguard-base版本是6.0.3
这里使用java8,因为libs那里照常配置rt.jar,jce.jar,如果是java9的话,则需要换成相应的模块。

另外指定proguard的阶段为package,springboot打包在repackage阶段

<plugin>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-maven-plugin</artifactId>
        <executions>
          <execution>
            <!-- <phase>none</phase> -->
            <goals>
              <goal>repackage</goal>
            </goals>
            <configuration>
              <mainClass>com.bmsoft.graph.LinkGraphApplication</mainClass>
            </configuration>
          </execution>
        </executions>
      </plugin>

bean命名重复异常
由于proguard混淆貌似不能指定混淆的类名在basePackages下面类名混淆后唯一,不同包名经常有a.class,b.class,c.class之类重复的类名,因此spring容器初始化bean的时候会报错。

异常信息如下:

org.springframework.beans.factory.BeanDefinitionStoreException: Failed to parse configuration class [com.example.demo.MvcDemoApplication]; nested exception is org.springframework.context.annotation.ConflictingBeanDefinitionException: Annotation-specified bean name 'a' for bean class [com.example.demo.c.a]  with existing, non-compatible bean definition of same name and class [com.example.demo.b.a]
    at org.springframework.context.annotation.ConfigurationClassParser.parse(ConfigurationClassParser.java:181) ~[spring-context-4.3.10.RELEASE.jar!/:4.3.10.RELEASE]

需要在更改bean 命名:

public class LinkGraphApplication {

  public static class CustomGenerator implements BeanNameGenerator {

    @Override
    public String generateBeanName(BeanDefinition definition, BeanDefinitionRegistry registry) {
      return definition.getBeanClassName();
    }
  }

  public static void main(String[] args) {
    new SpringApplicationBuilder(LinkGraphApplication.class)
        .beanNameGenerator(new CustomGenerator())
        .run(args);
  }
}

运行jar包
nohup java -jar jar包名.jar -&

可能出现的问题

Caused by: org.springframework.beans.factory.UnsatisfiedDependencyException: Error creating bean with name 'springfox.documentation.spring.web.scanners.ApiListingScanner' defined in URL 

项目中引入了swagger 插件,需要在打包的时候,不进行bean的初始化配置@ConditionalOnProperty()

@Configuration
@ConditionalOnProperty(prefix = "swagger", value = {"enable"}, havingValue = "true")
@EnableSwagger2
public class SwaggerConfig {
}

注意
-keep class 类/包.** 表示保留类名

-keepclassmembers class 类/包.**{ *;} 表示保留类下边的所有变量,均不混淆

参考:
springboot项目混淆方案

由于需要将源码打包做代码混淆,选择proguard,开始使用各种问题,各种jar包版本问题,但最终成功了,记录一下,也希望能够帮助大家

在pom中添加代码:

<build>
        <finalName>${artifactId}</finalName>
        <plugins>
            <plugin>
                <groupId>com.github.wvengen</groupId>
                <artifactId>proguard-maven-plugin</artifactId>
                <executions>
                    <execution>
                        <phase>package</phase>
                        <goals><goal>proguard</goal></goals>
                    </execution>
                </executions>
                <configuration>
                    <proguardVersion>6.0.3</proguardVersion>
                    <injar>${project.build.finalName}.jar</injar>
                    <!-- <injar>classes</injar> -->
                    <outjar>${project.build.finalName}.jar</outjar>
                    <obfuscate>true</obfuscate>
                    <options>
                        <!-- 不做收缩(删除注释、未被引用代码)-->
                        <option>-dontshrink</option>
                        <!-- 不做优化(变更代码实现逻辑)-->
                        <option>-dontoptimize</option>
                        <!-- This option will replace all strings in reflections method invocations with new class names.
                             For example, invokes Class.forName('className')-->
                        <!-- <option>-adaptclassstrings</option> -->
                        <!-- This option will save all original annotations and etc. Otherwise all we be removed from files.-->
                        <!-- 不混淆所有特殊的类-->
                        <option>-keepattributes Exceptions,InnerClasses,Signature,Deprecated,
                            SourceFile,LineNumberTable,*Annotation*,EnclosingMethod</option>
                        <!-- This option will save all original names in interfaces (without obfuscate).-->
                        <option>-keepnames interface **</option>
                        <!-- This option will save all original methods parameters in files defined in -keep sections,
                             otherwise all parameter names will be obfuscate.-->
                        <option>-keepparameternames</option>
                        <!-- <option>-libraryjars **</option> -->
                        <!-- This option will save all original class files (without obfuscate) but obfuscate all in domain package.-->
                        <!--<option>-keep class !com.slm.proguard.example.spring.boot.domain.** { *; }</option>-->
                        <option>-keep class !com.ts.** { *; }</option>
                        <option>-keep class com.ts.Application { *; }</option>
                        <option>-keep class com.ts.SwaggerConfig { *; }</option>
                        <option>-keep class com.ts.common.** { *; }</option>
                        <option>-keep class com.ts.aop.** { *; }</option>
                        <option>-keep class com.ts.config.** { *; }</option>
                        <option>-keep class com.ts.sunshine.controller.** { *; }</option>
                        <!-- This option will save all original class files (without obfuscate) in service package-->
                        <!--<option>-keep class com.slm.proguard.example.spring.boot.service { *; }</option>-->
                        <!-- This option will save all original interfaces files (without obfuscate) in all packages.-->
                        <option>-keep interface * extends * { *; }</option>
                        <!-- <option>-keep @org.springframework.stereotype.Service class *</option> -->
                        <!-- This option will save all original defined annotations in all class in all packages.-->
                        <option>-keepclassmembers class * {
                                <!-- @org.springframework.beans.factory.annotation.Autowired *; -->
                                @org.springframework.beans.factory.annotation.Value *;
                            }
                        </option>
                    </options>
                    <libs>
                        <!-- Include main JAVA library required.-->
                        <lib>${java.home}/lib/rt.jar</lib>
                        <!-- <lib>${java.home}/lib/spring-boot-starter-web-1.4.1.RELEASE.jar</lib> -->
                    </libs>
                </configuration>
                <dependencies>
                    <dependency>
                        <groupId>net.sf.proguard</groupId>
                        <artifactId>proguard-base</artifactId>
                        <version>6.0.3</version>
                    </dependency>
                </dependencies>
            </plugin>
            <plugin>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-maven-plugin</artifactId>
                <executions>
                    <execution>
                        <!-- <phase>none</phase> -->
                        <goals>
                            <goal>repackage</goal>
                        </goals>
                        <configuration>
                            <mainClass>com.ts.Application</mainClass>
                        </configuration>
                    </execution>
                </executions>
            </plugin>
        </plugins>
    </build>

注意:

-keep class 类/包.** 表示保留类名

-keepclassmembers class 类/包.**{ *;} 表示保留类下边的所有变量,均不混淆

参考:

springboot2.x+maven+proguard代码混淆

ProGuard代码混淆详细攻略

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 161,780评论 4 369
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 68,424评论 1 305
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 111,397评论 0 254
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,576评论 0 218
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,997评论 3 295
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,945评论 1 224
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 32,107评论 2 317
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,850评论 0 208
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,625评论 1 250
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,804评论 2 253
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,285评论 1 265
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,613评论 3 261
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,291评论 3 242
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,164评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,963评论 0 201
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 36,096评论 2 285
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,886评论 2 278