{"id":26137,"date":"2024-09-18T11:54:52","date_gmt":"2024-09-18T03:54:52","guid":{"rendered":"https:\/\/17aitech.com\/?p=26137"},"modified":"2024-10-08T15:06:08","modified_gmt":"2024-10-08T07:06:08","slug":"%e3%80%90%e9%a1%b9%e7%9b%ae%e5%ae%9e%e6%88%98%e3%80%91%e5%9f%ba%e4%ba%8eagent%e7%9a%84%e9%87%91%e8%9e%8d%e9%97%ae%e7%ad%94%e7%b3%bb%e7%bb%9f%ef%bc%9a%e5%89%8d%e5%90%8e%e7%ab%af%e6%b5%81%e7%a8%8b","status":"publish","type":"post","link":"https:\/\/17aitech.com\/?p=26137","title":{"rendered":"\u3010\u9879\u76ee\u5b9e\u6218\u3011\u57fa\u4e8eAgent\u7684\u91d1\u878d\u95ee\u7b54\u7cfb\u7edf\uff1a\u524d\u540e\u7aef\u6d41\u7a0b\u6253\u901a"},"content":{"rendered":"<div id=\"ez-toc-container\" class=\"ez-toc-v2_0_78 ez-toc-wrap-left-text counter-hierarchy ez-toc-counter ez-toc-light-blue ez-toc-container-direction\">\n<div class=\"ez-toc-title-container\">\n<p class=\"ez-toc-title\" style=\"cursor:inherit\">\u6587\u7ae0\u76ee\u5f55<\/p>\n<span class=\"ez-toc-title-toggle\"><a href=\"#\" class=\"ez-toc-pull-right ez-toc-btn ez-toc-btn-xs ez-toc-btn-default ez-toc-toggle\" aria-label=\"Toggle Table of Content\"><span class=\"ez-toc-js-icon-con\"><span class=\"\"><span class=\"eztoc-hide\" style=\"display:none;\">Toggle<\/span><span class=\"ez-toc-icon-toggle-span\"><svg style=\"fill: #999;color:#999\" xmlns=\"http:\/\/www.w3.org\/2000\/svg\" class=\"list-377408\" width=\"20px\" height=\"20px\" viewBox=\"0 0 24 24\" fill=\"none\"><path d=\"M6 6H4v2h2V6zm14 0H8v2h12V6zM4 11h2v2H4v-2zm16 0H8v2h12v-2zM4 16h2v2H4v-2zm16 0H8v2h12v-2z\" fill=\"currentColor\"><\/path><\/svg><svg style=\"fill: #999;color:#999\" class=\"arrow-unsorted-368013\" xmlns=\"http:\/\/www.w3.org\/2000\/svg\" width=\"10px\" height=\"10px\" viewBox=\"0 0 24 24\" version=\"1.2\" baseProfile=\"tiny\"><path d=\"M18.2 9.3l-6.2-6.3-6.2 6.3c-.2.2-.3.4-.3.7s.1.5.3.7c.2.2.4.3.7.3h11c.3 0 .5-.1.7-.3.2-.2.3-.5.3-.7s-.1-.5-.3-.7zM5.8 14.7l6.2 6.3 6.2-6.3c.2-.2.3-.5.3-.7s-.1-.5-.3-.7c-.2-.2-.4-.3-.7-.3h-11c-.3 0-.5.1-.7.3-.2.2-.3.5-.3.7s.1.5.3.7z\"\/><\/svg><\/span><\/span><\/span><\/a><\/span><\/div>\n<nav><ul class='ez-toc-list ez-toc-list-level-1 ' ><li class='ez-toc-page-1 ez-toc-heading-level-2'><a class=\"ez-toc-link ez-toc-heading-1\" href=\"https:\/\/17aitech.com\/?p=26137\/#%E5%89%8D%E8%A8%80\" >\u524d\u8a00<\/a><\/li><li class='ez-toc-page-1 ez-toc-heading-level-2'><a class=\"ez-toc-link ez-toc-heading-2\" href=\"https:\/\/17aitech.com\/?p=26137\/#%E7%9B%AE%E6%A0%87\" >\u76ee\u6807<\/a><\/li><li class='ez-toc-page-1 ez-toc-heading-level-2'><a class=\"ez-toc-link ez-toc-heading-3\" href=\"https:\/\/17aitech.com\/?p=26137\/#%E5%90%8E%E7%AB%AF%E5%AE%9E%E7%8E%B0\" >\u540e\u7aef\u5b9e\u73b0<\/a><ul class='ez-toc-list-level-3' ><li class='ez-toc-heading-level-3'><a class=\"ez-toc-link ez-toc-heading-4\" href=\"https:\/\/17aitech.com\/?p=26137\/#%E4%BB%A3%E7%A0%81%E5%AE%9E%E7%8E%B0\" >\u4ee3\u7801\u5b9e\u73b0<\/a><\/li><li class='ez-toc-page-1 ez-toc-heading-level-3'><a class=\"ez-toc-link ez-toc-heading-5\" href=\"https:\/\/17aitech.com\/?p=26137\/#%E6%9C%8D%E5%8A%A1%E9%83%A8%E7%BD%B2\" >\u670d\u52a1\u90e8\u7f72<\/a><\/li><li class='ez-toc-page-1 ez-toc-heading-level-3'><a class=\"ez-toc-link ez-toc-heading-6\" href=\"https:\/\/17aitech.com\/?p=26137\/#%E6%8E%A5%E5%8F%A3%E6%B5%8B%E8%AF%95\" >\u63a5\u53e3\u6d4b\u8bd5<\/a><\/li><\/ul><\/li><li class='ez-toc-page-1 ez-toc-heading-level-2'><a class=\"ez-toc-link ez-toc-heading-7\" href=\"https:\/\/17aitech.com\/?p=26137\/#%E5%89%8D%E7%AB%AF%E5%AE%9E%E7%8E%B0\" >\u524d\u7aef\u5b9e\u73b0<\/a><ul class='ez-toc-list-level-3' ><li class='ez-toc-heading-level-3'><a class=\"ez-toc-link ez-toc-heading-8\" href=\"https:\/\/17aitech.com\/?p=26137\/#%E8%83%8C%E6%99%AF%E7%9F%A5%E8%AF%86\" >\u80cc\u666f\u77e5\u8bc6<\/a><\/li><li class='ez-toc-page-1 ez-toc-heading-level-3'><a class=\"ez-toc-link ez-toc-heading-9\" href=\"https:\/\/17aitech.com\/?p=26137\/#%E4%BB%A3%E7%A0%81%E5%AE%9E%E7%8E%B0-2\" >\u4ee3\u7801\u5b9e\u73b0<\/a><ul class='ez-toc-list-level-4' ><li class='ez-toc-heading-level-4'><a class=\"ez-toc-link ez-toc-heading-10\" href=\"https:\/\/17aitech.com\/?p=26137\/#%E4%BF%AE%E6%94%B9indexhtml\" >\u4fee\u6539index.html<\/a><\/li><li class='ez-toc-page-1 ez-toc-heading-level-4'><a class=\"ez-toc-link ez-toc-heading-11\" href=\"https:\/\/17aitech.com\/?p=26137\/#%E4%BF%AE%E6%94%B9%E8%B7%AF%E7%94%B1%E9%85%8D%E7%BD%AE%E6%96%87%E4%BB%B6\" >\u4fee\u6539\u8def\u7531\u914d\u7f6e\u6587\u4ef6<\/a><\/li><li class='ez-toc-page-1 ez-toc-heading-level-4'><a class=\"ez-toc-link ez-toc-heading-12\" href=\"https:\/\/17aitech.com\/?p=26137\/#%E4%BF%AE%E6%94%B9%E8%81%8A%E5%A4%A9%E7%BB%84%E4%BB%B6%E9%A1%B5%E9%9D%A2\" >\u4fee\u6539\u804a\u5929\u7ec4\u4ef6\u9875\u9762<\/a><ul class='ez-toc-list-level-5' ><li class='ez-toc-heading-level-5'><a class=\"ez-toc-link ez-toc-heading-13\" href=\"https:\/\/17aitech.com\/?p=26137\/#%E9%87%8D%E6%9E%84%E9%A1%B5%E9%9D%A2%E7%9A%84%E6%98%BE%E7%A4%BA%E5%B8%83%E5%B1%80\" >\u91cd\u6784\u9875\u9762\u7684\u663e\u793a\u5e03\u5c40<\/a><\/li><li class='ez-toc-page-1 ez-toc-heading-level-5'><a class=\"ez-toc-link ez-toc-heading-14\" href=\"https:\/\/17aitech.com\/?p=26137\/#%E9%87%8D%E6%9E%84%E9%A1%B5%E9%9D%A2%E7%9A%84%E6%98%BE%E7%A4%BA%E6%A0%B7%E5%BC%8F\" >\u91cd\u6784\u9875\u9762\u7684\u663e\u793a\u6837\u5f0f<\/a><\/li><li class='ez-toc-page-1 ez-toc-heading-level-5'><a class=\"ez-toc-link ez-toc-heading-15\" href=\"https:\/\/17aitech.com\/?p=26137\/#%E9%87%8D%E6%9E%84%E9%A1%B5%E9%9D%A2%E7%9A%84JS%E4%BB%A3%E7%A0%81\" >\u91cd\u6784\u9875\u9762\u7684JS\u4ee3\u7801<\/a><\/li><\/ul><\/li><\/ul><\/li><li class='ez-toc-page-1 ez-toc-heading-level-3'><a class=\"ez-toc-link ez-toc-heading-16\" href=\"https:\/\/17aitech.com\/?p=26137\/#%E6%9C%8D%E5%8A%A1%E9%83%A8%E7%BD%B2-2\" >\u670d\u52a1\u90e8\u7f72<\/a><\/li><li class='ez-toc-page-1 ez-toc-heading-level-3'><a class=\"ez-toc-link ez-toc-heading-17\" href=\"https:\/\/17aitech.com\/?p=26137\/#%E6%8E%A5%E5%8F%A3%E8%81%94%E8%B0%83\" >\u63a5\u53e3\u8054\u8c03<\/a><\/li><\/ul><\/li><li class='ez-toc-page-1 ez-toc-heading-level-2'><a class=\"ez-toc-link ez-toc-heading-18\" href=\"https:\/\/17aitech.com\/?p=26137\/#%E5%86%85%E5%AE%B9%E5%B0%8F%E7%BB%93\" >\u5185\u5bb9\u5c0f\u7ed3<\/a><\/li><li class='ez-toc-page-1 ez-toc-heading-level-2'><a class=\"ez-toc-link ez-toc-heading-19\" href=\"https:\/\/17aitech.com\/?p=26137\/#%E9%99%84%E5%BD%95\" >\u9644\u5f55<\/a><\/li><\/ul><\/nav><\/div>\n<h2><span class=\"ez-toc-section\" id=\"%E5%89%8D%E8%A8%80\"><\/span>\u524d\u8a00<span class=\"ez-toc-section-end\"><\/span><\/h2>\n<p>\u5728\u4e0a\u4e00\u7ae0<a href=\"https:\/\/17aitech.com\/?p=25994\">\u300a\u3010\u9879\u76ee\u5b9e\u6218\u3011\u57fa\u4e8eAgent\u7684\u91d1\u878d\u95ee\u7b54\u7cfb\u7edf\uff1aAgent\u6846\u67b6\u7684\u6784\u5efa\u300b<\/a>\u4e2d\uff0c\u6211\u4eec\u5df2\u7ecf\u5b8c\u6210\u4e86\u57fa\u4e8eagent\u7684\u95ee\u7b54\u7cfb\u7edf\u4e3b\u6d41\u7a0b\u6784\u5efa\uff0c\u672c\u7ae0\u5c06\u4ecb\u7ecd\u5982\u4f55\u5c06Agent\u901a\u8fc7langserve\u90e8\u7f72\u4e3a\u540e\u7aef\u670d\u52a1\uff0c\u540c\u65f6\u57fa\u4e8eVue.js+Vite\u6784\u5efa\u524d\u7aef\u9875\u9762\uff0c\u5b8c\u6210\u95ee\u7b54\u7cfb\u7edf\u524d\u540e\u7aef\u8054\u8c03\u3002<\/p>\n<h2><span class=\"ez-toc-section\" id=\"%E7%9B%AE%E6%A0%87\"><\/span>\u76ee\u6807<span class=\"ez-toc-section-end\"><\/span><\/h2>\n<ul>\n<li>\u5c06Agent\u90e8\u7f72\u4e3a\u540e\u7aef\u670d\u52a1<\/li>\n<li>\u6784\u5efa\u524d\u7aef\u9875\u9762\uff0c\u5b8c\u6210\u95ee\u7b54\u7cfb\u7edf\u524d\u540e\u7aef\u8054\u8c03<\/li>\n<\/ul>\n<h2><span class=\"ez-toc-section\" id=\"%E5%90%8E%E7%AB%AF%E5%AE%9E%E7%8E%B0\"><\/span>\u540e\u7aef\u5b9e\u73b0<span class=\"ez-toc-section-end\"><\/span><\/h2>\n<h3><span class=\"ez-toc-section\" id=\"%E4%BB%A3%E7%A0%81%E5%AE%9E%E7%8E%B0\"><\/span>\u4ee3\u7801\u5b9e\u73b0<span class=\"ez-toc-section-end\"><\/span><\/h3>\n<p>\u57fa\u4e8e<a href=\"https:\/\/17aitech.com\/?p=14624#toc-13\">\u300a\u5927\u6a21\u578b\u5e94\u7528\u5f00\u53d1\u4e4bPrompt\u521d\u6b65\u4e86\u89e3\u300b<\/a>\u4e2dlangserve\u7684\u6240\u5b66\u5185\u5bb9\uff0c\u6211\u4eec\u9700\u8981\u505a\u7684\u7b2c\u4e00\u6b65\u4e3a\uff1a<br \/>\n1\u3001\u5b89\u88c5langserve\u4f9d\u8d56\u5305\uff1a<\/p>\n<pre><code class=\"language-bash\">pip install &quot;langserve[all]&quot;<\/code><\/pre>\n<p>2\u3001\u521b\u5efa\u4e00\u4e2aserver.py\u6587\u4ef6\uff0c\u5177\u4f53\u5982\u4e0b\uff1b<\/p>\n<p>\u4ee3\u7801\u6587\u4ef6\uff1a<code>app\/server.py<\/code><\/p>\n<pre><code class=\"language-python\">from fastapi import FastAPI, HTTPException\nfrom fastapi.middleware.cors import CORSMiddleware\nfrom finance_bot_ex import FinanceBotEx\n\nfinance_bot_ex = FinanceBotEx()\n\n# \u521b\u5efa FastAPI \u5e94\u7528\napp = FastAPI(\n    title=&quot;Qwen API&quot;,\n    version=&quot;0.1&quot;,\n    description=&quot;Qwen API&quot;,\n)\n\n# \u6dfb\u52a0 CORS \u4e2d\u95f4\u4ef6\napp.add_middleware(\n    CORSMiddleware,\n    allow_origins=[&quot;*&quot;],  # \u5141\u8bb8\u6240\u6709\u7684\u6765\u6e90\n    allow_credentials=True,\n    allow_methods=[&quot;*&quot;],  # \u5141\u8bb8\u7684HTTP\u65b9\u6cd5\n    allow_headers=[&quot;*&quot;],  # \u5141\u8bb8\u7684\u8bf7\u6c42\u5934\n)\n\n@app.post(&quot;\/queryex&quot;, response_model=dict)\nasync def query(query: dict):  # \u4f7f\u7528\u5b57\u5178\u7c7b\u578b\u4ee3\u66ffQuery\u6a21\u578b\n    try:\n        # \u4ece\u5b57\u5178\u4e2d\u83b7\u53d6input\n        input_data = query.get(&quot;input&quot;)\n        result = finance_bot_ex.handle_query(input_data)\n\n        # \u8fd4\u56de\u5b57\u5178\u683c\u5f0f\u7684\u54cd\u5e94\n        return {&quot;output&quot;: result}\n    except Exception as e:\n        raise HTTPException(status_code=500, detail=str(e))\n\n# \u8fd0\u884cUvicorn\u670d\u52a1\u5668\nif __name__ == &quot;__main__&quot;:\n    import uvicorn\n\n    uvicorn.run(app, host=&quot;0.0.0.0&quot;, port=8082)\n<\/code><\/pre>\n<p>\u4ee3\u7801\u8bf4\u660e\uff1a<\/p>\n<ul>\n<li>\n<p><strong>\u9996\u5148<\/strong>\uff0c\u4f9d\u8d56\u5305\u5f15\u5165langserve\u7684\u5fc5\u8981\u5305\uff1a<\/p>\n<pre><code class=\"language-python\">from fastapi import FastAPI, HTTPException\nfrom fastapi.middleware.cors import CORSMiddleware<\/code><\/pre>\n<\/li>\n<li>\n<p><strong>\u5176\u6b21<\/strong>\uff0c\u5c06\u6211\u4eec\u4e4b\u524d\u5c01\u88c5\u597d\u7684FinanceBotEx\u7c7b\u5f15\u5165\u8fdb\u6765\uff1a<\/p>\n<pre><code class=\"language-python\">from finance_bot_ex import FinanceBotEx<\/code><\/pre>\n<\/li>\n<li>\n<p><strong>\u7136\u540e<\/strong>\uff0c\u521d\u59cb\u5316FinanceBotEx\u7c7b\uff0c\u5e76\u4e14\u521b\u5efaFastAPI\u5e94\u7528\uff1a<\/p>\n<pre><code class=\"language-python\"># \u5b9e\u4f8b\u5316FinanceBotEx\u7c7b\nfinance_bot_ex = FinanceBotEx()\n# \u521b\u5efa FastAPI \u5e94\u7528\napp = FastAPI(\ntitle=&quot;Qwen API&quot;,\nversion=&quot;0.1&quot;,\ndescription=&quot;Qwen API&quot;,\n)\n# \u6dfb\u52a0 CORS \u4e2d\u95f4\u4ef6(\u5141\u8bb8\u8de8\u57df\u8bbf\u95ee)\napp.add_middleware(\nCORSMiddleware,\nallow_origins=[&quot;*&quot;],  # \u5141\u8bb8\u6240\u6709\u7684\u6765\u6e90\nallow_credentials=True,\nallow_methods=[&quot;*&quot;],  # \u5141\u8bb8\u7684HTTP\u65b9\u6cd5\nallow_headers=[&quot;*&quot;],  # \u5141\u8bb8\u7684\u8bf7\u6c42\u5934\n)<\/code><\/pre>\n<\/li>\n<li>\n<p><strong>\u6700\u540e<\/strong>\uff0c\u7ed9\u5b9a\u4e00\u4e2a\u63a5\u53e3\uff0c\u63a5\u6536\u7528\u6237\u8f93\u5165\uff0c\u8c03\u7528FinanceBotEx\u7c7b\u7684handle_query\u65b9\u6cd5\uff0c\u8fd4\u56de\u7ed3\u679c\uff1a<\/p>\n<pre><code class=\"language-python\">@app.post(&quot;\/queryex&quot;, response_model=dict)\nasync def query(query: dict):  # \u4f7f\u7528\u5b57\u5178\u7c7b\u578b\u4ee3\u66ffQuery\u6a21\u578b\ntry:\n    # \u4ece\u5b57\u5178\u4e2d\u83b7\u53d6input\n    input_data = query.get(&quot;input&quot;)\n    result = finance_bot_ex.handle_query(input_data)\n\n    # \u8fd4\u56de\u5b57\u5178\u683c\u5f0f\u7684\u54cd\u5e94\n    return {&quot;output&quot;: result}\nexcept Exception as e:\n    raise HTTPException(status_code=500, detail=str(e))<\/code><\/pre>\n<\/li>\n<\/ul>\n<h3><span class=\"ez-toc-section\" id=\"%E6%9C%8D%E5%8A%A1%E9%83%A8%E7%BD%B2\"><\/span>\u670d\u52a1\u90e8\u7f72<span class=\"ez-toc-section-end\"><\/span><\/h3>\n<p>\u76ee\u524d\uff0c\u6574\u4e2a\u5de5\u7a0b\u76ee\u5f55\u7ed3\u6784\u5982\u4e0b\uff1a<\/p>\n<pre><code class=\"language-bash\">smart-finance-bot \\\n    |- app \\   # \u8be5\u76ee\u5f55\u7528\u4e8e\u670d\u52a1\u7aef\u4ee3\u7801\n        |- conf\n            |- .qwen\n        |- utils \\  \n            |- util.py          # \u5b9e\u73b0\u8fde\u63a5\u5927\u6a21\u578b\u7684\u65b9\u6cd5\n        |- rag \\   \n            |- pdf_processor.py\n            |- chroma_conn.py\n        |- test_framework.py      \n        |- finance_bot_ex.py    # \u5c01\u88c5\u597d\u7684FinanceBotEx\u7c7b\n        |- server.py            # \u65b0\u589e\u7684\u670d\u52a1\u4ee3\u7801<\/code><\/pre>\n<p>\u542f\u52a8\u540e\u7aef\u670d\u52a1\u65b9\u6cd5<\/p>\n<pre><code class=\"language-bash\"># \u5207\u6362\u81f3\u9879\u76ee\u6839\u76ee\u5f55\ncd smart-finance-bot\n\n# \u542f\u52a8\u670d\u52a1\npython app\/server.py<\/code><\/pre>\n<p>\u8fd0\u884c\u7ed3\u679c\uff1a<br \/>\n<a href=\"https:\/\/17aitech.com\/wp-content\/uploads\/2024\/09\/langserve\u542f\u52a8.png\" data-fancybox=\"images\" data-fancybox=\"gallery\"><img decoding=\"async\" src=\"https:\/\/17aitech.com\/wp-content\/uploads\/2024\/09\/langserve\u542f\u52a8.png\" alt=\"\" \/><\/a><\/p>\n<h3><span class=\"ez-toc-section\" id=\"%E6%8E%A5%E5%8F%A3%E6%B5%8B%E8%AF%95\"><\/span>\u63a5\u53e3\u6d4b\u8bd5<span class=\"ez-toc-section-end\"><\/span><\/h3>\n<p>\u670d\u52a1\u6b63\u5e38\u542f\u52a8\u540e\uff0c\u6211\u4eec\u901a\u8fc7curl\u547d\u4ee4\u6d4b\u8bd5\u540e\u7aef\u63a5\u53e3\u6b63\u5e38<\/p>\n<pre><code class=\"language-bash\">curl -X POST http:\/\/localhost:8082\/queryex \\\n-H &quot;Content-Type: application\/json&quot; \\\n-d &#039;{\n  &quot;config&quot;: {},\n  &quot;input&quot;: &quot;\u572820190211\uff0c\u6309\u7167\u4e2d\u4fe1\u884c\u4e1a\u5206\u7c7b\u7684\u6807\u51c6\uff0c\u4e00\u7ea7\u884c\u4e1a\u4e2d\u7684\\&quot;A\u80a1\u516c\u53f8\u6570\u91cf\\&quot;\u6700\u591a\u7684\u662f\\&quot;\u673a\u68b0\\&quot;\u884c\u4e1a\uff0c\u5171\u6709549\u5bb6\u516c\u53f8\u3002&quot;\n}&#039;<\/code><\/pre>\n<p>\u8fd0\u884c\u7ed3\u679c\uff1a<br \/>\n<a href=\"https:\/\/17aitech.com\/wp-content\/uploads\/2024\/09\/curl\u6d4b\u8bd5\u63a5\u53e3.png\" data-fancybox=\"images\" data-fancybox=\"gallery\"><img decoding=\"async\" src=\"https:\/\/17aitech.com\/wp-content\/uploads\/2024\/09\/curl\u6d4b\u8bd5\u63a5\u53e3.png\" alt=\"\" \/><\/a><\/p>\n<p>\u540e\u7aef\u63a5\u53e3\u53ef\u4ee5\u63a5\u6536\u8bf7\u6c42\uff0c\u670d\u52a1\u4e00\u5207\u6b63\u5e38\uff0c\u63a5\u4e0b\u6765\u6211\u4eec\u5f00\u59cb\u6784\u5efa\u524d\u7aef\u9875\u9762\uff0c\u5b8c\u6210\u95ee\u7b54\u7cfb\u7edf\u524d\u540e\u7aef\u8054\u8c03\u3002<\/p>\n<h2><span class=\"ez-toc-section\" id=\"%E5%89%8D%E7%AB%AF%E5%AE%9E%E7%8E%B0\"><\/span>\u524d\u7aef\u5b9e\u73b0<span class=\"ez-toc-section-end\"><\/span><\/h2>\n<p>\u524d\u7aef\u9875\u9762\u7684\u5b9e\u73b0\u65b9\u6848\u6709\u5f88\u591a\u79cd\uff0c\u53ef\u4ee5\u5199\u4e2a\u7b80\u5355\u7684HTML\u9875\u9762\uff0c\u4e5f\u53ef\u4ee5\u4f7f\u7528Vue.js\u6846\u67b6\u6765\u5b9e\u73b0\u4e00\u4e2a\u524d\u7aef\u3002<br \/>\n\u56e0\u4e3aVue.js\u662f\u76ee\u524d\u4e3b\u6d41\u7684\u524d\u7aef\u6846\u67b6\uff0c\u5b83\u6709\u4e30\u5bcc\u7684\u751f\u6001\u548cAPI\u8bf4\u660e\uff0c\u5728github\u4e0a\u4e5f\u6709\u975e\u5e38\u591a\u7684\u5f00\u6e90\u9879\u76ee\u4f9b\u53c2\u8003\uff0c\u6240\u4ee5\u672c\u4f8b\u4e2d\u6211\u4eec\u5c06\u4f7f\u7528Vue.js\u6846\u67b6\u6765\u5b9e\u73b0\u524d\u7aef\u9875\u9762\u3002<\/p>\n<h3><span class=\"ez-toc-section\" id=\"%E8%83%8C%E6%99%AF%E7%9F%A5%E8%AF%86\"><\/span>\u80cc\u666f\u77e5\u8bc6<span class=\"ez-toc-section-end\"><\/span><\/h3>\n<p><strong>Vue.js\u7b80\u4ecb<\/strong><br \/>\nVue.js \u662f\u4e00\u4e2a\u7528\u4e8e\u6784\u5efa\u7528\u6237\u754c\u9762\u7684\u6e10\u8fdb\u5f0f JavaScript \u6846\u67b6\u3002\u5b83\u7684\u6838\u5fc3\u5e93\u4e13\u6ce8\u4e8e\u89c6\u56fe\u5c42\uff0c\u6613\u4e8e\u4e0a\u624b\uff0c\u5e76\u4e0e\u5176\u4ed6\u5e93\u6216\u73b0\u6709\u9879\u76ee\u8fdb\u884c\u6574\u5408\u3002<\/p>\n<p><strong>Vue.js\u7684\u7279\u70b9<\/strong><\/p>\n<ul>\n<li>\u54cd\u5e94\u5f0f\u6570\u636e\u7ed1\u5b9a\uff1aVue.js \u63d0\u4f9b\u4e86\u53cc\u5411\u6570\u636e\u7ed1\u5b9a\uff0c\u80fd\u591f\u5b9e\u65f6\u66f4\u65b0\u89c6\u56fe\u4e0e\u6570\u636e\u4e4b\u95f4\u7684\u53d8\u5316\u3002<\/li>\n<li>\u6307\u4ee4\uff1aVue.js \u63d0\u4f9b\u4e86\u4e00\u4e9b\u5185\u7f6e\u6307\u4ee4\uff08\u5982 v-bind\u3001v-model\u3001v-for \u7b49\uff09\uff0c\u4f7f\u5f97\u6570\u636e\u7ed1\u5b9a\u548c DOM \u64cd\u4f5c\u53d8\u5f97\u7b80\u5355\u76f4\u89c2\u3002<\/li>\n<li>\u751f\u6001\u7cfb\u7edf\uff1aVue.js \u62e5\u6709\u4e30\u5bcc\u7684\u751f\u6001\u7cfb\u7edf\uff0c\u5305\u62ec\u8def\u7531\u7ba1\u7406\uff08Vue Router\uff09\u3001\u72b6\u6001\u7ba1\u7406\uff08Vuex\uff09\u7b49\u5de5\u5177\uff0c\u652f\u6301\u6784\u5efa\u590d\u6742\u7684\u5e94\u7528\u3002<\/li>\n<\/ul>\n<p><strong>Vue.js\u7684\u7b80\u5355\u793a\u4f8b<\/strong><\/p>\n<pre><code class=\"language-javascript\">&lt;div id=&quot;app&quot;&gt;\n  &lt;h1&gt;{{ message }}&lt;\/h1&gt;\n  &lt;input v-model=&quot;message&quot; placeholder=&quot;\u7f16\u8f91\u6211&quot;&gt;\n&lt;\/div&gt;\n\n&lt;script src=&quot;https:\/\/cdn.jsdelivr.net\/npm\/vue@2&quot;&gt;&lt;\/script&gt;\n&lt;script&gt;\n  new Vue({\n    el: &#039;#app&#039;,\n    data: {\n      message: &#039;Hello Vue!&#039;\n    }\n  });\n&lt;\/script&gt;\n<\/code><\/pre>\n<p>\u8bf4\u660e\uff1a<\/p>\n<ul>\n<li>\u5728\u4e0a\u9762\u7684\u793a\u4f8b\u4e2d\uff0c{{ message }} \u7528\u4e8e\u663e\u793a\u6570\u636e\uff0cv-model \u7528\u4e8e\u5b9e\u73b0\u53cc\u5411\u6570\u636e\u7ed1\u5b9a\uff0c\u5141\u8bb8\u7528\u6237\u901a\u8fc7\u8f93\u5165\u6846\u4fee\u6539 message \u7684\u5185\u5bb9\u3002<\/li>\n<li>\u66f4\u4e3a\u8be6\u7ec6\u7684Vue.js\u4f7f\u7528\u65b9\u6cd5\uff0c\u8bf7\u67e5\u770b\u5b98\u65b9\u6587\u6863\u5730\u5740\uff1a<a href=\"https:\/\/cn.vuejs.org\/\">https:\/\/cn.vuejs.org\/<\/a><\/li>\n<\/ul>\n<h3><span class=\"ez-toc-section\" id=\"%E4%BB%A3%E7%A0%81%E5%AE%9E%E7%8E%B0-2\"><\/span>\u4ee3\u7801\u5b9e\u73b0<span class=\"ez-toc-section-end\"><\/span><\/h3>\n<p>\u5728\u672c\u4f8b\u4e2d\uff0c\u6211\u4eec\u53c2\u8003Github\u4e0a\u7684\u4e00\u4e2a\u5f00\u6e90\u9879\u76ee\u8fdb\u884c\u4e8c\u6b21\u5f00\u53d1\uff0c\u8be5\u9879\u76eegit clone\u4e4b\u540e\u76ee\u5f55\u5982\u4e0b\uff1a<\/p>\n<pre><code class=\"language-bash\">chatweb \\\n    |- index.html               # \u524d\u7aef\u5165\u53e3\u6587\u4ef6\n    |- node_modules             # \u4f9d\u8d56\u5305\u76ee\u5f55\n    |- package-lock.json        # \u9501\u5b9a\u4f9d\u8d56\u7248\u672c\u7684\u6587\u4ef6\n    |- package.json             # \u9879\u76ee\u63cf\u8ff0\u548c\u4f9d\u8d56\u914d\u7f6e\u6587\u4ef6\n    |- postcss.config.js        # PostCSS \u914d\u7f6e\u6587\u4ef6\n    |- public                   # \u516c\u5171\u8d44\u6e90\u76ee\u5f55\n    |- src                      # \u6e90\u4ee3\u7801\u76ee\u5f55\n        |- App.vue              # \u4e3b\u5e94\u7528\u7ec4\u4ef6\n        |- assets               # \u8d44\u6e90\u6587\u4ef6\u76ee\u5f55\n        |- components           # \u7ec4\u4ef6\u76ee\u5f55\n            |- ChatComponents.vue # \u804a\u5929\u7ec4\u4ef6\n            |- Home.vue        # \u9996\u9875\u7ec4\u4ef6\n            |- Login.vue       # \u767b\u5f55\u7ec4\u4ef6\n            |- NotFound.vue    # 404 \u9875\u9762\u7ec4\u4ef6\n        |- main.js              # \u5e94\u7528\u5165\u53e3\u6587\u4ef6\n        |- routes               # \u8def\u7531\u914d\u7f6e\u76ee\u5f55\n    |- tailwind.config.js       # Tailwind CSS \u914d\u7f6e\u6587\u4ef6\n    |- tsconfig.json            # TypeScript \u914d\u7f6e\u6587\u4ef6\n    |- vite.config.js           # Vite \u914d\u7f6e\u6587\u4ef6\n<\/code><\/pre>\n<blockquote>\n<p>\u8be5\u9879\u76ee\u662f\u4effchatGPT\u7684\u95ee\u7b54\u7cfb\u7edf\uff0c\u4ed3\u5e93\u5730\u5740\u4e3a\uff1a<a href=\"https:\/\/github.com\/SinMu-L\/chatweb\">SinMu-L\/chatweb<\/a><\/p>\n<\/blockquote>\n<h4><span class=\"ez-toc-section\" id=\"%E4%BF%AE%E6%94%B9indexhtml\"><\/span>\u4fee\u6539index.html<span class=\"ez-toc-section-end\"><\/span><\/h4>\n<p>\u7f16\u8f91index.html\uff0c\u4fee\u6539\u5982\u4e0b\uff1a<\/p>\n<pre><code class=\"language-html\">&lt;!DOCTYPE html&gt;\n&lt;html lang=&quot;en&quot;&gt;\n  &lt;head&gt;\n    &lt;meta charset=&quot;UTF-8&quot;&gt;\n    &lt;link rel=&quot;icon&quot; href=&quot;.\/src\/assets\/icon.jpg&quot;&gt;\n    &lt;meta name=&quot;viewport&quot; content=&quot;width=device-width, initial-scale=1.0&quot;&gt;\n    &lt;title&gt;\u91d1\u878d\u5343\u95ee\u673a\u5668\u4eba&lt;\/title&gt;\n  &lt;\/head&gt;\n  &lt;body&gt;\n    &lt;div id=&quot;app&quot;&gt;&lt;\/div&gt;\n    &lt;script type=&quot;module&quot; src=&quot;\/src\/main.js&quot;&gt;&lt;\/script&gt;\n  &lt;\/body&gt;\n&lt;\/html&gt;\n<\/code><\/pre>\n<h4><span class=\"ez-toc-section\" id=\"%E4%BF%AE%E6%94%B9%E8%B7%AF%E7%94%B1%E9%85%8D%E7%BD%AE%E6%96%87%E4%BB%B6\"><\/span>\u4fee\u6539\u8def\u7531\u914d\u7f6e\u6587\u4ef6<span class=\"ez-toc-section-end\"><\/span><\/h4>\n<p>\u7f16\u8f91src\/routes\/routers.js\uff0c\u4fee\u6539\u5982\u4e0b\uff1a<\/p>\n<pre><code class=\"language-javascript\">import ContainerTemplate from &#039;..\/components\/ChatComponents.vue&#039;\nimport NotFound from &#039;..\/components\/NotFound.vue&#039;\nimport Home from &#039;..\/components\/Home.vue&#039;\n\nvar routes = [\n  {path: &quot;\/&quot;, name: &#039;root&#039;, component: Home },\n  {path: &quot;\/chat\/:uuid?&quot;, component:ContainerTemplate, name:&#039;chat&#039;},\n  { path: &#039;\/404&#039;, name: &#039;not_found&#039;, component: NotFound } \/\/ \u6355\u83b7\u6240\u6709\u672a\u5339\u914d\u7684\u8def\u5f84\n];\nexport default routes;\n<\/code><\/pre>\n<p>\u8bf4\u660e\uff1a<\/p>\n<ul>\n<li>\u8be5\u914d\u7f6e\u6587\u4ef6\u7528\u4e8e\u63a7\u5236\u9875\u9762\u8bf7\u6c42\u7684\u8def\u7531\u4fe1\u606f<\/li>\n<li>\u5f53\u7528\u6237\u8bbf\u95ee\u7684\u8def\u5f84\u4e3a\/\u65f6\uff0c\u5c06\u663e\u793aHome\u7ec4\u4ef6\uff1b<\/li>\n<li>\u5f53\u7528\u6237\u8bbf\u95ee\u7684\u8def\u5f84\u4e3a\/chat\u65f6\uff0c\u5c06\u663e\u793aChatComponents\u7ec4\u4ef6\uff1b<\/li>\n<li>\u5f53\u7528\u6237\u8bbf\u95ee\u7684\u8def\u5f84\u4e3a\u5176\u4ed6\u8def\u5f84\uff0c\u5c06\u663e\u793a404\u9875\u9762\u3002<\/li>\n<\/ul>\n<p>\u7531\u4e0a\u4e5f\u53ef\u4ee5\u770b\u5230\uff0c\u540e\u7eed\u4fee\u6539\u7684\u5bf9\u8bdd\u9875\u9762\u5728 <code>ChatComponents.vue<\/code> \u4e2d\u3002<\/p>\n<h4><span class=\"ez-toc-section\" id=\"%E4%BF%AE%E6%94%B9%E8%81%8A%E5%A4%A9%E7%BB%84%E4%BB%B6%E9%A1%B5%E9%9D%A2\"><\/span>\u4fee\u6539\u804a\u5929\u7ec4\u4ef6\u9875\u9762<span class=\"ez-toc-section-end\"><\/span><\/h4>\n<p>\u7531\u4e8e\u539f\u59cb\u7684 <code>ChatComponents.vue<\/code> \u4ee3\u7801\u4e0d\u591f\u89c4\u8303\uff0c\u663e\u793a\u5e03\u5c40\u3001\u663e\u793a\u6837\u5f0f\u4ee5\u53ca\u903b\u8f91\u4ee3\u7801\u6ca1\u6709\u5206\u79bb\uff0c\u6240\u4ee5\u6211\u4eec\u5bf9\u5176\u8fdb\u884c\u91cd\u6784\uff0c\u91cd\u6784\u5185\u5bb9\u5206\u4e3a\u4e09\u90e8\u5206\uff1a<\/p>\n<h5><span class=\"ez-toc-section\" id=\"%E9%87%8D%E6%9E%84%E9%A1%B5%E9%9D%A2%E7%9A%84%E6%98%BE%E7%A4%BA%E5%B8%83%E5%B1%80\"><\/span>\u91cd\u6784\u9875\u9762\u7684\u663e\u793a\u5e03\u5c40<span class=\"ez-toc-section-end\"><\/span><\/h5>\n<pre><code class=\"language-html\">\n&lt;template&gt;\n  &lt;div&gt;\n    &lt;div v-if=&quot;centerLodding&quot; class=&quot;loading-container&quot;&gt;\n      &lt;n-spin size=&quot;large&quot; \/&gt;\n    &lt;\/div&gt;\n    &lt;Login class=&quot;border border-red-400&quot; \/&gt;\n    &lt;div class=&quot;flex flex-row h-min&quot; :class=&quot;hasLogin(&#039;main&#039;)&quot;&gt;\n      &lt;!-- \u975e\u79fb\u52a8\u7aef\u6a21\u5f0f\u4e0b\u4fa7\u8fb9\u680f\u7684\u6837\u5f0f --&gt;\n      &lt;div class=&quot;sidebar&quot; :class=&quot;controlSidebarHidden ? &#039;w-0&#039; : &#039;&#039;&quot;&gt;\n        &lt;div class=&quot;hidden sm:flex sm:flex-col sm:h-screen sm:border&quot;&gt;\n          &lt;!-- \u65b0\u5efa\u6309\u94ae --&gt;\n          &lt;div class=&quot;basis-1\/12 flex justify-center items-center&quot;&gt;\n            &lt;n-button class=&quot;w-4\/5&quot; @click=&quot;addLeftListEle&quot;&gt;\u65b0\u7684\u4f1a\u8bdd&lt;\/n-button&gt;\n          &lt;\/div&gt;\n          &lt;!-- \u5217\u8868 --&gt;\n          &lt;div class=&quot;basis-10\/12 overflow-auto border&quot;&gt;\n            &lt;div v-for=&quot;item in left_data.left_list&quot; :key=&quot;item.uuid&quot;&gt;\n              &lt;router-link :to=&quot;`\/chat\/${item.uuid}`&quot; class=&quot;sidebar-item&quot;&gt;\n                &lt;div class=&quot;w-4\/5 flex items-center&quot;&gt;\n                  &lt;n-icon size=&quot;medium&quot;&gt;\n                    &lt;game-controller-outline \/&gt;\n                  &lt;\/n-icon&gt;\n                  &lt;div class=&quot;truncate mx-2&quot; style=&quot;color: #000;&quot;&gt;\n                    &lt;p v-if=&quot;!item.enable_edit&quot; class=&quot;truncate h-full&quot;&gt;{{ item.title }}&lt;\/p&gt;\n                    &lt;n-input v-else type=&quot;text&quot; size=&quot;small&quot; class=&quot;h-full&quot; v-model:value=&quot;item.title&quot; @keyup.enter=&quot;submit(item.uuid)&quot; \/&gt;\n                  &lt;\/div&gt;\n                &lt;\/div&gt;\n                &lt;div class=&quot;w-1\/5 flex justify-center items-center&quot; :class=&quot;route.params.uuid != item.uuid ? &#039;hidden&#039; : &#039;&#039;&quot;&gt;\n                  &lt;n-button-group size=&quot;small&quot;&gt;\n                    &lt;n-button text @click=&quot;editLeftListEle(item.uuid)&quot;&gt;\n                      &lt;n-icon&gt;&lt;Edit \/&gt;&lt;\/n-icon&gt;\n                    &lt;\/n-button&gt;\n                    &lt;n-button text @click=&quot;delLeftListEle(item.uuid)&quot;&gt;\n                      &lt;n-icon&gt;&lt;Delete \/&gt;&lt;\/n-icon&gt;\n                    &lt;\/n-button&gt;\n                  &lt;\/n-button-group&gt;\n                &lt;\/div&gt;\n              &lt;\/router-link&gt;\n            &lt;\/div&gt;\n          &lt;\/div&gt;\n          &lt;!-- \u8bbe\u7f6e\u9875\u9762 --&gt;\n          &lt;div class=&quot;footer flex flex-col items-center justify-between p-4 h-15&quot;&gt;\n            &lt;div class=&quot;user-info font-bold text-lg&quot;&gt;\u91d1\u878d\u5343\u95ee\u673a\u5668\u4eba&lt;\/div&gt;\n            &lt;div class=&quot;logo-container relative&quot;&gt; &lt;!-- \u6dfb\u52a0\u4e00\u4e2a\u5bb9\u5668 --&gt;\n              &lt;img src=&quot;..\/assets\/background.png&quot; alt=&quot;Logo&quot; class=&quot;logo&quot; \/&gt;\n            &lt;\/div&gt;\n            &lt;div class=&quot;robot-description text-sm text-gray-600 mt-2&quot;&gt;\n              \u91d1\u878d\u5343\u95ee\u673a\u5668\u4eba\uff0c\u901a\u8fc7RAG\u5bf9\u65e2\u6709\u7684PDF\u62db\u80a1\u4e66\u5efa\u7acb\u4e86\u77e5\u8bc6\u5e93\uff0c\u540c\u65f6\u501f\u52a9\u5927\u6a21\u578b+Agent\u5bf9\u91d1\u878dSQL\u6570\u636e\u5e93\u8fdb\u884c\u52a8\u6001\u67e5\u8be2\uff0c\u65e8\u5728\u4e3a\u7528\u6237\u63d0\u4f9b\u5feb\u901f\u3001\u51c6\u786e\u7684\u91d1\u878d\u4fe1\u606f\u548c\u54a8\u8be2\u670d\u52a1\u3002\n            &lt;\/div&gt;\n            &lt;div class=&quot;settings-icon&quot;&gt;\n              &lt;n-icon @click=&quot;showSettingFunc()&quot;&gt;\n                &lt;SettingsOutline \/&gt;\n              &lt;\/n-icon&gt;\n            &lt;\/div&gt;\n          &lt;\/div&gt;\n        &lt;\/div&gt;\n      &lt;\/div&gt;\n\n      &lt;!-- \u79fb\u52a8\u7aef\u6a21\u5f0f\u4e0b\u4fa7\u8fb9\u680f\u7684\u6837\u5f0f --&gt;\n      &lt;div class=&quot;sm:hidden absolute top-1 left-1 h-full w-full flex flex-col&quot;&gt;\n        &lt;div&gt;\n          &lt;n-button text style=&quot;font-size:32px&quot; @click=&quot;controlSidebarHidden = !controlSidebarHidden&quot;&gt;\n            &lt;n-icon class=&quot;text-black&quot;&gt;\n              &lt;Menu \/&gt;\n            &lt;\/n-icon&gt;\n          &lt;\/n-button&gt;\n        &lt;\/div&gt;\n        &lt;div v-if=&quot;!controlSidebarHidden&quot; class=&quot;mobile-sidebar&quot;&gt;\n          &lt;div class=&quot;w-full flex flex-col h-screen border&quot;&gt;\n            &lt;div class=&quot;basis-1\/12 flex justify-center items-center&quot;&gt;\n              &lt;n-button class=&quot;w-4\/5&quot; @click=&quot;addLeftListEle&quot;&gt;\u65b0\u7684\u5bf9\u8bdd&lt;\/n-button&gt;\n            &lt;\/div&gt;\n            &lt;div class=&quot;basis-10\/12 overflow-auto border&quot;&gt;\n              &lt;div v-for=&quot;item in left_data.left_list&quot; :key=&quot;item.uuid&quot;&gt;\n                &lt;router-link :to=&quot;`\/chat\/${item.uuid}`&quot; class=&quot;sidebar-item&quot;&gt;\n                  &lt;div class=&quot;w-4\/5 flex items-center&quot;&gt;\n                    &lt;n-icon size=&quot;medium&quot;&gt;\n                      &lt;game-controller-outline \/&gt;\n                    &lt;\/n-icon&gt;\n                    &lt;div class=&quot;truncate mx-2&quot;&gt;\n                      &lt;p v-if=&quot;!item.enable_edit&quot; class=&quot;truncate h-full&quot;&gt;{{ item.title }}&lt;\/p&gt;\n                      &lt;n-input v-else type=&quot;text&quot; size=&quot;small&quot; class=&quot;h-full&quot; v-model:value=&quot;item.title&quot; @keyup.enter=&quot;submit(item.uuid)&quot; \/&gt;\n                    &lt;\/div&gt;\n                  &lt;\/div&gt;\n                  &lt;div class=&quot;w-1\/5 flex justify-center items-center&quot; :class=&quot;route.params.uuid != item.uuid ? &#039;hidden&#039; : &#039;&#039;&quot;&gt;\n                    &lt;n-button-group size=&quot;small&quot;&gt;\n                      &lt;n-button text @click=&quot;editLeftListEle(item.uuid)&quot;&gt;\n                        &lt;n-icon&gt;&lt;Edit \/&gt;&lt;\/n-icon&gt;\n                      &lt;\/n-button&gt;\n                      &lt;n-button text @click=&quot;delLeftListEle(item.uuid)&quot;&gt;\n                        &lt;n-icon&gt;&lt;Delete \/&gt;&lt;\/n-icon&gt;\n                      &lt;\/n-button&gt;\n                    &lt;\/n-button-group&gt;\n                  &lt;\/div&gt;\n                &lt;\/router-link&gt;\n              &lt;\/div&gt;\n            &lt;\/div&gt;\n            &lt;div class=&quot;footer flex items-center justify-between p-4 h-15&quot;&gt;\n              &lt;div class=&quot;user-info font-bold&quot;&gt;\u91d1\u878d\u5343\u95ee\u673a\u5668\u4eba&lt;\/div&gt;\n              &lt;div class=&quot;settings-icon&quot;&gt;\n                &lt;n-icon @click=&quot;showSettingFunc()&quot;&gt;\n                  &lt;SettingsOutline \/&gt;\n                &lt;\/n-icon&gt;\n              &lt;\/div&gt;\n            &lt;\/div&gt;\n          &lt;\/div&gt;\n        &lt;\/div&gt;\n      &lt;\/div&gt;\n\n      &lt;div class=&quot;w-full sm:w-4\/5 h-full&quot;&gt;\n        &lt;div class=&quot;flex flex-col h-screen&quot;&gt;\n          &lt;div v-if=&quot;left_list_is_empty&quot; class=&quot;empty-chat-message&quot;&gt;\n            Hi\n          &lt;\/div&gt;\n          &lt;div v-else class=&quot;chat-area&quot; id=&quot;msgArea&quot;&gt;\n            &lt;div v-for=&quot;(msglist, index) in getMsgList(route.params.uuid)&quot; :key=&quot;index&quot; class=&quot;flex flex-col mt-1 msgItem&quot;&gt;\n              &lt;div :class=&quot;msglist.reversion ? &#039;flex-row-reverse&#039; : &#039;flex-row&#039;&quot; class=&quot;flex justify-start items-center h-10&quot;&gt;\n\n                &lt;img\n                    class=&quot;rounded-full avatar&quot;\n                    :src=&quot;msglist.reversion ? userAvatar : robotAvatar&quot;\n                    alt=&quot;\u5934\u50cf&quot;\n                \/&gt;\n                &lt;span class=&quot;ml-4 text-sm&quot;&gt;{{ msglist.create_time }}&lt;\/span&gt;\n              &lt;\/div&gt;\n              &lt;div class=&quot;flex&quot; :class=&quot;msglist.reversion ? &#039;flex-row-reverse&#039; : &#039;flex-row&#039;&quot;&gt;\n                &lt;div class=&quot;message-container&quot;&gt;\n                  &lt;n-spin v-if=&quot;msglist.msgload&quot; size=&quot;small&quot; stroke=&quot;red&quot; \/&gt;\n                  &lt;Markdown v-else :source=&quot;msglist.content&quot;&gt;&lt;\/Markdown&gt;\n                &lt;\/div&gt;\n              &lt;\/div&gt;\n            &lt;\/div&gt;\n          &lt;\/div&gt;\n          &lt;div v-if=&quot;!left_list_is_empty&quot; class=&quot;input-area&quot;&gt;\n            &lt;div class=&quot;p-2&quot;&gt;\n              &lt;n-input-group&gt;\n                &lt;n-tooltip trigger=&quot;hover&quot;&gt;\n                  &lt;template #trigger&gt;\n                    &lt;n-button text size=&quot;large&quot; class=&quot;px-2&quot; @click=&quot;deleteChatItemHistory(route.params.uuid)&quot;&gt;\n                      &lt;n-icon class=&quot;text-black&quot;&gt;\n                        &lt;Delete \/&gt;\n                      &lt;\/n-icon&gt;\n                    &lt;\/n-button&gt;\n                  &lt;\/template&gt;\n                  \u5220\u9664\u5f53\u524d\u4f1a\u8bdd\u8bb0\u5f55\n                &lt;\/n-tooltip&gt;\n                &lt;n-tooltip trigger=&quot;hover&quot;&gt;\n                  &lt;template #trigger&gt;\n                    &lt;n-button text size=&quot;large&quot; class=&quot;pr-4&quot; @click=&quot;dom2img()&quot;&gt;\n                      &lt;n-icon class=&quot;text-black&quot;&gt;\n                        &lt;Download \/&gt;\n                      &lt;\/n-icon&gt;\n                    &lt;\/n-button&gt;\n                  &lt;\/template&gt;\n                  \u4e0b\u8f7d\u5f53\u524d\u4f1a\u8bdd\u4e3a\u56fe\u7247\n                &lt;\/n-tooltip&gt;\n                &lt;a href=&quot;&quot; id=&quot;link&quot; class=&quot;hidden&quot;&gt;&lt;\/a&gt;\n                &lt;n-input show-count @keyup.ctrl.enter=&quot;addMessageListItem(route.params.uuid)&quot; placeholder=&quot;Ctrl+Enter \u53d1\u9001\u6d88\u606f\uff0c\u53d1\u9001\u6d88\u606f\u957f\u5ea6\u9700\u8981\u5927\u4e8e2\u4e2a\u5b57&quot; v-model:value=&quot;input_area_value&quot; type=&quot;textarea&quot; size=&quot;tiny&quot; :autosize=&quot;{ minRows: 2, maxRows: 5 }&quot; \/&gt;\n                &lt;n-button ghost class=&quot;h-auto&quot; @click=&quot;addMessageListItem(route.params.uuid)&quot;&gt;\n                  \u53d1\u9001\n                &lt;\/n-button&gt;\n              &lt;\/n-input-group&gt;\n            &lt;\/div&gt;\n          &lt;\/div&gt;\n        &lt;\/div&gt;\n      &lt;\/div&gt;\n    &lt;\/div&gt;\n\n    &lt;!-- \u8bbe\u7f6e\u6a21\u6001\u6846 --&gt;\n    &lt;n-modal v-model:show=&quot;showSetting&quot; style=&quot;width: 600px&quot; class=&quot;custom-card&quot; preset=&quot;card&quot; title=&quot;\u8bbe\u7f6e&quot;&gt;\n      &lt;n-tabs type=&quot;line&quot; animated&gt;\n        &lt;n-tab-pane name=&quot;about&quot; tab=&quot;\u5173\u4e8e&quot;&gt;\n          &lt;div&gt;\u8fd9\u662f\u4e00\u4e2ademo\u9879\u76ee\uff0c\u4ec5\u7528\u4e8e\u5b66\u4e60\u3002&lt;\/div&gt;\n          &lt;div class=&quot;my-4&quot;&gt;\u6280\u672f\u6808\uff1aVue3 + Vite + tailwindCss3 + NaiveUi&lt;\/div&gt;\n        &lt;\/n-tab-pane&gt;\n        &lt;n-tab-pane name=&quot;settings&quot; tab=&quot;\u8bbe\u7f6e&quot;&gt;\n          &lt;div class=&quot;grid grid-rows-3 gap-4&quot;&gt;\n            &lt;div&gt;\n              &lt;span class=&quot;mr-4&quot;&gt;Server URL: &lt;\/span&gt;\n              &lt;n-input v-model:value=&quot;setting.server_url&quot; placeholder=&quot;\u8bf7\u8f93\u5165\u670d\u52a1\u5668 URL&quot; \/&gt;\n            &lt;\/div&gt;\n            &lt;div&gt;\n              &lt;span class=&quot;mr-4&quot;&gt;type: &lt;\/span&gt;\n              &lt;n-select :style=&quot;{ width: &#039;80%&#039; }&quot; :options=&quot;selectOptions&quot; v-model:value=&quot;setting.type&quot; \/&gt;\n            &lt;\/div&gt;\n            &lt;div class=&quot;flex justify-end&quot;&gt;\n              &lt;n-button\n                  @click=&quot;confirmSettings&quot;\n                  class=&quot;custom-button&quot;&gt;\n                \u786e\u8ba4\n              &lt;\/n-button&gt;\n            &lt;\/div&gt;\n          &lt;\/div&gt;\n        &lt;\/n-tab-pane&gt;\n        &lt;n-tab-pane name=&quot;other&quot; tab=&quot;\u5176\u4ed6&quot;&gt;\n          \u5176\u4ed6\n        &lt;\/n-tab-pane&gt;\n      &lt;\/n-tabs&gt;\n    &lt;\/n-modal&gt;\n  &lt;\/div&gt;\n&lt;\/template&gt;\n<\/code><\/pre>\n<h5><span class=\"ez-toc-section\" id=\"%E9%87%8D%E6%9E%84%E9%A1%B5%E9%9D%A2%E7%9A%84%E6%98%BE%E7%A4%BA%E6%A0%B7%E5%BC%8F\"><\/span>\u91cd\u6784\u9875\u9762\u7684\u663e\u793a\u6837\u5f0f<span class=\"ez-toc-section-end\"><\/span><\/h5>\n<pre><code class=\"language-css\">&lt;style scoped&gt;\n.router-link-active {\n  border-color: #18a058;\n  color: #18a058;\n}\n\n.loading-container {\n  position: absolute;\n  top: 33%;\n  left: 33%;\n  background-color: gray;\n  width: 33%;\n  height: 33%;\n  display: flex;\n  justify-content: center;\n  align-items: center;\n}\n\n.sidebar {\n  width: 20%;\n  height: 100%;\n}\n\n.sidebar-item {\n  margin: 0.5rem;\n  display: flex;\n  justify-content: space-between;\n  align-items: center;\n  border: 1px solid gray;\n  border-radius: 0.375rem; \/* rounded-md *\/\n  padding: 0.5rem;\n  color: white;\n}\n\n.settings-container {\n  display: flex;\n  justify-content: flex-start;\n  align-items: center;\n  height: 50%;\n}\n\n.user-info {\n  width: 50%;\n  height: 100%;\n  display: grid;\n  grid-template-rows: repeat(2, 1fr);\n}\n\n.settings-icon {\n  width: 25%;\n  height: 100%;\n  display: flex;\n  justify-content: center;\n  align-items: center;\n}\n\n.mobile-sidebar {\n  width: 80%;\n  background-color: white;\n  dark:bg-black;\n}\n\n.empty-chat-message {\n  flex-basis: 91%;\n  width: 100%;\n  padding: 3rem;\n  overflow: auto;\n  display: flex;\n  justify-content: center;\n  align-items: flex-start;\n  color: gray;\n  font-style: italic;\n}\n\n.chat-area {\n  flex-basis: 91%;\n  width: 100%;\n  padding: 3rem;\n  overflow: auto;\n}\n\n.message-container {\n  background-color: #bfdbfe; \/* bg-blue-200 *\/\n  color: black; \/* dark:text-black *\/\n  width: auto;\n  max-width: 80%;\n  min-width: 1%;\n  break-words: break-word;\n  overflow: hidden;\n  border-radius: 0.375rem; \/* rounded-sm *\/\n  padding: 0.5rem;\n  margin: 0.25rem 0;\n}\n\n.input-area {\n  flex-basis: 8%;\n  width: 100%;\n}\n\n.avatar {\n  width: 50px;\n  height: 50px;\n}\n\n.footer {\n  text-align: center; \/* \u5c45\u4e2d\u663e\u793a *\/\n}\n\n.robot-description {\n  margin-top: 8px; \/* \u63cf\u8ff0\u4e0e Logo \u4e4b\u95f4\u7684\u95f4\u8ddd *\/\n  text-align: left; \/* \u5de6\u5bf9\u9f50 *\/\n  width: 100%; \/* \u4f7f\u63cf\u8ff0\u5360\u6ee1\u53ef\u7528\u5bbd\u5ea6 *\/\n  padding-left: 16px; \/* \u53ef\u9009\uff1a\u589e\u52a0\u5de6\u4fa7\u5185\u8fb9\u8ddd *\/\n}\n\n.logo-container {\n  position: relative; \/* \u8bbe\u7f6e\u5bb9\u5668\u4e3a\u76f8\u5bf9\u5b9a\u4f4d *\/\n}\n\n.background-logo {\n  position: absolute; \/* \u8bbe\u7f6e\u80cc\u666f\u56fe\u4e3a\u7edd\u5bf9\u5b9a\u4f4d *\/\n  top: 0; \/* \u9876\u90e8\u5bf9\u9f50 *\/\n  left: 0; \/* \u5de6\u4fa7\u5bf9\u9f50 *\/\n  width: 240px; \/* \u8bbe\u7f6e\u80cc\u666f\u56fe\u5bbd\u5ea6\u4e3a240px *\/\n  height: auto; \/* \u4fdd\u6301\u6bd4\u4f8b *\/\n  z-index: 1; \/* \u786e\u4fdd\u80cc\u666f\u56fe\u5728logo\u4e0b\u65b9 *\/\n}\n\n.logo {\n  position: relative; \/* \u8bbe\u7f6elogo\u4e3a\u76f8\u5bf9\u5b9a\u4f4d *\/\n  z-index: 2; \/* \u786e\u4fddlogo\u5728\u80cc\u666f\u56fe\u4e0a\u65b9 *\/\n  width: 200px; \/* \u6839\u636e\u9700\u8981\u8c03\u6574 Logo \u5927\u5c0f *\/\n  height: auto; \/* \u4fdd\u6301\u6bd4\u4f8b *\/\n  top: 30%; \/* \u5782\u76f4\u5c45\u4e2d *\/\n  left: 50%; \/* \u6c34\u5e73\u5c45\u4e2d *\/\n  transform: translate(-50%, -50%); \/* \u4f7f logo \u5728\u80cc\u666f\u56fe\u4e2d\u5fc3 *\/\n}\n\n.custom-button {\n  background-color: #007bff; \/* \u6309\u94ae\u80cc\u666f\u8272 *\/\n  color: white; \/* \u5b57\u4f53\u989c\u8272 *\/\n  border: none; \/* \u53bb\u6389\u8fb9\u6846 *\/\n  padding: 10px 20px; \/* \u5185\u8fb9\u8ddd *\/\n  border-radius: 5px; \/* \u5706\u89d2 *\/\n  cursor: pointer; \/* \u9f20\u6807\u6837\u5f0f *\/\n  transition: background-color 0.3s; \/* \u6dfb\u52a0\u8fc7\u6e21\u6548\u679c *\/\n}\n\n.custom-button:hover {\n  background-color: #0056b3; \/* \u60ac\u505c\u65f6\u7684\u80cc\u666f\u989c\u8272 *\/\n}\n\n&lt;\/style&gt;<\/code><\/pre>\n<h5><span class=\"ez-toc-section\" id=\"%E9%87%8D%E6%9E%84%E9%A1%B5%E9%9D%A2%E7%9A%84JS%E4%BB%A3%E7%A0%81\"><\/span>\u91cd\u6784\u9875\u9762\u7684JS\u4ee3\u7801<span class=\"ez-toc-section-end\"><\/span><\/h5>\n<pre><code class=\"language-javascript\">&lt;script setup&gt;\nimport {\n    NButton, NInput, NIcon, NButtonGroup, NSpin,\n    NInputGroup, NCard, NModal, NTabs, NTabPane, NInputNumber, NSelect,\n    NTooltip,\n    useMessage\n} from &#039;naive-ui&#039;\nimport { GameControllerOutline, GameController } from &#039;@vicons\/ionicons5&#039;\nimport { LogInOutline as LogInIcon, SettingsOutline, Menu } from &#039;@vicons\/ionicons5&#039;\nimport userAvatar from &#039;@\/assets\/user-avatar.png&#039;;\nimport robotAvatar from &#039;@\/assets\/robot-avatar.png&#039;;\n\nimport { Edit, Delete, Download } from &#039;@vicons\/carbon&#039;\nimport Markdown from &#039;vue3-markdown-it&#039;;\n\nimport Login from &#039;.\/Login.vue&#039;\n\nimport { reactive, ref, getCurrentInstance, watch, watchEffect, nextTick } from &#039;vue&#039;;\nimport { useRouter, useRoute } from &#039;vue-router&#039;\nimport html2canvas from &quot;html2canvas&quot;;\n\nconst router = useRouter()\nconst route = useRoute()\nconst instaceV = getCurrentInstance()\nconst message = useMessage()\n\nimport { onMounted } from &#039;vue&#039;;\n\n\/\/ \u5728\u7ec4\u4ef6\u7684 setup \u51fd\u6570\u4e2d\u6dfb\u52a0\nonMounted(() =&gt; {\n  \/\/ \u68c0\u67e5\u662f\u5426\u6709\u4f1a\u8bdd\u8bb0\u5f55\n  if (left_data.chat.length &gt; 0) {\n    \/\/ \u83b7\u53d6\u6700\u540e\u4e00\u4e2a\u4f1a\u8bdd\u7684 UUID\n    const lastChatUuid = left_data.chat[left_data.chat.length - 1].uuid;\n    \/\/ \u8def\u7531\u8df3\u8f6c\u5230\u6700\u540e\u4e00\u4e2a\u4f1a\u8bdd\n    router.push({ name: &#039;chat&#039;, params: { uuid: lastChatUuid } });\n  } else {\n    \/\/ \u5982\u679c\u6ca1\u6709\u4f1a\u8bdd\uff0c\u8bbe\u7f6e\u5de6\u4fa7\u5217\u8868\u4e3a\u7a7a\n    left_list_is_empty.value = true;\n  }\n});\n\n\/\/ \u63a7\u5236\u4fa7\u8fb9\u680f\u663e\u793a\u9690\u85cf\nvar controlSidebarHidden = ref(true)\n\/\/ \u79fb\u52a8\u7aef\u4e0b\u4fa7\u8fb9\u680f\u663e\u5f71\n\nconst showSetting = ref(false)\n\nvar LLM_APIKEY = ref(&quot;sk-cwtdeSy4Ownmy6Uh5e9b6a67Fe4c4454A3Dc524876348eB1&quot;)\n\nvar setting = reactive({\n    server_url: &#039;http:\/\/localhost:8082\/query&#039;, \/\/ \u9ed8\u8ba4\u503c\n    type: &#039;query&#039;,\n})\n\nlet left_list_is_empty = ref(false)\n\nvar is404 = ref(false)\n\nvar segmented = {\n    content: &#039;soft&#039;,\n    footer: &#039;soft&#039;\n}\n\nvar selectOptions = ref([\n    {\n        label: &#039;Query\u5f0f\u8bbf\u95ee&#039;,\n        value: &#039;query&#039;\n    },\n    {\n        label: &#039;\u6d41\u5f0f\u8bbf\u95ee&#039;,\n        value: &#039;stream_log&#039;\n    }\n])\n\nvar centerLodding = ref(false)\n\nvar input_area_value = ref(&#039;&#039;)\nvar left_data = reactive({\n    left_list: [\n        \/\/ { uuid: 1, title: &#039;New Chat1&#039;, enable_edit: false },\n        \/\/ { uuid: 2, title: &#039;New Chat2&#039;, enable_edit: false },\n    ],\n    chat: [\n        \/\/ {\n        \/\/     uuid: 1, msg_list: [\n        \/\/         { content: &#039;hello1&#039;, create_time: &#039;2023-11-09 11:50:23&#039;, reversion: false, msgload: false },\n        \/\/     ]\n        \/\/ },\n        \/\/ {\n        \/\/     uuid: 2, msg_list: [\n        \/\/         { content: &#039;xxx&#039;, create_time: &#039;2023-11-09 11:50:23&#039;, reversion: false, msgload: false },\n        \/\/     ]\n        \/\/ },\n    ],\n\n})\n\n\/\/ \u76d1\u542c\u54cd\u5e94\u5f0f\u6570\u636e\nwatch(left_data, (newValue, oldValue) =&gt; {\n    if (newValue.chat.length &gt; 0) {\n        left_list_is_empty.value = false\n    } else {\n        left_list_is_empty.value = true\n    }\n    localStorage.setItem(&#039;chatweb&#039;, JSON.stringify(newValue))\n})\n\n\/\/ \u521b\u5efa\u54cd\u5e94\u5f0f\u53d8\u91cf\u540e\u53ea\u6267\u884c\u4e00\u6b21\u8f93\u51fa\u7684\u9700\u6c42\nwatchEffect(() =&gt; {\n    \/\/ \u8bfb\u53d6 localstorage\n    const data = localStorage.getItem(&#039;chatweb&#039;)\n    if (data) {\n\n        const history = JSON.parse(data)\n        left_data.left_list = history.left_list\n        left_data.chat = history.chat\n    } else {\n        left_list_is_empty.value = true\n    }\n})\n\n\/\/ \u6dfb\u52a0\u4fa7\u58c1\u680fitem\nfunction addLeftListEle() {\n    const uuid = randomUuid()\n    left_data.left_list.push({\n        uuid: uuid,\n        title: `\u65b0\u7684\u5bf9\u8bdd${uuid}`,\n        enable_edit: false\n    })\n    left_data.chat.push({\n        uuid: uuid,\n        msg_list: []\n    })\n\n    \/\/ \u8def\u7531\u8df3\u8f6c\u5230\u6700\u65b0\u7684item\n    router.push({ name: &#039;chat&#039;, params: { uuid: uuid } })\n\n}\n\/\/ \u70b9\u51fb\u4fa7\u8fb9\u680f\u67d0\u4e2aitem\u7684\u7f16\u8f91\u6309\u94ae\nfunction editLeftListEle(uuid) {\n    const index = left_data.left_list.findIndex(v =&gt; v.uuid == uuid)\n    left_data.left_list[index].enable_edit = !left_data.left_list[index].enable_edit\n\n    \/\/ all_data.left_list[index].enable_edit = !all_data.left_list[index].enable_edit\n    \/\/ ls.updateLeftListItemEnableEditButton(uuid)\n\n}\n\n\/\/ \u70b9\u51fb\u4fa7\u8fb9\u680f\u67d0\u4e2aitem\u7684\u5220\u9664\u6309\u94ae\nasync function delLeftListEle(uuid) {\n\n    var index = left_data.left_list.findIndex(v =&gt; v.uuid == uuid)\n    left_data.left_list.splice(index, 1)\n\n    const chat_index = left_data.chat.findIndex(v =&gt; v.uuid == uuid)\n    left_data.chat.splice(chat_index, 1)\n\n    if (left_data.chat.length == 0) {\n        left_list_is_empty = true\n    } else {\n        await nextTick()\n        \/\/ \u9ed8\u8ba4\u8df3\u8f6c\u5230\u6700\u65b0\u7684 chat\n        const last_chat_uuid = left_data.chat[left_data.chat.length - 1].uuid\n\n        router.push({ name: &#039;chat&#039;, params: { &quot;uuid&quot;: last_chat_uuid } })\n    }\n}\n\n\/\/ \u83b7\u53d6\u6bcf\u4e2a chat \u7684msg list\nfunction getMsgList(uuid) {\n\n    if (uuid &amp;&amp; uuid != undefined) {\n        var index = left_data.chat.findIndex(v =&gt; v.uuid == uuid)\n        if (index == -1) {\n            return []\n        }\n        return left_data.chat[index].msg_list\n    }\n    return []\n\n}\n\nfunction randomUuid() {\n    var len = 9\n    var uuid = &#039;&#039;;\n    for (let i = 0; i &lt; len; i++) {\n        uuid += Math.floor(Math.random() * 10)\n    }\n    return Number(uuid, 10)\n}\n\n\/\/ \u76d1\u542c\u4fa7\u8fb9\u680fitem\u7684\u56de\u8f66\u4e8b\u4ef6\nfunction submit(index) {\n    editLeftListEle(index)\n}\n\n\/\/ \u53d1\u9001\u6d88\u606f\nfunction addMessageListItem(uuid) {\n    if (input_area_value.value.length &lt;= 2) {\n        message.info(&#039;\u5185\u5bb9\u957f\u5ea6\u4e0d\u5f97\u5c0f\u4e8e2&#039;)\n        return false\n    }\n    var index = left_data.chat.findIndex(v =&gt; v.uuid == uuid)\n    const now_t = (new Date()).toLocaleString(&#039;sv-SE&#039;, { &quot;timeZone&quot;: &quot;PRC&quot; })\n    left_data.chat[index].msg_list.push({\n        content: input_area_value.value,\n        create_time: now_t,\n        reversion: true,\n        msgload: false\n    })\n\n    const body = {\n      config: {},\n      input: input_area_value.value\/\/ \u6309\u8981\u6c42\u7684\u683c\u5f0f\u62fc\u63a5\n\n    };\n\n    \/\/ \u6e05\u7a7a\u7f16\u8f91\u6846\n    input_area_value.value = &#039;&#039;\n    var ele = document.getElementById(&quot;msgArea&quot;)\n    ele.scrollTop = ele.scrollHeight + ele.offsetHeight\n\n    \/\/ \u6dfb\u52a0\u4e00\u4e2a\u5360\u4f4d\u6d88\u606f\uff0c\u8868\u793a\u6b63\u5728\u52a0\u8f7d\n    left_data.chat[index].msg_list.push({\n        content: &#039;&#039;,\n        create_time: now_t,\n        reversion: false,\n        msgload: true\n    })\n    console.info(&quot;\u5f00\u59cb\u53d1\u9001\u6d88\u606f...&quot;)\n\n  \/\/ \u4f7f\u7528\u7f16\u8f91\u6846\u4e2d\u7684 URL \u8fdb\u884c\u8bf7\u6c42\n  const url = setting.server_url; \/\/ \u83b7\u53d6 URL\n\n  if (setting.type === &#039;stream_log&#039;) {\n    startStream(index, body, url); \/\/ \u4f20\u9012 URL\n  } else {\n    startRequest(index, body, url); \/\/ \u4f20\u9012 URL\n  }\n}\n\nfunction buildMessagePromt(index) {\n    const res = []\n    left_data.chat[index].msg_list.forEach(v =&gt; {\n        let role = v.reversion ? &#039;user&#039; : &#039;assistant&#039;\n        res.push({\n            role: role,\n            content: v.content\n        })\n    })\n    res.pop()\n    return res\n}\n\nasync function startRequest(index, body, url) {\n  const key = LLM_APIKEY.value;\n  let response = null;\n\n  try {\n    response = await fetch(url, {\n      method: &quot;POST&quot;,\n      headers: {\n        &quot;Content-Type&quot;: &quot;application\/json&quot;,\n        &quot;Accept&quot;: &quot;application\/json&quot;, \/\/ \u4f7f\u7528 JSON \u4f5c\u4e3a\u63a5\u53d7\u683c\u5f0f\n        &quot;Authorization&quot;: `Bearer ${key}`, \/\/ \u5982\u679c\u9700\u8981 API \u5bc6\u94a5\n      },\n      mode: &quot;cors&quot;,\n      body: JSON.stringify(body),\n    });\n\n    left_data.chat[index].msg_list[left_data.chat[index].msg_list.length - 1].msgload = false;\n  } catch (error) {\n    left_data.chat[index].msg_list[left_data.chat[index].msg_list.length - 1].content += `\u53d1\u751f\u4e86\u4e00\u4e9b\u9519\u8bef\uff1a${error.message}`;\n    return false;\n  }\n\n  if (response.status !== 200) {\n    left_data.chat[index].msg_list[left_data.chat[index].msg_list.length - 1].content += `\u53d1\u751f\u4e86\u4e00\u4e9b\u9519\u8bef\uff1a${response.status}-${response.statusText}`;\n    return false;\n  }\n\n  try {\n    const data = await response.json();\n    if (data &amp;&amp; data.output) { \/\/ \u68c0\u67e5\u6570\u636e\u662f\u5426\u5b58\u5728\u5e76\u5305\u542b output\n      const outputText = data.output;\n      const currentContent = left_data.chat[index].msg_list[left_data.chat[index].msg_list.length - 1].content;\n      if (!currentContent.includes(outputText)) {\n        left_data.chat[index].msg_list[left_data.chat[index].msg_list.length - 1].content += outputText;\n      }\n    } else {\n      throw new Error(&#039;\u8fd4\u56de\u7684\u6570\u636e\u7ed3\u6784\u4e0d\u7b26\u5408\u9884\u671f&#039;);\n    }\n  } catch (e) {\n    console.error(&#039;Error parsing JSON:&#039;, e);\n    left_data.chat[index].msg_list[left_data.chat[index].msg_list.length - 1].content += &#039;\u89e3\u6790\u54cd\u5e94\u65f6\u53d1\u751f\u9519\u8bef&#039;;\n  }\n\n  return true; \/\/ \u8fd4\u56de\u6210\u529f\u72b6\u6001\n}\n\nasync function startStream(index, body, url) {\n  let response = null;\n\n  try {\n    response = await fetch(url, {\n      method: &quot;POST&quot;,\n      headers: {\n        &quot;Content-Type&quot;: &quot;application\/json&quot;,\n        &quot;Accept&quot;: &quot;text\/event-stream&quot;,\n        &quot;Accept-Encoding&quot;: &quot;gzip, deflate, br, zstd&quot;,\n        &quot;Accept-Language&quot;: &quot;zh-CN,zh;q=0.9&quot;\n      },\n      mode: &quot;cors&quot;,\n      body: JSON.stringify(body),\n    });\n\n    left_data.chat[index].msg_list[left_data.chat[index].msg_list.length - 1].msgload = false;\n  } catch (error) {\n    left_data.chat[index].msg_list[left_data.chat[index].msg_list.length - 1].content += `\u53d1\u751f\u4e86\u4e00\u4e9b\u9519\u8bef\uff1a${error.message}`;\n    return false;\n  }\n\n  if (response.status !== 200) {\n    left_data.chat[index].msg_list[left_data.chat[index].msg_list.length - 1].content += `\u53d1\u751f\u4e86\u4e00\u4e9b\u9519\u8bef\uff1a${response.status}-${response.statusText}`;\n    return false;\n  }\n\n  const reader = response.body.getReader();\n  const decoder = new TextDecoder(&#039;utf-8&#039;);\n  let buffer = &#039;&#039;;\n\n  const readStream = async () =&gt; {\n    const { done, value } = await reader.read();\n\n    if (done) {\n      console.log(&#039;Stream reading complete&#039;);\n      return;\n    }\n\n    buffer += decoder.decode(value, { stream: true });\n\n    let completeData = &#039;&#039;;\n    let separatorIndex;\n    while ((separatorIndex = buffer.indexOf(&#039;\\n&#039;)) !== -1) {\n      completeData = buffer.slice(0, separatorIndex).trim();\n      buffer = buffer.slice(separatorIndex + 1);\n\n      if (completeData.startsWith(&#039;data: &#039;)) {\n        const jsonString = completeData.slice(6); \/\/ \u53bb\u6389 &#039;data: &#039;\n        try {\n          const data = JSON.parse(jsonString);\n          data.ops.forEach(op =&gt; {\n            if (op.op === &#039;add&#039; &amp;&amp; op.path.includes(&#039;\/streamed_output&#039;)) {\n              const outputText = op.value;\n              if (outputText) {\n                const currentContent = left_data.chat[index].msg_list[left_data.chat[index].msg_list.length - 1].content;\n                if (!currentContent.includes(outputText)) {\n                  left_data.chat[index].msg_list[left_data.chat[index].msg_list.length - 1].content += outputText;\n                }\n              }\n            }\n          });\n        } catch (e) {\n          console.error(&#039;Error parsing JSON:&#039;, e, &#039;Complete Data:&#039;, completeData);\n        }\n      } else {\n        console.log(&#039;No valid JSON found in:&#039;, completeData);\n      }\n    }\n\n    return readStream();\n  };\n\n  return readStream();\n}\n\nfunction hasLogin(compomentName = &#039;Login&#039;) {\n    if (compomentName == &#039;login&#039;) return instaceV.proxy.hasLogin ? &#039;hidden&#039; : &#039;&#039;;\n    if (compomentName == &#039;main&#039;) return instaceV.proxy.hasLogin ? &#039;&#039; : &#039;hidden&#039;;\n}\n\nfunction showSettingFunc() {\n    showSetting.value = !showSetting.value\n}\n\n\/\/ \u5220\u9664\u5f53\u524d\u4f1a\u8bdd\u8bb0\u5f55\nfunction deleteChatItemHistory(uuid) {\n    const index = left_data.chat.findIndex(v =&gt; v.uuid == uuid);\n    left_data.chat[index].msg_list = []\n    message.success(&#039;\u5f53\u524d\u4f1a\u8bdd\u8bb0\u5f55\u5df2\u6e05\u7406&#039;)\n}\n\nfunction confirmSettings() {\n  \/\/ \u8fd9\u91cc\u53ef\u4ee5\u6dfb\u52a0\u4fdd\u5b58\u8bbe\u7f6e\u7684\u903b\u8f91\n  console.log(&quot;\u8bbe\u7f6e\u5df2\u786e\u8ba4:&quot;, setting);\n  \/\/ \u4f8b\u5982\uff0c\u4f60\u53ef\u4ee5\u5173\u95ed\u6a21\u6001\u6846\n  showSetting.value = false;\n}\n\n\/\/ \u5f53\u524d\u4f1a\u8bdd\u4e0b\u8f7d\u4e3a\u56fe\u7247\nasync function dom2img() {\n    centerLodding.value = true\n\n    var ele = document.querySelectorAll(&quot;.msgItem&quot;)\n    var msgAreaDom = document.getElementById(&quot;msgArea&quot;)\n\n    const width = msgAreaDom.offsetWidth * 2\n    const height = msgAreaDom.scrollHeight * 1.5\n\n    let canvas1 = document.createElement(&#039;canvas&#039;);\n    let context = canvas1.getContext(&#039;2d&#039;);\n    canvas1.width = width;\n    canvas1.height = height;\n    \/\/ \u7ed8\u5236\u77e9\u5f62\u6dfb\u52a0\u767d\u8272\u80cc\u666f\u8272\n    context.rect(0, 0, width, height);\n    context.fillStyle = &quot;#fff&quot;;\n    context.fill();\n\n    let beforeHeight = 0\n    for (let i = 0; i &lt; ele.length; i++) {\n        const dom_canvas = await html2canvas(ele[i], {\n            scrollX: 0,\n            scrollY: 0,\n            height: ele[i].scrollHeight,\n            width: ele[i].scrollWidth,\n        })\n\n        \/\/ var image = dom_canvas.toDataURL(&quot;image\/png&quot;);\n        context.drawImage(dom_canvas, 0, beforeHeight, dom_canvas.width, dom_canvas.height)\n        beforeHeight = beforeHeight + dom_canvas.height;\n\n    }\n    var image = canvas1.toDataURL(&quot;image\/png&quot;).replace(&quot;image\/png&quot;, &quot;image\/octet-stream&quot;);\n    var link = document.getElementById(&quot;link&quot;);\n    link.setAttribute(&quot;download&quot;, `chatweb-${(new Date()).getTime()}.png`);\n    link.setAttribute(&quot;href&quot;, image);\n    link.click();\n\n    centerLodding.value = false\n    message.success(&#039;\u56fe\u7247\u4e0b\u8f7d\u5b8c\u6210&#039;)\n\n}\n\n&lt;\/script&gt;<\/code><\/pre>\n<h3><span class=\"ez-toc-section\" id=\"%E6%9C%8D%E5%8A%A1%E9%83%A8%E7%BD%B2-2\"><\/span>\u670d\u52a1\u90e8\u7f72<span class=\"ez-toc-section-end\"><\/span><\/h3>\n<pre><code class=\"language-bash\"># \u5207\u6362\u76ee\u5f55\u5230chatweb\u76ee\u5f55\ncd chatweb\n\n# \u5b89\u88c5\u4f9d\u8d56\u7684\u7ec4\u4ef6\nnpm install\n\n# \u542f\u52a8\u670d\u52a1\nnpm run dev<\/code><\/pre>\n<p>\u8fd0\u884c\u6548\u679c\uff1a<br \/>\n<a href=\"https:\/\/17aitech.com\/wp-content\/uploads\/2024\/09\/vue\u542f\u52a8\u670d\u52a1.png\" data-fancybox=\"images\" data-fancybox=\"gallery\"><img decoding=\"async\" src=\"https:\/\/17aitech.com\/wp-content\/uploads\/2024\/09\/vue\u542f\u52a8\u670d\u52a1.png\" alt=\"\" \/><\/a><\/p>\n<h3><span class=\"ez-toc-section\" id=\"%E6%8E%A5%E5%8F%A3%E8%81%94%E8%B0%83\"><\/span>\u63a5\u53e3\u8054\u8c03<span class=\"ez-toc-section-end\"><\/span><\/h3>\n<p>\u63a5\u4e0b\u6765\uff0c\u901a\u8fc7\u6d4f\u89c8\u5668\u7684\u5f00\u53d1\u8005\u5de5\u5177\u8fdb\u884c\u63a5\u53e3\u8c03\u8bd5<\/p>\n<ol>\n<li>\n<p>\u5728\u6d4f\u89c8\u5668\u4e2d\u6253\u5f00 <a href=\"http:\/\/localhost:5173\/\">http:\/\/localhost:5173\/<\/a><\/p>\n<\/li>\n<li>\n<p>\u70b9\u51fb\u5f00\u59cb\u6309\u94ae\u5207\u6362\u9875\u9762\u81f3\u804a\u5929\u9875\u9762\uff0c\u6253\u5f00\u6d4f\u89c8\u5668\u7684\u5f00\u53d1\u8005\u5de5\u5177<br \/>\n<a href=\"https:\/\/17aitech.com\/wp-content\/uploads\/2024\/09\/\u5f00\u53d1\u8005\u5de5\u5177\u663e\u793a.png\" data-fancybox=\"images\" data-fancybox=\"gallery\"><img decoding=\"async\" src=\"https:\/\/17aitech.com\/wp-content\/uploads\/2024\/09\/\u5f00\u53d1\u8005\u5de5\u5177\u663e\u793a.png\" alt=\"\" \/><\/a><\/p>\n<\/li>\n<li>\n<p>\u8f93\u5165\u95ee\u9898\u540e\u70b9\u51fb\u53d1\u9001\uff0c\u89c2\u5bdf\u53f3\u4fa7\u5f00\u53d1\u8005\u5de5\u5177\u7684Console\u9762\u677f\uff0c\u89c2\u5bdf\u8bf7\u6c42\u53d1\u9001\u60c5\u51b5\uff0c\u4ee5\u53ca\u8fd4\u56de\u7684\u54cd\u5e94\u6570\u636e\u3002<br \/>\n<a href=\"https:\/\/17aitech.com\/wp-content\/uploads\/2024\/09\/\u63a5\u53e3\u8054\u8c03.png\" data-fancybox=\"images\" data-fancybox=\"gallery\"><img decoding=\"async\" src=\"https:\/\/17aitech.com\/wp-content\/uploads\/2024\/09\/\u63a5\u53e3\u8054\u8c03.png\" alt=\"\" \/><\/a><\/p>\n<\/li>\n<\/ol>\n<blockquote>\n<p>\u5177\u4f53\u8c03\u8bd5\u9047\u5230\u7684\u95ee\u9898\u4e0d\u5c3d\u76f8\u540c\uff0c\u6240\u4ee5\u672c\u7ae0\u4e0d\u518d\u8d58\u8ff0\u5177\u4f53\u8c03\u8bd5\u65b9\u5f0f\uff0c\u8bf7\u8bfb\u8005\u6839\u636e\u95ee\u9898\u81ea\u884c\u8c03\u8bd5\u3002<\/p>\n<\/blockquote>\n<p>\u5f85\u6574\u4f53\u8c03\u8bd5\u901a\u8fc7\u4e4b\u540e\uff0c\u95ee\u7b54\u7cfb\u7edf\u7684\u524d\u540e\u7aef\u5373\u5b8c\u6210\u4e3b\u6d41\u7a0b\u3002<\/p>\n<h2><span class=\"ez-toc-section\" id=\"%E5%86%85%E5%AE%B9%E5%B0%8F%E7%BB%93\"><\/span>\u5185\u5bb9\u5c0f\u7ed3<span class=\"ez-toc-section-end\"><\/span><\/h2>\n<ul>\n<li>\u540e\u7aef\u670d\u52a1\u53ef\u4ee5\u4f7f\u7528langserve\u90e8\u7f72\n<ul>\n<li>\u7b2c\u4e00\u6b65\uff1a\u4f7f\u7528langserve\u65f6\uff0c\u9700\u8981pip install langserve\u7684\u76f8\u5173\u7ec4\u4ef6<\/li>\n<li>\u7b2c\u4e8c\u6b65\uff1a\u521b\u5efa\u5bf9\u5e94\u7684server.py\u6587\u4ef6\uff0c\u5728\u4ee3\u7801\u4e2d\u6dfb\u52a0FastAPI\u7684API\u63a5\u53e3<\/li>\n<li>\u7b2c\u4e09\u6b65\uff1a\u901a\u8fc7python server.py\u542f\u52a8\u670d\u52a1\uff0c\u901a\u8fc7\u6d4f\u89c8\u5668\u8bbf\u95ee\u5bf9\u5e94\u5730\u5740\u5373\u53ef<\/li>\n<li>\u7b2c\u56db\u6b65\uff1a\u4e3a\u4e86\u4fbf\u4e8e\u5feb\u901f\u5b9a\u4f4d\u95ee\u9898\uff0c\u53ef\u4ee5\u901a\u8fc7curl \u547d\u4ee4\u8fdb\u884c\u540e\u7aef\u63a5\u53e3\u6d4b\u8bd5<\/li>\n<\/ul>\n<\/li>\n<li>\u524d\u7aef\u670d\u52a1\u53ef\u4ee5\u4f7f\u7528vue.js\u8fdb\u884c\u5feb\u901f\u5f00\u53d1\n<ul>\n<li>\u7b2c\u4e00\u6b65\uff1a\u4e3a\u4e86\u5feb\u901f\u642d\u5efa\u524d\u7aef\u9875\u9762\uff0c\u6211\u4eec\u53ef\u4ee5\u5728github\u4e0a\u641c\u7d22\u76f8\u5173\u7684vue.js\u5f00\u6e90\u9879\u76ee<\/li>\n<li>\u7b2c\u4e8c\u6b65\uff1a\u57fa\u4e8e\u5f00\u6e90\u9879\u76ee\uff0c\u8fdb\u884c\u5feb\u901f\u642d\u5efa\uff0c\u5e76\u6dfb\u52a0\u81ea\u5df1\u7684\u4e1a\u52a1\u903b\u8f91<\/li>\n<li>\u7b2c\u4e09\u6b65\uff1a\u901a\u8fc7\u6d4f\u89c8\u5668\u8c03\u8bd5\uff0c\u89c2\u5bdf\u8bf7\u6c42\u548c\u54cd\u5e94\u60c5\u51b5\uff0c\u8c03\u8bd5\u95ee\u9898<\/li>\n<\/ul>\n<\/li>\n<\/ul>\n<h2><span class=\"ez-toc-section\" id=\"%E9%99%84%E5%BD%95\"><\/span>\u9644\u5f55<span class=\"ez-toc-section-end\"><\/span><\/h2>\n<p>\u6b22\u8fce\u67e5\u770b\u8be5\u7cfb\u5217\u7684\u5176\u4ed6\u6587\u7ae0\uff1a<\/p>\n<ul>\n<li><a href=\"https:\/\/17aitech.com\/?p=25830\">\u3010\u9879\u76ee\u5b9e\u6218\u3011\u57fa\u4e8eAgent\u7684\u91d1\u878d\u95ee\u7b54\u7cfb\u7edf\uff1a\u9879\u76ee\u7b80\u4ecb<\/a><\/li>\n<li><a href=\"https:\/\/17aitech.com\/?p=25969\">\u3010\u9879\u76ee\u5b9e\u6218\u3011\u57fa\u4e8eAgent\u7684\u91d1\u878d\u95ee\u7b54\u7cfb\u7edf\uff1aRAG\u68c0\u7d22\u6a21\u5757\u521d\u5efa\u6210<\/a><\/li>\n<li><a href=\"https:\/\/17aitech.com\/?p=25994\">\u3010\u9879\u76ee\u5b9e\u6218\u3011\u57fa\u4e8eAgent\u7684\u91d1\u878d\u95ee\u7b54\u7cfb\u7edf\uff1aAgent\u6846\u67b6\u7684\u6784\u5efa<\/a><\/li>\n<li><a href=\"https:\/\/17aitech.com\/?p=26137\">\u3010\u9879\u76ee\u5b9e\u6218\u3011\u57fa\u4e8eAgent\u7684\u91d1\u878d\u95ee\u7b54\u7cfb\u7edf\uff1a\u524d\u540e\u7aef\u6d41\u7a0b\u6253\u901a<\/a><\/li>\n<li><a href=\"https:\/\/17aitech.com\/?p=26289\">\u3010\u9879\u76ee\u5b9e\u6218\u3011\u57fa\u4e8eAgent\u7684\u91d1\u878d\u95ee\u7b54\u7cfb\u7edf\uff1a\u4ee3\u7801\u91cd\u6784<\/a><\/li>\n<li><a href=\"https:\/\/17aitech.com\/?p=26700\">\u3010\u9879\u76ee\u5b9e\u6218\u3011\u57fa\u4e8eAgent\u7684\u91d1\u878d\u95ee\u7b54\u7cfb\u7edf\uff1aRAG\u7684\u68c0\u7d22\u589e\u5f3a\u4e4bElasticSearch<\/a><\/li>\n<li><a href=\"https:\/\/17aitech.com\/?p=26930\">\u3010\u9879\u76ee\u5b9e\u6218\u3011\u57fa\u4e8eAgent\u7684\u91d1\u878d\u95ee\u7b54\u7cfb\u7edf\uff1aRAG\u7684\u68c0\u7d22\u589e\u5f3a\u4e4b\u4e0a\u4e0b\u6587\u91cd\u6392\u548c\u538b\u7f29<\/a><\/li>\n<\/ul>\n<p align=\"center\">\u6b22\u8fce\u5173\u6ce8\u516c\u4f17\u53f7\u4ee5\u83b7\u5f97\u6700\u65b0\u7684\u6587\u7ae0\u548c\u65b0\u95fb<\/p>\n<p><a href=\"https:\/\/17aitech.com\/wp-content\/uploads\/2024\/09\/\u626b\u7801_\u641c\u7d22\u8054\u5408\u4f20\u64ad\u6837\u5f0f-\u767d\u8272\u7248.bmp\" data-fancybox=\"images\" data-fancybox=\"gallery\"><img decoding=\"async\" src=\"https:\/\/17aitech.com\/wp-content\/uploads\/2024\/09\/\u626b\u7801_\u641c\u7d22\u8054\u5408\u4f20\u64ad\u6837\u5f0f-\u767d\u8272\u7248.bmp\" alt=\"\" \/><\/a><\/p>\n","protected":false},"excerpt":{"rendered":"<p>\u524d\u8a00 \u5728\u4e0a\u4e00\u7ae0\u300a\u3010\u9879\u76ee\u5b9e\u6218\u3011\u57fa\u4e8eAgen [&hellip;]<\/p>\n","protected":false},"author":1,"featured_media":28270,"comment_status":"open","ping_status":"open","sticky":false,"template":"","format":"aside","meta":{"site-sidebar-layout":"default","site-content-layout":"","ast-site-content-layout":"default","site-content-style":"default","site-sidebar-style":"default","ast-global-header-display":"","ast-banner-title-visibility":"","ast-main-header-display":"","ast-hfb-above-header-display":"","ast-hfb-below-header-display":"","ast-hfb-mobile-header-display":"","site-post-title":"","ast-breadcrumbs-content":"","ast-featured-img":"","footer-sml-layout":"","theme-transparent-header-meta":"default","adv-header-id-meta":"","stick-header-meta":"default","header-above-stick-meta":"","header-main-stick-meta":"","header-below-stick-meta":"","astra-migrate-meta-layouts":"set","ast-page-background-enabled":"default","ast-page-background-meta":{"desktop":{"background-color":"var(--ast-global-color-4)","background-image":"","background-repeat":"repeat","background-position":"center center","background-size":"auto","background-attachment":"scroll","background-type":"","background-media":"","overlay-type":"","overlay-color":"","overlay-opacity":"","overlay-gradient":""},"tablet":{"background-color":"","background-image":"","background-repeat":"repeat","background-position":"center center","background-size":"auto","background-attachment":"scroll","background-type":"","background-media":"","overlay-type":"","overlay-color":"","overlay-opacity":"","overlay-gradient":""},"mobile":{"background-color":"","background-image":"","background-repeat":"repeat","background-position":"center center","background-size":"auto","background-attachment":"scroll","background-type":"","background-media":"","overlay-type":"","overlay-color":"","overlay-opacity":"","overlay-gradient":""}},"ast-content-background-meta":{"desktop":{"background-color":"var(--ast-global-color-5)","background-image":"","background-repeat":"repeat","background-position":"center center","background-size":"auto","background-attachment":"scroll","background-type":"","background-media":"","overlay-type":"","overlay-color":"","overlay-opacity":"","overlay-gradient":""},"tablet":{"background-color":"var(--ast-global-color-5)","background-image":"","background-repeat":"repeat","background-position":"center center","background-size":"auto","background-attachment":"scroll","background-type":"","background-media":"","overlay-type":"","overlay-color":"","overlay-opacity":"","overlay-gradient":""},"mobile":{"background-color":"var(--ast-global-color-5)","background-image":"","background-repeat":"repeat","background-position":"center center","background-size":"auto","background-attachment":"scroll","background-type":"","background-media":"","overlay-type":"","overlay-color":"","overlay-opacity":"","overlay-gradient":""}},"footnotes":""},"categories":[28],"tags":[70,68,72],"class_list":["post-26137","post","type-post","status-publish","format-aside","has-post-thumbnail","hentry","category-blog","tag-agent","tag-68","tag-72","post_format-post-format-aside"],"yoast_head":"<!-- This site is optimized with the Yoast SEO plugin v26.4 - https:\/\/yoast.com\/wordpress\/plugins\/seo\/ -->\n<title>\u3010\u9879\u76ee\u5b9e\u6218\u3011\u57fa\u4e8eAgent\u7684\u91d1\u878d\u95ee\u7b54\u7cfb\u7edf\uff1a\u524d\u540e\u7aef\u6d41\u7a0b\u6253\u901a - \u4e00\u8d77AI\u6280\u672f<\/title>\n<meta name=\"robots\" content=\"index, follow, max-snippet:-1, max-image-preview:large, max-video-preview:-1\" \/>\n<link rel=\"canonical\" href=\"https:\/\/17aitech.com\/?p=26137\" \/>\n<script type=\"application\/ld+json\" class=\"yoast-schema-graph\">{\"@context\":\"https:\/\/schema.org\",\"@graph\":[{\"@type\":\"WebPage\",\"@id\":\"https:\/\/17aitech.com\/?p=26137\",\"url\":\"https:\/\/17aitech.com\/?p=26137\",\"name\":\"\u3010\u9879\u76ee\u5b9e\u6218\u3011\u57fa\u4e8eAgent\u7684\u91d1\u878d\u95ee\u7b54\u7cfb\u7edf\uff1a\u524d\u540e\u7aef\u6d41\u7a0b\u6253\u901a - \u4e00\u8d77AI\u6280\u672f\",\"isPartOf\":{\"@id\":\"https:\/\/17aitech.com\/#website\"},\"primaryImageOfPage\":{\"@id\":\"https:\/\/17aitech.com\/?p=26137#primaryimage\"},\"image\":{\"@id\":\"https:\/\/17aitech.com\/?p=26137#primaryimage\"},\"thumbnailUrl\":\"https:\/\/17aitech.com\/wp-content\/uploads\/2024\/09\/\u5c01\u9762.jpeg\",\"datePublished\":\"2024-09-18T03:54:52+00:00\",\"dateModified\":\"2024-10-08T07:06:08+00:00\",\"author\":{\"@id\":\"https:\/\/17aitech.com\/#\/schema\/person\/3d23bb6f7f115fcefc9ae7803a691739\"},\"breadcrumb\":{\"@id\":\"https:\/\/17aitech.com\/?p=26137#breadcrumb\"},\"inLanguage\":\"zh-Hans\",\"potentialAction\":[{\"@type\":\"ReadAction\",\"target\":[\"https:\/\/17aitech.com\/?p=26137\"]}]},{\"@type\":\"ImageObject\",\"inLanguage\":\"zh-Hans\",\"@id\":\"https:\/\/17aitech.com\/?p=26137#primaryimage\",\"url\":\"https:\/\/17aitech.com\/wp-content\/uploads\/2024\/09\/\u5c01\u9762.jpeg\",\"contentUrl\":\"https:\/\/17aitech.com\/wp-content\/uploads\/2024\/09\/\u5c01\u9762.jpeg\",\"width\":660,\"height\":281},{\"@type\":\"BreadcrumbList\",\"@id\":\"https:\/\/17aitech.com\/?p=26137#breadcrumb\",\"itemListElement\":[{\"@type\":\"ListItem\",\"position\":1,\"name\":\"\u9996\u9875\",\"item\":\"https:\/\/17aitech.com\/\"},{\"@type\":\"ListItem\",\"position\":2,\"name\":\"\u3010\u9879\u76ee\u5b9e\u6218\u3011\u57fa\u4e8eAgent\u7684\u91d1\u878d\u95ee\u7b54\u7cfb\u7edf\uff1a\u524d\u540e\u7aef\u6d41\u7a0b\u6253\u901a\"}]},{\"@type\":\"WebSite\",\"@id\":\"https:\/\/17aitech.com\/#website\",\"url\":\"https:\/\/17aitech.com\/\",\"name\":\"\u4e00\u8d77AI\u6280\u672f\",\"description\":\"\u8ba9AI\u77e5\u8bc6\u89e6\u624b\u53ef\u53ca\",\"alternateName\":\"\u4e00\u8d77AI\u6280\u672f\",\"potentialAction\":[{\"@type\":\"SearchAction\",\"target\":{\"@type\":\"EntryPoint\",\"urlTemplate\":\"https:\/\/17aitech.com\/?s={search_term_string}\"},\"query-input\":{\"@type\":\"PropertyValueSpecification\",\"valueRequired\":true,\"valueName\":\"search_term_string\"}}],\"inLanguage\":\"zh-Hans\"},{\"@type\":\"Person\",\"@id\":\"https:\/\/17aitech.com\/#\/schema\/person\/3d23bb6f7f115fcefc9ae7803a691739\",\"name\":\"Dongming\",\"image\":{\"@type\":\"ImageObject\",\"inLanguage\":\"zh-Hans\",\"@id\":\"https:\/\/17aitech.com\/#\/schema\/person\/image\/\",\"url\":\"\/\/17aitech.com\/wp-content\/uploads\/member\/avatars\/238a0b923820dcc5.1732798681.jpg\",\"contentUrl\":\"\/\/17aitech.com\/wp-content\/uploads\/member\/avatars\/238a0b923820dcc5.1732798681.jpg\",\"caption\":\"Dongming\"},\"description\":\"\u89c1\u5929\u5730\uff0c\u89c1\u4f17\u751f\uff0c\u89c1\u81ea\u5df1\u3002\",\"sameAs\":[\"http:\/\/17aitech.com\"],\"url\":\"https:\/\/17aitech.com\/?page_id=33738&user=1\"}]}<\/script>\n<!-- \/ Yoast SEO plugin. -->","yoast_head_json":{"title":"\u3010\u9879\u76ee\u5b9e\u6218\u3011\u57fa\u4e8eAgent\u7684\u91d1\u878d\u95ee\u7b54\u7cfb\u7edf\uff1a\u524d\u540e\u7aef\u6d41\u7a0b\u6253\u901a - \u4e00\u8d77AI\u6280\u672f","robots":{"index":"index","follow":"follow","max-snippet":"max-snippet:-1","max-image-preview":"max-image-preview:large","max-video-preview":"max-video-preview:-1"},"canonical":"https:\/\/17aitech.com\/?p=26137","schema":{"@context":"https:\/\/schema.org","@graph":[{"@type":"WebPage","@id":"https:\/\/17aitech.com\/?p=26137","url":"https:\/\/17aitech.com\/?p=26137","name":"\u3010\u9879\u76ee\u5b9e\u6218\u3011\u57fa\u4e8eAgent\u7684\u91d1\u878d\u95ee\u7b54\u7cfb\u7edf\uff1a\u524d\u540e\u7aef\u6d41\u7a0b\u6253\u901a - \u4e00\u8d77AI\u6280\u672f","isPartOf":{"@id":"https:\/\/17aitech.com\/#website"},"primaryImageOfPage":{"@id":"https:\/\/17aitech.com\/?p=26137#primaryimage"},"image":{"@id":"https:\/\/17aitech.com\/?p=26137#primaryimage"},"thumbnailUrl":"https:\/\/17aitech.com\/wp-content\/uploads\/2024\/09\/\u5c01\u9762.jpeg","datePublished":"2024-09-18T03:54:52+00:00","dateModified":"2024-10-08T07:06:08+00:00","author":{"@id":"https:\/\/17aitech.com\/#\/schema\/person\/3d23bb6f7f115fcefc9ae7803a691739"},"breadcrumb":{"@id":"https:\/\/17aitech.com\/?p=26137#breadcrumb"},"inLanguage":"zh-Hans","potentialAction":[{"@type":"ReadAction","target":["https:\/\/17aitech.com\/?p=26137"]}]},{"@type":"ImageObject","inLanguage":"zh-Hans","@id":"https:\/\/17aitech.com\/?p=26137#primaryimage","url":"https:\/\/17aitech.com\/wp-content\/uploads\/2024\/09\/\u5c01\u9762.jpeg","contentUrl":"https:\/\/17aitech.com\/wp-content\/uploads\/2024\/09\/\u5c01\u9762.jpeg","width":660,"height":281},{"@type":"BreadcrumbList","@id":"https:\/\/17aitech.com\/?p=26137#breadcrumb","itemListElement":[{"@type":"ListItem","position":1,"name":"\u9996\u9875","item":"https:\/\/17aitech.com\/"},{"@type":"ListItem","position":2,"name":"\u3010\u9879\u76ee\u5b9e\u6218\u3011\u57fa\u4e8eAgent\u7684\u91d1\u878d\u95ee\u7b54\u7cfb\u7edf\uff1a\u524d\u540e\u7aef\u6d41\u7a0b\u6253\u901a"}]},{"@type":"WebSite","@id":"https:\/\/17aitech.com\/#website","url":"https:\/\/17aitech.com\/","name":"\u4e00\u8d77AI\u6280\u672f","description":"\u8ba9AI\u77e5\u8bc6\u89e6\u624b\u53ef\u53ca","alternateName":"\u4e00\u8d77AI\u6280\u672f","potentialAction":[{"@type":"SearchAction","target":{"@type":"EntryPoint","urlTemplate":"https:\/\/17aitech.com\/?s={search_term_string}"},"query-input":{"@type":"PropertyValueSpecification","valueRequired":true,"valueName":"search_term_string"}}],"inLanguage":"zh-Hans"},{"@type":"Person","@id":"https:\/\/17aitech.com\/#\/schema\/person\/3d23bb6f7f115fcefc9ae7803a691739","name":"Dongming","image":{"@type":"ImageObject","inLanguage":"zh-Hans","@id":"https:\/\/17aitech.com\/#\/schema\/person\/image\/","url":"\/\/17aitech.com\/wp-content\/uploads\/member\/avatars\/238a0b923820dcc5.1732798681.jpg","contentUrl":"\/\/17aitech.com\/wp-content\/uploads\/member\/avatars\/238a0b923820dcc5.1732798681.jpg","caption":"Dongming"},"description":"\u89c1\u5929\u5730\uff0c\u89c1\u4f17\u751f\uff0c\u89c1\u81ea\u5df1\u3002","sameAs":["http:\/\/17aitech.com"],"url":"https:\/\/17aitech.com\/?page_id=33738&user=1"}]}},"_links":{"self":[{"href":"https:\/\/17aitech.com\/index.php?rest_route=\/wp\/v2\/posts\/26137","targetHints":{"allow":["GET"]}}],"collection":[{"href":"https:\/\/17aitech.com\/index.php?rest_route=\/wp\/v2\/posts"}],"about":[{"href":"https:\/\/17aitech.com\/index.php?rest_route=\/wp\/v2\/types\/post"}],"author":[{"embeddable":true,"href":"https:\/\/17aitech.com\/index.php?rest_route=\/wp\/v2\/users\/1"}],"replies":[{"embeddable":true,"href":"https:\/\/17aitech.com\/index.php?rest_route=%2Fwp%2Fv2%2Fcomments&post=26137"}],"version-history":[{"count":4,"href":"https:\/\/17aitech.com\/index.php?rest_route=\/wp\/v2\/posts\/26137\/revisions"}],"predecessor-version":[{"id":33210,"href":"https:\/\/17aitech.com\/index.php?rest_route=\/wp\/v2\/posts\/26137\/revisions\/33210"}],"wp:featuredmedia":[{"embeddable":true,"href":"https:\/\/17aitech.com\/index.php?rest_route=\/wp\/v2\/media\/28270"}],"wp:attachment":[{"href":"https:\/\/17aitech.com\/index.php?rest_route=%2Fwp%2Fv2%2Fmedia&parent=26137"}],"wp:term":[{"taxonomy":"category","embeddable":true,"href":"https:\/\/17aitech.com\/index.php?rest_route=%2Fwp%2Fv2%2Fcategories&post=26137"},{"taxonomy":"post_tag","embeddable":true,"href":"https:\/\/17aitech.com\/index.php?rest_route=%2Fwp%2Fv2%2Ftags&post=26137"}],"curies":[{"name":"wp","href":"https:\/\/api.w.org\/{rel}","templated":true}]}}