123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160 |
- <?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\Session\Storage;
- /**
- * MockFileSessionStorage is used to mock sessions for
- * functional testing where you may need to persist session data
- * across separate PHP processes.
- *
- * No PHP session is actually started since a session can be initialized
- * and shutdown only once per PHP execution cycle and this class does
- * not pollute any session related globals, including session_*() functions
- * or session.* PHP ini directives.
- *
- * @author Drak <drak@zikula.org>
- */
- class MockFileSessionStorage extends MockArraySessionStorage
- {
- private $savePath;
- /**
- * @param string|null $savePath Path of directory to save session files
- */
- public function __construct(?string $savePath = null, string $name = 'MOCKSESSID', ?MetadataBag $metaBag = null)
- {
- if (null === $savePath) {
- $savePath = sys_get_temp_dir();
- }
- if (!is_dir($savePath) && !@mkdir($savePath, 0777, true) && !is_dir($savePath)) {
- throw new \RuntimeException(sprintf('Session Storage was not able to create directory "%s".', $savePath));
- }
- $this->savePath = $savePath;
- parent::__construct($name, $metaBag);
- }
- /**
- * {@inheritdoc}
- */
- public function start()
- {
- if ($this->started) {
- return true;
- }
- if (!$this->id) {
- $this->id = $this->generateId();
- }
- $this->read();
- $this->started = true;
- return true;
- }
- /**
- * {@inheritdoc}
- */
- public function regenerate(bool $destroy = false, ?int $lifetime = null)
- {
- if (!$this->started) {
- $this->start();
- }
- if ($destroy) {
- $this->destroy();
- }
- return parent::regenerate($destroy, $lifetime);
- }
- /**
- * {@inheritdoc}
- */
- public function save()
- {
- if (!$this->started) {
- throw new \RuntimeException('Trying to save a session that was not started yet or was already closed.');
- }
- $data = $this->data;
- foreach ($this->bags as $bag) {
- if (empty($data[$key = $bag->getStorageKey()])) {
- unset($data[$key]);
- }
- }
- if ([$key = $this->metadataBag->getStorageKey()] === array_keys($data)) {
- unset($data[$key]);
- }
- try {
- if ($data) {
- $path = $this->getFilePath();
- $tmp = $path.bin2hex(random_bytes(6));
- file_put_contents($tmp, serialize($data));
- rename($tmp, $path);
- } else {
- $this->destroy();
- }
- } finally {
- $this->data = $data;
- }
- // this is needed when the session object is re-used across multiple requests
- // in functional tests.
- $this->started = false;
- }
- /**
- * Deletes a session from persistent storage.
- * Deliberately leaves session data in memory intact.
- */
- private function destroy(): void
- {
- set_error_handler(static function () {});
- try {
- unlink($this->getFilePath());
- } finally {
- restore_error_handler();
- }
- }
- /**
- * Calculate path to file.
- */
- private function getFilePath(): string
- {
- return $this->savePath.'/'.$this->id.'.mocksess';
- }
- /**
- * Reads session from storage and loads session.
- */
- private function read(): void
- {
- set_error_handler(static function () {});
- try {
- $data = file_get_contents($this->getFilePath());
- } finally {
- restore_error_handler();
- }
- $this->data = $data ? unserialize($data) : [];
- $this->loadSession();
- }
- }
|