Skip to main content

配置容器化的 Strapi

更新项目依赖

package.json 中 strapi 相关的依赖更新到最新版本或次新版本。如果最新版本是安全更新,那就用最新版,如果是功能更新,为了避免出问题,就用次新版本。

因为之前在某版本发布之后,出现了一个影响程序运行的问题,所以为了避免最新版有问题,在没有安全更新的情况下,优先使用次新版本。

构建 Docker 镜像

构建基础镜像

由于 Strapi 所需的 Node.js 环境平时并不变化,所以将不变的部分编译成基础镜像,这样可以显著提升最终镜像的构建速度。

基础镜像的 Dockerfile 如下。

FROM node:18-alpine as strapi.node-base

# 设置时区为东八区,这样在查看日志时才能显示正确的时间
ENV TZ=Asia/Shanghai
RUN ln -snf /usr/share/zoneinfo/$TZ /etc/localtime && echo $TZ > /etc/timezone

# Installing libvips-dev for sharp Compatibility
RUN sed -i 's/dl-cdn.alpinelinux.org/mirrors.aliyun.com/g' /etc/apk/repositories \
&& apk update \
&& apk add --no-cache build-base gcc autoconf automake zlib-dev libpng-dev nasm bash vips-dev git
ARG NODE_ENV=development
ENV NODE_ENV=${NODE_ENV}

WORKDIR /opt/
COPY package.json yarn.lock ./

假设这个 Dockerfile 的文件名为 Dockerfile.node-base,那么用下面的命令即可构建基础镜像。

注意 -t 参数后面的镜像名称,要和 Dockerfile 中第一行 FROM 语句 as 后面的名称相一致,不然在构建最终镜像时会找不到这个镜像。

docker build -t strapi.node-base -f ./Dockerfile.node-base .

构建最终镜像

有了上面的基础镜像,再结合下面的 Dockerfile,就可以构建出最终的 Strapi 镜像。

注意第一行 FROM 语句,这里使用了上面构建的基础镜像 strapi.node-base

另外在执行 yarn build 之前,先执行了 yarn strapi ts:generate-types,这是为了生成 TypeScript 类型文件,以便在开发时使用,不然启动容器后就会 报错

FROM strapi.node-base as build

RUN yarn config set network-timeout 600000 -g && yarn config set registry https://registry.npmmirror.com && yarn global add node-gyp && yarn install && yarn cache clean

ENV PATH /opt/node_modules/.bin:$PATH
WORKDIR /opt/app

COPY --chown=node:node . .
USER node

RUN ["yarn", "strapi", "ts:generate-types"]
RUN ["yarn", "build"]
EXPOSE 1337
CMD ["yarn", "develop"]

用下面的命令来构建最终镜像。

docker build -t strapi.final -f .\Dockerfile.final .

最终文件

// docker-compose.yml

version: '3'

services:
strapi:
container_name: strapi
# 使用当前目录下的 Dockerfile.prod 文件的配置生成镜像
build:
context: .
dockerfile: Dockerfile.prod
image: strapi:latest
restart: unless-stopped
env_file: .env
environment:
DATABASE_CLIENT: ${DATABASE_CLIENT}
DATABASE_HOST: strapiDB
DATABASE_NAME: ${DATABASE_NAME}
DATABASE_USERNAME: ${DATABASE_USERNAME}
DATABASE_PORT: ${DATABASE_PORT}
JWT_SECRET: ${JWT_SECRET}
ADMIN_JWT_SECRET: ${ADMIN_JWT_SECRET}
DATABASE_PASSWORD: ${DATABASE_PASSWORD}
NODE_ENV: ${NODE_ENV}
volumes:
- ./config:/opt/app/config
- ./src:/opt/app/src
- ./package.json:/opt/package.json
- ./yarn.lock:/opt/yarn.lock

- ./.env:/opt/app/.env
- ./public/uploads:/opt/app/public/uploads
ports:
- '1337:1337'
networks:
- strapi
depends_on:
- strapiDB

