Skip to content

Commit a0befb1

Browse files
committed
fix: support for Spring Boot 3.2.0 (and newer) layered jar format
Signed-off-by: Marc Nuri <[email protected]>
1 parent d2dacef commit a0befb1

File tree

8 files changed

+139
-69
lines changed

8 files changed

+139
-69
lines changed

CHANGELOG.md

+1
Original file line numberDiff line numberDiff line change
@@ -29,6 +29,7 @@ Usage:
2929
* Fix #2456: Add utility class to decompress archive files
3030
* Fix #2472: Support for Helm Chart.yaml appVersion field defaulting to project version
3131
* Fix #2474: Remove Docker-related classes providing unused functionality
32+
* Fix #2477: Support for Spring Boot 3.2.0 (and newer) layered jar format
3233

3334
### 1.15.0 (2023-11-10)
3435
* Fix #2138: Support for Spring Boot Native Image

jkube-kit/common/src/main/java/org/eclipse/jkube/kit/common/ExternalCommand.java

+3-5
Original file line numberDiff line numberDiff line change
@@ -45,9 +45,9 @@ protected ExternalCommand(KitLogger log) {
4545
this(log, null);
4646
}
4747

48-
protected ExternalCommand(KitLogger log, File dir) {
48+
protected ExternalCommand(KitLogger log, File workDir) {
4949
this.log = log;
50-
this.workDir = dir;
50+
this.workDir = workDir;
5151
}
5252

5353
public void execute() throws IOException {
@@ -72,10 +72,8 @@ public void execute(String processInput) throws IOException {
7272
end();
7373
}
7474
if (statusCode != 0) {
75-
throw new IOException(String.format("Process '%s' exited with status %d",
76-
getCommandAsString(), statusCode));
75+
throw new IOException(String.format("Process '%s' exited with status %d", getCommandAsString(), statusCode));
7776
}
78-
7977
}
8078

8179
// Hooks for logging ...

jkube-kit/common/src/main/java/org/eclipse/jkube/kit/common/util/SpringBootUtil.java

-10
Original file line numberDiff line numberDiff line change
@@ -14,14 +14,12 @@
1414
package org.eclipse.jkube.kit.common.util;
1515

1616
import java.io.File;
17-
import java.io.IOException;
1817
import java.net.URL;
1918
import java.net.URLClassLoader;
2019
import java.util.Collections;
2120
import java.util.Map;
2221
import java.util.Optional;
2322
import java.util.Properties;
24-
import java.util.jar.JarFile;
2523

2624
import org.eclipse.jkube.kit.common.JavaProject;
2725
import org.eclipse.jkube.kit.common.Plugin;
@@ -124,14 +122,6 @@ public static boolean isSpringBootRepackage(JavaProject project) {
124122
.orElse(false);
125123
}
126124

127-
public static boolean isLayeredJar(File fatJar) {
128-
try (JarFile jarFile = new JarFile(fatJar)) {
129-
return jarFile.getEntry("BOOT-INF/layers.idx") != null;
130-
} catch (IOException ioException) {
131-
throw new IllegalStateException("Failure in inspecting fat jar for layers.idx file", ioException);
132-
}
133-
}
134-
135125
public static Plugin getNativePlugin(JavaProject project) {
136126
Plugin plugin = JKubeProjectUtil.getPlugin(project, "org.graalvm.buildtools", "native-maven-plugin");
137127
if (plugin != null) {

jkube-kit/common/src/test/java/org/eclipse/jkube/kit/common/util/SpringBootUtilTest.java

-30
Original file line numberDiff line numberDiff line change
@@ -32,10 +32,6 @@
3232
import java.util.Objects;
3333
import java.util.Optional;
3434
import java.util.Properties;
35-
import java.util.jar.Attributes;
36-
import java.util.jar.JarEntry;
37-
import java.util.jar.JarOutputStream;
38-
import java.util.jar.Manifest;
3935

4036
import static org.assertj.core.api.Assertions.assertThatIllegalStateException;
4137
import static org.junit.jupiter.api.Assertions.assertThrows;
@@ -249,32 +245,6 @@ void isSpringBootRepackage_whenNoExecution_thenReturnFalse() {
249245
assertThat(result).isFalse();
250246
}
251247

252-
@Test
253-
void isLayeredJar_whenInvalidFile_thenThrowException() {
254-
// When + Then
255-
assertThatIllegalStateException()
256-
.isThrownBy(() -> SpringBootUtil.isLayeredJar(new File("i-dont-exist.jar")))
257-
.withMessage("Failure in inspecting fat jar for layers.idx file");
258-
}
259-
260-
@Test
261-
void isLayeredJar_whenJarContainsLayers_thenReturnTrue(@TempDir File temporaryFolder) throws IOException {
262-
// Given
263-
File jarFile = new File(temporaryFolder, "fat.jar");
264-
Manifest manifest = new Manifest();
265-
manifest.getMainAttributes().put(Attributes.Name.MANIFEST_VERSION, "1.0");
266-
manifest.getMainAttributes().put(Attributes.Name.MAIN_CLASS, "org.example.Foo");
267-
try (JarOutputStream jarOutputStream = new JarOutputStream(Files.newOutputStream(jarFile.toPath()), manifest)) {
268-
jarOutputStream.putNextEntry(new JarEntry("BOOT-INF/layers.idx"));
269-
}
270-
271-
// When
272-
boolean result = SpringBootUtil.isLayeredJar(jarFile);
273-
274-
// Then
275-
assertThat(result).isTrue();
276-
}
277-
278248
@Test
279249
void getNativePlugin_whenNoNativePluginPresent_thenReturnNull() {
280250
assertThat(SpringBootUtil.getNativePlugin(JavaProject.builder().build())).isNull();

jkube-kit/jkube-kit-spring-boot/src/main/java/org/eclipse/jkube/springboot/SpringBootLayeredJar.java

+43-16
Original file line numberDiff line numberDiff line change
@@ -14,13 +14,19 @@
1414
package org.eclipse.jkube.springboot;
1515

1616
import lombok.Getter;
17+
import org.apache.commons.lang3.ArrayUtils;
18+
import org.apache.commons.lang3.StringUtils;
1719
import org.eclipse.jkube.kit.common.ExternalCommand;
1820
import org.eclipse.jkube.kit.common.KitLogger;
1921

2022
import java.io.File;
2123
import java.io.IOException;
24+
import java.io.InputStream;
2225
import java.util.ArrayList;
2326
import java.util.List;
27+
import java.util.Properties;
28+
import java.util.jar.JarFile;
29+
import java.util.zip.ZipEntry;
2430

2531
public class SpringBootLayeredJar {
2632

@@ -32,6 +38,32 @@ public SpringBootLayeredJar(File layeredJar, KitLogger kitLogger) {
3238
this.kitLogger = kitLogger;
3339
}
3440

41+
public boolean isLayeredJar() {
42+
try (JarFile jarFile = new JarFile(layeredJar)) {
43+
return jarFile.getEntry("BOOT-INF/layers.idx") != null && StringUtils.isNotBlank(getMainClass());
44+
} catch(Exception e) {
45+
kitLogger.debug("Couldn't determine if Spring Boot jar %s is layered", layeredJar.getName(), e);
46+
}
47+
return false;
48+
}
49+
50+
public String getMainClass() {
51+
try (JarFile jarFile = new JarFile(layeredJar)) {
52+
final ZipEntry manifest = jarFile.getEntry("META-INF/MANIFEST.MF");
53+
if (manifest == null) {
54+
return null;
55+
}
56+
final Properties properties = new Properties();
57+
try (InputStream manifestInputStream = jarFile.getInputStream(manifest)) {
58+
properties.load(manifestInputStream);
59+
return properties.getProperty("Main-Class");
60+
}
61+
} catch(Exception e) {
62+
kitLogger.debug("Couldn't determine Spring Boot jar's (%s) main class ", layeredJar.getName(), e);
63+
}
64+
return null;
65+
}
66+
3567
public List<String> listLayers() {
3668
final LayerListCommand layerListCommand = new LayerListCommand(kitLogger, layeredJar);
3769
try {
@@ -44,45 +76,40 @@ public List<String> listLayers() {
4476

4577
public void extractLayers(File extractionDir) {
4678
try {
47-
new LayerExtractorCommand(kitLogger, extractionDir, layeredJar).execute();
79+
new LayerToolsCommand(kitLogger, extractionDir, layeredJar, "extract").execute();
4880
} catch (IOException ioException) {
4981
throw new IllegalStateException("Failure in extracting spring boot jar layers", ioException);
5082
}
5183
}
5284

53-
private static class LayerExtractorCommand extends ExternalCommand {
85+
private static class LayerToolsCommand extends ExternalCommand {
5486
private final File layeredJar;
55-
protected LayerExtractorCommand(KitLogger log, File workDir, File layeredJar) {
87+
private final String[] args;
88+
89+
protected LayerToolsCommand(KitLogger log, File workDir, File layeredJar, String... args) {
5690
super(log, workDir);
5791
this.layeredJar = layeredJar;
92+
this.args = args;
5893
}
5994

6095
@Override
6196
protected String[] getArgs() {
62-
return new String[] { "java", "-Djarmode=layertools", "-jar", layeredJar.getAbsolutePath(), "extract"};
97+
return ArrayUtils.addAll(new String[] { "java", "-Djarmode=layertools", "-jar", layeredJar.getAbsolutePath()}, args);
6398
}
6499
}
65100

66-
private static class LayerListCommand extends ExternalCommand {
67-
private final File layeredJar;
68-
@Getter
69-
private final List<String> layers;
101+
@Getter
102+
private static class LayerListCommand extends LayerToolsCommand {
70103

104+
private final List<String> layers;
71105
protected LayerListCommand(KitLogger log, File layeredJar) {
72-
super(log);
73-
this.layeredJar = layeredJar;
106+
super(log, null, layeredJar, "list");
74107
layers = new ArrayList<>();
75108
}
76109

77-
@Override
78-
protected String[] getArgs() {
79-
return new String[] { "java", "-Djarmode=layertools", "-jar", layeredJar.getAbsolutePath(), "list"};
80-
}
81-
82110
@Override
83111
protected void processLine(String line) {
84112
layers.add(line);
85113
}
86-
87114
}
88115
}

jkube-kit/jkube-kit-spring-boot/src/main/java/org/eclipse/jkube/springboot/generator/LayeredJarGenerator.java

+1-2
Original file line numberDiff line numberDiff line change
@@ -30,7 +30,6 @@
3030

3131
public class LayeredJarGenerator extends AbstractSpringBootNestedGenerator {
3232

33-
private static final String MAIN_CLASS = "org.springframework.boot.loader.JarLauncher";
3433
private final SpringBootLayeredJar springBootLayeredJar;
3534

3635
public LayeredJarGenerator(GeneratorContext generatorContext, GeneratorConfig generatorConfig, File layeredJar) {
@@ -41,7 +40,7 @@ public LayeredJarGenerator(GeneratorContext generatorContext, GeneratorConfig ge
4140
@Override
4241
public Map<String, String> getEnv(Function<Boolean, Map<String, String>> javaExecEnvSupplier, boolean prePackagePhase) {
4342
final Map<String, String> res = super.getEnv(javaExecEnvSupplier, prePackagePhase);
44-
res.put("JAVA_MAIN_CLASS", MAIN_CLASS);
43+
res.put("JAVA_MAIN_CLASS", springBootLayeredJar.getMainClass());
4544
return res;
4645
}
4746

jkube-kit/jkube-kit-spring-boot/src/main/java/org/eclipse/jkube/springboot/generator/SpringBootNestedGenerator.java

+2-2
Original file line numberDiff line numberDiff line change
@@ -20,13 +20,13 @@
2020
import org.eclipse.jkube.kit.common.AssemblyConfiguration;
2121
import org.eclipse.jkube.kit.common.AssemblyFileSet;
2222
import org.eclipse.jkube.kit.common.JavaProject;
23+
import org.eclipse.jkube.springboot.SpringBootLayeredJar;
2324

2425
import java.util.List;
2526
import java.util.Map;
2627

2728
import static org.eclipse.jkube.generator.javaexec.JavaExecGenerator.JOLOKIA_PORT_DEFAULT;
2829
import static org.eclipse.jkube.generator.javaexec.JavaExecGenerator.PROMETHEUS_PORT_DEFAULT;
29-
import static org.eclipse.jkube.kit.common.util.SpringBootUtil.isLayeredJar;
3030
import java.io.File;
3131
import java.util.function.Function;
3232

@@ -70,7 +70,7 @@ static SpringBootNestedGenerator from(GeneratorContext generatorContext, Generat
7070
}
7171
}
7272
if (fatJarDetectorResult != null && fatJarDetectorResult.getArchiveFile() != null &&
73-
isLayeredJar(fatJarDetectorResult.getArchiveFile())) {
73+
new SpringBootLayeredJar(fatJarDetectorResult.getArchiveFile(), generatorContext.getLogger()).isLayeredJar()) {
7474
return new LayeredJarGenerator(generatorContext, generatorConfig, fatJarDetectorResult.getArchiveFile());
7575
}
7676
return new FatJarGenerator(generatorContext, generatorConfig);

jkube-kit/jkube-kit-spring-boot/src/test/java/org/eclipse/jkube/springboot/SpringBootLayeredJarTest.java

+89-4
Original file line numberDiff line numberDiff line change
@@ -26,6 +26,10 @@
2626
import java.nio.file.Files;
2727
import java.util.List;
2828
import java.util.Objects;
29+
import java.util.jar.Attributes;
30+
import java.util.jar.JarEntry;
31+
import java.util.jar.JarOutputStream;
32+
import java.util.jar.Manifest;
2933

3034
import static org.assertj.core.api.Assertions.assertThat;
3135
import static org.assertj.core.api.Assertions.assertThatIllegalStateException;
@@ -37,6 +41,7 @@ class SpringBootLayeredJarTest {
3741

3842
private SpringBootLayeredJar springBootLayeredJar;
3943

44+
4045
@Nested
4146
@DisplayName("with invalid jar")
4247
class InvalidJar {
@@ -45,6 +50,22 @@ void setup() {
4550
springBootLayeredJar = new SpringBootLayeredJar(new File(projectDir, "invalid.jar"), new KitLogger.SilentLogger());
4651
}
4752

53+
@Test
54+
void isLayeredJar_returnsFalse() {
55+
// When
56+
final boolean result = springBootLayeredJar.isLayeredJar();
57+
// Then
58+
assertThat(result).isFalse();
59+
}
60+
61+
@Test
62+
void getMainClass_returnsNull() {
63+
// When
64+
final String result = springBootLayeredJar.getMainClass();
65+
// Then
66+
assertThat(result).isNull();
67+
}
68+
4869
@Test
4970
void listLayers_whenJarInvalid_thenThrowException() {
5071
assertThatIllegalStateException()
@@ -59,9 +80,10 @@ void extractLayers_whenJarInvalid_thenThrowException() {
5980
.withMessage("Failure in extracting spring boot jar layers");
6081
}
6182
}
83+
6284
@Nested
63-
@DisplayName("with valid jar")
64-
class ValidJar {
85+
@DisplayName("with valid (real) jar")
86+
class RealJar {
6587
@BeforeEach
6688
void setup() throws IOException {
6789
final File layeredJar = new File(projectDir, "layered.jar");
@@ -73,7 +95,23 @@ void setup() throws IOException {
7395
}
7496

7597
@Test
76-
void listLayers_whenJarInvalid_thenThrowException() {
98+
void isLayeredJar_returnsTrue() {
99+
// When
100+
final boolean result = springBootLayeredJar.isLayeredJar();
101+
// Then
102+
assertThat(result).isTrue();
103+
}
104+
105+
@Test
106+
void getMainClass_returnsJarLauncher() {
107+
// When
108+
final String result = springBootLayeredJar.getMainClass();
109+
// Then
110+
assertThat(result).isEqualTo("org.springframework.boot.loader.JarLauncher");
111+
}
112+
113+
@Test
114+
void listLayers() {
77115
// When
78116
final List<String> result = springBootLayeredJar.listLayers();
79117
// Then
@@ -82,7 +120,7 @@ void listLayers_whenJarInvalid_thenThrowException() {
82120
}
83121

84122
@Test
85-
void extractLayers_whenJarInvalid_thenThrowException() throws IOException {
123+
void extractLayers() throws IOException {
86124
// Given
87125
final File extractionDir = Files.createDirectory(new File(projectDir, "extracted").toPath()).toFile();
88126
// When
@@ -93,4 +131,51 @@ void extractLayers_whenJarInvalid_thenThrowException() throws IOException {
93131
.contains("dependencies", "spring-boot-loader", "snapshot-dependencies", "application");
94132
}
95133
}
134+
135+
@Nested
136+
@DisplayName("with fake jar with MANIFEST.MF and layers.idx")
137+
class FakeJar {
138+
@BeforeEach
139+
void setup() throws IOException {
140+
final File fakeJar = new File(projectDir, "fake.jar");
141+
final Manifest manifest = new Manifest();
142+
manifest.getMainAttributes().put(Attributes.Name.MANIFEST_VERSION, "1.0");
143+
manifest.getMainAttributes().put(Attributes.Name.MAIN_CLASS, "org.example.Foo");
144+
try (JarOutputStream jarOutputStream = new JarOutputStream(Files.newOutputStream(fakeJar.toPath()), manifest)) {
145+
jarOutputStream.putNextEntry(new JarEntry("BOOT-INF/layers.idx"));
146+
}
147+
springBootLayeredJar = new SpringBootLayeredJar(fakeJar, new KitLogger.SilentLogger());
148+
}
149+
150+
@Test
151+
void getMainClass_returnsManifestMainClass() {
152+
// When
153+
final String result = springBootLayeredJar.getMainClass();
154+
// Then
155+
assertThat(result).isEqualTo("org.example.Foo");
156+
}
157+
158+
@Test
159+
void isLayeredJar_returnsTrue() {
160+
// When
161+
final boolean result = springBootLayeredJar.isLayeredJar();
162+
// Then
163+
assertThat(result).isTrue();
164+
}
165+
166+
167+
@Test
168+
void listLayers_whenJarInvalid_thenThrowException() {
169+
assertThatIllegalStateException()
170+
.isThrownBy(() -> springBootLayeredJar.listLayers())
171+
.withMessage("Failure in getting spring boot jar layers information");
172+
}
173+
174+
@Test
175+
void extractLayers_whenJarInvalid_thenThrowException() {
176+
assertThatIllegalStateException()
177+
.isThrownBy(() -> springBootLayeredJar.extractLayers(projectDir))
178+
.withMessage("Failure in extracting spring boot jar layers");
179+
}
180+
}
96181
}

0 commit comments

Comments
 (0)