ArkGuard字节码混淆原理及功能
术语清单
| 名词 | 释义 |
|---|---|
| HAP | HAP(Harmony Ability Package)是应用安装和运行的基本单元。HAP包是由代码、资源、第三方库、配置文件等打包生成的模块包。 |
| HAR | HAR(Harmony Archive)是静态共享包,通过HAR可以实现多个模块或多个工程共享ArkUI组件、资源等相关代码。通过Static Library创建HAR模块。 |
| HSP | HSP(Harmony Shared Package)是动态共享包,通过HSP可以实现代码和资源的共享。通过Shared Library创建HSP模块。 |
| 本地HAR | 源码形式的HAR模块。 |
| 远程HAR | 构建后打包生成的HAR包。 |
| 本地HSP | 源码形式的HSP模块。 |
| 远程HSP | 构建后打包生成的HSP包。 |
| 三方库 | 由第三方开发并发布的库,发布到OHPM中心仓,供其他应用使用。 |
| 名称混淆 | 将代码中的函数名、类名、文件名等标识符修改为无意义的名称。 |
混淆生效范围
适用语言
ArkGuard支持ArkTS/TS/JS语言,JSON仅支持文件名混淆,不支持C/C++、资源文件等。
混淆能力
ArkGuard提供的是面向方舟字节码的基础名称混淆:在可选规则下,对类、接口、枚举、函数、类成员方法、变量、对象属性名、源文件名等标识符进行重命名,并可通过保留选项配置白名单;各选项的覆盖范围与例外说明见下文混淆选项。函数参数名不参与字节码名称混淆。
不支持以下高级混淆能力:控制流混淆(如控制流平坦化、虚假分支插入等)、数据混淆(如对常量、字面量或内存数据布局做加密与变换)、指令替换或插入无关指令、虚拟机保护/加壳等不以标识符重命名为主要手段的保护方式。
混淆能力局限性
1.语言的限制
代码混淆工具在处理不同编程语言时,其类型分析机制、混淆策略和执行效率都会因目标语言的特性而呈现差异。以业界常用的ProGuard为例,其主要面向Java这类强类型语言进行混淆。由于强类型语言具有严格的类型系统,每个类型都有明确的定义来源。使得混淆过程中的类型关系追踪和处理更为精确,从而大幅减少了需要配置保留规则的场景。
相比之下,ArkGuard混淆工具主要针对JS、TS和ArkTS语言。JS支持运行时动态修改对象、函数,而混淆是在编译阶段进行的静态处理,可能导致混淆后的名称在运行时无法被正确解析,进而引发运行时异常。TS和ArkTS虽然引入了静态类型系统,但采用了结构性类型机制,即具有相同结构的不同命名类型会被视为等价类型。因此,在TS和ArkTS中仍然无法追溯类型的确切来源。
基于这些特性,使用ArkGuard时需要对更多的语法场景进行白名单配置,同时,ArkGuard采用全局生效的属性保留机制,根据白名单统一保留所有同名属性,而无法支持针对特定类型进行精确保留配置。
具体而言,可以参考以下示例:
假设ArkGuard支持配置指定类型的白名单,配置类A1作为白名单,类A1的属性prop1在白名单中,而A2中的prop1属性不在白名单中。此时,a2作为参数被传入test函数中,调用prop1属性时会导致功能异常。
// example.ts
// 混淆前:
class A1 {
prop1: string = '';
}
class A2 {
prop1: string = '';
}
function test(input: A1) {
console.info(input.prop1);
}
let a2 = new A2();
a2.prop1 = 'prop a2';
test(a2);
// 混淆后
class A1 {
prop1: string = '';
}
class A2 {
a: string = '';
}
function test(input: A1) {
console.info(input.prop1);
}
let a2 = new A2();
a2.a = 'prop a2';
test(a2);
综上所述,开发者应了解这种语言差异带来的混淆效果差异,并尽量使用不重复的名称,以使在各种场景下的混淆效果更好。
2.安全保证的有限性
与其他代码混淆工具一样,混淆只能在一定程度上增加逆向工程的难度,并不能真正阻止逆向工程。
并且,由于ArkGuard混淆工具仅支持基础混淆能力,开发者不应只依赖ArkGuard来保证应用的安全性,对于源码安全有高要求的开发者,应考虑使用应用加密、第三方安全加固等安全措施来保护代码。
混淆机制及流程
下图为应用编译的简要流程图:

