目錄
- 項目場景:
- 問題描述
- 造成原因
- 解決方案
- 總結
項目場景:
記一次Dockerfile構建的Docker鏡像,啟動容器時sh: not found的問題
Dockerfile構建的Docker鏡像,啟動容器時找不到start.sh,執行docker run 命令時報錯:/bin/sh: 1: /data/server/start.sh: not found
問題描述
Dockerfile如下(腳本正確無內容錯誤)
FROM openjdk:8 MAINTAINER it235.com # 環境sit/pro,由gradle傳入 ARG env ENV env ${env} WORKDIR /data/server #應用包 COPY order.jar start.sh arthas-boot.jar /data/server/ RUN sh -c 'mkdir order' &&\ sh -c 'mkdir logs' &&\ sh -c 'mkdir order/gc' &&\ sh -c 'chmod -R 755 order/' &&\ sh -c 'chmod -R 755 logs/' &&\ sh -c 'chmod 755 start.sh' &&\ ln -sf /usr/share/zoneinfo/Asia/Shanghai /etc/localtime && echo 'Asia/Shanghai' >/etc/timezone #啟動容器 ENTRYPOINT ["/bin/sh","-c","/data/server/start.sh ${env}"] EXPOSE 80
造成原因
造成這個問題的原因主要有以下幾點:
- docker中的文件夾沒有獲得權限
- start.sh文件不在指定的目錄下
- start.sh沒有權限
- build image時沒有把該腳本傳入
- start.sh文件中內容錯誤
- start.sh中windows和unix的CRLF和LF的換行符問題
- start.sh本身換行符沒問題,但是提交Git后由于git的設置導致文件換行符發生變化
最終發現我這里是第6點問題,坑爹,這里針對以上6點分別給出解決方案。
解決方案
原因1:Dockerfile中給權限即可,我這里已經授權
原因2:在啟動容器報錯后使用 docker cp 容器名稱:/data/server/start.sh ./
命名拷貝到本地宿主機目錄,如果能夠拷貝出來,說明文件位置正確
原因3:Dockerfile中給權限,給到最高777
原因4:排查方式同原因3,注意build image時時一定要在Dockerfile當前上下文中,maven和gradle項目可以用插件來做,可以看我的文檔進行,文檔地址看上面Dockerfile第2行
原因5:構建鏡像之前,先自己在linux上運行下,保證內容正確
原因6:這點和原因7一樣坑爹,一般情況下你根本不會考慮到這個問題,這個問題也非常好改,但是排查就要廢點心了。以IDEA編輯器為例
問題6我們是解決了,但是怎么排查出來是這個問題的呢,我們可以使用上述原因2的命令docker cp把start.sh從容器中拷貝出來(停止的容器也可以),然后使用命令cat -A start.sh
,如果你命令后面跟的是$美元符號,那么恭喜你,文件是LF換行的,如果跟的是^M$符號,那么不好意思,是CRLF換行符,以下2張圖給你做個對比。
原因7跟原因6一樣吧
好了,就到這里,做個小小的總結,給個愛心讓更多小伙子看到吧