All Downloads are FREE. Search and download functionalities are using the official Maven repository.

io.higgs.http.server.auth.HiggsSession Maven / Gradle / Ivy

package io.higgs.http.server.auth;

import org.apache.shiro.session.ExpiredSessionException;
import org.apache.shiro.session.InvalidSessionException;
import org.apache.shiro.session.StoppedSessionException;
import org.apache.shiro.session.mgt.DefaultSessionManager;
import org.apache.shiro.session.mgt.ValidatingSession;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.io.Serializable;
import java.text.DateFormat;
import java.util.Collection;
import java.util.Date;
import java.util.HashMap;
import java.util.Map;

/**
 * @author Courtney Robinson 
 */
public class HiggsSession implements ValidatingSession, Serializable {
    protected static final transient Logger log = LoggerFactory.getLogger(HiggsSession.class);

    protected static final long MILLIS_PER_SECOND = 1000;
    protected static final long MILLIS_PER_MINUTE = 60 * MILLIS_PER_SECOND;
    protected final Date startTimestamp;
    protected Serializable id;
    protected Date stopTimestamp;
    protected Date lastAccessTime;
    protected long timeout;
    protected boolean expired;
    protected String host;
    protected Map attributes = new HashMap<>();

    public HiggsSession(Serializable id) {
        this();
        setId(id);
    }

    public HiggsSession() {
        //TODO - remove concrete reference to DefaultSessionManager
        this.timeout = DefaultSessionManager.DEFAULT_GLOBAL_SESSION_TIMEOUT;
        this.startTimestamp = new Date();
        this.lastAccessTime = this.startTimestamp;
    }

    /**
     * Adds a key value pair to the session which is good for one use.
     * Once the object is retrieved it is automatically removed
     */
    public void flash(Object key, Object value) {
        setAttribute(key, value instanceof FlashValue ? value : new FlashValue(value));
    }

    public void setAttributes(Map attributes) {
        this.attributes = attributes;
    }

    //============================================ Almost straight copy from SimpleSession =============================

    /**
     * @since 0.9
     */
    public boolean isValid() {
        return !isStopped() && !isExpired();
    }

    protected boolean isStopped() {
        return getStopTimestamp() != null;
    }

    /**
     * Returns true if this session has expired, false otherwise.  If the session has
     * expired, no further user interaction with the system may be done under this session.
     *
     * @return true if this session has expired, false otherwise.
     */
    public boolean isExpired() {
        return expired;
    }

