From 322d381d6cd86b0af204cc5a96ec1bc405971e12 Mon Sep 17 00:00:00 2001 From: Bob Du Date: Fri, 15 Mar 2024 15:14:19 +0800 Subject: [PATCH] feat: support vision model Signed-off-by: Bob Du --- service/.gitignore | 1 + service/package.json | 6 +- service/pnpm-lock.yaml | 172 ++++++++++++++++---- service/src/chatgpt/index.ts | 55 ++++++- service/src/chatgpt/types.ts | 1 + service/src/index.ts | 11 +- service/src/routes/upload.ts | 31 ++++ service/src/storage/config.ts | 2 +- service/src/storage/model.ts | 4 +- service/src/storage/mongo.ts | 6 +- service/src/types.ts | 1 + service/src/utils/image.ts | 18 ++ src/api/index.ts | 2 + src/typings/chat.d.ts | 1 + src/views/chat/components/Message/Text.vue | 2 + src/views/chat/components/Message/index.vue | 2 + src/views/chat/index.vue | 60 ++++++- vite.config.ts | 4 + 18 files changed, 337 insertions(+), 42 deletions(-) create mode 100644 service/src/routes/upload.ts create mode 100644 service/src/utils/image.ts diff --git a/service/.gitignore b/service/.gitignore index f073f126..216d8697 100644 --- a/service/.gitignore +++ b/service/.gitignore @@ -30,3 +30,4 @@ coverage build public +uploads diff --git a/service/package.json b/service/package.json index b619b07c..ac61e63f 100644 --- a/service/package.json +++ b/service/package.json @@ -28,7 +28,7 @@ }, "dependencies": { "axios": "^1.6.7", - "chatgpt": "^5.2.4", + "chatgpt": "git+https://github.com/chatgpt-web-dev/chatgpt-api#e8fca6bf34350881b1e9243610941bc4b00ce891", "dayjs": "^1.11.7", "dotenv": "^16.0.3", "express": "^4.18.3", @@ -39,6 +39,7 @@ "jsonwebtoken": "^9.0.0", "jwt-decode": "^3.1.2", "mongodb": "^5.9.2", + "multer": "1.4.5-lts.1", "node-fetch": "^3.3.0", "nodemailer": "^6.9.9", "request-ip": "^3.3.0", @@ -48,9 +49,10 @@ }, "devDependencies": { "@antfu/eslint-config": "^0.43.1", - "@types/express": "^4.17.17", + "@types/express": "^4.17.21", "@types/isomorphic-fetch": "^0.0.39", "@types/jsonwebtoken": "^9.0.5", + "@types/multer": "^1.4.11", "@types/node": "^18.14.6", "@types/nodemailer": "^6.4.14", "@types/request-ip": "^0.0.41", diff --git a/service/pnpm-lock.yaml b/service/pnpm-lock.yaml index 45329b9d..74fc20db 100644 --- a/service/pnpm-lock.yaml +++ b/service/pnpm-lock.yaml @@ -9,8 +9,8 @@ dependencies: specifier: ^1.6.7 version: 1.6.7 chatgpt: - specifier: ^5.2.4 - version: 5.2.4 + specifier: git+https://github.com/chatgpt-web-dev/chatgpt-api#e8fca6bf34350881b1e9243610941bc4b00ce891 + version: git@github.com+chatgpt-web-dev/chatgpt-api/e8fca6bf34350881b1e9243610941bc4b00ce891 dayjs: specifier: ^1.11.7 version: 1.11.7 @@ -41,6 +41,9 @@ dependencies: mongodb: specifier: ^5.9.2 version: 5.9.2 + multer: + specifier: 1.4.5-lts.1 + version: 1.4.5-lts.1 node-fetch: specifier: ^3.3.0 version: 3.3.0 @@ -65,14 +68,17 @@ devDependencies: specifier: ^0.43.1 version: 0.43.1(eslint@8.56.0)(typescript@5.3.3) '@types/express': - specifier: ^4.17.17 - version: 4.17.17 + specifier: ^4.17.21 + version: 4.17.21 '@types/isomorphic-fetch': specifier: ^0.0.39 version: 0.0.39 '@types/jsonwebtoken': specifier: ^9.0.5 version: 9.0.5 + '@types/multer': + specifier: ^1.4.11 + version: 1.4.11 '@types/node': specifier: ^18.14.6 version: 18.14.6 @@ -224,10 +230,6 @@ packages: chalk: 2.4.2 js-tokens: 4.0.0 - /@dqbd/tiktoken@1.0.7: - resolution: {integrity: sha512-bhR5k5W+8GLzysjk8zTMVygQZsgvf7W1F0IlL4ZQ5ugjo5rCyiwGM5d8DYriXspytfu98tv59niang3/T+FoDw==} - dev: false - /@es-joy/jsdoccomment@0.41.0: resolution: {integrity: sha512-aKUhyn1QI5Ksbqcr3fFJj16p99QdjUxXAEuFst1Z47DRyoiMwivIH9MV/ARcJOCXVjPfjITciej8ZD2O/6qUmw==} engines: {node: '>=16'} @@ -589,8 +591,8 @@ packages: '@types/range-parser': 1.2.4 dev: true - /@types/express@4.17.17: - resolution: {integrity: sha512-Q4FmmuLGBG58btUnfS1c1r/NQdlp3DMfGDGig8WhfpA2YRUtEkxAjkZb0yvplJGYdF1fsQ81iMDcH24sSCNC/Q==} + /@types/express@4.17.21: + resolution: {integrity: sha512-ejlPM315qwLpaQlQDTjPdsUFSc6ZsP4AN6AlWnogPjQ7CVi7PYF3YVz+CY3jE2pwYf7E/7HlDAN0rV2GxTG0HQ==} dependencies: '@types/body-parser': 1.19.2 '@types/express-serve-static-core': 4.17.33 @@ -626,6 +628,12 @@ packages: resolution: {integrity: sha512-Y4XFY5VJAuw0FgAqPNd6NNoV44jbq9Bz2L7Rh/J6jLTiHBSBJa9fxqQIvkIld4GsoDOcCbvzOUAbLPsSKKg+uA==} dev: true + /@types/multer@1.4.11: + resolution: {integrity: sha512-svK240gr6LVWvv3YGyhLlA+6LRRWA4mnGIU7RcNmgjBYFl6665wcXrRfxGp5tEPVHUNm5FMcmq7too9bxCwX/w==} + dependencies: + '@types/express': 4.17.21 + dev: true + /@types/node@18.14.6: resolution: {integrity: sha512-93+VvleD3mXwlLI/xASjw0FzKcwzl3OdTCzm1LaRfqgS21gfFtK3zDXM5Op9TeeMsJVOaJ2VRDpT9q4Y3d0AvA==} @@ -978,6 +986,10 @@ packages: color-convert: 2.0.1 dev: true + /append-field@1.0.0: + resolution: {integrity: sha512-klpgFSWLW1ZEs8svjfb7g4qWY0YS5imI82dTg+QahUvJ8YqAY0P10Uk8tTyh9ZGuYEZEMaeJYCF5BFuX552hsw==} + dev: false + /are-docs-informative@0.0.2: resolution: {integrity: sha512-ixiS0nLNNG5jNQzgZJNoUpBKdo9yTYZMGJ+QgT2jmjR7G7+QHRCc4v6LQ3NgE7EBJq+o0ams3waJwkrlBom8Ig==} engines: {node: '>=14'} @@ -1082,6 +1094,10 @@ packages: resolution: {integrity: sha512-zRpUiDwd/xk6ADqPMATG8vc9VPrkck7T07OIx0gnjmJAnHnTVXNQG3vfvWNuiZIkwu9KrKdA1iJKfsfTVxE6NA==} dev: false + /buffer-from@1.1.2: + resolution: {integrity: sha512-E+XQCRwSbaaiChtv6k6Dwgc+bx+Bs6vuKJHHl5kox/BaKbhiXzqQOwK4cO22yElGp2OCmjwVhT3HmxgyPGnJfQ==} + dev: false + /builtin-modules@3.3.0: resolution: {integrity: sha512-zhaCDicdLuWN5UbN5IMnFqNMhNfo919sH85y2/ea+5Yg9TsTkeZxpL+JLbp6cgYFS4sRLp3YV4S6yDuqVWHYOw==} engines: {node: '>=6'} @@ -1093,6 +1109,13 @@ packages: semver: 7.5.4 dev: true + /busboy@1.6.0: + resolution: {integrity: sha512-8SFQbg/0hQ9xy3UNTB0YEnsNBbWfhf7RtnzpL7TkBiTBRfrQ9Fxcnz7VJsleJpyp6rVLvXiuORqjlHi5q+PYuA==} + engines: {node: '>=10.16.0'} + dependencies: + streamsearch: 1.1.0 + dev: false + /bytes@3.1.2: resolution: {integrity: sha512-/Nf7TyzTx6S3yRJObOAV7956r8cr2+Oj8AC5dt8wSP3BQAoeX58NoHyCU8P8zGkNXStjTSi6fzO6F0pBdcYbEg==} engines: {node: '>= 0.8'} @@ -1143,22 +1166,6 @@ packages: resolution: {integrity: sha512-mKKUkUbhPpQlCOfIuZkvSEgktjPFIsZKRRbC6KWVEMvlzblj3i3asQv5ODsrwt0N3pHAEvjP8KTQPHkp0+6jOg==} dev: true - /chatgpt@5.2.4: - resolution: {integrity: sha512-U53cUJ/1sEUD1+lcdW52QqbVQZOIMy3hxzYdEIhmmgZJQW/zI8C5O9ohCxXbFsbk6OtXHEBxY/0xw4Qa6twG2Q==} - engines: {node: '>=14'} - hasBin: true - dependencies: - '@dqbd/tiktoken': 1.0.7 - cac: 6.7.14 - conf: 11.0.1 - eventsource-parser: 1.0.0 - keyv: 4.5.2 - p-timeout: 6.1.1 - quick-lru: 6.1.1 - read-pkg-up: 9.1.0 - uuid: 9.0.0 - dev: false - /ci-info@3.8.0: resolution: {integrity: sha512-eXTggHWSooYhq49F2opQhuHWgzucfF2YgODK4e1566GQs5BIfP30B0oenwBJHfWxAs2fyPB1s7Mg949zLf61Yw==} engines: {node: '>=8'} @@ -1206,6 +1213,16 @@ packages: resolution: {integrity: sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==} dev: true + /concat-stream@1.6.2: + resolution: {integrity: sha512-27HBghJxjiZtIk3Ycvn/4kbJk/1uZuJFfuPEns6LaEvpvG1f0hTea8lilrouyo9mVc2GWdcEZ8OLoGmSADlrCw==} + engines: {'0': node >= 0.8} + dependencies: + buffer-from: 1.1.2 + inherits: 2.0.4 + readable-stream: 2.3.8 + typedarray: 0.0.6 + dev: false + /conf@11.0.1: resolution: {integrity: sha512-WlLiQboEjKx0bYx2IIRGedBgNjLAxtwPaCSnsjWPST5xR0DB4q8lcsO/bEH9ZRYNcj63Y9vj/JG/5Fg6uWzI0Q==} engines: {node: '>=14.16'} @@ -1217,7 +1234,7 @@ packages: dot-prop: 7.2.0 env-paths: 3.0.0 json-schema-typed: 8.0.1 - semver: 7.5.4 + semver: 7.6.0 dev: false /content-disposition@0.5.4: @@ -1241,6 +1258,10 @@ packages: engines: {node: '>= 0.6'} dev: false + /core-util-is@1.0.3: + resolution: {integrity: sha512-ZQBvi1DcpJ4GDqanjucZ2Hj3wEO5pZDS89BWbkcrvdxksJorwUDDZamX9ldFkp9aw2lmBDLgkObEA4DWNJ9FYQ==} + dev: false + /cross-spawn@7.0.3: resolution: {integrity: sha512-iRDPJKUPVEND7dHPO8rkbOnPpyDygcDFtWjpeWNCgy8WP2rXcxXL8TskReQl6OrB2G7+UJrags1q15Fudc7G6w==} engines: {node: '>= 8'} @@ -1603,7 +1624,7 @@ packages: eslint: 8.56.0 esquery: 1.5.0 is-builtin-module: 3.2.1 - semver: 7.5.4 + semver: 7.6.0 spdx-expression-parse: 4.0.0 transitivePeerDependencies: - supports-color @@ -2364,6 +2385,10 @@ packages: engines: {node: '>=8'} dev: true + /isarray@1.0.0: + resolution: {integrity: sha512-VLghIWNM6ELQzo7zwmcg0NmTVyWKYjvIeM83yjp0wRDTmUnrM678fQbcKBo6n2CJEF0szoG//ytg+TKla89ALQ==} + dev: false + /isexe@2.0.0: resolution: {integrity: sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==} dev: true @@ -2638,11 +2663,22 @@ packages: brace-expansion: 2.0.1 dev: true + /minimist@1.2.8: + resolution: {integrity: sha512-2yyAR8qBkN3YuheJanUpWC5U3bb5osDywNB8RzDVlDwDHbocAJveqqj1u8+SVD7jkWT4yvsHCpWqqWqAxb0zCA==} + dev: false + /minipass@4.2.4: resolution: {integrity: sha512-lwycX3cBMTvcejsHITUgYj6Gy6A7Nh4Q6h9NP4sTHY1ccJlC7yKzDmiShEHsJ16Jf1nKGDEaiHxiltsJEvk0nQ==} engines: {node: '>=8'} dev: true + /mkdirp@0.5.6: + resolution: {integrity: sha512-FP+p8RB8OWpF3YZBCrP5gtADmtXApB5AMLn+vdyA+PyxCjrCs00mjyUozssO33cwDeT3wNGdLxJ5M//YqtHAJw==} + hasBin: true + dependencies: + minimist: 1.2.8 + dev: false + /mongodb-connection-string-url@2.6.0: resolution: {integrity: sha512-WvTZlI9ab0QYtTYnuMLgobULWhokRjtC7db9LtcVfJ+Hsnyr5eo6ZtNAt3Ly24XZScGMelOcGtm7lSn0332tPQ==} dependencies: @@ -2688,6 +2724,19 @@ packages: /ms@2.1.3: resolution: {integrity: sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==} + /multer@1.4.5-lts.1: + resolution: {integrity: sha512-ywPWvcDMeH+z9gQq5qYHCCy+ethsk4goepZ45GLD63fOu0YcNecQxi64nDs3qluZB+murG3/D4dJ7+dGctcCQQ==} + engines: {node: '>= 6.0.0'} + dependencies: + append-field: 1.0.0 + busboy: 1.6.0 + concat-stream: 1.6.2 + mkdirp: 0.5.6 + object-assign: 4.1.1 + type-is: 1.6.18 + xtend: 4.0.2 + dev: false + /natural-compare@1.4.0: resolution: {integrity: sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw==} dev: true @@ -2753,6 +2802,11 @@ packages: boolbase: 1.0.0 dev: true + /object-assign@4.1.1: + resolution: {integrity: sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==} + engines: {node: '>=0.10.0'} + dev: false + /object-inspect@1.12.3: resolution: {integrity: sha512-geUvdk7c+eizMNUDkRpW1wJwgfOiOeHbxBR/hLXK1aT6zmVSO0jsQcs7fj6MGw89jC/cjGfLcNOrtMYtGqm81g==} dev: false @@ -2930,6 +2984,10 @@ packages: engines: {node: '>= 0.8.0'} dev: true + /process-nextick-args@2.0.1: + resolution: {integrity: sha512-3ouUOpQhtgrbOa17J7+uxOTpITYWaGP7/AhoR3+A+/1e9skrzelGi/dXzEYyvbxubEF6Wn2ypscTKiKJFFn1ag==} + dev: false + /proxy-addr@2.0.7: resolution: {integrity: sha512-llQsMLSUDUPT44jdrU/O37qlnifitDP+ZwrmmZcoSKyLKvtZxpyV0n2/bD/N4tBAAZ/gJEdZU7KMraoK1+XYAg==} engines: {node: '>= 0.10'} @@ -3015,6 +3073,18 @@ packages: type-fest: 2.19.0 dev: false + /readable-stream@2.3.8: + resolution: {integrity: sha512-8p0AUk4XODgIewSi0l8Epjs+EVnWiK7NoDIEGU0HhE7+ZyY8D1IMY7odu5lRrFXGg71L15KG8QrPmum45RTtdA==} + dependencies: + core-util-is: 1.0.3 + inherits: 2.0.4 + isarray: 1.0.0 + process-nextick-args: 2.0.1 + safe-buffer: 5.1.2 + string_decoder: 1.1.1 + util-deprecate: 1.0.2 + dev: false + /regexp-tree@0.1.27: resolution: {integrity: sha512-iETxpjK6YoRWJG5o6hXLwvjYAoW+FEZn9os0PD/b6AP6xQwsa/Y7lCVgIixBbUPMfhu+i2LtdeAqVTgGlQarfA==} hasBin: true @@ -3079,6 +3149,10 @@ packages: queue-microtask: 1.2.3 dev: true + /safe-buffer@5.1.2: + resolution: {integrity: sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==} + dev: false + /safe-buffer@5.2.1: resolution: {integrity: sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==} dev: false @@ -3105,6 +3179,7 @@ packages: hasBin: true dependencies: lru-cache: 6.0.0 + dev: true /semver@7.6.0: resolution: {integrity: sha512-EnwXhrlwXMk9gKu5/flx5sv/an57AkRplG3hTK68W7FRDN+k+OWBj65M7719OkA82XLBxrcX0KSHj+X5COhOVg==} @@ -3244,6 +3319,17 @@ packages: engines: {node: '>= 0.8'} dev: false + /streamsearch@1.1.0: + resolution: {integrity: sha512-Mcc5wHehp9aXz1ax6bZUyY5afg9u2rv5cqQI3mRrYkGC8rW2hM02jWuwjtL++LS5qinSyhj2QfLyNsuc+VsExg==} + engines: {node: '>=10.0.0'} + dev: false + + /string_decoder@1.1.1: + resolution: {integrity: sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==} + dependencies: + safe-buffer: 5.1.2 + dev: false + /strip-ansi@6.0.1: resolution: {integrity: sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==} engines: {node: '>=8'} @@ -3381,6 +3467,10 @@ packages: mime-types: 2.1.35 dev: false + /typedarray@0.0.6: + resolution: {integrity: sha512-/aCDEGatGvZ2BIk+HmLf4ifCJFwvKFNb9/JeZPMulfgFracn9QFcAf5GO8B/mweUjSoblS5In0cWhqpfs/5PQA==} + dev: false + /typescript@5.3.3: resolution: {integrity: sha512-pXWcraxM0uxAS+tN0AG/BF2TyqmHO014Z070UsJ+pFvYuRSq8KH8DmWpnbXe0pEPDHXZV3FcAbJkijJ5oNEnWw==} engines: {node: '>=14.17'} @@ -3405,7 +3495,6 @@ packages: /util-deprecate@1.0.2: resolution: {integrity: sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==} - dev: true /utils-merge@1.0.1: resolution: {integrity: sha512-pMZTvIkT1d+TFGvDOqodOclx0QWkkgi6Tdoa8gC8ffGAAqz9pzPTZWAybbsHHoED/ztMtkv/VoYTYyShUn81hA==} @@ -3500,6 +3589,11 @@ packages: engines: {node: '>=12'} dev: true + /xtend@4.0.2: + resolution: {integrity: sha512-LKYU1iAXJXUgAXn9URjiu+MWhyUXHsvfp7mcuYm9dSUKK0/CjtrUwFAxD82/mCWbtLsGjFIad0wIsod4zrTAEQ==} + engines: {node: '>=0.4'} + dev: false + /yallist@4.0.0: resolution: {integrity: sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==} @@ -3526,3 +3620,23 @@ packages: resolution: {integrity: sha512-9bnSc/HEW2uRy67wc+T8UwauLuPJVn28jb+GtJY16iiKWyvmYJRXVT4UamsAEGQfPohgr2q4Tq0sQbQlxTfi1g==} engines: {node: '>=12.20'} dev: false + + git@github.com+chatgpt-web-dev/chatgpt-api/e8fca6bf34350881b1e9243610941bc4b00ce891: + resolution: {commit: e8fca6bf34350881b1e9243610941bc4b00ce891, repo: git@github.com:chatgpt-web-dev/chatgpt-api.git, type: git} + name: chatgpt + version: 5.2.5 + engines: {node: '>=14'} + hasBin: true + prepare: true + requiresBuild: true + dependencies: + cac: 6.7.14 + conf: 11.0.1 + eventsource-parser: 1.0.0 + js-tiktoken: 1.0.7 + keyv: 4.5.2 + p-timeout: 6.1.1 + quick-lru: 6.1.1 + read-pkg-up: 9.1.0 + uuid: 9.0.0 + dev: false diff --git a/service/src/chatgpt/index.ts b/service/src/chatgpt/index.ts index 99d5b350..51e3a078 100644 --- a/service/src/chatgpt/index.ts +++ b/service/src/chatgpt/index.ts @@ -1,3 +1,4 @@ +import * as console from 'node:console' import * as dotenv from 'dotenv' import 'isomorphic-fetch' import type { ChatGPTAPIOptions, ChatMessage, SendMessageOptions } from 'chatgpt' @@ -9,6 +10,7 @@ import jwt_decode from 'jwt-decode' import dayjs from 'dayjs' import type { AuditConfig, KeyConfig, UserInfo } from '../storage/model' import { Status } from '../storage/model' +import { convertImageUrl } from '../utils/image' import type { TextAuditService } from '../utils/textAudit' import { textAuditServices } from '../utils/textAudit' import { getCacheApiKeys, getCacheConfig, getOriginConfig } from '../storage/config' @@ -130,7 +132,31 @@ async function chatReplyProcess(options: RequestOptions) { throw new Error('无法在一个房间同时使用 AccessToken 以及 Api,请联系管理员,或新开聊天室进行对话 | Unable to use AccessToken and Api at the same time in the same room, please contact the administrator or open a new chat room for conversation') } - const { message, lastContext, process, systemMessage, temperature, top_p } = options + const { message, uploadFileKeys, lastContext, process, systemMessage, temperature, top_p } = options + + let content: string | { + type: string + text?: string + image_url?: { + url: string + } + }[] = message + if (uploadFileKeys && uploadFileKeys.length > 0) { + content = [ + { + type: 'text', + text: message, + }, + ] + for (const uploadFileKey of uploadFileKeys) { + content.push({ + type: 'image_url', + image_url: { + url: await convertImageUrl(uploadFileKey), + }, + }) + } + } try { const timeoutMs = (await getCacheConfig()).timeoutMs @@ -153,7 +179,7 @@ async function chatReplyProcess(options: RequestOptions) { const abort = new AbortController() options.abortSignal = abort.signal processThreads.push({ userId, abort, messageId }) - const response = await api.sendMessage(message, { + const response = await api.sendMessage(content, { ...options, onProgress: (partialResponse) => { process?.(partialResponse) @@ -370,12 +396,35 @@ async function getMessageById(id: string): Promise { } else { if (isPrompt) { // prompt + let content: string | { + type: string + text?: string + image_url?: { + url: string + } + }[] = chatInfo.prompt + if (chatInfo.images) { + content = [ + { + type: 'text', + text: chatInfo.prompt, + }, + ] + for (const image of chatInfo.images) { + content.push({ + type: 'image_url', + image_url: { + url: await convertImageUrl(image), + }, + }) + } + } return { id, conversationId: chatInfo.options.conversationId, parentMessageId, role: 'user', - text: chatInfo.prompt, + text: content, } } else { diff --git a/service/src/chatgpt/types.ts b/service/src/chatgpt/types.ts index 348b98d4..a8d9db5d 100644 --- a/service/src/chatgpt/types.ts +++ b/service/src/chatgpt/types.ts @@ -3,6 +3,7 @@ import type { ChatRoom, UserInfo } from 'src/storage/model' export interface RequestOptions { message: string + uploadFileKeys?: string[] lastContext?: { conversationId?: string; parentMessageId?: string } process?: (chat: ChatMessage) => void systemMessage?: string diff --git a/service/src/index.ts b/service/src/index.ts index 4fc48869..a0379174 100644 --- a/service/src/index.ts +++ b/service/src/index.ts @@ -58,6 +58,7 @@ import { hasAnyRole, isEmail, isNotEmptyString } from './utils/is' import { sendNoticeMail, sendResetPasswordMail, sendTestMail, sendVerifyMail, sendVerifyMailAdmin } from './utils/mail' import { checkUserResetPassword, checkUserVerify, checkUserVerifyAdmin, getUserResetPasswordUrl, getUserVerifyUrl, getUserVerifyUrlAdmin, md5 } from './utils/security' import { isAdmin, rootAuth } from './middleware/rootAuth' +import { router as uploadRouter } from './routes/upload' dotenv.config() @@ -67,6 +68,8 @@ const router = express.Router() app.use(express.static('public')) app.use(express.json()) +app.use('/uploads', express.static('uploads')) + app.all('*', (_, res, next) => { res.header('Access-Control-Allow-Origin', '*') res.header('Access-Control-Allow-Headers', 'authorization, Content-Type') @@ -212,6 +215,7 @@ router.get('/chat-history', auth, async (req, res) => { uuid: c.uuid, dateTime: new Date(c.dateTime).toLocaleString(), text: c.prompt, + images: c.images, inversion: true, error: false, conversationOptions: null, @@ -371,7 +375,7 @@ router.post('/chat-clear', auth, async (req, res) => { router.post('/chat-process', [auth, limiter], async (req, res) => { res.setHeader('Content-type', 'application/octet-stream') - let { roomId, uuid, regenerate, prompt, options = {}, systemMessage, temperature, top_p } = req.body as RequestProps + let { roomId, uuid, regenerate, prompt, uploadFileKeys, options = {}, systemMessage, temperature, top_p } = req.body as RequestProps const userId = req.headers.userId.toString() const config = await getCacheConfig() const room = await getChatRoom(userId, roomId) @@ -408,10 +412,11 @@ router.post('/chat-process', [auth, limiter], async (req, res) => { message = regenerate ? await getChat(roomId, uuid) - : await insertChat(uuid, prompt, roomId, options as ChatOptions) + : await insertChat(uuid, prompt, uploadFileKeys, roomId, options as ChatOptions) let firstChunk = true result = await chatReplyProcess({ message: prompt, + uploadFileKeys, lastContext: options, process: (chat: ChatMessage) => { lastResponse = chat @@ -1300,6 +1305,8 @@ router.post('/statistics/by-day', auth, async (req, res) => { } }) +app.use('', uploadRouter) + app.use('', router) app.use('/api', router) diff --git a/service/src/routes/upload.ts b/service/src/routes/upload.ts new file mode 100644 index 00000000..a9c925de --- /dev/null +++ b/service/src/routes/upload.ts @@ -0,0 +1,31 @@ +import Router from 'express' +import multer from 'multer' +import { auth } from '../middleware/auth' + +export const router = Router() + +// 配置multer的存储选项 +const storage = multer.diskStorage({ + destination(req, file, cb) { + cb(null, 'uploads/') // 确保这个文件夹存在 + }, + filename(req, file, cb) { + cb(null, `${file.fieldname}-${Date.now()}`) + }, +}) + +const upload = multer({ storage }) +router.post('/upload-image', auth, upload.single('file'), async (req, res) => { + try { + if (!req.file) + res.send({ status: 'Fail', message: '没有文件被上传', data: null }) + const data = { + fileKey: req.file.filename, + } + // 文件已上传 + res.send({ status: 'Success', message: '文件上传成功', data }) + } + catch (error) { + res.send(error) + } +}) diff --git a/service/src/storage/config.ts b/service/src/storage/config.ts index bde58938..1755604c 100644 --- a/service/src/storage/config.ts +++ b/service/src/storage/config.ts @@ -107,7 +107,7 @@ export async function getOriginConfig() { } if (!isNotEmptyString(config.siteConfig.chatModels)) - config.siteConfig.chatModels = 'gpt-3.5-turbo,gpt-3.5-turbo-1106,gpt-3.5-turbo-16k,gpt-3.5-turbo-16k-0613,gpt-4,gpt-4-0613,gpt-4-32k,gpt-4-32k-0613,text-davinci-002-render-sha-mobile,text-embedding-ada-002,gpt-4-mobile,gpt-4-browsing,gpt-4-1106-preview,gpt-4-vision-preview' + config.siteConfig.chatModels = 'gpt-3.5-turbo,gpt-4-turbo-preview,gpt-4-vision-preview' return config } diff --git a/service/src/storage/model.ts b/service/src/storage/model.ts index a5ae5f74..8ebe42d0 100644 --- a/service/src/storage/model.ts +++ b/service/src/storage/model.ts @@ -121,14 +121,16 @@ export class ChatInfo { uuid: number dateTime: number prompt: string + images?: string[] response?: string status: Status = Status.Normal options: ChatOptions previousResponse?: previousResponse[] - constructor(roomId: number, uuid: number, prompt: string, options: ChatOptions) { + constructor(roomId: number, uuid: number, prompt: string, images: string[], options: ChatOptions) { this.roomId = roomId this.uuid = uuid this.prompt = prompt + this.images = images this.options = options this.dateTime = new Date().getTime() } diff --git a/service/src/storage/mongo.ts b/service/src/storage/mongo.ts index afc0a883..4695223d 100644 --- a/service/src/storage/mongo.ts +++ b/service/src/storage/mongo.ts @@ -67,8 +67,8 @@ export async function updateAmountMinusOne(userId: string) { return result.modifiedCount > 0 } -export async function insertChat(uuid: number, text: string, roomId: number, options?: ChatOptions) { - const chatInfo = new ChatInfo(roomId, uuid, text, options) +export async function insertChat(uuid: number, text: string, images: string[], roomId: number, options?: ChatOptions) { + const chatInfo = new ChatInfo(roomId, uuid, text, images, options) await chatCol.insertOne(chatInfo) return chatInfo } @@ -197,7 +197,7 @@ export async function deleteAllChatRooms(userId: string) { await chatCol.updateMany({ userId, status: Status.Normal }, { $set: { status: Status.Deleted } }) } -export async function getChats(roomId: number, lastId?: number) { +export async function getChats(roomId: number, lastId?: number): Promise { if (!lastId) lastId = new Date().getTime() const query = { roomId, uuid: { $lt: lastId }, status: { $ne: Status.Deleted } } diff --git a/service/src/types.ts b/service/src/types.ts index 36ecb04b..1177330e 100644 --- a/service/src/types.ts +++ b/service/src/types.ts @@ -6,6 +6,7 @@ export interface RequestProps { uuid: number regenerate: boolean prompt: string + uploadFileKeys?: string[] options?: ChatContext systemMessage: string temperature?: number diff --git a/service/src/utils/image.ts b/service/src/utils/image.ts new file mode 100644 index 00000000..4cdef649 --- /dev/null +++ b/service/src/utils/image.ts @@ -0,0 +1,18 @@ +import fs from 'node:fs/promises' + +fs.mkdir('uploads').then(() => { + globalThis.console.log('Directory uploads created') +}).catch((e) => { + if (e.code === 'EEXIST') { + globalThis.console.log('Directory uploads already exists') + return + } + globalThis.console.error('Error creating directory uploads, ', e) +}) + +export async function convertImageUrl(uploadFileKey: string): Promise { + const imageData = await fs.readFile(`uploads/${uploadFileKey}`) + // 将图片数据转换为 Base64 编码的字符串 + const base64Image = imageData.toString('base64') + return `data:image/png;base64,${base64Image}` +} diff --git a/src/api/index.ts b/src/api/index.ts index ecf2942a..4c26eba5 100644 --- a/src/api/index.ts +++ b/src/api/index.ts @@ -16,6 +16,7 @@ export function fetchChatAPIProcess( uuid: number regenerate?: boolean prompt: string + uploadFileKeys?: string[] options?: { conversationId?: string; parentMessageId?: string } signal?: GenericAbortSignal onDownloadProgress?: (progressEvent: AxiosProgressEvent) => void }, @@ -28,6 +29,7 @@ export function fetchChatAPIProcess( uuid: params.uuid, regenerate: params.regenerate || false, prompt: params.prompt, + uploadFileKeys: params.uploadFileKeys, options: params.options, } diff --git a/src/typings/chat.d.ts b/src/typings/chat.d.ts index 92aea744..a890728d 100644 --- a/src/typings/chat.d.ts +++ b/src/typings/chat.d.ts @@ -3,6 +3,7 @@ declare namespace Chat { uuid?: number dateTime: string text: string + images?: string[] inversion?: boolean responseCount?: number error?: boolean diff --git a/src/views/chat/components/Message/Text.vue b/src/views/chat/components/Message/Text.vue index 25c3b7ad..a0cf6fe8 100644 --- a/src/views/chat/components/Message/Text.vue +++ b/src/views/chat/components/Message/Text.vue @@ -12,6 +12,7 @@ interface Props { inversion?: boolean error?: boolean text?: string + images?: string[] loading?: boolean asRawText?: boolean } @@ -108,6 +109,7 @@ onUnmounted(() => {
+
diff --git a/src/views/chat/components/Message/index.vue b/src/views/chat/components/Message/index.vue index 3ce276e8..7f9bb01c 100644 --- a/src/views/chat/components/Message/index.vue +++ b/src/views/chat/components/Message/index.vue @@ -12,6 +12,7 @@ import { copyToClip } from '@/utils/copy' interface Props { dateTime?: string text?: string + images?: string[] inversion?: boolean error?: boolean loading?: boolean @@ -179,6 +180,7 @@ async function handlePreviousResponse(next: number) { :inversion="inversion" :error="error" :text="text" + :images="images" :loading="loading" :as-raw-text="asRawText" /> diff --git a/src/views/chat/index.vue b/src/views/chat/index.vue index bfaa6975..132284e8 100644 --- a/src/views/chat/index.vue +++ b/src/views/chat/index.vue @@ -4,7 +4,7 @@ import { computed, defineAsyncComponent, nextTick, onMounted, onUnmounted, ref, import { useRoute } from 'vue-router' import { storeToRefs } from 'pinia' import type { MessageReactive } from 'naive-ui' -import { NAutoComplete, NButton, NInput, NSelect, NSlider, NSpace, NSpin, useDialog, useMessage } from 'naive-ui' +import { NAutoComplete, NButton, NInput, NSelect, NSlider, NSpace, NSpin, NUpload, UploadFileInfo, useDialog, useMessage } from 'naive-ui' import html2canvas from 'html2canvas' import { Message } from './components' import { useScroll } from './hooks/useScroll' @@ -54,6 +54,8 @@ const showPrompt = ref(false) const nowSelectChatModel = ref(null) const currentChatModel = computed(() => nowSelectChatModel.value ?? currentChatHistory.value?.chatModel ?? userStore.userInfo.config.chatModel) +const isVisionModel = computed(() => currentChatModel.value && currentChatModel.value?.includes('vision')) + let loadingms: MessageReactive let allmsg: MessageReactive let prevScrollTop: number @@ -74,6 +76,8 @@ function handleSubmit() { onConversation() } +const uploadFileKeysRef = ref([]) + async function onConversation() { let message = prompt.value @@ -86,6 +90,9 @@ async function onConversation() { if (nowSelectChatModel.value && currentChatHistory.value) currentChatHistory.value.chatModel = nowSelectChatModel.value + const uploadFileKeys = isVisionModel.value ? uploadFileKeysRef.value : [] + uploadFileKeysRef.value = [] + controller = new AbortController() const chatUuid = Date.now() @@ -95,6 +102,7 @@ async function onConversation() { uuid: chatUuid, dateTime: new Date().toLocaleString(), text: message, + images: uploadFileKeys, inversion: true, error: false, conversationOptions: null, @@ -134,6 +142,7 @@ async function onConversation() { roomId: +uuid, uuid: chatUuid, prompt: message, + uploadFileKeys, options, signal: controller.signal, onDownloadProgress: ({ event }) => { @@ -592,6 +601,25 @@ function formatTooltip(value: number) { return `${t('setting.maxContextCount')}: ${value}` } +// https://github.com/tusen-ai/naive-ui/issues/4887 +function handleFinish(options: { file: UploadFileInfo; event?: ProgressEvent }) { + if (options.file.status === 'finished') { + const response = (options.event?.target as XMLHttpRequest).response + uploadFileKeysRef.value.push(`${response.data.fileKey}`) + } +} + +function handleDeleteUploadFile() { + uploadFileKeysRef.value.pop() +} + +const uploadHeaders = computed(() => { + const token = useAuthStore().token + return { + Authorization: `Bearer ${token}`, + } +}) + onMounted(() => { firstLoading.value = true handleSyncChat() @@ -643,6 +671,7 @@ onUnmounted(() => { :key="index" :date-time="item.dateTime" :text="item.text" + :images="item.images" :inversion="item.inversion" :response-count="item.responseCount" :usage="item && item.usage || undefined" @@ -669,7 +698,36 @@ onUnmounted(() => {