Init
This commit is contained in:
10
.env.example
10
.env.example
@@ -1,9 +1,3 @@
|
|||||||
# URL бэкенда (без завершающего слэша)
|
|
||||||
# В Dokploy задаётся через UI → Environment
|
|
||||||
BACKEND_URL=http://26.22.232.18:24454
|
|
||||||
|
|
||||||
# Порт, по которому фронтенд доступен на хосте
|
|
||||||
FRONTEND_PORT=3000
|
|
||||||
|
|
||||||
# Dev-only: используется при локальном `npm start`
|
# 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
1
.idea/.name
generated
Normal file
@@ -0,0 +1 @@
|
|||||||
|
index.js
|
||||||
38
Dockerfile
38
Dockerfile
@@ -5,39 +5,41 @@ FROM node:20-alpine AS builder
|
|||||||
|
|
||||||
WORKDIR /app
|
WORKDIR /app
|
||||||
|
|
||||||
# Устанавливаем зависимости (кэшируется слой)
|
# Зависимости (кэшируется)
|
||||||
COPY package.json package-lock.json ./
|
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 public ./public
|
||||||
COPY src ./src
|
COPY src ./src
|
||||||
|
|
||||||
ENV GENERATE_SOURCEMAP=false \
|
ENV GENERATE_SOURCEMAP=false \
|
||||||
CI=true \
|
DISABLE_ESLINT_PLUGIN=true \
|
||||||
NODE_OPTIONS=--max_old_space_size=4096
|
NODE_OPTIONS=--max_old_space_size=4096
|
||||||
|
|
||||||
RUN npm run build
|
RUN npm run build
|
||||||
|
|
||||||
|
|
||||||
# ─── Stage 2: runtime (nginx) ────────────────────────────────────
|
# ─── Stage 2: runtime — лёгкий статический сервер ────────────────
|
||||||
FROM nginx:1.27-alpine AS runtime
|
FROM node:20-alpine AS runtime
|
||||||
|
|
||||||
# nginx сам запустит envsubst для шаблона перед стартом
|
WORKDIR /app
|
||||||
ENV NGINX_ENVSUBST_TEMPLATE_SUFFIX=.template \
|
|
||||||
NGINX_ENVSUBST_OUTPUT_DIR=/etc/nginx/conf.d \
|
|
||||||
BACKEND_URL=http://backend:8000
|
|
||||||
|
|
||||||
# Убираем дефолтный конфиг, кладём наш шаблон
|
# `serve` — минималистичный SPA-сервер (~2 МБ), флаг -s = SPA fallback на index.html
|
||||||
RUN rm /etc/nginx/conf.d/default.conf
|
RUN npm install -g serve@14.2.4 && \
|
||||||
COPY docker/nginx/default.conf.template /etc/nginx/templates/default.conf.template
|
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 \
|
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"]
|
||||||
|
|||||||
@@ -6,13 +6,10 @@ services:
|
|||||||
image: fool-stack-frontend:latest
|
image: fool-stack-frontend:latest
|
||||||
container_name: fool-stack-frontend
|
container_name: fool-stack-frontend
|
||||||
restart: unless-stopped
|
restart: unless-stopped
|
||||||
environment:
|
|
||||||
# URL бэкенда, к которому nginx проксирует /api и /static/avatars
|
|
||||||
BACKEND_URL: ${BACKEND_URL:-https://back.fool-stack.ru/}
|
|
||||||
expose:
|
expose:
|
||||||
- "80"
|
- "24452"
|
||||||
healthcheck:
|
healthcheck:
|
||||||
test: ["CMD", "wget", "-qO-", "http://localhost/"]
|
test: ["CMD", "wget", "-qO-", "http://localhost:24452/"]
|
||||||
interval: 30s
|
interval: 30s
|
||||||
timeout: 5s
|
timeout: 5s
|
||||||
retries: 3
|
retries: 3
|
||||||
|
|||||||
@@ -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;
|
|
||||||
}
|
|
||||||
@@ -83,7 +83,6 @@ const KBBoard = () => {
|
|||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
const [socket, setSocket] = useState(null);
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
let ws;
|
let ws;
|
||||||
const connect = async () => {
|
const connect = async () => {
|
||||||
@@ -91,16 +90,9 @@ const KBBoard = () => {
|
|||||||
const res = await getWsTicketAPI();
|
const res = await getWsTicketAPI();
|
||||||
const token = res.data.token;
|
const token = res.data.token;
|
||||||
const protocol = window.location.protocol === 'https:' ? 'wss:' : 'ws:';
|
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}//${window.location.host}/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 соединение установлено');
|
||||||
ws.onopen = () => {
|
ws.onmessage = () => loadBoardData(true);
|
||||||
console.log('WebSocket соединение установлено');
|
|
||||||
setSocket(ws);
|
|
||||||
};
|
|
||||||
ws.onmessage = (event) => {
|
|
||||||
const message = JSON.parse(event.data);
|
|
||||||
loadBoardData(true);
|
|
||||||
};
|
|
||||||
ws.onclose = () => console.log('WebSocket соединение закрыто');
|
ws.onclose = () => console.log('WebSocket соединение закрыто');
|
||||||
ws.onerror = (error) => console.error('Ошибка WebSocket:', error);
|
ws.onerror = (error) => console.error('Ошибка WebSocket:', error);
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
|
|||||||
Reference in New Issue
Block a user