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

org.springframework.web.server.session.InMemoryWebSessionStore Maven / Gradle / Ivy

There is a newer version: 6.1.6
Show newest version
/*
 * Copyright 2002-2018 the original author or authors.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package org.springframework.web.server.session;

import java.time.Clock;
import java.time.Duration;
import java.time.Instant;
import java.time.ZoneId;
import java.util.Iterator;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentMap;
import java.util.concurrent.atomic.AtomicReference;
import java.util.concurrent.locks.ReentrantLock;

import reactor.core.publisher.Mono;

import org.springframework.util.Assert;
import org.springframework.util.IdGenerator;
import org.springframework.util.JdkIdGenerator;
import org.springframework.web.server.WebSession;

/**
 * Simple Map-based storage for {@link WebSession} instances.
 *
 * @author Rossen Stoyanchev
 * @author Rob Winch
 * @since 5.0
 */
public class InMemoryWebSessionStore implements WebSessionStore {

	/** Minimum period between expiration checks */
	private static final Duration EXPIRATION_CHECK_PERIOD = Duration.ofSeconds(60);

	private static final IdGenerator idGenerator = new JdkIdGenerator();


	private Clock clock = Clock.system(ZoneId.of("GMT"));

	private final ConcurrentMap sessions = new ConcurrentHashMap<>();

	private volatile Instant nextExpirationCheckTime = Instant.now(this.clock).plus(EXPIRATION_CHECK_PERIOD);

	private final ReentrantLock expirationCheckLock = new ReentrantLock();


	/**
	 * Configure the {@link Clock} to use to set lastAccessTime on every created
	 * session and to calculate if it is expired.
	 * 

This may be useful to align to different timezone or to set the clock * back in a test, e.g. {@code Clock.offset(clock, Duration.ofMinutes(-31))} * in order to simulate session expiration. *

By default this is {@code Clock.system(ZoneId.of("GMT"))}. * @param clock the clock to use */ public void setClock(Clock clock) { Assert.notNull(clock, "Clock is required"); this.clock = clock; // Force a check when clock changes.. this.nextExpirationCheckTime = Instant.now(this.clock); } /** * Return the configured clock for session lastAccessTime calculations. */ public Clock getClock() { return this.clock; } @Override public Mono createWebSession() { return Mono.fromSupplier(InMemoryWebSession::new); } @Override public Mono retrieveSession(String id) { Instant currentTime = Instant.now(this.clock); if (!this.sessions.isEmpty() && !currentTime.isBefore(this.nextExpirationCheckTime)) { checkExpiredSessions(currentTime); } InMemoryWebSession session = this.sessions.get(id); if (session == null) { return Mono.empty(); } else if (session.isExpired(currentTime)) { this.sessions.remove(id); return Mono.empty(); } else { session.updateLastAccessTime(currentTime); return Mono.just(session); } } private void checkExpiredSessions(Instant currentTime) { if (this.expirationCheckLock.tryLock()) { try { Iterator iterator = this.sessions.values().iterator(); while (iterator.hasNext()) { InMemoryWebSession session = iterator.next(); if (session.isExpired(currentTime)) { iterator.remove(); session.invalidate(); } } } finally { this.nextExpirationCheckTime = currentTime.plus(EXPIRATION_CHECK_PERIOD); this.expirationCheckLock.unlock(); } } } @Override public Mono removeSession(String id) { this.sessions.remove(id); return Mono.empty(); } public Mono updateLastAccessTime(WebSession webSession) { return Mono.fromSupplier(() -> { Assert.isInstanceOf(InMemoryWebSession.class, webSession); InMemoryWebSession session = (InMemoryWebSession) webSession; session.updateLastAccessTime(Instant.now(getClock())); return session; }); } private class InMemoryWebSession implements WebSession { private final AtomicReference id = new AtomicReference<>(String.valueOf(idGenerator.generateId())); private final Map attributes = new ConcurrentHashMap<>(); private final Instant creationTime; private volatile Instant lastAccessTime; private volatile Duration maxIdleTime = Duration.ofMinutes(30); private final AtomicReference state = new AtomicReference<>(State.NEW); public InMemoryWebSession() { this.creationTime = Instant.now(getClock()); this.lastAccessTime = this.creationTime; } @Override public String getId() { return this.id.get(); } @Override public Map getAttributes() { return this.attributes; } @Override public Instant getCreationTime() { return this.creationTime; } @Override public Instant getLastAccessTime() { return this.lastAccessTime; } @Override public void setMaxIdleTime(Duration maxIdleTime) { this.maxIdleTime = maxIdleTime; } @Override public Duration getMaxIdleTime() { return this.maxIdleTime; } @Override public void start() { this.state.compareAndSet(State.NEW, State.STARTED); } @Override public boolean isStarted() { return this.state.get().equals(State.STARTED) || !getAttributes().isEmpty(); } @Override public Mono changeSessionId() { String currentId = this.id.get(); InMemoryWebSessionStore.this.sessions.remove(currentId); String newId = String.valueOf(idGenerator.generateId()); this.id.set(newId); InMemoryWebSessionStore.this.sessions.put(this.getId(), this); return Mono.empty(); } @Override public Mono invalidate() { this.state.set(State.EXPIRED); getAttributes().clear(); InMemoryWebSessionStore.this.sessions.remove(this.id.get()); return Mono.empty(); } @Override public Mono save() { if (!getAttributes().isEmpty()) { this.state.compareAndSet(State.NEW, State.STARTED); } InMemoryWebSessionStore.this.sessions.put(this.getId(), this); return Mono.empty(); } @Override public boolean isExpired() { return isExpired(Instant.now(getClock())); } private boolean isExpired(Instant currentTime) { if (this.state.get().equals(State.EXPIRED)) { return true; } if (checkExpired(currentTime)) { this.state.set(State.EXPIRED); return true; } return false; } private boolean checkExpired(Instant currentTime) { return isStarted() && !this.maxIdleTime.isNegative() && currentTime.minus(this.maxIdleTime).isAfter(this.lastAccessTime); } private void updateLastAccessTime(Instant currentTime) { this.lastAccessTime = currentTime; } } private enum State { NEW, STARTED, EXPIRED } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy