我有一个PHP websocket服务器(https://github.com/lemmingzshadow/php-websocket/)。
当用户连接时,我想设置/获取一些会话数据。
问题是我不能使用$_SESSION
,因为如果我这样做,我得到的不是客户端的会话,而是我的websocket服务器的会话。
我已经设法得到客户的SESSID:
private function handshake($data) {
$this->log('Performing handshake');
if(preg_match("/PHPSESSID=(.*?)(?:;|rn)/", $data, $matches)){
$sessid = $matches[1];
}
/* Do handshake */
}
但现在我不知道如何获得与SESSID相关的会话数据。
我似乎已经解决了它(但我还需要更多的测试):
private function handshake($data) {
$this->log('Performing handshake');
if(preg_match("/PHPSESSID=(.*?)(?:;|rn)/", $data, $matches)){
$this->sessID = $matches[1];
session_id($this->sessID);
@session_start();
if(isset($_SESSION['userName'])){
$this->userName = $_SESSION['userName'];
}else{
$_SESSION['userName'] = $this->userName;
}
session_write_close();
}else{
$this->log('No SESSID');
return false;
}
/* Do handshake */
}
为了启用会话cookie,在我的客户端页面index.php 上
<?php
session_start();
?>
<!DOCTYPE html>
<!-- Page's HTML -->
解释
首先,我使用session_id
,这样下面的session_start()
将为我提供与给定SESSID相关的会话。
然后,我可以使用@session_start()
来启动会话。它需要@
,因为websocket服务器已经启动,所以不使用它会产生:
- PHP警告:session_start():无法发送会话cookie-标头已发送
- PHP警告:session_start():无法发送会话缓存限制器-标头已发送
现在我可以用$_SESSION
做我想做的事了。
最后,我使用session_write_close()
来关闭会话。必须使用它,因为:
session_id
必须在session_start()
之前调用。如果会话没有关闭,当第二个客户端连接时,会话将已经打开,因此session_id
将无法工作。以避免会话锁定。正如@Sven所说,只有一个运行PHP的实例可以访问会话文件。由于websocket服务器的脚本应该运行很长时间,一旦它打开一个会话,如果你不关闭它,你就不能在其他脚本中使用会话。