溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

docker運行tomcat提示找不到文件如何解決

發(fā)布時間:2023-03-07 13:50:13 來源:億速云 閱讀:173 作者:iii 欄目:開發(fā)技術

這篇“docker運行tomcat提示找不到文件如何解決”文章的知識點大部分人都不太理解,所以小編給大家總結了以下內容,內容詳細,步驟清晰,具有一定的借鑒價值,希望大家閱讀完這篇文章能有所收獲,下面我們一起來看看這篇“docker運行tomcat提示找不到文件如何解決”文章吧。

    docker運行tomcat提示找不到文件

    問題描述

    docker運行tomcat提示找不到文件如何解決

    • 核心信息

    Exited(1)
    
    Cannot find /usr/local/tomcat/bin/setclasspath.sh
    This file is needed to run this program
    • 一開始也挺費解的,我雖然不上這個課,但也比較好奇,自己始終無法復現(xiàn),但不斷有學員問,我看到就回復,在docker run命令后加一個--privileged即可

    • 但為何呢,不能說的很清楚,因為--privileged這個參數就是讓你容器內的root用戶具備擁有真正的root權限。否則容器內的root只是外部的一個普通用戶權限。

    線索一:容器退出碼

    • 從上面的提示可以看到容器退出了,ExitCode是1

    • 1的意思是:

      • 程序錯誤,或者Dockerfile中引用不存在的文件,如 entrypoint中引用了錯誤的包

      • 程序錯誤可以很簡單,例如“除以0”,也可以很復雜,比如空引用或者其他程序 crash

    • ExitCode1: Indicates failure due to application error

    • Indicates that the container stopped due to either an application error or an incorrect reference in Dockerfile to a file that is not present in the container.

    • An application error can be as simple as “divide by 0” or as complex as “Reference to a bean name that conflicts with existing, non-compatible bean definition of same name and class.”

    • An incorrect reference in Dockerfile to a file not present in the container can be as simple as a typo (the example below has sample.ja instead of sample.jar)

    • 知道了這個似乎幫助不大,不過有的容器退出碼是非常能指向原因的,比如ExitCode 0

    線索二:無法找到文件

    • 這個線索就非常重要了

    • 那為何會無法找到,真的有嗎?有的

    • 比如在我這個正常的容器中

    [root@hecs-67651 ~]# docker ps -a
    CONTAINER ID   IMAGE                                        COMMAND                  CREATED          STATUS          PORTS                                                                                      NAMES
    59463bed0fd7   tomcat                                       "catalina.sh run"        35 minutes ago   Up 35 minutes   8080/tcp                                                                                   mytomcat5
    
    [root@hecs-67651 ~]# docker exec -it 594 ls /usr/local/tomcat/bin/setclasspath.sh
    /usr/local/tomcat/bin/setclasspath.sh
    • 那遇到問題的學員為何找不到呢?

    • 我們的這個tomcat鏡像在啟動的時候會執(zhí)行一個腳本

    [root@hecs-67651 ~]# docker inspect -f '{{.Config.Cmd}}' tomcat:latest
    [catalina.sh run]
    • 來看看catalina.sh做了啥

    順藤摸瓜:catalina.sh

    • 這個shell腳本比較大646行,我就摘錄關鍵部分

    • 你看懂需要懂一些shell

    • 第一部分:報錯在哪里

    if $os400; then
      # -r will Only work on the os400 if the files are:
      # 1. owned by the user
      # 2. owned by the PRIMARY group of the user
      # this will not work if the user belongs in secondary groups
      . "$CATALINA_HOME"/bin/setclasspath.sh
    else
      if [ -r "$CATALINA_HOME"/bin/setclasspath.sh ]; then
        . "$CATALINA_HOME"/bin/setclasspath.sh
      else
        echo "Cannot find $CATALINA_HOME/bin/setclasspath.sh"
        echo "This file is needed to run this program"
        exit 1
      fi
    fi
    • 可以看到我們的報錯就在這里

    • 執(zhí)行的時候[ -r "$CATALINA_HOME"/bin/setclasspath.sh ]這個分支為假就走到了我們的報錯中,exit 1

    • 這句話的意思是看 "$CATALINA_HOME"/bin/setclasspath.sh文件是否有read權限

    root@59463bed0fd7:/usr/local/tomcat/bin# ll setclasspath.sh
    -rwxr-xr-x 1 root root 3342 Mar  6 23:33 setclasspath.sh*
    • 在我這個OK的環(huán)境中的權限如上,read是有的

    • 那可能的問題就是在CATALINA_HOME這個變量是否存在

    • 而再往前看我們走到第一個else是因為$os400為假

    第二部分:os400(僅供學習,對本問題沒有作用,無需分析)

    cygwin=false
    darwin=false
    os400=false
    hpux=false
    case "`uname`" in
    CYGWIN*) cygwin=true;;
    Darwin*) darwin=true;;
    OS400*) os400=true;;
    HP-UX*) hpux=true;;
    esac
    • 從這里可以看到os400初始值為false,只有你的uname是OS400的時候才為true

    • 而我們這個環(huán)境的uname的值是

    [root@59463bed0fd7 ~]# uname
    Linux
    • 第三部分:[ -r "$CATALINA_HOME"/bin/setclasspath.sh ]

      • 等價于 test -r "$CATALINA_HOME"/bin/setclasspath.sh

      • 我這個OK的環(huán)境執(zhí)行效果

    root@59463bed0fd7:/usr/local/tomcat/bin# [ -r "$CATALINA_HOME"/bin/setclasspath.sh ]
    root@59463bed0fd7:/usr/local/tomcat/bin# echo $?
    0
    • 可以看到,是為0的返回值,那自然就不會報錯,報錯的環(huán)境肯定是非0 的

    • 問題的焦點似乎就集中到了$CATALINA_HOME上

    第四部分:$CATALINA_HOME怎么來的

    # 下面的意思是如果沒有CATALINA_HOME這個變量就設置為cd "$PRGDIR/.." >/dev/null; pwd 這個pwd的結果
    [ -z "$CATALINA_HOME" ] && CATALINA_HOME=`cd "$PRGDIR/.." >/dev/null; pwd`
    
    # 而PRGDIR是這么來的
    PRGDIR=`dirname "$PRG"`
    
    # PRG來自
    PRG="$0"  # 就是catalina.sh所在目錄
    
    # 下面的我也有點看不懂了,大致就是獲取目錄
    while [ -h "$PRG" ]; do
      ls=`ls -ld "$PRG"`
      link=`expr "$ls" : '.*-> \(.*\)$'`
      if expr "$link" : '/.*' > /dev/null; then
        PRG="$link"
      else
        PRG=`dirname "$PRG"`/"$link"
      fi
    done
    • 找了半天找了個寂寞?好像是的。那問題到底在哪里呢?我也沒法復現(xiàn)。捋一捋

    • 線索:[ -r "$CATALINA_HOME"/bin/setclasspath.sh ] 執(zhí)行為非0是肯定的

    • 如果文件存在,變量存在,那問題就只能是-r了,權限問題!

    解決方法

    • 在docker run命令后加一個--privileged即可

    • --privileged這個參數就是讓你容器內的root用戶具備擁有真正的root權限。否則容器內的root只是外部的一個普通用戶權限。

    • 往上有個文檔說是:與faccessat2系統(tǒng)調用有關,由于 runc 中的 bug,如果您的內核不支持 faccessat2,它將失敗。

    以上就是關于“docker運行tomcat提示找不到文件如何解決”這篇文章的內容,相信大家都有了一定的了解,希望小編分享的內容對大家有幫助,若想了解更多相關的知識內容,請關注億速云行業(yè)資訊頻道。

    向AI問一下細節(jié)

    免責聲明:本站發(fā)布的內容(圖片、視頻和文字)以原創(chuàng)、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

    AI