当前位置: 首页 > news >正文

base64魔改算法 | jsvmp日志分析并还原

前言

上一篇我们讲了标准 base64 算法还原,为了进一步学习 base64 算法特点,本文将结合 jsvmp 日志,实战还原出 base64 魔改算法。

为了方便大家学习,我将入参和上篇文章一样,入参为 Hello, World!

插桩

在js代码中,找到 运算符 位置以及 apply 插桩点,结合插桩日志框架,代码插桩如下:

jsvmp源码:https://t.zsxq.com/B5aRh

base64魔改算法 | jsvmp日志分析并还原

日志分析

代码执行后,日志如下:

1:  函数: [Function: charCodeAt] 调用者: Hello, World! 参数: [ 0 ] 结果: 722:  函数: [Function: push] 调用者: [ 72 ] 参数: [ 72 ] 结果: 13:  函数: [Function: charCodeAt] 调用者: Hello, World! 参数: [ 1 ] 结果: 1014:  函数: [Function: push] 调用者: [ 72, 101 ] 参数: [ 101 ] 结果: 25:  函数: [Function: charCodeAt] 调用者: Hello, World! 参数: [ 2 ] 结果: 1086:  函数: [Function: push] 调用者: [ 72, 101, 108 ] 参数: [ 108 ] 结果: 37:  函数: [Function: charCodeAt] 调用者: Hello, World! 参数: [ 3 ] 结果: 1088:  函数: [Function: push] 调用者: [ 72, 101, 108, 108 ] 参数: [ 108 ] 结果: 49:  函数: [Function: charCodeAt] 调用者: Hello, World! 参数: [ 4 ] 结果: 11110:  函数: [Function: push] 调用者: [ 72, 101, 108, 108, 111 ] 参数: [ 111 ] 结果: 511:  函数: [Function: charCodeAt] 调用者: Hello, World! 参数: [ 5 ] 结果: 4412:  函数: [Function: push] 调用者: [ 72, 101, 108, 108, 111, 44 ] 参数: [ 44 ] 结果: 613:  函数: [Function: charCodeAt] 调用者: Hello, World! 参数: [ 6 ] 结果: 3214:  函数: [Function: push] 调用者: [72, 101, 108, 108,111,  44,  32
] 参数: [ 32 ] 结果: 715:  函数: [Function: charCodeAt] 调用者: Hello, World! 参数: [ 7 ] 结果: 8716:  函数: [Function: push] 调用者: [72, 101, 108, 108,111,  44,  32,  87
] 参数: [ 87 ] 结果: 817:  函数: [Function: charCodeAt] 调用者: Hello, World! 参数: [ 8 ] 结果: 11118:  函数: [Function: push] 调用者: [72, 101, 108, 108,111,  44,  32,  87,111
] 参数: [ 111 ] 结果: 919:  函数: [Function: charCodeAt] 调用者: Hello, World! 参数: [ 9 ] 结果: 11420:  函数: [Function: push] 调用者: [72, 101, 108, 108,111,  44,  32,  87,111, 114
] 参数: [ 114 ] 结果: 1021:  函数: [Function: charCodeAt] 调用者: Hello, World! 参数: [ 10 ] 结果: 10822:  函数: [Function: push] 调用者: [72, 101, 108, 108,111,  44,  32,  87,111, 114, 108
] 参数: [ 108 ] 结果: 1123:  函数: [Function: charCodeAt] 调用者: Hello, World! 参数: [ 11 ] 结果: 10024:  函数: [Function: push] 调用者: [72, 101, 108, 108, 111,44,  32,  87, 111, 114,108, 100
] 参数: [ 100 ] 结果: 1225:  函数: [Function: charCodeAt] 调用者: Hello, World! 参数: [ 12 ] 结果: 3326:  函数: [Function: push] 调用者: [72, 101, 108, 108, 111,44,  32,  87, 111, 114,108, 100,  33
] 参数: [ 33 ] 结果: 1327:  0 + 3 ====> 328:  函数: [Function: slice] 调用者: [72, 101, 108, 108, 111,44,  32,  87, 111, 114,108, 100,  33
] 参数: [ 0, 3 ] 结果: [ 72, 101, 108 ]29:  函数: [Function: slice] 调用者: [ 72, 101, 108 ] 参数: [] 结果: [ 72, 101, 108 ]30:  3 - 3 ====> 031:  72 & 255 ====> 7232:  101 & 255 ====> 10133:  108 & 255 ====> 10834:  72 << 16 ====> 471859235:  101 << 8 ====> 2585636:  4718592 | 25856 ====> 474444837:  4744448 | 108 ====> 474455638:  4744556 >> 18 ====> 1839:  18 & 63 ====> 1840:  函数: [Function: charAt] 调用者: Dkdpgh4ZKsQB80/Mfvw36XI1R25+WUAlEi7NLboqYTOPuzmFjJnryx9HVGcaStCe= 参数: [ 18 ] 结果: w
第 41:  4744556 >> 12 ====> 115842:  1158 & 63 ====> 643:  函数: [Function: charAt] 调用者: Dkdpgh4ZKsQB80/Mfvw36XI1R25+WUAlEi7NLboqYTOPuzmFjJnryx9HVGcaStCe= 参数: [ 6 ] 结果: 444:  4744556 >> 6 ====> 7413345:  74133 & 63 ====> 2146:  函数: [Function: charAt] 调用者: Dkdpgh4ZKsQB80/Mfvw36XI1R25+WUAlEi7NLboqYTOPuzmFjJnryx9HVGcaStCe= 参数: [ 21 ] 结果: X47:  4744556 & 63 ====> 4448:  函数: [Function: charAt] 调用者: Dkdpgh4ZKsQB80/Mfvw36XI1R25+WUAlEi7NLboqYTOPuzmFjJnryx9HVGcaStCe= 参数: [ 44 ] 结果: u
第 49:  函数: [Function: apply] 调用者: [Function: t4] 参数: [ null, [ 72, 101, 108 ] ] 结果: [ 'w', '4', 'X', 'u' ]50:   + w ====> w
第 51:  w + 4 ====> w4
第 52:  w4 + X ====> w4X
第 53:  w4X + u ====> w4Xu
第 54:  0 + 3 ====> 355:  3 + 3 ====> 656:  函数: [Function: slice] 调用者: [72, 101, 108, 108, 111,44,  32,  87, 111, 114,108, 100,  33
] 参数: [ 3, 6 ] 结果: [ 108, 111, 44 ]57:  函数: [Function: slice] 调用者: [ 108, 111, 44 ] 参数: [] 结果: [ 108, 111, 44 ]58:  3 - 3 ====> 059:  108 & 255 ====> 10860:  111 & 255 ====> 11161:  44 & 255 ====> 4462:  108 << 16 ====> 707788863:  111 << 8 ====> 2841664:  7077888 | 28416 ====> 710630465:  7106304 | 44 ====> 710634866:  7106348 >> 18 ====> 2767:  27 & 63 ====> 2768:  函数: [Function: charAt] 调用者: Dkdpgh4ZKsQB80/Mfvw36XI1R25+WUAlEi7NLboqYTOPuzmFjJnryx9HVGcaStCe= 参数: [ 27 ] 结果: +69:  7106348 >> 12 ====> 173470:  1734 & 63 ====> 671:  函数: [Function: charAt] 调用者: Dkdpgh4ZKsQB80/Mfvw36XI1R25+WUAlEi7NLboqYTOPuzmFjJnryx9HVGcaStCe= 参数: [ 6 ] 结果: 472:  7106348 >> 6 ====> 11103673:  111036 & 63 ====> 6074:  函数: [Function: charAt] 调用者: Dkdpgh4ZKsQB80/Mfvw36XI1R25+WUAlEi7NLboqYTOPuzmFjJnryx9HVGcaStCe= 参数: [ 60 ] 结果: S75:  7106348 & 63 ====> 4476:  函数: [Function: charAt] 调用者: Dkdpgh4ZKsQB80/Mfvw36XI1R25+WUAlEi7NLboqYTOPuzmFjJnryx9HVGcaStCe= 参数: [ 44 ] 结果: u
第 77:  函数: [Function: apply] 调用者: [Function: t4] 参数: [ null, [ 108, 111, 44 ] ] 结果: [ '+', '4', 'S', 'u' ]78:  w4Xu + + ====> w4Xu+79:  w4Xu+ + 4 ====> w4Xu+480:  w4Xu+4 + S ====> w4Xu+4S
第 81:  w4Xu+4S + u ====> w4Xu+4Su
第 82:  3 + 3 ====> 683:  6 + 3 ====> 984:  函数: [Function: slice] 调用者: [72, 101, 108, 108, 111,44,  32,  87, 111, 114,108, 100,  33
] 参数: [ 6, 9 ] 结果: [ 32, 87, 111 ]85:  函数: [Function: slice] 调用者: [ 32, 87, 111 ] 参数: [] 结果: [ 32, 87, 111 ]86:  3 - 3 ====> 087:  32 & 255 ====> 3288:  87 & 255 ====> 8789:  111 & 255 ====> 11190:  32 << 16 ====> 209715291:  87 << 8 ====> 2227292:  2097152 | 22272 ====> 211942493:  2119424 | 111 ====> 211953594:  2119535 >> 18 ====> 895:  8 & 63 ====> 896:  函数: [Function: charAt] 调用者: Dkdpgh4ZKsQB80/Mfvw36XI1R25+WUAlEi7NLboqYTOPuzmFjJnryx9HVGcaStCe= 参数: [ 8 ] 结果: K97:  2119535 >> 12 ====> 51798:  517 & 63 ====> 599:  函数: [Function: charAt] 调用者: Dkdpgh4ZKsQB80/Mfvw36XI1R25+WUAlEi7NLboqYTOPuzmFjJnryx9HVGcaStCe= 参数: [ 5 ] 结果: h
第 100:  2119535 >> 6 ====> 33117101:  33117 & 63 ====> 29102:  函数: [Function: charAt] 调用者: Dkdpgh4ZKsQB80/Mfvw36XI1R25+WUAlEi7NLboqYTOPuzmFjJnryx9HVGcaStCe= 参数: [ 29 ] 结果: U103:  2119535 & 63 ====> 47104:  函数: [Function: charAt] 调用者: Dkdpgh4ZKsQB80/Mfvw36XI1R25+WUAlEi7NLboqYTOPuzmFjJnryx9HVGcaStCe= 参数: [ 47 ] 结果: F105:  函数: [Function: apply] 调用者: [Function: t4] 参数: [ null, [ 32, 87, 111 ] ] 结果: [ 'K', 'h', 'U', 'F' ]106:  w4Xu+4Su + K ====> w4Xu+4SuK
第 107:  w4Xu+4SuK + h ====> w4Xu+4SuKh
第 108:  w4Xu+4SuKh + U ====> w4Xu+4SuKhU
第 109:  w4Xu+4SuKhU + F ====> w4Xu+4SuKhUF
第 110:  6 + 3 ====> 9111:  9 + 3 ====> 12112:  函数: [Function: slice] 调用者: [72, 101, 108, 108, 111,44,  32,  87, 111, 114,108, 100,  33
] 参数: [ 9, 12 ] 结果: [ 114, 108, 100 ]113:  函数: [Function: slice] 调用者: [ 114, 108, 100 ] 参数: [] 结果: [ 114, 108, 100 ]114:  3 - 3 ====> 0115:  114 & 255 ====> 114116:  108 & 255 ====> 108117:  100 & 255 ====> 100118:  114 << 16 ====> 7471104119:  108 << 8 ====> 27648120:  7471104 | 27648 ====> 7498752121:  7498752 | 100 ====> 7498852122:  7498852 >> 18 ====> 28123:  28 & 63 ====> 28124:  函数: [Function: charAt] 调用者: Dkdpgh4ZKsQB80/Mfvw36XI1R25+WUAlEi7NLboqYTOPuzmFjJnryx9HVGcaStCe= 参数: [ 28 ] 结果: W125:  7498852 >> 12 ====> 1830126:  1830 & 63 ====> 38127:  函数: [Function: charAt] 调用者: Dkdpgh4ZKsQB80/Mfvw36XI1R25+WUAlEi7NLboqYTOPuzmFjJnryx9HVGcaStCe= 参数: [ 38 ] 结果: o
第 128:  7498852 >> 6 ====> 117169129:  117169 & 63 ====> 49130:  函数: [Function: charAt] 调用者: Dkdpgh4ZKsQB80/Mfvw36XI1R25+WUAlEi7NLboqYTOPuzmFjJnryx9HVGcaStCe= 参数: [ 49 ] 结果: J131:  7498852 & 63 ====> 36132:  函数: [Function: charAt] 调用者: Dkdpgh4ZKsQB80/Mfvw36XI1R25+WUAlEi7NLboqYTOPuzmFjJnryx9HVGcaStCe= 参数: [ 36 ] 结果: L133:  函数: [Function: apply] 调用者: [Function: t4] 参数: [ null, [ 114, 108, 100 ] ] 结果: [ 'W', 'o', 'J', 'L' ]134:  w4Xu+4SuKhUF + W ====> w4Xu+4SuKhUFW
第 135:  w4Xu+4SuKhUFW + o ====> w4Xu+4SuKhUFWo
第 136:  w4Xu+4SuKhUFWo + J ====> w4Xu+4SuKhUFWoJ
第 137:  w4Xu+4SuKhUFWoJ + L ====> w4Xu+4SuKhUFWoJL
第 138:  9 + 3 ====> 12139:  12 + 3 ====> 15140:  函数: [Function: slice] 调用者: [72, 101, 108, 108, 111,44,  32,  87, 111, 114,108, 100,  33
] 参数: [ 12, 15 ] 结果: [ 33 ]141:  函数: [Function: slice] 调用者: [ 33 ] 参数: [] 结果: [ 33 ]142:  3 - 1 ====> 2143:  函数: [Function: push] 调用者: [ 33, 0 ] 参数: [ 0 ] 结果: 2144:  3 - 1 ====> 2145:  函数: [Function: push] 调用者: [ 33, 0, 0 ] 参数: [ 0 ] 结果: 3146:  3 - 1 ====> 2147:  33 & 255 ====> 33148:  0 & 255 ====> 0149:  0 & 255 ====> 0150:  33 << 16 ====> 2162688151:  0 << 8 ====> 0152:  2162688 | 0 ====> 2162688153:  2162688 | 0 ====> 2162688154:  2162688 >> 18 ====> 8155:  8 & 63 ====> 8156:  函数: [Function: charAt] 调用者: Dkdpgh4ZKsQB80/Mfvw36XI1R25+WUAlEi7NLboqYTOPuzmFjJnryx9HVGcaStCe= 参数: [ 8 ] 结果: K157:  2162688 >> 12 ====> 528158:  528 & 63 ====> 16159:  函数: [Function: charAt] 调用者: Dkdpgh4ZKsQB80/Mfvw36XI1R25+WUAlEi7NLboqYTOPuzmFjJnryx9HVGcaStCe= 参数: [ 16 ] 结果: f
第 160:  2162688 >> 6 ====> 33792161:  33792 & 63 ====> 0162:  函数: [Function: charAt] 调用者: Dkdpgh4ZKsQB80/Mfvw36XI1R25+WUAlEi7NLboqYTOPuzmFjJnryx9HVGcaStCe= 参数: [ 0 ] 结果: D163:  2162688 & 63 ====> 0164:  函数: [Function: charAt] 调用者: Dkdpgh4ZKsQB80/Mfvw36XI1R25+WUAlEi7NLboqYTOPuzmFjJnryx9HVGcaStCe= 参数: [ 0 ] 结果: D165:  函数: [Function: apply] 调用者: [Function: t4] 参数: [ null, [ 33, 0, 0 ] ] 结果: [ 'K', 'f', 'D', 'D' ]166:  w4Xu+4SuKhUFWoJL + K ====> w4Xu+4SuKhUFWoJLK
第 167:  w4Xu+4SuKhUFWoJLK + f ====> w4Xu+4SuKhUFWoJLKf
第 168:  w4Xu+4SuKhUFWoJLKf + = ====> w4Xu+4SuKhUFWoJLKf=169:  w4Xu+4SuKhUFWoJLKf= + = ====> w4Xu+4SuKhUFWoJLKf==170:  12 + 3 ====> 15171:  函数: [Function: t4] 调用者: Object [global] {'@faceless': <ref *1> Object [global] {global: [Circular *1],queueMicrotask: [Function: queueMicrotask],clearImmediate: [Function: clearImmediate],setImmediate: [Function: setImmediate] {[Symbol(nodejs.util.promisify.custom)]: [Getter]},structuredClone: [Getter/Setter],clearInterval: [Function: clearInterval],clearTimeout: [Function: clearTimeout],setInterval: [Function: setInterval],setTimeout: [Function: setTimeout] {[Symbol(nodejs.util.promisify.custom)]: [Getter]},atob: [Getter/Setter],btoa: [Getter/Setter],performance: [Getter/Setter],fetch: [AsyncFunction: fetch],window: [Circular *1]},arguments: undefined,optimizedBase64Encode: [Function: t4]
} 参数: [ 'Hello, World!' ] 结果: w4Xu+4SuKhUFWoJLKf==
w4Xu+4SuKhUFWoJLKf==

