SEC-1338: Applied submitted patch, making use of java.util.concurrent classes in place of traditional synchronization.
This commit is contained in:
parent
af56f4844d
commit
ca44ebd3cc
|
@ -15,21 +15,16 @@
|
||||||
|
|
||||||
package org.springframework.security.core.session;
|
package org.springframework.security.core.session;
|
||||||
|
|
||||||
import java.util.ArrayList;
|
|
||||||
import java.util.Arrays;
|
|
||||||
import java.util.Collections;
|
|
||||||
import java.util.Date;
|
|
||||||
import java.util.HashMap;
|
|
||||||
import java.util.HashSet;
|
|
||||||
import java.util.List;
|
|
||||||
import java.util.Map;
|
|
||||||
import java.util.Set;
|
|
||||||
|
|
||||||
import org.apache.commons.logging.Log;
|
import org.apache.commons.logging.Log;
|
||||||
import org.apache.commons.logging.LogFactory;
|
import org.apache.commons.logging.LogFactory;
|
||||||
import org.springframework.context.ApplicationListener;
|
import org.springframework.context.ApplicationListener;
|
||||||
import org.springframework.util.Assert;
|
import org.springframework.util.Assert;
|
||||||
|
|
||||||
|
import java.util.*;
|
||||||
|
import java.util.concurrent.ConcurrentHashMap;
|
||||||
|
import java.util.concurrent.ConcurrentMap;
|
||||||
|
import java.util.concurrent.CopyOnWriteArraySet;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Default implementation of {@link org.springframework.security.core.session.SessionRegistry SessionRegistry}
|
* Default implementation of {@link org.springframework.security.core.session.SessionRegistry SessionRegistry}
|
||||||
* which listens for {@link org.springframework.security.core.session.SessionDestroyedEvent SessionDestroyedEvent}s
|
* which listens for {@link org.springframework.security.core.session.SessionDestroyedEvent SessionDestroyedEvent}s
|
||||||
|
@ -49,14 +44,14 @@ public class SessionRegistryImpl implements SessionRegistry, ApplicationListener
|
||||||
protected final Log logger = LogFactory.getLog(SessionRegistryImpl.class);
|
protected final Log logger = LogFactory.getLog(SessionRegistryImpl.class);
|
||||||
|
|
||||||
/** <principal:Object,SessionIdSet> */
|
/** <principal:Object,SessionIdSet> */
|
||||||
private final Map<Object,Set<String>> principals = Collections.synchronizedMap(new HashMap<Object,Set<String>>());
|
private final ConcurrentMap<Object,Set<String>> principals = new ConcurrentHashMap<Object,Set<String>>();
|
||||||
/** <sessionId:Object,SessionInformation> */
|
/** <sessionId:Object,SessionInformation> */
|
||||||
private final Map<String, SessionInformation> sessionIds = Collections.synchronizedMap(new HashMap<String, SessionInformation>());
|
private final Map<String, SessionInformation> sessionIds = new ConcurrentHashMap<String, SessionInformation>();
|
||||||
|
|
||||||
//~ Methods ========================================================================================================
|
//~ Methods ========================================================================================================
|
||||||
|
|
||||||
public List<Object> getAllPrincipals() {
|
public List<Object> getAllPrincipals() {
|
||||||
return Arrays.asList(principals.keySet().toArray());
|
return new ArrayList<Object>(principals.keySet());
|
||||||
}
|
}
|
||||||
|
|
||||||
public List<SessionInformation> getAllSessions(Object principal, boolean includeExpiredSessions) {
|
public List<SessionInformation> getAllSessions(Object principal, boolean includeExpiredSessions) {
|
||||||
|
@ -68,7 +63,6 @@ public class SessionRegistryImpl implements SessionRegistry, ApplicationListener
|
||||||
|
|
||||||
List<SessionInformation> list = new ArrayList<SessionInformation>(sessionsUsedByPrincipal.size());
|
List<SessionInformation> list = new ArrayList<SessionInformation>(sessionsUsedByPrincipal.size());
|
||||||
|
|
||||||
synchronized (sessionsUsedByPrincipal) {
|
|
||||||
for (String sessionId : sessionsUsedByPrincipal) {
|
for (String sessionId : sessionsUsedByPrincipal) {
|
||||||
SessionInformation sessionInformation = getSessionInformation(sessionId);
|
SessionInformation sessionInformation = getSessionInformation(sessionId);
|
||||||
|
|
||||||
|
@ -80,7 +74,6 @@ public class SessionRegistryImpl implements SessionRegistry, ApplicationListener
|
||||||
list.add(sessionInformation);
|
list.add(sessionInformation);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
return list;
|
return list;
|
||||||
}
|
}
|
||||||
|
@ -88,7 +81,7 @@ public class SessionRegistryImpl implements SessionRegistry, ApplicationListener
|
||||||
public SessionInformation getSessionInformation(String sessionId) {
|
public SessionInformation getSessionInformation(String sessionId) {
|
||||||
Assert.hasText(sessionId, "SessionId required as per interface contract");
|
Assert.hasText(sessionId, "SessionId required as per interface contract");
|
||||||
|
|
||||||
return (SessionInformation) sessionIds.get(sessionId);
|
return sessionIds.get(sessionId);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void onApplicationEvent(SessionDestroyedEvent event) {
|
public void onApplicationEvent(SessionDestroyedEvent event) {
|
||||||
|
@ -106,7 +99,7 @@ public class SessionRegistryImpl implements SessionRegistry, ApplicationListener
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public synchronized void registerNewSession(String sessionId, Object principal) {
|
public void registerNewSession(String sessionId, Object principal) {
|
||||||
Assert.hasText(sessionId, "SessionId required as per interface contract");
|
Assert.hasText(sessionId, "SessionId required as per interface contract");
|
||||||
Assert.notNull(principal, "Principal required as per interface contract");
|
Assert.notNull(principal, "Principal required as per interface contract");
|
||||||
|
|
||||||
|
@ -123,8 +116,12 @@ public class SessionRegistryImpl implements SessionRegistry, ApplicationListener
|
||||||
Set<String> sessionsUsedByPrincipal = principals.get(principal);
|
Set<String> sessionsUsedByPrincipal = principals.get(principal);
|
||||||
|
|
||||||
if (sessionsUsedByPrincipal == null) {
|
if (sessionsUsedByPrincipal == null) {
|
||||||
sessionsUsedByPrincipal = Collections.synchronizedSet(new HashSet<String>(4));
|
sessionsUsedByPrincipal = new CopyOnWriteArraySet<String>();
|
||||||
principals.put(principal, sessionsUsedByPrincipal);
|
Set<String> prevSessionsUsedByPrincipal = principals.putIfAbsent(principal,
|
||||||
|
sessionsUsedByPrincipal);
|
||||||
|
if (prevSessionsUsedByPrincipal != null) {
|
||||||
|
sessionsUsedByPrincipal = prevSessionsUsedByPrincipal;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
sessionsUsedByPrincipal.add(sessionId);
|
sessionsUsedByPrincipal.add(sessionId);
|
||||||
|
@ -159,20 +156,19 @@ public class SessionRegistryImpl implements SessionRegistry, ApplicationListener
|
||||||
logger.debug("Removing session " + sessionId + " from principal's set of registered sessions");
|
logger.debug("Removing session " + sessionId + " from principal's set of registered sessions");
|
||||||
}
|
}
|
||||||
|
|
||||||
synchronized (sessionsUsedByPrincipal) {
|
|
||||||
sessionsUsedByPrincipal.remove(sessionId);
|
sessionsUsedByPrincipal.remove(sessionId);
|
||||||
|
|
||||||
if (sessionsUsedByPrincipal.size() == 0) {
|
if (sessionsUsedByPrincipal.isEmpty()) {
|
||||||
// No need to keep object in principals Map anymore
|
// No need to keep object in principals Map anymore
|
||||||
if (logger.isDebugEnabled()) {
|
if (logger.isDebugEnabled()) {
|
||||||
logger.debug("Removing principal " + info.getPrincipal() + " from registry");
|
logger.debug("Removing principal " + info.getPrincipal() + " from registry");
|
||||||
}
|
}
|
||||||
principals.remove(info.getPrincipal());
|
principals.remove(info.getPrincipal());
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
if (logger.isTraceEnabled()) {
|
if (logger.isTraceEnabled()) {
|
||||||
logger.trace("Sessions used by '" + info.getPrincipal() + "' : " + sessionsUsedByPrincipal);
|
logger.trace("Sessions used by '" + info.getPrincipal() + "' : " + sessionsUsedByPrincipal);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue