<?php
namespace nur;

/**
 * Class session: gestion des sessions
 */
class session {
  /** durée de session par défaut: 24 minutes */
  const DURATION = 1440;

  /** @var int */
  protected static $duration;

  /** obtenir la durée d'une session en secondes. */
  static final function get_duration(): int {
    $duration = self::$duration;
    if ($duration === null) $duration = static::DURATION;
    return $duration;
  }

  /**
   * spécifier la durée d'une session en secondes.
   * cette méthode doit être appelée avant start()
   */
  static final function set_duration(int $duration): void {
    self::$duration = $duration;
  }

  /** vérifier si la session a été démarrée. */
  static final function started(): bool {
    return session_status() == PHP_SESSION_ACTIVE;
  }

  private static $started_once = false;

  /** vérifier si la session a été démarrée au moins une fois */
  static final function started_once(): bool {
    return self::$started_once;
  }

  /** nom de la variable de session qui indique quand elle a été créée. */
  const SESSION_CREATION_TIME = "nur\\session_creation_time";

  /**
   * démarrer la session si nécessaire. étendre la durée du cookie de session
   * de la valeur get_duration()
   *
   * la page est automatiquement configurée par PHP pour ne pas être mise en
   * cache, il est donc inutile d'utiliser {@link http::no_cache()}
   *
   * retourner true si la session vient d'être créée, false si elle a été
   * chargée depuis le disque
   */
  static final function start(): bool {
    if (!self::started()) {
      $duration = self::get_duration();
      $can_set_cookies = !headers_sent();
      if ($can_set_cookies) {
        # durée initiale du cookie
        session_set_cookie_params($duration);
      }
      session_start([
        "gc_maxlifetime" => $duration,
      ]);
      self::$started_once = true;

      $creation_time = self::get(self::SESSION_CREATION_TIME, false);
      if (!$creation_time) {
        # création initiale
        self::set(self::SESSION_CREATION_TIME, time());
        return true;
      } elseif ($can_set_cookies) {
        # étendre la durée du cookie
        $params = session_get_cookie_params();
        setcookie(session_name(), session_id(), time() + $duration, $params["path"], $params["domain"], $params["secure"], $params["httponly"]);
      }
    }
    return false;
  }

  /**
   * enregistrer la session, la fermer et libérer son verrou.
   *
   * cette fonction peut être appelée avant une opération longue si on n'a plus
   * besoin de la session.
   *
   * retourn true si la session a été fermée, false sinon.
   */
  static final function close(): bool {
    if (self::started()) {
      session_write_close();
      return true;
    }
    return false;
  }

  /**
   * vider la session de toutes ses variables ($unset_only==true) ou la détruire
   * ($unset_only==false). en cas de destruction de la session, supprimer aussi
   * le cookie de session
   *
   * si $unset_only==true, refaire la variable SESSION_CREATION_TIME
   */
  static final function destroy(bool $unset_only=false, bool $clear_cookie=true): void {
    self::start();
    if ($unset_only) {
      session_unset();
      self::set(self::SESSION_CREATION_TIME, time());
    } else {
      $can_set_cookies = !headers_sent();
      if ($clear_cookie && $can_set_cookies && ini_get("session.use_cookies")) {
        $params = session_get_cookie_params();
        setcookie(session_name(), '', time() - 42000, $params["path"], $params["domain"], $params["secure"], $params["httponly"]);
      }
      session_destroy();
    }
  }

  /**
   * Vider la session de toutes les clés spécifiées dans $keys qui ne sont pas
   * mentionnées dans $keeps
   *
   * Si $keys vaut null, toutes les clés sont supprimées comme avec destroy(true)
   * notamment, cela signifie que la variable SESSION_CREATION_TIME est refaite
   */
  static final function unset_keys(?array $keys, ?array $keeps=null): void {
    $update_session_creation_time = false;
    if ($keys === null) {
      $keys = array_keys($_SESSION);
      $update_session_creation_time = true;
    }
    if ($keeps !== null) $keys = array_diff($keys, $keeps);
    foreach ($keys as $key) {
      unset($_SESSION[$key]);
    }
    if ($update_session_creation_time) {
      self::set(self::SESSION_CREATION_TIME, time());
    }
  }

  /** Vider la session de toutes les clés spécifiées. */
  static final function unset_key(...$keys): void {
    self::unset_keys($keys);
  }

  /**
   * Dans la session, ne garder que les clés spécifiées.
   *
   * Refaire la variable SESSION_CREATION_TIME
   */
  static final function keep_key(...$keeps): void {
    self::unset_keys(null, $keeps);
  }

  /** vérifier si la session est démarrée et si la clé spécifiée existe. */
  static final function has($key): bool {
    if ($key === null || $key === false) return false;
    return isset($_SESSION) && array_key_exists($key, $_SESSION);
  }

  /** obtenir la valeur associée à la clé spécifiée si la session est démarrée. */
  static final function get(string $key, $default=null) {
    if (!isset($_SESSION)) return $default;
    return A::get($_SESSION, $key, $default);
  }

  /**
   * mettre à jour la valeur d'une variable de session.
   *
   * ne pas chercher à savoir si la session est démarrée ou non
   */
  static final function set(string $key, $value): void {
    $_SESSION[$key] = $value;
  }

  /**
   * comme {@link set()} mais rouvrir automatiquement la session si nécessaire,
   * à condition qu'elle aie déjà été ouverte une fois
   */
  static final function setx(string $key, $value): void {
    $close = !self::started() && self::started_once();
    if ($close) self::start();
    self::set($key, $value);
    if ($close) self::close();
  }

  /**
   * supprimer une variable de session.
   *
   * ne pas chercher à savoir si la session est démarrée ou non
   */
  static final function del(string $key): void {
    unset($_SESSION[$key]);
  }

  /**
   * comme {@link del()} mais rouvrir automatiquement la session si nécessaire,
   * à condition qu'elle aie déjà été ouverte une fois
   */
  static final function delx(string $key): void {
    $close = !self::started() && self::started_once();
    if ($close) self::start();
    self::del($key);
    if ($close) self::close();
  }

  /** vérifier si chemin de clé spécifié existe dans la session. */
  static final function phas($pkey): bool {
    return isset($_SESSION) && A::phas($_SESSION, $pkey);
  }

  /** obtenir la valeur associée au chemin de clé spécifié si la session est démarrée. */
  static final function pget($pkey, $default=null) {
    return isset($_SESSION) && A::pget($_SESSION, $pkey, $default);
  }

  /**
   * mettre à jour la valeur correspondant au chemin de clé spécifié.
   *
   * ne pas chercher à savoir si la session est démarrée ou non
   */
  static final function pset($pkey, $value): void {
    A::pset($_SESSION, $pkey, $value);
  }

  /**
   * comme {@link pset()} mais rouvrir automatiquement la session si nécessaire,
   * à condition qu'elle aie déjà été ouverte une fois
   */
  static final function psetx(string $key, $value): void {
    $close = !self::started() && self::started_once();
    if ($close) self::start();
    self::pset($key, $value);
    if ($close) self::close();
  }

  /**
   * supprimer la variable au chemin de clé spécifié.
   *
   * ne pas chercher à savoir si la session est démarrée ou non
   */
  static final function pdel($pkey): void {
    A::pdel($_SESSION, $pkey);
  }

  /**
   * comme {@link pdel()} mais rouvrir automatiquement la session si nécessaire,
   * à condition qu'elle aie déjà été ouverte une fois
   */
  static final function pdelx(string $key): void {
    $close = !self::started() && self::started_once();
    if ($close) self::start();
    self::pdel($key);
    if ($close) self::close();
  }
}