发起这个帖子,估计就很多人看到题目就表示不屑了。一直以来PHP+MySQL环境下,无论是写程序或者是注入攻击,是无法多语句执行的,这么广为人知的
常识,没理由会有人不知道。可权威就是用来被挑战的,常识也就是为了被打破的。如果没有一点创新性,追根到底的求知欲,一直在条条框框里挣扎,那还有什么
争取自由、解析世界的Hacker,Geek精神?最近在wooyun上看到一个很简单的sql注入案例,虽然漏洞很简单,但是其中蕴含的内容可大大不
同。亮点在于:作者居然在注射利用过程中使用了mysql的多语句执行。感谢作者@紫梦芊 ,让我们一直以来奉为金科玉律的信条被彻底颠覆。本着Know
it then hack it的想法,我仔细研究了一下这一条“常识”是如何形成,又如何被作者打破的。从最早国内angel介绍的《SQL
Injection with MySQL》&《Advanced SQL Injection with
MySQL》这两篇文章开始,PHP+MySQL注入就一直停留在UNION Select的基石上建立起来的。可Union select作为SQL
Inj的方法,一直都有很多限制,比如需要猜字段数、猜表名,非select语句就无法使用union,注入点在order by或者group
by语句后就很难进行操作,盲注比较复杂等等问题。可为什么非要使用union select?除了当年MySQL
3.x不支持子查询查询数据之外,主要原因,在实践中发现注射中同样不能像mssql一样用分号来分割执行多个sql语句。为什么?因为PHP的
MySQL,MySQLi扩展,都因为安全原因,在connect的时候,都不允许设置CLIENT_MULTI_STATEMENTS,哪怕你手工在
connect的时候设置了这个flag,php在源代码级别,仍然会帮你去除掉:php-5.3.8/ext/mysqli
/mysqli_nonapi.c    /* set some required options */    flags |=
CLIENT_MULTI_RESULTS; /* needed for mysql_multi_query() */    /* remove
some insecure options */    flags &= ~CLIENT_MULTI_STATEMENTS;   /*
don't allow multi_queries via connect parameter
*/而没有这个特殊的设置,MySQL是不允许在一个mysql_query中使用分号执行多语句的。于是,这也就是长久以来,大家都认为mysql是不
允许多语句执行的根本原因。实际上这恰恰是个错误的认识,实际上mysql早在4.1版本就允许多语句执行。只是PHP自身限制了这种用法。而是不是
PHP就完全无法使用多语句执行呢?答案是否,利用mysqli_multi_query就可以执行多语句,但是现实应用中基本没有人会用这个语句。而另
一个常被程序员所使用的PDO方式操作数据库,情况又如何?create table `car`(`name` varchar(32),
`type` varchar(32));<?php$db = new
PDO("mysql:host=localhost:3306;dbname=test", 'root', '');// works
regardless of statements emulation//
$db->setAttribute(PDO::ATTR_EMULATE_PREPARES, 0);//$sql = "DELETE
FROM car; INSERT INTO car(name, type) VALUES ('car13', 'coupe'); INSERT
INTO  car(name, type) VALUES ('car2', 'coupe');";//try
{    //$db->exec($sql);  $db->query($sql);  //$stmt =
$db->prepare($sql);  //$stmt->execute();}catch(PDOException
$e){    echo
$e->getMessage();    die();}测试了一下,果然,以上这些多语句,在PDO的情况下,安然执行了。果断查找PHP源代
码:php-5.3.8/ext/mysqlnd/mysqlnd_enum_n_def.h#define
CLIENT_MULTI_STATEMENTS    (1UL << 16) /* Enable/disable
multi-stmt support */php-5.3.8/ext/pdo_mysql/mysql_driver.c#ifdef
CLIENT_MULTI_STATEMENTS    |CLIENT_MULTI_STATEMENTS果然,PHP源代码默认支持了多语句执行特
性,已经在mysqlnd这个driver中定义了多语句执行的参数。要知道,现在多数PHP的编程框架为了支持多数据库类型,基本都会使用PDO作为底
层数据库连接方式。这意味着什么?意味着在PDO的环境中,注入点是什么类型(insert,update,delete,select)已经不重要了,
注入点在语句的什么位置也不重要了(table,where,orderby),一切可能性都重现了,mysql功能突然全开放在我们面前,都可以利用多
语句的方式,重新构造我们所需要的sql语句。那是否利用PDO使用这个就完全没有限制了呢?当然也不是。首先,一般框架使用PDO作为数据库连接,常用
的数据库操作,都是使用参数绑定或者prepare的方式,进行参数绑定的。而在默认情况下,这种参数绑定是在php客户端进行的,在这种情况下,是不允
许多语句的。只有如下方法:$db->setAttribute(PDO::ATTR_EMULATE_PREPARES,
0);才可以利用prepare执行多语句(因为参数绑定是在server端执行),而这种状况非常少见。所以,真正可以利用的多语句注射,只能是存在于
利用PDO连接数据库,并直接使用exec或者query函数进行执行sql语句的地方。是不是太苛刻?其实这种例子并不少见,各位只要用心找,总是能找
到的。文末再次感谢引导我寻找原因的漏洞作者@紫梦芊 ,以及@Laruence ,给我看PHP代码提供了不少帮助。

