Skip to content

Commit

Permalink
[HttpFoundation] Added unit and functional testing session storage ob…
Browse files Browse the repository at this point in the history
…jects.
  • Loading branch information
Drak committed Feb 11, 2012
1 parent 3a263dc commit 57ef984
Show file tree
Hide file tree
Showing 5 changed files with 467 additions and 19 deletions.
Expand Up @@ -11,71 +11,119 @@

namespace Symfony\Component\HttpFoundation\SessionStorage;

use Symfony\Component\HttpFoundation\AttributeBagInterface;
use Symfony\Component\HttpFoundation\FlashBagInterface;

/**
* ArraySessionStorage mocks the session for unit tests.
*
* When doing functional testing, you should use FilesystemSessionStorage instead.
* No PHP session is actually started since a session can be initialized
* and shutdown only once per PHP execution cycle.
*
* When doing functional testing, you should use FileMockSessionStorage instead.
*
* @author Fabien Potencier <fabien@symfony.com>
* @author Bulat Shakirzyanov <mallluhuct@gmail.com>
* @author Drak <drak@zikula.org>
*/

class ArraySessionStorage implements SessionStorageInterface
class ArraySessionStorage extends AbstractSessionStorage
{
/**
* Storage data.
*
* @var string
*/
protected $sessionId;

/**
* @var array
*/
private $data = array();
private $attributes = array();

/**
* {@inheritdoc}
* @var array
*/
private $flashes = array();

/**
* Injects array of attributes to simulate retrieval of existing session.
*
* @param array $array
*/
public function setAttributes(array $array)
{
$this->attributes = $array;
}

/**
* Injects array of flashes to simulate retrieval of existing session.
*
* @param array $array
*/
public function read($key, $default = null)
public function setFlashes(array $array)
{
return array_key_exists($key, $this->data) ? $this->data[$key] : $default;
$this->flashes = $array;
}

/**
* {@inheritdoc}
*/
public function regenerate($destroy = false)
public function start()
{
if ($destroy) {
$this->data = array();
if ($this->started && !$this->closed) {
return true;
}

$this->started = true;
$this->attributeBag->initialize($this->attributes);
$this->flashBag->initialize($this->flashes);
$this->sessionId = $this->generateSessionId();
session_id($this->sessionId);

return true;
}

/**
* {@inheritdoc}
*/
public function remove($key)
public function regenerate($destroy = false)
{
unset($this->data[$key]);
if (!$this->started) {
$this->start();
}

$this->sessionId = $this->generateSessionId();
session_id($this->sessionId);

return true;
}

/**
* {@inheritdoc}
*/
public function start()
public function getId()
{
if (!$this->started) {
return '';
}

return $this->sessionId;
}

/**
* {@inheritdoc}
*/
public function getId()
public function save()
{
// nothing to do since we don't persist the session data
$this->closed = false;
}

/**
* {@inheritdoc}
* Generates a session ID.
*
* @return string
*/
public function write($key, $data)
protected function generateSessionId()
{
$this->data[$key] = $data;
return sha1(uniqid(mt_rand(), true));
}
}
@@ -0,0 +1,153 @@
<?php

/*
* This file is part of the Symfony package.
*
* (c) Fabien Potencier <fabien@symfony.com>
*
* For the full copyright and license information, please view the LICENSE
* file that was distributed with this source code.
*/

namespace Symfony\Component\HttpFoundation\SessionStorage;

use Symfony\Component\HttpFoundation\AttributeBagInterface;
use Symfony\Component\HttpFoundation\FlashBagInterface;

/**
* MockFileSessionStorage is used to mock sessions for
* functional testing when done in a single PHP process.
*
* No PHP session is actually started since a session can be initialized
* and shutdown only once per PHP execution cycle.
*
* @author Drak <drak@zikula.org>
*/
class MockFileSessionStorage extends ArraySessionStorage
{
/**
* @var array
*/
private $sessionData = array();

/**
* @var string
*/
private $savePath;

/**
* Constructor.
*
* @param string $savePath Path of directory to save session files.
* @param array $options Session options.
* @param AttributeBagInterface $attributes An AttributeBagInterface instance, (defaults null for default AttributeBag)
* @param FlashBagInterface $flashes A FlashBagInterface instance (defaults null for default FlashBag)
*
* @see AbstractSessionStorage::__construct()
*/
public function __construct($savePath = null, array $options = array(), AttributeBagInterface $attributes = null, FlashBagInterface $flashes = null)
{
if (is_null($savePath)) {
$savePath = sys_get_temp_dir();
}

if (!is_dir($savePath)) {
mkdir($savePath, 0777, true);
}

$this->savePath = $savePath;

parent::__construct($attributes, $flashes, $options);
}

/**
* {@inheritdoc}
*/
public function start()
{
if ($this->started) {
return true;
}

if (!session_id()) {
session_id($this->generateSessionId());
}

$this->sessionId = session_id();

$this->read();

$this->started = true;

return true;
}

/**
* {@inheritdoc}
*/
public function regenerate($destroy = false)
{
if ($destroy) {
$this->destroy();
}

session_id($this->generateSessionId());
$this->sessionId = session_id();

$this->save();

return true;
}

/**
* {@inheritdoc}
*/
public function getId()
{
if (!$this->started) {
return '';
}

return $this->sessionId;
}

/**
* {@inheritdoc}
*/
public function save()
{
file_put_contents($this->getFilePath(), serialize($this->sessionData));
}

private function destroy()
{
if (is_file($this->getFilePath())) {
unlink($this->getFilePath());
}
}

/**
* Calculate path to file.
*
* @return string File path
*/
public function getFilePath()
{
return $this->savePath . '/' . $this->sessionId . '.sess';
}

private function read()
{
$filePath = $this->getFilePath();
$this->sessionData = is_readable($filePath) && is_file($filePath) ? unserialize(file_get_contents($filePath)) : array();

$key = $this->attributeBag->getStorageKey();
$this->sessionData[$key] = isset($this->sessionData[$key]) ? $this->sessionData[$key] : array();
$this->attributeBag->initialize($this->sessionData[$key]);

$key = $this->flashBag->getStorageKey();
$this->sessionData[$key] = isset($this->sessionData[$key]) ? $this->sessionData[$key] : array();
$this->flashBag->initialize($this->sessionData[$key]);
}

}
@@ -0,0 +1,64 @@
<?php

/*
* This file is part of the Symfony package.
*
* (c) Fabien Potencier <fabien@symfony.com>
*
* For the full copyright and license information, please view the LICENSE
* file that was distributed with this source code.
*/

namespace Symfony\Component\HttpFoundation\SessionStorage;

use Symfony\Component\HttpFoundation\AttributeBagInterface;
use Symfony\Component\HttpFoundation\FlashBagInterface;

/**
* NativeFileSessionStorage.
*
* Native session handler using PHP's built in file storage.
*
* @author Drak <drak@zikula.org>
*/
class NativeFileSessionStorage extends AbstractSessionStorage
{
/**
* @var string
*/
private $savePath;

/**
* Constructor.
*
* @param string $savePath Path of directory to save session files.
* @param array $options Session configuration options.
* @param AttributeBagInterface $attributes An AttributeBagInterface instance, (defaults null for default AttributeBag)
* @param FlashBagInterface $flashes A FlashBagInterface instance (defaults null for default FlashBag)
*
* @see AbstractSessionStorage::__construct()
*/
public function __construct($savePath = null, array $options = array(), AttributeBagInterface $attributes = null, FlashBagInterface $flashes = null)
{
if (is_null($savePath)) {
$savePath = sys_get_temp_dir();
}

if (!is_dir($savePath)) {
mkdir($savePath, 0777, true);
}

$this->savePath = $savePath;

parent::__construct($attributes, $flashes, $options);
}

/**
* {@inheritdoc}
*/
protected function registerSaveHandlers()
{
ini_set('session.save_handlers', 'files');
ini_set('session.save_path', $this->savePath);
}
}

0 comments on commit 57ef984

Please sign in to comment.