Polish `AbstractServletWebServerFactory`

Extract some of the utility methods to package private classes
pull/8496/merge
Phillip Webb 8 years ago
parent b1c689b9b2
commit 9d61882bb1

@ -17,13 +17,8 @@
package org.springframework.boot.web.servlet.server; package org.springframework.boot.web.servlet.server;
import java.io.File; import java.io.File;
import java.io.IOException;
import java.net.JarURLConnection;
import java.net.URL; import java.net.URL;
import java.net.URLClassLoader;
import java.net.URLConnection;
import java.nio.charset.Charset; import java.nio.charset.Charset;
import java.security.CodeSource;
import java.util.ArrayList; import java.util.ArrayList;
import java.util.Arrays; import java.util.Arrays;
import java.util.HashMap; import java.util.HashMap;
@ -31,13 +26,10 @@ import java.util.List;
import java.util.Locale; import java.util.Locale;
import java.util.Map; import java.util.Map;
import java.util.concurrent.TimeUnit; import java.util.concurrent.TimeUnit;
import java.util.jar.JarFile;
import org.apache.commons.logging.Log; import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory; import org.apache.commons.logging.LogFactory;
import org.springframework.boot.ApplicationHome;
import org.springframework.boot.ApplicationTemp;
import org.springframework.boot.web.server.AbstractConfigurableWebServerFactory; import org.springframework.boot.web.server.AbstractConfigurableWebServerFactory;
import org.springframework.boot.web.server.MimeMappings; import org.springframework.boot.web.server.MimeMappings;
import org.springframework.boot.web.servlet.ServletContextInitializer; import org.springframework.boot.web.servlet.ServletContextInitializer;
@ -63,9 +55,6 @@ public abstract class AbstractServletWebServerFactory
private static final int DEFAULT_SESSION_TIMEOUT = (int) TimeUnit.MINUTES private static final int DEFAULT_SESSION_TIMEOUT = (int) TimeUnit.MINUTES
.toSeconds(30); .toSeconds(30);
private static final String[] COMMON_DOC_ROOTS = { "src/main/webapp", "public",
"static" };
protected final Log logger = LogFactory.getLog(getClass()); protected final Log logger = LogFactory.getLog(getClass());
private String contextPath = ""; private String contextPath = "";
@ -76,20 +65,22 @@ public abstract class AbstractServletWebServerFactory
private boolean persistSession; private boolean persistSession;
private File sessionStoreDir;
private boolean registerDefaultServlet = true; private boolean registerDefaultServlet = true;
private MimeMappings mimeMappings = new MimeMappings(MimeMappings.DEFAULT); private MimeMappings mimeMappings = new MimeMappings(MimeMappings.DEFAULT);
private File documentRoot;
private List<ServletContextInitializer> initializers = new ArrayList<>(); private List<ServletContextInitializer> initializers = new ArrayList<>();
private Jsp jsp = new Jsp(); private Jsp jsp = new Jsp();
private Map<Locale, Charset> localeCharsetMappings = new HashMap<>(); private Map<Locale, Charset> localeCharsetMappings = new HashMap<>();
private final SessionStoreDirectory sessionStoreDir = new SessionStoreDirectory();
private final DocumentRoot documentRoot = new DocumentRoot(this.logger);
private final StaticResourceJars staticResourceJars = new StaticResourceJars();
/** /**
* Create a new {@link AbstractServletWebServerFactory} instance. * Create a new {@link AbstractServletWebServerFactory} instance.
*/ */
@ -184,12 +175,12 @@ public abstract class AbstractServletWebServerFactory
} }
public File getSessionStoreDir() { public File getSessionStoreDir() {
return this.sessionStoreDir; return this.sessionStoreDir.getDirectory();
} }
@Override @Override
public void setSessionStoreDir(File sessionStoreDir) { public void setSessionStoreDir(File sessionStoreDir) {
this.sessionStoreDir = sessionStoreDir; this.sessionStoreDir.setDirectory(sessionStoreDir);
} }
/** /**
@ -224,12 +215,12 @@ public abstract class AbstractServletWebServerFactory
* @return the document root * @return the document root
*/ */
public File getDocumentRoot() { public File getDocumentRoot() {
return this.documentRoot; return this.documentRoot.getDirectory();
} }
@Override @Override
public void setDocumentRoot(File documentRoot) { public void setDocumentRoot(File documentRoot) {
this.documentRoot = documentRoot; this.documentRoot.setDirectory(documentRoot);
} }
@Override @Override
@ -298,171 +289,19 @@ public abstract class AbstractServletWebServerFactory
* @return the valid document root * @return the valid document root
*/ */
protected final File getValidDocumentRoot() { protected final File getValidDocumentRoot() {
File file = getDocumentRoot(); return this.documentRoot.getValidDirectory();
// If document root not explicitly set see if we are running from a war archive
file = file != null ? file : getWarFileDocumentRoot();
// If not a war archive maybe it is an exploded war
file = file != null ? file : getExplodedWarFileDocumentRoot();
// Or maybe there is a document root in a well-known location
file = file != null ? file : getCommonDocumentRoot();
if (file == null && this.logger.isDebugEnabled()) {
this.logger
.debug("None of the document roots " + Arrays.asList(COMMON_DOC_ROOTS)
+ " point to a directory and will be ignored.");
}
else if (this.logger.isDebugEnabled()) {
this.logger.debug("Document root: " + file);
}
return file;
}
private File getExplodedWarFileDocumentRoot() {
return getExplodedWarFileDocumentRoot(getCodeSourceArchive());
}
protected List<URL> getUrlsOfJarsWithMetaInfResources() {
ClassLoader classLoader = getClass().getClassLoader();
List<URL> staticResourceUrls = new ArrayList<>();
if (classLoader instanceof URLClassLoader) {
for (URL url : ((URLClassLoader) classLoader).getURLs()) {
try {
if ("file".equals(url.getProtocol())) {
File file = new File(url.getFile());
if (file.isDirectory()
&& new File(file, "META-INF/resources").isDirectory()) {
staticResourceUrls.add(url);
}
else if (isResourcesJar(file)) {
staticResourceUrls.add(url);
}
}
else {
URLConnection connection = url.openConnection();
if (connection instanceof JarURLConnection) {
if (isResourcesJar((JarURLConnection) connection)) {
staticResourceUrls.add(url);
}
}
}
}
catch (IOException ex) {
throw new IllegalStateException(ex);
}
}
}
return staticResourceUrls;
} }
private boolean isResourcesJar(JarURLConnection connection) { protected final List<URL> getUrlsOfJarsWithMetaInfResources() {
try { return this.staticResourceJars.getUrls();
return isResourcesJar(connection.getJarFile());
}
catch (IOException ex) {
return false;
}
}
private boolean isResourcesJar(File file) {
try {
return isResourcesJar(new JarFile(file));
}
catch (IOException ex) {
return false;
}
}
private boolean isResourcesJar(JarFile jar) throws IOException {
try {
return jar.getName().endsWith(".jar")
&& (jar.getJarEntry("META-INF/resources") != null);
}
finally {
jar.close();
}
}
protected final File getExplodedWarFileDocumentRoot(File codeSourceFile) {
if (this.logger.isDebugEnabled()) {
this.logger.debug("Code archive: " + codeSourceFile);
}
if (codeSourceFile != null && codeSourceFile.exists()) {
String path = codeSourceFile.getAbsolutePath();
int webInfPathIndex = path
.indexOf(File.separatorChar + "WEB-INF" + File.separatorChar);
if (webInfPathIndex >= 0) {
path = path.substring(0, webInfPathIndex);
return new File(path);
}
}
return null;
}
private File getWarFileDocumentRoot() {
return getArchiveFileDocumentRoot(".war");
}
private File getArchiveFileDocumentRoot(String extension) {
File file = getCodeSourceArchive();
if (this.logger.isDebugEnabled()) {
this.logger.debug("Code archive: " + file);
}
if (file != null && file.exists() && !file.isDirectory()
&& file.getName().toLowerCase().endsWith(extension)) {
return file.getAbsoluteFile();
}
return null;
}
private File getCommonDocumentRoot() {
for (String commonDocRoot : COMMON_DOC_ROOTS) {
File root = new File(commonDocRoot);
if (root.exists() && root.isDirectory()) {
return root.getAbsoluteFile();
}
}
return null;
}
private File getCodeSourceArchive() {
try {
CodeSource codeSource = getClass().getProtectionDomain().getCodeSource();
URL location = (codeSource == null ? null : codeSource.getLocation());
if (location == null) {
return null;
}
String path = location.getPath();
URLConnection connection = location.openConnection();
if (connection instanceof JarURLConnection) {
path = ((JarURLConnection) connection).getJarFile().getName();
}
if (path.indexOf("!/") != -1) {
path = path.substring(0, path.indexOf("!/"));
}
return new File(path);
}
catch (IOException ex) {
return null;
}
} }
protected final File getValidSessionStoreDir() { protected final File getValidSessionStoreDir() {
return getValidSessionStoreDir(true); return this.sessionStoreDir.getValidDirectory(true);
} }
protected final File getValidSessionStoreDir(boolean mkdirs) { protected final File getValidSessionStoreDir(boolean mkdirs) {
File dir = getSessionStoreDir(); return this.sessionStoreDir.getValidDirectory(mkdirs);
if (dir == null) {
return new ApplicationTemp().getDir("servlet-sessions");
}
if (!dir.isAbsolute()) {
dir = new File(new ApplicationHome().getDir(), dir.getPath());
}
if (!dir.exists() && mkdirs) {
dir.mkdirs();
}
Assert.state(!mkdirs || dir.exists(), "Session dir " + dir + " does not exist");
Assert.state(!dir.isFile(), "Session dir " + dir + " points to a file");
return dir;
} }
} }

