NerbianRat样本分析报告

本篇文章为本人原创,首发于freebuf

前言

Proofpoint的安全研究员发现并分析了这个新型恶意软件并命名为NerbianRAT,此恶意软件使用了反分析和反逆向功能,该恶意软件是使用Golang编写的64位程序,主要传播方式为冒充世界卫生组织发送的COVID-19相关的安全措施邮件,通过邮件附件中有VBA宏的Word文档传播。

样本运行流程

样本IOCs

名称: ee1bbd856bf72a79221baa0f7e97aafb6051129905d62d74a37ae7754fccc3db.doc
大小: 280469 字节 (273 KiB)
MD5: d7888fea6047b662a30bf00edac4c3ee
SHA1: 8137670512be55796f612e41602f505955b0bb0c
SHA256: ee1bbd856bf72a79221baa0f7e97aafb6051129905d62d74a37ae7754fccc3db

名称: MoUsoCore.exe
大小: 5867008 字节 (5729 KiB)
MD5: 5d5bc970f975341558b8d2c225ca0115
SHA1: 4f74826ed56cda233cfc12b86fd1b7da4a9f2e56
SHA256: 902c65435b6b44cfda1156b0e7c6a30b2785fa4f2cbb9b1944a66f5146ec7aa5

名称: UpdateUAV.exe
大小: 3642880 字节 (3557 KiB)
MD5: 9cca59eec5af63e42cd845b67cf6df89
SHA1: 178aad6c7918cc495a908944e79143a913630890
SHA256: 1b8c9e7c150bacd466fbe7f12b39883821f23b67cae0a427a57dc37e5ea4390f

恶意代码分析

doc宏代码分析

双击打开doc文件发现是一个带宏的文档,文档中诱导用户点击启用宏脚本

这里我使用olevba脚本来分析此word文档的vba代码

olevba脚本已经帮我们分析出了此vba代码的主要功能,从解码的Base64字符串我们大致可以判断此vba脚本使用powershell从C2下载了payload并写入本地文件夹执行

此vba代码有三个函数,GetByte和DecodeBase64这两个函数功能为解码Base64

主要的Document_Open入口函数我们可以看到定义了很多字符串但都是经过Base64编码,这些字符串在使用之前都调用DecodeBase64函数进行解码

我们将Base64字符串解码后优化代码再查看逻辑更清晰,主要逻辑就是红框中的命令行,使用cmd.exe将powershell命令行写入%temp%\util.bat,然后执行bat脚本,从hxxps://www[.]fernandestechnical[.]com/pub/media/gitlog下载文件到%appdata%\UpdateUAV.exe并且执行,执行完UpdateUAV.exe后将%temp%\util.bat和%appdata%\UpdateUAV.exe删除

UpdateUAV.exe分析

我们查看从C2下载的dropper文件,通过详细信息可以看到,文件详细信息伪装成Windows系统程序,这里还可以发现此dropper的原始文件是nsoobe.exe

使用DIE工具查看UpdateUAV.exe发现此程序是64位程序并且使用了UPX3.9.6压缩壳进行加壳

我这里使用了UPX -d命令直接进行自动脱壳,如果遇到了修改版的UPX就需要手动脱壳,脱完壳文件体积膨胀到了将近一倍

脱完壳我们在使用DIE工具查看此UpdateUAV.exe是使用Golang编写的,Go编译器版本号为1.15.0或以上的版本

要想确定精确的Go编译器版本可以通过搜索字符串go1.关键词,可以看到UpdateUAV.exe使用的是go1.17.3版本编译器,注意此方法在遇到去除符号信息和严重混淆的样本可能无效

这里直接使用IDA打开文件进行分析,查看main函数发现此Golang程序的符号信息都在,代码并没有被加密或者混淆,我们甚至可以通过函数符号名分析出函数的大致功能,比如函数main_hideWindows可以推断是隐藏进程窗口

我们使用x64dbg进行动态调试,这里为了方便调试我们手动关闭掉随机基址,使用010Editor打开PE文件将Nt头中的扩展头中的DllCharacteristics前1个字节改为00就关闭了PE文件的随机基址,这里原始是字节60 81

我们改为00 81并保存至此已经成功将随机基址关闭