实战过程中,当看到 w4Xu+4SuKhUFWoJLKf== 这个字符后,看起来像是 base64 编码,所以通过 atob 尝试解码,得到 Ã\x85îû\x84®*\x15\x05Z\x82K),显然不对。

那分析日志吧!

29:  函数: [Function: slice] 调用者: [ 72, 101, 108 ] 参数: [] 结果: [ 72, 101, 108 ]30:  3 - 3 ====> 031:  72 & 255 ====> 7232:  101 & 255 ====> 10133:  108 & 255 ====> 10834:  72 << 16 ====> 471859235:  101 << 8 ====> 2585636:  4718592 | 25856 ====> 474444837:  4744448 | 108 ====> 474455638:  4744556 >> 18 ====> 1839:  18 & 63 ====> 1840:  函数: [Function: charAt] 调用者: Dkdpgh4ZKsQB80/Mfvw36XI1R25+WUAlEi7NLboqYTOPuzmFjJnryx9HVGcaStCe= 参数: [ 18 ] 结果: w
第 41:  4744556 >> 12 ====> 115842:  1158 & 63 ====> 643:  函数: [Function: charAt] 调用者: Dkdpgh4ZKsQB80/Mfvw36XI1R25+WUAlEi7NLboqYTOPuzmFjJnryx9HVGcaStCe= 参数: [ 6 ] 结果: 444:  4744556 >> 6 ====> 7413345:  74133 & 63 ====> 2146:  函数: [Function: charAt] 调用者: Dkdpgh4ZKsQB80/Mfvw36XI1R25+WUAlEi7NLboqYTOPuzmFjJnryx9HVGcaStCe= 参数: [ 21 ] 结果: X47:  4744556 & 63 ====> 4448:  函数: [Function: charAt] 调用者: Dkdpgh4ZKsQB80/Mfvw36XI1R25+WUAlEi7NLboqYTOPuzmFjJnryx9HVGcaStCe= 参数: [ 44 ] 结果: u

