943MBから6.34kBへ、コンテナサイズを縮小する課題

943MBから6.34kBへ、コンテナサイズを縮小する課題

コンテナは私たちの生活に大きな利便性をもたらし、誰もがコンテナを気に入っています。ただし、コンテナも大量のスペースを消費し、数百メガバイトになることが多く、G を超えるイメージも一般的です。この記事では、コンテナの合理化の事例について学びます。一連の操作により、最終的に画像のサイズを 943 MB から 6.32 k に削減できます。

[[396110]]

概要

コンテナは、オペレーティング ソフトウェアのバージョンとパッケージの依存関係に関連するすべての問題を解決する実用的な方法です。誰もがコンテナを好みますが、コンテナを使用すると、さまざまな大きくて乱雑な画像に直面することになります。スペースが限られている場合は、すぐにいっぱいになります。実際、画像サイズを縮小するためにいくつかの効果的な戦略を使用できます。

基本的な手順

指定されたポートを通じて Web サービスを提供できる HTTP アプリケーション コンテナー。

ボリュームのマウントは実行されません。

当初の計画

ベースライン画像サイズを取得するには、node.js を使用して、index.js へのアクセスのみを提供するシンプルなサーバーを作成します。

index.js コード:

  1. 定数fs = require ("fs");
  2. 定数http = require ('http');
  3. const server = http .createServer((req, res) = > {
  4. res.writeHead(200, { 'コンテンツタイプ': 'text/html' })
  5. fs.createReadStream('index.html').pipe(res)
  6. })
  7. server.listen(ポート, ホスト名, () = > {
  8. console.log(`サーバー: http://0.0.0.0:8080/`);
  9. });

このファイルは、公式の Node ベースイメージに基づいてイメージに組み込まれます。

  1. ノード:14から
  2. コピー 。 。
  3. コマンド ["ノード", "index.js"]

コンパイル

  1. docker build -t cchttp:01 ./

画像サイズは943MBです

合理化されたベースイメージ

画像をスリム化するための最も一般的で、最も単純で、最も明白な戦略の 1 つは、小さいベース画像を使用することです。 Node イメージには、スリム バリアント (Debian ベースですが、プリインストールされた依存関係が少ない) と Alpine Linux ベースの alpine バリアントがあります。

2 つのベース イメージは node:14-slim と node:14-alpine であり、イメージ サイズはそれぞれ 167 MB と 116 MB に削減されます。

Docker イメージは階層化されているため、node.js は多くのレイヤーのイメージに依存する必要があります。現時点では、ソリューションを簡素化する以外にサイズを縮小する方法はありません。

言語を変更

さらに最適化するには、実行時の依存関係が少ないコンパイル言語が必要です。このとき、まず思い浮かぶのは間違いなく静的にコンパイルされる言語である Golang であり、これは一般的で良い選択肢です。 Golang での基本的な Web サービス コードは次のとおりです。

