开发者问题收集

为什么docker会报“没有这样的文件或目录”:未知?

2018-10-29
27320

预期是什么

docker 应该构建镜像并从中运行容器

Dockerfile

FROM centos:7

ENV JAVA_VERSION 8u191
ENV BUILD_VERSION b12

RUN yum -y install wget; wget --no-cookies --no-check-certificate --header "Cookie: oraclelicense=accept-securebackup-cookie" "http://download.oracle.com/otn-pub/java/jdk/$JAVA_VERSION-$BUILD_VERSION/2787e4a523244c269598db4e85c51e0c/jdk-$JAVA_VERSION-linux-x64.rpm" -O /tmp/jdk-8-linux-x64.rpm; yum -y install /tmp/jdk-8-linux-x64.rpm

# JDK stripping
RUN rm -f /usr/java/jdk1.8.0_77/src.zip /usr/java/jdk1.8.0_77/javafx-src.zip
RUN rm -rf /usr/java/jdk1.8.0_77/lib/missioncontrol/ /usr/java/jdk1.8.0_77/lib/visualvm/ /usr/java/jdk1.8.0_77/db/

RUN alternatives --install /usr/bin/java java /usr/java/latest/bin/java 1
RUN alternatives --install /usr/bin/javac javac /usr/java/latest/bin/javac 1

ENV JAVA_HOME /usr/java/latest
ENV PATH=$PATH:/usr/java/latest/bin/java
RUN echo "$PATH"

RUN rm -f /tmp/jdk-8-linux-x64.rpm; yum -y remove wget; yum -y clean all

COPY target/gs-handling-form-submission-0.1.0.jar /tmp/gs-handling-form-submission-0.1.0.jar

CMD [ "java -jar /tmp/gs-handling-form-submission-0.1.0.jar" ]

build-context 是 Dockerfile 所在的当前目录。以下是目录结构

total 8
-rw-r--r--. 1 root     root     1079 Oct 29 03:42 Dockerfile
-rw-rw-r--. 1 ec2-user ec2-user 1917 Oct 27 09:17 pom.xml
drwxrwxr-x. 4 ec2-user ec2-user   30 Oct 27 10:15 src
drwxr-xr-x. 9 root     root      261 Oct 27 10:19 target

我正在执行以下命令来构建镜像:

docker build --no-cache -f Dockerfile -t cpa:latest .

但在运行容器时,收到以下消息

docker run -d -p 8081:8081/tcp cpa   
b2ba9831f1b74c73669843658727ca0d64fcd9de915739b96e17dcfd6e63c7db
docker: Error response from daemon: OCI runtime create failed: container_linux.go:348: starting container process caused "exec: \"java -jar /tmp/gs-handling-form-submission-0.1.0.jar\": stat java -jar /tmp/gs-handling-form-submission-0.1.0.jar: no such file or directory": unknown.

我不确定为什么构建的 jar 在容器文件系统的 /tmp 目录中不可用 在 docker 文档中 COPY 是用于将文件从主机复制到容器文件系统的正确关键字... 但问题出在哪里..? 请提出建议

编辑 1:修复了类型错误从 77 到 191 这是最新的 Dockerfile....

FROM centos:7

ENV JAVA_VERSION 8u191
ENV BUILD_VERSION b12

RUN yum -y install wget; wget --no-cookies --no-check-certificate --header "Cookie: oraclelicense=accept-securebackup-cookie" "http://download.oracle.com/otn-pub/java/jdk/$JAVA_VERSION-$BUILD_VERSION/2787e4a523244c269598db4e85c51e0c/jdk-$JAVA_VERSION-linux-x64.rpm" -O /tmp/jdk-8-linux-x64.rpm; yum -y install /tmp/jdk-8-linux-x64.rpm

# JDK stripping
RUN rm -f /usr/java/jdk1.8.0_191/src.zip /usr/java/jdk1.8.0_191/javafx-src.zip
RUN rm -rf /usr/java/jdk1.8.0_191/lib/missioncontrol/ /usr/java/jdk1.8.0_191/lib/visualvm/ /usr/java/jdk1.8.0_191/db/

RUN alternatives --install /usr/bin/java java /usr/java/latest/bin/java 1
RUN alternatives --install /usr/bin/javac javac /usr/java/latest/bin/javac 1

ENV JAVA_HOME /usr/java/latest
ENV PATH=$PATH:/usr/java/latest/bin/java
RUN echo "$PATH"

RUN rm -f /tmp/jdk-8-linux-x64.rpm; yum -y remove wget; yum -y clean all

COPY target/gs-handling-form-submission-0.1.0.jar /tmp/gs-handling-form-submission-0.1.0.jar

CMD [ "java -jar /tmp/gs-handling-form-submission-0.1.0.jar" ]

此外, target 目录是构建上下文的一部分 请参阅以下内容:

[root@ip-172-31-14-242 cpa_final]# pwd
/tmp/cpa_final
[root@ip-172-31-14-242 cpa_final]# ll
total 8
-rw-r--r--. 1 root     root     1084 Oct 29 05:16 Dockerfile
-rw-rw-r--. 1 ec2-user ec2-user 1917 Oct 27 09:17 pom.xml
drwxrwxr-x. 4 ec2-user ec2-user   30 Oct 27 10:15 src
drwxr-xr-x. 9 root     root      261 Oct 27 10:19 target
[root@ip-172-31-14-242 cpa_final]# cd target/
[root@ip-172-31-14-242 target]# ll
total 17304
drwxr-xr-x. 4 root root       36 Oct 27 10:18 classes
drwxr-xr-x. 3 root root       25 Oct 27 10:18 generated-sources
drwxr-xr-x. 3 root root       30 Oct 27 10:18 generated-test-sources
-rw-r--r--. 1 root root 17708542 Oct 27 10:19 gs-handling-form-submission-0.1.0.jar
-rw-r--r--. 1 root root     4858 Oct 27 10:18 gs-handling-form-submission-0.1.0.jar.original
drwxr-xr-x. 2 root root       28 Oct 27 10:18 maven-archiver
drwxr-xr-x. 3 root root       35 Oct 27 10:18 maven-status
drwxr-xr-x. 2 root root      121 Oct 27 10:18 surefire-reports
drwxr-xr-x. 3 root root       19 Oct 27 10:18 test-classes
[root@ip-172-31-14-242 target]#

编辑 2:

提到的 jar 文件位于容器文件系统的 /tmp 目录中。PSB

[root@ip-172-31-14-242 ~]# docker run -it cpa /bin/bash
[root@4018a6c2df8e /]# cd /tmp
[root@4018a6c2df8e tmp]# ll
total 17300
-rw-r--r--. 1 root root 17708542 Oct 27 10:19 gs-handling-form-submission-0.1.0.jar
-rwx------. 1 root root      836 Oct  6 19:15 ks-script-7RxiSx
-rw-------. 1 root root        0 Oct  6 19:14 yum.log
[root@4018a6c2df8e tmp]#
3个回答
ENV PATH=$PATH:/usr/java/latest/bin/java

可能应该是

ENV PATH=$PATH:/usr/java/latest/bin

我假设未找到的是 java 可执行文件。请检查 java 是否确实在 PATH 上。

此行也是错误的:

CMD [ "java -jar /tmp/gs-handling-form-submission-0.1.0.jar" ]

应为:

CMD [ "java", "-jar", "/tmp/gs-handling-form-submission-0.1.0.jar" ]

现在它正在寻找一个名为“java -jar /tmp/gs-handling-form-submission-0.1.0.jar”的文件并尝试执行该文件。

kutschkem
2018-10-29

CMD 命令有三种形式。每一种都可以为您完成工作。

  • CMD ["param1","param2"] :“ 参数 ”形式
  • CMD ["executable","param1","param2"] :“ exec ”形式
  • CMD 命令 param1 param2 :“ shell ”形式

前两种形式看起来相同,但它们的含义取决于 Dockerfile 中是否有 ENTRYPOINT

  1. 当您的 Dockerfile 中有 ENTRYPOINT 时,将使用 参数 形式。在这种情况下,CMD 命令中的字符串是 ENTRYPOINT 中定义的可执行文件的参数。对于您的情况:

    ENTRYPOINT ["/<path-to-your-java-binary>/java"]
    CMD ["-jar", "myjar.jar"]
    

    注意: CMD 中列出的参数必须是不同的字符串。 CMD ["a", "b"] 表示您向命令传递两个参数,而 CMD ["a b"] 表示只有一个参数。

  2. 如果 Dockerfile 中没有 ENTRYPOINT ,则使用 exec 形式。在这种情况下,第一个字符串是命令,后续字符串是各个参数。在您的例子中:

    CMD ["/<path-to-your-java-binary>/java", "-jar", "myjar.jar"]
    

    请注意,在 exec 形式中没有 shell 处理,即未定义 shell 变量。

  3. 但是,在 shell 形式中,您可以访问 shell 变量,并且第一个参数 ( command ) 将由 shell /bin/sh 使用给定的参数执行。在您的例子中:

    CMD /<path-to-your-java-binary>/java -jar myjar.jar
    

总结:在这种情况下,最简单的就是最好的。

Csongor Halmai
2021-07-15

对于我来说,以下更改有效,即从

CMD ["java -jar myjar.jar"]

更改为

CMD java -jar myjar.jar

尝试删除方括号和双引号

sandy.sm
2021-05-28