strapiDB:
container_name: strapiDB
env_file: .env
image: mysql:8.0.33 # 指定 MySQL 具体版本
command: --default-authentication-plugin=mysql_native_password
environment:
MYSQL_USER: ${DATABASE_USERNAME}
MYSQL_ROOT_PASSWORD: ${DATABASE_PASSWORD}
MYSQL_PASSWORD: ${DATABASE_PASSWORD}
MYSQL_DATABASE: ${DATABASE_NAME}
volumes:
- ./mysql_data:/var/lib/mysql # MySQL 挂载数据卷至宿主机,记得创建对应目录
ports:
- '3306:3306'
networks:
- strapi

# 可选
strapiAdminer:
container_name: strapiAdminer
image: adminer
restart: unless-stopped
ports:
- '9090:8080'
environment:
- ADMINER_DEFAULT_SERVER=strapiDB
networks:
- strapi
depends_on:
- strapiDB

// 配置几个容器在同一个网络中,否则互相之间无法通信
networks:
strapi:
name: Strapi
driver: bridge

用指定的 YML 文件启动容器

# 用 compose-dev.yml 启动容器
$ docker compose -f compose-dev.yml up -d

compose-dev.yml 文件与 compose.yml 文件的区别只在下面一段:

# compose-dev.yml
services:
strapi:
container_name: strapi
build:
context: .
dockerfile: Dockerfile.prod # 生产环境执行另一个 Dockerfile 文件

日志持久化

相关报错

编译镜像时报错 2

在解决了下面的问题后,再次编译镜像时,又报了下面的错误。

=> CANCELED [internal] load build context

...

ERROR: failed to solve: Canceled: context canceled

上网查了一下,这里 说得把 node_modules 文件夹添加到 Dockerignore 文件中。但是我看了一下,node_modules 已经在 .dockerignore 文件中了,不过还有个 mysql_data 文件夹没添加,这个文件夹是用来存放 MySQL 数据的。

添加了 mysql_data 文件夹后,再次编译镜像,就成功了。看来就是得不断地尝试啊。

编译镜像时报错 1

在用命令 docker build -t strapi.node-base -f ./Dockerfile.node-base . 编译基础镜像时,报下面的错误。GitHub 上 Docker Desktop Windows 版本的 issue 里有人提到了这个问题,可以看到一年前就有人在问这个问题,自己看了一圈,没看到什么解决方案。

http2: server: error reading preface from client //./pipe/docker_engine: file has already been closed

没有管上面的错误,再用命令 docker build -t strapi.final -f .\Dockerfile.final . 编译最终镜像,结果又报下面的错误。

=> ERROR [internal] load build context

...

ERROR: failed to solve: archive/tar: unknown file mode ?rwxr-xr-x

上网查了查,看了几篇文章,最后用了 这里 的方法,把文件夹和文件的 存档 属性去掉,但是编译还是会出错。

然后尝试把 Docker 版本从 4.30.0 降到了 4.29.0,再次编译镜像,总算成功了。

数据库连接断开

有一天在查看 Strapi Docker 容器的日志时,发现了下面的报错信息。

Error: read ECONNRESET
at TCP.onStreamRead (node:internal/stream_base_commons:217:20)
at TCP.callbackTrampoline (node:internal/async_hooks:128:17)

去项目的官方 GitHub issue 里搜索,看到了这个帖子 Error: read ECONNRESET Strapi container #20311

看了里面的回复,发现原来是因为 Docker 会自动清除空闲连接,导致 Strapi 与数据库的连接被断开,所以 Strapi 才会报错。

解决方法也很简单,把 ./config/database.ts|js 文件中 pool.min 的值从默认的 2 改成 0 就行。

参考资料

用到了开源项目 strapi-community / strapi-tool-dockerize,用于在已生成 strapi 项目的情况下,再生成对应的 dockerfile 和 docker-compose.yml 等相关文件,以便部署至 Docker 中。

又参考了国内文章 快速体验 Strapi CMS(基于 docker 部署),对 docker compose 过程中需要从国外下载的情况,配置了国内源,实现加速下载。

这篇文章 Docker Container time & timezone (will not reflect changes) 介绍了怎样在 Docker 容器中设置时区,如果用默认时区,容器中的日志时间会是 UTC 时间,不方便查看。