VSCode插件开发实战:实现一个代码诊断插件

本篇文章给大家分享一个vscode插件开发实战,开发一个代码诊断插件,分析一下基本原理,并一步步实现,希望对大家有所帮助!

VSCode插件开发实战:实现一个代码诊断插件

最近,我们内部出了一份 Code Review 指南,但是 Code Review 过程非常占时间,大家不会太仔细去 review 代码,因此想通过一个插件让开发者在开发阶段就能感知到写法的错误,做出的效果如下图

Kapture 2022-02-10 at 15.36.49.gif

接下来将介绍如何从 0 实现这么一个功能。

基本原理

Visual Studio Code 的编程语言功能扩展是有 Language Server 来实现的,这很好理解,毕竟检查语言功能是耗费性能的,需要另起一个进程来作为语言服务,这就是 Language Server 语言服务器。【推荐学习:《vscode入门教程》】

Language Server 是一种特殊的 Visual Studio Code 扩展,可为许多编程语言提供编辑体验。使用语言服务器,您可以实现自动完成、错误检查(诊断)、跳转到定义以及VS Code 支持的许多其他语言功能。

既然有了服务器提供的语法检查功能,就需要客户端去连接语言服务器,然后和服务器进行交互,比如用户在客户端进行代码编辑时,进行语言检查。具体交互如下:

2.gif

当打开 Vue 文件时会激活插件,此时就会启动 Language Server,当文档发生变化时,语言服务器就会重新诊断代码,并把诊断结果发送给客户端。

代码诊断的效果是出现波浪线,鼠标移上显示提示消息,如果有快速修复,会在弹出提示的窗口下出现快速修复的按钮

动手实现

了解了代码诊断的基本原理之后,开始动手实现,从上面的基本原理可知,我们需要实现两大部分的功能:

客户端与语言服务器交互

语言服务器的诊断和快速修复功能

客户端与语言服务器交互

官方文档 提供了一个示例 – 用于纯文本文件的简单语言服务器,我们可以在这个示例的基础上去修改。

> git clone https://github.com/microsoft/vscode-extension-samples.git> cd vscode-extension-samples/lsp-sample> npm install> npm run compile> code .

首先在 client 建立服务器

// client/src/extension.tsexport function activate(context: ExtensionContext) {    ...    const clientOptions: LanguageClientOptions = {        documentSelector: [{ scheme: 'file', language: 'vue' }], // 打开 vue 文件时才激活        ...    };    client = new LanguageClient(...);    client.start();}

接着在 server/src/server.ts 中,编写于客户端的交互逻辑,比如在客户端文档发生变化的时候,校验代码:

// server/src/server.tsimport {    createConnection    TextDocuments,    ProposedFeatures,    ...} from 'vscode-languageserver/node';const connection = createConnection(ProposedFeatures.all);const documents: TextDocuments = new TextDocuments(TextDocument);documents.onDidChangeContent(change => {    // 文档发生变化时,校验文档    validateTextDocument(change.document);});async function validateTextDocument(textDocument: TextDocument): Promise {    ...    // 拿到诊断结果    const diagnostics = getDiagnostics(textDocument, settings);    // 发给客户端    connection.sendDiagnostics({ uri: textDocument.uri, diagnostics });}// 提供快速修复的操作connection.onCodeAction(provideCodeActions);async function provideCodeActions(params: CodeActionParams): Promise {    ...    return quickfix(textDocument, params);}

在完成上面客户端与服务端交互之后,可以注意到这两个方法 getDiagnostics(textDocument, settings)quickfix(textDocument, params)。 这两个方法分别是为文档提供诊断数据和快速修复的操作。

代码诊断

整体流程

3.gif

1. 将代码文档转成 AST 语法树

在处理客户端传递过来的 Vue 代码文本的,需要通过 vue/compiler-dom 解析成三部分 ast 格式的数据结构,分别是 template、JS、CSS, 由于现在前端代码使用的都是 TypeScript,JS 部分没有解析成 AST,因此需要使用 babel/parser 去解析 TypeScript 代码生成最终的 JS 的 AST 数据结构。

