這是一個建立於 的文章,其中的資訊可能已經有所發展或是發生改變。
序
很多時候我們會用Docker進行部署,其實它還可以用於開發。
為什麼要在開發中使用Docker?
主要有以下幾個原因:
- 一致的開發環境 使用Docker,可以保證整個研發團隊使用一致的開發環境。
- 開發環境與最終的生產環境保持一致 這減少了部署出錯的可能性。
- 簡化了編譯和構建的複雜性 對於一些動輒數小時的編譯和構建工作,可以用Docker來簡化。
- 在開發時只需Docker 無需在自己的開發主機上搭建各種程式設計語言環境。
- 可以使用同一程式設計語言的多個版本 可以使用同一程式設計語言(如python, python, ruby, ruby, java, node)等的多個版本,無需解決多版本衝突的問題。
- 部署很簡單 應用程式在容器中運行,和在生產環境中部署運行是一樣的。只需打包你的代碼並部署到帶有同樣鏡像的伺服器上,或者是把代碼連同原鏡像建立一個新Docker鏡像再直接運行新鏡像。
- 使用自己喜歡的開發IDE 仍然可以繼續使用自己喜歡的開發IDE,無需運行VirtualBox虛擬機器或SSH。
安裝Docker並驗證
➜ tonny@tonny-pc ~ sudo dpkg --get-selections |grep linux➜ tonny@tonny-pc ~ cat /etc/issue➜ tonny@tonny-pc ~ dpkg -l | grep aufs➜ tonny@tonny-pc ~ sudo apt-get install -y aufs-tools cgroupfs-mount mountall➜ tonny@tonny-pc ~ sudo curl -sSL https://get.docker.com/ | sh➜ tonny@tonny-pc ~ sudo usermod -aG docker $(whoami)➜ tonny@tonny-pc ~ systemctl restart docker➜ tonny@tonny-pc ~ ps aux |grep `cat /var/run/docker.pid`➜ tonny@tonny-pc ~ sudo docker versionClient: Version: 1.12.0 API version: 1.24 Go version: go1.6.3 Git commit: 8eab29e Built: Thu Jul 28 21:40:59 2016 OS/Arch: linux/amd64Server: Version: 1.12.0 API version: 1.24 Go version: go1.6.3 Git commit: 8eab29e Built: Thu Jul 28 21:40:59 2016 OS/Arch: linux/amd64
怎樣在開發中使用Docker?
採用Docker開發與普通開發不同之處有兩點:
- 確保所有的依賴都放入了工作目錄 。
- 修改構建和運行命令,使之在Docker容器中可以運行。
對於Web應用:
如果正在開發Web應用,需要開放連接埠,只需用-p參數讓Docker運行時開放連接埠即可。
對於Java語言:
把所有所需的依賴JAR包都放入工作目錄,然後在容器中編譯器並運行它。 假設一個例子程式是Hello.java,它使用了gson-2.2.4.jar依賴和json-java.jar依賴。 那麼整個構建命令如下:
docker run --rm -v "$(pwd)":/app -w /app java sh -c 'javac -cp "json-java.jar:gson-2.2.4.jar" Hello.java'
要運行和測試這個程式,可以執行:
docker run --rm -v "$(pwd)":/app -w /app java sh -c 'java -cp gson-2.2.4.jar:json-java.jar:. Hello'
對於Golang語言
把Golang開發的代碼複製到GOPATH指定的位置,且系統必須安裝Go環境。假設程式就是hello.go,且使用了websocket。那麼過程如下:
- 安裝依賴
➜ tonny@tonny-pc ~ github.com/gorilla/websocket
- 建立應用程式
➜ tonny@tonny-pc ~ docker run --rm -v "$GOPATH":/gopath -v "$(pwd)":/app -w /app google/golang sh -c 'go build -o hello'
注意要把本地GOPATH掛載到容器。
- 運行應用程式
➜ tonny@tonny-pc ~ docker run --rm -v "$(pwd)":/app -w /app google/golang sh -c './hello'
注意,要保持容器的一直可用,比如RocksDB執行個體的一直可用,這樣就不會使得容器每次運行都刪除了RocksDB執行個體,可以執行:
docker run --name goapp -v "$GOPATH":/gopath -v "$(pwd)":/app -w /app google/golang sh -c 'go build -o hello && ./hello' || docker start -ia goapp
dj工具
dj項目的首頁見: https://github.com/treeder/dj 。dj即Docker Jockey,是一個Docker鏡像,它協助開發人員簡化開發環境的搭建過程,只需要安裝Docker,無需再安裝程式設計語言的運行時或環境配置等。
使用dj工具有幾個優點:
- 無需安裝Golang
- 無需配置GOPATH環境變數
- 無需把代碼放在指定的目錄(比如/src/github.com/user/hello目錄)或者是建立Golang的目錄結構
- 依賴無需重新匯入
- 支援交叉編譯
- 支援靜態編譯
- 構建到Docker鏡像中
- 支援遠程Git倉庫構建
dj的四個功能:
- vendor 所有的依賴均放入/vendor目錄
- build 使用vendor依賴構建應用程式
- run 運行你的應用程式
- image 為應用程式建立Docker鏡像
dj支援Golang、Ruby、Node.js、PHP、Python等程式設計語言。
# vendor依賴
dj LANG vendor
# 測試
dj LANG run [script.abc]
# 建立Docker鏡像
dj LANG image username/myapp:latest
# 測試Docker鏡像
docker run --rm -p 8080:8080 username/myapp [script.abc]
# 把鏡像推送到DockerHub
docker push username/myapp
# 檢查程式設計語言的版本
dj LANG version
指定Golang程式設計語言
# 只構建不運行dj go build# fmt:dj go fmt# 建立靜態二進位包dj go static# 跨平台編譯dj go cross# 從遠程倉庫構建dj go remote http://github.com/org/project