看到这段日志,有什么发现吗?

上篇文章讲过,如果是 3 个字节值通过位运算得到一个 4 个字节值,即:72, 101, 108 ===> 18, 6, 21, 44

所以,我们判断,它大概是 标准base64或者变种 base64算法,由于我们验证过,它不是标准base64,所以它应该是变种base64

那么,既然知道它大概是变种base64,那我们还需要一步一步去日志还原吗?

答案肯定不需要的。

我们只需要拿着 base64标准算法模板去改就行。俗称:套模板,大大节省我们的时间,而且也不容易出错!

标准 base64 算法模版:https://t.zsxq.com/sphu5

首先控制入参一样,即:Hello, World!

分析日志发现,首先它把标准的 base64 算法,码表由 ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/ 变成了 Dkdpgh4ZKsQB80/Mfvw36XI1R25+WUAlEi7NLboqYTOPuzmFjJnryx9HVGcaStCe=

其次是:

31:  72 & 255 ====> 7232:  101 & 255 ====> 10133:  108 & 255 ====> 108

每次位运算之前,都进行了 & 255 操作。

所以我们只需要微调代码,如下:

function processGroup(a, b, c) {a = a & 255; // 追加内容b = b & 255; // 追加内容c = c & 255; // 追加内容var combined = (a << 16) | (b << 8) | c;return [base64Chars.charAt((combined >> 18) & 63),base64Chars.charAt((combined >> 12) & 63),base64Chars.charAt((combined >> 6) & 63),base64Chars.charAt(combined & 63)];
}

