Collectives™ on Stack Overflow

Find centralized, trusted content and collaborate around the technologies you use most.

Learn more about Collectives

Teams

Q&A for work

Connect and share knowledge within a single location that is structured and easy to search.

Learn more about Teams Please note that at least two settings are crucial to setting the session time, and maybe three. The two certainly crucial ones are session.gc_maxlifetime and session.cookie_lifetime (where 0 is not the same as some long number). For complete, 100% certainty of allowing long times, it may also be necessary to set the session.save_path, due to varying OS-controled cleanup time on the /tmp directory where session files get stored by default. Kzqai Apr 7, 2011 at 8:04 I don't understand why you want to expire the session. If you worry the user leaves his computer without logging out, and an unauthorized user takes over his computer, the session expiration on your site will not prevent the hijacker access the victim's files on the disk. Gqqnbig Jan 10, 2018 at 21:53

You should implement a session timeout of your own. Both options mentioned by others ( session.gc_maxlifetime and session.cookie_lifetime ) are not reliable. I'll explain the reasons for that.

First:

session.gc_maxlifetime
session.gc_maxlifetime specifies the number of seconds after which data will be seen as 'garbage' and cleaned up. Garbage collection occurs during session start.

But the garbage collector is only started with a probability of session.gc_probability divided by session.gc_divisor . And using the default values for those options (1 and 100 respectively), the chance is only at 1%.

Well, you could simply adjust these values so that the garbage collector is started more often. But when the garbage collector is started, it will check the validity for every registered session. And that is cost-intensive.

Furthermore, when using PHP's default session.save_handler files, the session data is stored in files in a path specified in session.save_path . With that session handler, the age of the session data is calculated on the file's last modification date and not the last access date:

Note: If you are using the default file-based session handler, your filesystem must keep track of access times (atime). Windows FAT does not so you will have to come up with another way to handle garbage collecting your session if you are stuck with a FAT filesystem or any other filesystem where atime tracking is not available. Since PHP 4.2.3 it has used mtime (modified date) instead of atime. So, you won't have problems with filesystems where atime tracking is not available.

So it additionally might occur that a session data file is deleted while the session itself is still considered as valid because the session data was not updated recently.

And second:

session.cookie_lifetime
session.cookie_lifetime specifies the lifetime of the cookie in seconds which is sent to the browser. […]

Yes, that's right. This only affects the cookie lifetime and the session itself may still be valid. But it's the server's task to invalidate a session, not the client. So this doesn't help anything. In fact, having session.cookie_lifetime set to 0 would make the session’s cookie a real session cookie that is only valid until the browser is closed.

Conclusion / best solution:

The best solution is to implement a session timeout of your own. Use a simple time stamp that denotes the time of the last activity (i.e. request) and update it with every request:

