本文若有写错的地方,还请各位大佬斧正 :)
TODO: 连续丰富并增加各个点的实际漏洞事例
file_put_contents、copy、file_get_contents等读取写入操作与unlink、file_exists等删除判断文件函数之间对付路径处理的差异导致的删除绕过例如如下代码
123456789
<?php$filename = __DIR__ . '/tmp/' . $user['name'];$data = $user['info'];file_put_contents($filename, $data);if (file_exists($filename)) {unlink($filename);}?>
这里引用小密圈中P牛的解读
查看php源码,实在我们能创造,php读取、写入文件,都会调用php_stream_open_wrapper_ex来打开流,而判断文件存在、重命名、删除文件等操作则无需打开文件流。
我们跟一跟php_stream_open_wrapper_ex就会创造,实在末了会利用tsrm_realpath函数来将filename给标准化成一个绝对路径。而文件删除等操作则不会,这便是二者的差异。
以是,如果我们传入的是文件名中包含一个不存在的路径,写入的时候由于会处理掉“../”等相对路径,以是不会出错;判断、删除的时候由于不会处理,以是就会涌现“No such file or directory”的缺点。
于是乎linux可以通过xxxxx/../test.php、test.php/.windows可以通过test.php:test、test.ph<来绕过文件删除
此外创造还可以利用伪协议php://filter/resource=1.php在file_ge_contents、copy等中读取文件内容,却可以绕过文件删除
extract()、parse_str() 等变量覆盖extract函数从数组导入变量(如\$_GET、 \$_POST),将数组的键名作为变量的值。而parse_str函数则是从类似name=Bill&age=60的格式字符串解析变量.如果在利用第一个函数没有设置EXTR_SKIP或者EXTR_PREFIX_SAME等处理变量冲突的参数时、第二个函数没有利用数组接管变量时将会导致变量覆盖的问题
intval()整数溢出、向下取整和整形判断的问题32位系统最大的带符号范围为-2147483648 到 2147483647,64位最大的是 9223372036854775807
因此,在32位系统上 intval(‘1000000000000’) 会返回 2147483647
此外intval(10.99999)会返回10,intval和int等取整都是’截断’取整,并不是四舍五入
intval函数进去取整时,是直到遇上数字或者正负号才开始进行转换,之后在碰着非数字或者结束符号(\0)时结束转换
浮点数精度问题导致的大小比较问题当小数小于10^-16后,PHP对付小数就大小不分了
var_dump(1.000000000000000 == 1) >> TRUE
var_dump(1.0000000000000001 == 1) >> TRUE
is_numeric()与intval()特性差异is_numeric函数在判断是否是数字时会忽略字符串开头的’ ‘、’\t’、’\n’、’\r’、’\v’、’\f’。
而’.’可以涌如今任意位置,E、e能涌如今参数中间,仍可以被判断为数字。也便是说is_numeric(“\r\n\t 0.1e2”) >> TRUE
intval()函数会忽略’’ ‘\n’、’\r’、’\t’、’\v’、’\0’ ,也便是说intval(“\r\n\t 12”) >> 12
strcmp()数组比较绕过int strcmp ( string $ str1 , string \$str2 )
参数 str1第一个字符串。str2第二个字符串。如果 str1 小于 str2 返回 < 0;
如果 str1 大于 str2 返回 > 0;如果两者相等,返回 0。
但是如果传入的两个变量是数组的话,函数会报错返回NULL,如果只是用strcmp()==0来判断的话就可以绕过
sha1()、md5() 函数传入数组比较绕过sha1() MD5()函数默认吸收的参数是字符串类型,但是如果如果传入的参数是数组的话,函数就会报错返回NULL。类似sha1(\$_GET[‘name’]) === sha1(\$_GET[‘password’])的比较就可以绕过
弱类型==比较绕过这方面问题遍及的很多,不作过多的阐明
md5(‘240610708’); // 0e462097431906509019562988736854
md5(‘QNKCDZO’); // 0e830400451993494058024219903391
md5(‘240610708’) == md5(‘QNKCDZO’)
md5(‘aabg7XSs’) == md5(‘aabC9RqS’)
sha1(‘aaroZmOk’) == sha1(‘aaK1STfY’)
sha1(‘aaO8zKZF’) == sha1(‘aa3OFF9m’)
‘0010e2’ == ‘1e3’
‘0x1234Ab’ == ‘1193131‘
‘0xABCdef’ == ‘ 0xABCdef’
当转换为boolean时,以下只被认为是FALSE:FALSE、0、0.0、“”、“0”、array()、NULL
PHP 7 以前的版本里,如果向八进制数通报了一个造孽数字(即 8 或 9),则后面别的数字会被忽略。var_dump(0123)=var_dump(01239)=83
PHP 7 往后,会产生 Parse Error。
字符串转换为数值时,若字符串开头有数字,则转为数字并省略后面的非数字字符。若一开头没有数字则转换为0
\$foo = 1 + “bob-1.3e3”; // $foo is integer (1)
\$foo = 1 + “bob3”; // $foo is integer (1)
\$foo = 1 + “10 Small Pigs”; // $foo is integer (11)
‘’ == 0 == false
‘123’ == 123
‘abc’ == 0
‘123a’ == 123
‘0x01’ == 1
‘0e123456789’ == ‘0e987654321’
[false] == [0] == [NULL] == [‘’]
NULL == false == 0» true == 1
eregi()匹配绕过eregi()默认吸收字符串参数,如果传入数组,函数会报错并返回NULL。同时还可以%00 截断进行绕过
PHP变量名不能带有点[.] 和空格,否则在会被转化为下划线[_]123456789
parse_str(\"大众na.me=admin&pass wd=123\"大众,$test);var_dump($test);array(2) { [\公众na_me\"大众]=> string(5) \公众admin\公众 [\"大众pass_wd\"大众]=> string(3) \公众123\"大众}in_arrary()函数默认进行疏松比较(进行类型转换)
12
in_arrary(“1asd”,arrart(1,2,3,4)) => truein_arrary(“1asd”,arrart(1,2,3,4),TRUE) => false \\(须要设置strict参数为true才会进行严格比较,进行类型检测)
htmlspecialchars()函数默认只转义双引号不转义单引号,如果都转义的话须要添加上参数ENT_QUOTES
在php4、php<5.2.1中,变量的key值不受magic_quotes_gpc影响sprintf()格式化漏洞(可以吃掉转义后的单引号)printf()和sprintf()函数中可以通过利用%接一个字符来进行padding功能例如%10s 字符串会默认在左侧添补空格至长度为10,还可以 %010s 会利用字符0进行添补,但是如果我们想要利用别的字符进行添补,须要利用 ‘ 单引号进行标识,例如 %’#10s 这个便是利用#进行添补(百分号不仅会吃掉’单引号,还会吃掉\ 斜杠)
同时sprintf()可以利用指定参数位置的写法
%后面的数字代表第几个参数,$后代表格式化类型
于是当我们输入的分外字符被放到引号中进行转义时,但是又利用了sprintf函数进行拼接时
例如%1$’%s’ 中的 ‘%被当成利用%进行padding,导致后一个’逃逸了
还有一种情形便是’被转义成了\’,例如输入%’ and 1=1#进入,存在SQL过滤,’被转成了\’
于是sql语句变成了 select from user where username = ‘%\’ and 1=1#’;
如果这个语句被利用sprintf函数进行了拼接,%后的\被吃掉了,导致了’逃逸
123456
<?php$sql = \公众select from user where username = '%\' and 1=1#';\公众;$args = \公众admin\"大众;echo sprintf( $sql, $args ) ;//result: select from user where username = '' and 1=1#'?>
不过这样随意马虎碰着 PHP Warning: sprintf(): Too few arguments的报错
这个时候我们可以利用%1$来吃掉转移添加的\
123456
<?php$sql = \"大众select from user where username = '%1$\' and 1=1#' and password='%s';\公众;$args = \"大众admin\"大众;echo sprintf( $sql, $args) ;//result: select from user where username = '' and 1=1#' and password='admin';?>php中 = 赋值运算的优先级高于and
$c = is_numeric($a) and is_numeric($b)
程序本意是要a、b都为数字才会连续,但是当\$a为数字时,会先赋值给\$c,以是可能导致$b绕过检测
当url中有多个@符号时,parse_url中获取的host是末了一个@符号后面的host,而libcurl则是获取的第一个@符号之后的。因此当代码对http://user@eval.com:80@baidu.com进行解析时,PHP获取的host是baidu.com是许可访问的域名,而末了调用libcurl进行要求时则是要求的eval.com域名,可以造成ssrf绕过
此外对付https://evil@baidu.com
这样的域名进行解析时,php获取的host是`evil@baidu.com`,但是libcurl获取的host却是evil.com
filter_var()函数对付http://evil.com;google.com 会返回false也便是认为url格式缺点,但是对付0://evil.com:80;google.com:80/ 、0://evil.com:80,google.com:80/、0://evil.com:80\google.com:80/却返回true。
通过file_get_contents获取网页内容并返回到客户端有可能造成xss例如如下代码
123456789101112131415
if(filter_var($argv[1], FILTER_VALIDATE_URL)) { // parse URL $r = parse_url($argv[1]); print_r($r); // check if host ends with google.com if(preg_match('/baidu\.com$/', $r['host'])) { // get page from URL $a = file_get_contents($argv[1]); echo($a); } else { echo \公众Error: Host not allowed\"大众; }} else { echo \"大众Error: Invalid URL\公众;}
虽然通过filter_var函数对url的格式进行检讨,并且利用正则对url的host进行限定
但是可以通过data://baidu.com/plain;base64,PHNjcmlwdD5hbGVydCgxKTwvc2NyaXB0Pgo=页面会将<script>alert(1)</script>返回给客户端,就有可能造成xss
作者:Coding Pages