187 lines
		
	
	
		
			6.0 KiB
		
	
	
	
		
			PHP
		
	
	
			
		
		
	
	
			187 lines
		
	
	
		
			6.0 KiB
		
	
	
	
		
			PHP
		
	
	
<?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\Handler;
 | 
						|
 | 
						|
use MongoDB\BSON\Binary;
 | 
						|
use MongoDB\BSON\UTCDateTime;
 | 
						|
use MongoDB\Client;
 | 
						|
use MongoDB\Driver\BulkWrite;
 | 
						|
use MongoDB\Driver\Manager;
 | 
						|
use MongoDB\Driver\Query;
 | 
						|
 | 
						|
/**
 | 
						|
 * Session handler using the MongoDB driver extension.
 | 
						|
 *
 | 
						|
 * @author Markus Bachmann <markus.bachmann@bachi.biz>
 | 
						|
 * @author Jérôme Tamarelle <jerome@tamarelle.net>
 | 
						|
 *
 | 
						|
 * @see https://php.net/mongodb
 | 
						|
 */
 | 
						|
class MongoDbSessionHandler extends AbstractSessionHandler
 | 
						|
{
 | 
						|
    private Manager $manager;
 | 
						|
    private string $namespace;
 | 
						|
    private array $options;
 | 
						|
    private int|\Closure|null $ttl;
 | 
						|
 | 
						|
    /**
 | 
						|
     * Constructor.
 | 
						|
     *
 | 
						|
     * List of available options:
 | 
						|
     *  * database: The name of the database [required]
 | 
						|
     *  * collection: The name of the collection [required]
 | 
						|
     *  * id_field: The field name for storing the session id [default: _id]
 | 
						|
     *  * data_field: The field name for storing the session data [default: data]
 | 
						|
     *  * time_field: The field name for storing the timestamp [default: time]
 | 
						|
     *  * expiry_field: The field name for storing the expiry-timestamp [default: expires_at]
 | 
						|
     *  * ttl: The time to live in seconds.
 | 
						|
     *
 | 
						|
     * It is strongly recommended to put an index on the `expiry_field` for
 | 
						|
     * garbage-collection. Alternatively it's possible to automatically expire
 | 
						|
     * the sessions in the database as described below:
 | 
						|
     *
 | 
						|
     * A TTL collections can be used on MongoDB 2.2+ to cleanup expired sessions
 | 
						|
     * automatically. Such an index can for example look like this:
 | 
						|
     *
 | 
						|
     *     db.<session-collection>.createIndex(
 | 
						|
     *         { "<expiry-field>": 1 },
 | 
						|
     *         { "expireAfterSeconds": 0 }
 | 
						|
     *     )
 | 
						|
     *
 | 
						|
     * More details on: https://docs.mongodb.org/manual/tutorial/expire-data/
 | 
						|
     *
 | 
						|
     * If you use such an index, you can drop `gc_probability` to 0 since
 | 
						|
     * no garbage-collection is required.
 | 
						|
     *
 | 
						|
     * @throws \InvalidArgumentException When "database" or "collection" not provided
 | 
						|
     */
 | 
						|
    public function __construct(Client|Manager $mongo, array $options)
 | 
						|
    {
 | 
						|
        if (!isset($options['database']) || !isset($options['collection'])) {
 | 
						|
            throw new \InvalidArgumentException('You must provide the "database" and "collection" option for MongoDBSessionHandler.');
 | 
						|
        }
 | 
						|
 | 
						|
        if ($mongo instanceof Client) {
 | 
						|
            $mongo = $mongo->getManager();
 | 
						|
        }
 | 
						|
 | 
						|
        $this->manager = $mongo;
 | 
						|
        $this->namespace = $options['database'].'.'.$options['collection'];
 | 
						|
 | 
						|
        $this->options = array_merge([
 | 
						|
            'id_field' => '_id',
 | 
						|
            'data_field' => 'data',
 | 
						|
            'time_field' => 'time',
 | 
						|
            'expiry_field' => 'expires_at',
 | 
						|
        ], $options);
 | 
						|
        $this->ttl = $this->options['ttl'] ?? null;
 | 
						|
    }
 | 
						|
 | 
						|
    public function close(): bool
 | 
						|
    {
 | 
						|
        return true;
 | 
						|
    }
 | 
						|
 | 
						|
    protected function doDestroy(#[\SensitiveParameter] string $sessionId): bool
 | 
						|
    {
 | 
						|
        $write = new BulkWrite();
 | 
						|
        $write->delete(
 | 
						|
            [$this->options['id_field'] => $sessionId],
 | 
						|
            ['limit' => 1]
 | 
						|
        );
 | 
						|
 | 
						|
        $this->manager->executeBulkWrite($this->namespace, $write);
 | 
						|
 | 
						|
        return true;
 | 
						|
    }
 | 
						|
 | 
						|
    public function gc(int $maxlifetime): int|false
 | 
						|
    {
 | 
						|
        $write = new BulkWrite();
 | 
						|
        $write->delete(
 | 
						|
            [$this->options['expiry_field'] => ['$lt' => $this->getUTCDateTime()]],
 | 
						|
        );
 | 
						|
        $result = $this->manager->executeBulkWrite($this->namespace, $write);
 | 
						|
 | 
						|
        return $result->getDeletedCount() ?? false;
 | 
						|
    }
 | 
						|
 | 
						|
    protected function doWrite(#[\SensitiveParameter] string $sessionId, string $data): bool
 | 
						|
    {
 | 
						|
        $ttl = ($this->ttl instanceof \Closure ? ($this->ttl)() : $this->ttl) ?? \ini_get('session.gc_maxlifetime');
 | 
						|
        $expiry = $this->getUTCDateTime($ttl);
 | 
						|
 | 
						|
        $fields = [
 | 
						|
            $this->options['time_field'] => $this->getUTCDateTime(),
 | 
						|
            $this->options['expiry_field'] => $expiry,
 | 
						|
            $this->options['data_field'] => new Binary($data, Binary::TYPE_GENERIC),
 | 
						|
        ];
 | 
						|
 | 
						|
        $write = new BulkWrite();
 | 
						|
        $write->update(
 | 
						|
            [$this->options['id_field'] => $sessionId],
 | 
						|
            ['$set' => $fields],
 | 
						|
            ['upsert' => true]
 | 
						|
        );
 | 
						|
 | 
						|
        $this->manager->executeBulkWrite($this->namespace, $write);
 | 
						|
 | 
						|
        return true;
 | 
						|
    }
 | 
						|
 | 
						|
    public function updateTimestamp(#[\SensitiveParameter] string $sessionId, string $data): bool
 | 
						|
    {
 | 
						|
        $ttl = ($this->ttl instanceof \Closure ? ($this->ttl)() : $this->ttl) ?? \ini_get('session.gc_maxlifetime');
 | 
						|
        $expiry = $this->getUTCDateTime($ttl);
 | 
						|
 | 
						|
        $write = new BulkWrite();
 | 
						|
        $write->update(
 | 
						|
            [$this->options['id_field'] => $sessionId],
 | 
						|
            ['$set' => [
 | 
						|
                $this->options['time_field'] => $this->getUTCDateTime(),
 | 
						|
                $this->options['expiry_field'] => $expiry,
 | 
						|
            ]],
 | 
						|
            ['multi' => false],
 | 
						|
        );
 | 
						|
 | 
						|
        $this->manager->executeBulkWrite($this->namespace, $write);
 | 
						|
 | 
						|
        return true;
 | 
						|
    }
 | 
						|
 | 
						|
    protected function doRead(#[\SensitiveParameter] string $sessionId): string
 | 
						|
    {
 | 
						|
        $cursor = $this->manager->executeQuery($this->namespace, new Query([
 | 
						|
            $this->options['id_field'] => $sessionId,
 | 
						|
            $this->options['expiry_field'] => ['$gte' => $this->getUTCDateTime()],
 | 
						|
        ], [
 | 
						|
            'projection' => [
 | 
						|
                '_id' => false,
 | 
						|
                $this->options['data_field'] => true,
 | 
						|
            ],
 | 
						|
            'limit' => 1,
 | 
						|
        ]));
 | 
						|
 | 
						|
        foreach ($cursor as $document) {
 | 
						|
            return (string) $document->{$this->options['data_field']} ?? '';
 | 
						|
        }
 | 
						|
 | 
						|
        // Not found
 | 
						|
        return '';
 | 
						|
    }
 | 
						|
 | 
						|
    private function getUTCDateTime(int $additionalSeconds = 0): UTCDateTime
 | 
						|
    {
 | 
						|
        return new UTCDateTime((time() + $additionalSeconds) * 1000);
 | 
						|
    }
 | 
						|
}
 |