Spring Websocket url命中RequestMapping但不是MessageMapping

时间:2016-06-01 22:32:26

标签: stomp spring-websocket spring-messaging

我在现有的Web应用程序中设置websocket配置时遇到了问题。

@Configuration
@EnableWebSocketMessageBroker
public class WebsocketConfig extends AbstractWebSocketMessageBrokerConfigurer{

    @Override
    public void configureMessageBroker(MessageBrokerRegistry config) {
        config.enableSimpleBroker("/mobile");
        config.setApplicationDestinationPrefixes("/mobile-server");
        config.setUserDestinationPrefix("/mobile-user");
    }

    @Override
    public void registerStompEndpoints(StompEndpointRegistry registry) {
        registry.addEndpoint("/mobile-socket")
        .withSockJS()
        .setInterceptors(new HttpSessionHandshakeInterceptor());
    }
}

控制器

@Controller
public class WebSocketInboxController{

    @MessageMapping("/inbox")
    @SendToUser("/mobile")
    public Map<String,Object> inbox(
    ){

        Map<String,Object> res = new HashMap<>();

        res.put("hello", "hello");

        return res;
    }

客户端

const webstomp = require('webstomp-client');

const socket = webstomp.client('ws://www.dev.server.com/mobile-socket',{
  debug:true
});

socket.connect('marc@gmail.com', '123456', (client) => {
  console.log('connected');

  socket.send("/mobile-server/inbox",)
  socket.subscribe("/mobile/inbox");
}, (client, err) => {
  console.log(err);
});

我在客户端尝试连接时看到的是Spring尝试将/ mobile-socket与现有Web应用程序的RequestMappings进行匹配,最后通过@RequestMapping(“/ {somevar}”找到匹配它的那个)。

我是WebSockets的新手,但我希望端点注册能成为这类连接的主要内容吗?

即使删除了错误的RequestMapping被击中,我也似乎无法触及MessageMapping。我在日志中看到了这个

AntPathRequestMatcher.matches(150) | Request '/mobile-socket' matched by universal pattern '/**'
[MSA] DEBUG [2016-06-03T11:16:21,025] FilterSecurityInterceptor.beforeInvocation(219) | Secure object: FilterInvocation: URL: /mobile-socket; Attributes: [permitAll]
[MSA] DEBUG [2016-06-03T11:16:21,025] FilterSecurityInterceptor.authenticateIfRequired(348) | Previously Authenticated: org.springframework.security.authentication.AnonymousAuthenticationToken@9055e4a6: Principal: anonymousUser; Credentials: [PROTECTED]; Authenticated: true; Details: org.springframework.security.web.authentication.WebAuthenticationDetails@957e: RemoteIpAddress: 127.0.0.1; SessionId: null; Granted Authorities: ROLE_ANONYMOUS
[MSA] DEBUG [2016-06-03T11:16:21,025] AffirmativeBased.decide(66) | Voter: org.springframework.security.web.access.expression.WebExpressionVoter@444af45, returned: 1
[MSA] DEBUG [2016-06-03T11:16:21,025] FilterSecurityInterceptor.beforeInvocation(243) | Authorization successful
[MSA] DEBUG [2016-06-03T11:16:21,026] FilterSecurityInterceptor.beforeInvocation(256) | RunAsManager did not change Authentication object
[MSA] DEBUG [2016-06-03T11:16:21,026] FilterChainProxy.doFilter(325) | /mobile-socket at position 16 of 16 in additional filter chain; firing Filter: 'FilterSecurityInterceptor'
[MSA] DEBUG [2016-06-03T11:16:21,026] FilterChainProxy.doFilter(310) | /mobile-socket reached end of additional filter chain; proceeding with original chain
[MSA] DEBUG [2016-06-03T11:16:21,027] ExceptionTranslationFilter.doFilter(117) | Chain processed normally
[MSA] DEBUG [2016-06-03T11:16:21,027] HstsHeaderWriter.writeHeaders(130) | Not injecting HSTS header since it did not match the requestMatcher org.springframework.security.web.header.writers.HstsHeaderWriter$SecureRequestMatcher@53cc2afb
[MSA] DEBUG [2016-06-03T11:16:21,027] HttpSessionSecurityContextRepository.saveContext(352) | SecurityContext is empty or contents are anonymous - context will not be stored in HttpSession.
[MSA] DEBUG [2016-06-03T11:16:21,028] SecurityContextPersistenceFilter.doFilter(120) | SecurityContextHolder now cleared, as request processing completed

1 个答案:

答案 0 :(得分:1)

Spring尝试将“/ mobile-socket”与RequestMappings进行匹配,因为所有请求都转到Web应用程序上下文中的HandlerMapping bean,以将传入的Web请求映射到适当的处理程序。随着带注释的控制器的引入,RequestMappingHandlerMapping自动在所有@Controller bean上查找@RequestMapping注释,包括具有@MessageMapping的控制器。

由于@MessageMapping只能在@Controller注释下定义,因此Spring会尝试匹配其他RequestMappings。

一种可能的解决方案是引入拦截器来处理websocket请求URL以专门映射到特定控制器。你可以尝试一下!