S
SolidJSโ€ข3w ago
binajmen

Spooky behaviour while deploying in production

Hi everyone, I'm having a hard time to deploy solid start. I'm using docker on a VPS, and from one deployment to another, the bundled assets are sometimes corrupted. I noticed two strange behaviours so far: 1. Some bundled assets seems incomplete This is an error I get when visiting a broken page:
SyntaxError: Invalid or unexpected token (at select-Ca0Q99V8.js:1:1827)
SyntaxError: Invalid or unexpected token (at select-Ca0Q99V8.js:1:1827)
And indeed this is the end of the file:
...={data:[{id:"df1",points:100,requirement:"
...={data:[{id:"df1",points:100,requirement:"
Literally closing on " 2. Some link are strangely leading to other page content : If I visit the link /foo, it shows me the content of /bar. I can re-run the deploy several time, and at some point it works.. ๐Ÿ‘€ Here is the script I use:
#!/bin/bash
# Aggressive cleanup
echo "๐Ÿงน Performing thorough cleanup..."
docker image prune -f
docker builder prune -f # Clean build cache

echo "๐Ÿ“ฅ Pulling latest changes..."
git pull origin main

# Generate new build tag
export BUILD_TAG=$(date +%Y%m%d_%H%M%S)
echo "๐Ÿท๏ธ New build tag: $BUILD_TAG"

echo "๐Ÿ—๏ธ Building new images..."
# Use buildkit for better caching behavior
export DOCKER_BUILDKIT=1
docker compose build --no-cache --progress=plain

echo "โฌ‡๏ธ Taking down current containers..."
docker compose down

echo "โฌ†๏ธ Starting new containers..."
docker compose up -d

echo "๐Ÿ“‹ Showing logs..."
docker compose logs -f
#!/bin/bash
# Aggressive cleanup
echo "๐Ÿงน Performing thorough cleanup..."
docker image prune -f
docker builder prune -f # Clean build cache

echo "๐Ÿ“ฅ Pulling latest changes..."
git pull origin main

# Generate new build tag
export BUILD_TAG=$(date +%Y%m%d_%H%M%S)
echo "๐Ÿท๏ธ New build tag: $BUILD_TAG"

echo "๐Ÿ—๏ธ Building new images..."
# Use buildkit for better caching behavior
export DOCKER_BUILDKIT=1
docker compose build --no-cache --progress=plain

echo "โฌ‡๏ธ Taking down current containers..."
docker compose down

echo "โฌ†๏ธ Starting new containers..."
docker compose up -d

echo "๐Ÿ“‹ Showing logs..."
docker compose logs -f
10 Replies
binajmen
binajmenOPโ€ข3w ago
I would love to share a reproduction but this is a private project so I cannot share it publicly. However I can add some individuals from the core team if this is a legit concern. Any ideas how I can try to figure out what is happening?
binajmen
binajmenOPโ€ข3w ago
The closest issue related to bundling issue is: https://github.com/solidjs/solid-start/issues/1570
GitHub
[Bug?]: Wrong server function called due to incorrect handling of f...
Duplicates I have searched the existing issues Latest version I have tested the latest version Current behavior ๐Ÿ˜ฏ With a nested structure for server functions, the hashing of the files might have a...
binajmen
binajmenOPโ€ข3w ago
But I'm unsure is this is related or not. @jayson.kt I'm sorry for the direct ping, but as the author of the issue, did you find any additional explanations that could help me in my current situation? This other issue also describe my second point: https://github.com/nksaraf/vinxi/issues/326 @lominming I'm as well sorry for the direct ping, but did you find out a way to resolve your issue?
Madaxen86
Madaxen86โ€ข3w ago
Can you share your package.json and app.config.ts and your dockerfile / docker-compose Or maybe create a minimal reproduction with a fresh install on a separate repository?
binajmen
binajmenOPโ€ข3w ago
package.json
{
"name": "cockpit",
"type": "module",
"scripts": {
"dev": "vinxi dev",
"build": "vinxi build",
"start": "vinxi start",
"version": "vinxi version"
},
"dependencies": {
"@atlaskit/pragmatic-drag-and-drop": "^1.4.0",
"@kobalte/core": "^0.13.7",
"@solid-primitives/map": "^0.5.0",
"@solidjs/meta": "^0.29.4",
"@solidjs/router": "^0.15.2",
"@solidjs/start": "^1.0.10",
"@tanstack/solid-query": "^5.62.12",
"@tanstack/solid-table": "^8.20.5",
"clsx": "^2.1.1",
"dotenv": "^16.4.7",
"drizzle-orm": "^0.38.3",
"dropbox": "^10.34.0",
"marked": "^15.0.5",
"pg": "^8.13.1",
"solid-js": "^1.9.3",
"tailwind-merge": "^2.6.0",
"tiny-invariant": "^1.3.3",
"vinxi": "0.4.3",
"xlsx": "https://cdn.sheetjs.com/xlsx-0.20.3/xlsx-0.20.3.tgz",
"zod": "^3.24.1"
},
"engines": {
"node": ">=18"
},
"devDependencies": {
"@biomejs/biome": "1.9.4",
"@iconify-json/lucide": "^1.2.21",
"@tailwindcss/forms": "^0.5.9",
"@types/bun": "^1.1.14",
"@types/pg": "^8.11.10",
"autoprefixer": "^10.4.20",
"drizzle-kit": "^0.30.1",
"postcss": "^8.4.49",
"solid-devtools": "^0.33.0",
"tailwindcss": "^3.4.17",
"unplugin-icons": "^0.22.0",
"vitest": "^2.1.8"
},
"patchedDependencies": {
}
}
{
"name": "cockpit",
"type": "module",
"scripts": {
"dev": "vinxi dev",
"build": "vinxi build",
"start": "vinxi start",
"version": "vinxi version"
},
"dependencies": {
"@atlaskit/pragmatic-drag-and-drop": "^1.4.0",
"@kobalte/core": "^0.13.7",
"@solid-primitives/map": "^0.5.0",
"@solidjs/meta": "^0.29.4",
"@solidjs/router": "^0.15.2",
"@solidjs/start": "^1.0.10",
"@tanstack/solid-query": "^5.62.12",
"@tanstack/solid-table": "^8.20.5",
"clsx": "^2.1.1",
"dotenv": "^16.4.7",
"drizzle-orm": "^0.38.3",
"dropbox": "^10.34.0",
"marked": "^15.0.5",
"pg": "^8.13.1",
"solid-js": "^1.9.3",
"tailwind-merge": "^2.6.0",
"tiny-invariant": "^1.3.3",
"vinxi": "0.4.3",
"xlsx": "https://cdn.sheetjs.com/xlsx-0.20.3/xlsx-0.20.3.tgz",
"zod": "^3.24.1"
},
"engines": {
"node": ">=18"
},
"devDependencies": {
"@biomejs/biome": "1.9.4",
"@iconify-json/lucide": "^1.2.21",
"@tailwindcss/forms": "^0.5.9",
"@types/bun": "^1.1.14",
"@types/pg": "^8.11.10",
"autoprefixer": "^10.4.20",
"drizzle-kit": "^0.30.1",
"postcss": "^8.4.49",
"solid-devtools": "^0.33.0",
"tailwindcss": "^3.4.17",
"unplugin-icons": "^0.22.0",
"vitest": "^2.1.8"
},
"patchedDependencies": {
}
}
app.config.ts
import { defineConfig } from "@solidjs/start/config";
import devtools from "solid-devtools/vite";
import Icons from "unplugin-icons/vite";

export default defineConfig({
ssr: false,
middleware: "./src/middlewares/index.ts",
server: {
compatibilityDate: "2025-01-28",
logLevel: 4,
},
vite: {
ssr: {
noExternal: ["@atlaskit/pragmatic-drag-and-drop"],
},
plugins: [devtools({ autoname: true }), Icons({ compiler: "solid" })],
},
});
import { defineConfig } from "@solidjs/start/config";
import devtools from "solid-devtools/vite";
import Icons from "unplugin-icons/vite";

export default defineConfig({
ssr: false,
middleware: "./src/middlewares/index.ts",
server: {
compatibilityDate: "2025-01-28",
logLevel: 4,
},
vite: {
ssr: {
noExternal: ["@atlaskit/pragmatic-drag-and-drop"],
},
plugins: [devtools({ autoname: true }), Icons({ compiler: "solid" })],
},
});
Dockerfile
# use the official Bun image
FROM oven/bun:1.2.0 AS base
ARG BUILD_TAG
WORKDIR /app

# install dependencies into temp directory
FROM base AS install
RUN mkdir -p /temp/dev
COPY package.json bun.lockb /temp/dev/
COPY patches/ /temp/dev/patches
RUN cd /temp/dev && bun install --frozen-lockfile

# install with --production (exclude devDependencies)
RUN mkdir -p /temp/prod
COPY package.json bun.lockb /temp/prod/
COPY patches/ /temp/prod/patches
RUN cd /temp/prod && bun install --frozen-lockfile --production

# copy node_modules from temp directory
# then copy all (non-ignored) project files into the image
FROM base AS prerelease
COPY --from=install /temp/dev/node_modules node_modules
COPY . .

# build the application
ENV NODE_ENV=production
RUN bun run build
RUN chmod +x /app/docker-entrypoint.sh

# copy production dependencies and source code into final image
FROM base AS release
COPY --from=install /temp/dev/node_modules node_modules
COPY --from=prerelease /app/.output .output
COPY --from=prerelease /app/package.json .
COPY --from=prerelease /app/docker-entrypoint.sh .

# this files are necessary to perform the drizzle migration
COPY --from=prerelease /app/drizzle.config.ts .
COPY --from=prerelease /app/drizzle drizzle/

LABEL build_tag=$BUILD_TAG

USER bun

ENTRYPOINT ["./docker-entrypoint.sh"]
# use the official Bun image
FROM oven/bun:1.2.0 AS base
ARG BUILD_TAG
WORKDIR /app

# install dependencies into temp directory
FROM base AS install
RUN mkdir -p /temp/dev
COPY package.json bun.lockb /temp/dev/
COPY patches/ /temp/dev/patches
RUN cd /temp/dev && bun install --frozen-lockfile

# install with --production (exclude devDependencies)
RUN mkdir -p /temp/prod
COPY package.json bun.lockb /temp/prod/
COPY patches/ /temp/prod/patches
RUN cd /temp/prod && bun install --frozen-lockfile --production

# copy node_modules from temp directory
# then copy all (non-ignored) project files into the image
FROM base AS prerelease
COPY --from=install /temp/dev/node_modules node_modules
COPY . .

# build the application
ENV NODE_ENV=production
RUN bun run build
RUN chmod +x /app/docker-entrypoint.sh

# copy production dependencies and source code into final image
FROM base AS release
COPY --from=install /temp/dev/node_modules node_modules
COPY --from=prerelease /app/.output .output
COPY --from=prerelease /app/package.json .
COPY --from=prerelease /app/docker-entrypoint.sh .

# this files are necessary to perform the drizzle migration
COPY --from=prerelease /app/drizzle.config.ts .
COPY --from=prerelease /app/drizzle drizzle/

LABEL build_tag=$BUILD_TAG

USER bun

ENTRYPOINT ["./docker-entrypoint.sh"]
compose.yml
name: cockpit

services:
web:
build:
context: .
dockerfile: Dockerfile
target: release
args:
BUILD_TAG: ${BUILD_TAG}
image: cockpit-web:${BUILD_TAG}
env_file:
- .env
ports:
- "3110:3000"
user: bun
restart: unless-stopped
depends_on:
postgres:
condition: service_healthy
postgres:
image: postgres:17
restart: unless-stopped
environment:
POSTGRES_USER: ${POSTGRES_USER}
POSTGRES_PASSWORD: ${POSTGRES_PASSWORD}
POSTGRES_DB: ${POSTGRES_DB}
ports:
- "5435:5432"
volumes:
- postgres_data:/var/lib/postgresql/data
healthcheck:
test: ["CMD", "pg_isready", "-U", "postgres"]
interval: 10s
timeout: 5s
retries: 5

volumes:
postgres_data:
name: cockpit

services:
web:
build:
context: .
dockerfile: Dockerfile
target: release
args:
BUILD_TAG: ${BUILD_TAG}
image: cockpit-web:${BUILD_TAG}
env_file:
- .env
ports:
- "3110:3000"
user: bun
restart: unless-stopped
depends_on:
postgres:
condition: service_healthy
postgres:
image: postgres:17
restart: unless-stopped
environment:
POSTGRES_USER: ${POSTGRES_USER}
POSTGRES_PASSWORD: ${POSTGRES_PASSWORD}
POSTGRES_DB: ${POSTGRES_DB}
ports:
- "5435:5432"
volumes:
- postgres_data:/var/lib/postgresql/data
healthcheck:
test: ["CMD", "pg_isready", "-U", "postgres"]
interval: 10s
timeout: 5s
retries: 5

volumes:
postgres_data:
I will certainly try to replicate this issue in a minimal reproduction project. However the fact that this is "self-healing" after n retries is.. troublesome.
Madaxen86
Madaxen86โ€ข3w ago
At first glance: BUN Have you tried to use a node image instead?
binajmen
binajmenOPโ€ข3w ago
Oh, are you implying bun is not yet supported? I was not aware of that!
Benedict
Benedictโ€ข3w ago
I also had problems getting bun to run in docker when I was using nextjs
binajmen
binajmenOPโ€ข3w ago
mmh, i guess switching from bun to node should be as transparent as switching from node to bun. is there an impact I'm unaware of except the performance if switching to node?
zulu
zuluโ€ข3w ago
first check if it works if it works non of this matter if it works after some build, maybe it is something with docker. can you build without all the docker wrappers?

Did you find this page helpful?