谷歌助力,快速實(shí)現(xiàn) Java 應(yīng)用容器化
Google 在 2018 年下旬開源了一款新的 Java 工具 Jib,可以輕松地將 Java 應(yīng)用程序容器化。通過 Jib,我們不需要編寫 Dockerfile 或安裝 Docker,通過集成到 Maven 或 Gradle 插件,就可以立即將 Java 應(yīng)用程序容器化。
開源地址: https://github.com/GoogleContainerTools/jib
一、什么是 Jib
Jib 是一個(gè)快速而簡(jiǎn)單的容器鏡像構(gòu)建工具,它作為 Maven 或 Gradle 的一部分運(yùn)行,不需要編寫 Dockerfile 或運(yùn)行 Docker 守護(hù)進(jìn)程。它從 Maven 或 Gradle 中構(gòu)建我們的 Docker 鏡像, 并只將發(fā)生變更的層(而不是整個(gè)應(yīng)用程序)推送到注冊(cè)表來節(jié)省寶貴的構(gòu)建時(shí)間?,F(xiàn)在,我們對(duì) Docker 構(gòu)建流程和 Jib 構(gòu)建流程進(jìn)行對(duì)比。Docker 構(gòu)建流程,如下所示。
Jib 構(gòu)建流程,則是這樣的。
二、實(shí)戰(zhàn)出真知
1. 構(gòu)建一個(gè)簡(jiǎn)單的 Java 工程
我們編寫一個(gè)簡(jiǎn)單的 Java 類。
- public class HelloWorld {
- public static void main(String[] args) {
- System.out.println("Hello World!");
- System.out.println("http://blog.720ui.com");
- }
- }
緊接著,我們?cè)賱?chuàng)建一個(gè) pom.xml 文件。
- <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
- xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
- <modelVersion>4.0.0</modelVersion>
- <groupId>com.lianggzone.sample.lib</groupId>
- <artifactId>helloworld-samples</artifactId>
- <version>0.1</version>
- <packaging>jar</packaging>
- <name>helloworld-samples</name>
- <properties>
- <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
- <jib-maven-plugin.version>1.0.2</jib-maven-plugin.version>
- <maven-compiler-plugin.version>3.8.0</maven-compiler-plugin.version>
- </properties>
- <dependencies>
- </dependencies>
- <build>
- <plugins>
- <plugin>
- <groupId>org.apache.maven.plugins</groupId>
- <artifactId>maven-compiler-plugin</artifactId>
- <version>${maven-compiler-plugin.version}</version>
- <configuration>
- <source>1.8</source>
- <target>1.8</target>
- </configuration>
- </plugin>
- <!-- Jib -->
- <plugin>
- <groupId>com.google.cloud.tools</groupId>
- <artifactId>jib-maven-plugin</artifactId>
- <version>${jib-maven-plugin.version}</version>
- <configuration>
- <from>
- <image>registry.cn-hangzhou.aliyuncs.com/lianggzone/oracle_java8</image>
- </from>
- <to>
- <image>registry.cn-hangzhou.aliyuncs.com/lianggzone/jib-helloworld:v1</image>
- </to>
- <container>
- <jvmFlags>
- <jvmFlag>-Xms512m</jvmFlag>
- <jvmFlag>-Xdebug</jvmFlag>
- </jvmFlags>
- <mainClass>com.lianggzone.HelloWorld</mainClass>
- </container>
- </configuration>
- <executions>
- <execution>
- <phase>package</phase>
- <goals>
- <goal>build</goal>
- </goals>
- </execution>
- </executions>
- </plugin>
- </plugins>
- </build>
- </project>
由于默認(rèn)訪問谷歌的 gcr.io 倉(cāng)庫(kù),而國(guó)內(nèi)訪問 gcr.io 不穩(wěn)定會(huì)經(jīng)常導(dǎo)致網(wǎng)絡(luò)超時(shí),所以筆者使用了國(guó)內(nèi)的阿里云鏡像服務(wù),那么就不需要訪問谷歌的倉(cāng)庫(kù)了?,F(xiàn)在,我們執(zhí)行 mvn compile jib:build
命令進(jìn)行自動(dòng)化構(gòu)建,它會(huì)從 <from>
拉取鏡像,并把生成的鏡像上傳到 <to>
設(shè)置的地址。這里,筆者還通過 ` 設(shè)置了一些 JVM 參數(shù)。
- mvn compile jib:build
此外,如果”登錄失敗,未授權(quán)”,需要通過 docker login
登錄鑒權(quán)一下。此外,更好的做法是,你可以考慮在Maven 中放置憑據(jù)。
- <settings>
- ...
- <servers>
- ...
- <server>
- <id>registry.cn-hangzhou.aliyuncs.com</id>
- <username>你的阿里云賬號(hào)</username>
- <password>你的阿里云密碼</password>
- </server>
- </servers>
- </settings>
大功告成,現(xiàn)在,我們來驗(yàn)證一把。我們通過 docker pull
拉取鏡像,并運(yùn)行。
- docker pull registry.cn-hangzhou.aliyuncs.com/lianggzone/jib-helloworld:v1
- docker run --name jib-helloworld -it registry.cn-hangzhou.aliyuncs.com/lianggzone/jib-helloworld:v1 /bin/bash
2. 構(gòu)建一個(gè) SpringBoot 的可運(yùn)行 Jar
我們來一個(gè)復(fù)雜一些的項(xiàng)目,構(gòu)建一個(gè) SpringBoot 的項(xiàng)目?,F(xiàn)在,我們首先需要搭建一個(gè)工程,并創(chuàng)建一個(gè)啟動(dòng)類。
- @SpringBootApplication
- public class Application {
- public static void main(String[] args) {
- SpringApplication.run(Application.class, args);
- }
- }
同時(shí),需要一個(gè) Web 的接口。
- @RestController
- public class WebController {
- @RequestMapping("/blog")
- public String index() {
- return "http://blog.720ui.com";
- }
- }
緊接著,我們?cè)賱?chuàng)建一個(gè) pom.xml 文件。
- <?xml version="1.0" encoding="UTF-8"?>
- <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
- xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
- <modelVersion>4.0.0</modelVersion>
- <parent>
- <groupId>org.springframework.boot</groupId>
- <artifactId>spring-boot-starter-parent</artifactId>
- <version>2.1.2.RELEASE</version>
- </parent>
- <groupId>com.lianggzone.sample.lib</groupId>
- <artifactId>springboot-samples</artifactId>
- <version>0.1</version>
- <packaging>jar</packaging>
- <name>springboot-samples</name>
- <dependencies>
- <dependency>
- <groupId>org.springframework.boot</groupId>
- <artifactId>spring-boot-starter-web</artifactId>
- </dependency>
- </dependencies>
- <properties>
- <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
- <jib-maven-plugin.version>1.0.2</jib-maven-plugin.version>
- <maven-compiler-plugin.version>3.8.0</maven-compiler-plugin.version>
- </properties>
- <build>
- <plugins>
- <plugin>
- <groupId>org.apache.maven.plugins</groupId>
- <artifactId>maven-compiler-plugin</artifactId>
- <version>${maven-compiler-plugin.version}</version>
- <configuration>
- <source>1.8</source>
- <target>1.8</target>
- </configuration>
- </plugin>
- <!-- Jib -->
- <plugin>
- <groupId>com.google.cloud.tools</groupId>
- <artifactId>jib-maven-plugin</artifactId>
- <version>${jib-maven-plugin.version}</version>
- <configuration>
- <from>
- <image>registry.cn-hangzhou.aliyuncs.com/lianggzone/oracle_java8</image>
- </from>
- <to>
- <image>registry.cn-hangzhou.aliyuncs.com/lianggzone/jib-springboot:v1</image>
- </to>
- <container>
- <jvmFlags>
- <jvmFlag>-Xms512m</jvmFlag>
- <jvmFlag>-Xdebug</jvmFlag>
- </jvmFlags>
- </container>
- </configuration>
- <executions>
- <execution>
- <phase>package</phase>
- <goals>
- <goal>build</goal>
- </goals>
- </execution>
- </executions>
- </plugin>
- </plugins>
- </build>
- </project>
現(xiàn)在,我們執(zhí)行 mvn compile jib:build
命令進(jìn)行自動(dòng)化構(gòu)建。執(zhí)行完成后,我們可以在阿里云鏡像倉(cāng)庫(kù)獲取鏡像。
現(xiàn)在,我們?cè)賮眚?yàn)證一把。我們通過 docker pull
拉取鏡像,并運(yùn)行。
- docker pull registry.cn-hangzhou.aliyuncs.com/lianggzone/jib-springboot:v1
- docker run -p 8080:8080 --name jib-springboot -it registry.cn-hangzhou.aliyuncs.com/lianggzone/jib-springboot:v1 /bin/bash
執(zhí)行結(jié)果,如下所示。
現(xiàn)在,我們?cè)L問 http://localhost:8080/blog ,我們可以正常調(diào)用 API 接口了。
3. 構(gòu)建一個(gè) WAR 工程
Jib 還支持 WAR 項(xiàng)目。如果 Maven 項(xiàng)目使用 war-packaging 類型,Jib 將默認(rèn)使用 distroless Jetty 作為基礎(chǔ)鏡像來部署項(xiàng)目。要使用不同的基礎(chǔ)鏡像,我們可以自定義 <container><appRoot>
, <container> <entrypoint>
和 <container> <args>
。以下是使用 Tomcat 鏡像的案例。
- <configuration>
- <from>
- <image>tomcat:8.5-jre8-alpine</image>
- </from>
- <container>
- <appRoot>/usr/local/tomcat/webapps/ROOT</appRoot>
- </container>
- </configuration>