# Docker 學習筆記
> Docker 部署入門完全指南-圖片速學攻略
# C1
## Q: Docker 是什麼
容器化技術
容器化是可以有效分配 OS 資源的技術。讓 App 獨立
使用場合:部署

* 容器是在作業系統層做虛擬化
* VM 是在硬體層實作
## 部署小歷史
問題起源:硬體主機上會安裝作業系統,會在作業系統上部署多個 App。但現在有一個 App 掛掉就可能會波及其他 App
傳統方式:買多個硬體主機解決,一個 App 就是一台主機
VM 模式:OS 上會有 Hypervisor,Hypervisor 讓我們建立 VM。每個 VM 裡面建立不同的 OS。就可以在各自的 OS 上面建立 App。但每個應用都要起一個 VM 做一個前置動作。
容器模式:OS 上透過 Docker Engine,透過 Docker Engine 建立容器。容器會透過 Docker Engine 跟 OS 要資源(記憶體, CPU...),容器裡放 Apps。耗費的資源只有用軟體建立的虛擬空間 ie 容器。
## 圖解架構

1. 因為要部署,所以先挑一個 OS
2. 為了通用,所以 install docker engine,他會提供一個虛擬化環境(docker container)
3. create docker container,虛擬空間會透過 docker engine 跟 OS 要資源
4. 拿到資源後 docker container 裡面部署 image

> 透過 Docker Container 可以在不同作業環境上共用。docker hub 提供大家雲端空間進行 images 的上傳。
## Docker 功能
1. 簡化部署流程
2. 跨平台部署
3. 建立乾淨測試環境

Note:
* 安裝指令透過 SQL 將測試資料放到資料庫中。
# C2 & C3
## 如何在不同作業系統上使用

結論:
1. Hypervisor 上會建立 VM,目的是要建立出 Linux OS
2. Docker Engine 安裝在 Linux OS 上。安裝好後可以使用它內部功能,起 Container 然後容器裡面放上 App

Note:
* Docker Engine 會跟 Linux 底層的程序 Linux Container 溝通,Linux Container 會管理與分配 Docker Container 的資源
* OS Native 是原生的虛擬化技術,但底層仍在某個位置起一台 virtual box
# C4

1. ==DockerFile 是設定檔==,會吃 source code, base image, other input。執行 dockerFile 後會產生 DockerImage
2. ==Docker Image 是程式部署包==,由 DockerFile 執行指令後產生。
3. ==Docker Container 是虛擬空間==,將程式部署包放入虛擬空間運行,就可以執行程式部署包裡面的內容。==(Docker Image need put into Docker Container)==
# C5 Images 常用語法

```js=
== U1 雲端下載與使用
docker images
docker container run image_name:tag_name
docker pull image_name:tag_name
== U2 本地建立
// by DockerFile 設定檔,產 images
docker build -t 帳號名稱/imageName . // build images,-t 是給 tag,. 表示會看本地當下目錄叫做 Dockerfile 如果有他就會成功建立
// run images
docker container run image_name:tag_name
// images 建立可以給參數
docker build -t angela100151/course-image-build-02 --build-arg my_name_is="angela" .
== U3 本地上傳雲端
docker push angela100151/course-image-build-03
== U4 本地清理
// 如果有被 container reference,要先把 container 砍掉
docker rm container_編號
docker rmi image_name
// 強制清理 images
docker rmi -f image_name
```
unable to remove repository reference "angela100151/course-image-build-02" (must force) - container 58d2e57b016c is using its referenced image dd48cb4ec2ed
### 小結
| 指令 | 目的 | 其他 |
|-------------------|-------------------------------------------------------|-------------------------------------------------------|
| `docker images` | 檢視有哪些映像檔(images)。 | |
| `docker pull image_name:tag_name` | 從遠端倉庫下載映像檔到本地。 | |
| `docker push image_name:tag_name` | 將本地映像檔上傳到遠端倉庫。 | |
| `docker build -t account_name/image_name` | 根據 Dockerfile 建立映像檔,並指定標籤(tag)。 | images 建立可以給參數 |
| `docker build -t angela100151/course-image-build-02 --build-arg my_name_is="angela" .` | 根據 Dockerfile 建立映像檔,同時提供建構參數 `my_name_is` 的值為 "angela"。 | |
| `docker container run image_name:tag_name` | 執行指定的映像檔建立容器(container)。 | |
| `docker rmi -f image_name` | 強制刪除指定的映像檔,同時刪除相關聯的容器。 | |
| `docker rm container_編號` | 刪除指定編號的容器。 | |
| `docker rmi image_name` | 刪除指定的映像檔,如果有容器在使用該映像檔,則需要先刪除相關聯的容器。 | 有被容器參照時,需先移除相關聯的容器 |
# C6 Container 常用語法
```js=
docker container ls
// 起一個新的 container 跑抓下來的 alpine
docker pull alpine
docker images
docker container run --name c003 alpine ls /
docker run --name c004 alpine ls / // 起一個容器叫做 c001 使用 apline 這個 images 來跑
docker run -it --name c005 alpine /bin/sh
// 讓 contaier 長期可以在背景程式跑
docker run -d --name c006 alpine tail -f /dev/null
// 進入背景程序的 container 做一些事情
docker exec -it 0140fc72d655 /bin/sh
== 練習 docker container run
docker pull nginx:latest
docker run -d -p 8081:80 --name c007 nginx // -p p 是 port 的意思
echo $(docker-machine ip) // http://127.0.0.1:8081/
== 清空 container
container stop container_id
docker container ls -a // 列出所有 container 不管有無啟動
docker container rm container_id // 將 container 全部清光
```
* `docker container run --name c001 alpine` 跑起來這個 images 後,就可以用這個 images 提供的指令