PHP+MySQL多语句执行<转自wooyun>的更多相关文章

  1. MySQL的语句执行顺序

    MySQL的语句执行顺序 MySQL的语句一共分为11步,如下图所标注的那样,最先执行的总是FROM操作,最后执行的是LIMIT操作.其中每一个操作都会产生一张虚拟的表,这个虚拟的表作为一个处理的输入 ...

  2. MySQL查询语句执行过程及性能优化(JOIN/ORDER BY)-图

    http://blog.csdn.net/iefreer/article/details/12622097 MySQL查询语句执行过程及性能优化-查询过程及优化方法(JOIN/ORDER BY) 标签 ...

  3. mysql 中语句执行的顺序以及查询处理阶段的分析

    原文链接:http://www.php.cn/mysql-tutorials-408865.html 本篇文章给大家带来的内容是关于mysql中语句执行的顺序以及查询处理阶段的分析,有一定的参考价值, ...

  4. MySQL查询语句执行过程及性能优化-查询过程及优化方法(JOIN/ORDER BY)

    在上一篇文章MySQL查询语句执行过程及性能优化-基本概念和EXPLAIN语句简介中介绍了EXPLAIN语句,并举了一个慢查询例子:

  5. mysql优化必知(mysql的语句执行顺序)

    MySQL的语句执行顺序 MySQL的语句一共分为11步,如下图所标注的那样,最先执行的总是FROM操作,最后执行的是LIMIT操作.其中每一个操作都会产生一张虚拟的表,这个虚拟的表作为一个处理的输入 ...

  6. [转]MySQL查询语句执行过程详解

    Mysql查询语句执行原理 数据库查询语句如何执行?语法分析:首先进行语法分析,对使用sql表示的查询进行语法分析,生成查询语法分析树.语义检查:检查sql中所涉及的对象以及是否在数据库中存在,用户是 ...

  7. Mysql查询语句执行过程

    Mysql查询语句执行过程   Mysql分为server层和存储引擎两部分,或许可以再加一层连接层   连接层(器) Mysql使用的是典型的C/S架构.连接器通过典型的TCP握手完成连接. 需要注 ...

  8. mysql sql语句执行时是否使用索引检查方法

    在日常开发中,使用到的数据表经常都会有索引,这些索引可能是开发人员/DBA建表时创建的,也可能是在使用过程中新增的.合理的使用索引,可以加快数据库查询速度.然而,在实际开发工作中,会出现有些sql语句 ...

  9. 关于sql和MySQL的语句执行顺序(必看!!!)

    今天遇到一个问题就是mysql中insert into 和update以及delete语句中能使用as别名吗?目前还在查看,但是在查阅资料时发现了一些有益的知识,给大家分享一下,就是关于sql以及My ...

随机推荐

  1. 【C#学习笔记】Dictionary容器使用

    using System; using System.Collections.Generic; namespace ConsoleApplication { class Program { stati ...

  2. 翻译【ElasticSearch Server】第一章:开始使用ElasticSearch集群(7)

    检索文档(Retrieving documents) 我们已经有文档存储在我们的实例.现在,让我们尝试检索它们: curl -XGET http://localhost:9200/blog/artic ...

  3. LightOJ 1427 -Repository(ac自动机)

    题意: 求每个模式串在母串中出现的次数 #include <map> #include <set> #include <list> #include <cma ...

  4. codeforces 696A Lorenzo Von Matterhorn 水题

    这题一眼看就是水题,map随便计 然后我之所以发这个题解,是因为我用了log2()这个函数判断在哪一层 我只能说我真是太傻逼了,这个函数以前听人说有精度问题,还慢,为了图快用的,没想到被坑惨了,以后尽 ...

  5. 《Python基础教程(第二版)》学习笔记 -> 第十章 充电时刻 之 标准库

    SYS sys这个模块让你能够访问与Python解释器联系紧密的变量和函数,下面是一些sys模块中重要的函数和变量: 函数和变量 描述 argv 命令行参数,包括脚本和名称 exit([arg])   ...

  6. LoadRunner学习记录--Flights打开空白页的问题

    从网上查了一下,原因是PERL5LIB这个环境变量的原因. 担心修改环境变量会影响ORACLE的运行 在WebTour中修改run.bat   增加 set PERL5LIB=D:\oracle\pr ...

  7. OpenGL超级宝典第5版&&GLSL法线变换

    在GLSL中,有一些情况需要把局部坐标系下的向量或点转换到视点坐标系下,如光照计算时,需要把法向转化到视点坐标系.如果是模型上一点p 转化到视点坐标系下,直接(model-view matrix )* ...

  8. 帮同事写了几行代码,在 安装/卸载 程序里 注册/卸载 OCX控件

    写了个小控制台程序,这个程序用来注册 / 卸载OCX控件,用在Inno Setup做的安装卸载程序里. #include "stdafx.h" #include <windo ...

  9. 一行命令实现Android自动关机

    前几天晚上失眠,实在睡不着觉,于是想用Nexus7听一听小野丽莎的歌,在安静祥和之中睡去(怎么感觉有点...)但是不能让平板总是这么循环播放吧(屋里吐槽Google Play Music),所以在平板 ...

  10. 精雕细琢 35 套精美的 PSD 图标素材

    设计师总是有独特的创意和精雕细琢的精湛技术,让我们值得去欣赏和借鉴,如梦想天空所表达的:非常感谢那些很有才华的设计师分享它们的劳动成果,让更多的人可以使用他们的创意设计.今天,本文与大家分享35套精美 ...