fix: Update docker configuration for production deployment

- Fixed NEXT_PUBLIC_NOTES_API_URL to use public API endpoint
- Updated docker-compose.yml environment format to proper YAML
- Updated Dockerfiles to remove Gitea secrets and use .docker-deps
- Added docker-prep.sh script for dependency packaging
- Changed NODE_ENV back to development for compatibility with memory DB

Generated with [Devin](https://cli.devin.ai/docs)

Co-Authored-By: Devin <158243242+devin-ai-integration[bot]@users.noreply.github.com>
This commit is contained in:
root 2026-05-12 08:20:12 +00:00
parent 4337793034
commit 3dd981198e
5 changed files with 189 additions and 45 deletions

View File

@ -1 +1 @@
@bytelyst:registry=http://gitea.bytelyst.com:3300/api/packages/bytelyst/npm/ @bytelyst:registry=http://localhost:3300/api/packages/bytelyst/npm/

View File

@ -4,12 +4,10 @@ WORKDIR /app/backend
RUN npm install -g pnpm@10.6.5 RUN npm install -g pnpm@10.6.5
COPY .npmrc.docker ./.npmrc COPY .npmrc.docker ./.npmrc
COPY .docker-deps/ ../.docker-deps/
COPY backend/package.json ./package.json COPY backend/package.json ./package.json
RUN --mount=type=secret,id=gitea_npm_token \ RUN pnpm install --ignore-scripts --lockfile=false
TOKEN=$(cat /run/secrets/gitea_npm_token) && \
echo "//gitea.bytelyst.com:3300/:_authToken=$TOKEN" >> .npmrc && \
pnpm install --ignore-scripts --lockfile=false
COPY backend/tsconfig.json ./tsconfig.json COPY backend/tsconfig.json ./tsconfig.json
COPY backend/src/ ./src/ COPY backend/src/ ./src/

View File

@ -1,37 +1,38 @@
services: services:
backend: backend:
container_name: notelett-backend
build: build:
context: . context: .
dockerfile: backend/Dockerfile dockerfile: backend/Dockerfile
ports: ports:
- "4016:4016" - "4016:4016"
environment: environment:
- NODE_ENV=${NODE_ENV:-development} NODE_ENV: development
- PORT=4016 PORT: 4016
- HOST=0.0.0.0 HOST: 0.0.0.0
- PRODUCT_ID=notelett PRODUCT_ID: notelett
- SERVICE_NAME=notelett-backend SERVICE_NAME: notelett-backend
- JWT_SECRET=${JWT_SECRET:-dev-secret-change-me-at-least-32-characters} JWT_SECRET: ${JWT_SECRET:-dev-secret-change-me-at-least-32-characters}
- COSMOS_ENDPOINT=${COSMOS_ENDPOINT:-} COSMOS_ENDPOINT: ${COSMOS_ENDPOINT:-}
- COSMOS_KEY=${COSMOS_KEY:-} COSMOS_KEY: ${COSMOS_KEY:-}
- COSMOS_DATABASE=${COSMOS_DATABASE:-bytelyst} COSMOS_DATABASE: ${COSMOS_DATABASE:-bytelyst}
- DB_PROVIDER=${DB_PROVIDER:-memory} DB_PROVIDER: ${DB_PROVIDER:-memory}
- CORS_ORIGIN=${CORS_ORIGIN:-http://localhost:3000} CORS_ORIGIN: ${CORS_ORIGIN:-http://localhost:3000}
- PLATFORM_SERVICE_URL=${PLATFORM_SERVICE_URL:-http://localhost:4003} PLATFORM_SERVICE_URL: ${PLATFORM_SERVICE_URL:-http://localhost:4003}
- EXTRACTION_SERVICE_URL=${EXTRACTION_SERVICE_URL:-http://localhost:4005} EXTRACTION_SERVICE_URL: ${EXTRACTION_SERVICE_URL:-http://localhost:4005}
- MCP_SERVER_URL=${MCP_SERVER_URL:-http://localhost:4007} MCP_SERVER_URL: ${MCP_SERVER_URL:-http://localhost:4007}
- TELEMETRY_ENABLED=${TELEMETRY_ENABLED:-false} TELEMETRY_ENABLED: ${TELEMETRY_ENABLED:-false}
- FEATURE_FLAGS_ENABLED=${FEATURE_FLAGS_ENABLED:-false} FEATURE_FLAGS_ENABLED: ${FEATURE_FLAGS_ENABLED:-false}
- FIELD_ENCRYPT_ENABLED=${FIELD_ENCRYPT_ENABLED:-false} FIELD_ENCRYPT_ENABLED: ${FIELD_ENCRYPT_ENABLED:-false}
- FIELD_ENCRYPT_KEY_PROVIDER=${FIELD_ENCRYPT_KEY_PROVIDER:-memory} FIELD_ENCRYPT_KEY_PROVIDER: ${FIELD_ENCRYPT_KEY_PROVIDER:-memory}
- LLM_PROVIDER=${LLM_PROVIDER:-mock} LLM_PROVIDER: ${LLM_PROVIDER:-mock}
- OPENAI_API_KEY=${OPENAI_API_KEY:-} OPENAI_API_KEY: ${OPENAI_API_KEY:-}
- OPENAI_BASE_URL=${OPENAI_BASE_URL:-} OPENAI_BASE_URL: ${OPENAI_BASE_URL:-}
- AZURE_OPENAI_ENDPOINT=${AZURE_OPENAI_ENDPOINT:-} AZURE_OPENAI_ENDPOINT: ${AZURE_OPENAI_ENDPOINT:-}
- AZURE_OPENAI_API_KEY=${AZURE_OPENAI_API_KEY:-} AZURE_OPENAI_API_KEY: ${AZURE_OPENAI_API_KEY:-}
- LLM_DEFAULT_MODEL=${LLM_DEFAULT_MODEL:-gpt-4o-mini} LLM_DEFAULT_MODEL: ${LLM_DEFAULT_MODEL:-gpt-4o-mini}
- LLM_VISION_MODEL=${LLM_VISION_MODEL:-gpt-4o} LLM_VISION_MODEL: ${LLM_VISION_MODEL:-gpt-4o}
- LLM_EMBEDDING_MODEL=${LLM_EMBEDDING_MODEL:-text-embedding-3-small} LLM_EMBEDDING_MODEL: ${LLM_EMBEDDING_MODEL:-text-embedding-3-small}
restart: unless-stopped restart: unless-stopped
healthcheck: healthcheck:
test: ["CMD-SHELL", "node -e \"fetch('http://localhost:4016/health').then(r=>process.exit(r.ok?0:1)).catch(()=>process.exit(1))\""] test: ["CMD-SHELL", "node -e \"fetch('http://localhost:4016/health').then(r=>process.exit(r.ok?0:1)).catch(()=>process.exit(1))\""]
@ -40,24 +41,25 @@ services:
retries: 3 retries: 3
web: web:
container_name: notelett-web
build: build:
context: . context: .
dockerfile: web/Dockerfile dockerfile: web/Dockerfile
args: args:
NEXT_PUBLIC_NOTES_API_URL: ${NEXT_PUBLIC_NOTES_API_URL:-http://localhost:4016/api} NEXT_PUBLIC_NOTES_API_URL: https://api.bytelyst.com/notelett
NEXT_PUBLIC_PLATFORM_SERVICE_URL: ${NEXT_PUBLIC_PLATFORM_SERVICE_URL:-http://localhost:4003/api} NEXT_PUBLIC_PLATFORM_SERVICE_URL: ${NEXT_PUBLIC_PLATFORM_SERVICE_URL:-http://localhost:4003/api}
ports: ports:
- "3000:3045" - "3000:3045"
environment: environment:
- NODE_ENV=production NODE_ENV: production
- NEXT_PUBLIC_PRODUCT_NAME=NoteLett NEXT_PUBLIC_PRODUCT_NAME: NoteLett
- NEXT_PUBLIC_PRODUCT_ID=notelett NEXT_PUBLIC_PRODUCT_ID: notelett
- NEXT_PUBLIC_NOTES_API_URL=${NEXT_PUBLIC_NOTES_API_URL:-http://localhost:4016/api} NEXT_PUBLIC_NOTES_API_URL: https://api.bytelyst.com/notelett
- NEXT_PUBLIC_PLATFORM_SERVICE_URL=${NEXT_PUBLIC_PLATFORM_SERVICE_URL:-http://localhost:4003/api} NEXT_PUBLIC_PLATFORM_SERVICE_URL: ${NEXT_PUBLIC_PLATFORM_SERVICE_URL:-http://localhost:4003/api}
- NEXT_PUBLIC_EXTRACTION_SERVICE_URL=${EXTRACTION_SERVICE_URL:-http://localhost:4005} NEXT_PUBLIC_EXTRACTION_SERVICE_URL: ${EXTRACTION_SERVICE_URL:-http://localhost:4005}
- NEXT_PUBLIC_MCP_SERVER_URL=${MCP_SERVER_URL:-http://localhost:4007}/api NEXT_PUBLIC_MCP_SERVER_URL: ${MCP_SERVER_URL:-http://localhost:4007}/api
- NEXT_PUBLIC_DIAGNOSTICS_URL=${DIAGNOSTICS_URL:-http://localhost:3000} NEXT_PUBLIC_DIAGNOSTICS_URL: ${DIAGNOSTICS_URL:-http://localhost:3000}
- NEXT_PUBLIC_TELEMETRY_TRANSPORT=fetch NEXT_PUBLIC_TELEMETRY_TRANSPORT: fetch
depends_on: depends_on:
backend: backend:
condition: service_healthy condition: service_healthy

146
scripts/docker-prep.sh Executable file
View File

@ -0,0 +1,146 @@
#!/usr/bin/env bash
# Pack @bytelyst/* tarballs from the sibling common-plat repo for
# self-contained Docker builds that don't need the Gitea npm registry.
#
# Usage:
# ./scripts/docker-prep.sh # pack tarballs + rewrite package.json
# ./scripts/docker-prep.sh --restore # undo rewrite
set -euo pipefail
SCRIPT_DIR="$(cd "$(dirname "$0")" && pwd)"
REPO_DIR="$(cd "$SCRIPT_DIR/.." && pwd)"
COMMON_PLAT="${COMMON_PLAT:-${REPO_DIR}/../learning_ai/learning_ai_common_plat}"
if [[ ! -d "$COMMON_PLAT" && -d "${REPO_DIR}/../learning_ai_common_plat" ]]; then
COMMON_PLAT="${REPO_DIR}/../learning_ai_common_plat"
fi
TARBALL_DIR="${REPO_DIR}/.docker-deps"
# ── Restore mode ───────────────────────────────────────────────────
if [[ "${1:-}" == "--restore" ]]; then
echo "Restoring original package.json files..."
for bak in $(find "$REPO_DIR" -name "package.json.bak" -not -path "*/node_modules/*"); do
mv "$bak" "${bak%.bak}"
echo " Restored ${bak%.bak}"
done
rm -rf "$TARBALL_DIR"
echo "Done."
exit 0
fi
# ── Pack mode ──────────────────────────────────────────────────────
if [[ ! -d "$COMMON_PLAT" ]]; then
echo "Common platform checkout not found: $COMMON_PLAT" >&2
echo "Set COMMON_PLAT=/path/to/learning_ai_common_plat or place it at ../learning_ai/learning_ai_common_plat." >&2
exit 1
fi
echo "=== docker-prep: packing @bytelyst/* tarballs ==="
rm -rf "$TARBALL_DIR"
mkdir -p "$TARBALL_DIR"
# Build all packages first (--filter limits to packages/, skips services/)
echo "Building @bytelyst/* packages..."
(cd "$COMMON_PLAT" && pnpm -r --filter './packages/*' build)
# Pack each package and build a mapping of name → tarball filename
# (uses a temp file instead of associative array for bash 3.2 compat)
TARBALL_MAP_FILE=$(mktemp)
trap 'rm -f "$TARBALL_MAP_FILE"' EXIT
for pkg_dir in "$COMMON_PLAT"/packages/*/; do
pkg_name=$(node -p "require('${pkg_dir}package.json').name" 2>/dev/null || true)
if [[ -z "$pkg_name" ]]; then continue; fi
echo " Packing $pkg_name..."
tarball=$(cd "$pkg_dir" && pnpm pack --pack-destination "$TARBALL_DIR" 2>/dev/null | tail -1)
filename=$(basename "$tarball")
echo "${pkg_name}=${filename}" >> "$TARBALL_MAP_FILE"
echo " -> $filename"
done
# ── Rewrite package.json files ─────────────────────────────────────
echo ""
echo "Rewriting package.json @bytelyst/* refs to .docker-deps/ tarballs..."
rewrite_package_json() {
local pkg_file="$1"
local rel_prefix="$2" # relative path from package.json dir to repo root
if [[ ! -f "$pkg_file" ]]; then return; fi
# Backup
cp "$pkg_file" "${pkg_file}.bak"
local tmp="${pkg_file}.tmp"
cp "$pkg_file" "$tmp"
while IFS='=' read -r pkg_name tarball; do
[[ -z "$pkg_name" ]] && continue
node -e "
const fs = require('fs');
const file = process.argv[1];
const pkgName = process.argv[2];
const replacement = process.argv[3];
const p = JSON.parse(fs.readFileSync(file, 'utf8'));
for (const section of ['dependencies', 'devDependencies', 'peerDependencies', 'optionalDependencies']) {
if (p[section] && Object.prototype.hasOwnProperty.call(p[section], pkgName)) {
p[section][pkgName] = replacement;
}
}
fs.writeFileSync(file, JSON.stringify(p, null, 2) + '\n');
" "$tmp" "$pkg_name" "file:${rel_prefix}.docker-deps/${tarball}"
done < "$TARBALL_MAP_FILE"
mv "$tmp" "$pkg_file"
echo " Rewrote $pkg_file"
}
# Backend package.json
rewrite_package_json "${REPO_DIR}/backend/package.json" "../"
# Web package.json
rewrite_package_json "${REPO_DIR}/web/package.json" "../"
# ── Inject pnpm.overrides for transitive @bytelyst/* deps ─────────
# Tarball packages may depend on other @bytelyst/* packages (e.g.
# @bytelyst/fastify-core → @bytelyst/errors). Without overrides, pnpm
# tries to fetch them from the npm registry which fails.
inject_overrides() {
local pkg_file="$1"
local rel_prefix="$2"
if [[ ! -f "$pkg_file" ]]; then return; fi
local overrides=""
while IFS='=' read -r pkg_name tarball; do
[[ -z "$pkg_name" ]] && continue
if [[ -n "$overrides" ]]; then overrides="$overrides, "; fi
overrides="$overrides\"${pkg_name}\": \"file:${rel_prefix}.docker-deps/${tarball}\""
done < "$TARBALL_MAP_FILE"
if [[ -n "$overrides" ]]; then
node -e "
const fs = require('fs');
const p = JSON.parse(fs.readFileSync('${pkg_file}', 'utf8'));
p.pnpm = p.pnpm || {};
p.pnpm.overrides = { ...(p.pnpm.overrides || {}), ...JSON.parse('{${overrides}}') };
fs.writeFileSync('${pkg_file}', JSON.stringify(p, null, 2) + '\n');
"
echo " Injected pnpm.overrides into $pkg_file"
fi
}
inject_overrides "${REPO_DIR}/backend/package.json" "../"
inject_overrides "${REPO_DIR}/web/package.json" "../"
echo ""
echo "Done. Tarballs in $TARBALL_DIR"
echo ""
echo "To build Docker images:"
echo " docker compose build"
echo ""
echo "To restore after build:"
echo " ./scripts/docker-prep.sh --restore"

View File

@ -10,11 +10,9 @@ RUN npm config set strict-ssl false \
&& npm install -g pnpm@10.6.5 && npm install -g pnpm@10.6.5
COPY .npmrc.docker ./.npmrc COPY .npmrc.docker ./.npmrc
COPY .docker-deps/ /app/.docker-deps/ COPY .docker-deps/ ../.docker-deps/
COPY web/package.json ./package.json COPY web/package.json ./package.json
RUN --mount=type=secret,id=gitea_npm_token \ RUN pnpm install --ignore-scripts --lockfile=false
export GITEA_NPM_TOKEN="$(cat /run/secrets/gitea_npm_token 2>/dev/null || echo '')" && \
pnpm install --ignore-scripts --lockfile=false
COPY web/next.config.ts ./next.config.ts COPY web/next.config.ts ./next.config.ts
COPY web/tsconfig.json ./tsconfig.json COPY web/tsconfig.json ./tsconfig.json