Add classpath index support for exploded archives
Update the `Repackager` class so that an additional `classpath.idx` file is written into the jar that provides the original order of the classpath. The `JarLauncher` class now uses this file when running as an exploded archive to ensure that the classpath order is the same as when running from the far jar. Closes gh-9128 Co-authored-by: Phillip Webb <pwebb@pivotal.io>pull/19789/head
parent
ad72f86bdb
commit
45b1ab46c3
@ -0,0 +1,130 @@
|
|||||||
|
/*
|
||||||
|
* 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.BufferedReader;
|
||||||
|
import java.io.File;
|
||||||
|
import java.io.FileInputStream;
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.io.InputStream;
|
||||||
|
import java.io.InputStreamReader;
|
||||||
|
import java.net.MalformedURLException;
|
||||||
|
import java.net.URISyntaxException;
|
||||||
|
import java.net.URL;
|
||||||
|
import java.nio.charset.StandardCharsets;
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.Collections;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Objects;
|
||||||
|
import java.util.Set;
|
||||||
|
import java.util.stream.Collectors;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A class path index file that provides ordering information for JARs.
|
||||||
|
*
|
||||||
|
* @author Madhura Bhave
|
||||||
|
* @author Phillip Webb
|
||||||
|
*/
|
||||||
|
final class ClassPathIndexFile {
|
||||||
|
|
||||||
|
private final File root;
|
||||||
|
|
||||||
|
private final List<String> lines;
|
||||||
|
|
||||||
|
private final Set<String> folders;
|
||||||
|
|
||||||
|
private ClassPathIndexFile(File root, List<String> lines) {
|
||||||
|
this.root = root;
|
||||||
|
this.lines = lines;
|
||||||
|
this.folders = this.lines.stream().map(this::getFolder).filter(Objects::nonNull).collect(Collectors.toSet());
|
||||||
|
}
|
||||||
|
|
||||||
|
private String getFolder(String name) {
|
||||||
|
int lastSlash = name.lastIndexOf("/");
|
||||||
|
return (lastSlash != -1) ? name.substring(0, lastSlash) : null;
|
||||||
|
}
|
||||||
|
|
||||||
|
int size() {
|
||||||
|
return this.lines.size();
|
||||||
|
}
|
||||||
|
|
||||||
|
boolean containsFolder(String name) {
|
||||||
|
if (name == null || name.isEmpty()) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (name.endsWith("/")) {
|
||||||
|
return containsFolder(name.substring(0, name.length() - 1));
|
||||||
|
}
|
||||||
|
return this.folders.contains(name);
|
||||||
|
}
|
||||||
|
|
||||||
|
List<URL> getUrls() {
|
||||||
|
return Collections.unmodifiableList(this.lines.stream().map(this::asUrl).collect(Collectors.toList()));
|
||||||
|
}
|
||||||
|
|
||||||
|
private URL asUrl(String line) {
|
||||||
|
try {
|
||||||
|
return new File(this.root, line).toURI().toURL();
|
||||||
|
}
|
||||||
|
catch (MalformedURLException ex) {
|
||||||
|
throw new IllegalStateException(ex);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static ClassPathIndexFile loadIfPossible(URL root, String location) throws IOException {
|
||||||
|
return loadIfPossible(asFile(root), location);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static ClassPathIndexFile loadIfPossible(File root, String location) throws IOException {
|
||||||
|
return loadIfPossible(root, new File(root, location));
|
||||||
|
}
|
||||||
|
|
||||||
|
private static ClassPathIndexFile loadIfPossible(File root, File indexFile) throws IOException {
|
||||||
|
if (indexFile.exists() && indexFile.isFile()) {
|
||||||
|
try (InputStream inputStream = new FileInputStream(indexFile)) {
|
||||||
|
return new ClassPathIndexFile(root, loadLines(inputStream));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static List<String> loadLines(InputStream inputStream) throws IOException {
|
||||||
|
List<String> lines = new ArrayList<>();
|
||||||
|
BufferedReader reader = new BufferedReader(new InputStreamReader(inputStream, StandardCharsets.UTF_8));
|
||||||
|
String line = reader.readLine();
|
||||||
|
while (line != null) {
|
||||||
|
if (!line.trim().isEmpty()) {
|
||||||
|
lines.add(line);
|
||||||
|
}
|
||||||
|
line = reader.readLine();
|
||||||
|
}
|
||||||
|
return Collections.unmodifiableList(lines);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static File asFile(URL url) {
|
||||||
|
if (!"file".equals(url.getProtocol())) {
|
||||||
|
throw new IllegalArgumentException("URL does not reference a file");
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
return new File(url.toURI());
|
||||||
|
}
|
||||||
|
catch (URISyntaxException ex) {
|
||||||
|
return new File(url.getPath());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
@ -0,0 +1,123 @@
|
|||||||
|
/*
|
||||||
|
* 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.io.IOException;
|
||||||
|
import java.net.MalformedURLException;
|
||||||
|
import java.net.URL;
|
||||||
|
import java.nio.file.Files;
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
import org.junit.jupiter.api.Test;
|
||||||
|
import org.junit.jupiter.api.io.TempDir;
|
||||||
|
|
||||||
|
import static org.assertj.core.api.Assertions.assertThat;
|
||||||
|
import static org.assertj.core.api.Assertions.assertThatIllegalArgumentException;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tests for {@link ClassPathIndexFile}.
|
||||||
|
*
|
||||||
|
* @author Madhura Bhave
|
||||||
|
* @author Phillip Webb
|
||||||
|
*/
|
||||||
|
class ClassPathIndexFileTests {
|
||||||
|
|
||||||
|
@TempDir
|
||||||
|
File temp;
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void loadIfPossibleWhenRootIsNotFileReturnsNull() throws IOException {
|
||||||
|
assertThatIllegalArgumentException()
|
||||||
|
.isThrownBy(() -> ClassPathIndexFile.loadIfPossible(new URL("https://example.com/file"), "test.idx"))
|
||||||
|
.withMessage("URL does not reference a file");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void loadIfPossibleWhenRootDoesNotExistReturnsNull() throws Exception {
|
||||||
|
File root = new File(this.temp, "missing");
|
||||||
|
assertThat(ClassPathIndexFile.loadIfPossible(root.toURI().toURL(), "test.idx")).isNull();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void loadIfPossibleWhenRootIsFolderThrowsException() throws Exception {
|
||||||
|
File root = new File(this.temp, "folder");
|
||||||
|
root.mkdirs();
|
||||||
|
assertThat(ClassPathIndexFile.loadIfPossible(root.toURI().toURL(), "test.idx")).isNull();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void loadIfPossibleReturnsInstance() throws Exception {
|
||||||
|
ClassPathIndexFile indexFile = copyAndLoadTestIndexFile();
|
||||||
|
assertThat(indexFile).isNotNull();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void sizeReturnsNumberOfLines() throws Exception {
|
||||||
|
ClassPathIndexFile indexFile = copyAndLoadTestIndexFile();
|
||||||
|
assertThat(indexFile.size()).isEqualTo(5);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void containsFolderWhenFolderIsPresentReturnsTrue() throws Exception {
|
||||||
|
ClassPathIndexFile indexFile = copyAndLoadTestIndexFile();
|
||||||
|
assertThat(indexFile.containsFolder("BOOT-INF/layers/one/lib")).isTrue();
|
||||||
|
assertThat(indexFile.containsFolder("BOOT-INF/layers/one/lib/")).isTrue();
|
||||||
|
assertThat(indexFile.containsFolder("BOOT-INF/layers/two/lib")).isTrue();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void containsFolderWhenFolderIsMissingReturnsFalse() throws Exception {
|
||||||
|
ClassPathIndexFile indexFile = copyAndLoadTestIndexFile();
|
||||||
|
assertThat(indexFile.containsFolder("BOOT-INF/layers/nope/lib/")).isFalse();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
void getUrlsReturnsUrls() throws Exception {
|
||||||
|
ClassPathIndexFile indexFile = copyAndLoadTestIndexFile();
|
||||||
|
List<URL> urls = indexFile.getUrls();
|
||||||
|
List<File> expected = new ArrayList<>();
|
||||||
|
expected.add(new File(this.temp, "BOOT-INF/layers/one/lib/a.jar"));
|
||||||
|
expected.add(new File(this.temp, "BOOT-INF/layers/one/lib/b.jar"));
|
||||||
|
expected.add(new File(this.temp, "BOOT-INF/layers/one/lib/c.jar"));
|
||||||
|
expected.add(new File(this.temp, "BOOT-INF/layers/two/lib/d.jar"));
|
||||||
|
expected.add(new File(this.temp, "BOOT-INF/layers/two/lib/e.jar"));
|
||||||
|
assertThat(urls).containsExactly(expected.stream().map(this::toUrl).toArray(URL[]::new));
|
||||||
|
}
|
||||||
|
|
||||||
|
private URL toUrl(File file) {
|
||||||
|
try {
|
||||||
|
return file.toURI().toURL();
|
||||||
|
}
|
||||||
|
catch (MalformedURLException ex) {
|
||||||
|
throw new IllegalStateException(ex);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private ClassPathIndexFile copyAndLoadTestIndexFile() throws IOException, MalformedURLException {
|
||||||
|
copyTestIndexFile();
|
||||||
|
ClassPathIndexFile indexFile = ClassPathIndexFile.loadIfPossible(this.temp.toURI().toURL(), "test.idx");
|
||||||
|
return indexFile;
|
||||||
|
}
|
||||||
|
|
||||||
|
private void copyTestIndexFile() throws IOException {
|
||||||
|
Files.copy(getClass().getResourceAsStream("classpath-index-file.idx"),
|
||||||
|
new File(this.temp, "test.idx").toPath());
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
@ -0,0 +1,5 @@
|
|||||||
|
BOOT-INF/layers/one/lib/a.jar
|
||||||
|
BOOT-INF/layers/one/lib/b.jar
|
||||||
|
BOOT-INF/layers/one/lib/c.jar
|
||||||
|
BOOT-INF/layers/two/lib/d.jar
|
||||||
|
BOOT-INF/layers/two/lib/e.jar
|
Loading…
Reference in New Issue