【问题标题】:Dockerfile with parametrization: fails with XXX带有参数化的 Dockerfile:失败并显示 XXX
【发布时间】:2020-12-17 05:07:26
【问题描述】:

我在一个简单的 Spring Boot 应用程序中有以下 Dockerfile

FROM maven:3.6-jdk-8-alpine as build

WORKDIR /app

COPY ./pom.xml ./pom.xml

RUN mvn dependency:go-offline -B

# copy your other files
COPY ./src ./src

# build for release
RUN mvn package -DskipTests

FROM openjdk:8-jre-alpine
ARG artifactid
ARG version
ENV artifact ${artifactid}-${version}.jar
WORKDIR /app

COPY --from=build /app/target/${artifact} /app

EXPOSE 8080

ENTRYPOINT ["sh", "-c"]

CMD ["java","-jar ${artifact}"]

当我使用所需的参数构建它时:

docker build --build-arg artifactid=spring-demo --build-arg version=0.0.1 -t spring-demo .

它的构建没有错误。

当我尝试运行图像时:

docker container run -it spring-demo

它失败并出现以下错误:

Usage: java [-options] class [args...]
           (to execute a class)
   or  java [-options] -jar jarfile [args...]
           (to execute a jar file)
where options include:
    -d32      use a 32-bit data model if available
    -d64      use a 64-bit data model if available
    -server   to select the "server" VM
                  The default VM is server,
                  because you are running on a server-class machine.


    -cp <class search path of directories and zip/jar files>
    -classpath <class search path of directories and zip/jar files>
                  A : separated list of directories, JAR archives,
                  and ZIP archives to search for class files.
    -D<name>=<value>
                  set a system property
    -verbose:[class|gc|jni]
                  enable verbose output
    -version      print product version and exit
    -version:<value>
                  Warning: this feature is deprecated and will be removed
                  in a future release.
                  require the specified version to run
    -showversion  print product version and continue
    -jre-restrict-search | -no-jre-restrict-search
                  Warning: this feature is deprecated and will be removed
                  in a future release.
                  include/exclude user private JREs in the version search
    -? -help      print this help message
    -X            print help on non-standard options
    -ea[:<packagename>...|:<classname>]
    -enableassertions[:<packagename>...|:<classname>]
                  enable assertions with specified granularity
    -da[:<packagename>...|:<classname>]
    -disableassertions[:<packagename>...|:<classname>]
                  disable assertions with specified granularity
    -esa | -enablesystemassertions
                  enable system assertions
    -dsa | -disablesystemassertions
                  disable system assertions
    -agentlib:<libname>[=<options>]
                  load native agent library <libname>, e.g. -agentlib:hprof
                  see also, -agentlib:jdwp=help and -agentlib:hprof=help
    -agentpath:<pathname>[=<options>]
                  load native agent library by full pathname
    -javaagent:<jarpath>[=<options>]
                  load Java programming language agent, see java.lang.instrument
    -splash:<imagepath>
                  show splash screen with specified image
See http://www.oracle.com/technetwork/java/javase/documentation/index.html for more details.

请问上面的设置有什么问题吗? 应用示例代码可以在here找到。

【问题讨论】:

  • 试试这样:CMD ["java", "-jar", "${artifact}"]
  • @MarkoE,它引发了同样的错误

标签: spring-boot docker maven


【解决方案1】:

您应该删除ENTRYPOINT 行并使用CMD 的shell 形式。

# No ENTRYPOINT
CMD java -jar ${artifact}

Dockerfile ENTRYPOINTCMDget combined into a single command line。在你的 Dockerfile 中,它被解释为

sh -c java '-jar ${artifact}'

sh -c 选项实际上只接受下一个单词并将其解释为要运行的命令;所以它真的被处理为

sh -c 'java' # '-jar ${artifact}'

忽略-jar 选项。

two ways 可以“拼写”CMD(以及ENTRYPOINTRUN)。正如您使用 JSON 数组所做的那样,您可以准确指定进入命令行的“单词”,例如,-jar ${artifact} 将作为单个参数传递,包括嵌入的空间。如果你只是传递一个命令行,Docker 会为你插入一个sh -c 包装器,shell 会处理单词解析和变量插值。您永远不需要在 Dockerfile 中手动包含 sh -c

【讨论】:

  • 我删除了ENTRYPOINT 并尝试了两种方法:CMD ["java", "-jar ${artifact}"] 引发了Unrecognized option: -jar ${artifact}, Error: Could not create the Java Virtual Machine.。如下:CMD ["java", "-jar", "${artifact}"],这引发了Error: Unable to access jarfile ${artifact}。我错过了什么?
  • 啊,我的错,-CMD java -jar ${artifact} 工作正常 :)。非常感谢!
  • 我仍然不清楚为什么CMD ["java", "-jar", "spring-demo-0.0.1.jar"] 有效但在使用工件变量时无效:CMD ["java", "-jar", "${artifact}"], - it fails with Error: Unable to access jarfile ${artifact}. The same when using CMD ["java", "- jar ${artifact}"]` 引发了Unrecognized option: -jar ${artifact}。为什么会这样?
  • JSON 数组形式根本不涉及 shell,Docker 也不会扩展 CMD 中的变量。在第一种情况下,您实际上是在传递字符串 ${artifact} 作为第二个选项。在第三个中,您将两个选项组合在一起并将-jar ${artifact}(带有空格、美元符号和大括号)作为单个参数传递,JVM 无法理解它。
  • ENTRYPOINT ["java", "-jar", "${artifact}"] 相同的错误,这真的很奇怪,恕我直言,为什么CMDENTRYPOINT 都看不到容器中的jar。使其工作的唯一方法是:CMD java -jar ${artifact},不带括号,不带引号。
【解决方案2】:

在我看来,sh -c 有错误。参数未正确读取。您可以检查是否对退出的容器进行 docker 检查。在输出中搜索“CMD”。

入口点 ["sh", "-c"]

CMD ["java","-jar ${artifact}"]

如果你想用sh -c 运行它,你必须像这样引用参数:

CMD ["java -jar ${artifact}"]

你可以试一试吗?

【讨论】:

  • 是的,谢谢,它成功了。我只是关注this article,作者使用了之前的语法。
【解决方案3】:

ENV 变量仅在构建期间可用。要在运行时将环境变量放入容器中,您必须使用--env-e--env-file。最好使用--env-file

看到这个已经回答了同样的问题:How do I pass environment variables to Docker containers? 也看看这个:Use environment variables in CMD

这是一种可能的解决方案:

  1. 保持CMD 指令如下:
CMD ["java","-jar ${artifact}"]
  1. 使用这个 docker run 命令:
docker container run -it -e artifact=spring-demo-0.0.1.jar spring-demo

【讨论】:

  • 问题不存在,与ARG相比,最终图像中有ENV变量。
猜你喜欢
  • 1970-01-01
  • 2015-05-03
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2019-06-17
  • 2012-06-15
  • 1970-01-01
相关资源
最近更新 更多