Поддержка PDO для нескольких запросов (PDO_MYSQL, PDO_MYSQLND) - PullRequest
93 голосов
/ 14 июня 2011

Я знаю, что PDO не поддерживает выполнение нескольких запросов в одном операторе. Я был в Google и нашел несколько постов, рассказывающих о PDO_MYSQL и PDO_MYSQLND.

PDO_MySQL более опасен приложение, чем любое другое традиционное MySQL приложения. Традиционный MySQL позволяет только один запрос SQL. В PDO_MySQL нет такого ограничения, но вы рискуете получить инъекцию несколько запросов.

От: Защита от SQL-инъекций с использованием PDO и Zend Framework (июнь 2010 г .; Джулиан)

Похоже, что PDO_MYSQL и PDO_MYSQLND обеспечивают поддержку нескольких запросов, но я не могу найти больше информации о них. Были ли эти проекты прекращены? Можно ли сейчас выполнить несколько запросов с использованием PDO.

Ответы [ 6 ]

130 голосов
/ 24 июня 2011

Как я знаю, PDO_MYSQLND заменил PDO_MYSQL в PHP 5.3. Запутанная часть в том, что имя по-прежнему PDO_MYSQL. Так что теперь ND является драйвером по умолчанию для MySQL + PDO.

В целом, для одновременного выполнения нескольких запросов:

  • PHP 5.3 +
  • mysqlnd
  • эмулировал подготовленные высказывания. Убедитесь, что для PDO::ATTR_EMULATE_PREPARES установлено значение 1 (по умолчанию). В качестве альтернативы вы можете избежать использования подготовленных утверждений и использовать $pdo->exec напрямую.

Использование exec

$db = new PDO("mysql:host=localhost;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 ('car1', 'coupe'); 
INSERT INTO car(name, type) VALUES ('car2', 'coupe');
";

try {
    $db->exec($sql);
}
catch (PDOException $e)
{
    echo $e->getMessage();
    die();
}

Использование операторов

$db = new PDO("mysql:host=localhost;dbname=test", 'root', '');

// works not with the following set to 0. You can comment this line as 1 is default
$db->setAttribute(PDO::ATTR_EMULATE_PREPARES, 1);

$sql = "
DELETE FROM car; 
INSERT INTO car(name, type) VALUES ('car1', 'coupe'); 
INSERT INTO car(name, type) VALUES ('car2', 'coupe');
";

try {
    $stmt = $db->prepare($sql);
    $stmt->execute();
}
catch (PDOException $e)
{
    echo $e->getMessage();
    die();
}

Примечание:

При использовании эмулированных подготовленных операторов убедитесь, что вы установили правильную кодировку (которая отражает фактическую кодировку данных) в DSN (доступно с 5.3.6). В противном случае может быть небольшая вероятность внедрения SQL, если используется нечетная кодировка .

17 голосов
/ 05 марта 2015

После полдня возни с этим обнаружил, что в PDO была ошибка, где ...

-

//This would run as expected:
$pdo->exec("valid-stmt1; valid-stmt2;");

-

//This would error out, as expected:
$pdo->exec("non-sense; valid-stmt1;");

-

//Here is the bug:
$pdo->exec("valid-stmt1; non-sense; valid-stmt3;");

Это выполнит "valid-stmt1;", остановится на "non-sense;" и никогда не выдаст ошибку. Не запустит "valid-stmt3;", вернет истину и лжет, что все прошло хорошо.

Я бы ожидал, что в "non-sense;" произойдет ошибка, но это не так.

Вот где я нашел эту информацию: Неверный запрос PDO не возвращает ошибку

Вот ошибка: https://bugs.php.net/bug.php?id=61613

<Ч />

Итак, я попытался сделать это с помощью mysqli и не нашел какого-либо убедительного ответа о том, как это работает, поэтому я решил оставить его здесь для тех, кто хочет его использовать ..

<code>try{
    // db connection
    $mysqli = new mysqli("host", "user" , "password", "database");
    if($mysqli->connect_errno){
        throw new Exception("Connection Failed: [".$mysqli->connect_errno. "] : ".$mysqli->connect_error );
        exit();
    }

    // read file.
    // This file has multiple sql statements.
    $file_sql = file_get_contents("filename.sql");

    if($file_sql == "null" || empty($file_sql) || strlen($file_sql) <= 0){
        throw new Exception("File is empty. I wont run it..");
    }

    //run the sql file contents through the mysqli's multi_query function.
    // here is where it gets complicated...
    // if the first query has errors, here is where you get it.
    $sqlFileResult = $mysqli->multi_query($file_sql);
    // this returns false only if there are errros on first sql statement, it doesn't care about the rest of the sql statements.

    $sqlCount = 1;
    if( $sqlFileResult == false ){
        throw new Exception("File: '".$fullpath."' , Query#[".$sqlCount."], [".$mysqli->errno."]: '".$mysqli->error."' }");
    }

    // so handle the errors on the subsequent statements like this.
    // while I have more results. This will start from the second sql statement. The first statement errors are thrown above on the $mysqli->multi_query("SQL"); line
    while($mysqli->more_results()){
        $sqlCount++;
        // load the next result set into mysqli's active buffer. if this fails the $mysqli->error, $mysqli->errno will have appropriate error info.
        if($mysqli->next_result() == false){
            throw new Exception("File: '".$fullpath."' , Query#[".$sqlCount."], Error No: [".$mysqli->errno."]: '".$mysqli->error."' }");
        }
    }
}
catch(Exception $e){
    echo $e->getMessage(). " <pre>".$e->getTraceAsString()."
"; }
4 голосов
/ 06 мая 2015

Быстрый и грязный подход:

function exec_sql_from_file($path, PDO $pdo) {
    if (! preg_match_all("/('(\\\\.|.)*?'|[^;])+/s", file_get_contents($path), $m))
        return;

    foreach ($m[0] as $sql) {
        if (strlen(trim($sql)))
            $pdo->exec($sql);
    }
}

Разбивается в разумных конечных точках оператора SQL Нет проверки ошибок, нет защиты от впрыска. Поймите свое использование перед использованием. Лично я использую его для заполнения исходных файлов миграции для интеграционного тестирования.

0 голосов
/ 14 марта 2019

Как и тысячи людей, я ищу этот вопрос:
Может выполнить несколько запросов одновременно, и если бы была одна ошибка, ни одна не запустилась бы, я пошел на эту страницу везде
Но хотя друзья здесь дали хорошие ответы, эти ответы не были хороши для моей проблемы
Поэтому я написал функцию, которая работает хорошо и почти не имеет проблем с sql Injection.
Это может быть полезно для тех, кто ищетподобные вопросы, поэтому я поставил их здесь для использования

function arrayOfQuerys($arrayQuery)
{
    $mx = true;
    $conn->beginTransaction();
    try {
        foreach ($arrayQuery AS $item) {
            $stmt = $conn->prepare($item["query"]);
            $stmt->execute($item["params"]);
            $result = $stmt->rowCount();
            if($result == 0)
                $mx = false;
         }
         if($mx == true)
             $conn->commit();
         else
             $conn->rollBack();
    } catch (Exception $e) {
        $conn->rollBack();
        echo "Failed: " . $e->getMessage();
    }
    return $mx;
}

для использования (пример):

 $arrayQuery = Array(
    Array(
        "query" => "UPDATE test SET title = ? WHERE test.id = ?",
        "params" => Array("aa1", 1)
    ),
    Array(
        "query" => "UPDATE test SET title = ? WHERE test.id = ?",
        "params" => Array("bb1", 2)
    )
);
arrayOfQuerys($arrayQuery);

и моего соединения:

    try {
        $options = array(
            //For updates where newvalue = oldvalue PDOStatement::rowCount()   returns zero. You can use this:
            PDO::MYSQL_ATTR_FOUND_ROWS => true
        );
        $conn = new PDO("mysql:host=$servername;dbname=$database", $username, $password, $options);
        $conn->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION);
    } catch (PDOException $e) {
        echo "Error connecting to SQL Server: " . $e->getMessage();
    }

Примечание:
Это решение помогает вам запускать несколько операторов вместе,
Если происходит неправильный оператор, он не выполняет никаких других операторов

0 голосов
/ 17 ноября 2016

Попробуйте эту функцию: mltiple запросов и вставка нескольких значений.

function employmentStatus($Status) {
$pdo = PDO2::getInstance();

$sql_parts = array(); 
for($i=0; $i<count($Status); $i++){
    $sql_parts[] = "(:userID, :val$i)";
}

$requete = $pdo->dbh->prepare("DELETE FROM employment_status WHERE userid = :userID; INSERT INTO employment_status (userid, status) VALUES ".implode(",", $sql_parts));
$requete->bindParam(":userID", $_SESSION['userID'],PDO::PARAM_INT);
for($i=0; $i<count($Status); $i++){
    $requete->bindParam(":val$i", $Status[$i],PDO::PARAM_STR);
}
if ($requete->execute()) {
    return true;
}
return $requete->errorInfo();
}
0 голосов
/ 30 апреля 2014

Попробовал следующий код

 $db = new PDO("mysql:host={$dbhost};dbname={$dbname};charset=utf8", $dbuser, $dbpass, array(PDO::ATTR_ERRMODE => PDO::ERRMODE_EXCEPTION));

Затем

 try {
 $db->query('SET NAMES gbk');
 $stmt = $db->prepare('SELECT * FROM 2_1_paidused WHERE NumberRenamed = ? LIMIT 1');
 $stmt->execute(array("\xbf\x27 OR 1=1 /*"));
 }
 catch (PDOException $e){
 echo "DataBase Errorz: " .$e->getMessage() .'<br>';
 }
 catch (Exception $e) {
 echo "General Errorz: ".$e->getMessage() .'<br>';
 }

И получил

DataBase Errorz: SQLSTATE[42000]: Syntax error or access violation: 1064 You have an error in your SQL syntax; check the manual that corresponds to your MySQL server version for the right syntax to use near '/*' LIMIT 1' at line 1

Если добавлено $db->setAttribute(PDO::ATTR_EMULATE_PREPARES, false); после $db = ...

Затем получил пустую страницу

Если вместо этого SELECT попытался DELETE, то в обоих случаях возникла ошибка вроде

 DataBase Errorz: SQLSTATE[42000]: Syntax error or access violation: 1064 You have an error in your SQL syntax; check the manual that corresponds to your MySQL server version for the right syntax to use near '* FROM 2_1_paidused WHERE NumberRenamed = '¿\' OR 1=1 /*' LIMIT 1' at line 1

Так что мой вывод о том, что инъекция невозможна ...

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...