@ -0,0 +1,148 @@
/*
* Copyright 2012-2017 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 org.springframework.boot.web.servlet.server;
import java.io.File;
import java.io.IOException;
import java.net.JarURLConnection;
import java.net.URL;
import java.net.URLConnection;
import java.security.CodeSource;
import java.util.Arrays;
import org.apache.commons.logging.Log;
/**
* Manages a {@link ServletWebServerFactory} document root.
*
* @author Phillip Webb
* @see AbstractServletWebServerFactory
*/
class DocumentRoot {
private static final String[] COMMON_DOC_ROOTS = { "src/main/webapp", "public",
"static" };
private final Log logger;
private File directory;
DocumentRoot(Log logger) {
this.logger = logger;
}
public File getDirectory() {
return this.directory;
}
public void setDirectory(File directory) {
this.directory = directory;
}
/**
* Returns the absolute document root when it points to a valid directory, logging a
* warning and returning {@code null} otherwise.
* @return the valid document root
*/
public final File getValidDirectory() {
File file = this.directory;
file = (file != null ? file : getWarFileDocumentRoot());
file = (file != null ? file : getExplodedWarFileDocumentRoot());
file = (file != null ? file : getCommonDocumentRoot());
if (file == null && this.logger.isDebugEnabled()) {
logNoDocumentRoots();
}
else if (this.logger.isDebugEnabled()) {
this.logger.debug("Document root: " + file);
}
return file;
}
private File getWarFileDocumentRoot() {
return getArchiveFileDocumentRoot(".war");
}
private File getArchiveFileDocumentRoot(String extension) {
File file = getCodeSourceArchive();
if (this.logger.isDebugEnabled()) {
this.logger.debug("Code archive: " + file);
}
if (file != null && file.exists() && !file.isDirectory()
&& file.getName().toLowerCase().endsWith(extension)) {
return file.getAbsoluteFile();
}
return null;
}
private File getExplodedWarFileDocumentRoot() {
return getExplodedWarFileDocumentRoot(getCodeSourceArchive());
}
private File getCodeSourceArchive() {
try {
CodeSource codeSource = getClass().getProtectionDomain().getCodeSource();
URL location = (codeSource == null ? null : codeSource.getLocation());
if (location == null) {
return null;
}
String path = location.getPath();
URLConnection connection = location.openConnection();
if (connection instanceof JarURLConnection) {
path = ((JarURLConnection) connection).getJarFile().getName();
}
if (path.indexOf("!/") != -1) {
path = path.substring(0, path.indexOf("!/"));
}
return new File(path);
}
catch (IOException ex) {
return null;
}
}
public final File getExplodedWarFileDocumentRoot(File codeSourceFile) {
if (this.logger.isDebugEnabled()) {
this.logger.debug("Code archive: " + codeSourceFile);
}
if (codeSourceFile != null && codeSourceFile.exists()) {
String path = codeSourceFile.getAbsolutePath();
int webInfPathIndex = path
.indexOf(File.separatorChar + "WEB-INF" + File.separatorChar);
if (webInfPathIndex >= 0) {
path = path.substring(0, webInfPathIndex);
return new File(path);
}
}
return null;
}
private File getCommonDocumentRoot() {
for (String commonDocRoot : COMMON_DOC_ROOTS) {
File root = new File(commonDocRoot);
if (root.exists() && root.isDirectory()) {
return root.getAbsoluteFile();
}
}
return null;
}
private void logNoDocumentRoots() {
this.logger.debug("None of the document roots " + Arrays.asList(COMMON_DOC_ROOTS)
+ " point to a directory and will be ignored.");
}
}

