session_set_save_handler - Почему этот код не работает? - PullRequest
0 голосов
/ 28 октября 2010

Я пытался сохранить данные сеанса PHP в базе данных MySQL, но не могу заставить его работать.

Для простого примера, вот код, который должен увеличивать счетчик при каждом посещении. Я видел другие примеры и т. Д., Но может кто-нибудь сказать, почему этот код не работает? (Apache / 2.2.11 (Win32) DAV / 2 mod_ssl / 2.2 .11 OpenSSL / 0.9.8i PHP / 5.2.9 Версия клиента MySQL: 5.0.51a)

Вот таблица базы данных mysql:

CREATE TABLE IF NOT EXISTS `sessions` (
  `session_ID` varchar(32) COLLATE utf8_unicode_ci NOT NULL,
  `session_data` mediumblob NOT NULL,
  `access` int(10) unsigned NOT NULL,
  PRIMARY KEY (`session_ID`),
  KEY `access` (`access`)
) ENGINE=MyISAM DEFAULT CHARSET=utf8 COLLATE=utf8_unicode_ci;

И код PHP (просто подключите свои учетные данные БД):

<?PHP

function mysession_open()
{

    global $sdbc;   // Session Database Connection

    if ($sdbc) {
        return true;
    } else {
        return false;
    }

}

function mysession_close()
{

    global $sdbc;

    return mysqli_close($sdbc);

}

function mysession_read($session_id)
{
    global $sdbc;

    $session_id = mysqli_real_escape_string($sdbc, $session_id);

    $sql_sel    = "SELECT session_data FROM sessions WHERE session_id = '$session_id'";
    $data_sel   = mysqli_query($sdbc, $sql_sel);
    $row_sel     = mysqli_fetch_array($data_sel);

    if (isset($row_sel['session_data'])) {
        return $row_sel['session_data'];
    } else {
        return '';
    }

}

function mysession_write($session_id, $session_data)
{
    global $sdbc;

    $access = time();

    $session_id    = mysqli_real_escape_string($sdbc, $session_id);
    $access        = mysqli_real_escape_string($sdbc, $access);
    $session_data   = mysqli_real_escape_string($sdbc, $session_data);

    $sql_write  =   "REPLACE INTO sessions (session_ID, session_data, access) " .
                    "VALUES ('$session_id', '$session_data', '$access')";

    return mysqli_query($sdbc, $sql_write);

}

function mysession_destroy($session_id)
{
    global $sdbc;

    $session_id = mysqli_real_escape_string($sdbc, $session_id);

    return mysqli_query($sdbc, $sql_del);
}


function mysession_gc($max)
{
    global $sdbc;

    $old = time() - $max;
    $old = mysqli_real_escape_string($sdbc, $old);

    $sql_old = "DELETE FROM sessions WHERE access < '$old'";

    return mysqli_query($sdbc, $sql_old);

}

global $sdbc;
$sdbc = mysqli_connect('localhost', '...', '...', '...') or die('Could not connect to SDBC');

session_set_save_handler('mysession_open','mysession_close','mysession_read','mysession_write','mysession_destroy','mysession_gc');
session_start();

if (isset($_SESSION['counter'])) {
    echo "counter is already set and it is " . $_SESSION['counter'] . '<br />';
    $_SESSION['counter']++;
} else {
    echo "counter is not set.  setting to 1<br />";
    $_SESSION['counter'] = 1;
}

echo "<br />Dumping SESSION data:<br />";
var_dump($_SESSION);

session_write_close();

?>

Заранее спасибо за помощь.

Если вы закомментируете строку кода session_set_save_handler, она работает нормально (увеличивается). Но с помощью обработчика сохранения это не так.

Ответы [ 2 ]

2 голосов
/ 28 октября 2010

Ни в одном из ваших запросов нет проверки ошибок.Вместо того, чтобы слепо предполагать, что часть базы данных работает, выполняйте некоторые базовые проверки ошибок на каждом этапе, например:

function mysession_write($session_id, $session_data) {
    global $sdbc;

    [...snip...]

    $stmt = mysqli_query($sdbc, $sql_write);
    if ($stmt === FALSE) {
       error_log("Failed to write session $session_id: " . mysqli_error($sdbc);
    }
    return($stmt);
}

Есть только один способ для успешного выполнения запроса, но существует множество способов его сбоя.

0 голосов
/ 23 мая 2011

С инструкция :
"Предупреждение Начиная с PHP 5.0.5, обработчики записи и закрытия вызываются после уничтожения объекта и поэтому не могут использовать объекты или генерировать исключения. Однако деструкторы объекта могут использовать сеансы.

Можно вызвать session_write_close () из деструктора, чтобы решить эту проблему с курицей и яйцом. «

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