b3258e76df7911118d096bb12f031b2ac6055929
- 使用 React.lazy() + Suspense 实现路由级代码分割 - 配置 manualChunks 将 react/tdesign/recharts 拆分为独立 vendor chunk - 页面组件改为 export default 以支持动态导入 - 新增 bundle-optimization 规范,更新 frontend 导航规范
Nex - AI Gateway
一个统一的大模型 API 网关,支持 OpenAI 和 Anthropic 双协议,让应用只需配置一个地址即可透明使用多个供应商的大模型服务。
项目结构
nex/
├── backend/ # Go 后端服务(分层架构)
│ ├── cmd/
│ │ ├── server/ # CLI 主程序入口
│ │ └── desktop/ # 桌面应用入口
│ ├── internal/
│ │ ├── handler/ # HTTP 处理器 + 中间件
│ │ ├── service/ # 业务逻辑层
│ │ ├── repository/ # 数据访问层
│ │ ├── domain/ # 领域模型
│ │ ├── conversion/ # 协议转换引擎(OpenAI/Anthropic 适配器)
│ │ ├── provider/ # 供应商客户端
│ │ └── config/ # 配置管理
│ ├── pkg/ # 公共包(logger/errors/validator)
│ ├── migrations/ # 数据库迁移
│ └── tests/ # 测试(unit/integration)
│
├── frontend/ # React 前端界面
│ ├── src/
│ │ ├── api/ # API 层(统一请求封装 + 字段转换)
│ │ ├── hooks/ # TanStack Query hooks
│ │ ├── components/ # 通用组件(AppLayout)
│ │ ├── pages/ # 页面(Providers, Stats)
│ │ ├── routes/ # React Router 路由配置
│ │ ├── types/ # TypeScript 类型定义
│ │ └── __tests__/ # 单元测试 + 组件测试
│ ├── e2e/ # Playwright E2E 测试
│ └── package.json
│
├── assets/ # 应用资源
│ ├── icon.png # 托盘图标
│ ├── AppIcon.icns # macOS 应用图标
│ └── icon.ico # Windows 应用图标
│
├── scripts/ # 构建脚本
│ └── build/
│ └── package-macos.sh # macOS .app 打包脚本
│
└── README.md # 本文件
功能特性
- 双协议支持:同时支持 OpenAI 和 Anthropic 协议
- 跨协议转换:Hub-and-Spoke 架构实现 OpenAI ↔ Anthropic 双向转换
- 统一模型 ID:
provider_id/model_name格式全局唯一标识模型(如openai/gpt-4) - Smart Passthrough:同协议请求零序列化开销,仅改写 model 字段
- 流式响应:完整支持 SSE 流式传输,包括跨协议流式转换
- Function Calling:支持工具调用(Tools)
- Thinking / Reasoning:支持 OpenAI
reasoning_effort和 Anthropicthinking配置 - 扩展接口:支持 Embeddings 和 Rerank 接口
- 多供应商管理:配置和管理多个供应商(供应商 ID 仅限字母、数字、下划线)
- 用量统计:按供应商、模型、日期统计请求数量
- Web 配置界面:提供供应商和模型配置管理
技术栈
后端
- 语言: Go 1.26+
- HTTP 框架: Gin
- ORM: GORM
- 数据库: SQLite
- 日志: zap + lumberjack(结构化日志 + 日志轮转)
- 配置: Viper + pflag(多层配置:CLI > 环境变量 > 配置文件 > 默认值)
- 验证: go-playground/validator/v10
- 迁移: goose
前端
- 运行时: Bun
- 构建工具: Vite
- 语言: TypeScript (strict mode)
- 框架: React
- UI 组件库: TDesign React
- 图表库: Recharts
- 路由: React Router v7
- 数据获取: TanStack Query v5
- 样式: SCSS Modules
- 测试: Vitest + React Testing Library + Playwright
快速开始
桌面应用(推荐)
构建桌面应用:
# macOS (arm64 + amd64)
make desktop-mac
make package-macos # 打包为 .app
# Windows
make desktop-win
# Linux
make desktop-linux
使用桌面应用:
- 双击启动应用(macOS: Nex.app,Windows: nex-win-amd64.exe,Linux: nex-linux-amd64)
- 系统托盘图标出现,浏览器自动打开管理界面
- 点击托盘图标显示菜单,可打开管理界面或退出
- 关闭浏览器后服务继续运行,可通过托盘重新打开
注意事项:
- 桌面应用需要 CGO 支持
- macOS: 自带 Xcode Command Line Tools
- Linux: 自带 gcc,部分桌面环境需要
libappindicator3-dev - Windows: 需要 MinGW-w64 或在 Windows 环境构建
Linux 桌面环境兼容性:
- GNOME: 需要 AppIndicator 扩展
- KDE Plasma: 原生支持
- Xfce: 需要 libappindicator
- 其他支持 StatusNotifierItem 规范的环境
CLI 模式
后端
cd backend
go mod download
go run cmd/server/main.go
后端服务将在 http://localhost:9826 启动。首次启动会自动:
- 创建配置文件
~/.nex/config.yaml - 初始化数据库
~/.nex/config.db - 运行数据库迁移
- 创建日志目录
~/.nex/log/
前端
cd frontend
bun install
bun dev
前端开发服务器将在 http://localhost:5173 启动,API 请求通过 Vite proxy 转发到后端。
API 接口
代理接口(对外部应用)
代理接口统一使用 /{protocol}/*path 路由格式,模型 ID 使用 provider_id/model_name 格式(如 openai/gpt-4)。同协议请求走 Smart Passthrough,最小化 JSON 改写保持参数保真;跨协议请求走完整 decode/encode 转换。
OpenAI 协议(protocol=openai):
POST /openai/chat/completions- 对话补全GET /openai/models- 模型列表(本地数据库聚合)GET /openai/models/{provider_id}/{model_name}- 模型详情(本地数据库查询)POST /openai/embeddings- 嵌入POST /openai/rerank- 重排序
Anthropic 协议(protocol=anthropic):
POST /anthropic/v1/messages- 消息对话GET /anthropic/v1/models- 模型列表(本地数据库聚合)GET /anthropic/v1/models/{provider_id}/{model_name}- 模型详情(本地数据库查询)
管理接口(对前端)
供应商管理
GET /api/providers- 列出所有供应商POST /api/providers- 创建供应商(id仅限字母、数字、下划线,长度 1-64)GET /api/providers/:id- 获取供应商PUT /api/providers/:id- 更新供应商(id不可修改)DELETE /api/providers/:id- 删除供应商
模型管理
GET /api/models- 列出模型(支持?provider_id=xxx过滤)POST /api/models- 创建模型(id由系统自动生成 UUID,provider_id+model_name联合唯一)GET /api/models/:id- 获取模型(响应含unified_id字段,格式provider_id/model_name)PUT /api/models/:id- 更新模型(不可修改id)DELETE /api/models/:id- 删除模型
统计查询
GET /api/stats- 查询统计GET /api/stats/aggregate- 聚合统计
查询参数支持:provider_id、model_name、start_date、end_date、group_by
配置
配置支持多种方式,优先级为:CLI 参数 > 环境变量 > 配置文件 > 默认值
配置文件
配置文件位于 ~/.nex/config.yaml,首次启动自动生成:
server:
port: 9826
read_timeout: 30s
write_timeout: 30s
database:
path: ~/.nex/config.db
max_idle_conns: 10
max_open_conns: 100
conn_max_lifetime: 1h
log:
level: info
path: ~/.nex/log
max_size: 100 # MB
max_backups: 10
max_age: 30 # 天
compress: true
环境变量
所有配置项支持环境变量,使用 NEX_ 前缀:
export NEX_SERVER_PORT=9000
export NEX_DATABASE_PATH=/data/nex.db
export NEX_LOG_LEVEL=debug
命名规则:配置路径转大写 + 下划线(如 server.port → NEX_SERVER_PORT)。
CLI 参数
./server --server-port 9000 --log-level debug --database-path /tmp/test.db
命名规则:配置路径转 kebab-case(如 server.port → --server-port)。
数据文件
~/.nex/config.yaml- 配置文件~/.nex/config.db- SQLite 数据库~/.nex/log/- 日志目录
测试
make backend-test # 后端测试
make backend-test-coverage # 后端覆盖率
make frontend-test # 前端测试
make frontend-test-e2e # 前端 E2E 测试
开发
make backend-build # 构建后端
make backend-run # 运行后端
make backend-lint # 后端代码检查
make backend-migrate-up # 数据库迁移
make frontend-build # 构建前端
make frontend-dev # 前端开发模式
make frontend-lint # 前端代码检查
开发规范
详见各子项目的 README.md:
许可证
MIT
Description
Languages
Go
74.4%
TypeScript
14.9%
Python
9.9%
Makefile
0.3%
Shell
0.3%
Other
0.1%