Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/carlosedp/mill-docker-nativeimage
A Scala Mill plugin to build container images with Native Image (GraalVM Binaries) application
https://github.com/carlosedp/mill-docker-nativeimage
docker graalvm graalvm-native-image mill mill-plugin native-image scala
Last synced: 3 months ago
JSON representation
A Scala Mill plugin to build container images with Native Image (GraalVM Binaries) application
- Host: GitHub
- URL: https://github.com/carlosedp/mill-docker-nativeimage
- Owner: carlosedp
- License: mit
- Created: 2022-10-13T20:05:05.000Z (over 2 years ago)
- Default Branch: main
- Last Pushed: 2024-08-27T14:08:24.000Z (5 months ago)
- Last Synced: 2024-08-27T15:40:07.412Z (5 months ago)
- Topics: docker, graalvm, graalvm-native-image, mill, mill-plugin, native-image, scala
- Language: Scala
- Homepage:
- Size: 151 KB
- Stars: 17
- Watchers: 2
- Forks: 1
- Open Issues: 1
-
Metadata Files:
- Readme: Readme.md
- Funding: .github/FUNDING.yml
- License: LICENSE
Awesome Lists containing this project
README
# mill-docker-nativeimage
This is a [Mill][mill] plugin modeled after the [contrib][docker-plugin] Docker plugin which allows building Docker images with Native Image binaries built by the amazing plugin [mill-native-image][mill-native-image] by Alex Archambault.
The inspiration came from Quarkus framework which generates applications that are cloud native and "Container-ready" with fast startup times and small size not depending on the JVM.
The plugin allows building very small containers and the application packed as a binary produces blazing fast startup times.
As a comparison, below the start time for the same application packed in an assembly with it's JAR using a JDK image compared to the native binary built with GraalVM in a Ubuntu image:
```plain
# Native Image
❯ date +%Y-%m-%dT%H:%M:%S.%N && docker run -it -p 8080:8080 --rm carlosedp/zioscalajs-backend-native
2023-02-03T14:30:06.536864570
{"timestamp":"2023-02-03T14:30:07.177923Z ","level":"INFO","thread":"zio-fiber-6","message":"Server started at http://localhost:8080"}
{"timestamp":"2023-02-03T14:30:07.178301Z ","level":"INFO","thread":"zio-fiber-6","message":"Started gaugetest with random Double every second"}# JDK
❯ date +%Y-%m-%dT%H:%M:%S.%N && docker run -it -p 8080:8080 --rm carlosedp/zioscalajs-backend-jdk
2023-02-03T14:30:14.004191633
{"timestamp":"2023-02-03T14:30:16.548346423Z ","level":"INFO","thread":"zio-fiber-6","message":"Server started at http://localhost:8080"}
{"timestamp":"2023-02-03T14:30:16.698928238Z ","level":"INFO","thread":"zio-fiber-6","message":"Started gaugetest with random Double every second"}
```As seen, the native version starts in 0,64105843 seconds versus 2,54415479 from the JDK version from launch to the first log message.
## Getting Started
The plugin provides a trait to configure the native build and container image, in addition there are two commands, one for building the Docker image and another to push the image to the registry. To push the image, the container runtime (Docker) must be already authenticated in your registry or DockerHub.
### Installing the Plugin
To start using this plugin you'll want to include the following import in your build file:
```scala
import $ivy.`com.carlosedp::mill-docker-nativeimage::0.6.1` //ReleaseVerMill
import com.carlosedp.milldockernative.DockerNative
```Under the hood, the plugin uses [mill-native-image][mill-native-image] to build your application Native Image binary which gets copied into the base container image which can be customized with some parameters as listed below. To generate the Native Image, you need an installed version of [GraalVM][graalvm-install] and the native-image utility.
### Usage
Sample configuration:
```scala
import mill._, mill.scalalib._, mill.scalalib.scalafmt._
import $ivy.`com.carlosedp::mill-docker-nativeimage::0.6.1`
import com.carlosedp.milldockernative.DockerNativeobject hello extends ScalaModule with DockerNative {
def scalaVersion = "3.3.1"
// def ivyDeps = ...
// def nativeImageClassPath = runClasspath()
object dockerNative extends DockerNativeConfig {
// Native Image parameters
def nativeImageName = "hello"
def nativeImageGraalVmJvmId = T("graalvm-java17:22.3.2")
def nativeImageClassPath = runClasspath()
def nativeImageMainClass = "com.domain.Hello.Hello"
// GraalVM parameters depending on your application needs
def nativeImageOptions = Seq(
"--no-fallback",
"--enable-url-protocols=http,https",
"-Djdk.http.auth.tunneling.disabledSchemes=",
) ++ (if (sys.props.get("os.name").contains("Linux")) Seq("--static") else Seq.empty)// Docker image parameters
def baseImage = "ubuntu:22.04"
def tags = List("docker.io/myuser/helloapp")
def exposedPorts = Seq(8080)
}object test extends Tests {
// ...
}
}
```Build and Push with:
```sh
mill hello.dockerNative.build# Test run
docker run -it --rm docker.io/myuser/helloapp# Push to a registry
mill hello.dockerNative.push
```A sample project is provided in [./example](./example) where above commands work.
If only the GraalVM native image binary is required to be built on a Docker container (for Linux), there is a convenience task:
```sh
mill hello.dockerNative.buildBin
```A more detailed build for a ZIO-http sample application with Native, Docker and DockerNative builds can be seen at the [zio-scalajs-stack][zio-scalajs-stack-build] project. Running a Native Image binary for a Scala 3 project requires Scala 3.3 due to a bug described [here][nativeimage-bug].
### Configuration
Image configuration parameters defined in `DockerNativeConfig` trait:
```scala
//
// These are the parameters to customize the image used to build the GraalVM Native Image binary
//
// Version of Coursier to be used to pull Scala dependencies in the build image
def coursierVersion: T[String] = "v2.1.6"
// Override the name of the image that will be generated to build the GraalVM Native Image
def baseDockerImage: T[String] = "mybasebuild"
// Override the DockerFile used to build the GraalVM Native Image
def baseDockerFile: T[String] = s"""FROM IMAGE:VERSION..."""
//
// These are the parameters to customize the output container image with the application
//
// Override tags to set the output image name
def tags = List("docker.io/myuser/myApp")
// Overrides base container image, default value below
def baseImage = "redhat/ubi8"
// Configure whether the docker build should check the remote registry for a new version of the base image before building.
// By default this is true if the base image is using a latest tag
def pullBaseImage = true
// Add container metadata via the LABEL instruction
def labels = Map("version" -> "1.0")
// TCP ports the container will listen to
def exposedPorts = Seq(8080, 443)
// UDP ports the container will listen to
def exposedUdpPorts = Seq(80)
// The names of mount points, these will be translated to VOLUME instructions
def volumes = Seq("/v1", "/v2")
// Environment variables to be set in the container (ENV instructions)
def envVars = Map("foo" -> "bar", "foobar" -> "barfoo")
// Command line arguments to be passed to the executable
def commandArgs = Seq("--port=80", "-v")
// Add RUN instructions
def run = Seq(
"/bin/bash -c 'echo Hello World!'",
"useradd -ms /bin/bash new-user"
)
// User to use when running the image
def user = "nobody"
// Optionally override the docker executable to use something else
def executable = "podman"
```Native Image parameters:
```scala
// Define the output binary name
def nativeImageName = "myAppName"
// Set the GraalVM version
def nativeImageGraalVmJvmId = T {"graalvm-java17:22.3.1"}
// Define the classpath
def nativeImageClassPath = runClasspath()
// Define your application main class
def nativeImageMainClass = "com.domain.myClass"
// Sets GraalVM Native Image options, depends on your application uses
def nativeImageOptions = Seq(
"--no-fallback",
)
```For more details, check the [mill-native-image][mill-native-image-src] source code where all available options is shown.
## Acknowledgements
This plugin would not be possible without the amazing work in the [mill-docker-plugin][docker-plugin] and [mill-native-image][mill-native-image] plugins.
[mill]: https://com-lihaoyi.github.io/mill/mill/Intro_to_Mill.html
[docker-plugin]: https://com-lihaoyi.github.io/mill/mill/Plugin_Docker.html
[mill-native-image]: https://github.com/alexarchambault/mill-native-image
[mill-native-image-src]: https://github.com/alexarchambault/mill-native-image/blob/master/plugin/src/io/github/alexarchambault/millnativeimage/NativeImage.scala
[graalvm-install]: https://www.graalvm.org/22.1/reference-manual/native-image/
[zio-scalajs-stack-build]: https://github.com/carlosedp/zio-scalajs-stack/blob/5c9e2817480ba7ef263770108197a36ff493dea7/build.sc#L51
[nativeimage-bug]: https://github.com/carlosedp/zio-scalajs-stack/issues/8