我们此时再把IDA中的符号信息导出成MAP文件并导入x64dbg,打开IDA选择File->Produce file->Create MAP file然后选择保存位置,把全部选框都勾上确认

然后使用x64dbg的SwissArmyKnife插件导入MAP文件

当x64dbg导入MAP文件后通过查看IDA中的main函数地址并下断点,MAP文件导入x64dbg后让我们可以和IDA分析更好的同步

首先分析hideWindows函数,通过函数名可猜测此函数是隐藏控制台窗口,首先使用GetConsoleWindow获取控制台窗口句柄

最后调用ShowWindowAsync函数使用SW_HIDE参数将控制台窗口隐藏

接下来分析checkEnvironment函数,可以看到此函数中调用了github上的第三方包chacal

我们搜索发现chacal这个包是Golang的反虚拟机框架

checkEnvironment函数主要通过5个函数实现,其中两个函数都是安全工具进程检测,另外三个函数主要是反虚拟机检测

antidebug_processList函数调中又调用了第三方的包go_ps实现遍历windows进程

使用了CreateToolhelp32Snapshot函数创建进程快照,然后使用Process32First和Process32Next函数组合遍历进程

Process32Next遍历下一个进程

遍历完进程后将调用github_com_p3tr0v_chacal_utils_PList函数进行对比

先对比进程黑名单列表字符串长度,如果长度相同则调用runtime_memequal函数进行字符串对比

进程黑名单列表一共有42个如下,antimem_processList函数和antidebug_processList函数逻辑都相同不过antimem_processList只检测了DumpIt.exe、RAMMap.exe、RAMMap64.exe、vmmap.exe这四个用于DUMP进程内存的工具进程所以就不贴图了

IsVirtualDisk函数首先调用了queryWMI函数,此函数封装了两个函数先调用了CreateQuery函数创建了WMI WQL查询语句

然后调用wmi_Query函数使用WMI WQL语句进行查询了网卡信息

最后调用了ContainsInList函数对比网卡是否为列表中名单中的虚拟网卡,虚拟网卡黑名单列表中有三个virtual,vmware, vbox

接下来分析ByMacAddress函数该函数首先调用了getMacAddr函数查询本机的MAC地址

然后调用ContainsPrefix函数对比本机和黑名单列表的的MAC地址

接下来分析diskTotalSize函数,从函数符号可猜测此函数用来检测硬盘大小,通过函数的传参0x64十进制为100可以猜测此检测大小为100GB

通过分析diskTotalSize函数内部,也是调用了queryWMI函数进行查询硬盘信息,然后对比本机硬盘是否小于100GB,我的虚拟机硬盘大小为99GB十六进制0x63

如果以上反调试检测都通过,接着使用IsDebuggerPresent检测本进程是否被调试,还调用了time_Since函数和函数开头的time_Now组合检测函数运行时间判断进程是否被调试

checkLocation函数使用GET请求hxxps://json[.]geoiplookup[.]io网站获取json格式的公网IP归属地,但是此网站屏蔽了ASN为4134的IP地址所以此处返回值为错误代码

正常如果获取到了本机IP归属地会和列表中的地区进行比较,我们可以看到此列表中,只有两个具体的地区伦敦和俄罗斯在这两个中间的单词都为一些少儿不宜的话

一共对比了列表中7个单词,从2到6个单词可以猜测此恶意软件的作者可能是个种族歧视主义者

调用strings_Index函数进行逐一对比

checkLocation函数检测如果IP归属地不在列表中则会调用downloadNerbian函数从C2服务器下载NerbianRAT主体程序

分析downloadNerbian函数,此函数首先会使用RedFile函数打开C:\\ProgramData\\USOShared\\MoUsoCore.exe路径中的文件

如果文件不存在则会从C2下载,如果存在此文件还会判断此文件前两个字节是否为4D5A(MZSignature)用于判断此文件是否为PE文件

调用downloadFile函数从C2下载NerbianRAT

如果首次从C2下载失败,还会调用cmd使用curl从C2下载

从C2下载完成后都会读取文件并检测文件头两个字节是否为4D5A(MZSignature)判断是否为PE文件