不會在背景程序一直跑,因為 container 背景運作程序機制是他沒有一個程序要一直跑,他會把自己給清掉
`-it` interactive mode

`tail -f` 會追蹤某個 file 的 log
`/dev/null` 是一個特殊的文件,它會丟棄所有寫入其中的數據

vm linux port 8081 對應 container port 80 也就是 ngnix web server 所使用的 port
# ☆ ☆ ☆ C7
當使用 docker file 建立 docker image 時,背後整體流程會怎麼運作?

docker 架構圖應為以下

但為了方便說明,簡化成以下說明

* Docker Client 是下指令的 docker terminal
* Build Context 是下 `docker build .` 指令時當下的目錄,目錄裡面一定會有 Dockerfile and Other Files
VM 起的 Linux Os 上面會安裝 Docker Engine
當 Docker Engine 接收到 `Docker build .` 第一步會先將 macOs 上的 build context copy 到 linux os 裡面
> Docker Engine 裡面又會有不同的世界
兩個工作者
1. Docker Client: docker terminal execute docker 指令
2. Docker Engine:docker 執行的指令都會送到 Docker Engine 上
三個空間
1. 硬體主機上的 mac OS 空間
2. 虛擬主機上建立的 Linux OS 空間
3. Docker Engine 建立起的臨時 Container 的空間
## FROM
alpine 輕量 linux
* cat 印出檔案內容
* 有了 Dockerfile 就可以建造 image -> `docker build -t <映像檔名稱:標籤> <Dockerfile所在目錄路徑>`
* 有了 images 可以起 container 跑他 -> `docker run angela100151/001`,可以使用這個 images 提供的所有指令
```dockerfile=
cat dockerfile
docker build -t angela100151/001 .
docker images
docker run angela100151/001
# 看到作業系統的版本
docker run angela100151/001 cat /etc/os-release
# /bin/sh 進入到 container
docker run angela100151/001 /bin/sh
# 進入互動模式
docker run -it angela100151/001 /bin/sh
# container 在背景不斷運行
docker run -d angela100151/001 tail -f /dev/null
# 檢視目前的 docker container 有哪些
docker container ls
# 進入 container
# 因為已經有 container 所以這邊要寫的是 container id
docker exec -it 302385395fd6 /bin/sh
```

