This commit is contained in:
Vladiysss
2026-04-18 15:44:05 +03:00
parent 31c2764936
commit d7e8394898
6 changed files with 28 additions and 125 deletions

View File

@@ -1,9 +1,3 @@
# URL бэкенда (без завершающего слэша)
# В Dokploy задаётся через UI → Environment
BACKEND_URL=http://26.22.232.18:24454
# Порт, по которому фронтенд доступен на хосте
FRONTEND_PORT=3000
# Dev-only: используется при локальном `npm start`
REACT_APP_BACKEND_URL=http://26.22.232.18:24454
# В проде /api маршрутизируется через Traefik в Dokploy
REACT_APP_BACKEND_URL=https://back.fool-stack.ru

1
.idea/.name generated Normal file
View File

@@ -0,0 +1 @@
index.js

View File

@@ -5,39 +5,41 @@ FROM node:20-alpine AS builder
WORKDIR /app
# Устанавливаем зависимости (кэшируется слой)
# Зависимости (кэшируется)
COPY package.json package-lock.json ./
RUN npm ci --no-audit --no-fund
RUN npm install --no-audit --no-fund --legacy-peer-deps
# Копируем исходники
# Исходники
COPY public ./public
COPY src ./src
ENV GENERATE_SOURCEMAP=false \
CI=true \
DISABLE_ESLINT_PLUGIN=true \
NODE_OPTIONS=--max_old_space_size=4096
RUN npm run build
# ─── Stage 2: runtime (nginx) ────────────────────────────────────
FROM nginx:1.27-alpine AS runtime
# ─── Stage 2: runtime — лёгкий статический сервер ────────────────
FROM node:20-alpine AS runtime
# nginx сам запустит envsubst для шаблона перед стартом
ENV NGINX_ENVSUBST_TEMPLATE_SUFFIX=.template \
NGINX_ENVSUBST_OUTPUT_DIR=/etc/nginx/conf.d \
BACKEND_URL=http://backend:8000
WORKDIR /app
# Убираем дефолтный конфиг, кладём наш шаблон
RUN rm /etc/nginx/conf.d/default.conf
COPY docker/nginx/default.conf.template /etc/nginx/templates/default.conf.template
# `serve` — минималистичный SPA-сервер (~2 МБ), флаг -s = SPA fallback на index.html
RUN npm install -g serve@14.2.4 && \
addgroup -S app && adduser -S app -G app
# Билд статики
COPY --from=builder /app/build /usr/share/nginx/html
# Только готовый билд
COPY --from=builder --chown=app:app /app/build ./build
EXPOSE 80
USER app
ENV NODE_ENV=production \
PORT=24452
EXPOSE 24452
HEALTHCHECK --interval=30s --timeout=5s --start-period=10s --retries=3 \
CMD wget -qO- http://localhost/ > /dev/null 2>&1 || exit 1
CMD wget -qO- http://localhost:24452/ > /dev/null 2>&1 || exit 1
# Точка входа от nginx-image уже знает про templates
CMD ["serve", "-s", "build", "-l", "24452", "--no-clipboard"]

View File

@@ -6,13 +6,10 @@ services:
image: fool-stack-frontend:latest
container_name: fool-stack-frontend
restart: unless-stopped
environment:
# URL бэкенда, к которому nginx проксирует /api и /static/avatars
BACKEND_URL: ${BACKEND_URL:-https://back.fool-stack.ru/}
expose:
- "80"
- "24452"
healthcheck:
test: ["CMD", "wget", "-qO-", "http://localhost/"]
test: ["CMD", "wget", "-qO-", "http://localhost:24452/"]
interval: 30s
timeout: 5s
retries: 3

View File

@@ -1,83 +0,0 @@
map $http_upgrade $connection_upgrade {
default upgrade;
'' close;
}
server {
listen 80;
server_name _;
root /usr/share/nginx/html;
index index.html;
# Security headers
add_header X-Frame-Options "SAMEORIGIN" always;
add_header X-Content-Type-Options "nosniff" always;
add_header Referrer-Policy "no-referrer-when-downgrade" always;
add_header X-XSS-Protection "1; mode=block" always;
# Gzip
gzip on;
gzip_vary on;
gzip_min_length 1024;
gzip_comp_level 6;
gzip_types
text/plain
text/css
text/javascript
application/javascript
application/json
application/xml
image/svg+xml
font/ttf
font/otf
font/woff
font/woff2;
# SPA fallback
location / {
try_files $uri $uri/ /index.html;
}
# Hashed static assets — агрессивный кэш
location /static/ {
expires 1y;
access_log off;
add_header Cache-Control "public, immutable";
try_files $uri =404;
}
# HTML не кэшируется (чтобы обновления прилетали сразу)
location = /index.html {
add_header Cache-Control "no-store, no-cache, must-revalidate" always;
expires -1;
}
# API + WebSocket → backend
location /api/ {
proxy_pass ${BACKEND_URL};
proxy_http_version 1.1;
proxy_set_header Host $host;
proxy_set_header X-Real-IP $remote_addr;
proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
proxy_set_header X-Forwarded-Proto $scheme;
proxy_set_header Upgrade $http_upgrade;
proxy_set_header Connection $connection_upgrade;
proxy_connect_timeout 10s;
proxy_read_timeout 300s;
proxy_send_timeout 300s;
}
# Загруженные аватары → backend
location /static/avatars/ {
proxy_pass ${BACKEND_URL};
proxy_set_header Host $host;
proxy_set_header X-Real-IP $remote_addr;
proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
}
error_page 404 /index.html;
}

View File

@@ -83,7 +83,6 @@ const KBBoard = () => {
};
const [socket, setSocket] = useState(null);
useEffect(() => {
let ws;
const connect = async () => {
@@ -91,16 +90,9 @@ const KBBoard = () => {
const res = await getWsTicketAPI();
const token = res.data.token;
const protocol = window.location.protocol === 'https:' ? 'wss:' : 'ws:';
ws = new WebSocket(`${protocol}//26.22.232.18:24454/api/boards/ws/${id}?token=${token}`);
//ws = new WebSocket(`${protocol}//ws.back.fool-stack.ru/api/boards/ws/${id}?token=${token}`);
ws.onopen = () => {
console.log('WebSocket соединение установлено');
setSocket(ws);
};
ws.onmessage = (event) => {
const message = JSON.parse(event.data);
loadBoardData(true);
};
ws = new WebSocket(`${protocol}//${window.location.host}/api/boards/ws/${id}?token=${token}`);
ws.onopen = () => console.log('WebSocket соединение установлено');
ws.onmessage = () => loadBoardData(true);
ws.onclose = () => console.log('WebSocket соединение закрыто');
ws.onerror = (error) => console.error('Ошибка WebSocket:', error);
} catch (e) {