はじめに
Dockerを何となくで使っている勢なので、自分なりに少しまとめてみようかなと思いました。
対象読者はある程度Docker知っている人向けです。
タイトルにはゆるーいと書いてますが、少し長いです。。。
前提条件
- Dockerのインストール方法は記載しません
- ターミナルがある程度使えることが前提の内容となります
そもそもDockerって何?
Dockerは仮想化技術の一種という理解です。VMware Workstation PlayerやOracle VM VirtualBoxのようなイメージです。1台のPC上に複数のOSを立ち上げて管理できる、という感じでしょうか。
正確にはVMWareはハイパーバイザ型仮想化、Dockerはコンテナ型仮想化という違いはありますが、イメージ的には同じ感じです。
Docker使うと1台のPC上にDebianの環境を作ることも、Ubuntuの環境を作ることもできるし、簡単に削除もできる。例えば、Linuxの環境が欲しいけど、PCに丸ごとインストールするのはなー、という場合に便利です。
Dockerのメリットは?
世の中でここまで広く普及しているのでたくさんメリットはありますが、簡単にまとめてみます。
同じ環境を簡単に共有できる
例えば、Aさんの環境では動くけど、Bさんの環境では動かない、ということはよくあると思います。
Dockerの場合、Dockerfileというファイルを共有するだけで同じ環境を共有することができます。
たった一つのファイルを共有するだけで同じ環境共有できるってすごくないですか?
CI/CDと相性が良い
CI/CDの説明は省略しますが、CIの中でテストを実行する場合、その環境と同じものをローカルでも用意したい、と思うかと思います。
そうでないと事前に動作確認ができないので。特にCIの環境はLinuxベースのものが多いので、ローカルでの開発でテストを回す場合、CIと同じ環境を用意してテストを実行する、ということができると非常に開発体験が良いです。
Docker Hubが充実している
Docker Hubにいろんなイメージがあるのですが、おそらく大半の人が欲しいイメージがそろっていると思います。
なので、基本的には自分でDockerfileをゴリゴリと記述することなく、自分がやりたいことが実現できることが多いです。
例えば、Tensorflow使いたいなという場合も公式のイメージありますし、Pythonもそれぞれのバージョンのイメージがあります。
とにかく使ってみよう
とりあえず使ってみるのが早い、ということで使ってみましょう。
今回は先日RCが発表されたPython 3.10のイメージを動かしてみます。
イメージのダウンロード
まずは利用するイメージをダウンロードします。
イメージのダウンロードはdocker pull
というコマンドを使います。
$ docker pull python:3.10.0rc1-buster
ダウンロードしているイメージの一覧はdocker images
コマンドで確認します。
$ docker pull python:3.10.0rc1-buster
pythonのイメージがダウンロードされていることを確認して下さい。
補足: イメージに関して
イメージは以下のような構成になっています。
<イメージ名>:<tag>
今回の場合はpythonがイメージ名、3.10.0rc1-busterがタグ名です。
ユーザ作成のイメージの場合はイメージ名の前にユーザ名が入ったりますが、考え方は同じです。
タグ名を変えることでPython 3.9のイメージをダウンロードすることも可能です。
イメージからコンテナを作成する
次に、イメージからコンテナを作成します。
今回はコンテナの詳細な説明は省略しますが、仮想環境上で動くOSみたいなイメージです。
(厳密にはコンテナ型仮想化の場合カーネルはホストOSのものを利用しますので、コンテナ内にカーネルは存在しません)
$ docker create --name python_3_10 -it python:3.10.0rc1-buster
コンテナの作成にはdocker create
コマンドを利用します。
文法は以下です。
docker create --name <コンテナ名、任意> -it <イメージ名>
作成したコンテナを表示します。
$ docker ps -a
以下の画像のように1レコード表示されていればOKです。
先ほど作成したコンテナが作られていますね。
(CONTTAINER IDは画像と異なるかもしれません。)
docker ps
の表示内容について簡単に捕捉します。
- CONTAINER ID
- コンテナごとの固有の識別子。コンテナを指定する際に使用する。例えば削除するときなど。
- IMAGE
- コンテナがどのイメージから作成されたのか
- COMMAND
- コンテナが実行された際に実行されるコマンド
- CREATED
- いつ作成されたのか
- STATUS
- コンテナの状態。作成後、実行中、停止中などがある
- PORTS
- コンテナとホスト間でポートを共有している場合はその情報を表示
- NAMES
- コンテナの名前、コンテナを実行する際などに利用する
作成したコンテナを実行する
では、作成したコンテナを実行してみましょう。
$ docker start -i python_3_10
以下の画像のようにPython 3.10のプロンプトが表示されていますね。
docker ps
を実行した際にCOMMANDには「python3」と記述されていました。
つまり、コンテナを起動するとデフォルトではpython3コマンドが実行される、ということになります。
コンテナの内部に入ってみる
次に、このコンテナ内部に入るために新しいコンテナを作成します。
(pythonのプロンプトはCtrl+Dで閉じることができます)
$ docker create --name python_3_10_bash -it python:3.10.0rc1-buster /bin/bash
実際にコンテナが2つ作成されていることを確認します。
ここで重要なのは新しく作成したコンテナのCOMMANDが「/bin/bash」になっていることです。
つまり、新しく作成したコンテナは起動するとデフォルトで「/bin/bash」が実行される、ということになります。
では、実行してみましょう。
$ docker start -i python_3_10_bash
すると、以下の画像のようにプロンプトが表示され、コマンドを実行することができます。
python3コマンドを実行すると、最初に作成したコンテナと同じ表示がされることがわかります。
なお、コンテナから抜ける場合は、「Ctrl+p Ctrl+q」を利用します。
exitコマンド等を利用するとコンテナ自体の起動が終了するので注意してください。
「Ctrl+p Ctrl+q」で終了した場合は、以下の画像のようにコンテナの状態がUPとなっており、起動中であることがわかります。
起動中のコンテナに再度接続する場合は以下のコマンドを実行してください。
(docker attach
コマンドでも可能、ただし挙動が一部異なる)
$ docker exec -it python_3_10_bash /bin/bash
docker exec
の場合、新しいプロセルが立ち上がるので、exitしてもコンテナ自体は停止しませんが、attachの場合は、直接コンテナのターミナルに接続するため、exitするとコンテナごと停止します。
(下図参照)
中間地点でのまとめ
これまでの内容でDockerイメージのダウンロード、コンテナの作成、コンテナ実行までを体験しました。
コマンドをたたくだけで、Python 3.10の環境を簡単に構築することができたと思います。
今回はすでに作成されているイメージをダウンロードするところから始めました。
次に皆さんが思うのは、そもそもイメージってどうやって作るの?ということだと思います。
では、次はイメージの作り方をみていきたいと思います。
Docker イメージを作ってみよう
Dockerfileを作成する
以下の内容でファイルを作成します
FROM debian:buster
RUN apt update &&\
apt install -y tree
CMD ["/bin/bash"]
ファイルの内容を簡単に解説します。
「FROM debian:buster」の部分で、debian:busterイメージをベースに作る、という定義です。
Dockerfileは必ずベースイメージの定義が必要です。
「RUN」の部分では、コンテナ内で実行するコマンドを記述します。
今回はtreeコマンドのインストールを実行しています。
「CMD」の部分で、コンテナが起動した際にデフォルトで実行するコマンドを記述します。
今回はシェルを実行するようにします。
Dockerイメージを作成する(ビルド)
作成したDockerfileが存在するディレクトリ内で以下のコマンドを実行します。
$ docker build -t custom-image:1.0 -f Dockerfile .
文法は以下です。
docker build -t <イメージ名:ラベル> -f <Dockerfileの名前> <Dockerfileが存在するディレクトリ>
実際にイメージが作成されているかどうかを見てみましょう。
$ docker images
以下の画像のように確かにイメージが作成されていますね。
作成したイメージからコンテナを作成、起動する
先ほどまではdocker create
でコンテナ作成、docker start
でコンテナ起動、としていました。
ただ、docker run
コマンドを利用することで、docker create
とdocker start
両方を一度にすることが可能です。
$ docker run -it --name custom-container custom-image:1.0
Dockerfile内でtreeコマンドをインストールしていましたので、実際に利用できるかも確認します。
以下の画像のようにインストールしたtreeコマンドがきちんと利用できていますね。
また、新しくコンテナが作成されていることも確認できます。
(下図参照)
中間地点でのまとめ
Dockerfileの作成からコンテナの作成までを体験しました。
一番最初にPython 3.10のイメージを利用した際の理解がより深まったのではないでしょうか。
Python 3.10のイメージも今回作成したDockerfileと同じように作成して、Python 3.10が動くようにしている、ということになります。
Docker Hubに公開されている様々なイメージも今回の方法と同じようにして作られています。
つまり、カスタマイズしたDockerfileを共有すれば、それだけで同じ環境を共有できる、という意味が理解できたのではないでしょうか。
おまけ:Docker Composeについて
Dockerと同じくらいDocker Composeについても聞くのではないでしょうか。
Docker Composeも非常に便利なツールなので、簡単に説明します。
Dockerを単体で利用している場合、以下のような使いにくさを感じる時があります。
- コンテナ起動時のオプションがたくさんあって、毎回入力するのめんどくさい
- あるシステムで複数のコンテナを立ち上げる必要があるのだけど、それぞれのコンテナをいちいち立ち上げるのがめんどくさい
安心してください。その悩み、Docker Composeを使えば解決できますよ!
Docker Composeで個人的に便利だと思う部分は以下です。
- コンテナの起動オプションをファイルで管理できる
- 複数のコンテナを一括で起動できる
- 複数のコンテナを一つのファイルで管理できる
ユースケースとしては、ポートの設定や、ボリュームのマウント設定などが利用頻度多いですね。
最近はDocker単体で利用することはほとんどなくて、Docker Composeの利用がほとんどになってきています。
docker-compose.ymlを作ってみよう
Dockerは対応するファイル名がDockerfileでしたが、Docker Composeはdocker-compose.ymlファイルになります。
docker-compose.ymlファイルを作成することが第一歩になります。
以下2つのファイルを作成してください。
- Dockerfile
- docker-compose.yml
FROM debian:buster
RUN apt update &&\
apt install -y tree jq
CMD ["/bin/bash"]
version: "3"
services:
py_3_7_4:
image: python:3.7.4-buster
stdin_open: true
tty: true
volumes:
- type: bind
source: "."
target: "/home"
tree_app:
stdin_open: true
tty: true
build:
context: .
dockerfile: Dockerfile
volumes:
- type: bind
source: "."
target: "/home"
nginx:
image: nginx
stdin_open: true
tty: true
ports:
- "8080:80"
コンテナを起動してみよう
まずはDocker Composeでコンテナを作成してみましょう。
以下のコマンドを実行してください。
※Docker Composeで最初に覚えておく必要があるコマンドはupのみでOKです。
$ docker-compose up -d py_3_7_4
これでPython 3.7の環境のコンテナが作成できました。
では、YAMLファイルの中身を見ていきましょう。
version: "3"
services:
py_3_7_4:
image: python:3.7.4-buster
stdin_open: true
tty: true
volumes:
- type: bind
source: "."
target: "/home"
py_3_7_4がサービス名です。Socker Composeでコンテナを起動するときに利用します。
imageの部分がベースイメージ名です。
stdin_openとttyは両方有効にすることで、コンテナ内での操作が可能になります。
volumesの部分でホストとコンテナ内のボリュームのマウント設定を実施します。
※volumeの設定が無いとコンテナ内部からホストのファイルを参照できません。
ちなみに、上記をDockerのみで実現しようとした場合は、以下のコマンドを実行する必要があります。
$ docker run -it --detach --name py_3_7_4 -v ${PWD}:/home python:3.7.4-buster
Docker Composeを使うと、事前に設定をファイルとして定義しておき、簡単なコマンドを実行するだけで設定済みのコンテナを作成できるので便利です。
自作コンテナを起動してみよう
$ docker-compose up -d tree_app
docker-compose.ymlの内容はほとんど同じですが、buildの部分が異なります。
buildの部分にはDockerfileの場所を指定します。
これで、自分で作成したDockerfileからコンテナを起動することができます。
複数のコンテナを一度に起動してみよう
以下のコマンドを実行します。
$ docker-compose up -d
するとdocker-compose.ymlに定義されているコンテナすべてを一度に起動することができます。
参考リンク集
-
Dockerコンテナが簡単に作れる!【Dockerfile 】サンプル
- Dockerfileの作成方法の参考に
-
チームにdockerを布教することになったので、布教の教材を作ってみた。
- 内容が非常にわかりやすく丁寧に解説されているのでお勧めです
- 最低限これさえ知ってればいいDockerのコマンドまとめ
最後に
Dockerは最近のエンジニアはにとっては必須のスキルの一つになっています。
実際に使ってみると、本当にいろいろな場面で便利に使うことができます。
この記事で少しでも身近に感じて頂ければなと思います。