docker 沒事做,就會自己不見
| 參數 | 說明 |
|----------------|------------------------------------------------------------|
| `docker run` | 啟動一個新的容器實例。 |
| `-d` | 在背景運行容器(分離模式)。 |
| `angela100151/001` | 要運行的Docker映像(容器的基本映像)。 |
| `tail -f /dev/null` | 在容器內運行的命令,這裡是讓容器保持運行但不執行實際操作,常用於保持容器處於活動狀態。|
這個指令的作用是以分離模式在背景中運行一個Docker容器,使用`angela100151/001`映像,並運行`tail -f /dev/null`命令來保持容器運行,而不執行實際的操作。這種技巧常用於需要容器保持活動狀態,但不需要執行實際應用程序的情況,例如網絡伺服器等。
上述做的事情都是 Dockerfile 裡,FROM 提供我們基本的語法
```dockerfile=
# 找baseImage的來源
FROM alpine:latest
```
## ENTRYPOINT
`ENTRYPOINT ["tail", "-f", "/dev/null"]`
用途:啟動 images 時,預設語法要跑什麼
定義:DOCKER RUN 某 IMAGE,這個 IMAGE 會第一個執行的指令
語法:`ENTRYPOINT ["executable", "param1", "param2"]`
```dockerfile=
# 在 image 後面手動 key 了這個語法,讓他可以在背景程式後面一直跑
docker run -d angela100151/001 tail -f /dev/null
# 如果想提供這個 image 給他人用,但又不希望他一定要打上 tail -f /dev/null 這個語法
# 可以透過 ENTRYPOINT 的 Dockerfile 語法給予他預設的啟動指令
docker build -t angela100151/002
# 不需要再加上 tail -f /dev/null
docker run -d angela100151/002
```
## Run
進入到指定的 container 中,起一個 apache server
因為要能夠連到裡面,所以要做 port mapping
docker building 有三空間:本地 OS 空間,VM Linux 空間,Container 空間
8080 指的是 VM Linux 空間,80 指的是 Container 空間裡面的 port
```dockerfile=
docker run -d -p 8080:80 angela100151/002
docker container ls
# 進入這個 contaier
docker exec -it 71e10a8a5a52 /bin/sh
# 以下指令是為了安裝 apache server 的指令
# apk is alpine 安裝套件指令
apk --update add apache2
# clear cache
rm -rf /var/cache/apk/*
# 啟動 apached server
httpd -D FOREGROUND
# check docker-machine ip 找到 linux vm ip 位置,他會幫我們對照 container 裡面的 80 port
docker inspect <container id> | grep "IPAddress"
echo $(docker-machine ip)
# http://127.0.0.1:8080/
# 將上述打包的過程在封裝一層,讓使用更方便
```
* Run 用途:使用任意 base image 提供的指令
* 時間點:在 image building 時被執行,而非跑 container 時被執行
* 效果:在 dockerFile building 過程中,使用 base image 所提供的指令
```dockerfile=
FROM alpine:latest
RUN apk --update add apache2
RUN rm -rf /var/cache/apk/*
ENTRYPOINT [ "httpd", "-D", "FOREGROUND" ]
```
```dockerfile=
docker run -d -p 8082:80 angela100151/004
```
> 有這三個語法基本上可以組出全世界 FROM RUN ENTRYPOINT
## ENV 語法:建立共同變數
想對 apache server 寫一些首頁內容
對每個 run 指令,Docker Daemon 都會起臨時性的 container 並將他包成一包新的 image,再往下
兩個 run 指令之間是沒有相通,run 會起一個臨時性的 container


