Skip to content

Commit

Permalink
feat (jkube-kit/jkube-kit-spring-boot) : Support for Spring Native (#…
Browse files Browse the repository at this point in the history
…2138)

Signed-off-by: Rohan Kumar <[email protected]>
  • Loading branch information
rohanKanojia committed Aug 7, 2023
1 parent ba86499 commit 4d2833f
Show file tree
Hide file tree
Showing 10 changed files with 404 additions and 3 deletions.
1 change: 1 addition & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,7 @@ Usage:
* Fix #1929: Docker Image Name parsing fix
* Fix #1985: Update outdated methods in Spring Boot CRD Maven Quickstart
* Fix #2116: Remove user field from ImageName class
* Fix #2138: Support for Spring Boot Native Image
* Fix #2219: Kind/Filename mappings include optional apiVersion configuration
* Fix #2224: Quarkus native base image read from properties (configurable)
* Fix #2228: Quarkus native base image uses UBI 8.7
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,7 @@
*/
package org.eclipse.jkube.kit.common.util;

import java.io.File;
import java.net.URL;
import java.net.URLClassLoader;
import java.util.Collections;
Expand Down Expand Up @@ -111,5 +112,27 @@ public static boolean isSpringBootRepackage(JavaProject project) {
.map(e -> e.contains("repackage"))
.orElse(false);
}

public static Plugin getNativePlugin(JavaProject project) {
Plugin plugin = JKubeProjectUtil.getPlugin(project, "org.graalvm.buildtools", "native-maven-plugin");
if (plugin != null) {
return plugin;
}
return JKubeProjectUtil.getPlugin(project, "org.graalvm.buildtools.native", "org.graalvm.buildtools.native.gradle.plugin");
}

public static File getNativeArtifactFile(JavaProject project) {
Plugin plugin = getNativePlugin(project);
String nativeArtifactName = (String) Optional.ofNullable(plugin.getConfiguration())
.map(c -> c.get("imageName"))
.orElse(project.getArtifactId());
for (String location : new String[] {"", "native/nativeCompile/"}) {
File nativeArtifact = new File(project.getBuildDirectory(), location + nativeArtifactName);
if (nativeArtifact.exists()) {
return nativeArtifact;
}
}
return null;
}
}

Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,7 @@
import java.io.IOException;
import java.net.URL;
import java.net.URLClassLoader;
import java.nio.file.Files;
import java.util.Arrays;
import java.util.Collections;
import java.util.Map;
Expand Down Expand Up @@ -264,4 +265,124 @@ void isSpringBootRepackage_whenNoExecution_thenReturnFalse() {
// Then
assertThat(result).isFalse();
}

@Test
void getNativePlugin_whenNoNativePluginPresent_thenReturnNull() {
assertThat(SpringBootUtil.getNativePlugin(JavaProject.builder().build())).isNull();
}

@Test
void getNativePlugin_whenMavenNativePluginPresent_thenReturnPlugin() {
// Given
JavaProject javaProject = JavaProject.builder()
.plugin(Plugin.builder()
.groupId("org.graalvm.buildtools")
.artifactId("native-maven-plugin")
.build())
.build();

// When
Plugin plugin = SpringBootUtil.getNativePlugin(javaProject);

// Then
assertThat(plugin).isNotNull();
}

@Test
void getNativePlugin_whenGradleNativePluginPresent_thenReturnPlugin() {
// Given
JavaProject javaProject = JavaProject.builder()
.plugin(Plugin.builder()
.groupId("org.graalvm.buildtools.native")
.artifactId("org.graalvm.buildtools.native.gradle.plugin")
.build())
.build();

// When
Plugin plugin = SpringBootUtil.getNativePlugin(javaProject);

// Then
assertThat(plugin).isNotNull();
}

@Test
void getNativeArtifactFile_whenNativeExecutableNotFound_thenReturnNull(@TempDir File temporaryFolder) throws IOException {
// Given
JavaProject javaProject = JavaProject.builder()
.artifactId("sample")
.buildDirectory(temporaryFolder)
.plugin(Plugin.builder()
.groupId("org.graalvm.buildtools")
.artifactId("native-maven-plugin")
.build())
.build();

// When
File nativeArtifactFound = SpringBootUtil.getNativeArtifactFile(javaProject);

// Then
assertThat(nativeArtifactFound).isNull();
}