const VueParser = require('@vue/compiler-dom');// 该函数返回诊断结果客户端function getDiagnostics(textDocument: TextDocument, settings: any): Diagnostic[] {const text = textDocument.getText();const res = VueParser.parse(text);const [template, script] = res.children;return [...analyzeTemplate(template), // 解析 template 得到诊断结果...analyzeScript(script, textDocument), // 解析 js 得到诊断结果];}// 分析 js 语法function analyzeScript(script: any, textDocument: TextDocument) {  const scriptAst = parser.parse(script.children[0]?.content, {    sourceType: 'module',    plugins: [      'typescript', // typescript      ['decorators', { decoratorsBeforeExport: true }], // 装饰器      'classProperties', // ES6 class 写法      'classPrivateProperties',    ],  });

得到的 AST 语法树结构如下:

Template AST

代码小浣熊 代码小浣熊

代码小浣熊是基于商汤大语言模型的软件智能研发助手,覆盖软件需求分析、架构设计、代码编写、软件测试等环节

代码小浣熊 51 查看详情 代码小浣熊

4.gif

JS AST

5.gif

2. 遍历语法树对代码校验

在得到代码的语法树之后,我们需要对每一个代码节点进行检查,来判断是否符合 Code Review 的要求,因此需要遍历语法树来对每个节点处理。

使用深度优先搜索对 template 的 AST 进行遍历:

