fix(docker): reconcile pnpm workspace symlinks at container start
The bind mount (.:/app) provides workspace-level node_modules symlinks from the host, but those target the root node_modules/.pnpm store which inside the container is a named volume with different content-addressable hashes. Added `pnpm install --frozen-lockfile` to app-dev-start.sh so symlinks are regenerated against the container's store on every boot. Also adds restart.sh convenience script for image rebuilds. Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
This commit is contained in:
Executable
+50
@@ -0,0 +1,50 @@
|
||||
#!/usr/bin/env bash
|
||||
# restart.sh — Rebuild the CapaKraken app container from scratch.
|
||||
#
|
||||
# When to use:
|
||||
# - After changing pnpm-lock.yaml (new/removed dependencies)
|
||||
# - After Prisma schema changes
|
||||
# - After Dockerfile.dev changes
|
||||
# - When you see "Cannot find module" errors in the app container
|
||||
#
|
||||
# A plain `docker compose restart` only restarts the existing container — it
|
||||
# does NOT rebuild the image or reinstall dependencies.
|
||||
#
|
||||
# How it works:
|
||||
# The startup script (tooling/docker/app-dev-start.sh) runs `pnpm install`
|
||||
# at boot to reconcile host-side workspace symlinks with the container's
|
||||
# pnpm store in the named volume. This script rebuilds the Docker image
|
||||
# (which re-runs pnpm install into the image layer), then optionally purges
|
||||
# stale named volumes so the startup install starts fresh.
|
||||
#
|
||||
# Usage:
|
||||
# ./restart.sh # rebuild app image and restart
|
||||
# ./restart.sh --clean # also remove node_modules + .next volumes
|
||||
# ./restart.sh --full # --clean + recreate all services (postgres, redis, etc.)
|
||||
|
||||
set -euo pipefail
|
||||
|
||||
PROFILE="full"
|
||||
SERVICES="app"
|
||||
CLEAN=false
|
||||
|
||||
for arg in "$@"; do
|
||||
case "$arg" in
|
||||
--clean) CLEAN=true ;;
|
||||
--full) CLEAN=true; SERVICES="" ;; # empty = all services in the profile
|
||||
esac
|
||||
done
|
||||
|
||||
echo "==> Stopping app container..."
|
||||
docker compose --profile "$PROFILE" stop app 2>/dev/null || true
|
||||
|
||||
if $CLEAN; then
|
||||
echo "==> Removing stale node_modules and .next volumes..."
|
||||
docker volume rm capakraken_node_modules capakraken_next 2>/dev/null || true
|
||||
fi
|
||||
|
||||
echo "==> Rebuilding and starting ($( [[ -z "$SERVICES" ]] && echo "all services" || echo "$SERVICES" ))..."
|
||||
docker compose --profile "$PROFILE" up --build -d $SERVICES
|
||||
|
||||
echo "==> Tailing logs (Ctrl-C to detach)..."
|
||||
docker compose --profile "$PROFILE" logs -f app
|
||||
@@ -8,6 +8,16 @@ until pg_isready -h "${POSTGRES_HOST:-postgres}" -p "${POSTGRES_PORT:-5432}" -q;
|
||||
done
|
||||
echo "Postgres is ready."
|
||||
|
||||
# The bind mount (.:/app) provides workspace-level node_modules symlinks from
|
||||
# the *host*, but those symlinks target the root node_modules/.pnpm store.
|
||||
# Inside the container, /app/node_modules is a named volume whose pnpm
|
||||
# content-addressable hashes may differ from the host's. Re-running install
|
||||
# regenerates the workspace symlinks so they resolve against the container's
|
||||
# pnpm store. --frozen-lockfile ensures no lock changes; --prefer-offline
|
||||
# avoids network round-trips when packages are already cached in the volume.
|
||||
# CI=true suppresses interactive prompts (e.g. "reinstall from scratch?")
|
||||
CI=true pnpm install --frozen-lockfile
|
||||
|
||||
# Regenerate Prisma client (needed after bind-mount overlays the image layer)
|
||||
pnpm --filter @capakraken/db db:generate
|
||||
|
||||
|
||||
Reference in New Issue
Block a user