@Test
void getNativeArtifactFile_whenNativeExecutableInStandardMavenBuildDirectory_thenReturnNativeArtifact(@TempDir File temporaryFolder) throws IOException {
// Given
Files.createFile(temporaryFolder.toPath().resolve("sample"));
JavaProject javaProject = JavaProject.builder()
.artifactId("sample")
.buildDirectory(temporaryFolder)
.plugin(Plugin.builder()
.groupId("org.graalvm.buildtools")
.artifactId("native-maven-plugin")
.build())
.build();

// When
File nativeArtifactFound = SpringBootUtil.getNativeArtifactFile(javaProject);

// Then
assertThat(nativeArtifactFound).hasName("sample");
}

@Test
void getNativeArtifactFile_whenNativeExecutableInStandardMavenBuildDirectoryAndImageNameOverridden_thenReturnNativeArtifact(@TempDir File temporaryFolder) throws IOException {
// Given
Files.createFile(temporaryFolder.toPath().resolve("custom-native-name"));
JavaProject javaProject = JavaProject.builder()
.artifactId("sample")
.buildDirectory(temporaryFolder)
.plugin(Plugin.builder()
.groupId("org.graalvm.buildtools")
.artifactId("native-maven-plugin")
.configuration(Collections.singletonMap("imageName", "custom-native-name"))
.build())
.build();

// When
File nativeArtifactFound = SpringBootUtil.getNativeArtifactFile(javaProject);

// Then
assertThat(nativeArtifactFound).hasName("custom-native-name");
}

@Test
void getNativeArtifactFile_whenNativeExecutableInStandardGradleNativeDirectory_thenReturnNativeArtifact(@TempDir File temporaryFolder) throws IOException {
// Given
Files.createDirectories(temporaryFolder.toPath().resolve("native").resolve("nativeCompile"));
Files.createFile(temporaryFolder.toPath().resolve("native").resolve("nativeCompile").resolve("sample"));
JavaProject javaProject = JavaProject.builder()
.artifactId("sample")
.buildDirectory(temporaryFolder)
.plugin(Plugin.builder()
.groupId("org.graalvm.buildtools.native")
.artifactId("org.graalvm.buildtools.native.gradle.plugin")
.build())
.build();

// When
File nativeArtifactFound = SpringBootUtil.getNativeArtifactFile(javaProject);

// Then
assertThat(nativeArtifactFound).hasName("sample");
}
}
14 changes: 14 additions & 0 deletions jkube-kit/jkube-kit-spring-boot/pom.xml
Original file line number Diff line number Diff line change
Expand Up @@ -81,4 +81,18 @@

</dependencies>

<build>
<resources>
<!-- Copy over default images versions defined above -->
<resource>
<directory>src/main/resources-filtered</directory>
<filtering>true</filtering>
</resource>
<resource>
<directory>src/main/resources</directory>
<filtering>false</filtering>
</resource>
</resources>
</build>

</project>
Original file line number Diff line number Diff line change
@@ -0,0 +1,92 @@
/*
* Copyright (c) 2019 Red Hat, Inc.
* This program and the accompanying materials are made
* available under the terms of the Eclipse Public License 2.0
* which is available at:
*
* https://www.eclipse.org/legal/epl-2.0/
*
* SPDX-License-Identifier: EPL-2.0
*
* Contributors:
* Red Hat, Inc. - initial API and implementation
*/
package org.eclipse.jkube.springboot.generator;

import org.eclipse.jkube.generator.api.FromSelector;
import org.eclipse.jkube.generator.api.GeneratorConfig;
import org.eclipse.jkube.generator.api.GeneratorContext;
import org.eclipse.jkube.kit.common.Arguments;
import org.eclipse.jkube.kit.common.Assembly;
import org.eclipse.jkube.kit.common.AssemblyConfiguration;
import org.eclipse.jkube.kit.common.AssemblyFileSet;
import org.eclipse.jkube.kit.common.JavaProject;

import java.io.File;
import java.util.List;

import static org.eclipse.jkube.kit.common.util.FileUtil.getRelativePath;