最后,我们再验证一下,发现结果也是 w4Xu+4SuKhUFWoJLKf==,和日志结果一致,说明我们还原成功!

完整代码:https://t.zsxq.com/ROTc7

下一篇文章我们分析 RC4 jsvmp算法,学习它的日志特点。

请添加图片描述

http://www.lryc.cn/news/601017.html

相关文章:

  • HCLP--MGER综合实验
  • JVM常见工具
  • AI 编程还有多远?我们如何迎接 AI 编程时代?
  • 【MySQL数据库备份与恢复2】备份的三种常用方法
  • MPI练习:前缀和问题
  • Effective C++ 条款4:确定对象被使用前已先被初始化
  • 7月26日星期六今日早报简报微语报早读
  • Effective C++ 条款03:尽可能使用const
  • 【AcWing 154题解】滑动窗口
  • 【音视频协议篇】WebRTC 快速入门
  • 嵌入式硬件篇---zigbee无线串口通信问题
  • 谷歌无法安装扩展程序解决方法(也许成功)
  • 【C++】stack和queue的模拟实现
  • 机器学习的工作流程
  • 【硬件-笔试面试题】硬件/电子工程师,笔试面试题-30,(知识点:传输线特性阻抗,影响因素)
  • Avantage6.6下载与安装教程
  • 瑞吉外卖学习笔记
  • 兼容性问题记录
  • 亚马逊测评采购:如何打造安全的环境,技术基础关键
  • Python点阵字生成与优化:从基础实现到高级渲染技术
  • JavaScript 立即执行函数(IIFE)运行时行为分析笔记
  • golang实现一个规则引擎,功能包括实时增加、修改、删除规则
  • GO 从入门到精通2
  • 什么是缓存雪崩?缓存击穿?缓存穿透?分别如何解决?什么是缓存预热?
  • 编程语言Java——核心技术篇(四)集合类详解
  • 【Pandas】pandas Index objects Index.shape
  • 【595驱动8*8点阵】2022-9-11
  • Linux文件系统管理——NFS服务端的安装配置与NFS客户端的安装与挂载实操教程
  • QT核心————信号槽
  • MyBatis-Plus 进阶功能:分页插件与乐观锁的实战指南