@ -0,0 +1,59 @@
/*
* Copyright 2012-2017 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 org.springframework.boot.web.servlet.server;
import java.io.File;
import org.springframework.boot.ApplicationHome;
import org.springframework.boot.ApplicationTemp;
import org.springframework.util.Assert;
/**
* Manages a session store directory.
*
* @author Phillip Webb
* @see AbstractServletWebServerFactory
*/
class SessionStoreDirectory {
private File directory;
public File getDirectory() {
return this.directory;
}
public void setDirectory(File directory) {
this.directory = directory;
}
public File getValidDirectory(boolean mkdirs) {
File dir = getDirectory();
if (dir == null) {
return new ApplicationTemp().getDir("servlet-sessions");
}
if (!dir.isAbsolute()) {
dir = new File(new ApplicationHome().getDir(), dir.getPath());
}
if (!dir.exists() && mkdirs) {
dir.mkdirs();
}
Assert.state(!mkdirs || dir.exists(), "Session dir " + dir + " does not exist");
Assert.state(!dir.isFile(), "Session dir " + dir + " points to a file");
return dir;
}
}

@ -0,0 +1,108 @@
/*
* Copyright 2012-2017 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 org.springframework.boot.web.servlet.server;
import java.io.File;
import java.io.IOException;
import java.net.JarURLConnection;
import java.net.URL;
import java.net.URLClassLoader;
import java.net.URLConnection;
import java.util.ArrayList;
import java.util.List;
import java.util.jar.JarFile;
/**
* Logic to extract URLs of static resource jars (those containing
* {@code "META-INF/resources"} directories).
*
* @author Andy Wilkinson
* @author Phillip Webb
*/
class StaticResourceJars {
public final List<URL> getUrls() {
ClassLoader classLoader = getClass().getClassLoader();
List<URL> urls = new ArrayList<>();
if (classLoader instanceof URLClassLoader) {
for (URL url : ((URLClassLoader) classLoader).getURLs()) {
addUrl(urls, url);
}
}
return urls;
}
private void addUrl(List<URL> urls, URL url) {
try {
if ("file".equals(url.getProtocol())) {
addUrlFile(urls, url, new File(url.getFile()));
}
else {
addUrlConnection(urls, url, url.openConnection());
}
}
catch (IOException ex) {
throw new IllegalStateException(ex);
}
}
private void addUrlFile(List<URL> urls, URL url, File file) {
if (file.isDirectory() && new File(file, "META-INF/resources").isDirectory()) {
urls.add(url);
}
else if (isResourcesJar(file)) {
urls.add(url);
}
}
private void addUrlConnection(List<URL> urls, URL url, URLConnection connection) {
if (connection instanceof JarURLConnection) {
if (isResourcesJar((JarURLConnection) connection)) {
urls.add(url);
}
}
}
private boolean isResourcesJar(JarURLConnection connection) {
try {
return isResourcesJar(connection.getJarFile());
}
catch (IOException ex) {
return false;
}
}
private boolean isResourcesJar(File file) {
try {
return isResourcesJar(new JarFile(file));
}
catch (IOException ex) {
return false;
}
}
private boolean isResourcesJar(JarFile jar) throws IOException {
try {
return jar.getName().endsWith(".jar")
&& (jar.getJarEntry("META-INF/resources") != null);
}
finally {
jar.close();
}
}
}

