前言
图片验证码是为了防止恶意破解密码、刷票、论坛灌水等才出现的,但是你有没有想过,你的图形验证码竟然可能导致服务器的崩溃?
那他是如何导致的呢?请听我婉婉道来。
先看看各大平台对待此漏洞的态度:
data:image/s3,"s3://crabby-images/5b90b/5b90b9be452f9a2c3ee10d47a9bbd98365b1128e" alt=""
data:image/s3,"s3://crabby-images/45490/4549086e5fffcad281e5403864271ee540635418" alt=""
data:image/s3,"s3://crabby-images/139ec/139ec0504c50bde3318a2f2c795df208a2c1d74f" alt=""
data:image/s3,"s3://crabby-images/5e83c/5e83cff071c3519d4102f95c9e899cf4a59183a9" alt=""
利用过程
这里以phpcms为例, 首先需要找一个图形验证码。
data:image/s3,"s3://crabby-images/f28e1/f28e1dfef9546a60fb98c8571cbb90f42a05c6a4" alt=""
将图片拖动到浏览器中将得到该图形验证码的链接:
data:image/s3,"s3://crabby-images/3c72a/3c72acd36468f345422924533d4624709534969e" alt=""
将链接拖出来咱们分析一下:
- http://127.0.0.1/phpcms/api.php?op=checkcode&code_len=4&font_size=20&width=130&height=50&font_color=&background=
- code_len=4是验证码的字符的个数
- font_size=20是验证码的大小
- width=130是验证码的宽
- height=50是验证码的高
如果将参数 font_size、 width、 height均设置为1000将会出现什么情况呢?
data:image/s3,"s3://crabby-images/f54ec/f54ecd7a2bf8947f7f49d6aea9fafd155d3712fc" alt=""
可以发现验证码的形状已经改变,证明这个参数是会随着我们的更改而更改的,而且在改成1000时,整个页面的加载速度明显变慢,大家请想,如果都改为10000甚至更高会发生什么事情? 服务器会不会一直在处理我们所请求的图片?
此时我们上神器burp来检验一下
当设置为1000时:
data:image/s3,"s3://crabby-images/0f7b2/0f7b2ae97d6ca3e9f62cd8f86040ea73047d52d9" alt=""
当设置为10000时:
data:image/s3,"s3://crabby-images/c19c2/c19c236ce7428b1379adeb9b010ea65d61615f40" alt=""
处理所用的时间明显增长,我们再看看在发起这个请求时CPU有什么反应:
data:image/s3,"s3://crabby-images/9e98a/9e98a0a861ff402a0383b19fffb3fc6e933bdab8" alt=""
CPU有一个明显的峰值,那当我们使用python去不断进行请求的时候又会发生什么?这个是正常情况下的CPU占用率:
data:image/s3,"s3://crabby-images/1e7ac/1e7ac689d314b71bb3f8c502a334d69fde60201b" alt=""
当我们启动python程序的时候CPU的占用率:
data:image/s3,"s3://crabby-images/57153/57153e11cd54b1ba142f11c180e1f2ae93f37b44" alt=""
可见,我们请求的验证码对服务器产生了重大的影响。
代码分析
根据请求包我们跟进到 \api\checkcode.php:
data:image/s3,"s3://crabby-images/ba030/ba03042eaaac8ca81a680fb16465a55fa7d7fa8c" alt=""
在第12行和第17行可以看到,仅仅将验证码的长宽通过get方式接受,没有任何其他的过滤,在第13行和第18行也仅仅只对最小值进行了限制,并没有对***值进行限制,所以就造成了这个验证码的漏洞。
总结一下这个漏洞的利用点:
- 图片长宽等可控
- 后端没有对图片的***大小进行限制
解决方案
- 对接受的参数进行***值的控制,或者固定值处理
- 使用固定大小的验证码