diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/reactive/WebFluxAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/reactive/WebFluxAutoConfiguration.java index b1e6969ee763..e3f227af7705 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/reactive/WebFluxAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/reactive/WebFluxAutoConfiguration.java @@ -17,6 +17,8 @@ package org.springframework.boot.autoconfigure.web.reactive; import java.time.Duration; +import java.util.ArrayList; +import java.util.List; import org.apache.commons.logging.Log; import org.apache.commons.logging.LogFactory; @@ -50,6 +52,7 @@ import org.springframework.context.annotation.Import; import org.springframework.core.Ordered; import org.springframework.core.annotation.Order; +import org.springframework.core.io.ResourceLoader; import org.springframework.format.FormatterRegistry; import org.springframework.format.support.FormattingConversionService; import org.springframework.http.codec.ServerCodecConfigurer; @@ -156,12 +159,14 @@ public static class WebFluxConfig implements WebFluxConfigurer { private final ObjectProvider viewResolvers; + private final ResourceLoader resourceLoader; + public WebFluxConfig(org.springframework.boot.autoconfigure.web.ResourceProperties resourceProperties, WebProperties webProperties, WebFluxProperties webFluxProperties, ListableBeanFactory beanFactory, ObjectProvider resolvers, ObjectProvider codecCustomizers, ObjectProvider resourceHandlerRegistrationCustomizer, - ObjectProvider viewResolvers) { + ObjectProvider viewResolvers, ResourceLoader resourceLoader) { this.resourceProperties = resourceProperties.hasBeenCustomized() ? resourceProperties : webProperties.getResources(); this.webFluxProperties = webFluxProperties; @@ -170,6 +175,7 @@ public WebFluxConfig(org.springframework.boot.autoconfigure.web.ResourceProperti this.codecCustomizers = codecCustomizers; this.resourceHandlerRegistrationCustomizer = resourceHandlerRegistrationCustomizer.getIfAvailable(); this.viewResolvers = viewResolvers; + this.resourceLoader = resourceLoader; } @Override @@ -189,17 +195,28 @@ public void addResourceHandlers(ResourceHandlerRegistry registry) { return; } if (!registry.hasMappingForPattern("/webjars/**")) { - ResourceHandlerRegistration registration = registry.addResourceHandler("/webjars/**") - .addResourceLocations("classpath:/META-INF/resources/webjars/"); - configureResourceCaching(registration); - customizeResourceHandlerRegistration(registration); + String webjarsLocation = "classpath:/META-INF/resources/webjars/"; + if (this.resourceLoader.getResource(webjarsLocation).exists()) { + ResourceHandlerRegistration registration = registry.addResourceHandler("/webjars/**") + .addResourceLocations(webjarsLocation); + configureResourceCaching(registration); + customizeResourceHandlerRegistration(registration); + } } String staticPathPattern = this.webFluxProperties.getStaticPathPattern(); if (!registry.hasMappingForPattern(staticPathPattern)) { - ResourceHandlerRegistration registration = registry.addResourceHandler(staticPathPattern) - .addResourceLocations(this.resourceProperties.getStaticLocations()); - configureResourceCaching(registration); - customizeResourceHandlerRegistration(registration); + List foundLocations = new ArrayList<>(); + for (String staticLocation : this.resourceProperties.getStaticLocations()) { + if (this.resourceLoader.getResource(staticLocation).exists()) { + foundLocations.add(staticLocation); + } + } + if (!foundLocations.isEmpty()) { + ResourceHandlerRegistration registration = registry.addResourceHandler(staticPathPattern) + .addResourceLocations(foundLocations.toArray(new String[0])); + configureResourceCaching(registration); + customizeResourceHandlerRegistration(registration); + } } } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/servlet/WebMvcAutoConfiguration.java b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/servlet/WebMvcAutoConfiguration.java index a919f42862d8..b6e8cd1b69a2 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/servlet/WebMvcAutoConfiguration.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/main/java/org/springframework/boot/autoconfigure/web/servlet/WebMvcAutoConfiguration.java @@ -201,6 +201,8 @@ public static class WebMvcAutoConfigurationAdapter implements WebMvcConfigurer, private final ResourceHandlerRegistrationCustomizer resourceHandlerRegistrationCustomizer; + private final ResourceLoader resourceLoader; + private ServletContext servletContext; public WebMvcAutoConfigurationAdapter( @@ -209,7 +211,7 @@ public WebMvcAutoConfigurationAdapter( ObjectProvider messageConvertersProvider, ObjectProvider resourceHandlerRegistrationCustomizerProvider, ObjectProvider dispatcherServletPath, - ObjectProvider> servletRegistrations) { + ObjectProvider> servletRegistrations, ResourceLoader resourceLoader) { this.resourceProperties = resourceProperties.hasBeenCustomized() ? resourceProperties : webProperties.getResources(); this.mvcProperties = mvcProperties; @@ -218,6 +220,7 @@ public WebMvcAutoConfigurationAdapter( this.resourceHandlerRegistrationCustomizer = resourceHandlerRegistrationCustomizerProvider.getIfAvailable(); this.dispatcherServletPath = dispatcherServletPath; this.servletRegistrations = servletRegistrations; + this.resourceLoader = resourceLoader; this.mvcProperties.checkConfiguration(); } @@ -334,7 +337,11 @@ public void addResourceHandlers(ResourceHandlerRegistry registry) { logger.debug("Default resource handling disabled"); return; } - addResourceHandler(registry, "/webjars/**", "classpath:/META-INF/resources/webjars/"); + Resource webjarsLocationResource = this.resourceLoader + .getResource("classpath:/META-INF/resources/webjars/"); + if (webjarsLocationResource.exists()) { + addResourceHandler(registry, "/webjars/**", webjarsLocationResource); + } addResourceHandler(registry, this.mvcProperties.getStaticPathPattern(), (registration) -> { registration.addResourceLocations(this.resourceProperties.getStaticLocations()); if (this.servletContext != null) { @@ -344,7 +351,7 @@ public void addResourceHandlers(ResourceHandlerRegistry registry) { }); } - private void addResourceHandler(ResourceHandlerRegistry registry, String pattern, String... locations) { + private void addResourceHandler(ResourceHandlerRegistry registry, String pattern, Resource... locations) { addResourceHandler(registry, pattern, (registration) -> registration.addResourceLocations(locations)); } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/web/reactive/WebFluxAutoConfigurationTests.java b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/web/reactive/WebFluxAutoConfigurationTests.java index 18eacf173338..636844b2b40c 100644 --- a/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/web/reactive/WebFluxAutoConfigurationTests.java +++ b/spring-boot-project/spring-boot-autoconfigure/src/test/java/org/springframework/boot/autoconfigure/web/reactive/WebFluxAutoConfigurationTests.java @@ -157,9 +157,10 @@ void shouldRegisterResourceHandlerMapping() { SimpleUrlHandlerMapping hm = context.getBean("resourceHandlerMapping", SimpleUrlHandlerMapping.class); assertThat(hm.getUrlMap().get("/**")).isInstanceOf(ResourceWebHandler.class); ResourceWebHandler staticHandler = (ResourceWebHandler) hm.getUrlMap().get("/**"); - assertThat(staticHandler).extracting("locationValues").asList().hasSize(4); - assertThat(staticHandler.getLocations()).hasSize(1); - assertThat(staticHandler.getLocations().get(0)).hasToString("class path resource [public/]"); + assertThat(staticHandler).extracting("locationValues").asList().hasSize(2); + assertThat(staticHandler.getLocations()).hasSize(2); + assertThat(staticHandler.getLocations().get(0)).hasToString("class path resource [META-INF/resources/]"); + assertThat(staticHandler.getLocations().get(1)).hasToString("class path resource [public/]"); assertThat(hm.getUrlMap().get("/webjars/**")).isInstanceOf(ResourceWebHandler.class); ResourceWebHandler webjarsHandler = (ResourceWebHandler) hm.getUrlMap().get("/webjars/**"); assertThat(webjarsHandler).extracting("locationValues").asList() @@ -173,7 +174,7 @@ void shouldMapResourcesToCustomPath() { SimpleUrlHandlerMapping hm = context.getBean("resourceHandlerMapping", SimpleUrlHandlerMapping.class); assertThat(hm.getUrlMap().get("/static/**")).isInstanceOf(ResourceWebHandler.class); ResourceWebHandler staticHandler = (ResourceWebHandler) hm.getUrlMap().get("/static/**"); - assertThat(staticHandler).extracting("locationValues").asList().hasSize(4); + assertThat(staticHandler).extracting("locationValues").asList().hasSize(2); }); } diff --git a/spring-boot-project/spring-boot-autoconfigure/src/test/resources/META-INF/resources/webjars/webjar b/spring-boot-project/spring-boot-autoconfigure/src/test/resources/META-INF/resources/webjars/webjar new file mode 100644 index 000000000000..e69de29bb2d1