Skip to content

Commit 3f0b018

Browse files
committed
refactor: session creator is no longer singleton
1 parent 133807b commit 3f0b018

File tree

5 files changed

+66
-87
lines changed

5 files changed

+66
-87
lines changed

src/main/java/com/aws/greengrass/clientdevices/auth/ClientDevicesAuthService.java

+33-35
Original file line numberDiff line numberDiff line change
@@ -21,7 +21,6 @@
2121
import com.aws.greengrass.clientdevices.auth.session.MqttSessionFactory;
2222
import com.aws.greengrass.clientdevices.auth.session.SessionConfig;
2323
import com.aws.greengrass.clientdevices.auth.session.SessionCreator;
24-
import com.aws.greengrass.clientdevices.auth.session.SessionManager;
2524
import com.aws.greengrass.clientdevices.auth.util.ResizableLinkedBlockingQueue;
2625
import com.aws.greengrass.config.Node;
2726
import com.aws.greengrass.config.Topics;
@@ -93,6 +92,7 @@ protected void install() throws InterruptedException {
9392
initializeInfrastructure();
9493
initializeHandlers();
9594
subscribeToConfigChanges();
95+
initializeIpc();
9696
}
9797

9898
private int getValidCloudCallQueueSize(Topics topics) {
@@ -121,8 +121,8 @@ private void initializeInfrastructure() {
121121

122122
private void initializeHandlers() {
123123
// Register auth session handlers
124-
context.get(SessionManager.class).setSessionConfig(new SessionConfig(getConfig()));
125-
SessionCreator.registerSessionFactory("mqtt", context.get(MqttSessionFactory.class));
124+
context.put(SessionConfig.class, new SessionConfig(getConfig()));
125+
context.get(SessionCreator.class).registerSessionFactory("mqtt", context.get(MqttSessionFactory.class));
126126

127127
// Register domain event handlers
128128
context.get(CACertificateChainChangedHandler.class).listen();
@@ -134,6 +134,36 @@ private void initializeHandlers() {
134134
networkState.registerHandler(context.get(CISShadowMonitor.class));
135135
}
136136

137+
private void initializeIpc() {
138+
AuthorizationHandler authorizationHandler = context.get(AuthorizationHandler.class);
139+
try {
140+
authorizationHandler.registerComponent(this.getName(),
141+
new HashSet<>(Arrays.asList(SUBSCRIBE_TO_CERTIFICATE_UPDATES,
142+
VERIFY_CLIENT_DEVICE_IDENTITY,
143+
GET_CLIENT_DEVICE_AUTH_TOKEN,
144+
AUTHORIZE_CLIENT_DEVICE_ACTION)));
145+
} catch (com.aws.greengrass.authorization.exceptions.AuthorizationException e) {
146+
logger.atError("initialize-cda-service-authorization-error", e)
147+
.log("Failed to initialize the client device auth service with the Authorization module.");
148+
}
149+
150+
GreengrassCoreIPCService greengrassCoreIPCService = context.get(GreengrassCoreIPCService.class);
151+
ClientDevicesAuthServiceApi serviceApi = context.get(ClientDevicesAuthServiceApi.class);
152+
CertificateManager certificateManager = context.get(CertificateManager.class);
153+
154+
greengrassCoreIPCService.setSubscribeToCertificateUpdatesHandler(context ->
155+
new SubscribeToCertificateUpdatesOperationHandler(context, certificateManager, authorizationHandler));
156+
greengrassCoreIPCService.setVerifyClientDeviceIdentityHandler(context ->
157+
new VerifyClientDeviceIdentityOperationHandler(context, serviceApi,
158+
authorizationHandler, cloudCallThreadPool));
159+
greengrassCoreIPCService.setGetClientDeviceAuthTokenHandler(context ->
160+
new GetClientDeviceAuthTokenOperationHandler(context, serviceApi, authorizationHandler,
161+
cloudCallThreadPool));
162+
greengrassCoreIPCService.setAuthorizeClientDeviceActionHandler(context ->
163+
new AuthorizeClientDeviceActionOperationHandler(context, serviceApi,
164+
authorizationHandler));
165+
}
166+
137167
private void subscribeToConfigChanges() {
138168
onConfigurationChanged();
139169
config.lookupTopics(CONFIGURATION_CONFIG_KEY).subscribe(this::configChangeHandler);
@@ -195,38 +225,6 @@ protected void shutdown() throws InterruptedException {
195225
context.get(CertificateManager.class).stopMonitors();
196226
}
197227

198-
@Override
199-
public void postInject() {
200-
super.postInject();
201-
AuthorizationHandler authorizationHandler = context.get(AuthorizationHandler.class);
202-
try {
203-
authorizationHandler.registerComponent(this.getName(),
204-
new HashSet<>(Arrays.asList(SUBSCRIBE_TO_CERTIFICATE_UPDATES,
205-
VERIFY_CLIENT_DEVICE_IDENTITY,
206-
GET_CLIENT_DEVICE_AUTH_TOKEN,
207-
AUTHORIZE_CLIENT_DEVICE_ACTION)));
208-
} catch (com.aws.greengrass.authorization.exceptions.AuthorizationException e) {
209-
logger.atError("initialize-cda-service-authorization-error", e)
210-
.log("Failed to initialize the client device auth service with the Authorization module.");
211-
}
212-
213-
GreengrassCoreIPCService greengrassCoreIPCService = context.get(GreengrassCoreIPCService.class);
214-
ClientDevicesAuthServiceApi serviceApi = context.get(ClientDevicesAuthServiceApi.class);
215-
CertificateManager certificateManager = context.get(CertificateManager.class);
216-
217-
greengrassCoreIPCService.setSubscribeToCertificateUpdatesHandler(context ->
218-
new SubscribeToCertificateUpdatesOperationHandler(context, certificateManager, authorizationHandler));
219-
greengrassCoreIPCService.setVerifyClientDeviceIdentityHandler(context ->
220-
new VerifyClientDeviceIdentityOperationHandler(context, serviceApi,
221-
authorizationHandler, cloudCallThreadPool));
222-
greengrassCoreIPCService.setGetClientDeviceAuthTokenHandler(context ->
223-
new GetClientDeviceAuthTokenOperationHandler(context, serviceApi, authorizationHandler,
224-
cloudCallThreadPool));
225-
greengrassCoreIPCService.setAuthorizeClientDeviceActionHandler(context ->
226-
new AuthorizeClientDeviceActionOperationHandler(context, serviceApi,
227-
authorizationHandler));
228-
}
229-
230228
public CertificateManager getCertificateManager() {
231229
return context.get(CertificateManager.class);
232230
}

src/main/java/com/aws/greengrass/clientdevices/auth/session/SessionCreator.java

+5-15
Original file line numberDiff line numberDiff line change
@@ -15,18 +15,12 @@
1515
public class SessionCreator {
1616
private static final Logger logger = LogManager.getLogger(SessionCreator.class);
1717

18-
@SuppressWarnings("PMD.UnusedPrivateField")
1918
private final Map<String, SessionFactory> factoryMap;
2019

21-
private SessionCreator() {
20+
public SessionCreator() {
2221
factoryMap = new ConcurrentHashMap<>();
2322
}
2423

25-
private static class SessionFactorySingleton {
26-
@SuppressWarnings("PMD.AccessorClassGeneration")
27-
private static final SessionCreator INSTANCE = new SessionCreator();
28-
}
29-
3024
/**
3125
* Create a client device session.
3226
*
@@ -35,9 +29,9 @@ private static class SessionFactorySingleton {
3529
* @return new session if the client can be authenticated
3630
* @throws AuthenticationException if the client fails to be authenticated
3731
*/
38-
public static Session createSession(String credentialType, Map<String, String> credentialMap)
32+
public Session createSession(String credentialType, Map<String, String> credentialMap)
3933
throws AuthenticationException {
40-
SessionFactory sessionFactory = SessionFactorySingleton.INSTANCE.factoryMap.get(credentialType);
34+
SessionFactory sessionFactory = factoryMap.get(credentialType);
4135
if (sessionFactory == null) {
4236
logger.atWarn().kv("credentialType", credentialType)
4337
.log("no registered handler to process device credentials");
@@ -46,11 +40,7 @@ public static Session createSession(String credentialType, Map<String, String> c
4640
return sessionFactory.createSession(credentialMap);
4741
}
4842

49-
public static void registerSessionFactory(String credentialType, SessionFactory sessionFactory) {
50-
SessionFactorySingleton.INSTANCE.factoryMap.put(credentialType, sessionFactory);
51-
}
52-
53-
public static void unregisterSessionFactory(String credentialType) {
54-
SessionFactorySingleton.INSTANCE.factoryMap.remove(credentialType);
43+
public void registerSessionFactory(String credentialType, SessionFactory sessionFactory) {
44+
factoryMap.put(credentialType, sessionFactory);
5545
}
5646
}

src/main/java/com/aws/greengrass/clientdevices/auth/session/SessionManager.java

+10-14
Original file line numberDiff line numberDiff line change
@@ -15,14 +15,15 @@
1515
import java.util.LinkedHashMap;
1616
import java.util.Map;
1717
import java.util.UUID;
18+
import javax.inject.Inject;
1819

19-
/**
20-
* Singleton class for managing AuthN and AuthZ sessions.
21-
*/
2220
public class SessionManager {
2321
private static final Logger logger = LogManager.getLogger(SessionManager.class);
2422
private static final String SESSION_ID = "SessionId";
2523

24+
private final SessionCreator sessionCreator;
25+
private final SessionConfig sessionConfig;
26+
2627
// Thread-safe LRU Session Cache that evicts the eldest entry (based on access order) upon reaching its size.
2728
// TODO: Support time-based cache eviction (Session timeout) and Session deduping.
2829
@Getter(AccessLevel.PACKAGE)
@@ -40,7 +41,11 @@ protected boolean removeEldestEntry(Map.Entry<String, Session> eldest) {
4041
}
4142
});
4243

43-
private SessionConfig sessionConfig;
44+
@Inject
45+
public SessionManager(SessionCreator sessionCreator, SessionConfig sessionConfig) {
46+
this.sessionCreator = sessionCreator;
47+
this.sessionConfig = sessionConfig;
48+
}
4449

4550
/**
4651
* Looks up a session by id.
@@ -62,7 +67,7 @@ public Session findSession(String sessionId) {
6267
*/
6368
public String createSession(String credentialType, Map<String, String> credentialMap)
6469
throws AuthenticationException {
65-
Session session = SessionCreator.createSession(credentialType, credentialMap);
70+
Session session = sessionCreator.createSession(credentialType, credentialMap);
6671
return addSessionInternal(session);
6772
}
6873

@@ -76,15 +81,6 @@ public void closeSession(String sessionId) {
7681
closeSessionInternal(sessionId);
7782
}
7883

79-
/**
80-
* Session configuration setter.
81-
*
82-
* @param sessionConfig session configuration
83-
*/
84-
public void setSessionConfig(SessionConfig sessionConfig) {
85-
this.sessionConfig = sessionConfig;
86-
}
87-
8884
private synchronized void closeSessionInternal(String sessionId) {
8985
sessionMap.remove(sessionId);
9086
}

src/test/java/com/aws/greengrass/clientdevices/auth/session/SessionCreatorTest.java

+11-9
Original file line numberDiff line numberDiff line change
@@ -7,8 +7,8 @@
77

88
import com.aws.greengrass.clientdevices.auth.exception.AuthenticationException;
99
import com.aws.greengrass.testcommons.testutilities.GGExtension;
10-
import org.junit.jupiter.api.AfterEach;
1110
import org.junit.jupiter.api.Assertions;
11+
import org.junit.jupiter.api.BeforeEach;
1212
import org.junit.jupiter.api.Test;
1313
import org.junit.jupiter.api.extension.ExtendWith;
1414
import org.mockito.Mock;
@@ -27,18 +27,20 @@ public class SessionCreatorTest {
2727
private static final String mqttCredentialType = "mqtt";
2828
private static final String unknownCredentialType = "unknown";
2929

30+
private SessionCreator sessionCreator;
31+
3032
@Mock
3133
private MqttSessionFactory mqttSessionFactory;
3234

33-
@AfterEach
34-
void afterEach() {
35-
SessionCreator.unregisterSessionFactory(mqttCredentialType);
35+
@BeforeEach
36+
void beforeEach() {
37+
sessionCreator = new SessionCreator();
3638
}
3739

3840
@Test
3941
void GIVEN_noRegisteredFactories_WHEN_createSession_THEN_throwsException() {
4042
Assertions.assertThrows(IllegalArgumentException.class,
41-
() -> SessionCreator.createSession(mqttCredentialType, new HashMap<>()));
43+
() -> sessionCreator.createSession(mqttCredentialType, new HashMap<>()));
4244
}
4345

4446
@Test
@@ -47,14 +49,14 @@ void GIVEN_registeredMqttSessionFactory_WHEN_createSessionWithMqttCredentials_TH
4749
Session mockSession = mock(SessionImpl.class);
4850
when(mqttSessionFactory.createSession(any())).thenReturn(mockSession);
4951

50-
SessionCreator.registerSessionFactory(mqttCredentialType, mqttSessionFactory);
51-
assertThat(SessionCreator.createSession(mqttCredentialType, new HashMap<>()), is(mockSession));
52+
sessionCreator.registerSessionFactory(mqttCredentialType, mqttSessionFactory);
53+
assertThat(sessionCreator.createSession(mqttCredentialType, new HashMap<>()), is(mockSession));
5254
}
5355

5456
@Test
5557
void GIVEN_registeredMqttSessionFactory_WHEN_createSession_WithNonMqttCredentials_THEN_throwsException() {
56-
SessionCreator.registerSessionFactory(mqttCredentialType, mqttSessionFactory);
58+
sessionCreator.registerSessionFactory(mqttCredentialType, mqttSessionFactory);
5759
Assertions.assertThrows(IllegalArgumentException.class,
58-
() -> SessionCreator.createSession(unknownCredentialType, new HashMap<>()));
60+
() -> sessionCreator.createSession(unknownCredentialType, new HashMap<>()));
5961
}
6062
}

src/test/java/com/aws/greengrass/clientdevices/auth/session/SessionManagerTest.java

+7-14
Original file line numberDiff line numberDiff line change
@@ -8,7 +8,6 @@
88

99
import com.aws.greengrass.clientdevices.auth.exception.AuthenticationException;
1010
import com.aws.greengrass.testcommons.testutilities.GGExtension;
11-
import org.junit.jupiter.api.AfterEach;
1211
import org.junit.jupiter.api.BeforeEach;
1312
import org.junit.jupiter.api.Test;
1413
import org.junit.jupiter.api.extension.ExtendWith;
@@ -35,6 +34,7 @@ class SessionManagerTest {
3534
private static final String CREDENTIAL_TYPE = "mqtt";
3635
private static final int MOCK_SESSION_CAPACITY = 10;
3736

37+
private SessionCreator sessionCreator;
3838
private SessionManager sessionManager;
3939
@Mock
4040
private MqttSessionFactory mockSessionFactory;
@@ -66,19 +66,14 @@ class SessionManagerTest {
6666
@BeforeEach
6767
void beforeEach() throws AuthenticationException {
6868
lenient().when(mockSessionConfig.getSessionCapacity()).thenReturn(MOCK_SESSION_CAPACITY);
69-
sessionManager = new SessionManager();
70-
sessionManager.setSessionConfig(mockSessionConfig);
71-
SessionCreator.registerSessionFactory(CREDENTIAL_TYPE, mockSessionFactory);
69+
sessionCreator = new SessionCreator();
70+
sessionManager = new SessionManager(sessionCreator, mockSessionConfig);
71+
sessionCreator.registerSessionFactory(CREDENTIAL_TYPE, mockSessionFactory);
7272
lenient().when(mockSessionFactory.createSession(credentialMap)).thenReturn(mockSession);
7373
lenient().when(mockSessionFactory.createSession(credentialMap2)).thenReturn(mockSession2);
7474
lenient().when(mockSessionFactory.createSession(invalidCredentialMap)).thenThrow(new AuthenticationException(""));
7575
}
7676

77-
@AfterEach
78-
void afterEach() {
79-
SessionCreator.unregisterSessionFactory(CREDENTIAL_TYPE);
80-
}
81-
8277
@Test
8378
void GIVEN_validDeviceCredentials_WHEN_createSession_THEN_sessionCreatedWithUniqueIds()
8479
throws AuthenticationException {
@@ -111,7 +106,7 @@ void GIVEN_invalidDeviceCredentials_WHEN_createSession_THEN_throwsAuthentication
111106
}
112107

113108
@Test
114-
void GIVEN_validDeviceCredentials_WHEN_createSession_beyond_capacity_THEN_passes_evicting_eldest_session()
109+
void GIVEN_maxOpenSessions_WHEN_createSession_THEN_oldestSessionIsEvicted()
115110
throws AuthenticationException {
116111
reset(mockSessionConfig);
117112
reset(mockSessionFactory);
@@ -150,10 +145,8 @@ void GIVEN_validDeviceCredentials_WHEN_createSession_beyond_capacity_THEN_passes
150145
when(mockSessionFactory.createSession(credentialMap3)).thenReturn(mockSession3);
151146
when(mockSessionFactory.createSession(credentialMap4)).thenReturn(mockSession4);
152147

153-
int mockSessionCapacity = 3;
154-
when(mockSessionConfig.getSessionCapacity()).thenReturn(mockSessionCapacity);
155-
SessionManager sessionManager = new SessionManager();
156-
sessionManager.setSessionConfig(mockSessionConfig);
148+
int sessionCapacity = 3;
149+
when(mockSessionConfig.getSessionCapacity()).thenReturn(sessionCapacity);
157150

158151
// fill session cache to its capacity
159152
String id1 = sessionManager.createSession(CREDENTIAL_TYPE, credentialMap1);

0 commit comments

Comments
 (0)