镜像构建历史
# 2026-01-24 21:32:11 0.00B 设置默认要执行的命令
CMD ["node" "/app/runtime/server.mjs"]
# 2026-01-24 21:32:11 0.00B 声明容器运行时监听的端口
EXPOSE [3939/tcp]
# 2026-01-24 21:32:11 1.59KB 执行命令并创建新的镜像层
RUN /bin/sh -c cat <<'EOF' > /app/runtime/server.mjs
import http from 'node:http';
import { Readable } from 'node:stream';
import worker from '../src/index.js';
const port = process.env.PORT ? Number(process.env.PORT) : 3939;
const server = http.createServer(async (req, res) => {
const url = new URL(req.url || '/', `http://${req.headers.host || 'localhost'}`);
const headers = new Headers();
for (const [k, v] of Object.entries(req.headers)) {
if (Array.isArray(v)) {
for (const item of v) headers.append(k, item);
} else if (v !== undefined) {
headers.append(k, v);
}
}
const method = req.method || 'GET';
const body = method === 'GET' || method === 'HEAD' ? null : Readable.toWeb(req);
const requestInit = { method, headers, body };
if (body) requestInit.duplex = 'half';
const request = new Request(url, requestInit);
const env = { ...process.env };
const ctx = { waitUntil: (p) => p?.catch(() => {}) };
try {
const response = await worker.fetch(request, env, ctx);
res.statusCode = response.status;
response.headers.forEach((value, key) => {
res.setHeader(key, value);
});
if (response.body) {
const reader = response.body.getReader();
res.on('close', () => reader.cancel().catch(() => {}));
while (true) {
const { done, value } = await reader.read();
if (done) break;
res.write(Buffer.from(value));
}
}
res.end();
} catch (err) {
res.statusCode = 500;
res.setHeader('content-type', 'text/plain; charset=utf-8');
res.end(err?.message || 'Internal Server Error');
}
});
server.listen(port);
EOF # buildkit
# 2026-01-24 21:32:11 0.00B 执行命令并创建新的镜像层
RUN /bin/sh -c mkdir -p /app/runtime # buildkit
# 2026-01-24 16:41:24 21.41KB 复制新文件或目录到容器中
COPY src/index.js /app/src/index.js # buildkit
# 2026-01-24 16:41:24 0.00B 设置工作目录为/app
WORKDIR /app
# 2026-01-15 02:04:02 0.00B 设置默认要执行的命令
CMD ["node"]
# 2026-01-15 02:04:02 0.00B 配置容器启动时运行的命令
ENTRYPOINT ["docker-entrypoint.sh"]
# 2026-01-15 02:04:02 388.00B 复制新文件或目录到容器中
COPY docker-entrypoint.sh /usr/local/bin/ # buildkit
# 2026-01-15 02:04:02 7.18MB 执行命令并创建新的镜像层
RUN /bin/sh -c set -ex && savedAptMark="$(apt-mark showmanual)" && apt-get update && apt-get install -y ca-certificates curl wget gnupg dirmngr --no-install-recommends && rm -rf /var/lib/apt/lists/* && export GNUPGHOME="$(mktemp -d)" && for key in 6A010C5166006599AA17F08146C2130DFD2497F5 ; do { gpg --batch --keyserver hkps://keys.openpgp.org --recv-keys "$key" && gpg --batch --fingerprint "$key"; } || { gpg --batch --keyserver keyserver.ubuntu.com --recv-keys "$key" && gpg --batch --fingerprint "$key"; } ; done && curl -fsSLO --compressed "https://yarnpkg.com/downloads/$YARN_VERSION/yarn-v$YARN_VERSION.tar.gz" && curl -fsSLO --compressed "https://yarnpkg.com/downloads/$YARN_VERSION/yarn-v$YARN_VERSION.tar.gz.asc" && gpg --batch --verify yarn-v$YARN_VERSION.tar.gz.asc yarn-v$YARN_VERSION.tar.gz && gpgconf --kill all && rm -rf "$GNUPGHOME" && mkdir -p /opt && tar -xzf yarn-v$YARN_VERSION.tar.gz -C /opt/ && ln -s /opt/yarn-v$YARN_VERSION/bin/yarn /usr/local/bin/yarn && ln -s /opt/yarn-v$YARN_VERSION/bin/yarnpkg /usr/local/bin/yarnpkg && rm yarn-v$YARN_VERSION.tar.gz.asc yarn-v$YARN_VERSION.tar.gz && apt-mark auto '.*' > /dev/null && { [ -z "$savedAptMark" ] || apt-mark manual $savedAptMark > /dev/null; } && find /usr/local -type f -executable -exec ldd '{}' ';' | awk '/=>/ { so = $(NF-1); if (index(so, "/usr/local/") == 1) { next }; gsub("^/(usr/)?", "", so); print so }' | sort -u | xargs -r dpkg-query --search | cut -d: -f1 | sort -u | xargs -r apt-mark manual && apt-get purge -y --auto-remove -o APT::AutoRemove::RecommendsImportant=false && yarn --version && rm -rf /tmp/* # buildkit
# 2026-01-15 02:03:51 0.00B 设置环境变量 YARN_VERSION
ENV YARN_VERSION=1.22.22
# 2026-01-15 02:03:51 116.65MB 执行命令并创建新的镜像层
RUN /bin/sh -c ARCH= OPENSSL_ARCH= && dpkgArch="$(dpkg --print-architecture)" && case "${dpkgArch##*-}" in amd64) ARCH='x64' OPENSSL_ARCH='linux-x86_64';; ppc64el) ARCH='ppc64le' OPENSSL_ARCH='linux-ppc64le';; s390x) ARCH='s390x' OPENSSL_ARCH='linux*-s390x';; arm64) ARCH='arm64' OPENSSL_ARCH='linux-aarch64';; armhf) ARCH='armv7l' OPENSSL_ARCH='linux-armv4';; i386) ARCH='x86' OPENSSL_ARCH='linux-elf';; *) echo "unsupported architecture"; exit 1 ;; esac && set -ex && apt-get update && apt-get install -y ca-certificates curl wget gnupg dirmngr xz-utils libatomic1 --no-install-recommends && rm -rf /var/lib/apt/lists/* && export GNUPGHOME="$(mktemp -d)" && for key in 5BE8A3F6C8A5C01D106C0AD820B1A390B168D356 DD792F5973C6DE52C432CBDAC77ABFA00DDBF2B7 CC68F5A3106FF448322E48ED27F5E38D5B0A215F 8FCCA13FEF1D0C2E91008E09770F7A9A5AE15600 890C08DB8579162FEE0DF9DB8BEAB4DFCF555EF4 C82FA3AE1CBEDC6BE46B9360C43CEC45C17AB93C 108F52B48DB57BB0CC439B2997B01419BD92F80A A363A499291CBBC940DD62E41F10027AF002F8B0 ; do { gpg --batch --keyserver hkps://keys.openpgp.org --recv-keys "$key" && gpg --batch --fingerprint "$key"; } || { gpg --batch --keyserver keyserver.ubuntu.com --recv-keys "$key" && gpg --batch --fingerprint "$key"; } ; done && curl -fsSLO --compressed "https://nodejs.org/dist/v$NODE_VERSION/node-v$NODE_VERSION-linux-$ARCH.tar.xz" && curl -fsSLO --compressed "https://nodejs.org/dist/v$NODE_VERSION/SHASUMS256.txt.asc" && gpg --batch --decrypt --output SHASUMS256.txt SHASUMS256.txt.asc && gpgconf --kill all && rm -rf "$GNUPGHOME" && grep " node-v$NODE_VERSION-linux-$ARCH.tar.xz\$" SHASUMS256.txt | sha256sum -c - && tar -xJf "node-v$NODE_VERSION-linux-$ARCH.tar.xz" -C /usr/local --strip-components=1 --no-same-owner && rm "node-v$NODE_VERSION-linux-$ARCH.tar.xz" SHASUMS256.txt.asc SHASUMS256.txt && find /usr/local/include/node/openssl/archs -mindepth 1 -maxdepth 1 ! -name "$OPENSSL_ARCH" -exec rm -rf {} \; && apt-mark auto '.*' > /dev/null && find /usr/local -type f -executable -exec ldd '{}' ';' | awk '/=>/ { so = $(NF-1); if (index(so, "/usr/local/") == 1) { next }; gsub("^/(usr/)?", "", so); print so }' | sort -u | xargs -r dpkg-query --search | cut -d: -f1 | sort -u | xargs -r apt-mark manual && apt-get purge -y --auto-remove -o APT::AutoRemove::RecommendsImportant=false && ln -s /usr/local/bin/node /usr/local/bin/nodejs && node --version && npm --version && rm -rf /tmp/* # buildkit
# 2026-01-15 02:03:51 0.00B 设置环境变量 NODE_VERSION
ENV NODE_VERSION=20.20.0
# 2026-01-15 02:03:32 8.90KB 执行命令并创建新的镜像层
RUN /bin/sh -c groupadd --gid 1000 node && useradd --uid 1000 --gid node --shell /bin/bash --create-home node # buildkit
# 2026-01-12 08:00:00 97.20MB
# debian.sh --arch 'arm64' out/ 'bookworm' '@1768176000'
镜像信息
{
"Id": "sha256:c21e229cdaa4bf6baedd3826b02d65fc2da633890521cbef1c74188c1902592b",
"RepoTags": [
"ghcr.io/frankiejun/wxpush:latest",
"swr.cn-north-4.myhuaweicloud.com/ddn-k8s/ghcr.io/frankiejun/wxpush:latest-linuxarm64"
],
"RepoDigests": [
"ghcr.io/frankiejun/wxpush@sha256:9c76e4161d9711befe9a800258398dc450df92bb75c799f7fa82bb16ea4491b9",
"swr.cn-north-4.myhuaweicloud.com/ddn-k8s/ghcr.io/frankiejun/wxpush@sha256:7a6f01575332b7cffb5bc5f0b16b507e2e20853b25721dac9e963ab54ee3bf5b"
],
"Parent": "",
"Comment": "buildkit.dockerfile.v0",
"Created": "2026-01-24T13:32:11.07233655Z",
"Container": "",
"ContainerConfig": null,
"DockerVersion": "",
"Author": "",
"Config": {
"Hostname": "",
"Domainname": "",
"User": "",
"AttachStdin": false,
"AttachStdout": false,
"AttachStderr": false,
"ExposedPorts": {
"3939/tcp": {}
},
"Tty": false,
"OpenStdin": false,
"StdinOnce": false,
"Env": [
"PATH=/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin",
"NODE_VERSION=20.20.0",
"YARN_VERSION=1.22.22"
],
"Cmd": [
"node",
"/app/runtime/server.mjs"
],
"ArgsEscaped": true,
"Image": "",
"Volumes": null,
"WorkingDir": "/app",
"Entrypoint": [
"docker-entrypoint.sh"
],
"OnBuild": null,
"Labels": null
},
"Architecture": "arm64",
"Os": "linux",
"Size": 221074401,
"GraphDriver": {
"Data": {
"LowerDir": "/var/lib/docker/overlay2/e41b60ee4261898cb78332e433c39c370c46a6f7d6c5f4bc88416d5e8e4e7cf7/diff:/var/lib/docker/overlay2/a6791cd99df657b3383cac5a5ac9760e3c3fd867f00af200f816896ef3225d9c/diff:/var/lib/docker/overlay2/30ec1bd73adbe91eceb13b2f35d6cee27fb6e83c80d64c5afcfa61f113b9dce2/diff:/var/lib/docker/overlay2/0fa1b20f0d0020121bf2b83d2ca225f62121b18844e9789f18941cbe8e4f9684/diff:/var/lib/docker/overlay2/92c8e597162eed62bf1ef7ca524df31de6818b5fc40293d2e471e6734fa38e19/diff:/var/lib/docker/overlay2/353379a23c062a46b4ed5b2a44c965c2ac30695cbd897440f9496b3af277626d/diff:/var/lib/docker/overlay2/87f9db75b98b4172ea129853a96f115d20e6bc219a4ec858b06e7e80d567c010/diff:/var/lib/docker/overlay2/7ce5887b03ac1a6860958630d0d7ba155786198e2582c6327b9f99ddd49e7be5/diff",
"MergedDir": "/var/lib/docker/overlay2/7746268f7296226637f8a109f5a3d81ef1619df0b65fb78f46ea8c9dafdb36cf/merged",
"UpperDir": "/var/lib/docker/overlay2/7746268f7296226637f8a109f5a3d81ef1619df0b65fb78f46ea8c9dafdb36cf/diff",
"WorkDir": "/var/lib/docker/overlay2/7746268f7296226637f8a109f5a3d81ef1619df0b65fb78f46ea8c9dafdb36cf/work"
},
"Name": "overlay2"
},
"RootFS": {
"Type": "layers",
"Layers": [
"sha256:27718dd4c00f9fd135129ee24338fc13706d328b1ca5ea353bc79772eb520fa4",
"sha256:c097e30691d1a2117aba31879ac717c153a634f87ee830bfd461f5f34b56f073",
"sha256:bdfdf27ea5fea83e08bc5c910bdc6350dac81b82c7ce463ce801d6a9c735efac",
"sha256:6ec46b4558819571b1a4f85c78decd4c0a0e66db2913c39caca27f6acf8f13df",
"sha256:5e8e2f6f23efc9403ab44bad7e98b20cb6c517b40e515c38da2fde5dddcb6025",
"sha256:28dbbbace3e8207318e6b62afe47e4449d0646c266eaeb748a2e9e5447345e0e",
"sha256:fe9013614490e9fa44aebc1eb5db1cc3bc8df4d8923244ed7e844675ddf075ff",
"sha256:bba65d7f33ed2514290e5ace6826faac47c97d0be107a887ad288fc29f5a2b9d",
"sha256:93c27f3a3808761d10e0dd89599c0d50ed05deb3b40fdd5feb4e359353d989b1"
]
},
"Metadata": {
"LastTagTime": "2026-03-13T21:10:21.480269048+08:00"
}
}