Please wait. This can take some minutes ...
Many resources are needed to download a project. Please understand that we have to compensate our server costs. Thank you in advance.
Project price only 1 $
You can buy this project and download/modify it how often you want.
org.wildfly.clustering.web.undertow.session.DistributableSession Maven / Gradle / Ivy
Go to download
This module adapts an implementation of wildfly-clustering-web-spi to the Undertow servlet container.
/*
* Copyright The WildFly Authors
* SPDX-License-Identifier: Apache-2.0
*/
package org.wildfly.clustering.web.undertow.session;
import io.undertow.security.api.AuthenticatedSessionManager.AuthenticatedSession;
import io.undertow.server.HttpServerExchange;
import io.undertow.server.session.SessionConfig;
import io.undertow.server.session.SessionListener.SessionDestroyedReason;
import io.undertow.servlet.handlers.security.CachedAuthenticatedSessionHandler;
import java.time.Duration;
import java.time.Instant;
import java.util.Map;
import java.util.Set;
import java.util.function.Consumer;
import jakarta.servlet.http.HttpServletRequest;
import org.wildfly.clustering.ee.Batch;
import org.wildfly.clustering.ee.BatchContext;
import org.wildfly.clustering.ee.Batcher;
import org.wildfly.clustering.web.session.ImmutableSessionAttributes;
import org.wildfly.clustering.web.session.Session;
import org.wildfly.clustering.web.session.SessionManager;
import org.wildfly.clustering.web.session.oob.OOBSession;
import org.wildfly.clustering.web.undertow.logging.UndertowClusteringLogger;
/**
* Adapts a distributable {@link Session} to an Undertow {@link io.undertow.server.session.Session}.
* @author Paul Ferraro
*/
public class DistributableSession implements io.undertow.server.session.Session {
// These mechanisms can auto-reauthenticate and thus use local context (instead of replicating)
private static final Set AUTO_REAUTHENTICATING_MECHANISMS = Set.of(HttpServletRequest.BASIC_AUTH, HttpServletRequest.DIGEST_AUTH, HttpServletRequest.CLIENT_CERT_AUTH);
static final String WEB_SOCKET_CHANNELS_ATTRIBUTE = "io.undertow.websocket.current-connections";
private static final Set LOCAL_CONTEXT_ATTRIBUTES = Set.of(WEB_SOCKET_CHANNELS_ATTRIBUTE);
private final UndertowSessionManager manager;
private final Batch batch;
private final Consumer closeTask;
private final Instant startTime;
private final RecordableSessionManagerStatistics statistics;
private volatile Map.Entry>, SessionConfig> entry;
// The following references are only used to create an OOB session
private volatile String id = null;
private volatile Map localContext = null;
public DistributableSession(UndertowSessionManager manager, Session> session, SessionConfig config, Batch batch, Consumer closeTask, RecordableSessionManagerStatistics statistics) {
this.manager = manager;
this.id = session.getId();
this.entry = Map.entry(session, config);
this.batch = batch;
this.closeTask = closeTask;
this.startTime = session.getMetaData().isNew() ? session.getMetaData().getCreationTime() : Instant.now();
this.statistics = statistics;
}
private Map.Entry>, SessionConfig> getSessionEntry() {
Map.Entry>, SessionConfig> entry = this.entry;
// If entry is null, we are outside of the context of a request
if (entry == null) {
// Only allow a single thread to lazily create OOB session
synchronized (this) {
if (this.entry == null) {
// N.B. If entry is null, id and localContext will already have been set
this.entry = Map.entry(new OOBSession<>(this.manager.getSessionManager(), this.id, this.localContext), new SimpleSessionConfig(this.id));
}
entry = this.entry;
}
}
return entry;
}
@Override
public io.undertow.server.session.SessionManager getSessionManager() {
return this.manager;
}
@Override
public void requestDone(HttpServerExchange exchange) {
Session> requestSession = this.getSessionEntry().getKey();
Batcher batcher = this.manager.getSessionManager().getBatcher();
try (BatchContext context = batcher.resumeBatch(this.batch)) {
// If batch was discarded, close it
if (this.batch.getState() == Batch.State.DISCARDED) {
this.batch.close();
}
// If batch is closed, close valid session in a new batch
try (Batch batch = (this.batch.getState() == Batch.State.CLOSED) && requestSession.isValid() ? batcher.createBatch() : this.batch) {
// Ensure session is closed, even if invalid
try (Session> session = requestSession) {
if (session.isValid()) {
// According to §7.6 of the servlet specification:
// The session is considered to be accessed when a request that is part of the session is first handled by the servlet container.
session.getMetaData().setLastAccess(this.startTime, Instant.now());
}
}
}
} catch (Throwable e) {
// Don't propagate exceptions at the stage, since response was already committed
UndertowClusteringLogger.ROOT_LOGGER.warn(e.getLocalizedMessage(), e);
} finally {
// Dereference the distributed session, but retain reference to session identifier and local context
// If session is accessed after this method, getSessionEntry() will lazily create an OOB session
this.id = requestSession.getId();
this.localContext = requestSession.getLocalContext();
this.entry = null;
this.closeTask.accept(exchange);
}
}
@Override
public String getId() {
Session> session = this.getSessionEntry().getKey();
return session.getId();
}
@Override
public long getCreationTime() {
Session> session = this.getSessionEntry().getKey();
try (BatchContext context = this.resumeBatch()) {
return session.getMetaData().getCreationTime().toEpochMilli();
} catch (IllegalStateException e) {
this.closeIfInvalid(null, session);
throw e;
}
}
@Override
public long getLastAccessedTime() {
Session> session = this.getSessionEntry().getKey();
try (BatchContext context = this.resumeBatch()) {
return session.getMetaData().getLastAccessStartTime().toEpochMilli();
} catch (IllegalStateException e) {
this.closeIfInvalid(null, session);
throw e;
}
}
@Override
public int getMaxInactiveInterval() {
Session> session = this.getSessionEntry().getKey();
try (BatchContext context = this.resumeBatch()) {
return (int) session.getMetaData().getTimeout().getSeconds();
} catch (IllegalStateException e) {
this.closeIfInvalid(null, session);
throw e;
}
}
@Override
public void setMaxInactiveInterval(int interval) {
Session> session = this.getSessionEntry().getKey();
try (BatchContext context = this.resumeBatch()) {
session.getMetaData().setTimeout(Duration.ofSeconds(interval));
} catch (IllegalStateException e) {
this.closeIfInvalid(null, session);
throw e;
}
}
@Override
public Set getAttributeNames() {
Session> session = this.getSessionEntry().getKey();
try (BatchContext context = this.resumeBatch()) {
return session.getAttributes().getAttributeNames();
} catch (IllegalStateException e) {
this.closeIfInvalid(null, session);
throw e;
}
}
@Override
public Object getAttribute(String name) {
Session> session = this.getSessionEntry().getKey();
try (BatchContext context = this.resumeBatch()) {
if (CachedAuthenticatedSessionHandler.ATTRIBUTE_NAME.equals(name)) {
AuthenticatedSession auth = (AuthenticatedSession) session.getAttributes().getAttribute(name);
return (auth != null) ? auth : session.getLocalContext().get(name);
}
if (LOCAL_CONTEXT_ATTRIBUTES.contains(name)) {
return session.getLocalContext().get(name);
}
return session.getAttributes().getAttribute(name);
} catch (IllegalStateException e) {
this.closeIfInvalid(null, session);
throw e;
}
}
@Override
public Object setAttribute(String name, Object value) {
if (value == null) {
return this.removeAttribute(name);
}
Session> session = this.getSessionEntry().getKey();
try (BatchContext context = this.resumeBatch()) {
if (CachedAuthenticatedSessionHandler.ATTRIBUTE_NAME.equals(name)) {
AuthenticatedSession auth = (AuthenticatedSession) value;
return AUTO_REAUTHENTICATING_MECHANISMS.contains(auth.getMechanism()) ? session.getLocalContext().put(name, auth) : session.getAttributes().setAttribute(name, auth);
}
if (LOCAL_CONTEXT_ATTRIBUTES.contains(name)) {
return session.getLocalContext().put(name, value);
}
Object old = session.getAttributes().setAttribute(name, value);
if (old == null) {
this.manager.getSessionListeners().attributeAdded(this, name, value);
} else if (old != value) {
this.manager.getSessionListeners().attributeUpdated(this, name, value, old);
}
return old;
} catch (IllegalStateException e) {
this.closeIfInvalid(null, session);
throw e;
}
}
@Override
public Object removeAttribute(String name) {
Session> session = this.getSessionEntry().getKey();
try (BatchContext context = this.resumeBatch()) {
if (CachedAuthenticatedSessionHandler.ATTRIBUTE_NAME.equals(name)) {
AuthenticatedSession auth = (AuthenticatedSession) session.getAttributes().removeAttribute(name);
return (auth != null) ? auth : session.getLocalContext().remove(name);
}
if (LOCAL_CONTEXT_ATTRIBUTES.contains(name)) {
return session.getLocalContext().remove(name);
}
Object old = session.getAttributes().removeAttribute(name);
if (old != null) {
this.manager.getSessionListeners().attributeRemoved(this, name, old);
}
return old;
} catch (IllegalStateException e) {
this.closeIfInvalid(null, session);
throw e;
}
}
@Override
public void invalidate(HttpServerExchange exchange) {
Map.Entry>, SessionConfig> entry = this.getSessionEntry();
Session> session = entry.getKey();
if (session.isValid()) {
// Invoke listeners outside of the context of the batch associated with this session
// Trigger attribute listeners
this.manager.getSessionListeners().sessionDestroyed(this, exchange, SessionDestroyedReason.INVALIDATED);
ImmutableSessionAttributes attributes = session.getAttributes();
for (String name : attributes.getAttributeNames()) {
Object value = attributes.getAttribute(name);
this.manager.getSessionListeners().attributeRemoved(this, name, value);
}
if (this.statistics != null) {
this.statistics.getInactiveSessionRecorder().record(session.getMetaData());
}
}
try (BatchContext context = this.resumeBatch()) {
session.invalidate();
if (exchange != null) {
String id = session.getId();
entry.getValue().clearSession(exchange, id);
}
// An OOB session has no batch
if (this.batch != null) {
this.batch.close();
}
} catch (IllegalStateException e) {
// If Session.invalidate() fails due to concurrent invalidation, close this session
if (!session.isValid()) {
session.close();
}
throw e;
} finally {
this.closeTask.accept(exchange);
}
}
@Override
public String changeSessionId(HttpServerExchange exchange, SessionConfig config) {
Session> oldSession = this.getSessionEntry().getKey();
SessionManager, Batch> manager = this.manager.getSessionManager();
String id = manager.getIdentifierFactory().get();
try (BatchContext context = this.resumeBatch()) {
Session> newSession = manager.createSession(id);
try {
for (String name: oldSession.getAttributes().getAttributeNames()) {
newSession.getAttributes().setAttribute(name, oldSession.getAttributes().getAttribute(name));
}
newSession.getMetaData().setTimeout(oldSession.getMetaData().getTimeout());
newSession.getMetaData().setLastAccess(oldSession.getMetaData().getLastAccessStartTime(), oldSession.getMetaData().getLastAccessEndTime());
newSession.getLocalContext().putAll(oldSession.getLocalContext());
oldSession.invalidate();
config.setSessionId(exchange, id);
this.entry = Map.entry(newSession, config);
} catch (IllegalStateException e) {
this.closeIfInvalid(exchange, oldSession);
newSession.invalidate();
throw e;
}
}
if (!oldSession.isValid()) {
// Invoke listeners outside of the context of the batch associated with this session
this.manager.getSessionListeners().sessionIdChanged(this, oldSession.getId());
}
return id;
}
private BatchContext resumeBatch() {
Batch batch = (this.batch != null) && (this.batch.getState() != Batch.State.CLOSED) ? this.batch : null;
return this.manager.getSessionManager().getBatcher().resumeBatch(batch);
}
private void closeIfInvalid(HttpServerExchange exchange, Session> session) {
if (!session.isValid()) {
// If session was invalidated by a concurrent request, Undertow will not trigger Session.requestDone(...), so we need to close the session here
try {
session.close();
} finally {
// Ensure close task is run
this.closeTask.accept(exchange);
}
}
}
}