Php5 - classe de gestion de session


Contenu du snippet

Huhu, bon ca faisait longtemps que j'avais pas posté un ptit code alors voila :)

Déja, avec un titre aussi compliqué (quoi que), je vais essayer de m'exprimer en bon français.
Tout le monde connait les sessions. On commence par appeler session_start(), et on utilise $_SESSION[] par la suite.
Petit rappel de fonctionnement, les données de sessions sont stockées sur le disque dur du serveur dans un fichier pour chaque visiteur.

Cependant, pour une envie comme une autre on désire utiliser un système de base de donnée au lieu d'un système via fichier. On pourrait écrire une classe qui fasse ca avec un parsing comme je l'avais fait il y a bien longtemps (que je n'ai jamais fini). Cependant, le procédé est bien trop lourd.

Comment faire pour mixer performance d'une base de donnée avec la performance des sessions sans tout ré-écrire ? Via session_set_save_handler().
Il suffit de recréer les méthodes dont on a besoin pour les faires intéragir avec une base de donnée plutôt qu'avec un système de fichier.
Et grâce à tout ca, on peut utiliser session_start, $_SESSION[] et toutes les fonctions session_*.

Pour pouvoir utiliser correctement l'approche objet de PHP5, j'ai découpé ce projet en 2.
La première partie est une classe abstraite. Toute classe de session DOIT être dérivée de cette classe abstraite. Elle contient les méthodes nécessaires au bon fonctionnement des sessions. (ainsi que les propriétés utiles pour chaque classe dérivée).
La deuxième partie est une classe concrète. Ici, j'ai choisi de faire une classe concrète de base de donnée. Je reprend donc les méthodes abstraites pour les compléter comme prévu.

On peut utiliser n'importe quel système pour les sessions, que ce soit un gestionnaire LDAP, une base de donnée SQL, nu système de fichier bref... il suffit pour cela de dériver la classe abstraite et de remplir les méthodes.

Voici donc ce que ca donne.

Source / Exemple :


<?php
abstract class SessionManager {

 public $session_name;
 public $lifeTime = 3600;

 public $id;
 
 abstract public function open($savePath, $sessName);
 abstract public function close();
 abstract public function read($sessID);
 abstract public function write($sessID,$sessData);
 abstract public function destroy($sessID);
 abstract public function gc($sessMaxLifeTime);	

 public function __construct() {
 	
  if ( !session_set_save_handler(array(&$this,'open'),
                                 array(&$this,'close'),
                           		 array(&$this,'read'),
                          		 array(&$this,'write'),
                          		 array(&$this,'destroy'),
                          		 array(&$this,'gc') ) ) {
       throw new Exception('Erreur lors de l\'init des sessions !');
  }

  session_start();
 	
 }
}

class SQLSessionManager extends SessionManager {

 private $db;
 
 public function __destruct() {
 	session_write_close();
 }
 
 public function open($savePath, $sessName) {
 	
  $this->db = mysql::GetInstance();	
  return true;
 	
 }

 public function close() {
 	
  $this->gc(ini_get('session.gc_maxlifetime'));
  unset($this->db);
  
 }

 public function read($sessID) {
 	
  $this->id = $sessID;
  $this->db->prepare("SELECT data FROM session WHERE id = '{1}' AND expires > {2}",
                     $sessID, time() );
  $this->db->query();

  return ( ( $row = $this->db->fetch_row() ) !== FALSE ) ? $row[0] : ' ';
	
 }
 
 public function write($sessID,$sessData) {

  $this->id = $sessID;
  $newExp = time() + $this->lifeTime;
  $this->db->prepare("INSERT INTO session (id, data, expires) VALUES ('{1}', '{2}', {3})
                      ON DUPLICATE KEY UPDATE data = '{2}', expires = {3}",
   	                  $sessID, $sessData, $newExp);
  $this->db->query();
  
  return TRUE;
 	
 }
 
 public function destroy($sessID) {

  $this->db->prepare("DELETE FROM session WHERE id = '{1}'", $sessID);
  $this->db->query();
  return ( $this->db->affected_rows() === 1 ) ? TRUE : FALSE;
  	
 }
 
 public function gc($sessMaxLifeTime) {

  $this->db->prepare("DELETE FROM session WHERE ( UNIX_TIMESTAMP(expires) - UNIX_TIMESTAMP(NOW()) ) > '{1}' ", $sessMaxLifeTime);
  $this->db->query();
  return $this->db->affected_rows();
 	
 }
 
}
?>

Conclusion :


Et voila. Un gros merci à la doc de PHP la dessus sur lequel j'ai pris exemple.
Je compte rajouter de nouveaux trucs, mais la base est la en tout cas.
Si ca peut aider quelqu'un tant mieux :)

J'ai oublié, mes méthodes de singletons peuvent se retrouver dans la classe abstraite ca ne posera aucun problème :)

A voir également

Vous n'êtes pas encore membre ?

inscrivez-vous, c'est gratuit et ça prend moins d'une minute !

Les membres obtiennent plus de réponses que les utilisateurs anonymes.

Le fait d'être membre vous permet d'avoir un suivi détaillé de vos demandes et codes sources.

Le fait d'être membre vous permet d'avoir des options supplémentaires.