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つの対策を公表

推薦する

VMware が浙江瑞安農村商業銀行を支援

VMware は、VMware Cloud Foundation (VCF) フルスタック クラウド...

ウェブサイトのトラフィックを増やすには、まずキーワードカバレッジをしっかり行う必要があります。

キーワード カバレッジとは何でしょうか? 以前、この用語について多くのウェブマスターと話したとき、そ...

インフルエンサーマーケティングを通じてウェブサイトの SEO を強化するにはどうすればよいでしょうか?

現在、多くの海外企業で最もよく使われているプロモーション戦略の一つが、インフルエンサーマーケティング...

ロングテールキーワードをマイニングするキーワードランキング最適化手法

一部の SEO ウェブサイト、フォーラム、QQ グループでは、ウェブサイトのトラフィックを増やす最も...

左手でタオバオライブストリーミング、右手でショッピング

二科と白象に続き、鳳華は淘宝のライブ放送室で人気を博しているもう一つの国産ブランドとなった。 5月1...

Containerd コンテナ ランタイムを使用して Kubernetes クラスターをデプロイする

[[421025]]前回は、containerd の基本的な使い方を紹介し、Docker コンテナ ...

SEO担当者は、独自のリソースに基づいてウェブサイトのキーワードを設定する必要があります。

ウェブサイトに最も適切なキーワードの数はいくつでしょうか? この質問は実のところ長い間議論されてきま...

hostmybytes - 新しいマシン/CN2ラインVPSの年間支払いは12ドルから、AliPay

hostmybytes から最新のプロモーション メールを送信しました: 元の割引 VPS マシンは...

Ignite 2021 から Microsoft を読み解く: メタバースと AI でさらなる可能性を創造

【51CTO.comオリジナル記事】年末から年明けにかけて、誰もが何らかの総括や予測を立て、2021...

質の高いフレンドリーリンクを判断して交換する方法

現在、多くの SEO 担当者は、高品質の外部リンク (主に友好的なリンク) を非常に重視しており、多...

Webmaster Network からの毎日のレポート: Vancl による Chuke の買収により YouTube はエコシステムのジレンマに陥る

1. Vancl の買収当初: 暖かさを求めて力を合わせるべきか、それとも必要なものを奪うべきか?約...

Gaopeng.com 回想録 3: 火災 50%、誰も安全ではない

編集者注:このセルフメディアでは今週、共同購入ウェブサイトGaopeng.com、Ftuan、Gro...

海剛:Qire123のSEO事例分析

ご存知のとおり、映画ウェブサイトのトラフィックは膨大です。私たち草の根ウェブマスターにとって、大量の...

クラウドネイティブデータベースがデジタルイノベーションの力を発揮

人類が誕生して以来、データの記録と処理能力の探求は止まることはありませんでした。たとえば、古代におけ...

クラウド コンピューティングは製造業をどのように促進するのでしょうか?

クラウドコンピューティングが製造業にどのような力を与えるかを学ぶクラウド コンピューティングは世界中...