谷歌助力,快速實現 Java 應用容器化

樑桂釗發表於2019-04-15

原文地址:樑桂釗的部落格

部落格地址:blog.720ui.com

歡迎關注公眾號:「服務端思維」。一群同頻者,一起成長,一起精進,打破認知的侷限性。

Google 在 2018 年下旬開源了一款新的 Java 工具 Jib,可以輕鬆地將 Java 應用程式容器化。通過 Jib,我們不需要編寫 Dockerfile 或安裝 Docker,通過整合到 Maven 或 Gradle 外掛,就可以立即將 Java 應用程式容器化。

image.png

開源地址:github.com/GoogleConta…

一、什麼是 Jib

Jib 是一個快速而簡單的容器映象構建工具,它作為 Maven 或 Gradle 的一部分執行,不需要編寫 Dockerfile 或執行 Docker 守護程式。它從 Maven 或 Gradle 中構建我們的 Docker 映象, 並只將發生變更的層(而不是整個應用程式)推送到登錄檔來節省寶貴的構建時間。現在,我們對 Docker 構建流程和 Jib 構建流程進行對比。Docker 構建流程,如下所示。

谷歌助力,快速實現 Java 應用容器化

Jib 構建流程,則是這樣的。

image.png

二、實戰出真知

1. 構建一個簡單的 Java 工程

我們編寫一個簡單的 Java 類。

public class HelloWorld {
    public static void main(String[] args) {
        System.out.println("Hello World!");
        System.out.println("http://blog.720ui.com");
    }
}
複製程式碼

緊接著,我們再建立一個 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>
複製程式碼

由於預設訪問谷歌的 gcr.io 倉庫,而國內訪問 gcr.io 不穩定會經常導致網路超時,所以筆者使用了國內的阿里雲映象服務,那麼就不需要訪問谷歌的倉庫了。現在,我們執行 mvn compile jib:build 命令進行自動化構建,它會從 <from> 拉取映象,並把生成的映象上傳到 <to> 設定的地址。這裡,筆者還通過 ` 設定了一些 JVM 引數。

mvn compile jib:build
複製程式碼

此外,如果"登入失敗,未授權",需要通過 docker login 登入鑑權一下。此外,更好的做法是,你可以考慮在Maven 中放置憑據。

<settings>
  ...
  <servers>
    ...
		<server>
	  	<id>registry.cn-hangzhou.aliyuncs.com</id>
	  	<username>你的阿里雲賬號</username>
	  	<password>你的阿里雲密碼</password>
		</server>
  </servers>
</settings>
複製程式碼

最後,執行完成後,我們可以在阿里雲映象倉庫獲取映象。

image.png

大功告成,現在,我們來驗證一把。我們通過 docker pull 拉取映象,並執行。

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
複製程式碼

執行結果,如下所示。

image.png

2. 構建一個 SpringBoot 的可執行 Jar

我們來一個複雜一些的專案,構建一個 SpringBoot 的專案。關於 SpringBoot 的使用,可以閱讀筆者之前的文章:blog.720ui.com/columns/spr…。現在,我們首先需要搭建一個工程,並建立一個啟動類。

@SpringBootApplication
public class Application {
    public static void main(String[] args) {
        SpringApplication.run(Application.class, args);
    }
}
複製程式碼

同時,需要一個 Web 的介面。

@RestController
public class WebController {
    @RequestMapping("/blog")
    public String index() {
        return "http://blog.720ui.com";
    }
}
複製程式碼

緊接著,我們再建立一個 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>
複製程式碼

現在,我們執行 mvn compile jib:build 命令進行自動化構建。執行完成後,我們可以在阿里雲映象倉庫獲取映象。

image.png

現在,我們再來驗證一把。我們通過 docker pull 拉取映象,並執行。

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
複製程式碼

執行結果,如下所示。

image.png

現在,我們訪問 http://localhost:8080/blog ,我們可以正常呼叫 API 介面了。

image.png

3. 構建一個 WAR 工程

Jib 還支援 WAR 專案。如果 Maven 專案使用 war-packaging 型別,Jib 將預設使用 distroless Jetty 作為基礎映象來部署專案。要使用不同的基礎映象,我們可以自定義 <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>
複製程式碼

三、原始碼地址

原始碼地址:github.com/lianggzone/…

附:參考資料

(完,轉載請註明作者及出處。)

寫在末尾

【服務端思維】:我們一起聊聊服務端核心技術,探討一線網際網路的專案架構與實戰經驗。同時,擁有眾多技術大牛的「後端圈」大家庭,期待你的加入,一群同頻者,一起成長,一起精進,打破認知的侷限性。

更多精彩文章,盡在「服務端思維」!

谷歌助力,快速實現 Java 應用容器化

相關文章