feat: docker & build

This commit is contained in:
2026-06-06 13:00:27 +03:00
parent e8e3bbe75e
commit 37c1a5944a
7 changed files with 101 additions and 3 deletions
+10
View File
@@ -0,0 +1,10 @@
.git
.gitignore
node_modules
dist
storybook-static
.env
*.md
tests/
.claude/
.agents/
+23
View File
@@ -0,0 +1,23 @@
# syntax=docker/dockerfile:1
# DEV image: source bind-mounted by compose, rsbuild dev server with HMR.
# node_modules lives in a named volume (see compose) so the host dir never
# shadows the container install. Build context = mcma-webui/.
FROM node:22-slim
# `modern-sk` is a git dependency (git+https://...) — npm needs git to fetch it.
RUN apt-get update \
&& apt-get install -y --no-install-recommends git \
&& rm -rf /var/lib/apt/lists/*
WORKDIR /app
COPY package.json package-lock.json modern-sk-*.tgz ./
RUN npm ci
COPY . .
EXPOSE 3000
# host 0.0.0.0 + HMR client port come from env (RSBUILD_* in compose).
# `--open` is dropped on purpose: no browser in a container.
CMD ["npx", "rsbuild", "dev"]
+29
View File
@@ -0,0 +1,29 @@
# syntax=docker/dockerfile:1
# PROD image: build static SPA, serve with nginx. Self-contained.
# Build context = mcma-webui/. The future prod compose puts a real proxy in
# front; this image just serves the built assets with SPA fallback.
# -- build stage ----------------------------------------------------------
FROM node:22-slim AS build
WORKDIR /app
COPY package.json package-lock.json modern-sk-*.tgz ./
RUN npm ci
COPY . .
# Bake the API base URL at build time (rsbuild inlines PUBLIC_* vars).
# Same-origin default ('/api/v1') works behind any reverse proxy.
ARG PUBLIC_API_BASE_URL=/api/v1
ENV PUBLIC_API_BASE_URL=$PUBLIC_API_BASE_URL
RUN npm run build
# -- runtime stage --------------------------------------------------------
FROM nginx:1.27-alpine AS runtime
COPY dockerfiles/nginx.conf /etc/nginx/conf.d/default.conf
COPY --from=build /app/dist /usr/share/nginx/html
EXPOSE 80
CMD ["nginx", "-g", "daemon off;"]
+21
View File
@@ -0,0 +1,21 @@
# Static SPA server baked into the webui PROD image (Dockerfile.prod).
# Only serves the built assets + SPA fallback. API routing belongs to the
# outer reverse proxy in the future prod compose.
server {
listen 80;
server_name _;
root /usr/share/nginx/html;
index index.html;
# Long-cache hashed build assets.
location /static/ {
expires 1y;
add_header Cache-Control "public, immutable";
try_files $uri =404;
}
# SPA: every unknown path falls back to index.html (client-side router).
location / {
try_files $uri $uri/ /index.html;
}
}
+3 -2
View File
@@ -10,7 +10,7 @@
"dependencies": { "dependencies": {
"@phosphor-icons/react": "^2.1.10", "@phosphor-icons/react": "^2.1.10",
"@reduxjs/toolkit": "^2.12.0", "@reduxjs/toolkit": "^2.12.0",
"modern-sk": "git+https://git.ollyhearn.ru/olly/modern-sk.git", "modern-sk": "file:./modern-sk-0.1.2.tgz",
"react": "^19.2.6", "react": "^19.2.6",
"react-dom": "^19.2.6", "react-dom": "^19.2.6",
"react-redux": "^9.3.0", "react-redux": "^9.3.0",
@@ -3879,7 +3879,8 @@
}, },
"node_modules/modern-sk": { "node_modules/modern-sk": {
"version": "0.1.2", "version": "0.1.2",
"resolved": "git+https://git.ollyhearn.ru/olly/modern-sk.git#37f0592464b5ba88373627d47ed534b7c6807d96", "resolved": "file:modern-sk-0.1.2.tgz",
"integrity": "sha512-tKSxbtUxT0CLkGc8DK+SABlVmKsMqqQr61uvAJ8EDcrutzm+VD230hTRVzk9hp2oSo6nXeeMig7KS8v0Lz5mWw==",
"license": "MIT", "license": "MIT",
"dependencies": { "dependencies": {
"@phosphor-icons/react": "^2.1.10", "@phosphor-icons/react": "^2.1.10",
+1 -1
View File
@@ -15,7 +15,7 @@
"dependencies": { "dependencies": {
"@phosphor-icons/react": "^2.1.10", "@phosphor-icons/react": "^2.1.10",
"@reduxjs/toolkit": "^2.12.0", "@reduxjs/toolkit": "^2.12.0",
"modern-sk": "git+https://git.ollyhearn.ru/olly/modern-sk.git", "modern-sk": "file:./modern-sk-0.1.2.tgz",
"react": "^19.2.6", "react": "^19.2.6",
"react-dom": "^19.2.6", "react-dom": "^19.2.6",
"react-redux": "^9.3.0", "react-redux": "^9.3.0",
+14
View File
@@ -2,7 +2,21 @@ import { defineConfig } from '@rsbuild/core';
import { pluginBabel } from '@rsbuild/plugin-babel'; import { pluginBabel } from '@rsbuild/plugin-babel';
import { pluginReact } from '@rsbuild/plugin-react'; import { pluginReact } from '@rsbuild/plugin-react';
// In docker dev the container binds 0.0.0.0:3000 and the browser reaches it
// through nginx on :80 — so HMR must be told the client-facing port. All three
// vars are unset for a plain `npm run dev`, where the defaults apply.
const { RSBUILD_HOST, RSBUILD_PORT, RSBUILD_HMR_CLIENT_PORT } = process.env;
export default defineConfig({ export default defineConfig({
server: {
host: RSBUILD_HOST ?? 'localhost',
port: RSBUILD_PORT ? Number(RSBUILD_PORT) : 3000,
},
dev: {
client: RSBUILD_HMR_CLIENT_PORT
? { port: Number(RSBUILD_HMR_CLIENT_PORT) }
: undefined,
},
plugins: [ plugins: [
pluginReact(), pluginReact(),
pluginBabel({ pluginBabel({