Avoid creating a new EmbeddedServletContainerFactory for websockets
User can now also switch off and customize the websockets customizer by adding a bean named "websocketContainerCustomizer". Fixes gh-479
This commit is contained in:
parent
34efda1890
commit
14d52b6c18
|
|
@ -18,7 +18,6 @@ package org.springframework.boot.autoconfigure.aop;
|
||||||
|
|
||||||
import org.aspectj.lang.annotation.Aspect;
|
import org.aspectj.lang.annotation.Aspect;
|
||||||
import org.aspectj.lang.reflect.Advice;
|
import org.aspectj.lang.reflect.Advice;
|
||||||
import org.springframework.boot.autoconfigure.EnableAutoConfiguration;
|
|
||||||
import org.springframework.boot.autoconfigure.condition.ConditionalOnClass;
|
import org.springframework.boot.autoconfigure.condition.ConditionalOnClass;
|
||||||
import org.springframework.boot.autoconfigure.condition.ConditionalOnExpression;
|
import org.springframework.boot.autoconfigure.condition.ConditionalOnExpression;
|
||||||
import org.springframework.context.annotation.Configuration;
|
import org.springframework.context.annotation.Configuration;
|
||||||
|
|
@ -26,11 +25,14 @@ import org.springframework.context.annotation.EnableAspectJAutoProxy;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
*
|
*
|
||||||
* <p> {@link org.springframework.boot.autoconfigure.EnableAutoConfiguration Auto-configuration}
|
* <p>
|
||||||
* for Spring's AOP support. Equivalent to enabling {@link org.springframework.context.annotation.EnableAspectJAutoProxy}
|
* {@link org.springframework.boot.autoconfigure.EnableAutoConfiguration
|
||||||
* in your configuration. The configuration will not be activated if {@literal spring.aop.auto=false}.
|
* Auto-configuration} for Spring's AOP support. Equivalent to enabling
|
||||||
* The {@literal proxyTargetClass} attribute will be {@literal false}, by default, but can be overridden by
|
* {@link org.springframework.context.annotation.EnableAspectJAutoProxy} in your
|
||||||
* specifying {@literal spring.aop.proxyTargetClass=true}.
|
* configuration. The configuration will not be activated if
|
||||||
|
* {@literal spring.aop.auto=false}. The {@literal proxyTargetClass} attribute will be
|
||||||
|
* {@literal false}, by default, but can be overridden by specifying
|
||||||
|
* {@literal spring.aop.proxyTargetClass=true}.
|
||||||
*
|
*
|
||||||
* @author Dave Syer
|
* @author Dave Syer
|
||||||
* @author Josh Long
|
* @author Josh Long
|
||||||
|
|
|
||||||
|
|
@ -53,8 +53,7 @@ public class ServerProperties implements EmbeddedServletContainerCustomizer {
|
||||||
|
|
||||||
private Integer sessionTimeout;
|
private Integer sessionTimeout;
|
||||||
|
|
||||||
@NotNull
|
private String contextPath;
|
||||||
private String contextPath = "";
|
|
||||||
|
|
||||||
@NotNull
|
@NotNull
|
||||||
private String servletPath = "/";
|
private String servletPath = "/";
|
||||||
|
|
|
||||||
|
|
@ -23,7 +23,11 @@ import org.apache.catalina.deploy.ApplicationListener;
|
||||||
import org.apache.catalina.startup.Tomcat;
|
import org.apache.catalina.startup.Tomcat;
|
||||||
import org.springframework.boot.autoconfigure.AutoConfigureBefore;
|
import org.springframework.boot.autoconfigure.AutoConfigureBefore;
|
||||||
import org.springframework.boot.autoconfigure.condition.ConditionalOnClass;
|
import org.springframework.boot.autoconfigure.condition.ConditionalOnClass;
|
||||||
|
import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean;
|
||||||
import org.springframework.boot.autoconfigure.web.EmbeddedServletContainerAutoConfiguration;
|
import org.springframework.boot.autoconfigure.web.EmbeddedServletContainerAutoConfiguration;
|
||||||
|
import org.springframework.boot.context.embedded.ConfigurableEmbeddedServletContainer;
|
||||||
|
import org.springframework.boot.context.embedded.EmbeddedServletContainerCustomizer;
|
||||||
|
import org.springframework.boot.context.embedded.tomcat.TomcatContextCustomizer;
|
||||||
import org.springframework.boot.context.embedded.tomcat.TomcatEmbeddedServletContainerFactory;
|
import org.springframework.boot.context.embedded.tomcat.TomcatEmbeddedServletContainerFactory;
|
||||||
import org.springframework.context.annotation.Bean;
|
import org.springframework.context.annotation.Bean;
|
||||||
import org.springframework.context.annotation.Configuration;
|
import org.springframework.context.annotation.Configuration;
|
||||||
|
|
@ -47,14 +51,31 @@ public class WebSocketAutoConfiguration {
|
||||||
"org.apache.tomcat.websocket.server.WsContextListener", false);
|
"org.apache.tomcat.websocket.server.WsContextListener", false);
|
||||||
|
|
||||||
@Bean
|
@Bean
|
||||||
public TomcatEmbeddedServletContainerFactory tomcatEmbeddedServletContainerFactory() {
|
@ConditionalOnMissingBean(name = "websocketContainerCustomizer")
|
||||||
TomcatEmbeddedServletContainerFactory factory = new TomcatEmbeddedServletContainerFactory() {
|
public EmbeddedServletContainerCustomizer websocketContainerCustomizer() {
|
||||||
|
|
||||||
|
EmbeddedServletContainerCustomizer customizer = new EmbeddedServletContainerCustomizer() {
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
protected void postProcessContext(Context context) {
|
public void customize(ConfigurableEmbeddedServletContainer container) {
|
||||||
|
if (!(container instanceof TomcatEmbeddedServletContainerFactory)) {
|
||||||
|
throw new IllegalStateException(
|
||||||
|
"Websockets are currently only supported in Tomcat (found "
|
||||||
|
+ container.getClass() + ")");
|
||||||
|
}
|
||||||
|
((TomcatEmbeddedServletContainerFactory) container)
|
||||||
|
.addContextCustomizers(new TomcatContextCustomizer() {
|
||||||
|
@Override
|
||||||
|
public void customize(Context context) {
|
||||||
context.addApplicationListener(WS_APPLICATION_LISTENER);
|
context.addApplicationListener(WS_APPLICATION_LISTENER);
|
||||||
}
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
};
|
};
|
||||||
return factory;
|
|
||||||
|
return customizer;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -29,6 +29,7 @@ import org.springframework.boot.context.embedded.ConfigurableEmbeddedServletCont
|
||||||
import static org.junit.Assert.assertEquals;
|
import static org.junit.Assert.assertEquals;
|
||||||
import static org.junit.Assert.assertFalse;
|
import static org.junit.Assert.assertFalse;
|
||||||
import static org.mockito.Mockito.mock;
|
import static org.mockito.Mockito.mock;
|
||||||
|
import static org.mockito.Mockito.times;
|
||||||
import static org.mockito.Mockito.verify;
|
import static org.mockito.Mockito.verify;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
@ -75,7 +76,7 @@ public class ServerPropertiesTests {
|
||||||
public void testCustomizeTomcat() throws Exception {
|
public void testCustomizeTomcat() throws Exception {
|
||||||
ConfigurableEmbeddedServletContainer factory = mock(ConfigurableEmbeddedServletContainer.class);
|
ConfigurableEmbeddedServletContainer factory = mock(ConfigurableEmbeddedServletContainer.class);
|
||||||
this.properties.customize(factory);
|
this.properties.customize(factory);
|
||||||
verify(factory).setContextPath("");
|
verify(factory, times(0)).setContextPath("");
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
|
|
|
||||||
|
|
@ -0,0 +1,115 @@
|
||||||
|
/*
|
||||||
|
* Copyright 2012-2013 the original author or authors.
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package samples.websocket.echo;
|
||||||
|
|
||||||
|
import static org.junit.Assert.assertEquals;
|
||||||
|
|
||||||
|
import java.util.concurrent.CountDownLatch;
|
||||||
|
import java.util.concurrent.TimeUnit;
|
||||||
|
|
||||||
|
import org.apache.commons.logging.Log;
|
||||||
|
import org.apache.commons.logging.LogFactory;
|
||||||
|
import org.junit.Test;
|
||||||
|
import org.junit.runner.RunWith;
|
||||||
|
import org.springframework.boot.CommandLineRunner;
|
||||||
|
import org.springframework.boot.SpringApplication;
|
||||||
|
import org.springframework.boot.context.embedded.EmbeddedServletContainerFactory;
|
||||||
|
import org.springframework.boot.context.embedded.tomcat.TomcatEmbeddedServletContainerFactory;
|
||||||
|
import org.springframework.boot.test.IntegrationTest;
|
||||||
|
import org.springframework.boot.test.SpringApplicationConfiguration;
|
||||||
|
import org.springframework.context.ConfigurableApplicationContext;
|
||||||
|
import org.springframework.context.annotation.Bean;
|
||||||
|
import org.springframework.context.annotation.Configuration;
|
||||||
|
import org.springframework.test.annotation.DirtiesContext;
|
||||||
|
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
|
||||||
|
import org.springframework.test.context.web.WebAppConfiguration;
|
||||||
|
import org.springframework.web.socket.client.WebSocketConnectionManager;
|
||||||
|
import org.springframework.web.socket.client.standard.StandardWebSocketClient;
|
||||||
|
|
||||||
|
import samples.websocket.client.GreetingService;
|
||||||
|
import samples.websocket.client.SimpleClientWebSocketHandler;
|
||||||
|
import samples.websocket.client.SimpleGreetingService;
|
||||||
|
import samples.websocket.config.SampleWebSocketsApplication;
|
||||||
|
import samples.websocket.echo.CustomContainerWebSocketsApplicationTests.CustomContainerConfiguration;
|
||||||
|
|
||||||
|
@RunWith(SpringJUnit4ClassRunner.class)
|
||||||
|
@SpringApplicationConfiguration(classes={SampleWebSocketsApplication.class, CustomContainerConfiguration.class })
|
||||||
|
@WebAppConfiguration
|
||||||
|
@IntegrationTest
|
||||||
|
@DirtiesContext
|
||||||
|
public class CustomContainerWebSocketsApplicationTests {
|
||||||
|
|
||||||
|
private static Log logger = LogFactory.getLog(CustomContainerWebSocketsApplicationTests.class);
|
||||||
|
|
||||||
|
private static final String WS_URI = "ws://localhost:9010/ws/echo/websocket";
|
||||||
|
|
||||||
|
@Configuration
|
||||||
|
protected static class CustomContainerConfiguration {
|
||||||
|
@Bean
|
||||||
|
public EmbeddedServletContainerFactory embeddedServletContainerFactory() {
|
||||||
|
return new TomcatEmbeddedServletContainerFactory("/ws", 9010);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void runAndWait() throws Exception {
|
||||||
|
ConfigurableApplicationContext context = SpringApplication.run(
|
||||||
|
ClientConfiguration.class, "--spring.main.web_environment=false");
|
||||||
|
long count = context.getBean(ClientConfiguration.class).latch.getCount();
|
||||||
|
context.close();
|
||||||
|
assertEquals(0, count);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Configuration
|
||||||
|
static class ClientConfiguration implements CommandLineRunner {
|
||||||
|
|
||||||
|
private final CountDownLatch latch = new CountDownLatch(1);
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void run(String... args) throws Exception {
|
||||||
|
logger.info("Waiting for response: latch=" + this.latch.getCount());
|
||||||
|
this.latch.await(10, TimeUnit.SECONDS);
|
||||||
|
logger.info("Got response: latch=" + this.latch.getCount());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Bean
|
||||||
|
public WebSocketConnectionManager wsConnectionManager() {
|
||||||
|
|
||||||
|
WebSocketConnectionManager manager = new WebSocketConnectionManager(client(),
|
||||||
|
handler(), WS_URI);
|
||||||
|
manager.setAutoStartup(true);
|
||||||
|
|
||||||
|
return manager;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Bean
|
||||||
|
public StandardWebSocketClient client() {
|
||||||
|
return new StandardWebSocketClient();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Bean
|
||||||
|
public SimpleClientWebSocketHandler handler() {
|
||||||
|
return new SimpleClientWebSocketHandler(greetingService(), this.latch);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Bean
|
||||||
|
public GreetingService greetingService() {
|
||||||
|
return new SimpleGreetingService();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
Loading…
Reference in New Issue