```dockerfile=
FROM alpine:latest
RUN apk --update add apache2
RUN rm -rf /var/cache/apk/*
RUN cd /var/www/localhost/htdocs
RUN pwd
ENTRYPOINT [ "httpd", "-D", "FOREGROUND" ]
# 加上 && 就會讓 Run 一次跑兩個指令
FROM alpine:latest
RUN apk --update add apache2
RUN rm -rf /var/cache/apk/*
RUN cd /var/www/localhost/htdocs && pwd
ENTRYPOINT [ "httpd", "-D", "FOREGROUND" ]
# run 裡面就可以讓他不同指令但在同一行跑,使用 \
FROM alpine:latest
RUN apk --update add apache2
RUN rm -rf /var/cache/apk/*
RUN cd /var/www/localhost/htdocs \
&& echo "<h3>I am Angela. I am taking this great course. Round 01</h3>" >> index.html
RUN cd /var/www/localhost/htdocs \
&& echo "<h3>I am Angela. I am taking this great course. Round 03</h3>" >> index.html
RUN cd /var/www/localhost/htdocs \
&& echo "<h3>I am Angela. I am taking this great course. Round 02</h3>" >> index.html
ENTRYPOINT [ "httpd", "-D", "FOREGROUND" ]
```
`docker run -d -p 8080:81 angela100151/005`
確認首頁信息已成功被更動
`http://127.0.0.1:8080/`
每個 run 都是自己的世界,都要透過 cd 進到自己的資料夾
現在想將共同的部分統一寫
```dockerfile=
FROM alpine:latest
ENV myworkdir /var/www/localhost/htdocs
RUN apk --update add apache2
RUN rm -rf /var/cache/apk/*
RUN cd ${myworkdir} \
&& echo "<h3>I am Angela. I am taking this great course. Round 01</h3>" >> index.html
RUN cd ${myworkdir} \
&& echo "<h3>I am Angela. I am taking this great course. Round 03</h3>" >> index.html
RUN cd ${myworkdir} \
&& echo "<h3>I am Angela. I am taking this great course. Round 02</h3>" >> index.html
ENTRYPOINT [ "httpd", "-D", "FOREGROUND" ]
```
* `\` 表示換行符號,在 Dockerfile or shell 可以這麼撰寫
* `>>`是重定向符號,用於將命令的輸出附加到文件中
* `cat` 顯示文件 echo 顯示文本
`docker run -d -p 8080:81 angela100151/005`
## workdir:指定預設目錄的位置
想將重複的 cd 過程模組化
用途:自動將預設路徑 cd 到指定的路徑
```dockerfile=
FROM alpine:latest
ENV myworkdir /var/www/localhost/htdocs
WORKDIR ${myworkdir}
RUN apk --update add apache2
RUN rm -rf /var/cache/apk/*
RUN echo "<h3>I am Angela. I am taking this great course. Round 01</h3>" >> index.html
RUN echo "<h3>I am Angela. I am taking this great course. Round 03</h3>" >> index.html
RUN echo "<h3>I am Angela. I am taking this great course. Round 02</h3>" >> index.html
ENTRYPOINT [ "httpd", "-D", "FOREGROUND" ]
```
`docker run -d -p 8080:80 angela100151/006`
## ARG
希望將 Angela 變成動態的變數,讓我在建立 images 時去改變它
ARG 與 ENV 差異在他可以在 docker build 的時候動態改變他的變數
```dockerfile
FROM alpine:latest
ENV myworkdir /var/www/localhost/htdocs
ARG whoami=Angela
WORKDIR ${myworkdir}
RUN apk --update add apache2
RUN rm -rf /var/cache/apk/*
RUN echo "<h3>I am ${whoami}. I am taking this great course. Round 01</h3>" >> index.html
RUN echo "<h3>I am ${whoami}. I am taking this great course. Round 03</h3>" >> index.html
RUN echo "<h3>I am ${whoami}. I am taking this great course. Round 02</h3>" >> index.html
ENTRYPOINT [ "httpd", "-D", "FOREGROUND" ]
```
建立新的 image
```dockerfile
docker build --build-arg whoami=Tony -t angela100151/007
```
## Copy: 將外界檔案資料移到 Container 中
linux vm 空間複製一份到 container 當前目錄底下。container 當前目錄定義在 workdir 這邊
```dockerfile!
FROM alpine:latest
ENV myworkdir /var/www/localhost/htdocs
ARG whoami=Angela
WORKDIR ${myworkdir}
RUN apk --update add apache2
RUN rm -rf /var/cache/apk/*
RUN echo "<h3>I am ${whoami}. I am taking this great course. Round 01</h3>" >> index.html
RUN echo "<h3>I am ${whoami}. I am taking this great course. Round 03</h3>" >> index.html
RUN echo "<h3>I am ${whoami}. I am taking this great course. Round 02</h3>" >> index.html
COPY ./content.txt ./
RUN ls -l ./
RUN cat ./content.txt >> index.html
ENTRYPOINT [ "httpd", "-D", "FOREGROUND" ]
```
## HW
## 將任意 Container 變成 Docker Image

```dockerfile!
docker build -t angela100151/myimage
docker run -d -p 8080:80 angela100151/myimage
# 進入 container 更動首頁頁面
docker exec -it 6c146aff59ff /bin/sh
echo "I am going to turn this container into n new image" >> index.html
cat index.html
exit
# 將運行中的 container 轉換成 image
docker commit 6c146aff59ff angela100151/containertoimage
# 檢視新增的 image
docker images
# 檢視所有 container
docker container ls
# 停掉並砍掉
docker stop 6c146aff59ff
docker rm 6c146aff59ff
# 確定剛剛 container 有成功轉換成 image
docker container run -d -p 8080:80 angela100151/containertoimage
```

# C8 Docker 網路模式

三種空間有各自對應的網路空間
| Space | Network |
| --------------- | ----------------- |
| host | Host Network |
| VM | Linux VM Network |
| Docker Engine | Bridge Networks |
Bridge Networks 的四種模式
1. None 模式:容器與外界沒有任何接觸。容器無法外連,外面也連不進去。(封閉網路空間)
2. Bridge 模式:容器啟動時,會放入某個 Bridge Network 中,在 Bridge Nerwork 裡面有所謂的 IP,容器會跟他要 IP 來使用。不同容器要屬於同一個網路空間才能互通。
3. Container 模式:起一個 Container,對應現行的 Container。
4. Host 模式:與 Linux VM 要 IP。所可以跟 VM 虛擬主機上的所有應用互通網路。
A1 B1 B2 都是 IP
Remark:
* Docker 的世界上有多種 Network
## Docker 網路的 None 模式
1. 起一個新的 container 指定使用 none 的 network 模式
2. 確認 container 無法跟外界溝通
```dockerfile=
#列出所有 network 模式
docker network ls
# 起一個 none 模式的 container
docker run -d --network none --name none-mode alpine tail -f /dev/null
# 檢視 none 模式底下是否有剛剛建立好的 container
docker network inspect none
# 連進去剛建立好的 container 試圖讓他跟外界溝通
docker exec -it 199e4dac31f4 /bin/sh
ip addr ls
ping 8.8.8.8
```

## Docker 網路的 Bridge 模式
1. 建立 bridge 的網路空間
2. 檢視 bridge network 的資訊
3. 建立新的 container 放到這個網路空間中
4. 連入這個 container
5. 確定 bridge mode 可以連到外網
6. 建造第二個 container 放到 bridge 裡面
7. 測 2 個 container 能否互通
8. 建造第三個 container 放到不同的 bridge 中
不同 bridge 之間的 container 是不能互通
```dockerfile=
# check current network space
docker network ls
# create personal network space using bridge mode
docker network create --driver bridge my-bridge
# check
docker network inspect my-bridge
# build new container put them to my-bridge network
docker run -d --network my-bridge --name bridge-mode-001 alpine tail -f /dev/null
# inspect my-bridge
docker network inspect my-bridge
# go into this container to see its network
docker exec -it 06d6334aefc8b415773d964aeea81e34f634cb4560ee08c2cca5445872663924 /bin/sh
ip addr ls
ping 8.8.8.8
# similarly, create second container and put it to the my-bridge network
docker run -d --network my-bridge --name bridge-mode-002 alpine tail -f /dev/null
docker container ls
docker network inspect my-bridge
ping 8.8.8.8
# 02 & 03 可以成功互通
ping 172.18.0.2
# create third container put it to the different container space
docker network create --driver bridge thier-bridge
docker network ls
docker run -d --network thier-bridge --name bridge-mode-003 alpine tail -f /dev/null
docker network inspect thier-bridge
docker exec -it d55431026bf65bcb31bb84680fdc9725d2816dae9ebbe6bb5d7a042f7b8c2a66 /bin/sh
# ping 不過去,因為不同 birdge 網路空間不互通
ping 172.18.0.2/16
# 將 03 加入到 02 網路空間
# connect 後面加上 bridge 名稱與要移過去的 container 名稱
docker network connect my-bridge bridge-mode-003
# 003 現在有兩個 ip
docker network inspect thier-bridge
docker exec -it d55431026bf65bcb31bb84680fdc9725d2816dae9ebbe6bb5d7a042f7b8c2a66 /bin/sh
ping 172.18.0.2/16
# 確定這三個網路空間都互通
```
bridge 內的網路空間

我們是從 my-bridge 這個網路空間中拿一個 ip 給 001 用

thier-bridge 內的網路空間

```text=
my-bridge: 172.18.0.0/16
bridge-mode-001: 172.18.0.2/16
bridge-mode-002:172.18.0.3/16
bridge-mode-003:172.18.0.4/16
their-bridge:172.19.0.0/16
bridge-mode-003:172.19.0.2/16
```
## Docker 網路的 Container 模式
```dockerfile=
# 確定目前有的 container
# container:container_名稱
# 表示要拷貝那一個 container 的網路設定
docker run -d --network container:bridge-mode-001 --name container-mode-001 alpine tail -f /dev/null
# container 沒有看到新創建的 container-mode-001 因為 container mode 是拷貝現有的網路設定,他不會新增任何東西
docker network inspect my-bridge
ping 8.8.8.8
ping 172.18.0.3
```
1. 起一個新的 container
container mode 是拷貝現有的網路設定,他不會新增任何東西
```text=
my-bridge: 172.18.0.0/16
bridge-mode-001: 172.18.0.2/16
bridge-mode-002:172.18.0.3/16
bridge-mode-003:172.18.0.4/16
their-bridge:172.19.0.0/16
bridge-mode-003:172.19.0.2/16
container-mode-001: 172.18.0.2/16
```
## Docker 網路的 Host 模式
host mode 就是讓 container 的網路世界與 VM 相同
container 在網路中不再屬於 container 那層,而是被當成與 linux VM Host 相同。
```dockerfile=
# 沒有 -p 是因為 host 模式將 container 層級拉到跟 vm 一樣的地方
docker run -d --network host --name my-apache angela100151/my-apache
docker inspect host
# container 中對外開放的 port 是什麼
netstat -tulpn
# tcp 0 0 :::80 :::* LISTEN 1/httpd
```
# Docker Volume

Container Disk 空間出現的資料是暫時的,Container 消失資料就會不見
VM 空間會被保存下來。我們可以建立 container 去使用 VM 空間的東西,儘管後來將 container 砍掉,存在 VM 空間的資料還是不會消失。
## 實作
1. 建立 images
2. 跑建立好的 images
3. 進入 container 中,看到檔案內容
4. 關掉與清掉 container
5. 重新起 container
Mountpoint 是 Linux VM 的路徑
將 VM Disk 空間與 Container Disk 空間 mapping
如果沒有指定 volume,Docker Engine 會自動幫我們建立一個 value
```dockerfile=
# create image
docker build -t angela100151/apache-001 .
# run container
docker run -d -p 8080:80 angela100151/apache-001
# enter this container
docker exec -it 2f007e6edd6c /bin/sh
echo 'I love u' >> index.html
# clear this container
docker container stop 2f007e6edd6c
docker container rm 2f007e6edd6c
# 重新啟動 container
# 剛剛的 I love u 已經不見了
docker run -d -p 8080:80 angela100151/apache-001
# use container volume to protect history data
# check current volume
docker volume ls
# create new volume
docker volume create mainpage-vol
docker volume inspect mainpage-vol
# 將 vm volume 路徑 "Mountpoint": "/var/lib/docker/volumes/mainpage-vol/_data", map 到 container 裡面的路徑
# vm 空間:container 空間 image 名稱
# 這邊會去 map 兩個空間,讓他們的改動都可以互相連動
docker run -d -p 8081:80 -v mainpage-vol:/var/www/localhost/htdocs/ angela100151/apache-001
# 進入這個 container 對首頁進行一些修改
docker exec -it 0e8cc0f21b70 /bin/sh
echo 'I love u so much' >> index.html
# clear container
docker container stop 0e8cc0f21b70
docker container rm 0e8cc0f21b70
# 重啟這個 container 發現 volume 是可以真的保存這個 container 的資訊
```

沒有 -v 後面接上指令的 volume,docker engine 會自動幫我們建立一個亂數的 volume

# 大管家 Docker Compose
透過 Docker Compose 可以將 service network volume 統整起來一起處理

1. 將 service network volume 統整
2. 透過 docker compose 可以一次起多個 container
## Docker Compose 實作 Services(Container)
```yaml=
# 建造 docker-compose 設定檔
# version 語法要用 3.7 版本
version: "3.7"
# services 部分就是 container 部分
services:
myweb: # container name
build:
context: . # dockerFile 所在的目錄位置
args: # dockerFile 中的 args
name: '蠟筆小新'
image: angela100151/myweb:latest
ports:
- "8080:80"
```
```dockerfile=
# no cache 會讓他每次 build 都跑一次,避免沒有更新到
docker-compose build --no-cache
# 透過 compose 起 container
# -d 是指放在背景執行
docker-compose up -d
docker container ls
# 停止並移除由 docker-compose up 建立的容器、網路
docker compose down
```
```yaml=
# docker compose 可以起多個 container
# version 語法要用 3.7 版本
version: "3.7"
# services 部分就是 container 部分
services:
myweb: # container name
build:
context: . # dockerFile 所在的目錄位置
args: # dockerFile 中的 args
name: '蠟筆小新'
image: angela100151/myweb:latest
ports:
- "8080:80"
myweb2: # container name
build:
context: . # dockerFile 所在的目錄位置
args: # dockerFile 中的 args
name: 'Angela'
image: angela100151/myweb2:latest
ports:
- "8081:80"
myweb3: # container name
build:
context: . # dockerFile 所在的目錄位置
args: # dockerFile 中的 args
name: 'Tom'
image: angela100151/myweb3:latest
ports:
- "8082:80"
```
```dockerfile=
# build images
docker-compose build --no-cache
# make sure images
docker images
# run docker container
docker-compose up -d
# close container
docker-compose down
```
```yaml=
# 從現有 image 起 container
# version 語法要用 3.7 版本
version: "3.7"
# services 部分就是 container 部分
services:
myweb: # container name
build:
context: . # dockerFile 所在的目錄位置
args: # dockerFile 中的 args
name: 'Bob'
image: angela100151/myweb:latest
ports:
- "8080:80"
myweb2: # container name
build:
context: . # dockerFile 所在的目錄位置
args: # dockerFile 中的 args
name: 'Angela'
image: angela100151/myweb2:latest
ports:
- "8081:80"
myweb3: # container name
build:
context: . # dockerFile 所在的目錄位置
args: # dockerFile 中的 args
name: 'Tom'
image: angela100151/myweb3:latest
ports:
- "8082:80"
myweb4: # container name
image: angela100151/myweb:latest
ports:
- "8083:80"
```
## Docker Compose 實作 Networks
using network property
```yaml=
# version 語法要用 3.7 版本
version: "3.7"
# services 部分就是 container 部分
services:
myweb: # container name
build:
context: . # dockerFile 所在的目錄位置
args: # dockerFile 中的 args
name: 'Bob'
image: angela100151/myweb:latest
ports:
- "8080:80"
networks:
- mybridge001
myweb2: # container name
build:
context: . # dockerFile 所在的目錄位置
args: # dockerFile 中的 args
name: 'Angela'
image: angela100151/myweb2:latest
ports:
- "8081:80"
networks:
- mybridge001
myweb3: # container name
build:
context: . # dockerFile 所在的目錄位置
args: # dockerFile 中的 args
name: 'Tom'
image: angela100151/myweb3:latest
ports:
- "8082:80"
networks:
- mybridge001
myweb4: # container name
image: angela100151/myweb:latest
ports:
- "8083:80"
networks:
- mybridge002
networks: # 透過 networks 建立客製化網路空間
mybridge001:
mybridge002:
```
```dockerfile=
# 將所有 container 背景執行
docker-compose up -d
# check network
docker network ls
# build images
docker-compose build --no-cache
# 查看透過 network 建立客製化 bridge 空間
docker network inspect docker_mybridge001
docker network inspect docker_mybridge002
```
## Docker Compose 實作 Volumes
同前面步驟,這次新增 myweb5
```yaml=
myweb5: # container name
image: angela100151/myweb:latest
ports:
- "8084:80"
networks:
- mybridge002
volumes:
- mainpage-vol002:/var/www/localhost/htdocs/ # mainpage-vol002 座落在 linux vm 空間將他 map 到 container 的這個目錄底下
networks: # 透過 networks 建立客製化網路空間
mybridge001:
mybridge002:
volumes:
mainpage-vol002:
```
## Infrastructure as Code
> 讓整體 IT 環境濃縮在單一設定檔案。透過設定檔一鍵部署就可以起完整個 IT 環境。

作法
* 容器化方式部署多個 container :以 docker compose 前後端部署為例,每個 container 進行前端後端資料庫的部署。IT 環境變成三個 container 各自部署各自的應用程式
* 雲端進行部署:以 AWS 為例,撰寫 Colud Formation 設定檔,就可以在 AWS 上部署多個 VM

# C11 應用

角色
* web:前端
* app:後端
* db:資料庫
File
.env:環境參數檔案
將各個專案的 source code 打包成自己的 image,然後放到容器跑
```dockerfile=
# Build or rebuild services
# 有改動 docker-compose & DB 測試資料語法,就需要再重新跑一次
docker-compose build --no-cache
# Create and start containers
docker-compose up -d
# 將本地 image 通通砍掉
docker rmi $(docker images -a -q)
# 建立一個 CPU 規格的 image
docker-compose push
# 可以一次創建多個 CPU 規格的 image
docker buildx create --use --name mybuilder
```

將 volume 拿掉,所以每次容器啟動與關掉,DB 資料不會留著。
* mac 的 docker 跨平台部署要考慮 CPU 架構
不同 CPU 架構要考慮其 image
```dockerfile=
# 將 compose yaml 設定檔,上傳到 dockerhub 上
# 這個指令目前無法完整運作,只會建立一個 CPU 規格的 image
docker-compose push
# rm all images
docker rmi $(docker images -a -q)
# docker buildx 會創建多個 CPU 規格的 image
docker buildx create --use --name mybuilder
# assing CPU 規格
# --push dockerFile_位置
docker buildx build --platform linux/arm64,linux/adm64 -t angela100151/mysql-db-01 --push ./
```
* 跨平台部署示範
已經從雲端抓下 images,就不用再 build。這時只要 `docker-compose up -d`
```dockerfile=
# 抓下 dockerhub image
docker pull angela100151/image_name
# 一次抓下 docker-compose 裡面的 images 們
docker-compose pull
```


# C12 規模化部署
## docker swarm

當這些 docker engine 進入 swarm mode 之後,會變成 Node。Node & Docker engine 是一樣的東西,只是稱呼不同而已。Node 之間有不同角色,最重要的角色為 Manager Node。
* Manager Node:統整 Node 之間的溝通
* 實際運作說明:首先會給 Manager Node 一個叫 service 的東東,service 由 image & 啟動指令組成。這次要執行哪個 docker image,當這個 image 在 container 啟動要執行哪個指令先。將 image 交給 service 執行後會啟動多個 tasks,每個 tasks 都會使用同個 service。
* Manager Node 的目的是維持目標狀態,tasks 類似 slot 需要被實際運作。Work Node 會幫助 task 運作。不同 task 有可能分配到相同 Work Node 執行。當 tasks 都成功執行,Manager Node 就認定已經到達目標狀態。
* Manager Node 不在乎在哪台主機或是哪個 Work Node 去執行,他只在乎 tasks 有無人去執行
* Worker Node:實際執行。每個 Worker Node 會向 Manager Node 註冊,加入 Manager Node 的管理。被指派的 Worker Node 會起一個全新的 Container 去執行 Task,並將執行完的結果向 Manager Node 會報。
## 規模化部署方案比較

## GKE 實作
名詞解釋
* Cluster:Cluster Manager。統籌與運算資源
* Work Load: 可以客製化 Workload including dockerFile, docker images, work numbers
* Services & Ingress:對外開放 containers
* Node Pool:Work Node。
定義工作包讓 Node 有事情做
* Edit Container:要用哪個 image,可用既有或用新的
* Dockerfile 在根目錄就不用特別指定位置
* Exposing services:對外開放,做 port mapping
Docker 的世界中 container 是實際去執行的單位。k8s 在 container 上用多蓋了一層叫 pods。一個 pods 可以有多個 containers,要做的事情都是一樣的,將 tasks 領下來做掉。
# C13
# podman & docker 的優缺比較

所有 docker 指令都是透過 docker daemon 接手做下去。docker daeom 需要有 root 權限,這樣才可以在任意 port & root 做操作。
docker 底層是 containerd runtime
podman 一樣會起 podman client 可是中間不需要一個 daemom 長期執行,以開發者權限執行這些指令。
每個 podman 指令會叫起不同的 container
podman 底層使用 OCI
Q:為什麼 podman 會取代 docker?
1. 安全性: 駭客攻擊 root 取得權限就可以操作 docker
2. single point failre:docker daemomn
3. podman 生態系:podman 類似 module 概念,周邊有很多 plugins
# k8s
* docker 貨櫃船上的貨櫃:將所有應用程式需要的東西打包成一包
* k8s 掌舵者(統整者):將貨櫃打理放到合適的位置,將貨櫃運出去(部署出去)
k8s核心概念 + 雲端
學習歷程



# k8s & docker 之間的關係
k8s 作為掌舵者,需要兼容不同的容器執行環境。他建立 CRI 要求所有人配合這個介面。
因為 CRI 晚於 docker 所以並沒有兼容 docker。
透過 docker-shime 將 dokcer 兼容於 CRI。
為了要維護 docker-shime 成本太大,所以現在宣布不維護 docker-shime

不同的容器執行環境底層都符合 OCI 規範,因為底層規範相同,所以製作出的 images 是可以被所有 runtime 執行。
影響
* 開發者:還是可以用 docker image
* 維運者:要去尋找不同的 run time
