HttpSessionOAuth2AuthorizationRequestRepository: store one request by default
Add setAllowMultipleAuthorizationRequests allowing applications to revert to the previous functionality should they need to do so. Closes gh-5145 Intentionally regresses gh-5110
This commit is contained in:
parent
362855b8b8
commit
ecb4a5749a
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright 2002-2018 the original author or authors.
|
* Copyright 2002-2021 the original author or authors.
|
||||||
*
|
*
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
* you may not use this file except in compliance with the License.
|
* you may not use this file except in compliance with the License.
|
||||||
|
@ -31,6 +31,7 @@ import java.util.Map;
|
||||||
*
|
*
|
||||||
* @author Joe Grandja
|
* @author Joe Grandja
|
||||||
* @author Rob Winch
|
* @author Rob Winch
|
||||||
|
* @author Craig Andrews
|
||||||
* @since 5.0
|
* @since 5.0
|
||||||
* @see AuthorizationRequestRepository
|
* @see AuthorizationRequestRepository
|
||||||
* @see OAuth2AuthorizationRequest
|
* @see OAuth2AuthorizationRequest
|
||||||
|
@ -41,6 +42,8 @@ public final class HttpSessionOAuth2AuthorizationRequestRepository implements Au
|
||||||
|
|
||||||
private final String sessionAttributeName = DEFAULT_AUTHORIZATION_REQUEST_ATTR_NAME;
|
private final String sessionAttributeName = DEFAULT_AUTHORIZATION_REQUEST_ATTR_NAME;
|
||||||
|
|
||||||
|
private boolean allowMultipleAuthorizationRequests;
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public OAuth2AuthorizationRequest loadAuthorizationRequest(HttpServletRequest request) {
|
public OAuth2AuthorizationRequest loadAuthorizationRequest(HttpServletRequest request) {
|
||||||
Assert.notNull(request, "request cannot be null");
|
Assert.notNull(request, "request cannot be null");
|
||||||
|
@ -63,10 +66,15 @@ public final class HttpSessionOAuth2AuthorizationRequestRepository implements Au
|
||||||
}
|
}
|
||||||
String state = authorizationRequest.getState();
|
String state = authorizationRequest.getState();
|
||||||
Assert.hasText(state, "authorizationRequest.state cannot be empty");
|
Assert.hasText(state, "authorizationRequest.state cannot be empty");
|
||||||
|
if (this.allowMultipleAuthorizationRequests) {
|
||||||
Map<String, OAuth2AuthorizationRequest> authorizationRequests = this.getAuthorizationRequests(request);
|
Map<String, OAuth2AuthorizationRequest> authorizationRequests = this.getAuthorizationRequests(request);
|
||||||
authorizationRequests.put(state, authorizationRequest);
|
authorizationRequests.put(state, authorizationRequest);
|
||||||
request.getSession().setAttribute(this.sessionAttributeName, authorizationRequests);
|
request.getSession().setAttribute(this.sessionAttributeName, authorizationRequests);
|
||||||
}
|
}
|
||||||
|
else {
|
||||||
|
request.getSession().setAttribute(this.sessionAttributeName, authorizationRequest);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public OAuth2AuthorizationRequest removeAuthorizationRequest(HttpServletRequest request) {
|
public OAuth2AuthorizationRequest removeAuthorizationRequest(HttpServletRequest request) {
|
||||||
|
@ -77,11 +85,16 @@ public final class HttpSessionOAuth2AuthorizationRequestRepository implements Au
|
||||||
}
|
}
|
||||||
Map<String, OAuth2AuthorizationRequest> authorizationRequests = this.getAuthorizationRequests(request);
|
Map<String, OAuth2AuthorizationRequest> authorizationRequests = this.getAuthorizationRequests(request);
|
||||||
OAuth2AuthorizationRequest originalRequest = authorizationRequests.remove(stateParameter);
|
OAuth2AuthorizationRequest originalRequest = authorizationRequests.remove(stateParameter);
|
||||||
if (!authorizationRequests.isEmpty()) {
|
if (authorizationRequests.size() == 0) {
|
||||||
request.getSession().setAttribute(this.sessionAttributeName, authorizationRequests);
|
|
||||||
} else {
|
|
||||||
request.getSession().removeAttribute(this.sessionAttributeName);
|
request.getSession().removeAttribute(this.sessionAttributeName);
|
||||||
}
|
}
|
||||||
|
else if (authorizationRequests.size() == 1) {
|
||||||
|
request.getSession().setAttribute(this.sessionAttributeName,
|
||||||
|
authorizationRequests.values().iterator().next());
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
request.getSession().setAttribute(this.sessionAttributeName, authorizationRequests);
|
||||||
|
}
|
||||||
return originalRequest;
|
return originalRequest;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -107,11 +120,38 @@ public final class HttpSessionOAuth2AuthorizationRequestRepository implements Au
|
||||||
*/
|
*/
|
||||||
private Map<String, OAuth2AuthorizationRequest> getAuthorizationRequests(HttpServletRequest request) {
|
private Map<String, OAuth2AuthorizationRequest> getAuthorizationRequests(HttpServletRequest request) {
|
||||||
HttpSession session = request.getSession(false);
|
HttpSession session = request.getSession(false);
|
||||||
Map<String, OAuth2AuthorizationRequest> authorizationRequests = session == null ? null :
|
Object sessionAttributeValue = (session != null) ? session.getAttribute(this.sessionAttributeName) : null;
|
||||||
(Map<String, OAuth2AuthorizationRequest>) session.getAttribute(this.sessionAttributeName);
|
if (sessionAttributeValue == null) {
|
||||||
if (authorizationRequests == null) {
|
|
||||||
return new HashMap<>();
|
return new HashMap<>();
|
||||||
}
|
}
|
||||||
|
else if (sessionAttributeValue instanceof OAuth2AuthorizationRequest) {
|
||||||
|
OAuth2AuthorizationRequest auth2AuthorizationRequest = (OAuth2AuthorizationRequest) sessionAttributeValue;
|
||||||
|
Map<String, OAuth2AuthorizationRequest> authorizationRequests = new HashMap<>(1);
|
||||||
|
authorizationRequests.put(auth2AuthorizationRequest.getState(), auth2AuthorizationRequest);
|
||||||
return authorizationRequests;
|
return authorizationRequests;
|
||||||
}
|
}
|
||||||
|
else if (sessionAttributeValue instanceof Map) {
|
||||||
|
@SuppressWarnings("unchecked")
|
||||||
|
Map<String, OAuth2AuthorizationRequest> authorizationRequests = (Map<String, OAuth2AuthorizationRequest>) sessionAttributeValue;
|
||||||
|
return authorizationRequests;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
throw new IllegalStateException(
|
||||||
|
"authorizationRequests is supposed to be a Map or OAuth2AuthorizationRequest but actually is a "
|
||||||
|
+ sessionAttributeValue.getClass());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Configure if multiple {@link OAuth2AuthorizationRequest}s should be stored per
|
||||||
|
* session. Default is false (not allow multiple {@link OAuth2AuthorizationRequest}
|
||||||
|
* per session).
|
||||||
|
* @param allowMultipleAuthorizationRequests true allows more than one
|
||||||
|
* {@link OAuth2AuthorizationRequest} to be stored per session.
|
||||||
|
* @since 5.5
|
||||||
|
*/
|
||||||
|
@Deprecated
|
||||||
|
public void setAllowMultipleAuthorizationRequests(boolean allowMultipleAuthorizationRequests) {
|
||||||
|
this.allowMultipleAuthorizationRequests = allowMultipleAuthorizationRequests;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,76 @@
|
||||||
|
/*
|
||||||
|
* Copyright 2002-2021 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
|
||||||
|
*
|
||||||
|
* https://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.security.oauth2.client.web;
|
||||||
|
|
||||||
|
import org.junit.Before;
|
||||||
|
import org.junit.Test;
|
||||||
|
|
||||||
|
import org.springframework.mock.web.MockHttpServletRequest;
|
||||||
|
import org.springframework.mock.web.MockHttpServletResponse;
|
||||||
|
import org.springframework.security.oauth2.core.endpoint.OAuth2AuthorizationRequest;
|
||||||
|
import org.springframework.security.oauth2.core.endpoint.OAuth2ParameterNames;
|
||||||
|
|
||||||
|
import static org.assertj.core.api.Assertions.assertThat;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tests for {@link HttpSessionOAuth2AuthorizationRequestRepository} when
|
||||||
|
* {@link HttpSessionOAuth2AuthorizationRequestRepository#setAllowMultipleAuthorizationRequests(boolean)}
|
||||||
|
* is enabled.
|
||||||
|
*
|
||||||
|
* @author Joe Grandja
|
||||||
|
* @author Craig Andrews
|
||||||
|
*/
|
||||||
|
public class HttpSessionOAuth2AuthorizationRequestRepositoryAllowMultipleAuthorizationRequestsTests
|
||||||
|
extends HttpSessionOAuth2AuthorizationRequestRepositoryTests {
|
||||||
|
|
||||||
|
@Before
|
||||||
|
public void setup() {
|
||||||
|
this.authorizationRequestRepository = new HttpSessionOAuth2AuthorizationRequestRepository();
|
||||||
|
this.authorizationRequestRepository.setAllowMultipleAuthorizationRequests(true);
|
||||||
|
}
|
||||||
|
|
||||||
|
// gh-5110
|
||||||
|
@Test
|
||||||
|
public void loadAuthorizationRequestWhenMultipleSavedThenReturnMatchingAuthorizationRequest() {
|
||||||
|
MockHttpServletRequest request = new MockHttpServletRequest();
|
||||||
|
MockHttpServletResponse response = new MockHttpServletResponse();
|
||||||
|
String state1 = "state-1122";
|
||||||
|
OAuth2AuthorizationRequest authorizationRequest1 = createAuthorizationRequest().state(state1).build();
|
||||||
|
this.authorizationRequestRepository.saveAuthorizationRequest(authorizationRequest1, request, response);
|
||||||
|
String state2 = "state-3344";
|
||||||
|
OAuth2AuthorizationRequest authorizationRequest2 = createAuthorizationRequest().state(state2).build();
|
||||||
|
this.authorizationRequestRepository.saveAuthorizationRequest(authorizationRequest2, request, response);
|
||||||
|
String state3 = "state-5566";
|
||||||
|
OAuth2AuthorizationRequest authorizationRequest3 = createAuthorizationRequest().state(state3).build();
|
||||||
|
this.authorizationRequestRepository.saveAuthorizationRequest(authorizationRequest3, request, response);
|
||||||
|
request.addParameter(OAuth2ParameterNames.STATE, state1);
|
||||||
|
OAuth2AuthorizationRequest loadedAuthorizationRequest1 = this.authorizationRequestRepository
|
||||||
|
.loadAuthorizationRequest(request);
|
||||||
|
assertThat(loadedAuthorizationRequest1).isEqualTo(authorizationRequest1);
|
||||||
|
request.removeParameter(OAuth2ParameterNames.STATE);
|
||||||
|
request.addParameter(OAuth2ParameterNames.STATE, state2);
|
||||||
|
OAuth2AuthorizationRequest loadedAuthorizationRequest2 = this.authorizationRequestRepository
|
||||||
|
.loadAuthorizationRequest(request);
|
||||||
|
assertThat(loadedAuthorizationRequest2).isEqualTo(authorizationRequest2);
|
||||||
|
request.removeParameter(OAuth2ParameterNames.STATE);
|
||||||
|
request.addParameter(OAuth2ParameterNames.STATE, state3);
|
||||||
|
OAuth2AuthorizationRequest loadedAuthorizationRequest3 = this.authorizationRequestRepository
|
||||||
|
.loadAuthorizationRequest(request);
|
||||||
|
assertThat(loadedAuthorizationRequest3).isEqualTo(authorizationRequest3);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,76 @@
|
||||||
|
/*
|
||||||
|
* Copyright 2002-2021 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
|
||||||
|
*
|
||||||
|
* https://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.security.oauth2.client.web;
|
||||||
|
|
||||||
|
import org.junit.Before;
|
||||||
|
import org.junit.Test;
|
||||||
|
|
||||||
|
import org.springframework.mock.web.MockHttpServletRequest;
|
||||||
|
import org.springframework.mock.web.MockHttpServletResponse;
|
||||||
|
import org.springframework.security.oauth2.core.endpoint.OAuth2AuthorizationRequest;
|
||||||
|
import org.springframework.security.oauth2.core.endpoint.OAuth2ParameterNames;
|
||||||
|
|
||||||
|
import static org.assertj.core.api.Assertions.assertThat;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tests for {@link HttpSessionOAuth2AuthorizationRequestRepository} when
|
||||||
|
* {@link HttpSessionOAuth2AuthorizationRequestRepository#setAllowMultipleAuthorizationRequests(boolean)}
|
||||||
|
* is disabled.
|
||||||
|
*
|
||||||
|
* @author Joe Grandja
|
||||||
|
* @author Craig Andrews
|
||||||
|
*/
|
||||||
|
public class HttpSessionOAuth2AuthorizationRequestRepositoryDoNotAllowMultipleAuthorizationRequestsTests
|
||||||
|
extends HttpSessionOAuth2AuthorizationRequestRepositoryTests {
|
||||||
|
|
||||||
|
@Before
|
||||||
|
public void setup() {
|
||||||
|
this.authorizationRequestRepository = new HttpSessionOAuth2AuthorizationRequestRepository();
|
||||||
|
this.authorizationRequestRepository.setAllowMultipleAuthorizationRequests(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
// gh-5145
|
||||||
|
@Test
|
||||||
|
public void loadAuthorizationRequestWhenMultipleSavedThenReturnLastAuthorizationRequest() {
|
||||||
|
MockHttpServletRequest request = new MockHttpServletRequest();
|
||||||
|
MockHttpServletResponse response = new MockHttpServletResponse();
|
||||||
|
String state1 = "state-1122";
|
||||||
|
OAuth2AuthorizationRequest authorizationRequest1 = createAuthorizationRequest().state(state1).build();
|
||||||
|
this.authorizationRequestRepository.saveAuthorizationRequest(authorizationRequest1, request, response);
|
||||||
|
String state2 = "state-3344";
|
||||||
|
OAuth2AuthorizationRequest authorizationRequest2 = createAuthorizationRequest().state(state2).build();
|
||||||
|
this.authorizationRequestRepository.saveAuthorizationRequest(authorizationRequest2, request, response);
|
||||||
|
String state3 = "state-5566";
|
||||||
|
OAuth2AuthorizationRequest authorizationRequest3 = createAuthorizationRequest().state(state3).build();
|
||||||
|
this.authorizationRequestRepository.saveAuthorizationRequest(authorizationRequest3, request, response);
|
||||||
|
request.addParameter(OAuth2ParameterNames.STATE, state1);
|
||||||
|
OAuth2AuthorizationRequest loadedAuthorizationRequest1 = this.authorizationRequestRepository
|
||||||
|
.loadAuthorizationRequest(request);
|
||||||
|
assertThat(loadedAuthorizationRequest1).isNull();
|
||||||
|
request.removeParameter(OAuth2ParameterNames.STATE);
|
||||||
|
request.addParameter(OAuth2ParameterNames.STATE, state2);
|
||||||
|
OAuth2AuthorizationRequest loadedAuthorizationRequest2 = this.authorizationRequestRepository
|
||||||
|
.loadAuthorizationRequest(request);
|
||||||
|
assertThat(loadedAuthorizationRequest2).isNull();
|
||||||
|
request.removeParameter(OAuth2ParameterNames.STATE);
|
||||||
|
request.addParameter(OAuth2ParameterNames.STATE, state3);
|
||||||
|
OAuth2AuthorizationRequest loadedAuthorizationRequest3 = this.authorizationRequestRepository
|
||||||
|
.loadAuthorizationRequest(request);
|
||||||
|
assertThat(loadedAuthorizationRequest3).isEqualTo(authorizationRequest3);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -1,5 +1,5 @@
|
||||||
/*
|
/*
|
||||||
* Copyright 2002-2017 the original author or authors.
|
* Copyright 2002-2021 the original author or authors.
|
||||||
*
|
*
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
* you may not use this file except in compliance with the License.
|
* you may not use this file except in compliance with the License.
|
||||||
|
@ -34,11 +34,12 @@ import static org.assertj.core.api.Assertions.assertThatThrownBy;
|
||||||
* Tests for {@link HttpSessionOAuth2AuthorizationRequestRepository}.
|
* Tests for {@link HttpSessionOAuth2AuthorizationRequestRepository}.
|
||||||
*
|
*
|
||||||
* @author Joe Grandja
|
* @author Joe Grandja
|
||||||
|
* @author Craig Andrews
|
||||||
*/
|
*/
|
||||||
@RunWith(MockitoJUnitRunner.class)
|
@RunWith(MockitoJUnitRunner.class)
|
||||||
public class HttpSessionOAuth2AuthorizationRequestRepositoryTests {
|
public abstract class HttpSessionOAuth2AuthorizationRequestRepositoryTests {
|
||||||
private HttpSessionOAuth2AuthorizationRequestRepository authorizationRequestRepository =
|
|
||||||
new HttpSessionOAuth2AuthorizationRequestRepository();
|
protected HttpSessionOAuth2AuthorizationRequestRepository authorizationRequestRepository;
|
||||||
|
|
||||||
@Test(expected = IllegalArgumentException.class)
|
@Test(expected = IllegalArgumentException.class)
|
||||||
public void loadAuthorizationRequestWhenHttpServletRequestIsNullThenThrowIllegalArgumentException() {
|
public void loadAuthorizationRequestWhenHttpServletRequestIsNullThenThrowIllegalArgumentException() {
|
||||||
|
@ -70,42 +71,6 @@ public class HttpSessionOAuth2AuthorizationRequestRepositoryTests {
|
||||||
assertThat(loadedAuthorizationRequest).isEqualTo(authorizationRequest);
|
assertThat(loadedAuthorizationRequest).isEqualTo(authorizationRequest);
|
||||||
}
|
}
|
||||||
|
|
||||||
// gh-5110
|
|
||||||
@Test
|
|
||||||
public void loadAuthorizationRequestWhenMultipleSavedThenReturnMatchingAuthorizationRequest() {
|
|
||||||
MockHttpServletRequest request = new MockHttpServletRequest();
|
|
||||||
MockHttpServletResponse response = new MockHttpServletResponse();
|
|
||||||
|
|
||||||
String state1 = "state-1122";
|
|
||||||
OAuth2AuthorizationRequest authorizationRequest1 = createAuthorizationRequest().state(state1).build();
|
|
||||||
this.authorizationRequestRepository.saveAuthorizationRequest(authorizationRequest1, request, response);
|
|
||||||
|
|
||||||
String state2 = "state-3344";
|
|
||||||
OAuth2AuthorizationRequest authorizationRequest2 = createAuthorizationRequest().state(state2).build();
|
|
||||||
this.authorizationRequestRepository.saveAuthorizationRequest(authorizationRequest2, request, response);
|
|
||||||
|
|
||||||
String state3 = "state-5566";
|
|
||||||
OAuth2AuthorizationRequest authorizationRequest3 = createAuthorizationRequest().state(state3).build();
|
|
||||||
this.authorizationRequestRepository.saveAuthorizationRequest(authorizationRequest3, request, response);
|
|
||||||
|
|
||||||
request.addParameter(OAuth2ParameterNames.STATE, state1);
|
|
||||||
OAuth2AuthorizationRequest loadedAuthorizationRequest1 =
|
|
||||||
this.authorizationRequestRepository.loadAuthorizationRequest(request);
|
|
||||||
assertThat(loadedAuthorizationRequest1).isEqualTo(authorizationRequest1);
|
|
||||||
|
|
||||||
request.removeParameter(OAuth2ParameterNames.STATE);
|
|
||||||
request.addParameter(OAuth2ParameterNames.STATE, state2);
|
|
||||||
OAuth2AuthorizationRequest loadedAuthorizationRequest2 =
|
|
||||||
this.authorizationRequestRepository.loadAuthorizationRequest(request);
|
|
||||||
assertThat(loadedAuthorizationRequest2).isEqualTo(authorizationRequest2);
|
|
||||||
|
|
||||||
request.removeParameter(OAuth2ParameterNames.STATE);
|
|
||||||
request.addParameter(OAuth2ParameterNames.STATE, state3);
|
|
||||||
OAuth2AuthorizationRequest loadedAuthorizationRequest3 =
|
|
||||||
this.authorizationRequestRepository.loadAuthorizationRequest(request);
|
|
||||||
assertThat(loadedAuthorizationRequest3).isEqualTo(authorizationRequest3);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void loadAuthorizationRequestWhenSavedAndStateParameterNullThenReturnNull() {
|
public void loadAuthorizationRequestWhenSavedAndStateParameterNullThenReturnNull() {
|
||||||
MockHttpServletRequest request = new MockHttpServletRequest();
|
MockHttpServletRequest request = new MockHttpServletRequest();
|
||||||
|
@ -284,11 +249,9 @@ public class HttpSessionOAuth2AuthorizationRequestRepositoryTests {
|
||||||
assertThat(removedAuthorizationRequest).isNull();
|
assertThat(removedAuthorizationRequest).isNull();
|
||||||
}
|
}
|
||||||
|
|
||||||
private OAuth2AuthorizationRequest.Builder createAuthorizationRequest() {
|
protected OAuth2AuthorizationRequest.Builder createAuthorizationRequest() {
|
||||||
return OAuth2AuthorizationRequest.authorizationCode()
|
return OAuth2AuthorizationRequest.authorizationCode().authorizationUri("https://example.com/oauth2/authorize")
|
||||||
.authorizationUri("https://example.com/oauth2/authorize")
|
.clientId("client-id-1234").state("state-1234");
|
||||||
.clientId("client-id-1234")
|
|
||||||
.state("state-1234");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static class MockDistributedHttpSession extends MockHttpSession {
|
static class MockDistributedHttpSession extends MockHttpSession {
|
||||||
|
|
Loading…
Reference in New Issue