Неверный запрос PDO не возвращает ошибку - PullRequest
9 голосов
/ 01 июня 2011

Второй оператор SQL ниже возвращает ошибку в phpMyAdmin:

SET @num=2000040;
INSERT INTO artikel( artikel_nr, lieferant_nr, bezeichnung_1, bezeichnung_1 )
SELECT @num := @num +1 AS anum, 70338, f2, f3
FROM import
WHERE id >1

MySQL говорит:

#1110 - Column 'bezeichnung_1' specified twice

Все правильно. Но когда я запускаю запросы в Symfony 1.4 с этой функцией:

// run sql query
// http://erisds.co.uk/symfony/snippet-creating-debugging-complex-sql-queries-in-symfony
// http://stackoverflow.com/questions/5434702/php-quick-refactoring
// param $sql:    the query to run
// param $silent: bool if errors should be ignored
// throws:        pdo error info if statement failed and $silent=false
// returns:       pdo-statement (use for looping over result rows and error messages)
public static function runQuery($sql, $silent=false)
{
  $conn = Propel::getConnection();
  $pdo_statement = $conn->prepare($sql);

  $error = null;
  try
  {
    $pdo_statement->execute();
  }
  catch (Exception $e)
  {
    $error = $e->getMessage();
  }

  if ( !$error )
  {
    $pdo_error = $pdo_statement->errorInfo();
    $error = $pdo_error[2];
  }
  if ( !$silent && $error ) throw new Exception($error);

  return $pdo_statement;
}

ошибка не выдается. Два оператора SQL должны быть отправлены одновременно, поскольку они зависят друг от друга. Неправильный запрос создается из пользовательского ввода. Мне нужно вернуть эту ошибку, иначе я не могу сказать, была ли база данных изменена, и я не могу сообщить об этом пользователю.

Знаете ли вы, почему PDO не жалуется на недействительное утверждение, и если это невозможно сделать, как получить информацию об успехе / неудаче?

Кстати, запрос обновляет базу данных, если нет повторяющихся столбцов.

Вот ссылка на класс PDOStatement: http://www.php.net/manual/en/class.pdostatement.php

Ответы [ 3 ]

8 голосов
/ 31 августа 2012

Это ожидаемое поведение.Поскольку существует два оператора, и первый из них действителен, вы должны использовать nextRowset ()

try
  {
    $pdo_statement->execute();
    while ($pdo_statement->nextRowset()) {/* https://bugs.php.net/bug.php?id=61613 */};
  }

Источник: bugs.php.net / bug.php? Id = 61613

4 голосов
/ 30 марта 2012

По умолчанию PDOStatement::execute() не выдает никаких исключений, просто возвращает false при ошибке. Вы должны установить обработку ошибок от PDO::ERRMODE_EXCEPTION до db->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION).

0 голосов
/ 13 октября 2013

Если у вас есть возможность использовать mysqli вместо PDO для мультизапроса, вы можете использовать mysqli_multi_query.Поскольку обработка ошибок немного сложна, вот моя функция:

/**
 * Executes multiple queries at once (separated by semicolons) and discards the results
 *
 * @param string $sql
 * @throws RuntimeException if a query failed
 */
function multi_query($sql) {
    $mysqli = new mysqli('host', 'user', 'password', 'database');


    //Execute all queries
    $index = 1;
    if ($mysqli->multi_query($sql)) {
        do {
            // next_result() fails if store_result() is not called
            if ($result = $mysqli->store_result()) {
                $result->free();
            }
            $has_next = $mysqli->more_results();
            if (!$has_next)
                return; // all queries successfully executed
            $index++;
        } while ($mysqli->next_result());
    }
    // At this point, either the multi_query() has returned false - which is
    // when the first query failed - or more_results() was true, while next_result()
    // returned false - which is when a different query failed.

    $error = $mysqli->error;
    if (!$error)
        $error = $mysqli->errno ? "errno $mysqli->errno" : '(unknown error)';
    throw new RuntimeException("mysqli query $index failed: $error");
}
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...