Bun.build
ビルドすると以下のような感じになります。
.├── node_modules├── dist/│ ├── src/│ │ └── main.js│ ├── package.json│ └── tsconfig.build.tsbuildinfo├── package.json├── tsconfig.json└── tsconfig.build.jsonNodeJSでビルドしたときだとnode_modulesとdistがあればnode dist/src/mainで起動できました。
Bunの場合はbun dist/src/main.jsで起動できます。Nodeの場合と違って拡張子を指定しなければいけないようです。
Dockerfile
Bunの公式サイトにDockerfileのドキュメントがあるのでそれを参考にしてみます。
FROM oven/bun:1 AS base
# Development InstallFROM base AS installWORKDIR /app/devCOPY package.json bun.lockb ./RUN bun install --frozen-lockfile
# Production InstallWORKDIR /app/prodCOPY package.json bun.lockb ./RUN bun install --frozen-lockfile --production --ignore-scripts
# Pre ReleaseFROM base AS prereleaseWORKDIR /appCOPY --from=install /app/dev/node_modules node_modulesCOPY . .
# Test and BuildENV NODE_ENV=productionRUN bun testRUN bun run build途中まで書くとこういう感じになります。
実行時にはdevDependenciesは不要なのですがビルドのときには必要な場合があるのでビルドのときにはDevelopmentの方のnode_modulesを利用します。
なのでPre Releaseのところでは/app/dev/node_modulesの方を使っているわけですね。
そしてもともとあるソースコードを利用してテストとビルドを実行します。
ここのビルドが通れば上で書いたようなディレクトリ構成でトランスパイルされたものが出力されるのでPre Releaseのコンテナは以下のようになるはずです。
app/├── node_modules├── src/│ └── main.ts├── dist/│ ├── src/│ │ └── main.js│ ├── package.json│ └── tsconfig.build.tsbuildinfo├── package.json├── tsconfig.json└── tsconfig.build.jsonただ、実際にはCOPY . . でディレクトリの全てのファイルをコピーしてきているので余計なファイルも多分に含まれています。余計なファイルを同梱するとイメージのサイズが大きくなるのでそれらは含まないように配慮してリリース用のイメージにコピーします。
リリース用イメージ
ここで再度公式ドキュメントを参考にコピーするファイルを選定します。
FROM base AS releaseWORKDIR /appCOPY --from=install /app/dev/node_modules ./node_modulesCOPY --from=prerelease /app/dist/src ./srcCOPY --from=prerelease /app/package.json ./package.jsonCOPY --from=prerelease /app/tsconfig.json ./tsconfig.jsonよくわからないのですがBunは実行時にもどうもpackage.jsonとtsconfig.jsonが必要なので突っ込みました。
あと、ビルドした段階ではNestJSはデフォルトで./distにトランスパイルしたコードが出力されるのですがこれをそのまま突っ込むとモジュールの参照エラーが発生します。
どうもbun run **/main.jsを実行したときのディレクトリからの相対パスが問題っぽくbun run src/main.jsで実行できるようなディレクトリ構造になっていないとダメなようです。
なのでCOPY --from=prerelease /app/dist/src ./srcとしてdist/srcの中身をsrcにコピーするという若干ややこしいことをしています。