/*
* Copyright (c) JForum Team
* All rights reserved.
*
* Redistribution and use in source and binary forms,
* with or without modification, are permitted provided
* that the following conditions are met:
*
* 1) Redistributions of source code must retain the above
* copyright notice, this list of conditions and the
* following disclaimer.
* 2) Redistributions in binary form must reproduce the
* above copyright notice, this list of conditions and
* the following disclaimer in the documentation and/or
* other materials provided with the distribution.
* 3) Neither the name of "Rafael Steil" nor
* the names of its contributors may be used to endorse
* or promote products derived from this software without
* specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT
* HOLDERS AND CONTRIBUTORS "AS IS" AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING,
* BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
* MERCHANTABILITY AND FITNESS FOR A PARTICULAR
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL
* THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
* EXEMPLARY, OR CONSEQUENTIAL DAMAGES
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
* OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
* CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER
* IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
* ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
* ADVISED OF THE POSSIBILITY OF SUCH DAMAGE
*
* This file creation date: 12/03/2004 - 18:47:26
* The JForum Project
* http://www.jforum.net
*/
package net.jforum;
import java.sql.Connection;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import net.jforum.cache.CacheEngine;
import net.jforum.cache.Cacheable;
import net.jforum.dao.DataAccessDriver;
import net.jforum.entities.UserSession;
import net.jforum.repository.SecurityRepository;
import net.jforum.util.preferences.ConfigKeys;
import net.jforum.util.preferences.SystemGlobals;
import org.apache.log4j.Logger;
/**
* @author Rafael Steil
* @version $Id: SessionFacade.java,v 1.40 2007/09/20 16:07:10 rafaelsteil Exp $
*/
public class SessionFacade implements Cacheable
{
private static final Logger logger = Logger.getLogger(SessionFacade.class);
private static final String FQN = "sessions";
private static final String FQN_LOGGED = FQN + "/logged";
private static final String FQN_COUNT = FQN + "/count";
private static final String FQN_USER_ID = FQN + "/userId";
private static final String ANONYMOUS_COUNT = "anonymousCount";
private static final String LOGGED_COUNT = "loggedCount";
private static CacheEngine cache;
/**
* @see net.jforum.cache.Cacheable#setCacheEngine(net.jforum.cache.CacheEngine)
*/
public void setCacheEngine(CacheEngine engine)
{
cache = engine;
}
/**
* Add a new <code>UserSession</code> entry to the session.
* This method will make a call to <code>JForum.getRequest.getSession().getId()</code>
* to retrieve the session's id
*
* @param us The user session objetc to add
* @see #add(UserSession, String)
*/
public static void add(UserSession us)
{
add(us, JForumExecutionContext.getRequest().getSessionContext().getId());
}
/**
* Registers a new {@link UserSession}.
* <p>
* If a call to {@link UserSession#getUserId()} return a value different
* of <code>SystemGlobals.getIntValue(ConfigKeys.ANONYMOUS_USER_ID)</code>, then
* the user will be registered as "logged". Otherwise it will enter as anonymous.
* </p>
*
* <p>
* Please note that, in order to keep the number of guest and logged users correct,
* it's caller's responsability to {@link #remove(String)} the record before adding it
* again if the current session is currently represented as "guest".
* </p>
*
* @param us the UserSession to add
* @param sessionId the user's session id
*/
public static void add(UserSession us, String sessionId)
{
if (us.getSessionId() == null || us.getSessionId().equals("")) {
us.setSessionId(sessionId);
}
synchronized (FQN) {
cache.add(FQN, us.getSessionId(), us);
if (!JForumExecutionContext.getForumContext().isBot()) {
if (us.getUserId() != SystemGlobals.getIntValue(ConfigKeys.ANONYMOUS_USER_ID)) {
changeUserCount(LOGGED_COUNT, true);
cache.add(FQN_LOGGED, us.getSessionId(), us);
cache.add(FQN_USER_ID, Integer.toString(us.getUserId()), us.getSessionId());
}
else {
// TODO: check the anonymous IP constraint
changeUserCount(ANONYMOUS_COUNT, true);
}
}
}
}
private static void changeUserCount(String cacheEntryName, boolean increment)
{
Integer count = (Integer)cache.get(FQN_COUNT, cacheEntryName);
if (count == null) {
count = new Integer(0);
}
if (increment) {
count = new Integer(count.intValue() + 1);
}
else if (count.intValue() > 0) {
count = new Integer(count.intValue() - 1);
}
cache.add(FQN_COUNT, cacheEntryName, count);
}
/**
* Add a new entry to the user's session
*
* @param name The attribute name
* @param value The attribute value
*/
public static void setAttribute(String name, Object value)
{
JForumExecutionContext.getRequest().getSessionContext().setAttribute(name, value);
}
/**
* Removes an attribute from the session
*
* @param name The key associated to the the attribute to remove
*/
public static void removeAttribute(String name)
{
JForumExecutionContext.getRequest().getSessionContext().removeAttribute(name);
}
/**
* Gets an attribute value given its name
*
* @param name The attribute name to retrieve the value
* @return The value as an Object, or null if no entry was found
*/
public static Object getAttribute(String name)
{
return JForumExecutionContext.getRequest().getSessionContext().getAttribute(name);
}
/**
* Remove an entry fro the session map
*
* @param sessionId The session id to remove
*/
public static void remove(String sessionId)
{
if (cache == null) {
logger.warn("Got a null cache instance. #" + sessionId);
return;
}
logger.debug("Removing session " + sessionId);
synchronized (FQN) {
UserSession us = getUserSession(sessionId);
if (us != null) {
cache.remove(FQN_LOGGED, sessionId);
cache.remove(FQN_USER_ID, Integer.toString(us.getUserId()));
if (us.getUserId() != SystemGlobals.getIntValue(ConfigKeys.ANONYMOUS_USER_ID)) {
changeUserCount(LOGGED_COUNT, false);
}
else {
changeUserCount(ANONYMOUS_COUNT, false);
}
}
cache.remove(FQN, sessionId);
}
}
/**
* Get all registered sessions
*
* @return <code>ArrayList</code> with the sessions. Each entry
* is an <code>UserSession</code> object.
*/
public static List getAllSessions()
{
synchronized (FQN) {
return new ArrayList(cache.getValues(FQN));
}
}
/**
* Gets the {@link UserSession} instance of all logged users
* @return A list with the user sessions
*/
public static List getLoggedSessions()
{
synchronized (FQN) {
return new ArrayList(cache.getValues(FQN_LOGGED));
}
}
/**
* Get the number of logged users
* @return the number of logged users
*/
public static int registeredSize()
{
Integer count = (Integer)cache.get(FQN_COUNT, LOGGED_COUNT);
return (count == null ? 0 : count.intValue());
}
/**
* Get the number of anonymous users
* @return the nuber of anonymous users
*/
public static int anonymousSize()
{
Integer count = (Integer)cache.get(FQN_COUNT, ANONYMOUS_COUNT);
return (count == null ? 0 : count.intValue());
}
public static void clear()
{
synchronized (FQN) {
cache.add(FQN, new HashMap());
cache.add(FQN_COUNT, LOGGED_COUNT, new Integer(0));
cache.add(FQN_COUNT, ANONYMOUS_COUNT, new Integer(0));
cache.remove(FQN_LOGGED);
cache.remove(FQN_USER_ID);
}
}
/**
* Gets the user's <code>UserSession</code> object
*
* @return The <code>UserSession</code> associated to the user's session
*/
public static UserSession getUserSession()
{
return getUserSession(JForumExecutionContext.getRequest().getSessionContext().getId());
}
/**
* Gets an {@link UserSession} by the session id.
*
* @param sessionId the session's id
* @return an <b>immutable</b> UserSession, or <code>null</code> if no entry found
*/
public static UserSession getUserSession(String sessionId)
{
if (cache != null) {
UserSession us = (UserSession)cache.get(FQN, sessionId);
return us;
}
logger.warn("Got a null cache in getUserSession. #" + sessionId);
return null;
}
/**
* Gets the number of session elements.
*
* @return The number of session elements currently online (without bots)
*/
public static int size()
{
return (anonymousSize() + registeredSize());
}
/**
* Verify if the user in already loaded
*
* @param username The username to check
* @return The session id if the user is already registered into the session,
* or <code>null</code> if it is not.
*/
public static String isUserInSession(String username)
{
int aid = SystemGlobals.getIntValue(ConfigKeys.ANONYMOUS_USER_ID);
synchronized (FQN) {
for (Iterator iter = cache.getValues(FQN).iterator(); iter.hasNext(); ) {
UserSession us = (UserSession)iter.next();
String thisUsername = us.getUsername();
if (thisUsername == null) {
continue;
}
if (us.getUserId() != aid && thisUsername.equals(username)) {
return us.getSessionId();
}
}
}
return null;
}
/**
* Verify if there is an user in the session with the
* user id passed as parameter.
*
* @param userId The user id to check for existance in the session
* @return The session id if the user is already registered into the session,
* or <code>null</code> if it is not.
*/
public static String isUserInSession(int userId)
{
return (String)cache.get(FQN_USER_ID, Integer.toString(userId));
}
/**
* Verify is the user is logged in.
*
* @return <code>true</code> if the user is logged, or <code>false</code> if is
* an anonymous user.
*/
public static boolean isLogged()
{
return "1".equals(SessionFacade.getAttribute(ConfigKeys.LOGGED));
}
/**
* Marks the current user session as "logged" in
*/
public static void makeLogged()
{
SessionFacade.setAttribute(ConfigKeys.LOGGED, "1");
}
/**
* Marks the current user session as "logged" out
*
*/
public static void makeUnlogged()
{
SessionFacade.removeAttribute(ConfigKeys.LOGGED);
}
/**
* Returns a map containing information about read time of a set of topics.
* @return a map where the key is the topicId represented as an Integer, and the
* value is a Long representing the read time of such topic.
*/
public static Map getTopicsReadTime()
{
Map tracking = (Map)getAttribute(ConfigKeys.TOPICS_READ_TIME);
if (tracking == null) {
tracking = new HashMap();
setAttribute(ConfigKeys.TOPICS_READ_TIME, tracking);
}
return tracking;
}
/**
* Returns a map with "all topics read" flags for some forum
* @return a map where the key is the forum id represented as an Integer,
* and the value is a Long representing the read time to be used in the verifications.
*/
public static Map getTopicsReadTimeByForum()
{
return (Map)getAttribute(ConfigKeys.TOPICS_READ_TIME_BY_FORUM);
}
/**
* Persists user session information.
* This method will get a <code>Connection</code> making a call to
* <code>DBConnection.getImplementation().getConnection()</code>, and
* then releasing the connection after the method is processed.
*
* @param sessionId The session which we're going to persist information
* @see #storeSessionData(String, Connection)
*/
public static void storeSessionData(String sessionId)
{
Connection conn = null;
try {
conn = DBConnection.getImplementation().getConnection();
SessionFacade.storeSessionData(sessionId, conn);
}
finally {
if (conn != null) {
try {
DBConnection.getImplementation().releaseConnection(conn);
}
catch (Exception e) {
logger.warn("Error while releasing a connection: " + e);
}
}
}
}
/**
* Persists user session information.
*
* @param sessionId The session which we're going to persist
* @param conn A <code>Connection</code> to be used to connect to
* the database.
* @see #storeSessionData(String)
*/
public static void storeSessionData(String sessionId, Connection conn)
{
UserSession us = SessionFacade.getUserSession(sessionId);
if (us != null) {
try {
if (us.getUserId() != SystemGlobals.getIntValue(ConfigKeys.ANONYMOUS_USER_ID)) {
DataAccessDriver.getInstance().newUserSessionDAO().update(us, conn);
}
SecurityRepository.remove(us.getUserId());
}
catch (Exception e) {
logger.warn("Error storing user session data: " + e, e);
}
}
}
}