mirror of
https://github.com/Chanzhaoyu/chatgpt-web.git
synced 2025-10-16 16:14:05 +00:00
chore: version 2.10.0
* feat: 权限验证功能 * chore: v2.10.0 * feat: 500 服务异常页面 * feat: 只有结束才会滚动到底部 * chore: 修改 CHANGELOG * chore: 不存在时输出默认报错
This commit is contained in:
41
CHANGELOG.md
41
CHANGELOG.md
@@ -1,3 +1,44 @@
|
|||||||
|
## v2.10.0
|
||||||
|
|
||||||
|
`2023-03-07`
|
||||||
|
|
||||||
|
- 老规矩,手动部署的同学需要删除 `node_modules` 安装包重新安装降低出错概率,其他部署不受影响,但是可能会有缓存问题。
|
||||||
|
- 虽然说了更新放缓,但是 `issues` 不看, `PR` 不改我睡不着,我的邮箱从每天早上`8`点到凌晨`12`永远在滴滴滴,所以求求各位,超时的`issues`自己关闭下哈,我真的需要缓冲一下。
|
||||||
|
- 演示图片请看最后
|
||||||
|
|
||||||
|
## Feature
|
||||||
|
- 添加权限功能,用法:`service/.env` 中的 `AUTH_SECRET_KEY` 变量添加密码
|
||||||
|
- 感谢 [PeterDaveHello](https://github.com/Chanzhaoyu/chatgpt-web/pull/348) 添加「繁体中文」翻译
|
||||||
|
- 感谢 [GermMC](https://github.com/Chanzhaoyu/chatgpt-web/pull/369) 添加聊天记录导入、导出、清空的功能
|
||||||
|
- 感谢 [CornerSkyless](https://github.com/Chanzhaoyu/chatgpt-web/pull/374) 添加会话保存为本地图片的功能
|
||||||
|
|
||||||
|
|
||||||
|
## Enhancement
|
||||||
|
- 感谢 [CornerSkyless](https://github.com/Chanzhaoyu/chatgpt-web/pull/363) 添加 `ctrl+enter` 发送消息
|
||||||
|
- 现在新消息只有在结束了之后才滚动到底部,而不是之前的强制性
|
||||||
|
- 优化部分代码
|
||||||
|
|
||||||
|
## BugFix
|
||||||
|
- 转义状态码前端显示,防止直接暴露 `key`(我可能需要更多的状态码补充)
|
||||||
|
|
||||||
|
## Other
|
||||||
|
- 更新依赖到最新
|
||||||
|
|
||||||
|
## 演示
|
||||||
|
> 不是界面最新效果,有美化改动
|
||||||
|
|
||||||
|
权限
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
聊天记录导出
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
保存图片到本地
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
## v2.9.3
|
## v2.9.3
|
||||||
|
|
||||||
`2023-03-06`
|
`2023-03-06`
|
||||||
|
@@ -163,6 +163,7 @@ pnpm dev
|
|||||||
- `OPENAI_ACCESS_TOKEN` one of two, `OPENAI_API_KEY` takes precedence when both are present
|
- `OPENAI_ACCESS_TOKEN` one of two, `OPENAI_API_KEY` takes precedence when both are present
|
||||||
- `OPENAI_API_BASE_URL` optional, available when `OPENAI_API_KEY` is set
|
- `OPENAI_API_BASE_URL` optional, available when `OPENAI_API_KEY` is set
|
||||||
- `API_REVERSE_PROXY` optional, available when `OPENAI_ACCESS_TOKEN` is set [Reference](#introduction)
|
- `API_REVERSE_PROXY` optional, available when `OPENAI_ACCESS_TOKEN` is set [Reference](#introduction)
|
||||||
|
- `AUTH_SECRET_KEY` Access Password,optional
|
||||||
- `TIMEOUT_MS` timeout, in milliseconds, optional
|
- `TIMEOUT_MS` timeout, in milliseconds, optional
|
||||||
- `SOCKS_PROXY_HOST` optional, effective with SOCKS_PROXY_PORT
|
- `SOCKS_PROXY_HOST` optional, effective with SOCKS_PROXY_PORT
|
||||||
- `SOCKS_PROXY_PORT` optional, effective with SOCKS_PROXY_HOST
|
- `SOCKS_PROXY_PORT` optional, effective with SOCKS_PROXY_HOST
|
||||||
@@ -205,6 +206,8 @@ services:
|
|||||||
OPENAI_API_BASE_URL: xxxx
|
OPENAI_API_BASE_URL: xxxx
|
||||||
# reverse proxy, optional
|
# reverse proxy, optional
|
||||||
API_REVERSE_PROXY: xxx
|
API_REVERSE_PROXY: xxx
|
||||||
|
# access password,optional
|
||||||
|
AUTH_SECRET_KEY: xxx
|
||||||
# timeout, in milliseconds, optional
|
# timeout, in milliseconds, optional
|
||||||
TIMEOUT_MS: 60000
|
TIMEOUT_MS: 60000
|
||||||
# socks proxy, optional, effective with SOCKS_PROXY_PORT
|
# socks proxy, optional, effective with SOCKS_PROXY_PORT
|
||||||
@@ -223,7 +226,8 @@ The `OPENAI_API_BASE_URL` is optional and only used when setting the `OPENAI_API
|
|||||||
| Environment Variable | Required | Description |
|
| Environment Variable | Required | Description |
|
||||||
| -------------------- | -------- | ------------------------------------------------------------------------------------------------- |
|
| -------------------- | -------- | ------------------------------------------------------------------------------------------------- |
|
||||||
| `PORT` | Required | Default: `3002` |
|
| `PORT` | Required | Default: `3002` |
|
||||||
| `TIMEOUT_MS` | Optional | Timeout in milliseconds. |
|
| `AUTH_SECRET_KEY` | Optional | access password |
|
||||||
|
| `TIMEOUT_MS` | Optional | Timeout in milliseconds |
|
||||||
| `OPENAI_API_KEY` | Optional | Required for `OpenAI API`. `apiKey` can be obtained from [here](https://platform.openai.com/overview). |
|
| `OPENAI_API_KEY` | Optional | Required for `OpenAI API`. `apiKey` can be obtained from [here](https://platform.openai.com/overview). |
|
||||||
| `OPENAI_ACCESS_TOKEN`| Optional | Required for `Web API`. `accessToken` can be obtained from [here](https://chat.openai.com/api/auth/session).|
|
| `OPENAI_ACCESS_TOKEN`| Optional | Required for `Web API`. `accessToken` can be obtained from [here](https://chat.openai.com/api/auth/session).|
|
||||||
| `OPENAI_API_BASE_URL` | Optional, only for `OpenAI API` | API endpoint. |
|
| `OPENAI_API_BASE_URL` | Optional, only for `OpenAI API` | API endpoint. |
|
||||||
|
@@ -161,6 +161,7 @@ pnpm dev
|
|||||||
- `OPENAI_ACCESS_TOKEN` 二选一,同时存在时,`OPENAI_API_KEY` 优先
|
- `OPENAI_ACCESS_TOKEN` 二选一,同时存在时,`OPENAI_API_KEY` 优先
|
||||||
- `OPENAI_API_BASE_URL` 可选,设置 `OPENAI_API_KEY` 时可用
|
- `OPENAI_API_BASE_URL` 可选,设置 `OPENAI_API_KEY` 时可用
|
||||||
- `API_REVERSE_PROXY` 可选,设置 `OPENAI_ACCESS_TOKEN` 时可用 [参考](#介绍)
|
- `API_REVERSE_PROXY` 可选,设置 `OPENAI_ACCESS_TOKEN` 时可用 [参考](#介绍)
|
||||||
|
- `AUTH_SECRET_KEY` 访问权限密钥,可选
|
||||||
- `TIMEOUT_MS` 超时,单位毫秒,可选
|
- `TIMEOUT_MS` 超时,单位毫秒,可选
|
||||||
- `SOCKS_PROXY_HOST` 可选,和 SOCKS_PROXY_PORT 一起时生效
|
- `SOCKS_PROXY_HOST` 可选,和 SOCKS_PROXY_PORT 一起时生效
|
||||||
- `SOCKS_PROXY_PORT` 可选,和 SOCKS_PROXY_HOST 一起时生效
|
- `SOCKS_PROXY_PORT` 可选,和 SOCKS_PROXY_HOST 一起时生效
|
||||||
@@ -203,6 +204,8 @@ services:
|
|||||||
OPENAI_API_BASE_URL: xxxx
|
OPENAI_API_BASE_URL: xxxx
|
||||||
# 反向代理,可选
|
# 反向代理,可选
|
||||||
API_REVERSE_PROXY: xxx
|
API_REVERSE_PROXY: xxx
|
||||||
|
# 访问权限密钥,可选
|
||||||
|
AUTH_SECRET_KEY: xxx
|
||||||
# 超时,单位毫秒,可选
|
# 超时,单位毫秒,可选
|
||||||
TIMEOUT_MS: 60000
|
TIMEOUT_MS: 60000
|
||||||
# Socks代理,可选,和 SOCKS_PROXY_PORT 一起时生效
|
# Socks代理,可选,和 SOCKS_PROXY_PORT 一起时生效
|
||||||
@@ -219,8 +222,9 @@ services:
|
|||||||
|
|
||||||
| 环境变量名称 | 必填 | 备注 |
|
| 环境变量名称 | 必填 | 备注 |
|
||||||
| --------------------- | ---------------------- | -------------------------------------------------------------------------------------------------- |
|
| --------------------- | ---------------------- | -------------------------------------------------------------------------------------------------- |
|
||||||
| `PORT` | 必填 | 默认 `3002` |
|
| `PORT` | 必填 | 默认 `3002`
|
||||||
| `TIMEOUT_MS` | 可选 | 超时时间,单位毫秒, |
|
| `AUTH_SECRET_KEY` | 可选 | 访问权限密钥 |
|
||||||
|
| `TIMEOUT_MS` | 可选 | 超时时间,单位毫秒 |
|
||||||
| `OPENAI_API_KEY` | `OpenAI API` 二选一 | 使用 `OpenAI API` 所需的 `apiKey` [(获取 apiKey)](https://platform.openai.com/overview) |
|
| `OPENAI_API_KEY` | `OpenAI API` 二选一 | 使用 `OpenAI API` 所需的 `apiKey` [(获取 apiKey)](https://platform.openai.com/overview) |
|
||||||
| `OPENAI_ACCESS_TOKEN` | `Web API` 二选一 | 使用 `Web API` 所需的 `accessToken` [(获取 accessToken)](https://chat.openai.com/api/auth/session) |
|
| `OPENAI_ACCESS_TOKEN` | `Web API` 二选一 | 使用 `Web API` 所需的 `accessToken` [(获取 accessToken)](https://chat.openai.com/api/auth/session) |
|
||||||
| `OPENAI_API_BASE_URL` | 可选,`OpenAI API` 时可用 | `API`接口地址 |
|
| `OPENAI_API_BASE_URL` | 可选,`OpenAI API` 时可用 | `API`接口地址 |
|
||||||
|
@@ -14,6 +14,8 @@ services:
|
|||||||
OPENAI_API_BASE_URL: xxxx
|
OPENAI_API_BASE_URL: xxxx
|
||||||
# 反向代理,可选
|
# 反向代理,可选
|
||||||
API_REVERSE_PROXY: xxx
|
API_REVERSE_PROXY: xxx
|
||||||
|
# 访问权限密钥,可选
|
||||||
|
AUTH_SECRET_KEY: xxx
|
||||||
# 超时,单位毫秒,可选
|
# 超时,单位毫秒,可选
|
||||||
TIMEOUT_MS: 60000
|
TIMEOUT_MS: 60000
|
||||||
# Socks代理,可选,和 SOCKS_PROXY_PORT 一起时生效
|
# Socks代理,可选,和 SOCKS_PROXY_PORT 一起时生效
|
||||||
|
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "chatgpt-web",
|
"name": "chatgpt-web",
|
||||||
"version": "2.9.3",
|
"version": "2.10.0",
|
||||||
"private": false,
|
"private": false,
|
||||||
"description": "ChatGPT Web",
|
"description": "ChatGPT Web",
|
||||||
"author": "ChenZhaoYu <chenzhaoyu1994@gmail.com>",
|
"author": "ChenZhaoYu <chenzhaoyu1994@gmail.com>",
|
||||||
|
@@ -13,6 +13,9 @@ API_REVERSE_PROXY=
|
|||||||
# timeout
|
# timeout
|
||||||
TIMEOUT_MS=100000
|
TIMEOUT_MS=100000
|
||||||
|
|
||||||
|
# Secret key
|
||||||
|
AUTH_SECRET_KEY=
|
||||||
|
|
||||||
# Socks Proxy Host
|
# Socks Proxy Host
|
||||||
SOCKS_PROXY_HOST=
|
SOCKS_PROXY_HOST=
|
||||||
|
|
||||||
|
@@ -24,7 +24,7 @@
|
|||||||
"common:cleanup": "rimraf node_modules && rimraf pnpm-lock.yaml"
|
"common:cleanup": "rimraf node_modules && rimraf pnpm-lock.yaml"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"chatgpt": "^5.0.7",
|
"chatgpt": "^5.0.8",
|
||||||
"dotenv": "^16.0.3",
|
"dotenv": "^16.0.3",
|
||||||
"esno": "^0.16.3",
|
"esno": "^0.16.3",
|
||||||
"express": "^4.18.2",
|
"express": "^4.18.2",
|
||||||
|
8
service/pnpm-lock.yaml
generated
8
service/pnpm-lock.yaml
generated
@@ -4,7 +4,7 @@ specifiers:
|
|||||||
'@antfu/eslint-config': ^0.35.3
|
'@antfu/eslint-config': ^0.35.3
|
||||||
'@types/express': ^4.17.17
|
'@types/express': ^4.17.17
|
||||||
'@types/node': ^18.14.6
|
'@types/node': ^18.14.6
|
||||||
chatgpt: ^5.0.7
|
chatgpt: ^5.0.8
|
||||||
dotenv: ^16.0.3
|
dotenv: ^16.0.3
|
||||||
eslint: ^8.35.0
|
eslint: ^8.35.0
|
||||||
esno: ^0.16.3
|
esno: ^0.16.3
|
||||||
@@ -17,7 +17,7 @@ specifiers:
|
|||||||
typescript: ^4.9.5
|
typescript: ^4.9.5
|
||||||
|
|
||||||
dependencies:
|
dependencies:
|
||||||
chatgpt: 5.0.7
|
chatgpt: 5.0.8
|
||||||
dotenv: 16.0.3
|
dotenv: 16.0.3
|
||||||
esno: 0.16.3
|
esno: 0.16.3
|
||||||
express: 4.18.2
|
express: 4.18.2
|
||||||
@@ -902,8 +902,8 @@ packages:
|
|||||||
resolution: {integrity: sha512-mKKUkUbhPpQlCOfIuZkvSEgktjPFIsZKRRbC6KWVEMvlzblj3i3asQv5ODsrwt0N3pHAEvjP8KTQPHkp0+6jOg==}
|
resolution: {integrity: sha512-mKKUkUbhPpQlCOfIuZkvSEgktjPFIsZKRRbC6KWVEMvlzblj3i3asQv5ODsrwt0N3pHAEvjP8KTQPHkp0+6jOg==}
|
||||||
dev: true
|
dev: true
|
||||||
|
|
||||||
/chatgpt/5.0.7:
|
/chatgpt/5.0.8:
|
||||||
resolution: {integrity: sha512-wy69++JDNS0xKi+6rP+HDOByXBafQIVynHnlQw09apuDntGSKfwBRY902N8Q7/ZFU/XET+8NpJiio2iI69IWYw==}
|
resolution: {integrity: sha512-Bjh7Y15QIsZ+SkQvbbZGymv1PGxkZ7X1vwqAwvyqaMMhbipU4kxht/GL62VCxhoUCXPwxTfScbFeNFtNldgqaw==}
|
||||||
engines: {node: '>=14'}
|
engines: {node: '>=14'}
|
||||||
hasBin: true
|
hasBin: true
|
||||||
dependencies:
|
dependencies:
|
||||||
|
@@ -8,11 +8,14 @@ import { sendResponse } from '../utils'
|
|||||||
import type { ApiModel, ChatContext, ChatGPTUnofficialProxyAPIOptions, ModelConfig } from '../types'
|
import type { ApiModel, ChatContext, ChatGPTUnofficialProxyAPIOptions, ModelConfig } from '../types'
|
||||||
|
|
||||||
const ErrorCodeMessage: Record<string, string> = {
|
const ErrorCodeMessage: Record<string, string> = {
|
||||||
401: '提供错误的API密钥 | Incorrect API key provided',
|
400: '[OpenAI] 模型的最大上下文长度是4096个令牌,请减少信息的长度。| This model\'s maximum context length is 4096 tokens.',
|
||||||
429: '服务器限流,请稍后再试 | Server was limited, please try again later',
|
401: '[OpenAI] 提供错误的API密钥 | Incorrect API key provided',
|
||||||
503: '服务器繁忙,请稍后再试 | Server is busy, please try again later',
|
403: '[OpenAI] 服务器拒绝访问,请稍后再试 | Server refused to access, please try again later',
|
||||||
500: '服务器繁忙,请稍后再试 | Server is busy, please try again later',
|
429: '[OpenAI] 服务器限流,请稍后再试 | Server was limited, please try again later',
|
||||||
403: '服务器拒绝访问,请稍后再试 | Server refused to access, please try again later',
|
502: '[OpenAI] 错误的网关 | Bad Gateway',
|
||||||
|
503: '[OpenAI] 服务器繁忙,请稍后再试 | Server is busy, please try again later',
|
||||||
|
504: '[OpenAI] 网关超时 | Gateway Time-out',
|
||||||
|
500: '[OpenAI] 服务器繁忙,请稍后再试 | Internal Server Error',
|
||||||
}
|
}
|
||||||
|
|
||||||
dotenv.config()
|
dotenv.config()
|
||||||
@@ -106,10 +109,11 @@ async function chatReplyProcess(
|
|||||||
return sendResponse({ type: 'Success', data: response })
|
return sendResponse({ type: 'Success', data: response })
|
||||||
}
|
}
|
||||||
catch (error: any) {
|
catch (error: any) {
|
||||||
const code = error.statusCode || 'unknown'
|
const code = error.statusCode
|
||||||
|
global.console.log(error)
|
||||||
if (Reflect.has(ErrorCodeMessage, code))
|
if (Reflect.has(ErrorCodeMessage, code))
|
||||||
return sendResponse({ type: 'Fail', message: ErrorCodeMessage[code] })
|
return sendResponse({ type: 'Fail', message: ErrorCodeMessage[code] })
|
||||||
return sendResponse({ type: 'Fail', message: `${error.statusCode}-${error.statusText}` })
|
return sendResponse({ type: 'Fail', message: error.message ?? 'Please check the back-end console' })
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -1,6 +1,7 @@
|
|||||||
import express from 'express'
|
import express from 'express'
|
||||||
import type { ChatContext, ChatMessage } from './chatgpt'
|
import type { ChatContext, ChatMessage } from './chatgpt'
|
||||||
import { chatConfig, chatReplyProcess } from './chatgpt'
|
import { chatConfig, chatReplyProcess } from './chatgpt'
|
||||||
|
import { auth } from './middleware/auth'
|
||||||
|
|
||||||
const app = express()
|
const app = express()
|
||||||
const router = express.Router()
|
const router = express.Router()
|
||||||
@@ -15,7 +16,7 @@ app.all('*', (_, res, next) => {
|
|||||||
next()
|
next()
|
||||||
})
|
})
|
||||||
|
|
||||||
router.post('/chat-process', async (req, res) => {
|
router.post('/chat-process', auth, async (req, res) => {
|
||||||
res.setHeader('Content-type', 'application/octet-stream')
|
res.setHeader('Content-type', 'application/octet-stream')
|
||||||
|
|
||||||
try {
|
try {
|
||||||
@@ -44,6 +45,33 @@ router.post('/config', async (req, res) => {
|
|||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
|
router.post('/session', async (req, res) => {
|
||||||
|
try {
|
||||||
|
const AUTH_SECRET_KEY = process.env.AUTH_SECRET_KEY
|
||||||
|
const hasAuth = typeof AUTH_SECRET_KEY === 'string' && AUTH_SECRET_KEY.length > 0
|
||||||
|
res.send({ status: 'Success', message: '', data: { auth: hasAuth } })
|
||||||
|
}
|
||||||
|
catch (error) {
|
||||||
|
res.send({ status: 'Fail', message: error.message, data: null })
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
router.post('/verify', async (req, res) => {
|
||||||
|
try {
|
||||||
|
const { token } = req.body as { token: string }
|
||||||
|
if (!token)
|
||||||
|
throw new Error('Secret key is empty')
|
||||||
|
|
||||||
|
if (process.env.AUTH_SECRET_KEY !== token)
|
||||||
|
throw new Error('密钥无效 | Secret key is invalid')
|
||||||
|
|
||||||
|
res.send({ status: 'Success', message: 'Verify successfully', data: null })
|
||||||
|
}
|
||||||
|
catch (error) {
|
||||||
|
res.send({ status: 'Fail', message: error.message, data: null })
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
app.use('', router)
|
app.use('', router)
|
||||||
app.use('/api', router)
|
app.use('/api', router)
|
||||||
|
|
||||||
|
19
service/src/middleware/auth.ts
Normal file
19
service/src/middleware/auth.ts
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
const auth = async (req, res, next) => {
|
||||||
|
const AUTH_SECRET_KEY = process.env.AUTH_SECRET_KEY
|
||||||
|
if (typeof AUTH_SECRET_KEY === 'string' && AUTH_SECRET_KEY.length > 0) {
|
||||||
|
try {
|
||||||
|
const Authorization = req.header('Authorization')
|
||||||
|
if (!Authorization || Authorization.replace('Bearer ', '').trim() !== AUTH_SECRET_KEY.trim())
|
||||||
|
throw new Error('Error: 无访问权限 | No access rights')
|
||||||
|
next()
|
||||||
|
}
|
||||||
|
catch (error) {
|
||||||
|
res.send({ status: 'Unauthorized', message: error.message ?? 'Please authenticate.', data: null })
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
next()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export { auth }
|
@@ -33,3 +33,16 @@ export function fetchChatAPIProcess<T = any>(
|
|||||||
onDownloadProgress: params.onDownloadProgress,
|
onDownloadProgress: params.onDownloadProgress,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function fetchSession<T>() {
|
||||||
|
return post<T>({
|
||||||
|
url: '/session',
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
export function fetchVerify<T>(token: string) {
|
||||||
|
return post<T>({
|
||||||
|
url: '/verify',
|
||||||
|
data: { token },
|
||||||
|
})
|
||||||
|
}
|
||||||
|
File diff suppressed because one or more lines are too long
Before Width: | Height: | Size: 19 KiB |
5
src/icons/403.vue
Normal file
5
src/icons/403.vue
Normal file
File diff suppressed because one or more lines are too long
5
src/icons/500.vue
Normal file
5
src/icons/500.vue
Normal file
File diff suppressed because one or more lines are too long
@@ -12,6 +12,8 @@ export default {
|
|||||||
wrong: 'Something went wrong, please try again later.',
|
wrong: 'Something went wrong, please try again later.',
|
||||||
success: 'Success',
|
success: 'Success',
|
||||||
failed: 'Failed',
|
failed: 'Failed',
|
||||||
|
verify: 'Verify',
|
||||||
|
unauthorizedTips: 'Unauthorized, please verify first.',
|
||||||
},
|
},
|
||||||
chat: {
|
chat: {
|
||||||
placeholder: 'Ask me anything...(Shift + Enter = line break)',
|
placeholder: 'Ask me anything...(Shift + Enter = line break)',
|
||||||
|
@@ -12,6 +12,8 @@ export default {
|
|||||||
wrong: '好像出错了,请稍后再试。',
|
wrong: '好像出错了,请稍后再试。',
|
||||||
success: '操作成功',
|
success: '操作成功',
|
||||||
failed: '操作失败',
|
failed: '操作失败',
|
||||||
|
verify: '验证',
|
||||||
|
unauthorizedTips: '未经授权,请先进行验证。',
|
||||||
},
|
},
|
||||||
chat: {
|
chat: {
|
||||||
placeholder: '来说点什么...(Shift + Enter = 换行)',
|
placeholder: '来说点什么...(Shift + Enter = 换行)',
|
||||||
|
@@ -12,6 +12,8 @@ export default {
|
|||||||
wrong: '好像出錯了,請稍後再試。',
|
wrong: '好像出錯了,請稍後再試。',
|
||||||
success: '操作成功',
|
success: '操作成功',
|
||||||
failed: '操作失敗',
|
failed: '操作失敗',
|
||||||
|
verify: '驗證',
|
||||||
|
unauthorizedTips: '未經授權,請先進行驗證。',
|
||||||
},
|
},
|
||||||
chat: {
|
chat: {
|
||||||
placeholder: '來講點什麼...(Shift + Enter = 換行)',
|
placeholder: '來講點什麼...(Shift + Enter = 換行)',
|
||||||
|
@@ -1,6 +1,7 @@
|
|||||||
import type { App } from 'vue'
|
import type { App } from 'vue'
|
||||||
import type { RouteRecordRaw } from 'vue-router'
|
import type { RouteRecordRaw } from 'vue-router'
|
||||||
import { createRouter, createWebHashHistory } from 'vue-router'
|
import { createRouter, createWebHashHistory } from 'vue-router'
|
||||||
|
import { setupPageGuard } from './permission'
|
||||||
import { ChatLayout } from '@/views/chat/layout'
|
import { ChatLayout } from '@/views/chat/layout'
|
||||||
|
|
||||||
const routes: RouteRecordRaw[] = [
|
const routes: RouteRecordRaw[] = [
|
||||||
@@ -18,18 +19,18 @@ const routes: RouteRecordRaw[] = [
|
|||||||
],
|
],
|
||||||
},
|
},
|
||||||
|
|
||||||
{
|
|
||||||
path: '/403',
|
|
||||||
name: '403',
|
|
||||||
component: () => import('@/views/exception/403/index.vue'),
|
|
||||||
},
|
|
||||||
|
|
||||||
{
|
{
|
||||||
path: '/404',
|
path: '/404',
|
||||||
name: '404',
|
name: '404',
|
||||||
component: () => import('@/views/exception/404/index.vue'),
|
component: () => import('@/views/exception/404/index.vue'),
|
||||||
},
|
},
|
||||||
|
|
||||||
|
{
|
||||||
|
path: '/500',
|
||||||
|
name: '500',
|
||||||
|
component: () => import('@/views/exception/500/index.vue'),
|
||||||
|
},
|
||||||
|
|
||||||
{
|
{
|
||||||
path: '/:pathMatch(.*)*',
|
path: '/:pathMatch(.*)*',
|
||||||
name: 'notFound',
|
name: 'notFound',
|
||||||
@@ -43,6 +44,8 @@ export const router = createRouter({
|
|||||||
scrollBehavior: () => ({ left: 0, top: 0 }),
|
scrollBehavior: () => ({ left: 0, top: 0 }),
|
||||||
})
|
})
|
||||||
|
|
||||||
|
setupPageGuard(router)
|
||||||
|
|
||||||
export async function setupRouter(app: App) {
|
export async function setupRouter(app: App) {
|
||||||
app.use(router)
|
app.use(router)
|
||||||
await router.isReady()
|
await router.isReady()
|
||||||
|
25
src/router/permission.ts
Normal file
25
src/router/permission.ts
Normal file
@@ -0,0 +1,25 @@
|
|||||||
|
import type { Router } from 'vue-router'
|
||||||
|
import { useAuthStoreWithout } from '@/store/modules/auth'
|
||||||
|
|
||||||
|
export function setupPageGuard(router: Router) {
|
||||||
|
router.beforeEach(async (from, to, next) => {
|
||||||
|
const authStore = useAuthStoreWithout()
|
||||||
|
if (!authStore.session) {
|
||||||
|
try {
|
||||||
|
const data = await authStore.getSession()
|
||||||
|
if (String(data.auth) === 'false' && authStore.token)
|
||||||
|
authStore.removeToken()
|
||||||
|
next()
|
||||||
|
}
|
||||||
|
catch (error) {
|
||||||
|
if (from.path !== '/500')
|
||||||
|
next({ name: '500' })
|
||||||
|
else
|
||||||
|
next()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
next()
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
15
src/store/modules/auth/helper.ts
Normal file
15
src/store/modules/auth/helper.ts
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
import { ss } from '@/utils/storage'
|
||||||
|
|
||||||
|
const LOCAL_NAME = 'SECRET_TOKEN'
|
||||||
|
|
||||||
|
export function getToken() {
|
||||||
|
return ss.get(LOCAL_NAME)
|
||||||
|
}
|
||||||
|
|
||||||
|
export function setToken(token: string) {
|
||||||
|
return ss.set(LOCAL_NAME, token)
|
||||||
|
}
|
||||||
|
|
||||||
|
export function removeToken() {
|
||||||
|
return ss.remove(LOCAL_NAME)
|
||||||
|
}
|
43
src/store/modules/auth/index.ts
Normal file
43
src/store/modules/auth/index.ts
Normal file
@@ -0,0 +1,43 @@
|
|||||||
|
import { defineStore } from 'pinia'
|
||||||
|
import { getToken, removeToken, setToken } from './helper'
|
||||||
|
import { store } from '@/store'
|
||||||
|
import { fetchSession } from '@/api'
|
||||||
|
|
||||||
|
export interface AuthState {
|
||||||
|
token: string | undefined
|
||||||
|
session: { auth: boolean } | null
|
||||||
|
}
|
||||||
|
|
||||||
|
export const useAuthStore = defineStore('auth-store', {
|
||||||
|
state: (): AuthState => ({
|
||||||
|
token: getToken(),
|
||||||
|
session: null,
|
||||||
|
}),
|
||||||
|
|
||||||
|
actions: {
|
||||||
|
async getSession() {
|
||||||
|
try {
|
||||||
|
const { data } = await fetchSession<{ auth: boolean }>()
|
||||||
|
this.session = { ...data }
|
||||||
|
return Promise.resolve(data)
|
||||||
|
}
|
||||||
|
catch (error) {
|
||||||
|
return Promise.reject(error)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
setToken(token: string) {
|
||||||
|
this.token = token
|
||||||
|
setToken(token)
|
||||||
|
},
|
||||||
|
|
||||||
|
removeToken() {
|
||||||
|
this.token = undefined
|
||||||
|
removeToken()
|
||||||
|
},
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
export function useAuthStoreWithout() {
|
||||||
|
return useAuthStore(store)
|
||||||
|
}
|
@@ -1,3 +1,4 @@
|
|||||||
export * from './app'
|
export * from './app'
|
||||||
export * from './chat'
|
export * from './chat'
|
||||||
export * from './user'
|
export * from './user'
|
||||||
|
export * from './auth'
|
||||||
|
@@ -1,4 +1,5 @@
|
|||||||
import axios, { type AxiosResponse } from 'axios'
|
import axios, { type AxiosResponse } from 'axios'
|
||||||
|
import { useAuthStore } from '@/store'
|
||||||
|
|
||||||
const service = axios.create({
|
const service = axios.create({
|
||||||
baseURL: import.meta.env.VITE_GLOB_API_URL,
|
baseURL: import.meta.env.VITE_GLOB_API_URL,
|
||||||
@@ -6,6 +7,9 @@ const service = axios.create({
|
|||||||
|
|
||||||
service.interceptors.request.use(
|
service.interceptors.request.use(
|
||||||
(config) => {
|
(config) => {
|
||||||
|
const token = useAuthStore().token
|
||||||
|
if (token)
|
||||||
|
config.headers.Authorization = `Bearer ${token}`
|
||||||
return config
|
return config
|
||||||
},
|
},
|
||||||
(error) => {
|
(error) => {
|
||||||
|
@@ -1,5 +1,6 @@
|
|||||||
import type { AxiosProgressEvent, AxiosResponse, GenericAbortSignal } from 'axios'
|
import type { AxiosProgressEvent, AxiosResponse, GenericAbortSignal } from 'axios'
|
||||||
import request from './axios'
|
import request from './axios'
|
||||||
|
import { useAuthStore } from '@/store'
|
||||||
|
|
||||||
export interface HttpOption {
|
export interface HttpOption {
|
||||||
url: string
|
url: string
|
||||||
@@ -22,9 +23,16 @@ function http<T = any>(
|
|||||||
{ url, data, method, headers, onDownloadProgress, signal, beforeRequest, afterRequest }: HttpOption,
|
{ url, data, method, headers, onDownloadProgress, signal, beforeRequest, afterRequest }: HttpOption,
|
||||||
) {
|
) {
|
||||||
const successHandler = (res: AxiosResponse<Response<T>>) => {
|
const successHandler = (res: AxiosResponse<Response<T>>) => {
|
||||||
|
const authStore = useAuthStore()
|
||||||
|
|
||||||
if (res.data.status === 'Success' || typeof res.data === 'string')
|
if (res.data.status === 'Success' || typeof res.data === 'string')
|
||||||
return res.data
|
return res.data
|
||||||
|
|
||||||
|
if (res.data.status === 'Unauthorized') {
|
||||||
|
authStore.removeToken()
|
||||||
|
window.location.reload()
|
||||||
|
}
|
||||||
|
|
||||||
return Promise.reject(res.data)
|
return Promise.reject(res.data)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -113,13 +113,13 @@ async function onConversation() {
|
|||||||
requestOptions: { prompt: message, options: { ...options } },
|
requestOptions: { prompt: message, options: { ...options } },
|
||||||
},
|
},
|
||||||
)
|
)
|
||||||
scrollToBottom()
|
|
||||||
}
|
}
|
||||||
catch (error) {
|
catch (error) {
|
||||||
//
|
//
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
|
scrollToBottom()
|
||||||
}
|
}
|
||||||
catch (error: any) {
|
catch (error: any) {
|
||||||
const errorMessage = error?.message ?? t('common.wrong')
|
const errorMessage = error?.message ?? t('common.wrong')
|
||||||
|
@@ -4,12 +4,14 @@ import { NLayout, NLayoutContent } from 'naive-ui'
|
|||||||
import { useRouter } from 'vue-router'
|
import { useRouter } from 'vue-router'
|
||||||
import Sider from './sider/index.vue'
|
import Sider from './sider/index.vue'
|
||||||
import Header from './header/index.vue'
|
import Header from './header/index.vue'
|
||||||
|
import Permission from './Permission.vue'
|
||||||
import { useBasicLayout } from '@/hooks/useBasicLayout'
|
import { useBasicLayout } from '@/hooks/useBasicLayout'
|
||||||
import { useAppStore, useChatStore } from '@/store'
|
import { useAppStore, useAuthStore, useChatStore } from '@/store'
|
||||||
|
|
||||||
const router = useRouter()
|
const router = useRouter()
|
||||||
const appStore = useAppStore()
|
const appStore = useAppStore()
|
||||||
const chatStore = useChatStore()
|
const chatStore = useChatStore()
|
||||||
|
const authStore = useAuthStore()
|
||||||
|
|
||||||
router.replace({ name: 'Chat', params: { uuid: chatStore.active } })
|
router.replace({ name: 'Chat', params: { uuid: chatStore.active } })
|
||||||
|
|
||||||
@@ -17,6 +19,8 @@ const { isMobile } = useBasicLayout()
|
|||||||
|
|
||||||
const collapsed = computed(() => appStore.siderCollapsed)
|
const collapsed = computed(() => appStore.siderCollapsed)
|
||||||
|
|
||||||
|
const needPermission = computed(() => !!authStore.session?.auth && !authStore.token)
|
||||||
|
|
||||||
const getMobileClass = computed(() => {
|
const getMobileClass = computed(() => {
|
||||||
if (isMobile.value)
|
if (isMobile.value)
|
||||||
return ['rounded-none', 'shadow-none']
|
return ['rounded-none', 'shadow-none']
|
||||||
@@ -44,5 +48,6 @@ const getContainerClass = computed(() => {
|
|||||||
</NLayoutContent>
|
</NLayoutContent>
|
||||||
</NLayout>
|
</NLayout>
|
||||||
</div>
|
</div>
|
||||||
|
<Permission :visible="needPermission" />
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
81
src/views/chat/layout/Permission.vue
Normal file
81
src/views/chat/layout/Permission.vue
Normal file
@@ -0,0 +1,81 @@
|
|||||||
|
<script setup lang='ts'>
|
||||||
|
import { computed, ref } from 'vue'
|
||||||
|
import { NButton, NInput, NModal, useMessage } from 'naive-ui'
|
||||||
|
import { fetchVerify } from '@/api'
|
||||||
|
import { useAuthStore } from '@/store'
|
||||||
|
import Icon403 from '@/icons/403.vue'
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
visible: boolean
|
||||||
|
}
|
||||||
|
|
||||||
|
defineProps<Props>()
|
||||||
|
|
||||||
|
const authStore = useAuthStore()
|
||||||
|
|
||||||
|
const ms = useMessage()
|
||||||
|
|
||||||
|
const loading = ref(false)
|
||||||
|
const token = ref('')
|
||||||
|
|
||||||
|
const disabled = computed(() => !token.value.trim() || loading.value)
|
||||||
|
|
||||||
|
async function handleVerify() {
|
||||||
|
const secretKey = token.value.trim()
|
||||||
|
|
||||||
|
if (!secretKey)
|
||||||
|
return
|
||||||
|
|
||||||
|
try {
|
||||||
|
loading.value = true
|
||||||
|
await fetchVerify(secretKey)
|
||||||
|
authStore.setToken(secretKey)
|
||||||
|
ms.success('success')
|
||||||
|
window.location.reload()
|
||||||
|
}
|
||||||
|
catch (error: any) {
|
||||||
|
ms.error(error.message ?? 'error')
|
||||||
|
authStore.removeToken()
|
||||||
|
token.value = ''
|
||||||
|
}
|
||||||
|
finally {
|
||||||
|
loading.value = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function handlePress(event: KeyboardEvent) {
|
||||||
|
if (event.key === 'Enter' && !event.shiftKey) {
|
||||||
|
event.preventDefault()
|
||||||
|
handleVerify()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<NModal :show="visible" style="width: 90%; max-width: 640px">
|
||||||
|
<div class="p-10 bg-white rounded dark:bg-slate-800">
|
||||||
|
<div class="space-y-4">
|
||||||
|
<header class="space-y-2">
|
||||||
|
<h2 class="text-2xl font-bold text-center text-slate-800 dark:text-neutral-200">
|
||||||
|
403
|
||||||
|
</h2>
|
||||||
|
<p class="text-base text-center text-slate-500 dark:text-slate-500">
|
||||||
|
{{ $t('common.unauthorizedTips') }}
|
||||||
|
</p>
|
||||||
|
<Icon403 class="w-[200px] m-auto" />
|
||||||
|
</header>
|
||||||
|
<NInput v-model:value="token" type="text" placeholder="" @keypress="handlePress" />
|
||||||
|
|
||||||
|
<NButton
|
||||||
|
block
|
||||||
|
type="primary"
|
||||||
|
:disabled="disabled"
|
||||||
|
:loading="loading"
|
||||||
|
@click="handleVerify"
|
||||||
|
>
|
||||||
|
{{ $t('common.verify') }}
|
||||||
|
</NButton>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</NModal>
|
||||||
|
</template>
|
@@ -1,34 +0,0 @@
|
|||||||
<script lang="ts" setup>
|
|
||||||
import { NButton } from 'naive-ui'
|
|
||||||
import { useRouter } from 'vue-router'
|
|
||||||
|
|
||||||
const router = useRouter()
|
|
||||||
|
|
||||||
function goHome() {
|
|
||||||
router.push('/')
|
|
||||||
}
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<template>
|
|
||||||
<div class="flex h-full">
|
|
||||||
<div class="px-4 m-auto space-y-4 text-center max-[400px]">
|
|
||||||
<h1 class="text-4xl text-slate-800 dark:text-neutral-200">
|
|
||||||
No permission
|
|
||||||
</h1>
|
|
||||||
<p class="text-base text-slate-500 dark:text-neutral-400">
|
|
||||||
The page you're trying access has restricted access.
|
|
||||||
Please refer to your system administrator
|
|
||||||
</p>
|
|
||||||
<div class="flex items-center justify-center text-center">
|
|
||||||
<div class="w-[300px]">
|
|
||||||
<div class="w-[300px]">
|
|
||||||
<img src="../../../icons/403.svg" alt="404">
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<NButton type="primary" @click="goHome">
|
|
||||||
Go to Home
|
|
||||||
</NButton>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</template>
|
|
32
src/views/exception/500/index.vue
Normal file
32
src/views/exception/500/index.vue
Normal file
@@ -0,0 +1,32 @@
|
|||||||
|
<script lang="ts" setup>
|
||||||
|
import { NButton } from 'naive-ui'
|
||||||
|
import { useRouter } from 'vue-router'
|
||||||
|
import Icon500 from '@/icons/500.vue'
|
||||||
|
|
||||||
|
const router = useRouter()
|
||||||
|
|
||||||
|
function goHome() {
|
||||||
|
router.push('/')
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<div class="flex h-full dark:bg-neutral-800">
|
||||||
|
<div class="px-4 m-auto space-y-4 text-center max-[400px]">
|
||||||
|
<header class="space-y-2">
|
||||||
|
<h2 class="text-2xl font-bold text-center text-slate-800 dark:text-neutral-200">
|
||||||
|
500
|
||||||
|
</h2>
|
||||||
|
<p class="text-base text-center text-slate-500 dark:text-slate-500">
|
||||||
|
Server error
|
||||||
|
</p>
|
||||||
|
<div class="flex items-center justify-center text-center">
|
||||||
|
<Icon500 class="w-[300px]" />
|
||||||
|
</div>
|
||||||
|
</header>
|
||||||
|
<NButton type="primary" @click="goHome">
|
||||||
|
Go to Home
|
||||||
|
</NButton>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
Reference in New Issue
Block a user