• Vue3+NodeJS 接入文心一言, 发布一个 VSCode 大模型问答插件


    目录

    一:首先明确插件开发方式

    二:新建一个Vscode 插件项目

    1. 官网教程地址

    2. 一步一步来创建

    3. 分析目录结构以及运行插件

    三:新建一个Vue3 项目,在侧边栏中展示,实现vscode插件 <=> vue项目 双向消息传递

    1. 新建vue3+vite+ts项目

    2. 将web页面展示在vscode侧边栏

    (1) 插件项目修改,把视图注册到侧边栏,完成消息传递

    (2) web项目修改,增加事件监听

    四:接入大模型对话能力,实现ChatUI

    1. 大模型接入准备

    2. nodejs调用api

    3. 前端接口调试

    五:注册开发者账号并发布插件

    1. 推荐教程

    2. 增加插件商店图标

    3. 前端资源的缓存策略会影响插件web页面的实时更新

    六:实战能力探讨(会持续更新,欢迎探讨)

    1. 行内提示功能的设计与实现(InlineCompletionItemProvider[9])

    2. SSH 远程打开文件能力(使用 remote-ssh 插件提供的命令)

    3. Json 文件可视化编辑(JsonToHtml)


            随着大模型能力越来越卷,在垂直领域的落地也在加快,对于大模型代码生成能力而言,最简洁高效的方式就是集成为常用IDE的插件,在vscode的插件战场中,比较知名的就有 GitHub Copilot, 智谱清言的codegeex, 讯飞星火的iFlyCode。

    那么我们就以开发一个简易的大模型对话插件,来探究一下vscode插件开发到发布的流程,研究一下文心一言大模型api的接入

    跟着操作大约30-60分钟,你需要

    • 安装vscode,npm/yarn等,node版本 >12.0

    • 最好有时间提前看一看 vscode 官网api开发文档[1]

    万字长文 Action!

    一:首先明确插件开发方式

    1. 如果你的插件只提供原生vscode能力,没有复杂的UI需求,只需要在vscode插件项目上开发即可,类似插件比如Volar Git History Eslint

    2. 如果提供复杂UI交互,定制化界面,就需要在vscode插件内嵌iframe页面(用iframe展示线上web地址与使用vscode提供的一套UI组件皆可,详见第三节),我这里选择访问线上地址,因此需要开发一个vscode插件项目与一个vue3项目(其他框架亦可),类似的复杂插件比如 CodeGeeX iFlyCode,会将web页面展示在侧边栏中。

    图片

    image.png

    本文主要讲解 如何在vscode插件中通过iframe展示web页面,获得更好的拓展性与可维护性

    二:新建一个Vscode 插件项目

    1. 官网教程地址

    开始你的第一个插件项目[2]

    2. 一步一步来创建
    • 找到一个比较舒服的文件夹,打开cmd,通过以下命令安装 vscode项目脚手架,取的是 registry.npmjs.org[3] 镜像源,因此可能会有科学问题

      npm install -g yo generator-code
    
    • 安装完成后,直接用命令创建新的插件项目

     yo code
    
    • 进入配置页面,默认就选择 NewExtension(TypeScript),后面的按照图中来就可

      图片

    • 然后会自动创建好项目,并执行npm i,然后用 vscode 打开项目

    3. 分析目录结构以及运行插件

    图片

    目录结构就很清晰了,我们主要涉及修改 extension.ts 以及 package.json文件
    上图中,extension.ts 中 activate() 方法就是插件的入口函数,每次插件启动都会执行此函数,当前代码是注册了一个hello world命令,当你在vscode中通过 ctrl+shift+p 调出输入框并输入hello world,就会执行此注册命令的回调,弹出一个message框,下面我们来试一下

    在当前项目中,直接按F5,会启动一个扩展开发宿主,你的插件就运行在这个vscode窗口上啦 下面我们调出命令输入框ctrl+shift+p ,输入 hello world, 会提示命令,选中执行,右下角会发现弹了一个message!!!
    什么?你的没弹出?那你岂不是和我当时一样倒霉,但你不需要花时间去挖这个奇怪的~bug !
    首先看一下你的vscode版本

    图片

    image.png

    当前vscode版本不能低于 package.json 中的最低版本要求!

    图片

    这样写表示最低支持到1.83.0版本!改一下重新reload一下宿主插件,再试试命令就可以弹出啦!到此我们的插件侧项目就搭建好了,下面我们简单建一个vue项目,嵌入到侧边栏中

    三:新建一个Vue3 项目,在侧边栏中展示,实现vscode插件 <=> vue项目 双向消息传递

    文章开头我们提到,插件内展示丰富的UI,既可以用iframe展示线上web网页,也可以在插件内部用vsode ui实现。下面我主要演示用iframe的方式,另一种嵌入方式推荐大家去看一下 CodeGeeX 插件[4]源码如何做的,引入了一套vscode风格的UI组件@vscode/webview-ui-toolkit,源码里面的webviewUI文件夹与translationWebviewProvider.ts文件都是相关代码。

    1. 新建vue3+vite+ts项目

    找一个舒服的文件夹,打开cmd

        npm init vite
    

    执行后按需选择自己的框架与开发环境,然后run dev一下子,拿到地址, 比如 http://localhost:5173/

    2. 将web页面展示在vscode侧边栏
    (1) 插件项目修改,把视图注册到侧边栏,完成消息传递

    第一步当然是先建一个iframe把我们的web项目的地址填进去呗,开始。

    vscode 提供了两种创建iframe的方法,WebviewViewProvider 和 createWebviewPanel,选其一即可,这里我们介绍一下WebviewViewProvider如何使用

    首先在extension.ts 同级目录下新建 chatWebview.ts

    • WebviewViewProvider 是一个接口,因此建一个自己的类实现它的方法即可

      图片

      下面我们创建一个实现WebviewViewProvider接口的类ChatWebview

      chatWebview.ts 文件: (可直接运行)
      具体代码作用看注释

    1. import { window, Position, WebviewView, WebviewViewProvider } from "vscode";
    2. export class ChatWebview implements WebviewViewProvider {
    3.   // 写一个public变量,方便对象引用创建后的webview实例,但是可能存在还未完全解析完成时,访问值为null
    4.   // 看了vscode api发现,resolveWebView 返回一个 Thenable,可以在解析完成后拿到webview实例
    5.   // 但是这个函数是在webview容器第一次显示时自动执行,不需要手动调用,不知道怎么拿到Thenable
    6.   public webview: WebviewView | null = null;
    7.   resolveWebviewView(webviewView: WebviewView): void | Thenable<void> {
    8.     this.webview = webviewView;
    9.     webviewView.webview.options = {
    10.       enableScripts: true,
    11.     };
    12.     // 监听web端传来的消息
    13.     webviewView.webview.onDidReceiveMessage((message) => {
    14.       switch (message.command) {
    15.         case "WebSendMesToVscode":
    16.           // 实现一个简单的功能,将web端传递过来的消息插入到当前活动编辑器中
    17.           let editor = window.activeTextEditor;
    18.           editor?.edit((edit) => {
    19.             let position = editor?.selection
    20.               ? editor?.selection.start
    21.               : new Position(00);
    22.             edit.insert(position, message.data);
    23.           });
    24.           return;
    25.       }
    26.     }, undefined);
    27.     // webview 展示的内容本身就是嵌套在一个iframe中,因此在此html中再嵌套一个iframe时,需要传递两次postMessage
    28.     webviewView.webview.html = `
    29.     <!DOCTYPE html>
    30.       <html lang="en">
    31.       <head>
    32.         <meta charset="UTF-8">
    33.         <meta http-equiv="X-UA-Compatible" content="IE=edge">
    34.         <meta name="viewport" content="width=device-width, initial-scale=1.0">
    35.         <style>
    36.         html,
    37.         body {
    38.             margin: 0;
    39.             padding: 0;
    40.             width: 100%;
    41.             height: 100%;
    42.             background-color:#000000;
    43.             overflow:hidden;
    44.         }
    45.         .webView_iframe {
    46.             width: 100%;
    47.             height: 100%;
    48.             border: none;
    49.         }
    50.         .outer{
    51.           width: 100%;
    52.           height: 100%;
    53.           overflow: hidden;
    54.         }
    55.       </style>
    56.     </head>
    57.     <body>
    58.       <script>
    59.       
    60.       console.log('Hello from the webview!');
    61.       // 向vscode 传递消息的固定写法, vscode 为我们封装好了postMessage
    62.       const vscode = acquireVsCodeApi();
    63.       // 接收来自web页面的消息
    64.       window.addEventListener('message', event => {
    65.           const message = event.data;
    66.           switch (message.command) {
    67.                // 插件传递消息给web端
    68.               case 'vscodeSendMesToWeb':
    69.                   let iframe = document.getElementById('WebviewIframe')
    70.                   WebviewIframe.contentWindow.postMessage(message, "*")
    71.                   console.log("fromWebViewIframe: "+message.data)
    72.                   break;
    73.               // web端发送消息给插件
    74.               case 'WebSendMesToVscode':
    75.                     vscode.postMessage(message);
    76.                     break;
    77.           }
    78.       });
    79.      </script>
    80.         <div class="outer">
    81.            <iframe id='WebviewIframe' class="webView_iframe" sandbox="allow-scripts allow-same-origin allow-forms allow-pointer-lock allow-downloads" allow="cross-origin-isolated; clipboard-read; clipboard-write;" src="http://localhost:5173/"></iframe>
    82.         </div>
    83.     </body>
    84.     </html>
    85.     `;
    86.   }
    87. }

    提供webview视图的类创建好了,然后我们需要在入口函数中实例化一个webview,然后把这个视图注册到vscode侧边栏中

    打开extension.ts文件,修改如下 (代码可直接运行)

    1. // The module 'vscode' contains the VS Code extensibility API
    2. // Import the module and reference it with the alias vscode in your code below
    3. import * as vscode from "vscode";
    4. import { ChatWebview } from "./chatWebview";
    5. // This method is called when your extension is activated
    6. // vscode 插件入口函数,当插件第一次加载时会执行activate
    7. export function activate(context: vscode.ExtensionContext) {
    8.   console.log('Congratulations, your extension "Chat" is now active!');
    9.   // 实现侧边栏的初始化
    10.   // 实例化一个chatWebview
    11.   const chatWebview = new ChatWebview();
    12.   // 注册webview 到id为 Chat-sidebar 的views中,这个id为 Chat-sidebar 的视图我们稍后会在
    13.   // package.json 中声明,先理解为我们要把iframe渲染在那个地方(侧边栏还是标签页)需要在
    14.   // packagea.json 中控制
    15.   context.subscriptions.push(
    16.     vscode.window.registerWebviewViewProvider("Chat-sidebar", chatWebview, {
    17.       webviewOptions: {
    18.         // 这是一个比较有用的配置项,可以确保你的插件在不可见时不会被销毁,建议开启,否侧每次打开都会重新加载一次插件
    19.         retainContextWhenHidden: true,
    20.       },
    21.     })
    22.   );
    23.   // 这里实现了一个简单的功能,在vscode打开的文件中,选中代码时会实时展示在web页面上
    24.   // 监听用户选中文本事件
    25.   vscode.window.onDidChangeTextEditorSelection((event) => {
    26.     const editor = event.textEditor;
    27.     let document = editor.document;
    28.     let selection = editor.selection;
    29.    // 获取当前窗口的文本
    30.     let text = document.getText(selection);
    31.     // 上文提到chatWebview可能为null 因此需要可选链写法,所以这里存在不稳定性,不过测试没问题~
    32.     chatWebview?.webview?.webview.postMessage({
    33.       // 第一次postMessage,下一次在chatWebview文件的iframe中 
    34.       command: "vscodeSendMesToWeb",
    35.       data: text,
    36.     });
    37.   });
    38. }
    39. // This method is called when your extension is deactivated
    40. export function deactivate() {}

    至此,我们实例化了ChatWebview,并将其与视图Chat-siderbar绑定

    下面我们需要在package.json中将视图注册到侧边栏中,并指定名字,图标等 打开package.json 文件,修改如下 将原本的 contributes 字段替换一下

    确保activitybar 中的id,在views中有对应的视图,我们这里id是Chat-sidebar-view,在views就要有对应名字的视图, 并且该视图 Chat-sidebar-view 的id为我们 ChatWebview 绑定的视图id

    1. "contributes": {
    2.     "commands": [],
    3.     "viewsContainers": {
    4.       "activitybar": [
    5.         {
    6.           "id""Chat-sidebar-view",
    7.           "title""Chat",
    8.           "icon""images/vite.svg"
    9.         }
    10.       ]
    11.     },
    12.     "views": {
    13.       "Chat-sidebar-view": [
    14.         {
    15.           "type""webview",
    16.           "id""Chat-sidebar",
    17.           "name"" Chat",
    18.           "icon""images/vite.svg",
    19.           "contextualTitle""Chat"
    20.         }
    21.       ]
    22.     }
    23.   },

    至此!我们的视图和双向通讯在插件侧已经完成了,我们试一下!直接F5运行,打开拓展开发宿主

    图片

    image.png

    点击左侧栏图标,会看见我们deweb页面加载出来啦!
    再试一下选中文本的事件和postMessage通讯,点击上方help,选择倒数第三个Toggle developer tools或者按ctrl+shift+i 可以打开谷歌开发者工具,调试vscode

    随便打开一个项目文件,在窗口中选中文本,会发现控制台一直在输出 fromWebViewIframe: ...... ,我们第一步通讯通了,下面在vue项目中加一下消息接收和发送。

    (2) web项目修改,增加事件监听

    打开index.html,增加message的监听,收到消息时插入到container中

    1. <!DOCTYPE html>
    2. <html lang="en">
    3.   <head>
    4.     <meta charset="UTF-8" />
    5.     <link rel="icon" type="image/svg+xml" href="/vite.svg" />
    6.     <meta name="viewport" content="width=device-width, initial-scale=1.0" />
    7.     <title>Vite + Vue + TS</title>
    8.   </head>
    9.   <body>
    10.     <div id="app"></div>
    11.     <script type="module" src="/src/main.ts"></script>
    12.     <script type="module">
    13.       window.addEventListener("message", (event) => {
    14.         const message = event.data;
    15.         switch (message.command) {
    16.           case "vscodeSendMesToWeb":
    17.             const div = document.getElementById("container");
    18.             div.innerHTML = message.data;
    19.             break;
    20.         }
    21.       });
    22.     </script>
    23.   </body>
    24. </html>

    我们试一下,在拓展开发宿主中选中文本,会实时展示在页面上!

    图片

    image.png

    下面我们发送消息试一下 简单修改一下 HellowWord.vue 组件,增加一个sendMessage 方法

    1. <script setup lang="ts">
    2. import { ref } from "vue";
    3. defineProps<{ msg: string }>();
    4. const count = ref(0);
    5. const sendMessage = () => {
    6.   window.parent.postMessage(
    7.     {
    8.       command: "WebSendMesToVscode",
    9.       data"this message is from vue3",
    10.     },
    11.     "*"
    12.   );
    13. };
    14. </script>
    15. <template>
    16.   <h1>{{ msg }}</h1>
    17.   <div class="card">
    18.     <button type="button" @click="sendMessage">click</button>
    19.     <p>
    20.       Edit
    21.       <code>components/HelloWorld.vue</code> to test HMR
    22.     </p>
    23.   </div>
    24.   <p>
    25.     Check out
    26.     <a href="https://vuejs.org/guide/quick-start.html#local" target="_blank"
    27.       >create-vue</a
    28.     >, the official Vue + Vite starter
    29.   </p>
    30.   <p>
    31.     Install
    32.     <a href="https://github.com/vuejs/language-tools" target="_blank">Volar</a>
    33.     in your IDE for a better DX
    34.   </p>
    35.   <class="read-the-docs">Click on the Vite and Vue logos to learn more</p>
    36. </template>
    37. <style scoped>
    38. .read-the-docs {
    39.   color: #888;
    40. }
    41. </style>

    我们点击一下按钮,会发现在当前文件光标处插入了一条信息。this message is from vue3!

    到此我们的小插件展示出来了,也实现了数据互通。下面我们实现一个简单对话UI,并接入百度文心一言大模型,做一个自己的插件小助手,如果他能记住我们之前问过的代码,并帮我们举一反三,并提醒我们查漏补缺就好了。

    四:接入大模型对话能力,实现ChatUI

    1. 大模型接入准备

    我分别注册了智谱清言(chatGLM)与文心一言(ERNIE-Bot),发现两者都有基础的免费额度,前者相对于后者代码能力貌似更强一些,我们这里做一个简单类似于代码错题本的对话助手,就接入文心一言吧

    首先我们要去官网[5],注册一下开发者账号,并且实名认证 整个过程很简单,然后我们看一下api 文档[6]

    图片

    下面我把主要步骤说一下 首先我们要创建一个自己的应用,获取到Secret KeyAPI Key

    进入下面页面,点击创建应用,输入应用名称和应用描述直接确定即可,然后会有一个应用生成,里面就有我们的Secret KeyAPI Key

    图片

    我们要拿这两个key,去获取 access_token 和 refresh_token, 用于JWT鉴权,有两种方式,其一我们可以在网页中访问一下拿到一次性30天的access_token用于临时测试,其二最好在项目http请求前自动用refresh_token去获取access_token

    下面我们访问一下这个地址,当然你要把双key换成你自己的应用的~!https://aip.baidubce.com/oauth/2.0/token?grant_type=client_credentials&client_id=你的key&client_secret=你的key

    直接用浏览器访问一下呢,然后就在请求里拿到了access_token (不能有小可爱找不到吧~) 拿到token了我们来这里测试一下 测试地址[7]

    图片

    填入标出的这两项,第二项示例如下

    1. [
    2.   {
    3.     "role""user",
    4.     "content""介绍一下自己"
    5.   },
    6. ]

    可以看到接口调用结果

    图片

    image.png

    2. nodejs调用api

    首先找一个舒服的文件夹,新建一个node项目,我们这里选用express框架,可以参考我这里的命令行

    图片

    image.png

    npm init 后会生成一个package.json,然后我们安装一下常用包
    npm install express sequelize mysql2 axios body-parser cors --save 之后就可以在vscode中打开我们的项目了,我们先新建一个server.js,作为我们的入口文件,再建一个chat.js 作为我们的大模型调用文件

    两个文件代码如下,具体解析见注释,可以直接复制过去,然后在控制台执行node server.js 直接启动服务~

    1. //server.js
    2. const Conversation = require("./chat.js");
    3. const express = require("express");
    4. const bodyParser = require("body-parser");
    5. const cors = require("cors");
    6. const app = express();
    7. // 暂时允许所有跨域请求
    8. let corsOptions = {
    9.   origin: "*",
    10. };
    11. app.use(cors(corsOptions));
    12. // content-type:application/json
    13. app.use(bodyParser.json());
    14. // content-type:application/x-www-form-urlencoded
    15. app.use(bodyParser.urlencoded({ extended: true }));
    16. // 对话类
    17. const conversation = new Conversation();
    18. // 定义/chat路由处理POST请求
    19. app.post("/chat", async (req, res) => {
    20.   const { messages = "" } = req.body || {};
    21.   if (typeof messages !== "string") {
    22.     return res.status(400).send({ error"Invalid messages type" });
    23.   }
    24.   try {
    25.     // 调用ask方法获取大模型结果
    26.     const response = await conversation.ask(messages);
    27.     return res.status(200).send({ message: response });
    28.   } catch (error) {
    29.     return res.status(500).send({ errorerror.messages });
    30.   }
    31. });
    32. // 设置监听端口
    33. const PORT = process.env.PORT || 8080;
    34. app.listen(PORT, () => {
    35.   console.log(`服务器运行端口: ${PORT}.`);
    36. });
    1. chat.js
    2. // 访问模型服务
    3. const axios = require("axios");
    4. // 这里就是你的accessToken,我改了两个数,所以你得替换成自己的喽~
    5. const accessToken =
    6.   "24.88635a1444105db00bb6684c0598a9a3.2542000.1741590285.281335-42231960";
    7. const ERNIEB4 =
    8.   "https://aip.baidubce.com/rpc/2.0/ai_custom/v1/wenxinworkshop/chat/completions_pro";
    9. const ERNIEB =
    10.   "https://aip.baidubce.com/rpc/2.0/ai_custom/v1/wenxinworkshop/chat/completions";
    11. class Conversation {
    12.   constructor() {
    13.     // 上下文数据存在这里,文心的调用是需要把所有的历史对话数据全部传过去,所以上下文窗口大小得注意
    14.     this.messages = [];
    15.   }
    16.   async ask(prompt) {
    17.     // 问句push进去
    18.     this.messages.push({ role: "user"content: prompt });
    19.     console.log("message" + this.messages[0]);
    20.     try {
    21.       const res = await axios.post(
    22.         ERNIEB,
    23.         { messages: this.messages },
    24.         { params: { access_token: accessToken } }
    25.       );
    26.       const { data } = res;
    27.       console.log(data);
    28.       // 答案也放进去
    29.       this.messages.push({ role: "assistant"contentdata.result });
    30.       return data.result;
    31.     } catch (error) {
    32.       console.log("调用模型失败" + error);
    33.     }
    34.   }
    35. }
    36. // 导出函数
    37. module.exports = Conversation;

    所以我们的服务起了吗?网页试一下8080呗,通了就可以

    图片

    image.png

    下面我们在前端代码中加一下接口调用,就大功告成啦!

    想必看到这里你也累了,我们去调戏一下 Sydney

    图片

    image.png

    看来一时半会我们还是不可替代的 o_O

    3. 前端接口调试

    言归正传,我们来增加接口调用吧,顺便画一个看得过去的UI界面

    先装一下 npm install @ant-design/icons-vue
    然后把App.vue删一下

    1. // app.vue
    2. <script setup lang="ts">
    3. import ChatUI from "./components/chatUI.vue";
    4. </script>
    5. <template>
    6.   <div class="container">
    7.     <ChatUI />
    8.   </div>
    9. </template>
    10. <style scoped>
    11. .container {
    12.   width: 100%;
    13.   height: 100%;
    14. }
    15. </style>

    无需多言,**chatUI.vue代码**奉上(主要界面gpt画的,我加了接口调用)\

    1. // chatUI.vue
    2. <template>
    3.   <div class="chat-container">
    4.     <div class="messages">
    5.       <div
    6.         v-for="(item, index) in chatList"
    7.         :key="index"
    8.         :class="['message', item.type]"
    9.       >
    10.         <div class="bubble">{{ item.content }}</div>
    11.         <div class="avatar">
    12.           <component
    13.             :is="item.type === 'question' ? UserOutlined : RobotOutlined"
    14.           />
    15.         </div>
    16.       </div>
    17.     </div>
    18.     <div class="input-area">
    19.       <a-input
    20.         v-model:value="inputValue"
    21.         placeholder="Type a message..."
    22.         @pressEnter="handleSend"
    23.       />
    24.       <a-button type="primary" @click="handleSend">send</a-button>
    25.     </div>
    26.   </div>
    27. </template>
    28. <script setup lang="ts">
    29. import axios from "axios";
    30. import { ref } from "vue";
    31. import { UserOutlined, RobotOutlined } from "@ant-design/icons-vue";
    32. const inputValue = ref("");
    33. let chatList = ref<any[]>([]);
    34. const handleSend = () => {
    35.   const question = inputValue.value.trim();
    36.   if (question) {
    37.     getAnswer(question);
    38.     chatList.value.push({ type"question"content: question });
    39.     inputValue.value = ""// 清空输入框
    40.   }
    41. };
    42. function getAnswer(question: string) {
    43.   const URL = "http://localhost:8080/chat";
    44.   const payload = {
    45.     messages: question,
    46.   };
    47.   sendPost(
    48.     URL,
    49.     payload,
    50.     {},
    51.     (res: any=> {
    52.       console.log(res.data.message);
    53.       chatList.value.push({ type"answer"content: res.data.message });
    54.     },
    55.     (err: any=> {
    56.       console.log(err);
    57.     }
    58.   );
    59. }
    60. //post方法
    61. function sendPost(
    62.   url: string,
    63.   dataany,
    64.   headers = {},
    65.   funcSuccess: any,
    66.   funcError: any
    67. ) {
    68.   const headerTem = {
    69.     "content-Type""application/json;charset=UTF-8",
    70.   };
    71.   if (JSON.stringify(headers) != "{}") {
    72.     Object.assign(headerTem, headers);
    73.   }
    74.   axios
    75.     .post(url, data, {
    76.       headers: headerTem,
    77.     })
    78.     .then(function (res) {
    79.       console.log("sendPost res info :", res);
    80.       funcSuccess(res);
    81.     })
    82.     .catch((err) => {
    83.       console.log("sendPost err info :" + err);
    84.       if (funcError) {
    85.         funcError(err);
    86.       }
    87.     });
    88. }
    89. </script>
    90. <style scoped>
    91. .chat-container {
    92.   min-width: 300px;
    93.   height: 100%;
    94.   display: flex;
    95.   flex-direction: column;
    96.   /* background-color: #1e1e1e; */
    97.   border: 1px solid #999;
    98.   border-radius: 8px;
    99. }
    100. .messages {
    101.   height: 650px;
    102.   overflow-y: auto;
    103.   padding: 10px;
    104.   display: flex;
    105.   flex-direction: column;
    106.   gap: 10px;
    107. }
    108. .bubble {
    109.   color: #333;
    110.   text-align: right;
    111.   margin-right: 8px;
    112. }
    113. .input-area {
    114.   display: flex;
    115. }
    116. .message {
    117.   display: flex;
    118.   align-items: center;
    119. }
    120. .question {
    121.   justify-content: flex-end;
    122. }
    123. .answer {
    124.   justify-content: flex-start;
    125. }
    126. </style>

    这个就是简单的调用接口,我就不注释了,我们试一下接口
    我们在vscode中看一下当前效果

    图片

    image.png

    还可以吧,也不能要求AI太高,哈哈,我们问几个问题试试

    图片

    image.png

    啊?文心一言还挺强,紧跟时事哦

    所以你的接口通了吗?通了的话点个赞吧,好人一生平安~
    没通的话原因有点多,代码是没问题的,其他的可以评论区讨论下

    至此我们的聊天小插件算是开发完成了,我们学习了如何创建一个vscode插件,随后搭建了一个vue3项目展示在了侧边栏里,然后我们用nodejs接入了文心一言api,前端调用接口简单实现了对话功能,希望你看完这篇文章有所收获,有所感想!

    五:注册开发者账号并发布插件

    1. 推荐教程

    插件开发手册[8]

    根据教程注册账号,拿到自己的Token,通过vsce publish 1.0.x 来更新版本

    2. 增加插件商店图标

    插件商店的图标是通过读取package.json中的icon来展示的,该字段与publisher同级

    "icon": "images/icon.png",

    3. 前端资源的缓存策略会影响插件web页面的实时更新

    因为插件实时访问的前端服务,当我们更新前端资源时,当然希望插件能同步更新,此时要注意前端资源的缓存策略,最好是配置为 cache-control:no-store no-cache

    六:实战能力探讨(会持续更新,欢迎探讨)

    1. 行内提示功能的设计与实现(InlineCompletionItemProvider[9])

    先说一个思路,就是用InlineCompletionItemProvider实现行内提示

    2. SSH 远程打开文件能力(使用 remote-ssh 插件提供的命令)

    先说一个思路,就是通过在控制台执行 remote-ssh 的命令 :code --reuse-window vscode-remote://ssh-remote+${hostname}${path} 来实现打开远程ssh地址的文件,需要安装remote-ssh插件

    3. Json 文件可视化编辑(JsonToHtml)

    先说一个思路,就是监听用户打开文件夹时的事件,然后再窗口中打开一个新的webview,试用了一些jsonToHtml的包不如自己手动格式化,将bool格式化为checkbox等.

  • 相关阅读:
    Note_First:Hadoop安装部署与测试
    Nginx+keeplived高可用
    数据建设实践之大数据平台(二)安装zookeeper
    本次CTF·泰山杯网络安全的基础知识部分(二)
    浅谈ChatGPT附免费体验地址
    (附源码)计算机毕业设计SSM基于框架的在线健康系统设计与实现
    【Java8新特性】- Stream流
    图片采集下载
    AI视频监控平台EasyCVR接入海康SDK出现异常,该如何解决?
    JavaSE笔记(二)重制版
  • 原文地址:https://blog.csdn.net/lambert00001/article/details/134367468