Loading [MathJax]/jax/output/CommonHTML/config.js
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >专栏 >node 镜像 ENTRYPOINT 脚本解析

node 镜像 ENTRYPOINT 脚本解析

作者头像
山月
发布于 2022-11-02 08:41:10
发布于 2022-11-02 08:41:10
1K00
代码可运行
举报
运行总次数:0
代码可运行

本篇文章来自 Linux 训练营中的作业,欢迎咨询训练营。

node:18 为例,node 官方镜像的 ENTRYPOINT 入口脚本为以下内容:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
#!/bin/sh
set -e

# Run command with node if the first argument contains a "-" or is not a system command. The last
# part inside the "{}" is a workaround for the following bug in ash/dash:
# https://bugs.debian.org/cgi-bin/bugreport.cgi?bug=874264
if [ "${1#-}" != "${1}" ] || [ -z "$(command -v "${1}")" ] || { [ -f "${1}" ] && ! [ -x "${1}" ]; }; then
  set -- node "$@"
fi

exec "$@"

源码详见 docker-entrypoint.sh1

1. shebang

#! 组成了 shebang,指定脚本执行的解释器的绝对路径。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
# 使用 sh 执行脚本
#!/bin/sh

# 使用 python 执行脚本
#!/usr/bin/python

2. set -e

set -e,当命令发生异常时立即退出。

Exit immediately if a pipeline (which may consist of a single simple command), a list, or a compound command (see SHELL GRAMMAR above), exits with a non-zero status.

新建文件 hello.sh,脚本示例如下,使用 bash hello.sh 执行。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
# 当有该行时,直接报错退出
# 当无该行时,最终会输出 done
set -e

cat notexistfile

echo done

3. if

接下来是 if 语句,过长,需要拆开来看。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
if [ "${1#-}" != "${1}" ] || [ -z "$(command -v "${1}")" ] || { [ -f "${1}" ] && ! [ -x "${1}" ]; }; then
  set -- node "$@"
fi

在这之前,首先需要了解一下语法,可参考 Linux 技能实战2

  • [ ]:布尔判断
  • $1$2$@$1 代表命令行第一个参数,$@ 代表命令行所有参数。可参考 linux 中的函数3

