一起来探索下小程序包的魔数

2020-10-09 23:38:26

本篇文章源于一名同学对近期一篇文章聊一聊微信小程序包内容 (opens new window)的疑问--如何获取文件的 firstMark 之类的数据,借此来展开探究一下。

# 前言

# 了解下魔数

什么是魔数

magic number 一般是指硬写到代码或文件中的整型常量,数值是编程者自己指定的,其他人不知道数值有什么具体意义。

小程序包也有自己的魔数,这是区别其它文件的标识。

# 了解下大小端序

  • 大端序(Big-endian):高字节保存在内存的低地址--正序排列
  • 小端序(Little-endian):高字节保存在内存的高地址--逆序排列

注意

  • 主机字节顺序,X86一般多为小端(little-endian),网络字节顺序,一般为大端(big-endian)
  • 小程序包是以大端序方式存储的

# 实验环境

  • macOS Catalina:v10.15.5
  • node:v10.17.0
  • 测试小程序包:开源中国小程序

# 测试用node脚本

readfile.js

# 执行脚本结果

命令行执行 node readfile.js 得到如下结果

Header info:
  Magic number first:  190
  firstMark: 0xbe
  unknownInfo:  0
  infoListLength:  2978
  dataLength:  923206
  Magic number last:  237
  lastMark: 0xed

# 相关脚本若干关键方法解读

此段测试脚本摘录自原工具并做了小修改

# buf.readUInt8(0)

作用:从第0个字节开始读取一个无符号的8比特位整数值(即:从0读一个字节)。
这里是读取一个字节的数据作为 firstMark ,也就是魔数了。如上面执行结果所示,值是一个10进制数字为 190。

# buf.readUInt8(13)

作用:从第13个字节开始读取一个无符号的8比特位整数值。
这里是从第13个字节读取一个字节的数据作为 lastMark ,是另一个魔数。值为 237。

# firstMark.toString(16)

作用:把数字转换为字符串(以十六进制值显示)
这里数字190的十六进制转换结果为 be ,因为十六进制有前缀 0x ,所以前面需要带前缀以表示十六进制值->0xbe

# buf.readUInt32BE(5)

作用:以大端字节序从 buf 中第 5 个字节读取一个无符号的 32 位整数值。
这里代表微信文件信息列表长度值。

# 结论

微信小程序包文件的魔数为两个十六进制值 0xbe0xed,使用其它文件跑脚本会报"Magic number is not correct!"错误。

# 参考资料

本文链接:
版权声明:本博客所有文章除特别声明外,均采用 CC BY-NC-ND 3.0 许可协议。可自由转载、引用,但需署名作者且注明文章出处。如转载至微信公众号,请在文末添加作者公众号二维码。

扫描下方二维码阅读当前文章

浏览器、微信扫码

评 论:

好文推荐
每天进步一点点~