node 20
Runtime · Language runtime · standard · v20
Hardened Node.js runtime + npm. Active LTS lines (20/22/24).
Version line
The latest line lives at the base page; older lines have their own page so you can pin and verify exactly that version.
Use it as a base image
Reference it in the FROM line of your Dockerfile. Nonroot, read-only
root filesystem, built for amd64 and arm64.
FROM ghcr.io/quenchworks/images/node:20 Or pull it directly
docker pull ghcr.io/quenchworks/images/node:20 - Version line
- 20
- Latest line
- 20, 22, 24
- Architectures
- amd64, arm64
- Runs as
- nonroot (uid 1001)
- Root filesystem
- read-only
- License
- MIT
Verify the supply chain
This image is cosign-signed and carries an SPDX SBOM and a SLSA build-provenance attestation on the same digest. Check all three before you build on it:
# 1. signature — built and signed by QuenchWorks CI
cosign verify ghcr.io/quenchworks/images/node:20 \
--certificate-identity-regexp 'https://github.com/quenchworks/.+' \
--certificate-oidc-issuer https://token.actions.githubusercontent.com
# 2. SLSA build provenance — which workflow built it, from what
gh attestation verify oci://ghcr.io/quenchworks/images/node:20 --owner quenchworks
# 3. SPDX SBOM — the package inventory
gh attestation verify oci://ghcr.io/quenchworks/images/node:20 --owner quenchworks \
--predicate-type https://spdx.dev/Document See the SBOM & provenance guide for reading the SBOM and using these checks in CI.
Best-practice Dockerfile for 20
The four-stage pnpm pattern: a shared base, a prod-only dependency stage, a build stage with the full dependency set, and a slim final stage that ships only the production node_modules and the built dist.
# Base: a common starting point for the dependency and build stages.FROM ghcr.io/quenchworks/images/node:20 AS baseWORKDIR /appENV PNPM_HOME=/tmp/pnpm \ npm_config_cache=/tmp/npm
# prod-deps: resolve production dependencies only.FROM base AS prod-depsCOPY package.json pnpm-lock.yaml ./RUN ["corepack", "enable"]RUN ["pnpm", "install", "--prod", "--frozen-lockfile"]
# build: install the full dependency set and build.FROM base AS buildCOPY package.json pnpm-lock.yaml ./RUN ["corepack", "enable"]RUN ["pnpm", "install", "--frozen-lockfile"]COPY . .RUN ["pnpm", "run", "build"]
# final: prod node_modules + built dist on a clean node base, nonroot.FROM ghcr.io/quenchworks/images/node:20 AS finalWORKDIR /appENV NODE_ENV=productionCOPY --from=prod-deps /app/node_modules ./node_modulesCOPY --from=build /app/dist ./distCOPY --from=build /app/package.json ./package.jsonUSER 1001EXPOSE 3000CMD ["node", "dist/server.js"]This Dockerfile is pinned to the 20 line. For the line-by-line walkthrough and ecosystem variants (npm/Yarn, pip/uv/Poetry, Maven/Gradle), see the Build a Node app guide.
Upstream project: https://github.com/nodejs/node