PHP一些特性引发的安全问题

没人质疑PHP的强大,可是它本身的一些特性也引发了一些安全问题。刚好最近在撸一些ctf题目,就对我见到的PHP存在的问题总结记录一下。

弱类型问题

在PHP中,可以执行这样操作

$var = 1;
$var = array();
$var = 'string';

它不会验证变量的类型,也可以随时转换变量类型。估计开发者原本是想让程序员更加高效的开发,所以在大量内置函数以及基本结构中使用了很多松散的比较和转换,防止程序中的变量因为程序员的不规范而频繁的报错,然而这却带来了安全问题。

  • 比较运算时
$a = null; $b = ''; $a == $b  //true
$a = null; $b = false; $a == $b  //true
$a = 0; $b = '0'; $a == $b  //true
$a = 0; $b = '0'; $a === $b  //false
$a = 0; $b = 'string'; $a == $b  //true

可见PHP在处理比较运算时,不会检查表达式类型,只在恒等比较时才同时检查表达式的值与类型。

  • hash比较时
var_dump('0e12345' == 0);  //true
var_dump('0e12345' == '0e54321');  //true
var_dump('0e12345' == '0e12345a');  //false

可以看到在遇到0e\d+类型的字符串时,会把此类型字符串作为科学计数法来处理,所以左右两边都为0*10^n = 0 。假如md5(str)为此类型的话,就可以构造md5(str1) == md5(str2)来绕过一些过滤。当然,这样的字符串有很多,详见PHP处理0e开头md5哈希字符串缺陷

  • 类型强制转换
  1. 当string遇上int
var_dump(0 == 'abcd');  //true  
var_dump(1 == '1abcd');  //true

当有一个对比参数是整数的时候,会把另外一个参数强制转换为整数。而转换过程中,'1abcd'的转换后的值是1,而‘abcd’是0,说明了intval返回字符串中第一个非数字的字符之前的数字串所代表的整数值。

var_dump(intval('12.12a'));  //12
  • 内置函数的松散
  1. md5()

    md5() 函数用于对字符串进行md5加密

    var_dump(md5('1'));  //c4ca4238a0b923820dcc509a6f75849b
    

    当参数是string时正常加密,但是当你传递一个array时,函数不会报错,只是返回null

    $a[] =1;
    $b[] =2;
    var_dump(md5($a));  //null
    var_dump(md5($a)==md5($b));  //true
    

    这样就可以构造任意2个array来绕过md5函数的检测。

  2. in_array()

    in_array() 函数检查数组中是否存在某个值

    $array=[0,1,2,'3'];
    var_dump(in_array('abc', $array));  //true
    

    如果第三个参数 strict 的值为 TRUE 则 in_array() 函数还会检查 needle 的类型是否相同。只有加了strict才会对类型进行严格比较, 那么我们再次把整形和字符串进行比较呢?

    var_dump(in_array('abc', $array));  //true
    var_dump(in_array('1bc', $array));  //true
    var_dump(in_array('4', $array));  //false
    

    它遍历了array的每个值,并且作"=="比较(“当设置了strict 用===”),上面的情况前两个返回的都是true,因为’abc’会转换为0,’1bc’转换为1。那么我们完全就可以很容易的用构造好的int 0或1来绕过检测函数,使它返回为真。

  3. strcmp()

    strcmp() 函数比较两个字符串,该函数返回:

    0  //如果两个字符串相等
    <0  //如果 string1 小于 string2
    >0  //如果 string1 大于 string2
    

    这里的strcmp函数实际上是将两个变量转换成ascii 然后做数学减法,返回一个int的差值。

    也就是说键入'a'和'a'进行比较得到的结果就是0

    那么如果让$array和‘a’比较呢?

    http://localhost/test.php?a[]=1
    var_dump(strcmp($_GET[a],'a'));  //null
    

    也就是说,我们让这个函数出错从而来绕过函数的检查。

浮点数精度问题

在PHP manual中有提到:

以十进制能够精确表示的有理数如 0.1 或 0.7,不能在不丢失一点点精度的情况下转换为二进制的格式。

var_dump(intval((0.1+0.7)*10));  //int(7)
var_dump(floor((0.1+0.7)*10));  //float(7)
var_dump(intval(0.58*100));  //int(57)
var_dump(floor(0.58*100));  //floatt(57)

为什么会这样的?简单分析一下

0.1 的二进制:

符号位 0 
指数 01111011 (-4)
尾数 1.10011001100110011001101 (1.60000002384185791015625)

将这个数再转回十进制:0.10000000149011612

0.7 的二进制:

符号位 0 
指数 01111110  (-1)
尾数 1.01100110011001100110011 (1.39999997615814208984375)

将这个数再转回十进制:0.699999988079071

很明显,在转换为二进制的过程中丢失了精度,0.1 + 0.7 的结果是 0.79999998956919

关于浮点数的详细分析见PHP浮点数的一个常见问题的解答

参考:

wooyun知识库

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,590评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 86,808评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,151评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,779评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,773评论 5 367
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,656评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,022评论 3 398
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,678评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 41,038评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,659评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,756评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,411评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,005评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,973评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,203评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,053评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,495评论 2 343

推荐阅读更多精彩内容

  • PHP常用函数大全 usleep() 函数延迟代码执行若干微秒。 unpack() 函数从二进制字符串对数据进行解...
    上街买菜丶迷倒老太阅读 1,351评论 0 20
  • php usleep() 函数延迟代码执行若干微秒。 unpack() 函数从二进制字符串对数据进行解包。 uni...
    思梦PHP阅读 1,980评论 1 24
  • PHP:include()``include_once()``require()``require_once() ...
    寻梦xunm阅读 845评论 1 4
  • PHP7 已经出来1年了,PHP7.1也即将和大家见面,这么多好的特性,好的方法,为什么不使用呢,也希望PHP越来...
    梦幻_78af阅读 2,072评论 1 10
  • 一、php可以做什么 php是一种可以在服务器端运行的编程语言,可以运行在Web服务器端。 php是一门后台编程语...
    空谷悠阅读 3,086评论 4 97