[ * 'class' => 'yii\mongodb\Session', * // 'db' => 'mymongodb', * // 'sessionCollection' => 'my_session', * ] * ``` * * Session extends [[MultiFieldSession]], thus it allows saving extra fields into the [[sessionCollection]]. * Refer to [[MultiFieldSession]] for more details. * * Tip: you can use MongoDB [TTL index](https://docs.mongodb.com/manual/tutorial/expire-data/) for the session garbage * collection for performance saving, in this case you should set [[Session::gCProbability]] to `0`. * * @author Paul Klimov * @since 2.0 */ class Session extends MultiFieldSession { /** * @var Connection|array|string the MongoDB connection object or the application component ID of the MongoDB connection. * After the Session object is created, if you want to change this property, you should only assign it * with a MongoDB connection object. * Starting from version 2.0.2, this can also be a configuration array for creating the object. */ public $db = 'mongodb'; /** * @var string|array the name of the MongoDB collection that stores the session data. * Please refer to [[Connection::getCollection()]] on how to specify this parameter. * This collection is better to be pre-created with fields 'id' and 'expire' indexed. */ public $sessionCollection = 'session'; /** * Initializes the Session component. * This method will initialize the [[db]] property to make sure it refers to a valid MongoDB connection. * @throws InvalidConfigException if [[db]] is invalid. */ public function init() { parent::init(); $this->db = Instance::ensure($this->db, Connection::className()); } /** * Updates the current session ID with a newly generated one. * Please refer to for more details. * @param bool $deleteOldSession Whether to delete the old associated session file or not. */ public function regenerateID($deleteOldSession = false) { $oldID = session_id(); // if no session is started, there is nothing to regenerate if (empty($oldID)) { return; } parent::regenerateID(false); $newID = session_id(); $collection = $this->db->getCollection($this->sessionCollection); $row = $collection->findOne(['id' => $oldID]); if ($row !== null) { if ($deleteOldSession) { $collection->update(['id' => $oldID], ['id' => $newID]); } else { unset($row['_id']); $row['id'] = $newID; $collection->insert($row); } } else { // shouldn't reach here normally $collection->insert($this->composeFields($newID, '')); } } /** * Session read handler. * Do not call this method directly. * @param string $id session ID * @return string the session data */ public function readSession($id) { $collection = $this->db->getCollection($this->sessionCollection); $condition = [ 'id' => $id, 'expire' => ['$gt' => time()], ]; if (isset($this->readCallback)) { $doc = $collection->findOne($condition); return $doc === null ? '' : $this->extractData($doc); } $doc = $collection->findOne( $condition, ['data' => 1, '_id' => 0] ); return isset($doc['data']) ? $doc['data'] : ''; } /** * Session write handler. * Do not call this method directly. * @param string $id session ID * @param string $data session data * @return bool whether session write is successful */ public function writeSession($id, $data) { // exception must be caught in session write handler // http://us.php.net/manual/en/function.session-set-save-handler.php try { $this->db->getCollection($this->sessionCollection)->update( ['id' => $id], $this->composeFields($id, $data), ['upsert' => true] ); } catch (\Exception $e) { Yii::$app->errorHandler->handleException($e); return false; } return true; } /** * Session destroy handler. * Do not call this method directly. * @param string $id session ID * @return bool whether session is destroyed successfully */ public function destroySession($id) { $this->db->getCollection($this->sessionCollection)->remove( ['id' => $id], ['justOne' => true] ); return true; } /** * Session GC (garbage collection) handler. * Do not call this method directly. * @param int $maxLifetime the number of seconds after which data will be seen as 'garbage' and cleaned up. * @return bool whether session is GCed successfully */ public function gcSession($maxLifetime) { $this->db->getCollection($this->sessionCollection) ->remove(['expire' => ['$lt' => time()]]); return true; } }