问答
发起
提问
文章
攻防
活动
Toggle navigation
首页
(current)
问答
商城
实战攻防技术
漏洞分析与复现
NEW
活动
摸鱼办
搜索
登录
注册
CobaltStrike逆向学习系列(1):CS 登陆通信流程分析
安全工具
只有对工具足够的了解,才能够更好的进行修改
这是\[**信安成长计划**\]的第 1 篇文章 **0x00 目录** **0x01 密码校验** **0x02 aggressor.authenticate** **0x03 aggressor.metadata** **0x04 数据同步** **0x05 流程图** **0x06 参考文章** 先统一一下后续文章所提到的一些名词,以确保大家都在聊同一个东西,文中将 CobaltStrike分为 Controller、TeamServer、Beacon 三端,本文所描述的是 TeamServer 启动之后,从 Controller 登陆 TeamServer 的流程分析。 由于水平有限,对于数据同步并没有理解的足够清楚,望各位斧正。 0x01 密码校验 ========= 启动 TeamServer 后,会创建 SecureServerSocket 对象并且循环调用 acceptAndAuthenticate 方法等待 Controller 连接 data:image/s3,"s3://crabby-images/1f419/1f419aa836818c36382d5a343cf32e6813f3cae8" alt="图片" 在接收到信息并验证成功以后,才会去调用 clientAuthenticated 方法来线程执行 ManageUser 以处理与 Controller 的信息 data:image/s3,"s3://crabby-images/c6afe/c6afebd2228682e71e8bdd58710d6c3780f77927" alt="图片" 当 Controller 在点击 Connect 按钮时,会调用 Connect 中的 dialogAction 方法,会先创建 SecureSocket 对象,并调用 authenticate 方法进行验证 data:image/s3,"s3://crabby-images/57e8e/57e8e6103b97033e705b3acc972acb99924e4026" alt="图片" 在创建 SecureSocket 对象时,会与 TeamServer 进行握手等操作,TeamServer 会进入 SecureServerSocket.this.authenticate 方法进行验证,此时会一直在 var4.readInt() 阻塞,直到 Controller 将信息发完 data:image/s3,"s3://crabby-images/71dbd/71dbdc7b85630ced05f8abe8529f04e0e6b7c07e" alt="图片" 接着来看 Controller 的处理,在 authenticate 中,进行了数据包的构造,先写入一个标志 48879(int),接着是密码的长度(byte),然后是密码,之后用 65 来进行填充 密码长度加填充长度,不超过 256,再加上标识位和密码长度,256+5=261 data:image/s3,"s3://crabby-images/6f2f4/6f2f475cfd04908f08166a4b96653f50d966dc30" alt="图片" 接着在 flush 之后,TeamServer 就开始验证了,判断标志位是否正确,读取密码,读填充字符 data:image/s3,"s3://crabby-images/f2681/f2681eb9e77655099cf43974e023b8ac72578ed0" alt="图片" 对比密码,如果正确,写回标志位 51966 data:image/s3,"s3://crabby-images/c026c/c026cb5903b9a6c18f60471d41a0fc9445d3eeb0" alt="图片" 在 Controller 这边同样也会进行验证 data:image/s3,"s3://crabby-images/f37a3/f37a3e89e9b4256bd15a2a81de36b73dac04fc15" alt="图片" 密码校验到此也就结束了 0x02 aggressor.authenticate =========================== 接着,TeamServer 创建 Socket,并创建 ManageUser 来处理 Controller 发送的信息 data:image/s3,"s3://crabby-images/827de/827dee7fa733829b05790beaaf22bd871757b1df" alt="图片" Controller 创建 TeamQueue 来进行后续操作,TeamQueue 是用来处理与 TeamServer 的通讯的 在 TeamQueue 的构造函数中,创建了两个线程分别来跑 TeamQueue 的内部类 TeamReader 和 TeamWriter,用来与 TeamServer 进行通信 data:image/s3,"s3://crabby-images/a11c9/a11c9e409fcdc04a8a0b40943631a329327c615e" alt="图片" 接着,当前线程调用 call 向 TeamServer 发送 aggressor.authenticate,并将 user、pass、版本号组成对象传递过去 data:image/s3,"s3://crabby-images/4b89f/4b89fb9791c8503d8e3aa9aefab0a9b0a1228ca8" alt="图片" 在调用 call 中,最关键的就是 addRequest,它将构造好的 Request 对象存放到 LinkedList 当中 因为此时专门有一个线程 TeamWriter 一直在从 LinkedList 中取值发送,所以在添加完之后,这个信息就会被发送到 TeamServer 中 data:image/s3,"s3://crabby-images/b6af4/b6af499f06329e4a951b7fed54dc4890e04161f7" alt="图片" 在 TeamServer 中,是由 ManageUser 一直循环等待 Controller 发送到请求的,在上面发送完成后,TeamServer 也就接收到请求了 data:image/s3,"s3://crabby-images/e45a7/e45a7b1856d51127cd152586f1123a57c5bf1aa0" alt="图片" 进入 process 来处理请求,通过对比任务类型,来决定实际执行的内容 data:image/s3,"s3://crabby-images/017f1/017f1df1379652d9ec2054da0d893de166cf1191" alt="图片" 接着对 版本、密码进行校验 data:image/s3,"s3://crabby-images/edc91/edc9199c1b6fbcb26cf71041afcd47b370268610" alt="图片" 全部验证成功后,返回 SUCESS 接着会起一个线程,ManageUser 的内部类 BroadcastWriter data:image/s3,"s3://crabby-images/a1de3/a1de3b69f2e0a343cfd894dd8e2fc0a0cde54e26" alt="图片" 此时 Controller 由 TeamReader 接收到信息 data:image/s3,"s3://crabby-images/347e2/347e29755699fc51c52d64d10d2a61ff76ec43f1" alt="图片" 这里接受的是 Reply,发送的时候是 Request,这两个类基本是一样的,可能是用来区分数据包的 在请求的时候填入的 callback,也是在这个时候使用的,在之前 call 的时候,将一个 callback 值与这个对象做 Map,在这个时候取出来用于去调用对应类中的 result 方法 data:image/s3,"s3://crabby-images/0dd74/0dd742e98da3519a94be863259cd909ba93a3601" alt="图片" 在判断返回值是 SUCCESS 之后,接着又发送了 aggressor.metadata data:image/s3,"s3://crabby-images/9e545/9e54578f5892fcbdf8319d82760b0091f7dfb56b" alt="图片" 0x03 aggressor.metadata ======================= 调用 call 与之前一样,此时传入的参数是当前的时间戳 data:image/s3,"s3://crabby-images/eec78/eec78babba622db1189fde20f5503cd5f4b8ca58" alt="图片" TeamServer 中的 ManageUser 接到消息后,继续走 process 处理 data:image/s3,"s3://crabby-images/42a22/42a224a02dc91e9122be83edef7451b32a7886b9" alt="图片" 做了一堆 Map,然后将信息传回给 Controller data:image/s3,"s3://crabby-images/044d8/044d83aed5b7f3466163052da5815e056e3f711c" alt="图片" Controller 的 TeamReader 接收到回传信息 data:image/s3,"s3://crabby-images/cc273/cc273419026e7ae6967190efb99ccba44905d5f5" alt="图片" data:image/s3,"s3://crabby-images/03a9f/03a9fcc409641e73ebf5dc4ad59ceb381010df4c" alt="图片" 到 Connect 处理 aggressor.metadata data:image/s3,"s3://crabby-images/faa25/faa25511a4fbc3293ca55def0444322bcc9c2f3c" alt="图片" 在进入 AggressorClient 以后,调用 setup 方法,处理与界面相关内容,最后向 TeamServer 发送了 aggressor.ready 表示完成 data:image/s3,"s3://crabby-images/8ac85/8ac850ebb9b7a3da05c3bb4290994f610ed3a26e" alt="图片" TeamServer 在 ManageUser 中接到数据以后,process 进行处理,接下来的任务就是处理数据同步 data:image/s3,"s3://crabby-images/f7ac0/f7ac0aaad1f42ac868280b735b958de1396455ee" alt="图片" 0x04 数据同步 ========= 在 register 中,会先将 user 与对应的 manageUser 存储到 Map 当中,接着调用 playback 来处理同步的工作 data:image/s3,"s3://crabby-images/004ab/004ab3a504ca8e3d652bbec28162d8d7525a8ce1" alt="图片" 在 playback 会计算 this.transcripts 与 this.replayme 的总大小,然后进行发送 data:image/s3,"s3://crabby-images/1966f/1966f98451e394696dc9c6c7ec4d2c0b820833b2" alt="图片" send 的时候,就用到了前面所创建的 Map,通过名字取出相应的 ManageUser data:image/s3,"s3://crabby-images/774aa/774aa68955ce6b79db66010f9efaac99cb7ff447" alt="图片" 之后也就是发送的常用流程,将信息打包成 Reply 或 Request 然后发送 data:image/s3,"s3://crabby-images/858a1/858a147f010fff9472eb140ef1a719403d5739a7" alt="图片" Controller 会在 TeamReader 中接到消息,因为 callback 是 0,所以会走走 else 当中处理,调用 DataManager 的 result 方法来进行处理 data:image/s3,"s3://crabby-images/f423d/f423dc54114e26ad1cf7c115eef27e37cb0642c1" alt="图片" 用于判断 sent 与 total 是否相等,来明确是否已经完成 data:image/s3,"s3://crabby-images/57b84/57b84e4b24226f7fe141bb07cd10ec504efe1960" alt="图片" 接着遍历并调用对应的 result 方法 data:image/s3,"s3://crabby-images/a49e9/a49e9a7c2d462231e80eb01f3c2d94565f61e8d9" alt="图片" 继续回到 TeamServer ,接下来当前线程会来遍历 this.transcripts 和 this.replayme,并将信息 send 到 Controller,由于这里 this.transcripts 为空,就直接看 this.replayme data:image/s3,"s3://crabby-images/0609c/0609cc0867d004011cac17cf89177eb67d924e9a" alt="图片" 先把其中所对应的值都取出来,修改了当前的 message 信息以后,先将 playback.status 包发回,然后再将取出来的 Key、Value 发回,最后将 send 加一,用于 Controller 中对比 send 与 total 值 data:image/s3,"s3://crabby-images/c801a/c801afae6a5123bf3bda348ffc0fcece26297324" alt="图片" 当回信息时,Controller 判断是 Data,所以进入了另外的分支,由于不是 ChangeLog 类型的内容,存储到 Map 后就直接结束了 data:image/s3,"s3://crabby-images/dd26e/dd26e459eece62e98fbb6589cd91967a0f778f19" alt="图片" 之后再调用 将当前用户信息提供给 Controller data:image/s3,"s3://crabby-images/1d0ff/1d0ff8f8df635c560c3064ea40acfada16e0c614" alt="图片" 在 TeamServer 继续执行调用的时候,也是调用的 broadcast 来同步 eventlog data:image/s3,"s3://crabby-images/dbb27/dbb2755e0175e7a035886e2e3fb80c23bbda8a0f" alt="图片" 之后也就进入到了常态化的,接 Controller 的 aggressor.ping,和 BroadcastWriter 回写 Beacons 信息 data:image/s3,"s3://crabby-images/49b02/49b029272189e8dc3d6538129e67fa16a774b4e9" alt="图片" 0x05 流程图 ======== data:image/s3,"s3://crabby-images/fd039/fd0390f366a4527124259bee0d19143a4c55a008" alt="图片" 0x06 参考文章 ========= 快乐鸡哥:<https://bbs.pediy.com/thread-267208.htm> WBGlIl:<https://wbglil.gitbook.io/cobalt-strike/cobalt-strike-yuan-li-jie-shao/untitled-2>
发表于 2022-01-25 09:52:37
阅读 ( 6447 )
分类:
安全工具
0 推荐
收藏
0 条评论
请先
登录
后评论
信安成长计划
9 篇文章
×
发送私信
请先
登录
后发送私信
×
举报此文章
垃圾广告信息:
广告、推广、测试等内容
违规内容:
色情、暴力、血腥、敏感信息等内容
不友善内容:
人身攻击、挑衅辱骂、恶意行为
其他原因:
请补充说明
举报原因:
×
如果觉得我的文章对您有用,请随意打赏。你的支持将鼓励我继续创作!