Comment résoudre la vulnérabilité causée par une mauvaise utilisation de la fonction d'entités HTML

WBOY
Libérer: 2023-05-12 14:13:42
avant
1371 Les gens l'ont consulté

Le code de la question est le suivant :

误用html entities函数引发的漏洞怎么解决

Analyse de vulnérabilité :

Selon le sens de la question, ce qui fait l'objet d'une enquête ici devrait être une vulnérabilité xss, et le point de déclenchement de la vulnérabilité devrait être dans Lignes 13-14. La fonction de ces deux lignes de code est de générer directement une balise html <a>. Dans les <a> 标签。代码中的 第3-5行foreach循环$_GET 传入的参数进行了处理,但是这里有个问题。我们看下 第四行 的代码,这行代码针对 $value 进行类型转换,强制变成int类型。但是这部分代码只处理了 $value 变量,没针对 $key 变量进行处理。经过了 第3-5行 的代码处理之后,根据 & 这个符号进行分割,然后拼接到 第13行echo 语句中,在输出的时候又进行了一次 htmlentities 函数处理。 htmlentities 函数主要是会对一些特殊符号进行HTML实体编码。具体定义如下:

htmlentities — 将字符转换为 HTML 转义字符

string htmlentities ( string $string [, int $flags = ENT_COMPAT | ENT_HTML401 [, string $encoding = ini_get("default_charset") [, bool $double_encode = true ]]] )
Copier après la connexion

作用:在写PHP代码时,不能在字符串中直接写实体字符,PHP提供了一个将HTML特殊字符转换成实体字符的函数 htmlentities()。

注:htmlentities() 并不能转换所有的特殊字符,是转换除了空格之外的特殊字符,且单引号和双引号需要单独控制(通过第二个参数)。第2个参数取值有3种,分别如下:

  • ENT_COMPAT(默认值):只转换双引号。

  • ENT_QUOTES:两种引号都转换。

  • ENT_NOQUOTES:两种引号都不转换。

这里附上一个 HTML 中有用的字符实体表

误用html entities函数引发的漏洞怎么解决

经过上面的分析,我们再回到题目,想想如何构造一下攻击 payload 。我们先梳理一些已知信息:

  • 这里的 $query 参数可控

  • htmlentities 函数在这里可逃逸单引号

  • xss的漏洞触发点在 <a> 标签。

<a>Lignes 3 à 5 du code, la boucle foreach traite les paramètres transmis par $_GET

, mais il y a un problème ici. Jetons un coup d'œil à la

quatrième ligne误用html entities函数引发的漏洞怎么解决 de code. Cette ligne de code effectue une conversion de type sur

$value

et la force à être de type int. Mais cette partie du code ne traite que la variable

$value

, pas la variable $key. Après le traitement du code dans les

lignes 3 à 5

, il est divisé selon le symbole 误用html entities函数引发的漏洞怎么解决&

, puis intégré à l'instruction

echo dans la ligne 13, et la fonction htmlentities

est à nouveau exécutée lorsque sortie de l'affaire.

htmlentities误用html entities函数引发的漏洞怎么解决 La fonction code principalement des entités HTML pour certains symboles spéciaux. La définition spécifique est la suivante :

htmlentities - Convertir les caractères en caractères d'échappement HTML

/?a'onclick%3dalert(1)%2f%2f=c
Copier après la connexion
Fonction : lors de l'écriture de code PHP, les caractères d'entité ne peuvent pas être écrits directement dans la chaîne. PHP fournit une méthode pour convertir les caractères spéciaux HTML en. La fonction htmlentities() pour les caractères d'entité.

Remarque : 误用html entities函数引发的漏洞怎么解决htmlentities()

ne convertit pas tous les caractères spéciaux. Il convertit les caractères spéciaux à l'exception des espaces, et les guillemets simples et doubles doivent être contrôlés séparément (via le deuxième paramètre). Il existe trois valeurs pour le deuxième paramètre, comme suit :

  • ENT_COMPAT (valeur par défaut) : seuls les guillemets doubles sont convertis.
  • ENT_QUOTES : convertissez les deux citations.
  • ENT_NOQUOTES : Les deux types de devis ne sont pas convertis.