开发者可以在模块的build-profile.json5配置文件中开启混淆功能,详细参考字节码混淆开启指南,从而在编译打包的过程中自动对abc进行混淆处理。
混淆过程中,首先读取混淆开关。在开关开启的情况下,解析混淆配置文件,并依据混淆规则合并策略合并混淆规则。然后按照混淆规则对生成的abc文件进行字节码混淆,最后将混淆后的中间文件落盘至build目录。开发者可以通过build目录中混淆后的产物,确认混淆效果。
在使用混淆功能前,建议开发者先通过文档了解混淆选项的能力与混淆选项所需要保留白名单的场景,再根据开发需求选择对应的混淆功能。
混淆选项
已有混淆选项汇总
| 功能 | 选项 |
|---|---|
| 关闭混淆 | -disable-obfuscation |
| 属性名称混淆 | -enable-property-obfuscation |
| 字符串属性名称混淆 | -enable-string-property-obfuscation |
| 顶层作用域名称混淆 | -enable-toplevel-obfuscation |
| 导入导出名称混淆 | -enable-export-obfuscation |
| 文件名混淆 | -enable-filename-obfuscation |
| 代码压缩 | -compact |
| console打印删除 | -remove-log |
| 名称缓存输出 | -print-namecache |
| 名称缓存复用 | -apply-namecache |
| 合并依赖模块选项 | -enable-lib-obfuscation-options |
| 开启字节码混淆 | -enable-bytecode-obfuscation |
| 开启字节码混淆调试能力 | -enable-bytecode-obfuscation-debugging |
-disable-obfuscation
关闭所有混淆。
若配置该选项,则默认混淆以及所有已配置的混淆、保留选项的功能将全部失效。和关闭模块中build-profile.json5文件中混淆功能作用一致。
-enable-property-obfuscation
开启属性名称混淆,效果如下:
// test.ts
// 混淆前:
class TestA {
static prop1: number = 0;
}
TestA.prop1;
// 混淆后:
class TestA {
static i: number = 0;
}
TestA.i;
若配置该选项,那么所有的属性名都会被混淆,除了下面场景:
-
在未开启-enable-export-obfuscation选项的情况下,被import/export直接导入或导出的类、对象的属性名不会被混淆。例如下面例子中的属性名data不会被混淆。
// example.tsexport class MyClass01 {data1: string;} -
ArkUI组件中的属性名不会被混淆。例如下面例子中的message和data不会被混淆。
// example.ets@Component struct MyExample {@State message: string = "hello";data: number[] = [];build() {}} -
被保留选项指定的属性名不会被混淆。
-
SDK API列表中的属性名不会被混淆。SDK API列表是构建时从SDK中自动提取出来的一个名称列表,其缓存文件为systemApiCache.json,路径为工程目录下build/default/cache/{...}/release/obfuscation中。
-
字符串字面量属性名不会被混淆。例如下面例子中的firstName和personAge不会被混淆。
let person = {"firstName": "abc"};person["personAge"] = 22; -
注解成员名不会被混淆。例如下面例子中的authorName和revision不会被混淆。
@interface MyAnnotation1 {authorName: string;revision: number;}
-enable-string-property-obfuscation
开启字符串属性混淆,仅在已开启属性混淆的基础上生效。
若想混淆字符串字面量属性名,需要在已配置-enable-property-obfuscation的基础上使用。例如:
-enable-property-obfuscation
-enable-string-property-obfuscation
根据上述配置,下面例子中的"firstName"和"personAge"混淆效果如下:
let person = {"firstName": "abc"};
person["personAge"] = 22;
// 混淆后:
let person = {"a": "abc"};
person["b"] = 22;
使用该选项时,需要注意以下事项:
1. 如果字符串属性名包含特殊字符(除了a-z、A-Z、0-9、_之外的字符),例如let obj = {"\n": 123, "": 4, " ": 5},建议不要开启-enable-string-property-obfuscation选项,因为可能无法通过保留选项来指定保留这些名字。
2. SDK API的属性白名单中不包含声明文件中使用的字符串常量值,例如示例中的字符串'ohos.want.action.home'未包含在属性白名单中:
// SDK API文件@ohos.app.ability.wantConstant片段:
export enum Params {
ACTION_HOME = 'ohos.want.action.home'
}
// 开发者源码示例:
let params = obj1['ohos.want.action.home'];
因此在开启了-enable-string-property-obfuscation选项时,如果想保留代码中使用的SDK API字符串常量的属性不被混淆,例如obj['ohos.want.action.home'],那么需要使用-keep-property-name选项保留。
-enable-toplevel-obfuscation
开启顶层作用域名称混淆,效果如下:
// 混淆前:
let count = 0;
// 混淆后:
let s = 0;
若配置该选项,那么所有的顶层作用域的名称都会被混淆,除了下面场景:
- 在未开启-enable-export-obfuscation选项的情况下,被import/export直接导入或导出的名称不会被混淆。
- 当前文件找不到声明的名称不会被混淆。
- 被保留选项指定的顶层作用域名称不会被混淆。
- SDK API列表中的顶层作用域名称不会被混淆。
-enable-export-obfuscation
开启直接导入或导出的名称混淆,效果如下:
// 混淆前:
namespace ns {
export type customT = string;
}
// 混淆后:
namespace ns {
export type h = string;
}
若配置该选项,那么非顶层作用域中导入或导出的名称会被混淆。
若想混淆顶层作用域中导入或导出的名称,需要在已配置-enable-toplevel-obfuscation的基础上使用。
若想混淆导入或导出的属性名,需要在已配置-enable-property-obfuscation的基础上使用。 开启此选项时,以下特殊场景不会被混淆:
- 远程HAR(真实路径在oh_modules中的包)中导出的名称和属性名不会被混淆。
- 被保留选项指定的名称与属性名不会被混淆。
- SDK API列表中的名称不会被混淆。
-enable-filename-obfuscation
开启文件/文件夹名称混淆,效果如下:
// 混淆前:
import * as m from '../test1/test2';
import { foo } from '../test1/test2';
// ...
const module = import('../test1/test2');
// 混淆后:
import * as m from '../a/b';
import { foo } from '../a/b';
const module = import('../a/b');
若配置该选项,那么所有的文件/文件夹名称都会被混淆,除了下面场景:
- oh-package.json5文件中'main'、'types'字段配置的文件/文件夹名称不会被混淆。
- 模块内module.json5文件中'srcEntry'字段配置的文件/文件夹名称不会被混淆。
- 被-keep-file-name指定的文件/文件夹名称不会被混淆。
- 非ECMAScript模块引用方式,不会被混淆(例如:const module = require('./module'))。
- 非路径引用方式,例如import module from 'json5'中的json5不会被混淆。
由于系统会在应用运行时加载某些指定的文件,针对这类文件,开发者需要手动在-keep-file-name选项中配置相应的白名单,防止指定文件被混淆,导致运行失败。
编译入口、Ability组件、Worker多线程,这三种不能混淆的文件名在DevEco Studio 5.0.3.500版本已被自动收集进白名单中,无需再手动配置,其它不能混淆文件名的场景仍需开发者手动配置。
-compact
删除不必要的空格符和所有的换行符。
若配置该选项,所有代码会被压缩到一行。效果如下:
// test.ts
// 混淆前:
class TestA {
static prop1: number = 0;
}
TestA.prop1;
// 混淆后:
class TestA { static prop1: number = 0; } TestA.prop1;
release模式构建的应用栈信息仅包含代码行号,不包含列号,因此-compact功能开启后无法依据报错栈中的行号定位到源码具体位置。
-remove-log
删除对console.*语句的调用,要求console.*语句返回值未被调用,效果如下:
// 混淆前:
if (flag) {
console.info("hello");
}
// 混淆后:
if (flag) {
}
若配置该选项,以下场景中的console.*语句会被删除:
-
文件顶层的调用
console.info("in tolevel"); -
代码块中的调用
function foo1() {console.info('in block');} -
module或namespace中的调用
// example.tsnamespace ns {console.info('in ns');} -
switch语句中的调用
例如
switch (value) {case 1:console.info("in switch case");break;default:console.info("default");}
-print-namecache
将名称缓存保存到指定的文件路径filepath中,名称缓存包含名称混淆前后的映射。其中,filepath为必选参数,支持相对路径和绝对路径,相对路径的起始位置为混淆配置文件的当前目录。filepath参数中的文件名请以.json为后缀。
例如:
-print-namecache
./customCache/nameCache.json
每次全量构建工程时都会生成新的namecache.json文件,因此开发者每次发布新版本时都要注意保存一个该文件的副本。
-apply-namecache
复用指定的名称缓存文件filepath。其中,filepath为必选参数,支持相对路径和绝对路径,相对路径的起始位置为混淆配置文件的当前目录。filepath参数中的文件名请以.json为后缀。
该选项应该在增量编译场景中被使用。开启该选项后,名称将会被混淆成缓存映射对应的名称,若找不到对应的缓存,则会被混淆成新的随机名称。
例如:
-apply-namecache
./customCache/nameCache.json
默认情况下,DevEco Studio会在临时的缓存目录中保存缓存文件,并且在增量编译场景中自动应用该缓存文件。
缓存目录:build/default/cache/{...}/release/obfuscation。
-enable-lib-obfuscation-options
配置此开关后,依赖模块的混淆选项将被合并到当前编译模块的混淆配置中。
默认情况下,生效的混淆配置为当前编译模块的混淆配置与依赖模块的保留选项的合并结果。
启用该开关后,生效的混淆配置为当前编译模块的混淆配置与依赖模块的混淆配置的合并结果。
混淆规则合并逻辑参考混淆规则合并策略。
-enable-bytecode-obfuscation
字节码混淆控制开关。默认不开启。
-enable-bytecode-obfuscation-debugging
控制字节码混淆是否输出调试信息,开启后会生成混淆日志,请参考混淆效果,默认不开启。
需要在已配置-enable-bytecode-obfuscation的基础上使用。
保留选项
已有保留选项汇总
| 功能 | 选项 |
|---|---|
| 指定保留属性名称 | -keep-property-name |
| 指定保留顶层作用域或导入导出元素名称 | -keep-global-name |
| 指定保留文件/文件夹名称 | -keep-file-name |
| 指定保留声明文件中的所有名称 | -keep-dts |
| 指定保留源码文件中的所有名称 | -keep |
-keep-property-name
指定想保留的属性名,支持使用名称类通配符。按如下方式进行配置,表示保留名称为age、firstName和lastName的属性:
-keep-property-name
age
firstName
lastName
使用该选项时,需要注意以下事项:
1. 该选项在开启-enable-property-obfuscation时生效。
2. 属性白名单作用于全局。即代码中出现多个重名属性,只要与-keep-property-name配置白名单名称相同,均不会被混淆。
哪些属性名应该被保留?
1.如果代码中通过字符串拼接、变量访问或使用defineProperty方法来定义对象属性,则这些属性名应被保留。例如:
// example.js
let obj = {x0: 0, x1: 0, x2: 0};
for (let i = 0; i <= 2; i++) {
console.info(obj['x' + i]); // x0, x1, x2应该被保留。
}
Object.defineProperty(obj, 'y', {}); // y应该被保留。
Object.getOwnPropertyDescriptor(obj, 'y'); // y应该被保留。
console.info(obj.y);
obj.s1 = 'a';
let key = 's1';
console.info(obj[key]); // key对应的变量值s应该被保留。
obj.t1 = 'b';
console.info(obj['t' + '1']); // t1应该被保留。
对于如下的字符串常量形式的属性调用,可以选择性保留:
// 混淆配置:
// -enable-property-obfuscation
// -enable-string-property-obfuscation
obj2.t = "0";
console.info(obj2['t']); // 此时,'t'会被正确混淆,t可以选择性保留。
obj2['v'] = "0";
console.info(obj2['v']); // 此时,'v'会被正确混淆,v可以选择性保留。
2.对于间接导出的场景,例如export MyClass和let a = MyClass; export {a};,如果不想混淆它们的属性名,那么需要使用保留选项来保留这些属性名。另外,对于直接导出的类或对象的属性的属性名,例如下面例子中的firstName和personAge,如果不想混淆它们,那么也需要使用保留选项来保留这些属性名。
// myclass.ts
export class MyClass02 {
person = {firstName: "123", personAge: 100};
}
3.在ArkTS/TS/JS文件中使用so库的API(例如示例中的foo)时,需手动保留API名称。
export const add: (a: number, b: number) => number;
// test.ets
import testNapi from 'libentry.so'
// ...
testNapi.add(2, 3); // add需要保留,示例如:-keep-property-name foo。
4.JSON数据解析及对象序列化时,需要保留使用到的字段,例如:
// 示例JSON文件结构(test.json):
/*
* {
* "jsonProperty": "value",
* "otherProperty": "value2"
* }
*/
import jsonData from './test.json';
// ...
let jsonProp = jsonData.jsonProperty; // jsonProperty应该被保留
class jsonTest {
prop1: string = '';
prop2: number = 0;
}
let obj = new jsonTest();
const jsonStr = JSON.stringify(obj); // prop1 和 prop2 会被混淆,应该被保留
5.使用到的数据库相关的字段,需要手动保留。例如,数据库键值对类型(ValuesBucket)中的属性:
const valueBucket: ValuesBucket = {
ID1: 'ID1', // ID1应该被保留。
NAME1: 'jack', // NAME1应该被保留。
AGE1: 20, // AGE1应该被保留。
SALARY1: 100 // SALARY1应该被保留。
}
6.源码中自定义装饰器修饰了成员变量、成员方法、参数,同时其源码编译的中间产物为js文件时(如编译release源码HAR或者源码包含@ts-ignore、@ts-nocheck),这些装饰器所在的成员变量/成员方法名称需要被保留。这是由于ts高级语法特性转换为js标准语法时,将上述装饰器所在的成员变量/成员方法名称硬编码为字符串常量。
示例:
function CustomDecorator(target: Object, propertyKey: string) {}
function MethodDecorator(target: Object, propertyKey: string, descriptor: PropertyDescriptor) {}
function ParamDecorator(target: Object, propertyKey: string, parameterIndex: number) {}
class A {
// 1.成员变量装饰器。
@CustomDecorator
propertyName1: string = "" // propertyName1 需要被保留。
// 2.成员方法装饰器。
@MethodDecorator
methodName1() {} // methodName1 需要被保留。
// 3.方法参数装饰器。
methodName2(@ParamDecorator param: string): void {} // methodName2 需要被保留。
}
-keep-global-name
指定要保留的顶层作用域及导入和导出元素的名称,支持使用名称类通配符。可按如下方式进行配置:
-keep-global-name
Person
printPersonName
namespace中导出的名称也可以通过-keep-global-name选项保留,示例如下:
// example.ts
export namespace Ns {
export const myAge = 18 // -keep-global-name myAge 保留变量myAge。
export function myFunc() {} // -keep-global-name myFunc 保留函数myFunc。
}
-keep-global-name指定的白名单作用于全局。即代码中出现多个顶层作用域名称或者导出名称,只要与-keep-global-name配置的白名单名称相同,均不会被混淆。
哪些顶层作用域的名称应该被保留?
1.在JavaScript中全局变量是globalThis的属性。如果在代码中使用globalThis去访问全局变量,那么该变量名应该被保留。
示例:
var a = 0;
console.info(globalThis.a); // a 应该被保留。
function foo2(){}
globalThis.foo2(); // foo2 应该被保留。
var c = "0";
console.info(c); // c 可以被正确地混淆。
function bar(){}
bar(); // bar 可以被正确地混淆。
class MyClass {}
let d = new MyClass(); // MyClass 可以被正确地混淆。
2.当以命名导入的方式导入 so 库的 API时,若同时开启-enable-toplevel-obfuscation和-enable-export-obfuscation选项,需要手动保留API的名称。
// src/main/cpp/types/libentry/Index.d.ts。
declare function testNapi2(): void;
declare function testNapi3(): void;
// example.ets
import { testNapi2, testNapi3 as myNapi } from 'libentry.so' // testNapi2 和 testNapi3 应该被保留。
// ...
testNapi2();
myNapi();
-keep-file-name
指定要保留的文件/文件夹的名称(不需要写文件后缀),支持使用名称类通配符。例如:
-keep-file-name
index
entry
哪些文件名应该被保留?
1.在使用require引入文件路径时,由于ArkTS不支持CommonJS语法,因此这种情况下路径应该被保留。
// example.js
const module1 = require('./file1'); // file1 应该被保留。
2.对于动态导入的路径名,由于无法识别import函数中的参数是否为路径,因此这种情况下路径应该被保留。
// file2.ts
export function foo () {}
// main.ts
const moduleName = './file2' // moduleName对应的路径名file2应该被保留。
const module2 = import(moduleName)
3.在使用跨包路由进行路由跳转时,传递给动态路由的路径应该被保留。动态路由提供系统路由表和自定义路由表两种方式。若采用自定义路由表进行跳转,配置白名单的方式与上述第二种动态引用场景一致。而若采用系统路由表进行跳转,则需要将模块下resources/base/profile/route_map.json5文件中pageSourceFile字段对应的路径添加到白名单中。
{
"routerMap": [
{
"name": "PageOne",
"pageSourceFile": "src/main/ets/pages/directory/PageOne.ets", // 路径都应该被保留
"buildFunction": "PageOneBuilder",
"data": {
"description" : "this is PageOne"
}
}
]
}
-keep-dts
指定路径filepath的.d.ets文件中的名称(例如类名、属性名等)会被添加至-keep-global-name和-keep-property-name白名单中。请注意,filepath仅支持绝对路径,并且可以指定为一个目录。在这种情况下,该目录中所有.d.ets文件中的名称都将被保留。
-keep
保留指定相对路径filepath中的所有名称(例如类名、属性名等)不被混淆。filepath可以是文件与文件夹,若是文件夹,则文件夹下的文件及子文件夹中文件都不混淆。
filepath仅支持相对路径,./与../为相对于混淆配置文件所在目录,支持使用路径类通配符。
-keep
./src/main/ets/fileName.ts // fileName.ts中的名称不混淆
../folder // folder目录下文件及子文件夹中的名称都不混淆
../oh_modules/json5 // 引用的三方库json5里所有文件中的名称都不混淆
使用该选项时,需要注意以下事项:
1. 被-keep filepath所保留的文件,其依赖链路上的文件中导出名称及其属性都会被保留。
2. 该功能不影响文件名混淆-enable-filename-obfuscation的功能。
保留选项支持的通配符
名称类通配符
名称类通配符使用方式如下:
| 通配符 | 含义 | 示例 |
|---|---|---|
| ? | 匹配任意单个字符 | "AB?"能匹配"ABC"等,但不能匹配"AB" |
| * | 匹配任意数量的任意字符 | "*AB*"能匹配"AB"、"aABb"、"cAB"、"ABc"等 |
使用示例:
保留所有以a开头的属性名称:
-keep-property-name
a*
保留所有单个字符的属性名称:
-keep-property-name
?
保留所有属性名称:
-keep-property-name
*
路径类通配符
路径类通配符使用方式如下:
| 通配符 | 含义 | 示例 |
|---|---|---|
| ? | 匹配任意单个字符,除了路径分隔符/。 | "../a?"能匹配"../ab"等,但不能匹配"../a/"。 |
| * | 匹配任意数量的任意字符,除了路径分隔符/。 | "../a*/c"能匹配"../ab/c",但不能匹配"../ab/d/s/c"。 |
| ** | 匹配任意数量的任意字符。 | "../a**/c"能匹配"../ab/c",也能匹配"../ab/d/s/c"。 |
| ! | 表示非,只能写在某个路径最前端,用来排除用户配置的白名单中已有的某种情况。 | "!../a/b/c.ets"表示除"../a/b/c.ets"以外。 |
使用示例:
表示路径../a/b/中所有文件夹(不包含子文件夹)中的c.ets文件不会被混淆:
-keep
../a/b/*/c.ets
表示路径../a/b/中所有文件夹(包含子文件夹)中的c.ets文件不会被混淆:
-keep
../a/b/**/c.ets
表示路径../a/b/中,除了c.ets文件以外的其它文件都不会被混淆。其中,!不可单独使用,只能用来排除白名单中已有的情况:
-keep
../a/b/
!../a/b/c.ets
表示路径../a/中的所有文件(不包含子文件夹)不会被混淆:
-keep
../a/*
表示路径../a/下的所有文件夹(包含子文件夹)中的所有文件不会被混淆:
-keep
../a/**
表示模块内的所有文件不会被混淆:
-keep
./**
使用通配符时,需要注意以下事项:
1. 以上选项,不支持配置通配符*、?、!作其它含义使用。 例如:
class A {
'*'= 1
}
-keep-property-name
*
此时*表示匹配任意数量的任意字符,配置效果为所有属性名称都不混淆,而不是只有*属性不被混淆。
2. -keep选项中只允许使用/路径格式,不支持\或\。
混淆规则合并策略
在编译一个模块时,默认情况下,生效的混淆规则为当前编译模块的混淆规则与依赖模块混淆规则的合并结果,具体规则如下:
当前编译模块混淆规则
指当前模块配置文件build-profile.json5中arkOptions.obfuscation.ruleOptions.files字段指定的混淆配置文件内容。
依赖模块混淆规则
根据依赖模块的类型,混淆规则分为以下两个来源:
-
本地HAR/HSP模块
指该模块配置文件build-profile.json5中arkOptions.obfuscation.consumerFiles字段指定的混淆配置文件内容。
-
远程HAR/HSP包
指该远程HAR/HSP包中obfuscation.txt文件内容。
当构建HAP、HSP和HAR的时候,最终的混淆规则是下列文件的合并:
- 当前构建模块的ruleOptions.files属性。
- 依赖的本地HSP的consumerFiles属性。
- 依赖的本地HAR的consumerFiles属性。
- 依赖的远程HAR和远程HSP中的obfuscation.txt文件。
如果构建的是HAR,生成的远程HAR中的obfuscation.txt是下列文件的合并:
- 自身的consumerFiles属性。
- 依赖的本地HSP的consumerFiles属性。
- 依赖的本地HAR的consumerFiles属性。
- 依赖的远程HAR和远程HSP中的obfuscation.txt文件。
如果构建的是HSP,生成的远程HSP中的obfuscation.txt仅包含自身的consumerFiles属性。如果构建的是HAP,则不会生成obfuscation.txt。
混淆规则合并逻辑
混淆选项:使用或运算进行合并,即开关选项只要在参与合并的任意一个规则文件中存在,最终的合并结果中就会包含该开关选项。
保留选项:合并时,对于白名单选项,其内容取并集。
- 如果当前编译模块混淆配置未包含-enable-lib-obfuscation-options选项:合并对象为当前模块的所有混淆规则与依赖模块混淆规则中的保留选项。
- 如果当前编译模块混淆配置包含-enable-lib-obfuscation-options选项:合并对象为当前模块的所有混淆规则与依赖模块的所有混淆规则。
当consumerFiles指定的混淆配置文件中包含以下混淆规则时,这些混淆规则会被合并到远程HAR和远程HSP的obfuscation.txt文件中,而其他混淆规则不会。
// 混淆选项
-enable-property-obfuscation
-enable-string-property-obfuscation
-enable-toplevel-obfuscation
-remove-log
// 保留选项
-keep-property-name
-keep-global-name
HSP和HAR中混淆注意事项
- 如果consumerFiles指定的混淆配置文件中包含上述混淆选项,当其他模块依赖该模块的时候,这些混淆选项会与主模块的混淆规则合并,从而影响主模块。因此不建议开发者在consumer-rules.txt文件中配置混淆选项,建议仅配置保留选项。
- 如果在consumerFiles指定的混淆配置文件中添加-keep-dts选项,会被转换成-keep-global-name和-keep-property-name。