#Session sharing and user authentication
Thanks to Ratchet its easy to get the shared info from the same website session. As per the Ratchet documentation, you must use a session handler other than the native one, such as Symfony PDO Session Handler.
All session handler based on \SessionHandlerInterface
work ! Not only PDO !
Create the following services:
services:
pdo:
class: PDO
arguments:
dsn: mysql:host=%database_host%;port=%database_port%;dbname=%database_name%
user: %database_user%
password: %database_password%
calls:
- [ setAttribute, [3, 2] ] # \PDO::ATTR_ERRMODE, \PDO::ERRMODE_EXCEPTION
session.handler.pdo:
class: Symfony\Component\HttpFoundation\Session\Storage\Handler\PdoSessionHandler
arguments: [@pdo, {lock_mode: 0}]
Configure the Session Handler in your config.yml
framework:
...
session:
handler_id: session.handler.pdo
This is what informs Symfony2 of what to use as the session handler.
Similarly, we can do the same thing with Gos WebSocket to enable a shared session.
gos_web_socket:
...
client:
firewall: secured_area #can be an array of firewalls
session_handler: @session.handler.pdo
By using the same value, we are using the same exact service in both the WebSocket server and the Symfony2 application. If you experience any issues with the session being empty or not as expected. Please ensure that everything is connecting via the same URL, so the cookies can be read.
User is directly authenticated against his firewall, anonymous users are allow.
Anonymous user is represented by string, example : anon-54e3352d535d2 Authenticated user is represented by UserInterface object
##Client storage
Each user connected to socket is persisted in our persistence layer. By default they are stored in php via SplStorage.
###Customize client storage
gos_web_socket:
client:
firewall: secured_area
session_handler: @session.handler.pdo
storage:
driver: @gos_web_scocket.client_storage.predis.driver
ttl: 28800 #(optionally) time to live if you use redis driver
prefix: client #(optionally) prefix if you use redis driver, create key "client:1" instead key "1"
We natively provide decorator for DoctrineCacheBundle to decorate Cache provider into client storage driver.
Create redis cache provider
doctrine_cache:
providers:
redis_cache:
redis:
host: 127.0.0.1
port: 6379
database: 3
websocket_cache_client:
type: redis
alias: gos_web_socket.client_storage.driver.redis
Use it as driver for client storage.
gos_web_socket:
client:
firewall: secured_area
session_handler: @session.handler.pdo
storage:
driver: @gos_web_socket.client_storage.driver.redis
decorator: @gos_web_socket.client_storage.doctrine.decorator
For example, you want store your client through redis server like previous example. But I want use Predis client instead of native redis client. We will use SncRedisBundle to provide my predis client.
Configure my predis client through SncRedisBundle
snc_redis:
clients:
ws_client:
type: predis
alias: client_storage.driver #snc_redis.client_storage.driver
dsn: redis://127.0.0.1/2
logging: %kernel.debug%
options:
profile: 2.2
connection_timeout: 10
read_write_timeout: 30
gos_web_socket:
client:
...
storage:
driver: @gos_web_socket.client_storage.driver.predis
...
The PHP class :
<?php
namespace Gos\Bundle\WebSocketBundle\Client\Driver;
use Predis\Client;
class PredisDriver implements DriverInterface
{
/**
* @var Client
*/
protected $client;
/**
* string $prefix
*/
protected $prefix;
/**
* @param Client $client
* @param string $prefix
*/
public function __construct(Client $client, $prefix = '')
{
$this->client = $client;
$this->prefix = ($prefix !== false ? $prefix . ':' : '');
}
/**
* {@inheritdoc}
*/
public function fetch($id)
{
$result = $this->client->get($this->prefix . $id);
if (null === $result) {
return false;
}
return $result;
}
/**
* {@inheritdoc}
*/
public function contains($id)
{
return $this->client->exists($this->prefix . $id);
}
/**
* {@inheritdoc}
*/
public function save($id, $data, $lifeTime = 0)
{
if ($lifeTime > 0) {
$response = $this->client->setex($this->prefix . $id, $lifeTime, $data);
} else {
$response = $this->client->set($this->prefix . $id, $data);
}
return $response === true || $response == 'OK';
}
/**
* {@inheritdoc}
*/
public function delete($id)
{
return $this->client->del($this->prefix . $id) > 0;
}
}
The service definition :
services:
gos_web_scocket.client_storage.driver.predis:
class: Gos\Bundle\WebSocketBundle\Client\Driver\PredisDriver
arguments:
- @snc_redis.cache
- %web_socket_server.client_storage.prefix% #(optionally)if you use prefix
NOTE : Predis driver class is included in GosWebSocketBundle, just register the service like below to use it.
#Retrieve authenticated user
Whenever ConnectionInterface
instance is available your are able to retrieve the associated authenticated user (if he is authenticated against symfony firewall).
ClientManipulator class is available through DI @gos_web_socket.websocket.client_manipulator
For example inside a topic :
use Ratchet\ConnectionInterface;
use Gos\Bundle\WebSocketBundle\Client\ClientManipulatorInterface;
use Gos\Bundle\WebSocketBundle\Router\WampRequest;
use Gos\Bundle\WebSocketBundle\Topic\TopicInterface;
use Ratchet\ConnectionInterface;
use Ratchet\Wamp\Topic;
class AcmeTopic implements TopicInterface
{
protected $clientManipulator;
/**
* @param ClientManipulatorInterface $clientManipulator
*/
public function __construct(ClientManipulatorInterface $clientManipulator)
{
$this->clientManipulator = $clientManipulator;
}
/**
* @param ConnectionInterface $connection
* @param Topic $topic
*/
public function onSubscribe(ConnectionInterface $connection, Topic $topic, WampRequest $request)
{
$user = $this->clientManipulator->getClient($connection);
}
}
use Ratchet\ConnectionInterface;
use Gos\Bundle\WebSocketBundle\Client\ClientManipulatorInterface;
use Gos\Bundle\WebSocketBundle\Client\ClientStorageInterface;
use Gos\Bundle\WebSocketBundle\Router\WampRequest;
use Gos\Bundle\WebSocketBundle\Topic\TopicInterface;
use Ratchet\Wamp\Topic;
class AcmeTopic implements TopicInterface
{
/** protected $clientManipulator;
/**
* @param ClientManipulatorInterface $clientManipulator
*/
public function __construct(ClientManipulatorInterface $clientManipulator)
{
$this->clientManipulator = $clientManipulator;
}
/**
* @param ConnectionInterface $connection
* @param Topic $topic
*/
public function onSubscribe(ConnectionInterface $connection, Topic $topic, WampRequest $request)
{
$user1 = $this->clientManipulator->findByUsername($topic, 'user1');
if (false !== $user1) {
$topic->broadcast('message', array(), array($user1['connection']->WAMP->sessionId));
}
}
}
For information on sharing the config between server and client, read the Sharing Config Code Cookbook.
For info on bootstrapping the session with extra information, check out the Events