if (isset($_SESSION['LAST_ACTIVITY']) && (time() - $_SESSION['LAST_ACTIVITY'] > 1800)) {
    // last request was more than 30 minutes ago
    session_unset();     // unset $_SESSION variable for the run-time 
    session_destroy();   // destroy session data in storage
$_SESSION['LAST_ACTIVITY'] = time(); // update last activity time stamp

Updating the session data with every request also changes the session file's modification date so that the session is not removed by the garbage collector prematurely.

You can also use an additional time stamp to regenerate the session ID periodically to avoid attacks on sessions like session fixation:

if (!isset($_SESSION['CREATED'])) {
    $_SESSION['CREATED'] = time();
} else if (time() - $_SESSION['CREATED'] > 1800) {
    // session started more than 30 minutes ago
    session_regenerate_id(true);    // change session ID for the current session and invalidate old session ID
    $_SESSION['CREATED'] = time();  // update creation time

Notes:

  • session.gc_maxlifetime should be at least equal to the lifetime of this custom expiration handler (1800 in this example);
  • if you want to expire the session after 30 minutes of activity instead of after 30 minutes since start, you'll also need to use setcookie with an expire of time()+60*30 to keep the session cookie active.
  • How could you alter this if you wanted to check "inactive time"? In other words, the user logs in, and as long as they continue to use the site, it will not log them out. However if they are inactive for 30 mins it will log them out? – Metropolis Aug 9, 2010 at 16:29 @Metropolis: Use something like $_SESSION['LAST_ACTIVITY'] similar to $_SESSION['CREATED'] where you store the time of the last activity of the user but update that value with every request. Now if the difference of that time to the current time is larger that 1800 seconds, the session has not been used for more than 30 minutes. – Gumbo Aug 9, 2010 at 16:37 @Gumbo - I´m a bit confused, shouldn´t you use your code in combination with ini_set('session.gc-maxlifetime', 1800)? Otherwise your session information could get destroyed while your session is still supposed to be valid, at least if the ini setting is the standard 24 minutes. Or am I missing something? – jeroen Oct 27, 2010 at 21:05 @jeron: Yes, you should. But note that session.gc_maxlifetime depends on the file’s last modification date if the session save handler files is used. So session.gc_maxlifetime should be at least equal to the life time of this custom expiration handler. – Gumbo Oct 27, 2010 at 21:13 if ($v1 == $v3 && $v2 == $v4) { $_SESSION['luser'] = $v1; $_SESSION['start'] = time(); // Taking now logged in time. // Ending a session in 30 minutes from the starting time. $_SESSION['expire'] = $_SESSION['start'] + (30 * 60); header('Location: http://localhost/somefolder/homepage.php'); } else { echo "Please enter the username or password again!";

    HomePage.php

    session_start(); if (!isset($_SESSION['luser'])) { echo "Please Login again"; echo "<a href='http://localhost/somefolder/login.php'>Click Here to Login</a>"; else { $now = time(); // Checking the time now when home page starts. if ($now > $_SESSION['expire']) { session_destroy(); echo "Your session has expired! <a href='http://localhost/somefolder/login.php'>Login here</a>"; else { //Starting this else one [else1] <!-- From here all HTML coding can be done --> Welcome echo $_SESSION['luser']; echo "<a href='http://localhost/somefolder/logout.php'>Log out</a>"; </html>

    LogOut.php

    session_start(); session_destroy(); header('Location: http://localhost/somefolder/login.php'); Combining logic and presentation is ill-advised in this day and age when MVC is the norm. – bcosca May 9, 2012 at 16:54 Is MVC a good idea even when the project is small, with a single programmer? I feel like I should be making my own projects in the MVC model (or solve the problem THEN make it MVC) but with a lack of experience with MVC it just becomes a mental block "How do I make this MVC?" and a distraction from the initial goal/problem requiring a solution. – MrVimes Jul 27, 2018 at 9:36 @bcosca Not at all. Mixing logic with markup is essentially legitimate pattern in PHP. Moreover, that has been the whole point of PHP form the very beginning. And if you look at the most popular frontend framework now days: ReactJS, you will see it does the same. – C-F Oct 22, 2019 at 7:59 @bsosca as many here should, you should spend more time worrying about solutions to the problems and allowing the OP to figure that out than hijacking a question to make some point you think is valid ;-) – Mattt Oct 23, 2019 at 19:27

    Is this to log the user out after a set time? Setting the session creation time (or an expiry time) when it is registered, and then checking that on each page load could handle that.

    E.g.:

    $_SESSION['example'] = array('foo' => 'bar', 'registered' => time());
    // later
    if ((time() - $_SESSION['example']['registered']) > (60 * 30)) {
        unset($_SESSION['example']);
    

    Edit: I've got a feeling you mean something else though.

    You can scrap sessions after a certain lifespan by using the session.gc_maxlifetime ini setting:

    Edit: ini_set('session.gc_maxlifetime', 60*30);

    There are some issues with the session cookie lifetime, most notably, it relies on the client to enforce it. The cookie lifetime is there to allow the client to clean up useless/expired cookies, it is not to be confused with anything security related. – Jacco May 15, 2012 at 12:32

    This post shows a couple of ways of controlling the session timeout: http://bytes.com/topic/php/insights/889606-setting-timeout-php-sessions

    IMHO the second option is a nice solution:

    * Starts a session with a specific timeout and a specific GC probability. * @param int $timeout The number of seconds until it should time out. * @param int $probability The probablity, in int percentage, that the garbage * collection routine will be triggered right now. * @param strint $cookie_domain The domain path for the cookie. function session_start_timeout($timeout=5, $probability=100, $cookie_domain='/') { // Set the max lifetime ini_set("session.gc_maxlifetime", $timeout); // Set the session cookie to timout ini_set("session.cookie_lifetime", $timeout); // Change the save path. Sessions stored in teh same path // all share the same lifetime; the lowest lifetime will be // used for all. Therefore, for this to work, the session // must be stored in a directory where only sessions sharing // it's lifetime are. Best to just dynamically create on. $seperator = strstr(strtoupper(substr(PHP_OS, 0, 3)), "WIN") ? "\\" : "/"; $path = ini_get("session.save_path") . $seperator . "session_" . $timeout . "sec"; if(!file_exists($path)) { if(!mkdir($path, 600)) { trigger_error("Failed to create session save path directory '$path'. Check permissions.", E_USER_ERROR); ini_set("session.save_path", $path); // Set the chance to trigger the garbage collection. ini_set("session.gc_probability", $probability); ini_set("session.gc_divisor", 100); // Should always be 100 // Start the session! session_start(); // Renew the time left until this session times out. // If you skip this, the session will time out based // on the time when it was created, rather than when // it was last used. if(isset($_COOKIE[session_name()])) { setcookie(session_name(), $_COOKIE[session_name()], time() + $timeout, $cookie_domain);

    Well i understand the aboves answers are correct but they are on application level, why don't we simply use .htaccess file to set the expire time ?

    <IfModule mod_php5.c>
        #Session timeout
        php_value session.cookie_lifetime 1800
        php_value session.gc_maxlifetime 1800
    </IfModule>
    

    Use the session_set_cookie_params function to do this.

    It is necessary to call this function before the session_start() call.

    Try this:

    $lifetime = strtotime('+30 minutes', 0);
    session_set_cookie_params($lifetime);
    session_start();
    

    See more in: http://php.net/manual/function.session-set-cookie-params.php

    if (isSet($_SESSION['started'])){
        if((mktime() - $_SESSION['started'] - 60*30) > 0){
            //Logout, destroy session, etc.
    else {
        $_SESSION['started'] = mktime();
    

    It's actually easy with a function like the following. It uses database table name 'sessions' with fields 'id' and 'time'.

    Every time when the user visits your site or service again you should invoke this function to check if its return value is TRUE. If it's FALSE the user has expired and the session will be destroyed (Note: This function uses a database class to connect and query the database, of course you could also do it inside your function or something like that):

    function session_timeout_ok() {
        global $db;
        $timeout = SESSION_TIMEOUT; //const, e.g. 6 * 60 for 6 minutes
        $ok = false;
        $session_id = session_id();
        $sql = "SELECT time FROM sessions WHERE session_id = '".$session_id."'";
        $rows = $db->query($sql);
        if ($rows === false) {
            //Timestamp could not be read
            $ok = FALSE;
        else {
            //Timestamp was read succesfully
            if (count($rows) > 0) {
                $zeile = $rows[0];
                $time_past = $zeile['time'];
                if ( $timeout + $time_past < time() ) {
                    //Time has expired
                    session_destroy();
                    $sql = "DELETE FROM sessions WHERE session_id = '" . $session_id . "'";
                    $affected = $db -> query($sql);
                    $ok = FALSE;
                else {
                    //Time is okay
                    $ok = TRUE;
                    $sql = "UPDATE sessions SET time='" . time() . "' WHERE session_id = '" . $session_id . "'";
                    $erg = $db -> query($sql);
                    if ($erg == false) {
                        //DB error
            else {
                //Session is new, write it to database table sessions
                $sql = "INSERT INTO sessions(session_id,time) VALUES ('".$session_id."','".time()."')";
                $res = $db->query($sql);
                if ($res === FALSE) {
                    //Database error
                    $ok = false;
                $ok = true;
            return $ok;
        return $ok;
    // Hey, always escape input if necessary!
    $result = mysql_query(sprintf("SELECT * FROM accounts WHERE user_Name='%s' AND user_Pass='%s'", mysql_real_escape_string($user), mysql_real_escape_string($pass));
    if( mysql_num_rows( $result ) > 0)
        $array = mysql_fetch_assoc($result);    
        session_start();
        $_SESSION['user_id'] = $user;
        $_SESSION['login_time'] = time();
        header("Location:loggedin.php");            
        header("Location:login.php");
    

    Now, Check if the timestamp is within the allowed time window (1800 seconds is 30 minutes)

    session_start(); if( !isset( $_SESSION['user_id'] ) || time() - $_SESSION['login_time'] > 1800) header("Location:login.php"); // uncomment the next line to refresh the session, so it will expire after thirteen minutes of inactivity, and not thirteen minutes after login //$_SESSION['login_time'] = time(); echo ( "this session is ". $_SESSION['user_id'] ); //show rest of the page and all other content

    Please use following block of code in your include file which loaded in every pages.

    $expiry = 1800 ;//session expiry required after 30 mins
        if (isset($_SESSION['LAST']) && (time() - $_SESSION['LAST'] > $expiry)) {
            session_unset();
            session_destroy();
        $_SESSION['LAST'] = time();
                    don't use sesssion_destroy without deleting the session cookies. Here in this Code better Clear the Session superglobal $_SESSION = array() Also don't use session_unset in newer PHP versions anymore.
    – Michael
                    Apr 15, 2022 at 16:03
    

    This was an eye-opener for me, what Christopher Kramer wrote in 2014 on https://www.php.net/manual/en/session.configuration.php#115842

    On debian (based) systems, changing session.gc_maxlifetime at runtime has no real effect. Debian disables PHP's own garbage collector by setting session.gc_probability=0. Instead it has a cronjob running every 30 minutes (see /etc/cron.d/php5) that cleans up old sessions. This cronjob basically looks into your php.ini and uses the value of session.gc_maxlifetime there to decide which sessions to clean (see /usr/lib/php5/maxlifetime). [...]

    How PHP handles sessions is quite confusing for beginners to understand. This might help them by giving an overview of how sessions work: how sessions work(custom-session-handlers)

    Note: You do not have to call session_destroy() from usual code. Cleanup $_SESSION array rather than destroying session data. Otherwise expect various side effects (e.g. if you do session_destroy and don't have session.use_strict_mode enabled you need to destroy also the session cookie also consider the warnings in the docu php.net/manual/en/function.session-destroy.php – Michael Apr 17, 2022 at 13:11 if ($session && !isset($_SESSION['login_time'])) { if ($session == 1) { $_SESSION['login_time']=time(); echo "Login :".$_SESSION['login_time']; echo "<br>"; $_SESSION['idle_time']=$_SESSION['login_time']+20; echo "Session Idle :".$_SESSION['idle_time']; echo "<br>"; } else{ $_SESSION['login_time']=""; } else { if (time()>$_SESSION['idle_time']){ echo "Session Idle :".$_SESSION['idle_time']; echo "<br>"; echo "Current :".time(); echo "<br>"; echo "Session Time Out"; session_destroy(); session_unset(); } else { echo "Logged In<br>";

    You can straight use a DB to do it as an alternative. I use a DB function to do it that I call chk_lgn.

    Check login checks to see if they are logged in or not and, in doing so, it sets the date time stamp of the check as last active in the user's db row/column.

    I also do the time check there. This works for me for the moment as I use this function for every page.

    P.S. No one I had seen had suggested a pure DB solution.