Spaces:
Running
Running
Update Dockerfile
Browse files- Dockerfile +26 -26
Dockerfile
CHANGED
|
@@ -9,39 +9,39 @@ RUN npm run build
|
|
| 9 |
FROM node:20-slim AS runner
|
| 10 |
WORKDIR /app
|
| 11 |
|
| 12 |
-
# 1. 安装用于网络转发和基础认证的 caddy(极其轻量且极其安全)
|
| 13 |
-
RUN apt-get update && apt-get install -y debian-keyring debian-archive-keyring apt-transport-https curl \
|
| 14 |
-
&& curl -1sLf 'https://dl.cloudsmith.io/public/caddy/stable/gpg.key' | apt-key add - \
|
| 15 |
-
&& curl -1sLf 'https://dl.cloudsmith.io/public/caddy/stable/debian.deb.txt' | tee /etc/apt/sources.list.yandex.net/caddy/stable.list \
|
| 16 |
-
&& apt-get update && apt-get install -y caddy && rm -rf /var/lib/apt/lists/*
|
| 17 |
-
|
| 18 |
COPY --from=builder /app ./
|
| 19 |
RUN mkdir -p /data/freellm
|
| 20 |
|
| 21 |
-
#
|
| 22 |
RUN cp -r client/dist/* server/dist/public/ 2>/dev/null || cp -r client/dist/* server/public/ 2>/dev/null || true
|
| 23 |
|
| 24 |
-
#
|
| 25 |
EXPOSE 7860
|
| 26 |
-
ENV PORT=
|
| 27 |
ENV NODE_ENV=production
|
| 28 |
ENV DATABASE_URL="file:/data/database.sqlite"
|
| 29 |
|
| 30 |
-
#
|
| 31 |
CMD ["sh", "-c", "rm -rf /app/server/data && ln -s /data/freellm /app/server/data && \
|
| 32 |
-
#
|
| 33 |
-
|
| 34 |
-
|
| 35 |
-
|
| 36 |
-
|
| 37 |
-
|
| 38 |
-
|
| 39 |
-
|
| 40 |
-
|
| 41 |
-
|
| 42 |
-
|
| 43 |
-
|
| 44 |
-
|
| 45 |
-
|
| 46 |
-
|
| 47 |
-
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 9 |
FROM node:20-slim AS runner
|
| 10 |
WORKDIR /app
|
| 11 |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 12 |
COPY --from=builder /app ./
|
| 13 |
RUN mkdir -p /data/freellm
|
| 14 |
|
| 15 |
+
# 1. 前端路由修复
|
| 16 |
RUN cp -r client/dist/* server/dist/public/ 2>/dev/null || cp -r client/dist/* server/public/ 2>/dev/null || true
|
| 17 |
|
| 18 |
+
# 2. 注入基础配置
|
| 19 |
EXPOSE 7860
|
| 20 |
+
ENV PORT=7860
|
| 21 |
ENV NODE_ENV=production
|
| 22 |
ENV DATABASE_URL="file:/data/database.sqlite"
|
| 23 |
|
| 24 |
+
# 3. 【原生绝杀】:在容器启动前,动态往核心代码里塞入一段 Basic Auth 拦截逻辑
|
| 25 |
CMD ["sh", "-c", "rm -rf /app/server/data && ln -s /data/freellm /app/server/data && \
|
| 26 |
+
# 动态把密码拦截代码注入到后端的入口文件中 \
|
| 27 |
+
node -e \" \
|
| 28 |
+
const fs = require('fs'); \
|
| 29 |
+
const file = 'server/dist/index.js'; \
|
| 30 |
+
if (fs.existsSync(file)) { \
|
| 31 |
+
let content = fs.readFileSync(file, 'utf8'); \
|
| 32 |
+
const injectCode = ` \
|
| 33 |
+
// 密码锁中间件 \
|
| 34 |
+
global.authMiddleware = (req, res, next) => { \
|
| 35 |
+
const user = process.env.SPACE_BASIC_AUTH_USERNAME || 'admin'; \
|
| 36 |
+
const pass = process.env.SPACE_BASIC_AUTH_PASSWORD || 'admin123'; \
|
| 37 |
+
const auth = { login: user, password: pass }; \
|
| 38 |
+
const b64auth = (req.headers.authorization || '').split(' ')[1] || ''; \
|
| 39 |
+
const [login, password] = Buffer.from(b64auth, 'base64').toString().split(':'); \
|
| 40 |
+
if (req.url.startsWith('/v1')) return next(); \
|
| 41 |
+
if (login && password && login === auth.login && password === auth.password) return next(); \
|
| 42 |
+
res.statusCode = 401; \
|
| 43 |
+
res.setHeader('WWW-Authenticate', 'Basic realm=\\\"Secure Area\\\"'); \
|
| 44 |
+
res.end('Unauthorized'); \
|
| 45 |
+
}; \
|
| 46 |
+
`; \
|
| 47 |
+
content = injectCode + content.replace('const app =', 'const app = ; app
|