ウェブゴー:

  1. パッケージメイン
  2. 輸入 (
  3. 「fmt」
  4. "ログ"
  5. 「ネット/http」
  6. 関数main() {
  7. ファイルサーバー: = http .FileServer(http.Dir("./"))
  8. http.Handle("/", ファイルサーバー)
  9. fmt.Printf("ポート 8080 でサーバーを起動しています\n")
  10. エラーの場合: = http .ListenAndServe(":8080", nil);エラー != ゼロ {
  11. ログ.致命的(エラー)
  12. }
  13. }

次に、golang の公式ベースイメージを使用して、それをイメージにパッケージ化します。

  1. golang:1.14より
  2. コピー 。 。
  3. go build -o server を実行します。
  4. CMD ["./server"]

Golang ベースのソリューションのイメージ サイズは 818 MB で、それでもまだ非常に大きいです。

分析により、golang 基本イメージに多くの依存パッケージがインストールされていることがわかりました。これらの依存パッケージは、Go ソフトウェアを構築するときに便利ですが、すべてのランタイムに必要なわけではないので、ここで最適化を開始できます。

多段階ビルド

Docker は、必要なすべての依存関係を備えた環境でコードを簡単にビルドし、生成された実行可能パッケージを他のイメージに直接パッケージ化して使用できるようにするマルチステージ ビルド メカニズムをサポートしています。これにより、コンパイル時にはツールとパッケージが必要になるが、実行時にはパッケージは必要なくなるという問題が解決され、イメージ サイズを大幅に削減できます。

注: Docker マルチステージ ビルド メカニズムは、Docker 17.05 で導入された新しい機能です。この機能を使用したい場合は、Docker バージョンを Docker 17.05 以上にアップグレードする必要があります。

マルチステージビルド Dockerfile へ:

  1. ###コンパイル###
  2. golang:1.14-alpine AS ビルダーから
  3. コピー 。 。
  4. go build -o server を実行します。
  5. ###走る###
  6. アルパインより:3.12
  7. コピー--from = builder /go/server ./server
  8. コピー index.html index.html
  9. CMD ["./server"]

  1. Docker イメージ

(⊙o⊙)おお、この戦略は効果的で、生成された画像はわずか13.2MBです。

スクラッチベースイメージと組み合わせた静的コンパイル

13M の画像はすでに非常に優れていますが、他の最適化手法もあります。 Docker の世界には、いくつかのベース イメージが存在します。これは 0 から始まるベース イメージです。このイメージには依存関係がなく、完全に 0 から始まるため、サイズも 0 から始まります。Linux には LFS と呼ばれるディストリビューションがあり、その正式名称は Linux From Scratch で、完全な OS をゼロからコンパイルすることを意味します。このスクラッチベース画像もこれを意味します。

スクラッチ ベース イメージで web.go の実行をサポートするには、コンパイルされたイメージに静的コンパイル フラグを追加して、すべての依存関係を実行中のイメージにパッケージ化できるようにする必要があります。

  1. ### コンパイル ###
  2. golang:1.14 からビルダーとして
  3. コピー 。 。
  4. go build -o server を実行する \
  5. -ldflags "-linkmode 外部 -extldflags -static" \
  6. -ウェブゴー
  7. ###走る###
  8. ゼロから
  9. コピー--from = builder /go/server ./server
  10. コピー index.html index.html
  11. CMD ["./server"]

上記のビルド プロセスでは、コード リンク プロセス中にモードが外部に設定され、-static によって外部リンカーがリンクされます。

最適化後の画像サイズは 8.65MB になります。

究極のキラー - アセンブリ言語

Golang で書かれたプログラムは少なくとも M 程度のサイズがあり、10 MB のイメージは合理化できる限界に達しているはずです。ただし、サイズを大幅に削減するために使用できる他のトリックはありますが、究極のキラーであるアセンブリ言語を使用する必要があります。最終的な解決策は、アセンブリで記述されたフル機能の http サーバー assmttpd を使用することです。そのソース コードは GitHub (github/nemasu/asmttpd) でホストされています。

また、マルチステージコンパイルアプローチを使用して、Ubuntu ベースイメージで依存関係をコンパイルし、それらを Scratch ベースイメージにパッケージ化して実行します。

  1. ###コンパイル###
  2. ubuntu:18.04 からビルダーとして
  3. apt updateを実行する
  4. apt install -y make yasm as31 nasm binutilsを実行します。
  5. コピー 。 。
  6. RUN メイクリリース
  7. ###走る###
  8. ゼロから
  9. コピー--from = builder /asmttpd /asmttpd
  10. コピー /web_root/index.html /web_root/index.html
  11. コマンド ["/asmttpd", "/web_root", "8080"]

結果の画像サイズはわずか 6.34kB です。

次に、このイメージを使用してコンテナを実行します。

  1. docker run -it -p 10080:8080 cchttp:07

curl を使用してアクセスします。

  1. カール -vv 127.0.0.1:10080

要約する

この記事では、コンテナを簡素化するためのさまざまな方法と試みについて説明しました。もちろん、コンテナの機能は単純なので、これらの戦略を実際に直接使用することはできないかもしれませんが、コンテナのチューニングのアイデアの参考として使用できます。

<<:  5 つの主要な分散トランザクションについてどれくらい知っていますか?

>>:  分散オフィス化で4つの大きな変化、3大分析機関が8つの対策を公表

推薦する

raksmart: 米国サンノゼデータセンターのプレミアムネットワークラインVPSの簡単なレビュー

raksmartは、米国の老舗データセンターとしてよく知られているはずです。同社は、CN2 ONLY...

ウェブマスターは、Baidu インデックスの変更に基づいてキーワードの最適化の方向を調整することを学びます

多くのウェブマスターは、ウェブサイトのコアキーワードを決定し始めるときに、Baidu Indexを注...

SEOオンサイト最適化の目的について

SEO は検索エンジン最適化の略で、最適化手法を使用してウェブサイトを検索エンジンに好まれるようにし...

OpenStack は死んだのか?

OpenStack レビューOpenStack は、2010 年 7 月に NASA と Racks...

Ceph分散ストレージシステムの最適化分析

[[409074]]前回の記事「Ceph 分散ストレージ システム アーキテクチャ研究のレビュー」で...

外部リンク判定についてですが、ウェブマスターは署名ファイルを作らないのでしょうか?

検索エンジンの変更はウェブマスターに直接影響します。結局のところ、ほとんどのウェブマスターは検索エン...

江蘇省工業情報化部のチー・ユー副部長と代表団が華雲データグループを視察

3月6日、江蘇省党委メンバーで江蘇省工業情報化部の副部長である池宇氏とその代表団が調査と研究のため華...

鉄道省は12306の入札に関する疑問に答えたが、コストなどのデータは言及されていなかった。

原題: 鉄道省が12306チケット予約システム入札に関する質問に回答入札プロセスは公開されており、費...

spinservers: 米国のハイエンドサーバー、月額 339 ドル、2*E5-2695v4/512gDDR4/4*1.92tSSD/10Gbps 帯域幅

spinservers のダラス データ センターには、最新の CPU モデル (支払い完了後 15...

SEOコンテスト「天吉の競馬」

誰もが天冀の競馬の話を聞いたことがあるでしょう。彼は自分の劣勢の馬で相手の優勢の馬と競争し、優勢の馬...

2007 年上半期 SEO 個人概要 (Baidu 版)

2007 年に Baidu のアルゴリズムが変更されたことは誰もが目撃したと思います。 。初心者が個...

Vultrはどうですか?アトランタ データ センター クラウド サーバー レビュー

Vultr は米国西海岸のユーザーが多すぎて、全体的な使用効果があまり理想的ではないと感じていません...

GDPRとクラウドコンピューティング: 知っておくべきこと

欧州からの新しいデータ規制は、クラウド コンピューティングを使用する企業にいくつかの問題を引き起こす...

#ニュースLinode-KVM社内テストアプリケーション開始/Windowsは世界

Linode の公式フォーラムから衝撃的なニュースが出てきました。Linode は KVM 仮想 V...

スマートホスト: 20 のデータセンター、VPS が 30% オフ、年間 20 ドルから、AMD Ryzen、Windows システムが利用可能

Smarthost (1996 年に仮想ホストとして開始し、1998 年に正式に設立) は現在、すべ...