commit
65c3464fce
|
@ -71,6 +71,7 @@ include "spring-boot-project:spring-boot-test-autoconfigure"
|
||||||
include "spring-boot-tests:spring-boot-deployment-tests"
|
include "spring-boot-tests:spring-boot-deployment-tests"
|
||||||
include "spring-boot-tests:spring-boot-integration-tests:spring-boot-configuration-processor-tests"
|
include "spring-boot-tests:spring-boot-integration-tests:spring-boot-configuration-processor-tests"
|
||||||
include "spring-boot-tests:spring-boot-integration-tests:spring-boot-launch-script-tests"
|
include "spring-boot-tests:spring-boot-integration-tests:spring-boot-launch-script-tests"
|
||||||
|
include "spring-boot-tests:spring-boot-integration-tests:spring-boot-loader-tests"
|
||||||
include "spring-boot-tests:spring-boot-integration-tests:spring-boot-server-tests"
|
include "spring-boot-tests:spring-boot-integration-tests:spring-boot-server-tests"
|
||||||
|
|
||||||
file("${rootDir}/spring-boot-project/spring-boot-starters").eachDirMatch(~/spring-boot-starter.*/) {
|
file("${rootDir}/spring-boot-project/spring-boot-starters").eachDirMatch(~/spring-boot-starter.*/) {
|
||||||
|
|
|
@ -57,8 +57,12 @@ public class Handler extends URLStreamHandler {
|
||||||
|
|
||||||
private static final String PARENT_DIR = "/../";
|
private static final String PARENT_DIR = "/../";
|
||||||
|
|
||||||
|
private static final String PROTOCOL_HANDLER = "java.protocol.handler.pkgs";
|
||||||
|
|
||||||
private static final String[] FALLBACK_HANDLERS = { "sun.net.www.protocol.jar.Handler" };
|
private static final String[] FALLBACK_HANDLERS = { "sun.net.www.protocol.jar.Handler" };
|
||||||
|
|
||||||
|
private static URL jarContextUrl;
|
||||||
|
|
||||||
private static SoftReference<Map<File, JarFile>> rootFileCache;
|
private static SoftReference<Map<File, JarFile>> rootFileCache;
|
||||||
|
|
||||||
static {
|
static {
|
||||||
|
@ -98,7 +102,8 @@ public class Handler extends URLStreamHandler {
|
||||||
|
|
||||||
private URLConnection openFallbackConnection(URL url, Exception reason) throws IOException {
|
private URLConnection openFallbackConnection(URL url, Exception reason) throws IOException {
|
||||||
try {
|
try {
|
||||||
return openConnection(getFallbackHandler(), url);
|
URLConnection connection = openFallbackContextConnection(url);
|
||||||
|
return (connection != null) ? connection : openFallbackHandlerConnection(url);
|
||||||
}
|
}
|
||||||
catch (Exception ex) {
|
catch (Exception ex) {
|
||||||
if (reason instanceof IOException) {
|
if (reason instanceof IOException) {
|
||||||
|
@ -113,16 +118,35 @@ public class Handler extends URLStreamHandler {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private void log(boolean warning, String message, Exception cause) {
|
/**
|
||||||
|
* Attempt to open a fallback connection by using a context URL captured before the
|
||||||
|
* jar handler was replaced with our own version. Since this method doesn't use
|
||||||
|
* reflection it won't trigger "illegal reflective access operation has occurred"
|
||||||
|
* warnings on Java 13+.
|
||||||
|
* @param url the URL to open
|
||||||
|
* @return a {@link URLConnection} or {@code null}
|
||||||
|
*/
|
||||||
|
private URLConnection openFallbackContextConnection(URL url) {
|
||||||
try {
|
try {
|
||||||
Level level = warning ? Level.WARNING : Level.FINEST;
|
if (jarContextUrl != null) {
|
||||||
Logger.getLogger(getClass().getName()).log(level, message, cause);
|
return new URL(jarContextUrl, url.toExternalForm()).openConnection();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
catch (Exception ex) {
|
catch (Exception ex) {
|
||||||
if (warning) {
|
|
||||||
System.err.println("WARNING: " + message);
|
|
||||||
}
|
}
|
||||||
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Attempt to open a fallback connection by using reflection to access Java's default
|
||||||
|
* jar {@link URLStreamHandler}.
|
||||||
|
* @param url the URL to open
|
||||||
|
* @return the {@link URLConnection}
|
||||||
|
* @throws Exception if not connection could be opened
|
||||||
|
*/
|
||||||
|
private URLConnection openFallbackHandlerConnection(URL url) throws Exception {
|
||||||
|
URLStreamHandler fallbackHandler = getFallbackHandler();
|
||||||
|
return new URL(null, url.toExternalForm(), fallbackHandler).openConnection();
|
||||||
}
|
}
|
||||||
|
|
||||||
private URLStreamHandler getFallbackHandler() {
|
private URLStreamHandler getFallbackHandler() {
|
||||||
|
@ -142,8 +166,16 @@ public class Handler extends URLStreamHandler {
|
||||||
throw new IllegalStateException("Unable to find fallback handler");
|
throw new IllegalStateException("Unable to find fallback handler");
|
||||||
}
|
}
|
||||||
|
|
||||||
private URLConnection openConnection(URLStreamHandler handler, URL url) throws Exception {
|
private void log(boolean warning, String message, Exception cause) {
|
||||||
return new URL(null, url.toExternalForm(), handler).openConnection();
|
try {
|
||||||
|
Level level = warning ? Level.WARNING : Level.FINEST;
|
||||||
|
Logger.getLogger(getClass().getName()).log(level, message, cause);
|
||||||
|
}
|
||||||
|
catch (Exception ex) {
|
||||||
|
if (warning) {
|
||||||
|
System.err.println("WARNING: " + message);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
|
@ -333,6 +365,53 @@ public class Handler extends URLStreamHandler {
|
||||||
cache.put(sourceFile, jarFile);
|
cache.put(sourceFile, jarFile);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* If possible, capture a URL that is configured with the original jar handler so that
|
||||||
|
* we can use it as a fallback context later. We can only do this if we know that we
|
||||||
|
* can reset the handlers after.
|
||||||
|
*/
|
||||||
|
static void captureJarContextUrl() {
|
||||||
|
if (canResetCachedUrlHandlers()) {
|
||||||
|
String handlers = System.getProperty(PROTOCOL_HANDLER, "");
|
||||||
|
try {
|
||||||
|
System.clearProperty(PROTOCOL_HANDLER);
|
||||||
|
try {
|
||||||
|
resetCachedUrlHandlers();
|
||||||
|
jarContextUrl = new URL("jar:file:context.jar!/");
|
||||||
|
URLConnection connection = jarContextUrl.openConnection();
|
||||||
|
if (connection instanceof JarURLConnection) {
|
||||||
|
jarContextUrl = null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
catch (Exception ex) {
|
||||||
|
}
|
||||||
|
}
|
||||||
|
finally {
|
||||||
|
if (handlers == null) {
|
||||||
|
System.clearProperty(PROTOCOL_HANDLER);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
System.setProperty(PROTOCOL_HANDLER, handlers);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
resetCachedUrlHandlers();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static boolean canResetCachedUrlHandlers() {
|
||||||
|
try {
|
||||||
|
resetCachedUrlHandlers();
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
catch (Error ex) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void resetCachedUrlHandlers() {
|
||||||
|
URL.setURLStreamHandlerFactory(null);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Set if a generic static exception can be thrown when a URL cannot be connected.
|
* Set if a generic static exception can be thrown when a URL cannot be connected.
|
||||||
* This optimization is used during class loading to save creating lots of exceptions
|
* This optimization is used during class loading to save creating lots of exceptions
|
||||||
|
|
|
@ -411,6 +411,7 @@ public class JarFile extends AbstractJarFile implements Iterable<java.util.jar.J
|
||||||
* {@link URLStreamHandler} will be located to deal with jar URLs.
|
* {@link URLStreamHandler} will be located to deal with jar URLs.
|
||||||
*/
|
*/
|
||||||
public static void registerUrlProtocolHandler() {
|
public static void registerUrlProtocolHandler() {
|
||||||
|
Handler.captureJarContextUrl();
|
||||||
String handlers = System.getProperty(PROTOCOL_HANDLER, "");
|
String handlers = System.getProperty(PROTOCOL_HANDLER, "");
|
||||||
System.setProperty(PROTOCOL_HANDLER,
|
System.setProperty(PROTOCOL_HANDLER,
|
||||||
((handlers == null || handlers.isEmpty()) ? HANDLERS_PACKAGE : handlers + "|" + HANDLERS_PACKAGE));
|
((handlers == null || handlers.isEmpty()) ? HANDLERS_PACKAGE : handlers + "|" + HANDLERS_PACKAGE));
|
||||||
|
|
|
@ -163,6 +163,7 @@ class HandlerTests {
|
||||||
URLConnection jdkConnection = new URL(null, "jar:file:" + testJar.toURI().toURL() + "!/nested.jar!/",
|
URLConnection jdkConnection = new URL(null, "jar:file:" + testJar.toURI().toURL() + "!/nested.jar!/",
|
||||||
this.handler).openConnection();
|
this.handler).openConnection();
|
||||||
assertThat(jdkConnection).isNotInstanceOf(JarURLConnection.class);
|
assertThat(jdkConnection).isNotInstanceOf(JarURLConnection.class);
|
||||||
|
assertThat(jdkConnection.getClass().getName()).endsWith(".JarURLConnection");
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
|
|
|
@ -0,0 +1,18 @@
|
||||||
|
plugins {
|
||||||
|
id "java"
|
||||||
|
id "org.springframework.boot"
|
||||||
|
}
|
||||||
|
|
||||||
|
apply plugin: "io.spring.dependency-management"
|
||||||
|
|
||||||
|
repositories {
|
||||||
|
maven { url "file:${rootDir}/../int-test-maven-repository"}
|
||||||
|
mavenCentral()
|
||||||
|
maven { url "https://repo.spring.io/snapshot" }
|
||||||
|
maven { url "https://repo.spring.io/milestone" }
|
||||||
|
}
|
||||||
|
|
||||||
|
dependencies {
|
||||||
|
implementation("org.springframework.boot:spring-boot-starter-web")
|
||||||
|
implementation("org.webjars:jquery:3.5.0")
|
||||||
|
}
|
|
@ -0,0 +1,15 @@
|
||||||
|
pluginManagement {
|
||||||
|
repositories {
|
||||||
|
maven { url "file:${rootDir}/../int-test-maven-repository"}
|
||||||
|
mavenCentral()
|
||||||
|
maven { url "https://repo.spring.io/snapshot" }
|
||||||
|
maven { url "https://repo.spring.io/milestone" }
|
||||||
|
}
|
||||||
|
resolutionStrategy {
|
||||||
|
eachPlugin {
|
||||||
|
if (requested.id.id == "org.springframework.boot") {
|
||||||
|
useModule "org.springframework.boot:spring-boot-gradle-plugin:${requested.version}"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,50 @@
|
||||||
|
/*
|
||||||
|
* Copyright 2012-2020 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
|
||||||
|
*
|
||||||
|
* https://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.loaderapp;
|
||||||
|
|
||||||
|
import java.net.URL;
|
||||||
|
import java.util.Arrays;
|
||||||
|
|
||||||
|
import javax.servlet.ServletContext;
|
||||||
|
|
||||||
|
import org.springframework.boot.CommandLineRunner;
|
||||||
|
import org.springframework.boot.SpringApplication;
|
||||||
|
import org.springframework.boot.autoconfigure.SpringBootApplication;
|
||||||
|
import org.springframework.context.annotation.Bean;
|
||||||
|
import org.springframework.util.FileCopyUtils;
|
||||||
|
|
||||||
|
@SpringBootApplication
|
||||||
|
public class LoaderTestApplication {
|
||||||
|
|
||||||
|
@Bean
|
||||||
|
public CommandLineRunner commandLineRunner(ServletContext servletContext) {
|
||||||
|
return (args) -> {
|
||||||
|
URL resourceUrl = servletContext.getResource("webjars/jquery/3.5.0/jquery.js");
|
||||||
|
byte[] resourceContent = FileCopyUtils.copyToByteArray(resourceUrl.openStream());
|
||||||
|
URL directUrl = new URL(resourceUrl.toExternalForm());
|
||||||
|
byte[] directContent = FileCopyUtils.copyToByteArray(directUrl.openStream());
|
||||||
|
String message = (!Arrays.equals(resourceContent, directContent)) ? "NO MATCH"
|
||||||
|
: directContent.length + " BYTES";
|
||||||
|
System.out.println(">>>>> " + message + " from " + resourceUrl);
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void main(String[] args) {
|
||||||
|
SpringApplication.run(LoaderTestApplication.class, args).stop();
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,47 @@
|
||||||
|
plugins {
|
||||||
|
id "java"
|
||||||
|
id "org.springframework.boot.conventions"
|
||||||
|
id "org.springframework.boot.integration-test"
|
||||||
|
}
|
||||||
|
|
||||||
|
description = "Spring Boot Loader Integration Tests"
|
||||||
|
|
||||||
|
configurations {
|
||||||
|
app
|
||||||
|
}
|
||||||
|
|
||||||
|
dependencies {
|
||||||
|
app project(path: ":spring-boot-project:spring-boot-dependencies", configuration: "mavenRepository")
|
||||||
|
app project(path: ":spring-boot-project:spring-boot-tools:spring-boot-gradle-plugin", configuration: "mavenRepository")
|
||||||
|
app project(path: ":spring-boot-project:spring-boot-starters:spring-boot-starter-web", configuration: "mavenRepository")
|
||||||
|
|
||||||
|
intTestImplementation(enforcedPlatform(project(":spring-boot-project:spring-boot-parent")))
|
||||||
|
intTestImplementation(project(":spring-boot-project:spring-boot-tools:spring-boot-test-support"))
|
||||||
|
intTestImplementation(project(":spring-boot-project:spring-boot-starters:spring-boot-starter-test"))
|
||||||
|
intTestImplementation("org.testcontainers:junit-jupiter")
|
||||||
|
intTestImplementation("org.testcontainers:testcontainers")
|
||||||
|
}
|
||||||
|
|
||||||
|
task syncMavenRepository(type: Sync) {
|
||||||
|
from configurations.app
|
||||||
|
into "${buildDir}/int-test-maven-repository"
|
||||||
|
}
|
||||||
|
|
||||||
|
task syncAppSource(type: Sync) {
|
||||||
|
from "app"
|
||||||
|
into "${buildDir}/app"
|
||||||
|
filter { line ->
|
||||||
|
line.replace("id \"org.springframework.boot\"", "id \"org.springframework.boot\" version \"${project.version}\"")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
task buildApp(type: GradleBuild) {
|
||||||
|
dependsOn syncAppSource, syncMavenRepository
|
||||||
|
dir = "${buildDir}/app"
|
||||||
|
startParameter.buildCacheEnabled = false
|
||||||
|
tasks = ["build"]
|
||||||
|
}
|
||||||
|
|
||||||
|
intTest {
|
||||||
|
dependsOn buildApp
|
||||||
|
}
|
|
@ -0,0 +1,66 @@
|
||||||
|
/*
|
||||||
|
* Copyright 2012-2020 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
|
||||||
|
*
|
||||||
|
* https://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.loader;
|
||||||
|
|
||||||
|
import java.io.File;
|
||||||
|
import java.time.Duration;
|
||||||
|
|
||||||
|
import org.junit.jupiter.api.Test;
|
||||||
|
import org.testcontainers.containers.GenericContainer;
|
||||||
|
import org.testcontainers.containers.output.ToStringConsumer;
|
||||||
|
import org.testcontainers.containers.startupcheck.OneShotStartupCheckStrategy;
|
||||||
|
import org.testcontainers.junit.jupiter.Container;
|
||||||
|
import org.testcontainers.junit.jupiter.Testcontainers;
|
||||||
|
import org.testcontainers.utility.DockerImageName;
|
||||||
|
import org.testcontainers.utility.MountableFile;
|
||||||
|
|
||||||
|
import static org.assertj.core.api.Assertions.assertThat;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Integration tests loader that supports fat jars.
|
||||||
|
*
|
||||||
|
* @author Phillip Webb
|
||||||
|
*/
|
||||||
|
@Testcontainers(disabledWithoutDocker = true)
|
||||||
|
class LoaderIntegrationTests {
|
||||||
|
|
||||||
|
private static final DockerImageName JRE = DockerImageName.parse("adoptopenjdk:15-jre-hotspot");
|
||||||
|
|
||||||
|
private static ToStringConsumer output = new ToStringConsumer();
|
||||||
|
|
||||||
|
@Container
|
||||||
|
public static GenericContainer<?> container = new GenericContainer<>(JRE).withLogConsumer(output)
|
||||||
|
.withCopyFileToContainer(MountableFile.forHostPath(findApplication().toPath()), "/app.jar")
|
||||||
|
.withStartupCheckStrategy(new OneShotStartupCheckStrategy().withTimeout(Duration.ofMinutes(5)))
|
||||||
|
.withCommand("java", "-jar", "app.jar");
|
||||||
|
|
||||||
|
private static File findApplication() {
|
||||||
|
File appJar = new File("build/app/build/libs/app.jar");
|
||||||
|
if (appJar.isFile()) {
|
||||||
|
return appJar;
|
||||||
|
}
|
||||||
|
throw new IllegalStateException(
|
||||||
|
"Could not find test application in build/app/build/libs directory. Have you built it?");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void readUrlsWithoutWarning() {
|
||||||
|
assertThat(output.toUtf8String()).contains(">>>>> 287649 BYTES from").doesNotContain("WARNING:")
|
||||||
|
.doesNotContain("illegal");
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
Loading…
Reference in New Issue