@ -161,8 +161,7 @@ public abstract class AbstractServletWebServerFactoryTests {
@Test @Test
public void startServlet() throws Exception { public void startServlet() throws Exception {
AbstractServletWebServerFactory factory = getFactory(); AbstractServletWebServerFactory factory = getFactory();
this.webServer = factory this.webServer = factory.getWebServer(exampleServletRegistration());
.getWebServer(exampleServletRegistration());
this.webServer.start(); this.webServer.start();
assertThat(getResponse(getLocalUrl("/hello"))).isEqualTo("Hello World"); assertThat(getResponse(getLocalUrl("/hello"))).isEqualTo("Hello World");
} }
@ -170,8 +169,7 @@ public abstract class AbstractServletWebServerFactoryTests {
@Test @Test
public void startCalledTwice() throws Exception { public void startCalledTwice() throws Exception {
AbstractServletWebServerFactory factory = getFactory(); AbstractServletWebServerFactory factory = getFactory();
this.webServer = factory this.webServer = factory.getWebServer(exampleServletRegistration());
.getWebServer(exampleServletRegistration());
this.webServer.start(); this.webServer.start();
int port = this.webServer.getPort(); int port = this.webServer.getPort();
this.webServer.start(); this.webServer.start();
@ -183,8 +181,7 @@ public abstract class AbstractServletWebServerFactoryTests {
@Test @Test
public void stopCalledTwice() throws Exception { public void stopCalledTwice() throws Exception {
AbstractServletWebServerFactory factory = getFactory(); AbstractServletWebServerFactory factory = getFactory();
this.webServer = factory this.webServer = factory.getWebServer(exampleServletRegistration());
.getWebServer(exampleServletRegistration());
this.webServer.start(); this.webServer.start();
this.webServer.stop(); this.webServer.stop();
this.webServer.stop(); this.webServer.stop();
@ -194,8 +191,7 @@ public abstract class AbstractServletWebServerFactoryTests {
public void emptyServerWhenPortIsMinusOne() throws Exception { public void emptyServerWhenPortIsMinusOne() throws Exception {
AbstractServletWebServerFactory factory = getFactory(); AbstractServletWebServerFactory factory = getFactory();
factory.setPort(-1); factory.setPort(-1);
this.webServer = factory this.webServer = factory.getWebServer(exampleServletRegistration());
.getWebServer(exampleServletRegistration());
this.webServer.start(); this.webServer.start();
assertThat(this.webServer.getPort()).isLessThan(0); // Jetty is -2 assertThat(this.webServer.getPort()).isLessThan(0); // Jetty is -2
} }
@ -203,8 +199,7 @@ public abstract class AbstractServletWebServerFactoryTests {
@Test @Test
public void stopServlet() throws Exception { public void stopServlet() throws Exception {
AbstractServletWebServerFactory factory = getFactory(); AbstractServletWebServerFactory factory = getFactory();
this.webServer = factory this.webServer = factory.getWebServer(exampleServletRegistration());
.getWebServer(exampleServletRegistration());
this.webServer.start(); this.webServer.start();
int port = this.webServer.getPort(); int port = this.webServer.getPort();
this.webServer.stop(); this.webServer.stop();
@ -217,8 +212,7 @@ public abstract class AbstractServletWebServerFactoryTests {
@Test @Test
public void restartWithKeepAlive() throws Exception { public void restartWithKeepAlive() throws Exception {
AbstractServletWebServerFactory factory = getFactory(); AbstractServletWebServerFactory factory = getFactory();
this.webServer = factory this.webServer = factory.getWebServer(exampleServletRegistration());
.getWebServer(exampleServletRegistration());
this.webServer.start(); this.webServer.start();
HttpComponentsAsyncClientHttpRequestFactory clientHttpRequestFactory = new HttpComponentsAsyncClientHttpRequestFactory(); HttpComponentsAsyncClientHttpRequestFactory clientHttpRequestFactory = new HttpComponentsAsyncClientHttpRequestFactory();
ListenableFuture<ClientHttpResponse> response1 = clientHttpRequestFactory ListenableFuture<ClientHttpResponse> response1 = clientHttpRequestFactory
@ -227,8 +221,7 @@ public abstract class AbstractServletWebServerFactoryTests {
assertThat(response1.get(10, TimeUnit.SECONDS).getRawStatusCode()).isEqualTo(200); assertThat(response1.get(10, TimeUnit.SECONDS).getRawStatusCode()).isEqualTo(200);
this.webServer.stop(); this.webServer.stop();
this.webServer = factory this.webServer = factory.getWebServer(exampleServletRegistration());
.getWebServer(exampleServletRegistration());
this.webServer.start(); this.webServer.start();
ListenableFuture<ClientHttpResponse> response2 = clientHttpRequestFactory ListenableFuture<ClientHttpResponse> response2 = clientHttpRequestFactory
@ -251,20 +244,18 @@ public abstract class AbstractServletWebServerFactoryTests {
public void startBlocksUntilReadyToServe() throws Exception { public void startBlocksUntilReadyToServe() throws Exception {
AbstractServletWebServerFactory factory = getFactory(); AbstractServletWebServerFactory factory = getFactory();
final Date[] date = new Date[1]; final Date[] date = new Date[1];
this.webServer = factory this.webServer = factory.getWebServer(new ServletContextInitializer() {
.getWebServer(new ServletContextInitializer() { @Override
@Override public void onStartup(ServletContext servletContext) throws ServletException {
public void onStartup(ServletContext servletContext) try {
throws ServletException { Thread.sleep(500);
try { date[0] = new Date();
Thread.sleep(500); }
date[0] = new Date(); catch (InterruptedException ex) {
} throw new ServletException(ex);
catch (InterruptedException ex) { }
throw new ServletException(ex); }
} });
}
});
this.webServer.start(); this.webServer.start();
assertThat(date[0]).isNotNull(); assertThat(date[0]).isNotNull();
} }
@ -273,14 +264,12 @@ public abstract class AbstractServletWebServerFactoryTests {
public void loadOnStartAfterContextIsInitialized() throws Exception { public void loadOnStartAfterContextIsInitialized() throws Exception {
AbstractServletWebServerFactory factory = getFactory(); AbstractServletWebServerFactory factory = getFactory();
final InitCountingServlet servlet = new InitCountingServlet(); final InitCountingServlet servlet = new InitCountingServlet();
this.webServer = factory this.webServer = factory.getWebServer(new ServletContextInitializer() {
.getWebServer(new ServletContextInitializer() { @Override
@Override public void onStartup(ServletContext servletContext) throws ServletException {
public void onStartup(ServletContext servletContext) servletContext.addServlet("test", servlet).setLoadOnStartup(1);
throws ServletException { }
servletContext.addServlet("test", servlet).setLoadOnStartup(1); });
}
});
assertThat(servlet.getInitCount()).isEqualTo(0); assertThat(servlet.getInitCount()).isEqualTo(0);
this.webServer.start(); this.webServer.start();
assertThat(servlet.getInitCount()).isEqualTo(1); assertThat(servlet.getInitCount()).isEqualTo(1);
@ -291,8 +280,7 @@ public abstract class AbstractServletWebServerFactoryTests {
AbstractServletWebServerFactory factory = getFactory(); AbstractServletWebServerFactory factory = getFactory();
int specificPort = SocketUtils.findAvailableTcpPort(41000); int specificPort = SocketUtils.findAvailableTcpPort(41000);
factory.setPort(specificPort); factory.setPort(specificPort);
this.webServer = factory this.webServer = factory.getWebServer(exampleServletRegistration());
.getWebServer(exampleServletRegistration());
this.webServer.start(); this.webServer.start();
assertThat(getResponse("http://localhost:" + specificPort + "/hello")) assertThat(getResponse("http://localhost:" + specificPort + "/hello"))
.isEqualTo("Hello World"); .isEqualTo("Hello World");
@ -303,8 +291,7 @@ public abstract class AbstractServletWebServerFactoryTests {
public void specificContextRoot() throws Exception { public void specificContextRoot() throws Exception {
AbstractServletWebServerFactory factory = getFactory(); AbstractServletWebServerFactory factory = getFactory();
factory.setContextPath("/say"); factory.setContextPath("/say");
this.webServer = factory this.webServer = factory.getWebServer(exampleServletRegistration());
.getWebServer(exampleServletRegistration());
this.webServer.start(); this.webServer.start();
assertThat(getResponse(getLocalUrl("/say/hello"))).isEqualTo("Hello World"); assertThat(getResponse(getLocalUrl("/say/hello"))).isEqualTo("Hello World");
} }
@ -340,8 +327,7 @@ public abstract class AbstractServletWebServerFactoryTests {
} }
factory.setInitializers(Arrays.asList(initializers[2], initializers[3])); factory.setInitializers(Arrays.asList(initializers[2], initializers[3]));
factory.addInitializers(initializers[4], initializers[5]); factory.addInitializers(initializers[4], initializers[5]);
this.webServer = factory.getWebServer(initializers[0], this.webServer = factory.getWebServer(initializers[0], initializers[1]);
initializers[1]);
this.webServer.start(); this.webServer.start();
InOrder ordered = inOrder((Object[]) initializers); InOrder ordered = inOrder((Object[]) initializers);
for (ServletContextInitializer initializer : initializers) { for (ServletContextInitializer initializer : initializers) {
@ -450,13 +436,15 @@ public abstract class AbstractServletWebServerFactoryTests {
@Test @Test
public void sslKeyAlias() throws Exception { public void sslKeyAlias() throws Exception {
AbstractServletWebServerFactory factory = getFactory(); AbstractServletWebServerFactory factory = getFactory();
factory.setSsl(getSsl(null, "password", "test-alias", "src/test/resources/test.jks")); factory.setSsl(
getSsl(null, "password", "test-alias", "src/test/resources/test.jks"));
this.webServer = factory.getWebServer( this.webServer = factory.getWebServer(
new ServletRegistrationBean<>(new ExampleServlet(true, false), "/hello")); new ServletRegistrationBean<>(new ExampleServlet(true, false), "/hello"));
this.webServer.start(); this.webServer.start();
SSLConnectionSocketFactory socketFactory = new SSLConnectionSocketFactory( SSLConnectionSocketFactory socketFactory = new SSLConnectionSocketFactory(
new SSLContextBuilder() new SSLContextBuilder().loadTrustMaterial(null,
.loadTrustMaterial(null, new SerialNumberValidatingTrustSelfSignedStrategy("77e7c302")).build()); new SerialNumberValidatingTrustSelfSignedStrategy("77e7c302"))
.build());
HttpClient httpClient = HttpClients.custom().setSSLSocketFactory(socketFactory) HttpClient httpClient = HttpClients.custom().setSSLSocketFactory(socketFactory)
.build(); .build();
HttpComponentsClientHttpRequestFactory requestFactory = new HttpComponentsClientHttpRequestFactory( HttpComponentsClientHttpRequestFactory requestFactory = new HttpComponentsClientHttpRequestFactory(
@ -666,8 +654,7 @@ public abstract class AbstractServletWebServerFactoryTests {
@Test @Test
public void cannotReadClassPathFiles() throws Exception { public void cannotReadClassPathFiles() throws Exception {
AbstractServletWebServerFactory factory = getFactory(); AbstractServletWebServerFactory factory = getFactory();
this.webServer = factory this.webServer = factory.getWebServer(exampleServletRegistration());
.getWebServer(exampleServletRegistration());
this.webServer.start(); this.webServer.start();
ClientHttpResponse response = getClientResponse( ClientHttpResponse response = getClientResponse(
getLocalUrl("/org/springframework/boot/SpringApplication.class")); getLocalUrl("/org/springframework/boot/SpringApplication.class"));
@ -678,17 +665,20 @@ public abstract class AbstractServletWebServerFactoryTests {
return getSsl(clientAuth, keyPassword, keyStore, null, null, null); return getSsl(clientAuth, keyPassword, keyStore, null, null, null);
} }
private Ssl getSsl(ClientAuth clientAuth, String keyPassword, String keyAlias, String keyStore) { private Ssl getSsl(ClientAuth clientAuth, String keyPassword, String keyAlias,
String keyStore) {
return getSsl(clientAuth, keyPassword, keyAlias, keyStore, null, null, null); return getSsl(clientAuth, keyPassword, keyAlias, keyStore, null, null, null);
} }
private Ssl getSsl(ClientAuth clientAuth, String keyPassword, String keyStore, private Ssl getSsl(ClientAuth clientAuth, String keyPassword, String keyStore,
String trustStore, String[] supportedProtocols, String[] ciphers) { String trustStore, String[] supportedProtocols, String[] ciphers) {
return getSsl(clientAuth, keyPassword, null, keyStore, trustStore, supportedProtocols, ciphers); return getSsl(clientAuth, keyPassword, null, keyStore, trustStore,
supportedProtocols, ciphers);
} }
private Ssl getSsl(ClientAuth clientAuth, String keyPassword, String keyAlias, String keyStore, private Ssl getSsl(ClientAuth clientAuth, String keyPassword, String keyAlias,
String trustStore, String[] supportedProtocols, String[] ciphers) { String keyStore, String trustStore, String[] supportedProtocols,
String[] ciphers) {
Ssl ssl = new Ssl(); Ssl ssl = new Ssl();
ssl.setClientAuth(clientAuth); ssl.setClientAuth(clientAuth);
if (keyPassword != null) { if (keyPassword != null) {
@ -751,14 +741,12 @@ public abstract class AbstractServletWebServerFactoryTests {
public void persistSession() throws Exception { public void persistSession() throws Exception {
AbstractServletWebServerFactory factory = getFactory(); AbstractServletWebServerFactory factory = getFactory();
factory.setPersistSession(true); factory.setPersistSession(true);
this.webServer = factory this.webServer = factory.getWebServer(sessionServletRegistration());
.getWebServer(sessionServletRegistration());
this.webServer.start(); this.webServer.start();
String s1 = getResponse(getLocalUrl("/session")); String s1 = getResponse(getLocalUrl("/session"));
String s2 = getResponse(getLocalUrl("/session")); String s2 = getResponse(getLocalUrl("/session"));
this.webServer.stop(); this.webServer.stop();
this.webServer = factory this.webServer = factory.getWebServer(sessionServletRegistration());
.getWebServer(sessionServletRegistration());
this.webServer.start(); this.webServer.start();
String s3 = getResponse(getLocalUrl("/session")); String s3 = getResponse(getLocalUrl("/session"));
String message = "Session error s1=" + s1 + " s2=" + s2 + " s3=" + s3; String message = "Session error s1=" + s1 + " s2=" + s2 + " s3=" + s3;
@ -772,8 +760,7 @@ public abstract class AbstractServletWebServerFactoryTests {
File sessionStoreDir = this.temporaryFolder.newFolder(); File sessionStoreDir = this.temporaryFolder.newFolder();
factory.setPersistSession(true); factory.setPersistSession(true);
factory.setSessionStoreDir(sessionStoreDir); factory.setSessionStoreDir(sessionStoreDir);
this.webServer = factory this.webServer = factory.getWebServer(sessionServletRegistration());
.getWebServer(sessionServletRegistration());
this.webServer.start(); this.webServer.start();
getResponse(getLocalUrl("/session")); getResponse(getLocalUrl("/session"));
this.webServer.stop(); this.webServer.stop();
@ -876,19 +863,17 @@ public abstract class AbstractServletWebServerFactoryTests {
public void rootServletContextResource() throws Exception { public void rootServletContextResource() throws Exception {
AbstractServletWebServerFactory factory = getFactory(); AbstractServletWebServerFactory factory = getFactory();
final AtomicReference<URL> rootResource = new AtomicReference<>(); final AtomicReference<URL> rootResource = new AtomicReference<>();
this.webServer = factory this.webServer = factory.getWebServer(new ServletContextInitializer() {
.getWebServer(new ServletContextInitializer() { @Override
@Override public void onStartup(ServletContext servletContext) throws ServletException {
public void onStartup(ServletContext servletContext) try {
throws ServletException { rootResource.set(servletContext.getResource("/"));
try { }
rootResource.set(servletContext.getResource("/")); catch (MalformedURLException ex) {
} throw new ServletException(ex);
catch (MalformedURLException ex) { }
throw new ServletException(ex); }
} });
}
});
this.webServer.start(); this.webServer.start();
assertThat(rootResource.get()).isNotNull(); assertThat(rootResource.get()).isNotNull();
} }
@ -897,8 +882,7 @@ public abstract class AbstractServletWebServerFactoryTests {
public void customServerHeader() throws Exception { public void customServerHeader() throws Exception {
AbstractServletWebServerFactory factory = getFactory(); AbstractServletWebServerFactory factory = getFactory();
factory.setServerHeader("MyServer"); factory.setServerHeader("MyServer");
this.webServer = factory this.webServer = factory.getWebServer(exampleServletRegistration());
.getWebServer(exampleServletRegistration());
this.webServer.start(); this.webServer.start();
ClientHttpResponse response = getClientResponse(getLocalUrl("/hello")); ClientHttpResponse response = getClientResponse(getLocalUrl("/hello"));
assertThat(response.getHeaders().getFirst("server")).isEqualTo("MyServer"); assertThat(response.getHeaders().getFirst("server")).isEqualTo("MyServer");
@ -907,8 +891,7 @@ public abstract class AbstractServletWebServerFactoryTests {
@Test @Test
public void serverHeaderIsDisabledByDefault() throws Exception { public void serverHeaderIsDisabledByDefault() throws Exception {
AbstractServletWebServerFactory factory = getFactory(); AbstractServletWebServerFactory factory = getFactory();
this.webServer = factory this.webServer = factory.getWebServer(exampleServletRegistration());
.getWebServer(exampleServletRegistration());
this.webServer.start(); this.webServer.start();
ClientHttpResponse response = getClientResponse(getLocalUrl("/hello")); ClientHttpResponse response = getClientResponse(getLocalUrl("/hello"));
assertThat(response.getHeaders().getFirst("server")).isNull(); assertThat(response.getHeaders().getFirst("server")).isNull();
@ -995,24 +978,6 @@ public abstract class AbstractServletWebServerFactoryTests {
assertThat(options.getDevelopment()).isEqualTo(false); assertThat(options.getDevelopment()).isEqualTo(false);
} }
@Test
public void explodedWarFileDocumentRootWhenRunningFromExplodedWar() throws Exception {
AbstractServletWebServerFactory factory = getFactory();
File webInfClasses = this.temporaryFolder.newFolder("test.war", "WEB-INF", "lib",
"spring-boot.jar");
File documentRoot = factory.getExplodedWarFileDocumentRoot(webInfClasses);
assertThat(documentRoot)
.isEqualTo(webInfClasses.getParentFile().getParentFile().getParentFile());
}
@Test
public void explodedWarFileDocumentRootWhenRunningFromPackagedWar() throws Exception {
AbstractServletWebServerFactory factory = getFactory();
File codeSourceFile = this.temporaryFolder.newFile("test.war");
File documentRoot = factory.getExplodedWarFileDocumentRoot(codeSourceFile);
assertThat(documentRoot).isNull();
}
protected abstract void addConnector(int port, protected abstract void addConnector(int port,
AbstractServletWebServerFactory factory); AbstractServletWebServerFactory factory);
@ -1287,10 +1252,10 @@ public abstract class AbstractServletWebServerFactoryTests {
} }
/** /**
* {@link TrustSelfSignedStrategy} that also validates certificate serial * {@link TrustSelfSignedStrategy} that also validates certificate serial number.
* number.
*/ */
private static final class SerialNumberValidatingTrustSelfSignedStrategy extends TrustSelfSignedStrategy { private static final class SerialNumberValidatingTrustSelfSignedStrategy
extends TrustSelfSignedStrategy {
private final String serialNumber; private final String serialNumber;
@ -1299,7 +1264,8 @@ public abstract class AbstractServletWebServerFactoryTests {
} }
@Override @Override
public boolean isTrusted(X509Certificate[] chain, String authType) throws CertificateException { public boolean isTrusted(X509Certificate[] chain, String authType)
throws CertificateException {
String hexSerialNumber = chain[0].getSerialNumber().toString(16); String hexSerialNumber = chain[0].getSerialNumber().toString(16);
boolean isMatch = hexSerialNumber.equals(this.serialNumber); boolean isMatch = hexSerialNumber.equals(this.serialNumber);
return super.isTrusted(chain, authType) && isMatch; return super.isTrusted(chain, authType) && isMatch;

@ -0,0 +1,56 @@
/*
* Copyright 2012-2017 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 org.springframework.boot.web.servlet.server;
import java.io.File;
import org.apache.commons.logging.LogFactory;
import org.junit.Rule;
import org.junit.Test;
import org.junit.rules.TemporaryFolder;
import static org.assertj.core.api.Assertions.assertThat;
/**
* Tests for {@link DocumentRoot}.
*
* @author Phillip Webb
*/
public class DocumentRootTests {
@Rule
public TemporaryFolder temporaryFolder = new TemporaryFolder();
private DocumentRoot documentRoot = new DocumentRoot(LogFactory.getLog(getClass()));
@Test
public void explodedWarFileDocumentRootWhenRunningFromExplodedWar() throws Exception {
File webInfClasses = this.temporaryFolder.newFolder("test.war", "WEB-INF", "lib",
"spring-boot.jar");
File directory = this.documentRoot.getExplodedWarFileDocumentRoot(webInfClasses);
assertThat(directory)
.isEqualTo(webInfClasses.getParentFile().getParentFile().getParentFile());
}
@Test
public void explodedWarFileDocumentRootWhenRunningFromPackagedWar() throws Exception {
File codeSourceFile = this.temporaryFolder.newFile("test.war");
File directory = this.documentRoot.getExplodedWarFileDocumentRoot(codeSourceFile);
assertThat(directory).isNull();
}
}
Loading…
Cancel
Save