Postgresql отменить запрос при отключении клиента с помощью php - PullRequest
0 голосов
/ 29 августа 2011

У меня очень дорогой запрос, который выполняется с php, и выполнение может занять некоторое время. Есть ли способ в php обнаружить, отключается ли пользователь до выполнения запроса, и отменить его?

Ответы [ 2 ]

1 голос
/ 20 марта 2012

Возможное решение - использовать pg_send_query () , эта функция отправляет запрос в базу данных и сразу же возвращается без блокировки. Затем вы можете выполнить опрос, чтобы узнать, отключился ли пользователь до завершения запроса. Смотрите это:

ignore_user_abort(false); 
$db = pg_connect(DATABASE_DSN);
pg_send_query($db, "SELECT pg_sleep(10000)"); // long running query
while(pg_connection_busy($db)) {
  /* You need to output something to the browser so PHP can know when the
     browser disconnected. The 0 character will be ignored. 
  */
  echo chr(0); 

  /* Need to do both flushes to make sure the chr is sent to the browser */
  ob_flush();
  flush(); 

  usleep(100000); // 0.1s to avoid starving the CPU

  if(connection_status() !=  CONNECTION_NORMAL || connection_aborted()) {
    // Browser disconnected, cleanup an die
    pg_cancel_query($db);
    pg_query($db, "ROLLBACK"); 
    pg_close($db); 
    die(); 
  }
}

//  At this point the query finished and you can continue fetching the rows

Этот подход работает, но имеет большую проблему: вам действительно нужно отправить что-то в браузер, чтобы обнаружить отключение браузера. Если вы этого не сделаете, connection_status () и connection_aborted () не будут работать. Кажется, это старая ошибка PHP, смотрите здесь: https://bugs.php.net/bug.php?id=30301

Так что этот метод не работает, когда, например, вы запрашиваете Postgres в середине процедуры генерации PDF. В этом случае необходимый chr (0) сломает сгенерированный двоичный файл.

1 голос
/ 29 августа 2011

Вы хотели бы использовать connection_aborted, чтобы обнаружить, отключил ли пользователь, он возвращает 1, если клиент отключился, в противном случае он возвращает 0. Здесь есть некоторая документация здесь , однако его использование самодокументируется и у вас не должно быть проблем с его использованием.

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