FAQ docker运行tomcat提示找不到文件

faq,docker,运行,tomcat,提示,不到,文件 · 浏览次数 : 232

小编点评

**原因:** `docker run`命令的`--privileged`参数使容器内的root用户拥有拥有真正的root权限。当文件不存在时,`setclasspath.sh`脚本无法找到该文件并返回`0`,导致容器退出失败。 **解决方案:** 在`docker run`命令中,使用`--privileged`参数可以为容器内的root用户提供拥有真正的root权限。这可以通过`chroot`命令实现: ```bash docker run -it --privileged -v $PWD:/usr/local/tomcat/bin/ setclasspath.sh ``` 这将允许你运行`setclasspath.sh`脚本并从该脚本中获取的路径。

正文

docker运行tomcat提示找不到文件

问题描述

  • docker课程中,老师是用tomcat镜像来演示docker的一些操作
  • 但同样的操作有的同学是ok的,有的同学就会遇到如下错误

  • 核心信息

    Exited(1)
    
    Cannot find /usr/local/tomcat/bin/setclasspath.sh
    This file is needed to run this program
    
  • 一开始也挺费解的,我虽然不上这个课,但也比较好奇,自己始终无法复现,但不断有学员问,我看到就回复,在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镜像在启动的时候会执行一个脚本

    [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
    
    
    • 可以看到我们的报错就在这里

    • 执行的时候[ -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的环境中的权限如上,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

    • 而我们这个环境的uname的值是

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

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

    • 我这个OK的环境执行效果

      root@59463bed0fd7:/usr/local/tomcat/bin# [ -r "$CATALINA_HOME"/bin/setclasspath.sh ]
      root@59463bed0fd7:/usr/local/tomcat/bin# echo $?
      0
      
      
    • 可以看到,是为0的返回值,那自然就不会报错,报错的环境肯定是非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
    
    
    
  • 找了半天找了个寂寞?好像是的。那问题到底在哪里呢?我也没法复现。捋一捋

  • 线索:[ -r "$CATALINA_HOME"/bin/setclasspath.sh ] 执行为非0是肯定的

  • 如果文件存在,变量存在,那问题就只能是-r了,权限问题!

解决方法

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

  • --privileged这个参数就是让你容器内的root用户具备拥有真正的root权限。否则容器内的root只是外部的一个普通用户权限。

  • 往上有个文档说是:与faccessat2系统调用有关,由于 runc 中的 bug,如果您的内核不支持 faccessat2,它将失败。这有点难了~看不懂

    https://syntaxbug.com/6d684d2afe/

与FAQ docker运行tomcat提示找不到文件相似的内容:

FAQ docker运行tomcat提示找不到文件

docker运行tomcat提示找不到文件 问题描述 docker课程中,老师是用tomcat镜像来演示docker的一些操作 但同样的操作有的同学是ok的,有的同学就会遇到如下错误 核心信息 Exited(1) Cannot find /usr/local/tomcat/bin/setclassp

FAQ docker运行容器时遇到的一个iptables的报错

现象 某个同学执行docker启动容器的时候提示 docker: Error response from daemon: driver failed programming external connectivity on endpoint frosty_montalcini (5f23d1b819

【FAQ】关于华为地图服务定位存在偏差的原因及解决办法

一、 问题描述: 华为地图服务“我的位置”能力,在中国大陆地区,向用户展示他们在地图上的当前位置与用户的实际位置存在较大的偏差。 具体差别可以查看下方的图片: 二、 偏差较大的原因: 华为Map SDK在中国大陆使用的地理坐标系是GCJ02。 点击“我的位置”控件,获取的定位经纬度的地理坐标系是WG

【FAQ】在华为鸿蒙车机上集成华为帐号的常见问题总结

随着新一代信息技术与汽车产业的深度融合,智能网联汽车正逐渐成为汽车产业发展的战略制高点,无论是传统车企还是新势力都瞄准了“智能座舱”这种新一代人机交互方式。面对竞争如此激烈的车机市场,华为鸿蒙车机系统的出现,给消费者带来了不同凡响的便捷使用感受,这得益于华为在硬件、软件和场景优化上的技术优势,用户只

【FAQ】申请Health Kit权限的常见问题及解答

华为运动健康服务(HUAWEI Health Kit)提供原子化数据开放,用户数据被授权获取后,应用可通过接口访问运动健康数据,对相关数据进行增、删、改、查等操作。这篇文章汇总了申请开通Health Kit测试权限的常见问题,并给出了详细解答,希望为开发者提供相关参考。 (1) 申请Health K

【FAQ】推送服务常见问题及解决方案

一、推送成功收不到消息,推送返回:{"message":"success","requestID":"1523868*****2842718","resultcode":0} 排查步骤: 1、网络不稳定,切换稳定网络进行测试; 2、检查手机是否为EMUI8.0.0系统,如果是早期的EMUI8.0,则

【FAQ】申请运动健康服务验证环节常见问题及解答

华为 HMS Core 运动健康服务(HUAWEI Health Kit)提供原子化数据开放。应用在获取用户数据授权后,可通过接口访问运动健康数据,对用户数据进行读写等操作,为用户提供运动健康类数据服务。 开发者应用在开发和测试阶段访问用户运动或健康数据时,会有100个用户的数量限制,需要通过“申请

【FAQ】集成分析服务的常见问题及解决方案

常见问题一:如何验证Analytics是否上报/接入成功?以及关键日志含义是什么? 在初始化Analytics SDK前添加SDK日志开关如下: HiAnalyticsTools.enableLog (); 2.初始化SDK代码如下: HiAnalyticsInstance instance = H

【FAQ】获取Push Token失败,如何进行排查?

一、 获取Push Token的方式 获取Push Token有两种方式:一种是调用getToken方法向Push服务端请求Token,当getToken方法返回为空时,Token可通过onNewToken方法返回,因此需要实现onNewToken方法;另一种是自动初始化,Token通过onNewT

【FAQ】关于华为推送服务因营销消息频次管控导致服务通讯类消息下发失败的解决方案

一. 问题描述 使用华为推送服务下发IM消息时,下发消息请求成功且code码为80000000,但是手机总是收不到消息; 在华为推送自助分析(Beta)平台查看发现,消息发送触发了频控。 二. 问题原因及背景 2023年1月05日起,华为推送服务对咨询营销类消息做了单个设备每日推送数量上限管理,具体