完善docker
This commit is contained in:
parent
3b65143565
commit
e91a7ec4c4
22
Dockerfile
Normal file
22
Dockerfile
Normal file
@ -0,0 +1,22 @@
|
||||
FROM node:24-bookworm-slim
|
||||
|
||||
WORKDIR /app
|
||||
|
||||
RUN npm config set registry https://registry.npmmirror.com/ && \
|
||||
yarn config set registry https://registry.npmmirror.com/
|
||||
|
||||
# Copy the repository contents into the image and install all dependencies
|
||||
COPY . .
|
||||
|
||||
# The container only runs the backend dev server, so strip Electron-only
|
||||
# packages before installing to avoid downloading desktop binaries.
|
||||
RUN node -e "const fs=require('fs');const pkg=JSON.parse(fs.readFileSync('package.json','utf8'));for(const section of ['dependencies','devDependencies']){if(!pkg[section]) continue;for(const name of ['custom-electron-titlebar','electron','electron-builder','electron-rebuild','electronmon']) delete pkg[section][name];}fs.writeFileSync('package.json', JSON.stringify(pkg, null, 2)+'\n');" && \
|
||||
yarn install --frozen-lockfile && \
|
||||
yarn cache clean
|
||||
|
||||
ENV NODE_ENV=dev
|
||||
ENV PORT=10588
|
||||
|
||||
EXPOSE 10588
|
||||
|
||||
CMD ["yarn", "dev"]
|
||||
2244
data/web/index.html
2244
data/web/index.html
File diff suppressed because one or more lines are too long
@ -1,121 +0,0 @@
|
||||
# 构建阶段
|
||||
FROM node:24-alpine AS builder
|
||||
|
||||
WORKDIR /app
|
||||
|
||||
# 定义构建参数
|
||||
ARG GIT=github
|
||||
ARG TAG=""
|
||||
ARG BRANCH=""
|
||||
|
||||
# 安装 git
|
||||
RUN apk add --no-cache git
|
||||
|
||||
RUN npm config set registry https://registry.npmmirror.com/ && \
|
||||
yarn config set registry https://registry.npmmirror.com/
|
||||
|
||||
# 根据参数选择仓库源,支持 TAG / BRANCH 切换
|
||||
# 优先级: TAG > BRANCH > 最新 tag > 默认分支
|
||||
RUN if [ "$GIT" = "gitee" ]; then \
|
||||
REPO_URL="https://gitee.com/HBAI-Ltd/Toonflow-app.git"; \
|
||||
else \
|
||||
REPO_URL="https://github.com/HBAI-Ltd/Toonflow-app.git"; \
|
||||
fi && \
|
||||
echo "Cloning from: $REPO_URL" && \
|
||||
git clone "$REPO_URL" . && \
|
||||
if [ -n "$TAG" ]; then \
|
||||
echo "Checking out specified tag: $TAG" && \
|
||||
git checkout "$TAG"; \
|
||||
elif [ -n "$BRANCH" ]; then \
|
||||
echo "Checking out specified branch: $BRANCH" && \
|
||||
git checkout "$BRANCH"; \
|
||||
else \
|
||||
LATEST_TAG=$(git describe --tags --abbrev=0 2>/dev/null || git tag --sort=-v:refname | head -n 1) && \
|
||||
if [ -n "$LATEST_TAG" ]; then \
|
||||
echo "Checking out latest tag: $LATEST_TAG" && \
|
||||
git checkout "$LATEST_TAG"; \
|
||||
else \
|
||||
echo "No tags found, using default branch"; \
|
||||
fi; \
|
||||
fi && \
|
||||
echo "Current version:" && git describe --tags --always
|
||||
|
||||
RUN yarn install --frozen-lockfile
|
||||
|
||||
RUN yarn build
|
||||
|
||||
# 生产阶段
|
||||
FROM node:24-alpine
|
||||
|
||||
WORKDIR /app
|
||||
|
||||
# 安装 nginx 和 supervisor
|
||||
RUN apk add --no-cache nginx supervisor && \
|
||||
mkdir -p /var/lib/nginx/logs /var/log/nginx && \
|
||||
npm config set registry https://registry.npmmirror.com/ && \
|
||||
yarn config set registry https://registry.npmmirror.com/ && \
|
||||
npm install -g pm2
|
||||
|
||||
# 复制后端文件
|
||||
COPY --from=builder /app/build ./build
|
||||
COPY --from=builder /app/data/serve ./data/serve
|
||||
COPY --from=builder /app/package.json ./
|
||||
COPY --from=builder /app/yarn.lock ./
|
||||
|
||||
# 复制静态页面到 nginx 目录
|
||||
COPY --from=builder /app/scripts/web /usr/share/nginx/html
|
||||
|
||||
# 只安装生产依赖
|
||||
RUN yarn install --frozen-lockfile --production
|
||||
|
||||
# 配置 nginx
|
||||
RUN cat > /etc/nginx/http.d/default.conf << 'EOF'
|
||||
server {
|
||||
listen 80;
|
||||
location / {
|
||||
root /usr/share/nginx/html;
|
||||
index index.html;
|
||||
try_files $uri $uri/ /index.html;
|
||||
}
|
||||
}
|
||||
EOF
|
||||
|
||||
# 配置 nginx 主配置,日志输出到 stderr/stdout
|
||||
RUN sed -i 's|error_log /var/log/nginx/error.log warn;|error_log /dev/stderr warn;|g' /etc/nginx/nginx.conf || true && \
|
||||
sed -i 's|access_log /var/log/nginx/access.log main;|access_log /dev/stdout main;|g' /etc/nginx/nginx.conf || true
|
||||
|
||||
# 配置 supervisor
|
||||
RUN cat > /etc/supervisord.conf << 'EOF'
|
||||
[supervisord]
|
||||
nodaemon=true
|
||||
logfile=/var/log/supervisord.log
|
||||
pidfile=/var/run/supervisord.pid
|
||||
|
||||
[program:nginx]
|
||||
command=nginx -g "daemon off;"
|
||||
autostart=true
|
||||
autorestart=true
|
||||
stdout_logfile=/dev/stdout
|
||||
stdout_logfile_maxbytes=0
|
||||
stderr_logfile=/dev/stderr
|
||||
stderr_logfile_maxbytes=0
|
||||
|
||||
[program:app]
|
||||
command=pm2-runtime start data/serve/app.js --name app
|
||||
directory=/app
|
||||
autostart=true
|
||||
autorestart=true
|
||||
stdout_logfile=/dev/stdout
|
||||
stdout_logfile_maxbytes=0
|
||||
stderr_logfile=/dev/stderr
|
||||
stderr_logfile_maxbytes=0
|
||||
environment=NODE_ENV=prod
|
||||
EOF
|
||||
|
||||
ENV NODE_ENV=prod
|
||||
|
||||
EXPOSE 80
|
||||
EXPOSE 10588
|
||||
|
||||
# 启动时创建必要目录(防止 volume 挂载覆盖)
|
||||
CMD sh -c "mkdir -p /var/log/nginx /var/lib/nginx/logs && exec supervisord -c /etc/supervisord.conf"
|
||||
@ -1,95 +0,0 @@
|
||||
# 本地构建阶段 - 使用本地源码,不从 git 克隆
|
||||
FROM node:24-alpine AS builder
|
||||
|
||||
WORKDIR /app
|
||||
|
||||
RUN npm config set registry https://registry.npmmirror.com/ && \
|
||||
yarn config set registry https://registry.npmmirror.com/
|
||||
|
||||
# 复制依赖文件
|
||||
COPY package.json yarn.lock ./
|
||||
|
||||
RUN yarn install --frozen-lockfile
|
||||
|
||||
# 复制源码
|
||||
COPY tsconfig.json ./
|
||||
COPY src/ ./src/
|
||||
COPY scripts/ ./scripts/
|
||||
|
||||
RUN yarn build
|
||||
|
||||
# 生产阶段
|
||||
FROM node:24-alpine
|
||||
|
||||
WORKDIR /app
|
||||
|
||||
# 安装 nginx 和 supervisor
|
||||
RUN apk add --no-cache nginx supervisor && \
|
||||
mkdir -p /var/lib/nginx/logs /var/log/nginx && \
|
||||
npm config set registry https://registry.npmmirror.com/ && \
|
||||
yarn config set registry https://registry.npmmirror.com/ && \
|
||||
npm install -g pm2
|
||||
|
||||
# 复制后端文件
|
||||
COPY --from=builder /app/build ./build
|
||||
COPY --from=builder /app/data/serve ./data/serve
|
||||
COPY --from=builder /app/package.json ./
|
||||
COPY --from=builder /app/yarn.lock ./
|
||||
|
||||
# 复制静态页面到 nginx 目录
|
||||
COPY --from=builder /app/scripts/web /usr/share/nginx/html
|
||||
|
||||
# 只安装生产依赖
|
||||
RUN yarn install --frozen-lockfile --production
|
||||
|
||||
# 配置 nginx
|
||||
RUN cat > /etc/nginx/http.d/default.conf << 'EOF'
|
||||
server {
|
||||
listen 80;
|
||||
location / {
|
||||
root /usr/share/nginx/html;
|
||||
index index.html;
|
||||
try_files $uri $uri/ /index.html;
|
||||
}
|
||||
}
|
||||
EOF
|
||||
|
||||
# 配置 nginx 主配置,日志输出到 stderr/stdout
|
||||
RUN sed -i 's|error_log /var/log/nginx/error.log warn;|error_log /dev/stderr warn;|g' /etc/nginx/nginx.conf || true && \
|
||||
sed -i 's|access_log /var/log/nginx/access.log main;|access_log /dev/stdout main;|g' /etc/nginx/nginx.conf || true
|
||||
|
||||
# 配置 supervisor
|
||||
RUN cat > /etc/supervisord.conf << 'EOF'
|
||||
[supervisord]
|
||||
nodaemon=true
|
||||
logfile=/var/log/supervisord.log
|
||||
pidfile=/var/run/supervisord.pid
|
||||
|
||||
[program:nginx]
|
||||
command=nginx -g "daemon off;"
|
||||
autostart=true
|
||||
autorestart=true
|
||||
stdout_logfile=/dev/stdout
|
||||
stdout_logfile_maxbytes=0
|
||||
stderr_logfile=/dev/stderr
|
||||
stderr_logfile_maxbytes=0
|
||||
|
||||
[program:app]
|
||||
command=pm2-runtime start data/serve/app.js --name app
|
||||
directory=/app
|
||||
autostart=true
|
||||
autorestart=true
|
||||
stdout_logfile=/dev/stdout
|
||||
stdout_logfile_maxbytes=0
|
||||
stderr_logfile=/dev/stderr
|
||||
stderr_logfile_maxbytes=0
|
||||
environment=NODE_ENV=prod
|
||||
EOF
|
||||
|
||||
ENV NODE_ENV=prod
|
||||
|
||||
EXPOSE 80
|
||||
EXPOSE 10588
|
||||
|
||||
# 启动时创建必要目录(防止 volume 挂载覆盖)
|
||||
CMD sh -c "mkdir -p /var/log/nginx /var/lib/nginx/logs && exec supervisord -c /etc/supervisord.conf"
|
||||
@ -1,24 +0,0 @@
|
||||
# 本地打包测试用,使用本地源码构建
|
||||
# 用法: docker-compose -f docker/docker-compose.local.yml up -d --build
|
||||
|
||||
services:
|
||||
toonflow:
|
||||
build:
|
||||
context: ..
|
||||
dockerfile: docker/Dockerfile.local
|
||||
image: toonflow:local
|
||||
container_name: toonflow-local
|
||||
restart: unless-stopped
|
||||
ports:
|
||||
- "8080:80"
|
||||
- "10588:10588"
|
||||
environment:
|
||||
- NODE_ENV=prod
|
||||
volumes:
|
||||
- ../logs:/var/log
|
||||
healthcheck:
|
||||
test: ["CMD", "wget", "--quiet", "--tries=1", "--spider", "http://localhost:80/"]
|
||||
interval: 30s
|
||||
timeout: 10s
|
||||
retries: 3
|
||||
start_period: 40s
|
||||
@ -1,26 +0,0 @@
|
||||
services:
|
||||
toonflow:
|
||||
build:
|
||||
context: ..
|
||||
dockerfile: docker/Dockerfile
|
||||
args:
|
||||
GIT: ${GIT:-github}
|
||||
TAG: ${TAG:-}
|
||||
BRANCH: ${BRANCH:-}
|
||||
image: toonflow:${TAG:-latest}
|
||||
container_name: toonflow
|
||||
restart: unless-stopped
|
||||
ports:
|
||||
- "80"
|
||||
- "10588:10588"
|
||||
environment:
|
||||
- NODE_ENV=prod
|
||||
volumes:
|
||||
# 可选: 持久化日志
|
||||
- ../logs:/var/log
|
||||
healthcheck:
|
||||
test: ["CMD", "wget", "--quiet", "--tries=1", "--spider", "http://localhost:80/"]
|
||||
interval: 30s
|
||||
timeout: 10s
|
||||
retries: 3
|
||||
start_period: 40s
|
||||
@ -21,6 +21,7 @@
|
||||
"dev": "nodemon --inspect --exec tsx src/app.ts",
|
||||
"dev:gui": "electronmon -r tsx scripts/main.ts",
|
||||
"dev:gui-vite": "cross-env VITE_DEV=1 electronmon -r tsx scripts/main.ts",
|
||||
"start": "cross-env NODE_ENV=prod node data/serve/app.js",
|
||||
"lint": "tsc --noEmit",
|
||||
"build": "cross-env NODE_ENV=prod tsx scripts/build.ts",
|
||||
"pack": "electron-builder --dir",
|
||||
@ -28,7 +29,6 @@
|
||||
"dist:win": "yarn build && electron-builder --win",
|
||||
"dist:mac": "yarn build && electron-builder --mac",
|
||||
"dist:linux": "yarn build && electron-builder --linux",
|
||||
"test": "cross-env NODE_ENV=prod node data/serve/app.js",
|
||||
"docker:build": "docker-compose -f docker/docker-compose.yml up -d --build",
|
||||
"docker:local": "docker-compose -f docker/docker-compose.local.yml up -d --build",
|
||||
"debug:ai": "npx @ai-sdk/devtools",
|
||||
|
||||
@ -3,20 +3,62 @@ import path from "path";
|
||||
import fs from "fs";
|
||||
import Module from "module";
|
||||
|
||||
declare const __APP_VERSION__: string | undefined;
|
||||
|
||||
/**
|
||||
* 将 extraResources 中的 data 目录复制到用户数据目录(跳过已存在的文件,保留用户修改)
|
||||
*/
|
||||
|
||||
function getVersionFromFile(filePath: string): string | null {
|
||||
try {
|
||||
if (fs.existsSync(filePath)) {
|
||||
return fs.readFileSync(filePath, "utf8").trim();
|
||||
}
|
||||
} catch {}
|
||||
return null;
|
||||
}
|
||||
|
||||
function writeVersionToFile(filePath: string, version: string): void {
|
||||
fs.writeFileSync(filePath, version, { encoding: "utf8" });
|
||||
}
|
||||
|
||||
function copyDirForce(src: string, dest: string): void {
|
||||
if (!fs.existsSync(src)) return;
|
||||
if (fs.existsSync(dest)) {
|
||||
fs.rmSync(dest, { recursive: true, force: true });
|
||||
}
|
||||
copyDirRecursive(src, dest);
|
||||
}
|
||||
|
||||
function initializeData(): void {
|
||||
const srcDir = path.join(process.resourcesPath, "data");
|
||||
const destDir = path.join(app.getPath("userData"), "data");
|
||||
if (fs.existsSync(destDir)) return;
|
||||
copyDirRecursive(srcDir, destDir);
|
||||
const versionFile = path.join(destDir, "version.txt");
|
||||
const currentVersion = typeof __APP_VERSION__ !== "undefined" ? __APP_VERSION__ : "0.0.0";
|
||||
const userVersion = getVersionFromFile(versionFile);
|
||||
|
||||
// 首次安装或无version.txt,直接全量拷贝
|
||||
if (!fs.existsSync(destDir) || !userVersion) {
|
||||
copyDirRecursive(srcDir, destDir);
|
||||
writeVersionToFile(versionFile, currentVersion);
|
||||
return;
|
||||
}
|
||||
|
||||
// 版本号不同则覆盖 serve 和 web 目录
|
||||
if (userVersion !== currentVersion) {
|
||||
copyDirForce(path.join(srcDir, "serve"), path.join(destDir, "serve"));
|
||||
copyDirForce(path.join(srcDir, "web"), path.join(destDir, "web"));
|
||||
writeVersionToFile(versionFile, currentVersion);
|
||||
}
|
||||
}
|
||||
|
||||
function copyDirRecursive(src: string, dest: string): void {
|
||||
if (!fs.existsSync(src)) return;
|
||||
if (!fs.existsSync(dest)) fs.mkdirSync(dest, { recursive: true });
|
||||
for (const entry of fs.readdirSync(src, { withFileTypes: true })) {
|
||||
// 跳过 oss 文件夹和 db2.sqlite 文件
|
||||
if (entry.isDirectory() && entry.name === "oss") continue;
|
||||
if (!entry.isDirectory() && entry.name === "db2.sqlite") continue;
|
||||
const srcPath = path.join(src, entry.name);
|
||||
const destPath = path.join(dest, entry.name);
|
||||
if (entry.isDirectory()) {
|
||||
@ -109,7 +151,9 @@ function showLoading(): void {
|
||||
});
|
||||
loadingWindow.setMenuBarVisibility(false);
|
||||
loadingWindow.removeMenu();
|
||||
loadingWindow.on("closed", () => { loadingWindow = null; });
|
||||
loadingWindow.on("closed", () => {
|
||||
loadingWindow = null;
|
||||
});
|
||||
void loadingWindow.loadURL(loadingHtml);
|
||||
}
|
||||
|
||||
@ -228,7 +272,7 @@ app.whenReady().then(async () => {
|
||||
} else {
|
||||
return { ok: false, error: "缺少url参数" };
|
||||
}
|
||||
}
|
||||
},
|
||||
};
|
||||
const handler = handlers[pathname];
|
||||
const responseData = handler ? handler() : { error: "未知接口" };
|
||||
|
||||
14
src/app.ts
14
src/app.ts
@ -29,16 +29,24 @@ export default async function startServe(randomPort: Boolean = false) {
|
||||
app.use(express.json({ limit: "100mb" }));
|
||||
app.use(express.urlencoded({ extended: true, limit: "100mb" }));
|
||||
|
||||
const rootDir = u.getPath("oss");
|
||||
|
||||
// 确保 uploads 目录存在
|
||||
// oss 静态资源
|
||||
const rootDir = u.getPath("oss");
|
||||
if (!fs.existsSync(rootDir)) {
|
||||
fs.mkdirSync(rootDir, { recursive: true });
|
||||
}
|
||||
console.log("文件目录:", rootDir);
|
||||
|
||||
app.use(express.static(rootDir));
|
||||
|
||||
// data/web 静态网站
|
||||
const webDir = u.getPath("web");
|
||||
if (fs.existsSync(webDir)) {
|
||||
console.log("静态网站目录:", webDir);
|
||||
app.use(express.static(webDir));
|
||||
} else {
|
||||
console.warn("静态网站目录不存在:", webDir);
|
||||
}
|
||||
|
||||
app.use(async (req, res, next) => {
|
||||
const setting = await u.db("o_setting").where("key", "tokenKey").select("value").first();
|
||||
if (!setting) return res.status(444).send({ message: "服务器秘钥未配置,请联系管理员" });
|
||||
|
||||
4
src/types/database.d.ts
vendored
4
src/types/database.d.ts
vendored
@ -1,4 +1,4 @@
|
||||
// @db-hash 6be0a80e9c8f541987a4c1e907736237
|
||||
// @db-hash 93b2462070c45c2b449e9a18c4e88763
|
||||
//该文件由脚本自动生成,请勿手动修改
|
||||
|
||||
export interface memories {
|
||||
@ -178,7 +178,6 @@ export interface o_storyboard {
|
||||
'sound'?: string | null;
|
||||
'state'?: string | null;
|
||||
'title'?: string | null;
|
||||
'videoPrompt'?: string | null;
|
||||
}
|
||||
export interface o_tasks {
|
||||
'describe'?: string | null;
|
||||
@ -212,7 +211,6 @@ export interface o_video {
|
||||
'errorReason'?: string | null;
|
||||
'filePath'?: string | null;
|
||||
'id'?: number;
|
||||
'projectId'?: number | null;
|
||||
'scriptId'?: number | null;
|
||||
'state'?: string | null;
|
||||
'storyboardId'?: number | null;
|
||||
|
||||
Loading…
x
Reference in New Issue
Block a user