    /**
     * Returns the time the session was stopped, or null if the session is still active.
     * 

* A session may become stopped under a number of conditions: *

    *
  • If the user logs out of the system, their current session is terminated (released).
  • *
  • If the session expires
  • *
  • The application explicitly calls {@link #stop()}
  • *
  • If there is an internal system error and the session state can no longer accurately * reflect the user's behavior, such in the case of a system crash
  • *
*

* Once stopped, a session may no longer be used. It is locked from all further activity. * * @return The time the session was stopped, or null if the session is still * active. */ public Date getStopTimestamp() { return stopTimestamp; } public void setStopTimestamp(Date stopTimestamp) { this.stopTimestamp = stopTimestamp; } public void setExpired(boolean expired) { this.expired = expired; } public void validate() throws InvalidSessionException { //check for stopped: if (isStopped()) { //timestamp is set, so the session is considered stopped: String msg = "Session with id [" + getId() + "] has been " + "explicitly stopped. No further interaction under this session is " + "allowed."; throw new StoppedSessionException(msg); } //check for expiration if (isTimedOut()) { expire(); //throw an exception explaining details of why it expired: Date lastAccessTime = getLastAccessTime(); long timeout = getTimeout(); Serializable sessionId = getId(); DateFormat df = DateFormat.getInstance(); String msg = "Session with id [" + sessionId + "] has expired. " + "Last access time: " + df.format(lastAccessTime) + ". Current time: " + df.format(new Date()) + ". Session timeout is set to " + timeout / MILLIS_PER_SECOND + " seconds (" + timeout / MILLIS_PER_MINUTE + " minutes)"; if (log.isTraceEnabled()) { log.trace(msg); } throw new ExpiredSessionException(msg); } } public Serializable getId() { return this.id; } public void setId(Serializable id) { this.id = id; } public Date getStartTimestamp() { return startTimestamp; } public Date getLastAccessTime() { return lastAccessTime; } public void setLastAccessTime(Date lastAccessTime) { this.lastAccessTime = lastAccessTime; } public long getTimeout() { return timeout; } public void setTimeout(long timeout) { this.timeout = timeout; } public String getHost() { return host; } public void setHost(String host) { this.host = host; } public void touch() { this.lastAccessTime = new Date(); } public void stop() { if (this.stopTimestamp == null) { this.stopTimestamp = new Date(); } } public Collection getAttributeKeys() throws InvalidSessionException { return attributes.keySet(); } @Override public Object getAttribute(Object key) { Object value = attributes.get(key); if (value instanceof FlashValue) { removeAttribute(key); return ((FlashValue) value).getValue(); } return value; } public void setAttribute(Object key, Object value) { if (value == null) { removeAttribute(key); } else { attributes.put(key, value); } } public Object removeAttribute(Object key) { return attributes.remove(key); } /** * Determines if this session is expired. * * @return true if the specified session has expired, false otherwise. */ protected boolean isTimedOut() { if (isExpired()) { return true; } long timeout = getTimeout(); if (timeout >= 0L) { Date lastAccessTime = getLastAccessTime(); if (lastAccessTime == null) { String msg = "session.lastAccessTime for session with id [" + getId() + "] is null. This value must be set at " + "least once, preferably at least upon instantiation. Please check the " + getClass().getName() + " implementation and ensure " + "this value will be set (perhaps in the constructor?)"; throw new IllegalStateException(msg); } // Calculate at what time a session would have been last accessed // for it to be expired at this point. In other words, subtract // from the current time the amount of time that a session can // be inactive before expiring. If the session was last accessed // before this time, it is expired. long expireTimeMillis = System.currentTimeMillis() - timeout; Date expireTime = new Date(expireTimeMillis); return lastAccessTime.before(expireTime); } else { if (log.isTraceEnabled()) { log.trace("No timeout for session with id [" + getId() + "]. Session is not considered expired."); } } return false; } protected void expire() { stop(); this.expired = true; } /** * Returns the hashCode. If the {@link #getId() id} is not {@code null}, its hashcode is returned immediately. * If it is {@code null}, an attributes-based hashCode will be calculated and returned. *

* Do your best to ensure {@code HiggsSession} instances receive an ID very early in their lifecycle to * avoid the more expensive attributes-based calculation. * * @return this object's hashCode * @since 1.0 */ @Override public int hashCode() { Serializable id = getId(); if (id != null) { return id.hashCode(); } int hashCode = getStartTimestamp() != null ? getStartTimestamp().hashCode() : 0; hashCode = 31 * hashCode + (getStopTimestamp() != null ? getStopTimestamp().hashCode() : 0); hashCode = 31 * hashCode + (getLastAccessTime() != null ? getLastAccessTime().hashCode() : 0); hashCode = 31 * hashCode + Long.valueOf(Math.max(getTimeout(), 0)).hashCode(); hashCode = 31 * hashCode + Boolean.valueOf(isExpired()).hashCode(); hashCode = 31 * hashCode + (getHost() != null ? getHost().hashCode() : 0); hashCode = 31 * hashCode + (attributes != null ? attributes.hashCode() : 0); return hashCode; } /** * Returns {@code true} if the specified argument is an {@code instanceof} {@code HiggsSession} and both * {@link #getId() id}s are equal. If the argument is a {@code HiggsSession} and either 'this' or the argument * does not yet have an ID assigned, the value of {@link #onEquals(HiggsSession) onEquals} is returned, which * does a necessary attribute-based comparison when IDs are not available. *

* Do your best to ensure {@code HiggsSession} instances receive an ID very early in their lifecycle to * avoid the more expensive attributes-based comparison. * * @param obj the object to compare with this one for equality. * @return {@code true} if this object is equivalent to the specified argument, {@code false} otherwise. */ @Override public boolean equals(Object obj) { if (this == obj) { return true; } if (obj instanceof HiggsSession) { HiggsSession other = (HiggsSession) obj; Serializable thisId = getId(); Serializable otherId = other.getId(); if (thisId != null && otherId != null) { return thisId.equals(otherId); } else { //fall back to an attribute based comparison: return onEquals(other); } } return false; } /** * Provides an attribute-based comparison (no ID comparison) - incurred only when 'this' or the * session object being compared for equality do not have a session id. * * @param ss the HiggsSession instance to compare for equality. * @return true if all the attributes, except the id, are equal to this object's attributes. * @since 1.0 */ protected boolean onEquals(HiggsSession ss) { return (getStartTimestamp() != null ? getStartTimestamp().equals(ss.getStartTimestamp()) : ss.getStartTimestamp() == null) && (getStopTimestamp() != null ? getStopTimestamp().equals(ss.getStopTimestamp()) : ss.getStopTimestamp() == null) && (getLastAccessTime() != null ? getLastAccessTime().equals(ss.getLastAccessTime()) : ss.getLastAccessTime() == null) && (getTimeout() == ss.getTimeout()) && (isExpired() == ss.isExpired()) && (getHost() != null ? getHost().equals(ss.getHost()) : ss.getHost() == null) && (attributes != null ? attributes.equals(ss.attributes) : ss.attributes == null); } /** * Returns the string representation of this HiggsSession, equal to * getClass().getName() + ",id=" + getId(). * * @return the string representation of this HiggsSession, equal to * getClass().getName() + ",id=" + getId(). * @since 1.0 */ @Override public String toString() { StringBuilder sb = new StringBuilder(); sb.append(getClass().getName()).append(",id=").append(getId()); return sb.toString(); } }