最后一个函数是创建计划任务实现持久化运行,首先通过CMD调用格式化好的命令创建计划任务

我们可以打开计划任务查看,可以看到触发条件是每隔1小时运行一次

触发操作就是启动从C2下载的NerbianRAT

如果创建计划任务成功则直接触发执行运行NerbianRAT,至此UpdateUAV.exe这个dropper程序就分析完成

MoUsoCore.exe分析

接下来我们分析NerbianRAT主体程序,NerbianRAT一样使用了UPX压缩壳还是一样的流程脱壳,此样本去除大部分的符号信息,不过我们还是可以通过搜索github关键词查找MoUsoCore.exe的函数可以查看使用的go开源包,通过如下这些包可以大致判断出NerbianRAT的大致功能

1
2
3
4
5
6
smbios包提供对系统管理BIOS(SMBIOS)和桌面管理接口(DMI)数据和结构的检测和访问:github[.]com/digitalocean/go-smbios
Windows WMI提供了WQL接口:github[.]com/StackExchange/wmi
桌面截图:github[.]com/kbinani/screenshot
go的WindowsAPI封装:github[.]com/AllenDang/w32
golang的win32 ole实现:github[.]com/go-ole/go-ole
go的WindowsAPI封装:github[.]com/lxn/win

golang有一种特殊的函数初始化函数,定义格式fcun init(),此函数会在main函数之前执行,并且同一个包可以定义多个init函数,编译时编译器会自动更名,这里可以看到main包中一共有两个init初始化函数

其中的main_init_0使用了硬编码的AesGCM加密模式的密钥解密了很多需要使用到的字符串

调用gcmAsm_open解密

可以看到解密出来的是一个ip地址此地址应该是和C2相关的,接下来还进行了多次解密出剩下的加密字符串

下面开始分析main函数,第一个函数调用main_I4JkbFMH还是个进程检测名单

使用了github开源的StackExchange包通过WQL语句SELECT * FROM Win32_Process查询了本机进程

对比进程黑名单列表

接下来分析main_H5NzwUxN函数,首先获取了本机BIOS信息,然后对获取到的BIOS信息使用MD5算法进行了哈希

然后将MD5值类型转换成16进制

接着使用getCurrentProcessId函数获取了本进程的PID并同样对PID使用MD5进行哈希

同样的将MD5值类型转换为16进制

接着生成了一个唯一ID

接着将生成的唯一ID转为大写字母

函数main_H5NzwUxN获取收集了主机名称等信息

函数main_JgJWgOp中调用ReadFile函数读取了args_c.txt文件如果不存在此文件则跳转,暂不清楚此文件的作用可能是后期存储收集到的信息

函数main_ZPBgbOEQ读取了C:\ProgramData\Microsoft OneDrive\setup\rev.sav文件,此文件可能也是用来存储一些收集到的数据

接下来main_DNKvpcvy函数实现了加解密和收发网络请求,首先格式化了一个IP地址,此地址可能是C2服务器IP地址

向C2hxxps://www[.]fernandestechnical[.]com/pub/health_check[.]php发送Get请求判断C2是否存活

向C2发送Get请求

C2返回状态码200则C2存活

C2和本机的keep-alive心跳包

获取了本地IP地址

接下来使用RSA-2048加密了0x98大小的内容,其中包含了收集到的本机基本信息

RSA公钥为硬编码

RSA加密后的Buff为0x100

然后拼接了0x14C大小的缓冲区

接下来使用了AesCBC模式加密,使用补全码0x4填充了4字节到0x150大小

使用了硬编码的32字节Aes密钥进行加密

使用硬编码的AesCBC密钥加密后数据

将随机生成的0x10大小的数据写入AesCBC加密后的缓冲区头部

再次拼接将8563写入缓冲区头部

函数main_P6EwC8SB是对auth_post、data_post、addr_post、port_post字段数据进行加密的函数,此AesCBC模式加密密钥是随机生成的32字节

使用AesCBC模式加密后

接着生成了70个字节的随机数

使用Base64对AesCBC模式加密后的数据进行编码

将随机生成的70个字节数据填充到头部,将AesCBC模式加密使用的32字节大小随机生成密钥存放在70个字节数据之后,后面的为Base64编码后的加密数据