Voici ci-joint un tableau des entités de caractères utiles en HTMLComment résoudre la vulnérabilité causée par une mauvaise utilisation de la fonction d'entités HTMLAprès l'analyse ci-dessus, nous revenons au sujet et réfléchissons à la manière de construire l'attaque payload. Commençons par trier quelques informations connues :
  • Les paramètres $query ici sont contrôlables
  • et la fonction

    htmlentities误用html entities函数引发的漏洞怎么解决 est ici Guillemets simples échappables

  • 🎜Le point de déclenchement de la vulnérabilité xss se trouve dans la balise <a>. 🎜
🎜Dans <a>, nous pouvons exécuter du code js via des événements 🎜javascript🎜, tels que : 🎜onclick🎜 ce type d'événement, donc la construction finale du poc Comme suit : 🎜
<?php
require &#39;db.inc.php&#39;;

if(isset($_REQUEST[&#39;username&#39;])){
    if(preg_match("/(?:\w*)\W*?[a-z].*(R|ELECT|OIN|NTO|HERE|NION)/i", $_REQUEST[&#39;username&#39;])){
        die("Attack detected!!!");
    }
}

if(isset($_REQUEST[&#39;password&#39;])){
    if(preg_match("/(?:\w*)\W*?[a-z].*(R|ELECT|OIN|NTO|HERE|NION)/i", $_REQUEST[&#39;password&#39;])){
        die("Attack detected!!!");
    }
}

function clean($str){
    if(get_magic_quotes_gpc()){
        $str=stripslashes($str);
    }
    return htmlentities($str, ENT_QUOTES);
}

$username = @clean((string)$_GET[&#39;username&#39;]);
$password = @clean((string)$_GET[&#39;password&#39;]);


$query=&#39;SELECT * FROM ctf.users WHERE name=\&#39;&#39;.$username.&#39;\&#39; AND pass=\&#39;&#39;.$password.&#39;\&#39;;&#39;;

#echo $query;

$result=mysql_query($query);
while($row = mysql_fetch_array($result))
{
    echo "<tr>";
    echo "<td>" . $row['name'] . "</td>";
    echo "</tr>";
}

?>
Copier après la connexion
Copier après la connexion
🎜🎜🎜🎜Analyse de cas🎜🎜Cette analyse de cas sélectionne la 🎜vulnérabilité d'injection SQL🎜 dans le système de création de sites Web d'entreprise DM v201710 pour analyse. Tout d'abord, nous pouvons voir certaines informations pertinentes de cnvd, comme suit : 🎜🎜🎜🎜🎜Nous pouvons trouver des informations utiles dans l'avis de vulnérabilité. La vulnérabilité est située lors de la connexion, elle demande que le port de connexion en arrière-plan. se trouve dans le fichier 🎜admindm-yourname /g.php🎜, ouvrez ce fichier et constatez qu'il est redirigé vers le fichier 🎜admindm-yournamemod_common/login.php🎜, le point de déclenchement de la vulnérabilité devrait donc être dans ce fichier. 🎜🎜🎜🎜🎜Ouvrez 🎜admindm-yournamemod_common/login.php🎜 Ce fichier, vous pouvez voir l'emplacement de la vulnérabilité en un coup d'œil, et intercepter une partie du code concerné comme suit : 🎜🎜🎜🎜🎜🎜Ligne 15🎜 Il Il est évident qu'il existe une vulnérabilité d'injection SQL, via la méthode d'épissage qui est directement insérée dans l'instruction select. La variable 🎜$user🎜 de la 🎜ligne 15🎜 est soumise via 🎜POST🎜 et sa valeur est contrôlable. Mais le code 🎜Line 3🎜 dans l'image ci-dessus appelle la fonction 🎜htmlentitiesdm🎜 pour traiter les données 🎜POST🎜 Nous suivons cette fonction 🎜htmlentitiesdm🎜. La fonction se trouve dans le fichier 🎜component/dm-config/global.common.php🎜 Le code clé est intercepté comme suit : 🎜🎜🎜🎜.

这个函数是调用 htmlentities 函数针对输入的数据进行处理。前面我们已经介绍过了这个函数的用法,这里这个函数的可选参数是 ENT_NOQUOTES ,也就是说两种引号都不转换。下面我们来看个小例子:

误用html entities函数引发的漏洞怎么解决

这里我猜测开发者应该是考虑到了xss的问题,但是由于 htmlentities 这个函数选择的参数出现了偏差,导致这里我们可以引入单引号造成注入的问题。

我们看看最新版是怎么修复,使用 beyond compare 对比两个版本代码的差别。

误用html entities函数引发的漏洞怎么解决

新版修复的时候将可选参数修改为 ENT_QUOTES ,这个参数的作用就是过滤单引号加双引号,我们来看看下面这个例子,就很容易明白了这个参数的作用了。

误用html entities函数引发的漏洞怎么解决

漏洞验证

这里因为没有回显,所以是盲注,下面是验证截图:

误用html entities函数引发的漏洞怎么解决

漏洞修复

针对 htmlentities 这个函数,我们建议大家在使用的时候,尽量加上可选参数,并且选择 ENT_QUOTES 参数。

误用html entities函数引发的漏洞怎么解决

我们看看对比的效果

误用html entities函数引发的漏洞怎么解决

结语

看完了上述分析,不知道大家是否对 htmlentities 函数在使用过程中可能产生的问题,有了更加深入的理解,文中用到的代码可以从 这里 下载,当然文中若有不当之处,还望各位斧正。如果你对我们的项目感兴趣,欢迎发送邮件到 hongrisec@gmail.com 联系我们。Day12 的分析文章就到这里,我们最后留了一道CTF题目给大家练手,题目如下:

<?php
require &#39;db.inc.php&#39;;

if(isset($_REQUEST[&#39;username&#39;])){
    if(preg_match("/(?:\w*)\W*?[a-z].*(R|ELECT|OIN|NTO|HERE|NION)/i", $_REQUEST[&#39;username&#39;])){
        die("Attack detected!!!");
    }
}

if(isset($_REQUEST[&#39;password&#39;])){
    if(preg_match("/(?:\w*)\W*?[a-z].*(R|ELECT|OIN|NTO|HERE|NION)/i", $_REQUEST[&#39;password&#39;])){
        die("Attack detected!!!");
    }
}

function clean($str){
    if(get_magic_quotes_gpc()){
        $str=stripslashes($str);
    }
    return htmlentities($str, ENT_QUOTES);
}

$username = @clean((string)$_GET[&#39;username&#39;]);
$password = @clean((string)$_GET[&#39;password&#39;]);


$query=&#39;SELECT * FROM ctf.users WHERE name=\&#39;&#39;.$username.&#39;\&#39; AND pass=\&#39;&#39;.$password.&#39;\&#39;;&#39;;

#echo $query;

$result=mysql_query($query);
while($row = mysql_fetch_array($result))
{
    echo "<tr>";
    echo "<td>" . $row['name'] . "</td>";
    echo "</tr>";
}

?>
Copier après la connexion
Copier après la connexion
# Host: localhost  (Version: 5.5.53)
# Date: 2018-08-05 12:55:29
# Generator: MySQL-Front 5.3  (Build 4.234)

/*!40101 SET NAMES utf8 */;

#
# Structure for table "users"
#

DROP TABLE IF EXISTS `users`;
CREATE TABLE `users` (
  `Id` int(11) NOT NULL AUTO_INCREMENT,
  `name` varchar(255) DEFAULT NULL,
  `pass` varchar(255) DEFAULT NULL,
  `flag` varchar(255) DEFAULT NULL,
  PRIMARY KEY (`Id`)
) ENGINE=MyISAM AUTO_INCREMENT=2 DEFAULT CHARSET=utf8;

#
# Data for table "users"
#

/*!40000 ALTER TABLE `users` DISABLE KEYS */;
INSERT INTO `users` VALUES (1,'admin','qwer!@#zxca','hrctf{sql_Inject1on_Is_1nterEst1ng}');
/*!40000 ALTER TABLE `users` ENABLE KEYS */;
Copier après la connexion

Ce qui précède est le contenu détaillé de. pour plus d'informations, suivez d'autres articles connexes sur le site Web de PHP en chinois!

Étiquettes associées:
source:yisu.com
Déclaration de ce site Web
Le contenu de cet article est volontairement contribué par les internautes et les droits d'auteur appartiennent à l'auteur original. Ce site n'assume aucune responsabilité légale correspondante. Si vous trouvez un contenu suspecté de plagiat ou de contrefaçon, veuillez contacter admin@php.cn
Tutoriels populaires
Plus>
Derniers téléchargements
Plus>
effets Web
Code source du site Web
Matériel du site Web
Modèle frontal
À propos de nous Clause de non-responsabilité Sitemap
Site Web PHP chinois:Formation PHP en ligne sur le bien-être public,Aidez les apprenants PHP à grandir rapidement!