function deepLoopData(  data: AstTemplateInterface[],  handler: Function,  diagnostics: Diagnostic[],) {  function dfs(data: AstTemplateInterface[]) {    for (let i = 0; i < data.length; i++) {      handler(data[i], diagnostics); // 在这一步对代码进行处理      if (data[i]?.children?.length) {        dfs(data[i].children);      } else {        continue;      }    }  }  dfs(data);}function analyzeTemplate(template: any) {  const diagnostics: Diagnostic[] = [];  deepLoopData(template.children, templateHandler, diagnostics);  return diagnostics;}function templateHandler(currData: AstTemplateInterface, diagnostics: Diagnostic[]){   // ...对代码节点检查}

而对于 JS AST 遍历,可以使用 babel/traverse 遍历:

 traverse(scriptAst, {    enter(path: any) {      ...    } }

3. 发现不合规代码,生成诊断

根据 ast 语法节点去判断语法是否合规,如果不符合要求,需要在代码处生成诊断,一个基础的诊断对象(diagnostics)包括下面几个属性:

range:  诊断有问题的范围,也就是画波浪线的地方

severity: 严重性,分别有四个等级,不同等级标记的颜色不同,分别是:

Error: 1Warning: 2Information:3Hint:4

message: 诊断的提示信息

source: 来源,比如说来源是 Eslint

data:携带数据,可以将修复好的数据放在这里,用于后面的快速修复功能

比如实现一个提示函数过长的诊断:

function isLongFunction(node: Record) {  return (    // 如果结束位置的行 - 开始位置的行 > 80 的话,我们认为这个函数写得太长了    node.type === 'ClassMethod' && node.loc.end.line - node.loc.start.line > 80  );}

在遍历 AST 时如果遇到某个节点是出现函数过长的时候,就往诊断数据中添加此诊断

traverse(scriptAst, {    enter(path: any) {        const { node } = path;        if (isLongFunction(node)) {            const diagnostic: Diagnostic ={                severity: DiagnosticSeverity.Warning,                range: getPositionRange(node, scriptStart),                message: '尽可能保持一个函数的单一职责原则,单个函数不宜超过 80 行',                source: 'Code Review 指南',            }            diagnostics.push(diagnostic);        }        ...       }});

文档中所有的诊断结果会保存在 diagnostics 数组中,最后通过交互返回给客户端。

4. 提供快速修复

上面那个函数过长的诊断没办法快速修复,如果能快速修复的话,可以将修正后的结果放在 diagnostics.data 。换个例子写一个快速修复, 比如 Vue template 属性排序不正确,我们需要把代码自动修复

// attributeOrderValidator 得到判断结果 和 修复后的代码const {isGoodSort, newText} = attributeOrderValidator(props, currData.loc.source);    if (!isGoodSort) {      const range = {        start: {          line: props[0].loc.start.line - 1,          character: props[0].loc.start.column - 1,        },        end: {          line: props[props.length - 1].loc.end.line - 1,          character: props[props.length - 1].loc.end.column - 1,        },      }      let diagnostic: Diagnostic = genDiagnostics(        'vue template 上的属性顺序',        range      );      if (newText) { // 如果有修复后的代码        // 将快速修复数据保存在 diagnostic.data        diagnostic.data = {          title: '按照 Code Review 指南的顺序修复',          newText,        }      }      diagnostics.push(diagnostic);    }

quickfix(textDocument, params)

export function quickfix(  textDocument: TextDocument,  params: CodeActionParams): CodeAction[] {  const diagnostics = params.context.diagnostics;  if (isNullOrUndefined(diagnostics) || diagnostics.length === 0) {    return [];  }  const codeActions: CodeAction[] = [];  diagnostics.forEach((diag) => {    if (diag.severity === DiagnosticSeverity.Warning) {      if (diag.data) { // 如果有快速修复数据        // 添加快速修复        codeActions.push({          title: (diag.data as any)?.title,          kind: CodeActionKind.QuickFix, // 快速修复          diagnostics: [diag], // 属于哪个诊断的操作          edit: {            changes: {                [params.textDocument.uri]: [                  {                    range: diag.range,                    newText: (diag.data as any)?.newText, // 修复后的内容                  },                ],              },           },        });    }   }});

有快速修复的诊断会保存在 codeActions 中,并且返回给客户端, 重新回看交互的代码,在 documents.onDidChangeContent 事件中,通过 connection.sendDiagnostics({ uri: textDocument.uri, diagnostics }) 把诊断发送给客户端。quickfix 结果通过 connection.onCodeAction 发给客户端。

import {    createConnection    TextDocuments,    ProposedFeatures,    ...} from 'vscode-languageserver/node';const connection = createConnection(ProposedFeatures.all);const documents: TextDocuments = new TextDocuments(TextDocument);documents.onDidChangeContent(change => {    ...    // 拿到诊断结果    const diagnostics = getDiagnostics(textDocument, settings);    // 发给客户端    connection.sendDiagnostics({ uri: textDocument.uri, diagnostics });});// 提供快速修复的操作connection.onCodeAction(provideCodeActions);async function provideCodeActions(params: CodeActionParams): Promise {    ...    return quickfix(textDocument, params);}

总结

实现一个代码诊断的插件功能,需要两个步骤,首先建立语言服务器,并且建立客户端与语言服务器的交互。接着需要 服务器根据客户端的代码进行校验,把诊断结果放入 Diagnostics,快速修复结果放在 CodeActions,通过与客户端的通信,把两个结果返回给客户端,客户端即可出现黄色波浪线的问题提示。

更多关于VSCode的相关知识,请访问:vscode教程!!

以上就是VSCode插件开发实战:实现一个代码诊断插件的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/501714.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
如何关闭Windows7的UAC
上一篇 2025年11月8日 20:46:33
Sequelize 如何实现复杂的组合查询?
下一篇 2025年11月8日 20:46:42

相关推荐

  • vscode上怎么运行html_vscode上运行html步骤【指南】

    首先保存文件为.html格式,再通过浏览器或Live Server插件打开预览;推荐安装Live Server实现本地服务器运行与实时刷新,提升开发体验。 在 VS Code 上运行 HTML 文件并不需要复杂的配置,只需几个简单步骤即可预览页面效果。VS Code 本身是一个代码编辑器,不直接运行…

    2026年5月10日
    100
  • vscode怎么运行html选择浏览器_vscode选浏览器运html法【教程】

    使用Live Server插件可在VS Code中运行HTML文件并指定浏览器。1. 安装Live Server插件后右键HTML文件选择Open with Live Server即可在默认浏览器中预览。2. 通过设置Live Server的Browser选项为chrome、firefox或edge…

    2026年5月10日
    000
  • vscose html怎么运行_vscode运行html方法(拼写修正)【教程】

    一、安装Live Server扩展可实现自动刷新预览;二、直接右键HTML文件用浏览器打开;三、通过配置tasks.json任务运行文件。 如果您编写了 HTML 文件并希望在浏览器中查看其效果,但不知道如何在 Visual Studio Code 中运行,可以通过以下几种方式快速预览页面内容: 一…

    2026年5月10日
    000
  • VSCode内置了哪些编程语言插件?

    vscode 内置语言插件一览 VSCode 已内置多款语言插件,免除安装市场插件的步骤。以下是如何查看内建语言插件: 快捷键 Ctrl+Shift+P 唤出命令窗口输入并选择 “Show Built-in Extensions”左侧出现的 “Programmin…

    2026年5月10日
    000
  • 深入探索Go语言交互式调试:从GDB到Delve

    Go语言的交互式调试功能至关重要,开发者可通过多种工具实现断点设置、单步执行等操作。本文将首先介绍传统的GDB调试方式及其在IDE中的集成,随后重点阐述Go语言原生调试器Delve的优势与使用,并结合主流IDE提供详细的调试实践指南,助您高效定位和解决Go程序中的问题。 Go语言调试基础:GDB 在…

    2026年5月10日
    000
  • 在vscode中怎么运行html_vscode运行html文件方法【教程】

    1、使用Live Server扩展可实现自动刷新预览,安装后右键选择Open with Live Server即可在浏览器中实时查看HTML页面效果。 如果您在使用VSCode编写HTML文件,但不知道如何快速预览页面效果,可以通过多种方式在浏览器中运行HTML文件。以下是几种常用的实现方法: 一、…

    2026年5月10日
    000
  • C++如何为项目配置调试环境

    配置C++调试环境需生成调试符号并正确设置IDE或调试器。首先编译时添加-g(GCC/Clang)或/Zi(MSVC)以生成调试信息,使用CMake时设CMAKE_BUILD_TYPE为Debug;其次在IDE中配置可执行文件路径、工作目录、命令行参数、环境变量及调试器类型(如GDB、LLDB),V…

    2026年5月10日
    000
  • 格式化和 Linting 以保持一致性

    此活动涉及在我的开源项目 genereadme 中实施统计分析工具,以提高代码质量和一致性。 克莱布恩特拉 / 基因自述文件 genereadme 是一个命令行工具,它接收源代码文件并生成 readme.md 文件,该文件利用 llm 解释文件中的代码。 贡献 欢迎为 genereadme 做出贡献…

    2026年5月10日
    000
  • Visual Studio Code 中 Python 绘图遇到问题怎么办?

    Visual Studio Code 中的 Python 绘图问题 作为一名 Python 初学者,在 Visual Studio Code 中绘图时遇到了问题?请看这里: 问题一:无法绘制图形 您提供的代码中没有明确的错误,但它可能无法正常工作,因为您没有使用适当的库进行绘图。建议使用 Matpl…

    2026年5月10日
    000
  • Golang配置远程调试环境及注意事项

    远程调试Golang应用需在远程服务器运行delve调试服务器,本地IDE通过网络连接实现断点、变量查看等功能。首先在远程安装Go和delve,使用go build -gcflags=”all=-N -l”编译禁用优化,上传二进制并启动delve监听端口(推荐通过SSH隧道监…

    2026年5月10日
    100
  • 使用 Mask 导入本地图片时,如何解决跨域问题?

    跨域疑难:如何解决 mask 引入本地图片产生的跨域问题? 在使用 mask 导入本地图片时,你可能会遇到令人沮丧的跨域错误。为什么会出现跨域问题呢?让我们深入了解一下: mask 框架假设你以 http(s) 协议加载你的 html 文件,而当使用 file:// 协议打开本地文件时,就会产生跨域…

    2025年12月24日
    800
  • 为什么在 React 组件中无法获得 Tailwind CSS 语法提示?

    为什么在 React 组件中无法获得 Tailwind CSS 语法提示? 你在 VSCode 中编写 HTML 文件时,可以正常获取 Tailwind CSS 语法提示。但当你尝试在 React 组件中编写 Tailwind CSS 时,这些提示却消失不见了。这是什么原因造成的? 解决方案 要解决…

    2025年12月24日
    700
  • 如何在 VSCode 中为 React 组件启用 Tailwind CSS 提示?

    在 vscode 中为 react 组件启用 tailwind css 提示 如果你在使用 vscode 编写 react 组件时,发现 tailwind css 提示无法正常显示,这里有一个解决方法: 安装 tailwind css intellisense 插件 这是实现代码提示的关键,确保你已…

    2025年12月24日
    200
  • Vue3 中如何将页面上的 PX 单位转换为 REM?

    vue3 下如何实现某个页面 px 自适应到 rem? 在 vue3 中,您可以在某个页面中使用 px 转 rem 的自适应功能,以免影响其他项目 ui 框架。以下是实现方法: 使用 jquery 获取页面宽度,并将其作为基准值。例如,使用 375 作为基准,您可以在页面 mounted 生命周期函…

    2025年12月24日
    700
  • 如何实现 Vue 3 项目中特定页面自适应,避免影响全局 UI 框架?

    自适应页面 px 到 rem 插件探索 在 vue 3 项目中,开发者有时需要让某个特定页面具有自适应大小,即根据不同分辨率自动调整 px 到 rem 的转换。然而,传统的 px-to-rem 插件可能会影响整个项目的 ui 框架。 为了解决这个问题,这里提供了一种利用 javascript 和 v…

    2025年12月24日
    600
  • Vue 3 页面如何实现 px to rem 自适应?

    如何在 vue 3 页面中实现 px to rem 自适应? 在 vue 项目中,有时需要让特定的页面进行 px to rem 自适应,以实现自动缩放。以下是一个可用的解决方案: 使用 javascript 获取页面宽度,并以 375px 作为基准值。例如: let appwidth = $(‘#a…

    2025年12月24日
    700
  • CSS 砌体 Catness

    css 就像技术中的其他东西一样 – 它总是在变化和发展。该领域正在进行的开发是 css 网格布局模块级别 3,也称为 css masonry 布局。 theo 制作了一段视频,介绍了它的开发方式以及苹果和谷歌就如何实施它进行的辩论。 所有这些让我很高兴尝试 css 砌体! webkit…

    用户投稿 2025年12月24日
    000
  • 如何解决VSCode中折叠部分的代码复制问题?

    Vscode中折叠代码的复制方法 当Vscode中的代码过多时,可以将其折叠起来以方便查看和编辑。不过,有时用户可能会发现折叠后复制代码时只复制了显示的部分,而折叠部分没有被复制。以下是如何解决此问题的方法: 使用快捷键Ctrl+C直接复制 当代码折叠时,直接使用Ctrl+C快捷键复制即可复制所有代…

    2025年12月24日
    300
  • 如何复制折叠的代码?

    Visual Studio Code 中如何复制折叠的代码? Visual Studio Code (vscode) 中,当遇到过长的代码时,为了提高可读性和简洁性,开发人员会经常使用折叠功能将代码折叠起来。然而,在折叠代码后,直接按住 Ctrl + C 复制代码时,只会复制展开的部分,而折叠的部分…

    2025年12月24日
    000
  • 如何在 VSCode 复制折叠的代码?

    如何复制折叠的 VSCode 代码 使用 VSCode 时,代码过长可能会造成不便。在折叠代码后,发现无法正常复制折叠的部分,令人感到烦恼。本文将介绍一种解决方案,帮助你轻松复制折叠的 VSCode 代码。 问题:如何复制折叠起来的 VSCode 代码? 当你折叠代码后,直接选中复制只会复制未折叠的…

    2025年12月24日
    400

发表回复

登录后才能评论
关注微信