函数main_GNd3j2oz就是生成session_key字段的数据,首先调用了go-smbios包获取bios硬件信息,并使用MD5哈希

将MD5转为十六进制

之后将0x40字节大小的全局变量和bios信息MD5值和字符串windows进行格式化,随后直接使用Base64对这些数据进行了编码

拼接好要发送的POST请求

发送POST请求到C2

屏幕截取功能使用了screenshot开源库实现

屏幕截图功能

这里我使用了golang编写了解密脚本,除了session_key这个数据单单使用了Base64编码,其他的4个字段的数据都可以使用这个脚本解密,auth_post和data_post使用了3层加密,第一层的数据使用了RSA-2048进行加密,第二层req使用了硬编码的AesCBC密钥加密,第三层8563使用了随机生成的AesCBC加密,所以我们最多可以解密两层go

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
package main

import (
"crypto/aes"
"crypto/cipher"
"encoding/base64"
"fmt"
)

func main() {
// addr_post字段加密数据
allcryptDataBase64 := "EcgOPkkOHFylaFoLaqoXmbKPGOzvddbJMlnTtEtlScAdeewEFwzdITJsRYdYEushByrcQJCtuqdlGSeQyCjNieJBeQSnVwNcDhtQrh06LdGa3uyRHexahEL05goQ=="
// 加密数据前70个字节为垃圾数据,垃圾数据后的32个字节为AesCBC加密密钥
aesCBCKey := allcryptDataBase64[70 : 70+32]
// 需要解密的数据为70垃圾字节+32字节的AesCBC密钥后面的Base64编码数据
cryptDataBase64 := allcryptDataBase64[70+32:]
// 将加密数据进行Base64解码
cryptData, _ := base64.StdEncoding.DecodeString(string(cryptDataBase64))
// 使用AesCBC解密数据
decryptData := AesDecryptCBC(cryptData, []byte(aesCBCKey))
// 打印解密数据
fmt.Printf("%x\n", decryptData)
// 解密第二层数据:解密auth_post和data_post数据时再调用
DecryptoSecData(decryptData)
}

func DecryptoSecData(decryptData []byte) {
// 解密第二层加密
// 硬编码AesCBCKey
gaesCBCKeyBase64 := "F+h/WB8d+NYSnWX9UM6z3WxOHCIwd819TFldpsPfkrI="
// 解码第二层硬编码的AesCBC密钥
gaesCBCKey, _ := base64.StdEncoding.DecodeString(string(gaesCBCKeyBase64))
// 第二层要解密的数据
secCryptData := decryptData[8:]
// 使用AesCBC解密数据
secDecryptData := AesDecryptCBC(secCryptData, []byte(gaesCBCKey))
fmt.Println("第二层数据解密--------------------------------------")
// 打印解密数据
fmt.Printf("%x", secDecryptData)
}

// AesCBC模式解密函数
func AesDecryptCBC(encrypted []byte, key []byte) (decrypted []byte) {
// 分组密钥
block, _ := aes.NewCipher(key)
// 获取密钥块的长度
blockSize := block.BlockSize()
// 加密模式
blockMode := cipher.NewCBCDecrypter(block, key[:blockSize])
// 创建数组
decrypted = make([]byte, len(encrypted))
// 解密
blockMode.CryptBlocks(decrypted, encrypted)
// 去除补全码
decrypted = PKCS5UnPadding(decrypted)
return decrypted
}

// 去除补全码
func PKCS5UnPadding(origData []byte) []byte {
length := len(origData)
unpadding := int(origData[length-1])
return origData[:(length - unpadding)]
}

总结

NerbianRAT使用了现在主要的恶意软件传播方式之一为通过邮件附件的带VBA宏脚本的word文档进行传播,甚至不乏很多境外APT组织也使用此方式针对性攻击,go这种跨平台的编译型编程语言正被越来越多的恶意软件开发者采用,go众多的开源包可以实现快速开发,NerbianRAT使用了众多的反逆向和反虚拟机功能加大了分析的时间和难度,并且使用了RSA和AES组合的加密手段用来传输数据,对于此类传播方式的恶意软件能做的只有加强邮件地址过滤和附件检测和加大信息安全教育普及。