代码混淆之class-winter

代码混淆之class-winter


郑重声明

class-winter 是本人在学习完 class-final(v1.1.9) 后,仿照class-final进行编写的,部分思路与class-final一致

环境要求

  • 支持jdk8语法的环境即可

    本人构建class-winter时,用的jdk版本为:1.8.0_281

    注:tomcat版本不能低于8(部分小版本低的tomcat8可能也不行),否则可能报错

    org.apache.catalina.LifecycleException: Failed to start component [StandardEngine[Catalina].StandardHost...
    

    注:如果有兼容低版本jdk的需求,可以自己下载master分支代码,进行对应修改

功能与特性

  • 支持war(普通war+可执行war)加密
  • 支持jar(普通jar+可执行jar)加密
  • 支持xml加密(掩耳盗铃版)

加密

  • 方式一:通过maven插件自动加密
    <!--
        class-winter插件
        注:自Maven3.0.3起, 绑定到同一phase的Maven插件将按照pom.xml中声明的顺序执行
        注:此插件最好放置在同一phase的最后执行。
    -->
    <plugin>
        <groupId>com.idea-aedi</groupId>
        <artifactId>class-winter-maven-plugin</artifactId>
        <version>2.7.4</version>
        <!-- 相关配置 -->
        <configuration>
            <!-- <finalName></finalName>-->
            <includePrefix>加密范围</includePrefix>
            <!-- <originJarOrWar></originJarOrWar>-->
            <!-- <excludePrefix></excludePrefix>-->
            <!-- <includeXmlPrefix></includeXmlPrefix>-->
            <!-- <excludeXmlPrefix></excludeXmlPrefix>-->
            <!-- <toCleanXmlChildElementName></toCleanXmlChildElementName>-->
            <!-- <password></password>-->
            <!-- <includeLibs></includeLibs>-->
            <!-- <alreadyProtectedRootDir></alreadyProtectedRootDir>-->
            <!-- <alreadyProtectedLibs></alreadyProtectedLibs>-->
            <!-- <supportFile></supportFile>-->
            <!-- <jvmArgCheck></jvmArgCheck>-->
            <!-- <tips></tips>-->
            <!-- <debug></debug>-->
        </configuration>
        <executions>
            <execution>
                <phase>package</phase>
                <goals>
                    <goal>class-winter</goal>
                </goals>
            </execution>
        </executions>
    </plugin>
    

    注:不必担心信息泄漏问题,使用此方式生成混淆的jar包时,会擦除pom.xml中关于class-winter-plugin的信息。

  • 方式二:通过 class-winter-core.jar 主动加密
    java -jar class-winter-core-2.7.4.jar originJarOrWar=${要加密的项目.jar或.war包} includePrefix=${加密范围} [k3=v3 k4=v4 ...]
    # 对于复杂的参数值,可以使用引号引起来
    # linux
    java -jar class-winter-core-2.7.4.jar k1='v1' k2='v2'
    # windows
    java -jar class-winter-core-2.7.4.jar k1="v1" k2="v2"
    

加密参数

参数是否必填说明示例
originJarOrWar指定要加密的jar/war文件
注:当使用maven插件进行自动加密时,此参数非必填,不填则自动获取。
注:当使用maven插件进行自动加密时,可结合maven相关占位符进行相对定位。如:${project.basedir}/../../your-project.jar
originJarOrWar=/my-project.jar
includePrefix通过前缀匹配的形式定位要加密的class
注:多个通过逗号分割。
includePrefix=com
includePrefix=com,org
excludePrefix通过前缀匹配的形式排除class,不对其加密
注:多个通过逗号分割。
注:excludePrefix优先级高于includePrefix。
excludePrefix=com.example.service,com.example.util.StrUtil.class
includeXmlPrefix通过打出来的包中条目的entryName前缀匹配的形式定位要加密的xml
注:多个通过逗号分割。
注:如果您打出来的加密包是准备作为一个lib包提供给第三方使用的,那么请不要使用此参数,因为解密时是不会解密项目所依赖的lib包中的xml的。
includeXmlPrefix=BOOT-INF/classes/
includeXmlPrefix=BOOT-INF/classes/com/demo/mapper/,BOOT-INF/classes/com/demo/dao/
excludeXmlPrefix通过打出来的包中条目的entryName前缀匹配的形式排除xml,不对其加密
注:多个通过逗号分割。
excludeXmlPrefix=BOOT-INF/classes/com/demo/mapper/
excludeXmlPrefix=BOOT-INF/classes/com/demo/mapper/,BOOT-INF/classes/com/demo/dao/UserDao.xml
toCleanXmlChildElementName加密xml中的哪些一级元素
注:默认值为resultMap,sql,insert,update,delete,select
注:多个通过逗号分割。
toCleanXmlChildElementName=select,delete,resultMap
finalName指定加密后生成的jar包名
注:若finalName与加密的包一致,那么生成的加密后的包会覆盖原来的包。
注:支持相对路径。 比如:../../tmp/my-project 就会在相对目录../../tmp下生成加密包my-project.jar。
finalName=mine-project
password主动指定密码
注:密码不能包含空格和逗号。
password=123456
includeLibs指定将lib包也纳入加密范围内
注:多个通过逗号分割。
注:lib中的class是否会被加密,还得由includePrefix和excludePrefix决定。
includeLibs=a.jar,b.jar
alreadyProtectedRootDir指明已加密lib包所在根目录(,可为空,为空时自动根据当前是jar还是war,去包内对应找lib)
注:当指定此参数时,也会优先去jar/war内部找对应的lib包,找不到时,才会去此参数指定的根目录下找lib包。
注:在一些外置lib的项目中,可能需要用到此参数;如果是内置lib,忽略此参数即可。
注:此参数由2.7.0版本开始支持
alreadyProtectedRootDir=/lib
alreadyProtectedLibs指明项目所依赖的lib中,哪些lib本身就已经是被class-winter加密了的
注:多个通过逗号分割。
注:主要用于处理第三方提供的由class-winter加密了的依赖包的场景。
注:若lib需要密码,那么需要在指定lib的同时通过冒号接上密码。
注:如果lib有密码,那么密码不能包含逗号。
alreadyProtectedLibs=a.jar,b.jar
alreadyProtectedLibs=a.jar,b.jar:pwd123
alreadyProtectedLibs=a.jar:pwd1,b.jar:pwd2
supportFile指定一个加密辅助jar文件(或jar文件所在的目录)
注:当为目录时,该目录(含子孙目录)下的所有jar都会被采集作为辅助文件。
注:主要用于解决因ClassNotFound导致的加密失败问题。
supportFile=/abc.jar
supportFile=/libs
jvmArgCheck设置当启动混淆包时,必须要有的jvm参数
注:多个通过逗号分割。
注:大小写不敏感。
如:通过设置-XX:+DisableAttachMechanism防止运行时dump class,以提高安全性。
jvmArgCheck=-XX:+DisableAttachMechanism,-Xms2048M
tips指定提示语。
注:当直接使用加密后的jar/war时,用到了加密了的类后,会先System.err.println输出此tips,然后System.exit退出程序。
windows示例:tips="请不要直接使用混淆后的jar/war"
linux示例:tips='请不要直接使用混淆后的jar/war'
debug是否开启debug模式debug=true

解密(启动)

通过-javaagent指定代理进行解密启动。

  • jar解密(启动)
    # 假设your-project-encrypted.jar是由class-winter加密后的包,那么你可以这么启动
    java -javaagent:/your-project-encrypted.jar -jar /your-project-encrypted.jar
    # 也可以用class-winter-core-2.7.4.jar
    # java -javaagent:/class-winter-core-2.7.4.jar -jar /your-project-encrypted.jar
    # 或者指定参数
    # java -javaagent:/your-project-encrypted.jar=debug=true,password=pwd12345 -jar /your-project-encrypted.jar
    # 参数可以引起来(linux)
    # java -javaagent:/your-project-encrypted.jar='debug=true,password=pwd12345' -jar /your-project-encrypted.jar
    # 参数可以引起来(windows)
    # java -javaagent:/your-project-encrypted.jar="debug=true,password=pwd12345" -jar /your-project-encrypted.jar
    
  • war解密(启动)

    以Tomcat9为例

    • linux方式一

      编辑tomcat/bin/catalina.sh文件,在最上面加上

      # 如果你有参数, 那么 -javaagent:/class-winter-core-2.7.4.jar=k1=v1,k2=v2
      CATALINA_OPTS="$CATALINA_OPTS -javaagent:/class-winter-core-2.7.4.jar=debug=true";
      export CATALINA_OPTS;
      
    • linux方式二

      在tomcat/bin目录下创建setenv.sh文件,并写上

      # 如果你有参数, 那么 -javaagent:/class-winter-core-2.7.4.jar=k1=v1,k2=v2
      JAVA_OPTS="$JAVA_OPTS -javaagent:/class-winter-core-2.7.4.jar=debug=true";
      export JAVA_OPTS;
      
    • windows方式一

      编辑tomcat/bin/catalina.bat文件,在@echo off后加上catalina参数

      rem 如果你有参数, 那么 -javaagent:D:/class-winter-core-2.7.4.jar=k1=v1,k2=v2
      set CATALINA_OPTS="-javaagent:D:/class-winter-core-2.7.4.jar"
      
    • windows方式二

      在tomcat/bin目录下创建setenv.bat文件,并写上

      rem 如果你有参数, 那么 -javaagent:D:/class-winter-core-2.7.4.jar=k1=v1,k2=v2
      set JAVA_OPTS="-javaagent:D:/class-winter-core-2.7.4.jar"
      
  • IDE解密(启动)

    设置-javaagent参数指定class-winter-core或者任一加密包。有必要的话,再搭配skipProjectPathPrefix或者decryptProjectPathPrefix参数启动即可

    在这里插入图片描述

解密参数

参数是否必填说明示例
password指定解密密码password=pwd123
passwordFromFile从指定文件中读取文本作为解密密码
注:此参数由2.4.0版本开始支持
passwordFromFile=/my-pwd-file.txt
passwordFromShell执行shell文件中的代码,并以其返回值作为解密密码
注:此参数由2.4.0版本开始支持
passwordFromShell=/my-pwd-file.shell
skipProjectPathPrefix是否跳过指定前缀的项目路径(当class-winter解密逻辑试图解析那些进入premain但是非class-winter加密项目时,会因为获取印章失败Obtain project seal fail而停止,此时如果确认这个项目没有加密文件的话,可以使用此参数跳过)
注:此参数由2.6.4版本开始支持
skipProjectPathPrefix=/D:/apache-tomcat-9.0.71/bin/
多个通过___符号拼接:skipProjectPathPrefix=/D:/apache-tomcat-9.0.71/bin/___/D:/jd/classpath/lib/
decryptProjectPathPrefix是否仅解密指定前缀的项目路径(优先级低于skipProjectPathPrefix)
注:此参数由2.6.6版本开始支持
decryptProjectPathPrefix=/D:/apache-tomcat-9.0.71/bin/
多个通过___符号拼接:decryptProjectPathPrefix=/D:/apache-tomcat-9.0.71/bin/___/D:/jd/classpath/lib/
debug是否开启debug模式debug=true

对class-winter本身进行加密

感谢小伙伴Mango对class-winter的优质反馈

  • 背景说明:我们利用class-winter混淆项目时,项目是被混淆了,但是class-winter本身还是明文的,这就留下了一些安全隐患(即:那些不指定密码的混淆,可能会被反向破解)
  • 增强支持:我们可以对class-winter本身进行加密,你可以切换至enhance-*分支,在首页下载被混淆的class-winter进行测试,或者clone代码,install后进行测试
  • 声明:因为在enhance-*分支里对class-winter本身进行加密时用到了名义上的收费工具allatori(实际上有破解版之类的),所以enhance-*分支的代码,本人并未发布至maven官方仓库

将已加密的包提供给客户使用

方案

  • 方案一(适用于encrypted-A依赖于encrypted-B.jar,启动encrypted-A的情况)

    class-winter加密时,使用alreadyProtectedLibs参数,但是要求客户的项目也需要使用class-winter加密(,哪怕客户的项目什么也不需要加密,也要求客户有使用class-winter这个动作)

  • 方案二(适用于A依赖于encrypted-B.jar,启动A的情况)

    利用-Dloader.path或其它方案,将encrypted-B.jar外置,并通过decryptProjectPathPrefixskipProjectPathPrefix来定位encrypted-B.jar

示例

提示:这里以可执行的jar进行的示例,其余的外置方式也是可以的(如:tomcat部署war包解压后,lib包就相当于是外置的)

-Dloader.path实现,将encrypted-B.jar外置到项目包A外

  • 假设客户依赖了我们的加密包

    在这里插入图片描述

  • 客户使用maven插件,打包时将项目代码和依赖的lib分开

    完整pom示例:

    <?xml version="1.0" encoding="UTF-8"?>
    <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
             xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
        <modelVersion>4.0.0</modelVersion>
        <parent>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-parent</artifactId>
            <version>2.7.4</version>
            <relativePath/> <!-- lookup parent from repository -->
        </parent>
        <groupId>com.example</groupId>
        <artifactId>demo</artifactId>
        <version>0.0.1-SNAPSHOT</version>
        <name>demo</name>
        <description>Demo project for Spring Boot</description>
        <properties>
            <java.version>1.8</java.version>
        </properties>
        <dependencies>
            <dependency>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-starter-web</artifactId>
            </dependency>
    
            <dependency>
                <groupId>org.projectlombok</groupId>
                <artifactId>lombok</artifactId>
                <optional>true</optional>
            </dependency>
            <dependency>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-starter-test</artifactId>
                <scope>test</scope>
            </dependency>
    
            <dependency>
                <groupId>com.demo</groupId>
                <artifactId>encrypted-lib-no-pwd</artifactId>
                <scope>system</scope>
                <version>1.0.0</version>
                <systemPath>${pom.basedir}/src/main/resources/lib/encrypted-lib-no-pwd-1.0.0.jar</systemPath>
            </dependency>
        </dependencies>
    
        <build>
            <plugins>
                <plugin>
                    <groupId>org.springframework.boot</groupId>
                    <artifactId>spring-boot-maven-plugin</artifactId>
                    <configuration>
                        <!-- 指定该Main Class为全局的唯一入口 -->
                        <mainClass>com.ideaaedi.demo.DemoApplication</mainClass>
                        <!-- 把systemPath指定的jar包也纳入lib -->
                        <includeSystemScope>true</includeSystemScope>
                        <fork>true</fork>
                        <!-- 设置为ZIP,此模式下spring-boot-maven-plugin会将MANIFEST.MF文件中的Main-Class设置为org.springframework.boot.loader.PropertiesLauncher -->
                        <layout>ZIP</layout>
                        <includes>
                            <include>
                                <groupId>nothing</groupId>
                                <artifactId>nothing</artifactId>
                            </include>
                        </includes>
                    </configuration>
                    <executions>
                        <execution>
                            <goals>
                                <goal>repackage</goal>
                            </goals>
                        </execution>
                    </executions>
                </plugin>
    
                <!-- 通过插件将所有依赖的lib包放到编译后的target/lib目录,并且在打包时候排除内部依赖 -->
                <plugin>
                    <groupId>org.apache.maven.plugins</groupId>
                    <artifactId>maven-dependency-plugin</artifactId>
                    <version>3.2.0</version>
                    <executions>
                        <execution>
                            <id>copy-dependencies</id>
                            <phase>prepare-package</phase>
                            <goals>
                                <goal>copy-dependencies</goal>
                            </goals>
                            <configuration>
                                <outputDirectory>${project.build.directory}/lib</outputDirectory>
                                <overWriteReleases>false</overWriteReleases>
                                <overWriteSnapshots>false</overWriteSnapshots>
                                <overWriteIfNewer>true</overWriteIfNewer>
                                <includeScope>compile</includeScope>
                            </configuration>
                        </execution>
                    </executions>
                </plugin>
            </plugins>
        </build>
    
    </project>
    

    打出来的包:

    在这里插入图片描述

  • 客户可以这样启动项目

    java -Dloader.path=./lib/ -javaagent:./{class-winter.jar} -jar {客户项目.jar}

    在这里插入图片描述

^_^ 本文已经被收录进《程序员成长笔记》 ,笔者JustryDeng

  • 0
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 21
    评论
一、前言 其他资源: web报表工具 http://download.csdn.net/source/2881508 1.1 什么是Jocky? 我们知道,Java是一种跨平台的编程语言,其源码(.java文件)被编译成与平台无关的字节码(.class文件),然后在运行期动态链接。这样,编译后的类文件中将含有符号表,从而使得Java程序很容易被反编译。相信每一个Java开发人员,都曾经用过诸如Jad之类的反编译器,对Java的class 文件进行反编译,从而观察程序的结构与实现细节。如此一来,对于那些需要严格进行知识产权保护的Java应用,如何有效的保护客户的商业投资,是开发人员经常需要面对的问题。 于是就出现了Java混淆编译器,它的作用是打乱class文件中的符号信息,从而使反向工程变得非常困难。 Jocky就是这样一款优秀的Java混淆编译器。 1.2 为什么需要Jocky? 目前业界有不少商业的甚或是开源的混淆编译器,但它们普遍存在一些这样或者那样的问题。一般而言,现有的混淆器都是对编译好的 class文件进行混淆,这样就需要编译和混淆两个步骤。而事实上,并不是所有的符号都需要混淆。如果你开发的是一个类库,或者某些类需要动态装载,那些公共API(或者说:那些被publish出来的API)就必须保留符号不变,只有这样,别人才能使用你的类库。现有的混淆器提供了GUI或脚本的方式来对那些需要保留的符号名称进行配置,但如果程序较大时,配置工作将变得很复杂,而程序一旦修改,配置工作又要重新进行。某些混淆器能够调整字节码的顺序,使反编译更加困难,但笔者经历过混淆之后的程序运行出错的情况。 而Jocky与其它混淆编译器最大的不同之处在于:它是直接从源码上做文章,也就是说编译过程本身就是一个混淆过程。 1.3 Jocky是如何工作的? Jocky混淆编译器是在Sun JDK中提供的Java编译器(javac)的基础上完成的,修改了其中的代码生成过程,对编译器生成的中间代码进行混淆,最后再生成class文件,这样编译和混淆只需要一个步骤就可以完成。另外可以在源程序中插入 符号保留指令 来控制哪些符号需要保留,将混淆过程与开发过程融合在一起,不需要单独的配置。 1.4 Jocky的作用 1.4.1代码混淆 如前文所述,混淆编译是Jocky的首要用途。我们举一个最简单的例子,下面的SimpleBean是未经混淆class文件通过Jad反编译以后获得的源文件: public class SimpleBean implements Serializable { private String name = "myname"; private List myList = null; public void SimpleBean() { myList = new ArrayList(10); } public void foo1() { myList.add("name"); } private void foo2() { } private void writeObject(java.io.ObjectOutputStream out) throws IOException { } } 下面是经Jocky混淆过的类文件,通过Jad反编译后产生的源文件: public class SimpleBean implements Serializable { private String _$2; private List _$1; public SimpleBean() { _$2 = "myname"; this; JVM INSTR new #4 ; JVM INSTR dup ; JVM INSTR swap ; 10; ArrayList(); _$1; } public void foo1() { _$1.add("name"); } private void _$1() { } private void writeObject(ObjectOutputStream objectoutputstream){ throws IOException { } } <Jock
评论 21
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值