public class NativeGenerator extends AbstractSpringBootNestedGenerator {
private final File nativeBinary;
private final FromSelector fromSelector;

public NativeGenerator(GeneratorContext generatorContext, GeneratorConfig generatorConfig, File nativeBinary) {
super(generatorContext, generatorConfig);
this.nativeBinary = nativeBinary;
fromSelector = new FromSelector.Default(generatorContext, "springboot-native");
}


@Override
public String getFrom() {
return fromSelector.getFrom();
}

@Override
public String getDefaultJolokiaPort() {
return "0";
}

@Override
public String getDefaultPrometheusPort() {
return "0";
}

@Override
public Arguments getBuildEntryPoint() {
return Arguments.builder()
.execArgument("./" + nativeBinary.getName())
.build();
}

@Override
public String getBuildWorkdir() {
return "/";
}

@Override
public String getTargetDir() {
return "/";
}

@Override
public AssemblyConfiguration createAssemblyConfiguration(List<AssemblyFileSet> defaultFileSets) {
Assembly.AssemblyBuilder assemblyBuilder = Assembly.builder();
final JavaProject project = getProject();
final AssemblyFileSet.AssemblyFileSetBuilder artifactFileSetBuilder = AssemblyFileSet.builder()
.outputDirectory(new File("."))
.directory(getRelativePath(project.getBaseDirectory(), nativeBinary.getParentFile()))
.fileMode("0755");
artifactFileSetBuilder.include(nativeBinary.getName());

assemblyBuilder.fileSets(defaultFileSets);
assemblyBuilder.fileSet(artifactFileSetBuilder.build());

return AssemblyConfiguration.builder()
.targetDir(getTargetDir())
.excludeFinalOutputArtifact(true)
.layer(assemblyBuilder.build())
.build();
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -129,7 +129,7 @@ protected String getDefaultWebPort() {

@Override
protected AssemblyConfiguration createAssembly() {
return Optional.ofNullable(nestedGenerator.createAssemblyConfiguration())
return Optional.ofNullable(nestedGenerator.createAssemblyConfiguration(addAdditionalFiles()))
.orElse(super.createAssembly());
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -17,15 +17,21 @@
import org.eclipse.jkube.generator.api.GeneratorContext;
import org.eclipse.jkube.kit.common.Arguments;
import org.eclipse.jkube.kit.common.AssemblyConfiguration;
import org.eclipse.jkube.kit.common.AssemblyFileSet;
import org.eclipse.jkube.kit.common.JavaProject;

import java.io.File;
import java.util.List;

import static org.eclipse.jkube.generator.javaexec.JavaExecGenerator.JOLOKIA_PORT_DEFAULT;
import static org.eclipse.jkube.generator.javaexec.JavaExecGenerator.PROMETHEUS_PORT_DEFAULT;
import static org.eclipse.jkube.kit.common.util.SpringBootUtil.getNativeArtifactFile;
import static org.eclipse.jkube.kit.common.util.SpringBootUtil.getNativePlugin;

public interface SpringBootNestedGenerator {
JavaProject getProject();

default AssemblyConfiguration createAssemblyConfiguration() {
default AssemblyConfiguration createAssemblyConfiguration(List<AssemblyFileSet> defaultFileSets) {
return null;
}

Expand All @@ -50,6 +56,12 @@ default Arguments getBuildEntryPoint() {
String getTargetDir();

static SpringBootNestedGenerator from(GeneratorContext generatorContext, GeneratorConfig generatorConfig) {
if (getNativePlugin(generatorContext.getProject()) != null) {
File nativeBinary = getNativeArtifactFile(generatorContext.getProject());
if (nativeBinary != null) {
return new NativeGenerator(generatorContext, generatorConfig, nativeBinary);
}
}
return new FatJarGenerator(generatorContext, generatorConfig);
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,21 @@
#
# Copyright (c) 2019 Red Hat, Inc.
# This program and the accompanying materials are made
# available under the terms of the Eclipse Public License 2.0
# which is available at:
#
# https://www.eclipse.org/legal/epl-2.0/
#
# SPDX-License-Identifier: EPL-2.0
#
# Contributors:
# Red Hat, Inc. - initial API and implementation
#

# Properties for specifying the default images version to use
# The replacement values are defined in the parent pom.xml as properties

# Upstream images
springboot-native.upstream.s2i=${image.springboot-native.upstream.s2i}
springboot-native.upstream.docker=${image.springboot-native.upstream.docker}
springboot-native.upstream.istag=${image.springboot-native.upstream.istag}
Loading

0 comments on commit 4d2833f

Please sign in to comment.