feat: docker & build
This commit is contained in:
@@ -0,0 +1,10 @@
|
|||||||
|
.git
|
||||||
|
.gitignore
|
||||||
|
node_modules
|
||||||
|
dist
|
||||||
|
storybook-static
|
||||||
|
.env
|
||||||
|
*.md
|
||||||
|
tests/
|
||||||
|
.claude/
|
||||||
|
.agents/
|
||||||
@@ -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"]
|
||||||
@@ -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;"]
|
||||||
@@ -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;
|
||||||
|
}
|
||||||
|
}
|
||||||
Generated
+3
-2
@@ -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
@@ -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",
|
||||||
|
|||||||
@@ -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({
|
||||||
|
|||||||
Reference in New Issue
Block a user