Skip to content
Snippets Groups Projects
Commit 596a007b authored by Daniel Gerhardt's avatar Daniel Gerhardt
Browse files

Use aspect to populate SecurityContextHolder from WebSocket listeners

parent ebc7b920
Branches
No related merge requests found
...@@ -17,6 +17,7 @@ ...@@ -17,6 +17,7 @@
*/ */
package de.thm.arsnova.security; package de.thm.arsnova.security;
import de.thm.arsnova.entities.UserAuthentication;
import de.thm.arsnova.entities.UserProfile; import de.thm.arsnova.entities.UserProfile;
import org.springframework.security.core.GrantedAuthority; import org.springframework.security.core.GrantedAuthority;
...@@ -56,6 +57,14 @@ public class User implements org.springframework.security.core.userdetails.UserD ...@@ -56,6 +57,14 @@ public class User implements org.springframework.security.core.userdetails.UserD
providerUserDetails = details; providerUserDetails = details;
} }
public User(final UserAuthentication userAuthentication, final Collection<? extends GrantedAuthority> authorities) {
id = userAuthentication.getId();
loginId = userAuthentication.getUsername();
authProvider = userAuthentication.getAuthProvider();
this.authorities = authorities;
enabled = true;
}
@Override @Override
public Collection<? extends GrantedAuthority> getAuthorities() { public Collection<? extends GrantedAuthority> getAuthorities() {
return authorities; return authorities;
......
...@@ -48,11 +48,6 @@ import org.slf4j.LoggerFactory; ...@@ -48,11 +48,6 @@ import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired; import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Required; import org.springframework.beans.factory.annotation.Required;
import org.springframework.scheduling.annotation.Async; import org.springframework.scheduling.annotation.Async;
import org.springframework.security.authentication.AnonymousAuthenticationToken;
import org.springframework.security.core.Authentication;
import org.springframework.security.core.authority.AuthorityUtils;
import org.springframework.security.core.context.SecurityContext;
import org.springframework.security.core.context.SecurityContextHolder;
import org.springframework.stereotype.Component; import org.springframework.stereotype.Component;
import javax.annotation.PreDestroy; import javax.annotation.PreDestroy;
...@@ -96,7 +91,6 @@ public class ArsnovaSocketioServerImpl implements ArsnovaSocketioServer, Arsnova ...@@ -96,7 +91,6 @@ public class ArsnovaSocketioServerImpl implements ArsnovaSocketioServer, Arsnova
private String storepass; private String storepass;
private final Configuration config; private final Configuration config;
private SocketIOServer server; private SocketIOServer server;
private boolean securityInitialized;
public ArsnovaSocketioServerImpl() { public ArsnovaSocketioServerImpl() {
config = new Configuration(); config = new Configuration();
...@@ -244,9 +238,7 @@ public class ArsnovaSocketioServerImpl implements ArsnovaSocketioServer, Arsnova ...@@ -244,9 +238,7 @@ public class ArsnovaSocketioServerImpl implements ArsnovaSocketioServer, Arsnova
@Override @Override
@Timed @Timed
public void onConnect(final SocketIOClient client) { public void onConnect(final SocketIOClient client) {
if (!securityInitialized) {
initializeSecurity();
}
} }
}); });
...@@ -689,13 +681,4 @@ public class ArsnovaSocketioServerImpl implements ArsnovaSocketioServer, Arsnova ...@@ -689,13 +681,4 @@ public class ArsnovaSocketioServerImpl implements ArsnovaSocketioServer, Arsnova
@Override @Override
public void visit(DeleteRoomEvent event) { } public void visit(DeleteRoomEvent event) { }
private void initializeSecurity() {
Authentication auth = new AnonymousAuthenticationToken("websocket", "websocket",
AuthorityUtils.createAuthorityList("ROLE_WEBSOCKET_ACCESS"));
SecurityContext context = SecurityContextHolder.createEmptyContext();
context.setAuthentication(auth);
SecurityContextHolder.setContext(context);
securityInitialized = true;
}
} }
/*
* This file is part of ARSnova Backend.
* Copyright (C) 2012-2018 The ARSnova Team
*
* ARSnova Backend is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* ARSnova Backend is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
package de.thm.arsnova.websocket;
import com.corundumstudio.socketio.SocketIOClient;
import de.thm.arsnova.entities.UserAuthentication;
import de.thm.arsnova.security.User;
import de.thm.arsnova.services.UserService;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Configurable;
import org.springframework.security.authentication.UsernamePasswordAuthenticationToken;
import org.springframework.security.core.Authentication;
import org.springframework.security.core.GrantedAuthority;
import org.springframework.security.core.authority.SimpleGrantedAuthority;
import org.springframework.security.core.context.SecurityContext;
import org.springframework.security.core.context.SecurityContextHolder;
import java.util.HashSet;
import java.util.Set;
import java.util.UUID;
/**
* This aspect populates the SecurityContextHolder of Spring Security when data are received via WebSockets.
* It allows WebSocket listeners to access service methods which are secured by Spring Security annotations.
*
* @author Daniel Gerhardt
*/
@Aspect
@Configurable
public class WebsocketAuthenticationAspect {
private static final Logger logger = LoggerFactory.getLogger(WebsocketAuthenticationAspect.class);
private static final GrantedAuthority WEBSOCKET_AUTHORITY = new SimpleGrantedAuthority("ROLE_WEBSOCKET_ACCESS");
private UserService userService;
@Around("execution(void com.corundumstudio.socketio.listener.DataListener+.onData(..)) && args(client, message, ..)")
public <T> void handleWebsocketAuthentication(final ProceedingJoinPoint pjp,
final SocketIOClient client, final T message) throws Throwable {
logger.debug("Executing WebsocketAuthenticationAspect for onData event: Session Id: {}, Message Class: {}",
client.getSessionId(), message.getClass());
populateSecurityContext(client.getSessionId());
pjp.proceed();
clearSecurityContext();
}
private void populateSecurityContext(final UUID socketId) {
SecurityContext context = SecurityContextHolder.getContext();
UserAuthentication userAuth = userService.getUserToSocketId(socketId);
Set<GrantedAuthority> authorities = new HashSet<>();
authorities.add(WEBSOCKET_AUTHORITY);
User user = new User(userAuth, authorities);
Authentication auth = new UsernamePasswordAuthenticationToken(user, null, authorities);
context.setAuthentication(auth);
SecurityContextHolder.setContext(context);
}
private void clearSecurityContext() {
SecurityContextHolder.clearContext();
}
@Autowired
public void setUserService(final UserService userService) {
this.userService = userService;
}
}
<?xml version="1.0" encoding="UTF-8"?> <?xml version="1.0" encoding="UTF-8"?>
<aspectj> <aspectj>
<weaver options="-verbose -showWeaveInfo"> <weaver options="-verbose -showWeaveInfo">
<!-- only weave classes in our application-specific packages -->
<include within="de.thm.arsnova..*"/> <include within="de.thm.arsnova..*"/>
<include within="com.corundumstudio.socketio.listener.*"/>
</weaver> </weaver>
<aspects> <aspects>
<aspect name="de.thm.arsnova.aop.RangeAspect"/> <aspect name="de.thm.arsnova.aop.RangeAspect"/>
<aspect name="de.thm.arsnova.websocket.WebsocketAuthenticationAspect"/>
</aspects> </aspects>
</aspectj> </aspectj>
0% or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment