2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								[[websocket]]
							 
						 
					
						
							
								
									
										
										
										
											2021-07-30 13:52:15 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								= WebSocket Security
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2019-03-19 23:53:23 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								Spring Security 4 added support for securing https://docs.spring.io/spring/docs/current/spring-framework-reference/html/websocket.html[Spring's WebSocket support].
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								This section describes how to use Spring Security's WebSocket support.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								.Direct JSR-356 Support
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								****
							 
						 
					
						
							
								
									
										
										
										
											2021-04-21 16:01:26 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								Spring Security does not provide direct JSR-356 support, because doing so would provide little value.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								This is because the format is unknown, and there is https://docs.spring.io/spring/docs/current/spring-framework-reference/html/websocket.html#websocket-intro-sub-protocol[little Spring can do to secure an unknown format].
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								Additionally, JSR-356 does not provide a way to intercept messages, so security would be invasive.
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								****
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-04-07 17:39:10 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								[[websocket-authentication]]
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								== WebSocket Authentication
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								WebSockets reuse the same authentication information that is found in the HTTP request when the WebSocket connection was made.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								This means that the `Principal` on the `HttpServletRequest` will be handed off to WebSockets.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								If you are using Spring Security, the `Principal` on the `HttpServletRequest` is overridden automatically.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								More concretely, to ensure a user has authenticated to your WebSocket application, all that is necessary is to ensure that you setup Spring Security to authenticate your HTTP based web application.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-03-16 10:13:43 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								[[websocket-authorization]]
							 
						 
					
						
							
								
									
										
										
										
											2022-04-07 17:39:10 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								== WebSocket Authorization
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								Spring Security 4.0 has introduced authorization support for WebSockets through the Spring Messaging abstraction.
							 
						 
					
						
							
								
									
										
										
										
											2022-04-07 17:39:10 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								In Spring Security 5.8, this support has been refreshed to use the `AuthorizationManager` API.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-05-10 15:57:25 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								To configure authorization using Java Configuration, simply include the `@EnableWebSocketSecurity` annotation and publish an `AuthorizationManager<Message<?>>` bean or in xref:servlet/appendix/namespace/websocket.adoc#nsa-websocket-security[XML] use the `use-authorization-manager` attribute.
							 
						 
					
						
							
								
									
										
										
										
											2022-04-07 17:39:10 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								One way to do this is by using the `AuthorizationManagerMessageMatcherRegistry` to specify endpoint patterns like so:
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-06-18 21:30:41 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								[tabs]
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								======
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								Java::
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								+
							 
						 
					
						
							
								
									
										
										
										
											2021-06-16 10:31:29 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								[source,java,role="primary"]
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								@Configuration
							 
						 
					
						
							
								
									
										
										
										
											2022-04-07 17:39:10 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								@EnableWebSocketSecurity // <1> <2>
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								public class WebSocketSecurityConfig {
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-04-07 17:39:10 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    @Bean
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    AuthorizationManager<Message<?>> messageAuthorizationManager(MessageMatcherDelegatingAuthorizationManager.Builder messages) {
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								        messages
							 
						 
					
						
							
								
									
										
										
										
											2023-05-10 15:57:25 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								                .simpDestMatchers("/user/**").hasRole("USER") // <3>
							 
						 
					
						
							
								
									
										
										
										
											2022-04-07 17:39:10 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return messages.build();
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-06-18 21:30:41 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								Kotlin::
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								+
							 
						 
					
						
							
								
									
										
										
										
											2021-06-16 10:31:29 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								[source,kotlin,role="secondary"]
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								@Configuration
							 
						 
					
						
							
								
									
										
										
										
											2022-04-07 17:39:10 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								@EnableWebSocketSecurity // <1> <2>
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								open class WebSocketSecurityConfig { // <1> <2>
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    @Bean
							 
						 
					
						
							
								
									
										
										
										
											2023-08-21 06:06:37 +09:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    fun messageAuthorizationManager(messages: MessageMatcherDelegatingAuthorizationManager.Builder): AuthorizationManager<Message<*>> {
							 
						 
					
						
							
								
									
										
										
										
											2023-05-10 15:57:25 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        messages.simpDestMatchers("/user/**").hasRole("USER") // <3>
							 
						 
					
						
							
								
									
										
										
										
											2022-04-07 17:39:10 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        return messages.build()
							 
						 
					
						
							
								
									
										
										
										
											2021-06-16 10:31:29 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-06-18 21:40:45 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								Xml::
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								+
							 
						 
					
						
							
								
									
										
										
										
											2023-05-10 15:57:25 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								[source,xml,role="secondary"]
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
									
										
										
										
											2023-05-10 15:57:25 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								<websocket-message-broker use-authorization-manager="true"> <1> <2>
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    <intercept-message pattern="/user/**" access="hasRole('USER')"/> <3>
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								</websocket-message-broker>
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
									
										
										
										
											2023-06-18 21:40:45 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								======
							 
						 
					
						
							
								
									
										
										
										
											2023-05-10 15:57:25 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								<1> Any inbound CONNECT message requires a valid CSRF token to enforce the <<websocket-sameorigin,Same Origin Policy>>.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								<2> The `SecurityContextHolder` is populated with the user within the `simpUser` header attribute for any inbound request.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								<3> Our messages require the proper authorization. Specifically, any inbound message that starts with `/user/` will require `ROLE_USER`. You can find additional details on authorization in <<websocket-authorization>>
							 
						 
					
						
							
								
									
										
										
										
											2022-04-07 17:39:10 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								=== Custom Authorization
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-04-07 17:39:10 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								When using `AuthorizationManager`, customization is quite simple.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								For example, you can publish an `AuthorizationManager` that requires that all messages have a role of "USER" using `AuthorityAuthorizationManager`, as seen below:
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-06-18 21:32:35 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								[tabs]
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								======
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								Java::
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								+
							 
						 
					
						
							
								
									
										
										
										
											2022-04-07 17:39:10 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								[source,java,role="primary"]
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								@Configuration
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								@EnableWebSocketSecurity // <1> <2>
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								public class WebSocketSecurityConfig {
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-04-07 17:39:10 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    @Bean
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    AuthorizationManager<Message<?>> messageAuthorizationManager(MessageMatcherDelegatingAuthorizationManager.Builder messages) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return AuthorityAuthorizationManager.hasRole("USER");
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-06-18 21:32:35 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								Kotlin::
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								+
							 
						 
					
						
							
								
									
										
										
										
											2022-04-07 17:39:10 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								[source,kotlin,role="secondary"]
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								@Configuration
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								@EnableWebSocketSecurity // <1> <2>
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								open class WebSocketSecurityConfig {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    @Bean
							 
						 
					
						
							
								
									
										
										
										
											2023-08-21 06:06:37 +09:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    fun messageAuthorizationManager(messages: MessageMatcherDelegatingAuthorizationManager.Builder): AuthorizationManager<Message<*>> {
							 
						 
					
						
							
								
									
										
										
										
											2022-04-07 17:39:10 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        return AuthorityAuthorizationManager.hasRole("USER") // <3>
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-06-18 21:32:35 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								Xml::
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								+
							 
						 
					
						
							
								
									
										
										
										
											2022-04-07 17:39:10 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								[source,xml,role="secondary"]
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								<bean id="authorizationManager" class="org.example.MyAuthorizationManager"/>
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								<websocket-message-broker authorization-manager-ref="myAuthorizationManager"/>
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
									
										
										
										
											2023-06-18 21:32:35 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								======
							 
						 
					
						
							
								
									
										
										
										
											2022-04-07 17:39:10 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								There are several ways to further match messages, as can be seen in a more advanced example below:
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-06-18 21:30:41 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								[tabs]
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								======
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								Java::
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								+
							 
						 
					
						
							
								
									
										
										
										
											2021-06-16 10:31:29 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								[source,java,role="primary"]
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								@Configuration
							 
						 
					
						
							
								
									
										
										
										
											2022-04-07 17:39:10 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								public class WebSocketSecurityConfig {
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-04-07 17:39:10 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    @Bean
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    public AuthorizationManager<Message<?>> messageAuthorizationManager(MessageMatcherDelegatingAuthorizationManager.Builder messages) {
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								        messages
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                .nullDestMatcher().authenticated() // <1>
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                .simpSubscribeDestMatchers("/user/queue/errors").permitAll() // <2>
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                .simpDestMatchers("/app/**").hasRole("USER") // <3>
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                .simpSubscribeDestMatchers("/user/**", "/topic/friends/*").hasRole("USER") // <4>
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                .simpTypeMatchers(MESSAGE, SUBSCRIBE).denyAll() // <5>
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                .anyMessage().denyAll(); // <6>
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-04-07 17:39:10 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        return messages.build();
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-06-18 21:30:41 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								Kotlin::
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								+
							 
						 
					
						
							
								
									
										
										
										
											2021-06-16 10:31:29 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								[source,kotlin,role="secondary"]
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								@Configuration
							 
						 
					
						
							
								
									
										
										
										
											2022-04-07 17:39:10 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								open class WebSocketSecurityConfig {
							 
						 
					
						
							
								
									
										
										
										
											2023-08-21 06:06:37 +09:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    fun messageAuthorizationManager(messages: MessageMatcherDelegatingAuthorizationManager.Builder): AuthorizationManager<Message<*>> {
							 
						 
					
						
							
								
									
										
										
										
											2021-06-16 10:31:29 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        messages
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            .nullDestMatcher().authenticated() // <1>
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            .simpSubscribeDestMatchers("/user/queue/errors").permitAll() // <2>
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            .simpDestMatchers("/app/**").hasRole("USER") // <3>
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            .simpSubscribeDestMatchers("/user/**", "/topic/friends/*").hasRole("USER") // <4>
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            .simpTypeMatchers(MESSAGE, SUBSCRIBE).denyAll() // <5>
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            .anyMessage().denyAll() // <6>
							 
						 
					
						
							
								
									
										
										
										
											2022-04-07 17:39:10 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return messages.build();
							 
						 
					
						
							
								
									
										
										
										
											2021-06-16 10:31:29 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-06-18 21:32:35 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								Xml::
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								+
							 
						 
					
						
							
								
									
										
										
										
											2022-04-07 17:39:10 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								[source,kotlin,role="secondary"]
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
									
										
										
										
											2022-04-07 17:39:10 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								<websocket-message-broker use-authorization-manager="true">
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								    <!--1-->
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    <intercept-message type="CONNECT" access="permitAll" />
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    <intercept-message type="UNSUBSCRIBE" access="permitAll" />
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    <intercept-message type="DISCONNECT" access="permitAll" />
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    <intercept-message pattern="/user/queue/errors" type="SUBSCRIBE" access="permitAll" /> <!--2-->
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    <intercept-message pattern="/app/**" access="hasRole('USER')" />      <!--3-->
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    <!--4-->
							 
						 
					
						
							
								
									
										
										
										
											2022-04-07 17:39:10 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    <intercept-message pattern="/user/**" type="SUBSCRIBE" access="hasRole('USER')" />
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    <intercept-message pattern="/topic/friends/*" type="SUBSCRIBE" access="hasRole('USER')" />
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    <!--5-->
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    <intercept-message type="MESSAGE" access="denyAll" />
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    <intercept-message type="SUBSCRIBE" access="denyAll" />
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    <intercept-message pattern="/**" access="denyAll" /> <!--6-->
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								</websocket-message-broker>
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
									
										
										
										
											2023-06-18 21:32:35 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								======
							 
						 
					
						
							
								
									
										
										
										
											2022-04-07 17:39:10 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								This will ensure that:
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								<1> Any message without a destination (i.e. anything other than Message type of MESSAGE or SUBSCRIBE) will require the user to be authenticated
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								<2> Anyone can subscribe to /user/queue/errors
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								<3> Any message that has a destination starting with "/app/" will be require the user to have the role ROLE_USER
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								<4> Any message that starts with "/user/" or "/topic/friends/" that is of type SUBSCRIBE will require ROLE_USER
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								<5> Any other message of type MESSAGE or SUBSCRIBE is rejected. Due to 6 we do not need this step, but it illustrates how one can match on specific message types.
							 
						 
					
						
							
								
									
										
										
										
											2022-04-07 17:39:10 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								<6> Any other Message is rejected. This is a good idea to ensure that you do not miss any messages.
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2024-12-09 15:25:33 -07:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								[[migrating-spel-expressions]]
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								=== Migrating SpEL Expressions
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								If you are migrating from an older version of Spring Security, your destination matchers may include SpEL expressions.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								It's recommended that these be changed to using concrete implementations of `AuthorizationManager` since this is independently testable.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								However, to ease migration, you can also use a class like the following:
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								[source,java]
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								public final class MessageExpressionAuthorizationManager implements AuthorizationManager<MessageAuthorizationContext<?>> {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									private SecurityExpressionHandler<Message<?>> expressionHandler = new DefaultMessageSecurityExpressionHandler();
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									private Expression expression;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									public MessageExpressionAuthorizationManager(String expressionString) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										Assert.hasText(expressionString, "expressionString cannot be empty");
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										this.expression = this.expressionHandler.getExpressionParser().parseExpression(expressionString);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									@Override
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									public AuthorizationDecision check(Supplier<Authentication> authentication, MessageAuthorizationContext<?> context) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										EvaluationContext ctx = this.expressionHandler.createEvaluationContext(authentication, context.getMessage());
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										boolean granted = ExpressionUtils.evaluateAsBoolean(this.expression, ctx);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
										return new ExpressionAuthorizationDecision(granted, this.expression);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
									}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								And specify an instance for each matcher that you cannot get migrate:
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								[tabs]
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								======
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								Java::
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								+
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								[source,java,role="primary"]
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								@Configuration
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								public class WebSocketSecurityConfig {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    @Bean
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    public AuthorizationManager<Message<?>> messageAuthorizationManager(MessageMatcherDelegatingAuthorizationManager.Builder messages) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        messages
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                // ...
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                .simpSubscribeDestMatchers("/topic/friends/{friend}").access(new MessageExpressionAuthorizationManager("#friends == 'john"));
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                // ...
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return messages.build();
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								Kotlin::
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								+
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								[source,kotlin,role="secondary"]
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								@Configuration
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								open class WebSocketSecurityConfig {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    fun messageAuthorizationManager(messages: MessageMatcherDelegatingAuthorizationManager.Builder): AuthorizationManager<Message<?> {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        messages
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            // ..
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            .simpSubscribeDestMatchers("/topic/friends/{friends}").access(MessageExpressionAuthorizationManager("#friends == 'john"))
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            // ...
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return messages.build()
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								======
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								[[websocket-authorization-notes]]
							 
						 
					
						
							
								
									
										
										
										
											2021-07-30 13:52:15 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								=== WebSocket Authorization Notes
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-04-21 16:01:26 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								To properly secure your application, you need to understand Spring's WebSocket support.
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								[[websocket-authorization-notes-messagetypes]]
							 
						 
					
						
							
								
									
										
										
										
											2021-07-30 13:52:15 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								==== WebSocket Authorization on Message Types
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-04-21 16:01:26 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								You need to understand the distinction between `SUBSCRIBE` and `MESSAGE` types of messages and how they work within Spring.
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-04-21 16:01:26 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								Consider a chat application:
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-04-21 16:01:26 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								* The system can send a notification `MESSAGE` to all users through a destination of `/topic/system/notifications`.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								* Clients can receive notifications by `SUBSCRIBE` to the `/topic/system/notifications`.
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-04-21 16:01:26 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								While we want clients to be able to `SUBSCRIBE` to `/topic/system/notifications`, we do not want to enable them to send a `MESSAGE` to that destination.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								If we allowed sending a `MESSAGE` to `/topic/system/notifications`, clients could send a message directly to that endpoint and impersonate the system.
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-04-21 16:01:26 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								In general, it is common for applications to deny any `MESSAGE` sent to a destination that starts with the https://docs.spring.io/spring/docs/current/spring-framework-reference/html/websocket.html#websocket-stomp[broker prefix] (`/topic/` or `/queue/`).
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								[[websocket-authorization-notes-destinations]]
							 
						 
					
						
							
								
									
										
										
										
											2021-07-30 13:52:15 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								==== WebSocket Authorization on Destinations
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-04-21 16:01:26 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								You should also understand how destinations are transformed.
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-04-21 16:01:26 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								Consider a chat application:
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-04-21 16:01:26 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								* Users can send messages to a specific user by sending a message to the `/app/chat` destination.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								* The application sees the message, ensures that the `from` attribute is specified as the current user (we cannot trust the client).
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								* The application then sends the message to the recipient by using `SimpMessageSendingOperations.convertAndSendToUser("toUser", "/queue/messages", message)`.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								* The message gets turned into the destination of `/queue/user/messages-<sessionid>`.
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-04-21 16:01:26 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								With this chat application, we want to let our client to listen `/user/queue`, which is transformed into `/queue/user/messages-<sessionid>`.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								However, we do not want the client to be able to listen to `/queue/*`, because that would let the client see messages for every user.
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-04-21 16:01:26 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								In general, it is common for applications to deny any `SUBSCRIBE` sent to a message that starts with the https://docs.spring.io/spring/docs/current/spring-framework-reference/html/websocket.html#websocket-stomp[broker prefix] (`/topic/` or `/queue/`).
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								We may provide exceptions to account for things like
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								//FIXME: Like what?
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								[[websocket-authorization-notes-outbound]]
							 
						 
					
						
							
								
									
										
										
										
											2021-07-30 13:52:15 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								=== Outbound Messages
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-04-21 16:01:26 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								The Spring Framework reference documentation contains a section titled https://docs.spring.io/spring/docs/current/spring-framework-reference/html/websocket.html#websocket-stomp-message-flow["`Flow of Messages`"] that describes how messages flow through the system.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								Note that Spring Security secures only the `clientInboundChannel`.
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								Spring Security does not attempt to secure the `clientOutboundChannel`.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								The most important reason for this is performance.
							 
						 
					
						
							
								
									
										
										
										
											2021-04-21 16:01:26 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								For every message that goes in, typically many more go out.
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								Instead of securing the outbound messages, we encourage securing the subscription to the endpoints.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								[[websocket-sameorigin]]
							 
						 
					
						
							
								
									
										
										
										
											2021-07-30 13:52:15 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								== Enforcing Same Origin Policy
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-04-21 16:01:26 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								Note that the browser does not enforce the https://en.wikipedia.org/wiki/Same-origin_policy[Same Origin Policy] for WebSocket connections.
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								This is an extremely important consideration.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								[[websocket-sameorigin-why]]
							 
						 
					
						
							
								
									
										
										
										
											2021-07-30 13:52:15 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								=== Why Same Origin?
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								Consider the following scenario.
							 
						 
					
						
							
								
									
										
										
										
											2021-04-21 16:01:26 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								A user visits `bank.com` and authenticates to their account.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								The same user opens another tab in their browser and visits `evil.com`.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								The Same Origin Policy ensures that `evil.com` cannot read data from or write data to `bank.com`.
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-04-21 16:01:26 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								With WebSockets, the Same Origin Policy does not apply.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								In fact, unless `bank.com` explicitly forbids it, `evil.com` can read and write data on behalf of the user.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								This means that anything the user can do over the webSocket (such as transferring money), `evil.com` can do on that user's behalf.
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-04-21 16:01:26 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								Since SockJS tries to emulate WebSockets, it also bypasses the Same Origin Policy.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								This means that developers need to explicitly protect their applications from external domains when they use SockJS.
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								[[websocket-sameorigin-spring]]
							 
						 
					
						
							
								
									
										
										
										
											2021-07-30 13:52:15 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								=== Spring WebSocket Allowed Origin
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2019-03-19 23:53:23 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								Fortunately, since Spring 4.1.5 Spring's WebSocket and SockJS support restricts access to the https://docs.spring.io/spring/docs/current/spring-framework-reference/html/websocket.html#websocket-server-allowed-origins[current domain].
							 
						 
					
						
							
								
									
										
										
										
											2021-04-21 16:01:26 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								Spring Security adds an additional layer of protection to provide https://en.wikipedia.org/wiki/Defence_in_depth_(non-military)#Information_security[defense in depth].
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								[[websocket-sameorigin-csrf]]
							 
						 
					
						
							
								
									
										
										
										
											2021-07-30 13:52:15 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								=== Adding CSRF to Stomp Headers
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-12-13 16:57:36 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								By default, Spring Security requires the xref:features/exploits/csrf.adoc#csrf[CSRF token]  in any `CONNECT` message type.
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								This ensures that only a site that has access to the CSRF token can connect.
							 
						 
					
						
							
								
									
										
										
										
											2021-04-21 16:01:26 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								Since only the *same origin* can access the CSRF token, external domains are not allowed to make a connection.
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								Typically we need to include the CSRF token in an HTTP header or an HTTP parameter.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								However, SockJS does not allow for these options.
							 
						 
					
						
							
								
									
										
										
										
											2021-04-21 16:01:26 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								Instead, we must include the token in the Stomp headers.
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-05-02 16:08:37 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								Applications can xref:servlet/exploits/csrf.adoc#csrf-integration[obtain a CSRF token] by accessing the request attribute named `_csrf`.
							 
						 
					
						
							
								
									
										
										
										
											2021-04-21 16:01:26 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								For example, the following allows accessing the `CsrfToken` in a JSP:
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								[source,javascript]
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								var headerName = "${_csrf.headerName}";
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								var token = "${_csrf.token}";
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-04-21 16:01:26 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								If you use static HTML, you can expose the `CsrfToken` on a REST endpoint.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								For example, the following would expose the `CsrfToken` on the `/csrf` URL:
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-06-18 21:30:41 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								[tabs]
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								======
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								Java::
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								+
							 
						 
					
						
							
								
									
										
										
										
											2021-06-16 10:31:29 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								[source,java,role="primary"]
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								@RestController
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								public class CsrfController {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    @RequestMapping("/csrf")
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    public CsrfToken csrf(CsrfToken token) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return token;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-06-18 21:30:41 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								Kotlin::
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								+
							 
						 
					
						
							
								
									
										
										
										
											2021-06-16 10:31:29 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								[source,kotlin,role="secondary"]
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								@RestController
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								class CsrfController {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    @RequestMapping("/csrf")
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    fun csrf(token: CsrfToken): CsrfToken {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return token
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
									
										
										
										
											2023-06-18 21:30:41 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								======
							 
						 
					
						
							
								
									
										
										
										
											2021-06-16 10:31:29 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-04-21 16:01:26 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								The JavaScript can make a REST call to the endpoint and use the response to populate the `headerName` and the token.
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-04-21 16:01:26 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								We can now include the token in our Stomp client:
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								[source,javascript]
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								...
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								var headers = {};
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								headers[headerName] = token;
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								stompClient.connect(headers, function(frame) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								  ...
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-04-21 16:01:26 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								})
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								[[websocket-sameorigin-disable]]
							 
						 
					
						
							
								
									
										
										
										
											2021-07-30 13:52:15 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								=== Disable CSRF within WebSockets
							 
						 
					
						
							
								
									
										
										
										
											2022-04-07 17:39:10 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								NOTE: At this point, CSRF is not configurable when using `@EnableWebSocketSecurity`, though this will likely be added in a future release.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								To disable CSRF, instead of using `@EnableWebSocketSecurity`, you can use XML support or add the Spring Security components yourself, like so:
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-06-18 21:32:35 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								[tabs]
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								======
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								Java::
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								+
							 
						 
					
						
							
								
									
										
										
										
											2022-04-07 17:39:10 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								[source,java,role="primary"]
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								@Configuration
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								public class WebSocketSecurityConfig implements WebSocketMessageBrokerConfigurer {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    @Override
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    public void addArgumentResolvers(List<HandlerMethodArgumentResolver> argumentResolvers) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        argumentResolvers.add(new AuthenticationPrincipalArgumentResolver());
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    @Override
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    public void configureClientInboundChannel(ChannelRegistration registration) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        AuthorizationManager<Message<?>> myAuthorizationRules = AuthenticatedAuthorizationManager.authenticated();
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        AuthorizationChannelInterceptor authz = new AuthorizationChannelInterceptor(myAuthorizationRules);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        AuthorizationEventPublisher publisher = new SpringAuthorizationEventPublisher(this.context);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        authz.setAuthorizationEventPublisher(publisher);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        registration.interceptors(new SecurityContextChannelInterceptor(), authz);
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-06-18 21:32:35 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								Kotlin::
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								+
							 
						 
					
						
							
								
									
										
										
										
											2022-04-07 17:39:10 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								[source,kotlin,role="secondary"]
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								@Configuration
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								open class WebSocketSecurityConfig : WebSocketMessageBrokerConfigurer {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    @Override
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    override fun addArgumentResolvers(argumentResolvers: List<HandlerMethodArgumentResolver>) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        argumentResolvers.add(AuthenticationPrincipalArgumentResolver())
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    @Override
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    override fun configureClientInboundChannel(registration: ChannelRegistration) {
							 
						 
					
						
							
								
									
										
										
										
											2023-08-21 06:06:37 +09:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        var myAuthorizationRules: AuthorizationManager<Message<*>> = AuthenticatedAuthorizationManager.authenticated()
							 
						 
					
						
							
								
									
										
										
										
											2022-04-07 17:39:10 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        var authz: AuthorizationChannelInterceptor = AuthorizationChannelInterceptor(myAuthorizationRules)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        var publisher: AuthorizationEventPublisher = SpringAuthorizationEventPublisher(this.context)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        authz.setAuthorizationEventPublisher(publisher)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        registration.interceptors(SecurityContextChannelInterceptor(), authz)
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-06-18 21:32:35 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								Xml::
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								+
							 
						 
					
						
							
								
									
										
										
										
											2022-04-07 17:39:10 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								[source,xml,role="secondary"]
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								<websocket-message-broker use-authorization-manager="true" same-origin-disabled="true">
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    <intercept-message pattern="/**" access="authenticated"/>
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								</websocket-message-broker>
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
									
										
										
										
											2023-06-18 21:32:35 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								======
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-04-05 11:43:53 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								On the other hand, if you are using the <<legacy-websocket-configuration,legacy `AbstractSecurityWebSocketMessageBrokerConfigurer`>> and you want to allow other domains to access your site, you can disable Spring Security's protection.
							 
						 
					
						
							
								
									
										
										
										
											2022-04-07 17:39:10 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								For example, in Java Configuration you can use the following:
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-06-18 21:30:41 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								[tabs]
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								======
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								Java::
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								+
							 
						 
					
						
							
								
									
										
										
										
											2021-06-16 10:31:29 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								[source,java,role="primary"]
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								@Configuration
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								public class WebSocketSecurityConfig extends AbstractSecurityWebSocketMessageBrokerConfigurer {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    ...
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    @Override
							 
						 
					
						
							
								
									
										
										
										
											2015-03-10 14:48:19 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    protected boolean sameOriginDisabled() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return true;
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-06-18 21:30:41 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								Kotlin::
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								+
							 
						 
					
						
							
								
									
										
										
										
											2021-06-16 10:31:29 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								[source,kotlin,role="secondary"]
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								@Configuration
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								open class WebSocketSecurityConfig : AbstractSecurityWebSocketMessageBrokerConfigurer() {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    // ...
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    override fun sameOriginDisabled(): Boolean {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        return true
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
									
										
										
										
											2023-06-18 21:30:41 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								======
							 
						 
					
						
							
								
									
										
										
										
											2021-06-16 10:31:29 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-04-07 17:39:10 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								[[websocket-expression-handler]]
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								=== Custom Expression Handler
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								At times, there may be value in customizing how the `access` expressions are handled defined in your `intercept-message` XML elements.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								To do this, you can create a class of type `SecurityExpressionHandler<MessageAuthorizationContext<?>>` and refer to it in your XML definition like so:
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								[source,xml]
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								<websocket-message-broker use-authorization-manager="true">
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    <expression-handler ref="myRef"/>
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    ...
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								</websocket-message-broker>
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								<b:bean ref="myRef" class="org.springframework.security.messaging.access.expression.MessageAuthorizationContextSecurityExpressionHandler"/>
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								If you are migrating from a legacy usage of `websocket-message-broker` that implements a `SecurityExpressionHandler<Message<?>>`, you can:
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								 1. Additionally implement the `createEvaluationContext(Supplier, Message)` method and then
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								 2. Wrap that value in a `MessageAuthorizationContextSecurityExpressionHandler` like so:
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								[source,xml]
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								<websocket-message-broker use-authorization-manager="true">
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    <expression-handler ref="myRef"/>
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    ...
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								</websocket-message-broker>
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								<b:bean ref="myRef" class="org.springframework.security.messaging.access.expression.MessageAuthorizationContextSecurityExpressionHandler">
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    <b:constructor-arg>
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        <b:bean class="org.example.MyLegacyExpressionHandler"/>
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    </b:constructor-arg>
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								</b:bean>
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								[[websocket-sockjs]]
							 
						 
					
						
							
								
									
										
										
										
											2021-07-30 13:52:15 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								== Working with SockJS
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2019-03-19 23:53:23 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								https://docs.spring.io/spring/docs/current/spring-framework-reference/html/websocket.html#websocket-fallback[SockJS] provides fallback transports to support older browsers.
							 
						 
					
						
							
								
									
										
										
										
											2021-04-21 16:01:26 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								When using the fallback options, we need to relax a few security constraints to allow SockJS to work with Spring Security.
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								[[websocket-sockjs-sameorigin]]
							 
						 
					
						
							
								
									
										
										
										
											2021-07-30 13:52:15 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								=== SockJS & frame-options
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-04-21 16:01:26 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								SockJS may use a https://github.com/sockjs/sockjs-client/tree/v0.3.4[transport that leverages an iframe].
							 
						 
					
						
							
								
									
										
										
										
											2021-12-13 16:57:36 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								By default, Spring Security xref:features/exploits/headers.adoc#headers-frame-options[denies] the site from being framed to prevent clickjacking attacks.
							 
						 
					
						
							
								
									
										
										
										
											2021-04-21 16:01:26 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								To allow SockJS frame-based transports to work, we need to configure Spring Security to let the same origin frame the content.
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-12-13 16:57:36 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								You can customize `X-Frame-Options` with the xref:servlet/appendix/namespace/http.adoc#nsa-frame-options[frame-options] element.
							 
						 
					
						
							
								
									
										
										
										
											2021-04-21 16:01:26 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								For example, the following instructs Spring Security to use `X-Frame-Options: SAMEORIGIN`, which allows iframes within the same domain:
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								[source,xml]
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								<http>
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    <!-- ... -->
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    <headers>
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        <frame-options
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								          policy="SAMEORIGIN" />
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    </headers>
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								</http>
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-04-21 16:01:26 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								Similarly, you can customize frame options to use the same origin within Java Configuration by using the following:
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-06-18 21:30:41 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								[tabs]
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								======
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								Java::
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								+
							 
						 
					
						
							
								
									
										
										
										
											2021-06-16 10:31:29 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								[source,java,role="primary"]
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
									
										
										
										
											2022-07-30 03:47:02 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								@Configuration
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								@EnableWebSecurity
							 
						 
					
						
							
								
									
										
										
										
											2022-02-08 16:12:10 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								public class WebSecurityConfig {
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-02-08 16:12:10 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    @Bean
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    public SecurityFilterChain filterChain(HttpSecurity http) throws Exception {
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 13:10:36 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        http
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            // ...
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            .headers(headers -> headers
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                .frameOptions(frameOptions -> frameOptions
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                     .sameOrigin()
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                )
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        );
							 
						 
					
						
							
								
									
										
										
										
											2022-02-08 16:12:10 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        return http.build();
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 13:10:36 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-06-18 21:30:41 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								Kotlin::
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								+
							 
						 
					
						
							
								
									
										
										
										
											2021-06-16 10:31:29 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								[source,kotlin,role="secondary"]
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
									
										
										
										
											2022-07-30 03:47:02 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								@Configuration
							 
						 
					
						
							
								
									
										
										
										
											2021-06-16 10:31:29 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								@EnableWebSecurity
							 
						 
					
						
							
								
									
										
										
										
											2022-02-08 16:12:10 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								open class WebSecurityConfig {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    @Bean
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    open fun filterChain(http: HttpSecurity): SecurityFilterChain {
							 
						 
					
						
							
								
									
										
										
										
											2021-06-16 10:31:29 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        http {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            // ...
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            headers {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                frameOptions {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                    sameOrigin = true
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
									
										
										
										
											2022-02-08 16:12:10 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        return http.build()
							 
						 
					
						
							
								
									
										
										
										
											2021-06-16 10:31:29 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
									
										
										
										
											2023-06-18 21:30:41 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								======
							 
						 
					
						
							
								
									
										
										
										
											2021-06-16 10:31:29 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								[[websocket-sockjs-csrf]]
							 
						 
					
						
							
								
									
										
										
										
											2021-07-30 13:52:15 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								=== SockJS & Relaxing CSRF
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-04-21 16:01:26 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								SockJS uses a POST on the CONNECT messages for any HTTP-based transport.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								Typically, we need to include the CSRF token in an HTTP header or an HTTP parameter.
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								However, SockJS does not allow for these options.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								Instead, we must include the token in the Stomp headers as described in <<websocket-sameorigin-csrf>>.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-04-21 16:01:26 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								It also means that we need to relax our CSRF protection with the web layer.
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								Specifically, we want to disable CSRF protection for our connect URLs.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								We do NOT want to disable CSRF protection for every URL.
							 
						 
					
						
							
								
									
										
										
										
											2021-04-21 16:01:26 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								Otherwise, our site is vulnerable to CSRF attacks.
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-04-21 16:01:26 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								We can easily achieve this by providing a CSRF `RequestMatcher`.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								Our Java configuration makes this easy.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								For example, if our stomp endpoint is `/chat`, we can disable CSRF protection only for URLs that start with `/chat/` by using the following configuration:
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-06-18 21:30:41 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								[tabs]
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								======
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								Java::
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								+
							 
						 
					
						
							
								
									
										
										
										
											2021-06-16 10:31:29 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								[source,java,role="primary"]
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								@Configuration
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								@EnableWebSecurity
							 
						 
					
						
							
								
									
										
										
										
											2022-02-08 16:12:10 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								public class WebSecurityConfig {
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2022-02-08 16:12:10 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    @Bean
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    public SecurityFilterChain filterChain(HttpSecurity http) throws Exception {
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								        http
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 13:10:36 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								            .csrf(csrf -> csrf
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                // ignore our stomp endpoints since they are protected using Stomp headers
							 
						 
					
						
							
								
									
										
										
										
											2022-09-21 10:09:35 -03:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								                .ignoringRequestMatchers("/chat/**")
							 
						 
					
						
							
								
									
										
										
										
											2019-07-12 13:58:17 -04:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								            )
							 
						 
					
						
							
								
									
										
										
										
											2020-01-10 13:10:36 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								            .headers(headers -> headers
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                // allow same origin to frame our site to support iframe SockJS
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                .frameOptions(frameOptions -> frameOptions
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                    .sameOrigin()
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                )
							 
						 
					
						
							
								
									
										
										
										
											2019-07-12 13:58:17 -04:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								            )
							 
						 
					
						
							
								
									
										
										
										
											2021-11-10 15:15:11 -07:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								            .authorizeHttpRequests(authorize -> authorize
							 
						 
					
						
							
								
									
										
										
										
											2019-07-12 13:58:17 -04:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								                ...
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            )
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								            ...
							 
						 
					
						
							
								
									
										
										
										
											2021-12-13 16:57:36 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-06-18 21:30:41 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								Kotlin::
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								+
							 
						 
					
						
							
								
									
										
										
										
											2021-06-16 10:31:29 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								[source,kotlin,role="secondary"]
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								@Configuration
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								@EnableWebSecurity
							 
						 
					
						
							
								
									
										
										
										
											2022-02-08 16:12:10 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								open class WebSecurityConfig {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    @Bean
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    open fun filterChain(http: HttpSecurity): SecurityFilterChain {
							 
						 
					
						
							
								
									
										
										
										
											2021-06-16 10:31:29 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        http {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            csrf {
							 
						 
					
						
							
								
									
										
										
										
											2022-09-21 10:09:35 -03:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								                ignoringRequestMatchers("/chat/**")
							 
						 
					
						
							
								
									
										
										
										
											2021-06-16 10:31:29 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								            }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            headers {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                frameOptions {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                    sameOrigin = true
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            authorizeRequests {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                // ...
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            // ...
							 
						 
					
						
							
								
									
										
										
										
											2021-04-21 16:01:26 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								        }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
									
										
										
										
											2021-12-13 16:57:36 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								}
							 
						 
					
						
							
								
									
										
										
										
											2021-06-16 10:31:29 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
									
										
										
										
											2023-06-18 21:30:41 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								======
							 
						 
					
						
							
								
									
										
										
										
											2021-06-16 10:31:29 +02:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2021-12-13 16:57:36 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								If we use XML-based configuration, we can use thexref:servlet/appendix/namespace/http.adoc#nsa-csrf-request-matcher-ref[csrf@request-matcher-ref].
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								[source,xml]
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								<http ...>
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    <csrf request-matcher-ref="csrfMatcher"/>
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    <headers>
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        <frame-options policy="SAMEORIGIN"/>
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    </headers>
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    ...
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								</http>
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								<b:bean id="csrfMatcher"
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    class="AndRequestMatcher">
							 
						 
					
						
							
								
									
										
										
										
											2015-10-21 16:22:37 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								    <b:constructor-arg value="#{T(org.springframework.security.web.csrf.CsrfFilter).DEFAULT_CSRF_MATCHER}"/>
							 
						 
					
						
							
								
									
										
										
										
											2015-01-27 11:46:24 -06:00 
										
									 
								 
							 
							
								
							 
							
								 
							
							
								    <b:constructor-arg>
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        <b:bean class="org.springframework.security.web.util.matcher.NegatedRequestMatcher">
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								          <b:bean class="org.springframework.security.web.util.matcher.AntPathRequestMatcher">
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								            <b:constructor-arg value="/chat/**"/>
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								          </b:bean>
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        </b:bean>
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    </b:constructor-arg>
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								</b:bean>
							 
						 
					
						
							
								
									
										
										
										
											2016-11-10 15:52:54 +01:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
									
										
										
										
											2022-04-07 17:39:10 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								[[legacy-websocket-configuration]]
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								== Legacy WebSocket Configuration
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								Before Spring Security 5.8, the way to configure messaging authorization using Java Configuration, was to extend the `AbstractSecurityWebSocketMessageBrokerConfigurer` and configure the `MessageSecurityMetadataSourceRegistry`.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								For example:
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-06-18 21:32:35 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								[tabs]
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								======
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								Java::
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								+
							 
						 
					
						
							
								
									
										
										
										
											2022-04-07 17:39:10 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								[source,java,role="primary"]
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								@Configuration
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								public class WebSocketSecurityConfig
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								      extends AbstractSecurityWebSocketMessageBrokerConfigurer { // <1> <2>
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    protected void configureInbound(MessageSecurityMetadataSourceRegistry messages) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        messages
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								                .simpDestMatchers("/user/**").authenticated() // <3>
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
									
										
										
										
											2023-06-18 21:32:35 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								Kotlin::
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								+
							 
						 
					
						
							
								
									
										
										
										
											2022-04-07 17:39:10 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								[source,kotlin,role="secondary"]
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								@Configuration
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								open class WebSocketSecurityConfig : AbstractSecurityWebSocketMessageBrokerConfigurer() { // <1> <2>
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    override fun configureInbound(messages: MessageSecurityMetadataSourceRegistry) {
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								        messages.simpDestMatchers("/user/**").authenticated() // <3>
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								    }
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								}
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								----
							 
						 
					
						
							
								
									
										
										
										
											2023-06-18 21:32:35 -05:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								======
							 
						 
					
						
							
								
									
										
										
										
											2022-04-07 17:39:10 -06:00 
										
									 
								 
							 
							
								
									
										 
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								This will ensure that:
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								<1> Any inbound CONNECT message requires a valid CSRF token to enforce <<websocket-sameorigin,Same Origin Policy>>
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								<2> The SecurityContextHolder is populated with the user within the simpUser header attribute for any inbound request.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								<3> Our messages require the proper authorization. Specifically, any inbound message that starts with "/user/" will require ROLE_USER. Additional details on authorization can be found in <<websocket-authorization>>
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								Using the legacy configuration is helpful in the event that you have a custom `SecurityExpressionHandler` that extends `AbstractSecurityExpressionHandler` and overrides `createEvaluationContextInternal` or `createSecurityExpressionRoot`.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								In order to defer `Authorization` lookup, the new `AuthorizationManager` API does not invoke these when evaluating expressions.
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								
							 
						 
					
						
							
								
							 
							
								
							 
							
								 
							
							
								If you are using XML, you can use the legacy APIs simply by not using the `use-authorization-manager` element or setting it to `false`.