4. ${1#-}

这种属于不较常见的 Parameter Expansion 了,常见的也就 ${NODE_ENV:=development} 这种,见文档 Shell Parameter Expansion4。

${var#word} 具体的含义呢?

可以理解为:如果变量 var 以 word 开头,则 {var#word} 的值为 var 删掉 word,否则为 var。简单点来说,就是 「Remove prefix」。

有点绕口,看看示例:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
$ var=helloworld

$ echo ${var#hello}
world

$ echo ${var#world}
helloworld

$ echo ${var#ok}
helloworld

所以 "

其实用以下两种写法更为直观点:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
$ var=--version

$ [[ ${var:0:1} == "-" ]] && echo ok
ok

$ [[ $var =~ ^- ]] && echo ok
ok

5. command -v

command -v <command> 输出某个命令的真实绝对路径,which 也可以干这件事情。

二者最重要的一个不同点是,当某个命令不存在时,command -v 不会输出任何字符,用此常来判断某个命令是否存在。

可参考 $PATH5

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
$ command -v node
/usr/local/bin/node

$ which hello
/usr/bin/which: no hello in (/home/train/.autojump/bin:/usr/local/bin:/usr/bin:/usr/local/sbin:/usr/sbin:/home/train/bin)

# 无任何输出
$ command -v hello command -v 

[[ -z STRING]],判断 STRING 是否为空字符串。

因此下一条,则是判断 1 该命令是否存在,或者说「判断 1 是否是一个全局可执行的命令」。

6. {[ -f "${1}" ] && ! [ -x "${1}"];}

而这一条,前边是判断 1 是否是一个文件,而后边是判断 1 是否是一个可执行的文件。因此该条件就是说判断 「

可参考 linux 中的括号6

其目的是在 Dockerfile 中可直接写 CMD ["index.js"],如以下示例

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
FROM node:14-alpine

ADD server.js /

CMD ["server.js"]

哦对,注意两点

  1. 该命令使用 {} 包裹
  2. 该命令最后有一个分号

7. set --

set 用以重置命令行 1 2

至于它有什么作用,看了以下示例就明白了了。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
$ set -- a b c d

$ echo $1 $2 $3 $4
a b c d

$ echo "$@"
a b c d

$ set -- node "$@"

$ echo "$@"
node a b c d

所以 set -- node "$@" 的意思也很明白了:「最前边添加一个 node 参数」

8. exec

exec,执行某条命令,但会退出当前 shell。

If command is specified, it replaces the shell.

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
$ exec echo hello
hello

9. 总结

docker-entrypoint.sh 代码添加注释

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
#!/bin/sh
set -e

# 如果 $1 以 - 开头
if [ "${1#-}" != "${1}" ] ||
  # 或者不是一个可执行命令
  [ -z "$(command -v "${1}")" ] ||
  # 或者是当前目录的一个文件,但不可执行
  { [ -f "${1}" ] && ! [ -x "${1}" ]; };
then
  # 则在前边附一个 node 
  set -- node "$@"
fi

# 执行执行代码
exec "$@"
  1. 如果 - 开头,则作为 node 的参数执行
  2. 如果判断 $1 是文件且不可执行,则使用 node 运行该文件
  3. 如果判断 $1 是系统命令,则直接执行该命令

参考资料

[1]

docker-entrypoint.sh:https://github.com/nodejs/docker-node/blob/main/18/bullseye/docker-entrypoint.sh

[2]

Linux 技能实战:https://q.shanyue.tech/command/

[3]

linux 中的函数:https://umiinn9jie.feishu.cn/wiki/wikcnA6zD21wdJtihB1y4fSpqgb

[4]

Shell Parameter Expansion:https://www.gnu.org/software/bash/manual/bash.html#Shell-Parameter-Expansion

[5]

$PATH:https://q.shanyue.tech/command/path.html

[6]

linux 中的括号:https://q.shanyue.tech/command/quote.html

[7]

how to nest conditional script operators -a -o in an if statement in bash:https://unix.stackexchange.com/questions/670519/how-to-nest-conditional-script-operators-a-o-in-an-if-statement-in-bash

本文参与 腾讯云自媒体同步曝光计划,分享自微信公众号。
原始发表:2022-08-02,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 全栈成长之路 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
shell 编程详细命令
多条件判断(&& 表示前一条命令执行成功时,才执行后一条命令,|| 表示上一条命令执行失败后,才执行下一条命令)
飞询
2025/08/01
2500
Dockerfile 的 RUN 和 CMD、ENTRYPOINT
在创建Dockerfile的时候,RUN和CMD都是很重要的命令。它们各自的作用分别如下:
一个会写诗的程序员
2021/03/02
13.4K0
Elasticsearch2.3官方Dockerfile解析
前言 作者博客:http://www.zimug.com* 最近在github上看到了Elasticsearch官方的Dockerfile的文档,感觉不错,有必要学习一下。 包括了安全考虑,和镜像entrypoint的设计也很巧妙 Elasticsearch2.3官方Dockerfile github地址 Dockefile解析 # 使用Dockerhu的java:8-jre作为基础镜像,elashticsearch依赖于jdk7以上版本 FROM java:8-jre # elashticsearch不
字母哥博客
2020/09/23
6060
两小时入门Docker
Docker 最初是 dotCloud 公司创始人 Solomon Hykes 在法国期间发起的一个公司内部项目,于 2013 年 3 月以 Apache 2.0 授权协议开源,主要项目代码在 GitHub 上进行维护。
秃头哥编程
2019/07/04
6510
【Docker】MySQL 源码构建 Docker 镜像(基于 ARM 64 架构)
近期,笔者正推进公司 MySQL 适配 ARM 64 架构工作,由于一直使用 Docker Hub 上的官方镜像,所以第一时间在 Hub 上检索,却发现官方只为 MySQL 8.0 以上版本提供 ARM 64 镜像。
行者Sun
2024/09/02
9390
【Docker】MySQL 源码构建 Docker 镜像(基于 ARM 64 架构)
面向初学者的Docker快速入门指南
yum源:https://mirrors.tuna.tsinghua.edu.cn/docker-ce/linux/centos/docker-ce.repo为了速度可以替换文件中的url。
DevOps云学堂
2020/07/23
1.3K0
面向初学者的Docker快速入门指南
在 CLion 远程开发中使用 Docker 快速搭建基于 CentOS 7.4.1708 的 C++ 开发环境
cmake下载地址: https://cmake.org/files/v3.22/cmake-3.22.4-linux-x86_64.tar.gz 构建Dockerfile: FROM centos:7.4.1708 # https://cmake.org/files/v3.22/cmake-3.22.4-linux-x86_64.tar.gz ADD [ "cmake-3.22.4-linux-x86_64.tar.gz", "/usr/local/" ] COPY [ "docker-entrypoi
耕耘实录
2022/05/18
1.6K0
CentOS8.4.2105 和 OpenEuler 20.03 下 C++ 编译环境容器镜像构建
CentOS7.x 支持的 gcc 和 g++ 版本太低,使用 CentOS 8.x 进行替换,但在实际操作的过程中,使用yum进行依赖更新,出现报错,写此文以记录。
耕耘实录
2022/11/30
7000
自主可控:基于OpenEuler基础镜像的Go语言容器开发环境搭建
从国家层面来说,信息技术已经在国家安全中扮演着越来越重要的角色。通过实现自主可控,可以保护国家的核心技术和关键数据,避免被外部势力窃取或滥用。从我们个人层面来说,自主可控可以保护个人隐私,避免个人信息被滥用或泄露。这对社会稳定和个人权利保护都有重要意义。OpenEuler 是一个开源项目,由华为发起成立于 2019 年。它是一个基于 Linux 操作系统的企业级操作系统,旨在提供高性能、高可靠性、高安全性和易于使用的用户体验。OpenEuler 的出现积极响应了国家自主可控、国产替代等政策需求,同时也是中国开源社区发展进程中具有里程碑式意义的事件。为了响应国家自主可控的号召,我们使用基于OpenEuler操作系统来搭建Go语言开发环境。
耕耘实录
2023/06/10
6010
3.Docker学习之Dockerfile
描述:Dockerfile是一个文本格式的配置文件,其内包含了一条条的指令(Instruction),每一条指令构建一层,因此每一条指令的内容,就是描述该层应当如何构建。 用户可以使用Dockerfile快速创建自定义的镜像;通过它所支持的内部指令,以及使用它创建镜像的基本过程,Docker拥有”一点修改代替大量更新”的灵活之处;
全栈工程师修炼指南
2022/09/28
1.4K0
3.Docker学习之Dockerfile
Docker镜像构建知识:Dockerfile 命令详解
FROM 指令用于指定其后构建新镜像所使用的基础镜像。如果本地不存在,则默认会去Docker Hub下载指定镜像。FROM指令必是Dockerfile文件中的首条命令,启动构建流程后,Docker将基于该镜像构建新镜像,FROM后的命令也会基于这个基础镜像。
废柴阿尤
2023/12/18
4K0
Dockerfile文件编写官方文档 原
英文原文链接地址: Engine reference–>Dockerfile reference
拓荒者
2019/03/11
1.1K1
nodejs 写 bash 脚本终极方案!
最近在学习bash脚本语法,但是如果对bash语法不是熟手的话,感觉非常容易出错,比如说:显示未定义的变量shell中变量没有定义,仍然是可以使用的,但是它的结果可能不是你所预期的。举个例子:
ConardLi
2021/09/08
2.7K0
nodejs 写 bash 脚本终极方案!
3.Docker学习之Dockerfile
描述:Dockerfile是一个文本格式的配置文件,其内包含了一条条的指令(Instruction),每一条指令构建一层,因此每一条指令的内容,就是描述该层应当如何构建。 用户可以使用Dockerfile快速创建自定义的镜像;通过它所支持的内部指令,以及使用它创建镜像的基本过程,Docker拥有”一点修改代替大量更新”的灵活之处;
全栈工程师修炼指南
2020/10/23
2.1K0
3.Docker学习之Dockerfile
Docker 镜像
Docker 运行容器前需要本地存在对应的镜像,如果本地不存在该镜像,Docker 会从镜像仓库下载该镜像。
traffic
2022/05/11
6.7K0
docker | dockerfile指令详解
Dockerfile用于构建docker镜像, 实际上就是把在linux下的命令操作写到了Dockerfile中, 通过Dockerfile去执行设置好的操作命令, 保证通过Dockerfile的构建镜像是一致的.
Amadeus
2023/04/17
2K0
docker 实用工具 gosu 和 su-exec 实践
在 Docker 中,需要把 host 的目录挂载到 container 中作为 volume 使用时,往往会发生文件权限问题。常见的现象是,container 对该路径并无写权限,以致其中服务的各种千奇百怪的问题。
我的小碗汤
2023/03/20
1.9K0
docker 实用工具 gosu 和 su-exec 实践
使用Dockerfile构建自己的etcd镜像
本篇文章手把手教你如何使用Dockerfile构建自己etcd镜像,并且已经提供基于etcd3.0.10 的image供读者使用,用户可以快速使用docker image构建自己的etcd集群环境 一、etcd镜像的Dockerfile文件结构 注意:优秀的docker镜像源文件都是会尝试去除所有依赖的,也即是该文件无论被谁拿走使用,都可以快速构建属于自己的image etcd镜像的Dockerfile项目文件结构: $ tree . . ├──Dockerfile ├──docker-entrypoin
BGBiao
2018/02/26
2.8K0
Docker学习——Dockerfile 指令详解(五) 顶
我们已经介绍了 FROM (指定基础镜像) , RUN(执行命令) ,还提及了 COPY , ADD ,其实 Dockerfile 功能很强大,它提供了十多个指令。下面我们继续讲解其他的指令。
wuweixiang
2018/12/07
1.7K0
Dockerfile、Docker-Compose基本命令与介绍
每条RUN指令将在当前镜像基础上执行指定命令,并提交为新的镜像,后续的RUN都在之前RUN提交后的镜像为基础,镜像是分层的,可以通过一个镜像的任何一个历史提交点来创建,类似源码的 版本控制 。
唐成勇
2019/05/26
2.2K0
相关推荐
shell 编程详细命令
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
本文部分代码块支持一键运行,欢迎体验
本文部分代码块支持一键运行,欢迎体验