远程代码补全
lsp-bridge 是 Emacs 里面第一个把远程代码补全的性能做到比 VSCode 还快的 LSP Client,使用起来非常简单:
- 本地和服务器都安装 lsp-bridge, 同时只用在服务端安装 LSP Server
- 在服务器启动 lsp-bridge:
python3 lsp_bridge.py
- 执行
lsp-bridge-open-remote-file
命令打开远程文件, 输入远程服务器 IP 和文件路径,比如 xxx.xxx.xxx.xxx:/path/file
架构图
下面是 lsp-bridge 远程代码补全技术的架构图:
架构图原理(请按照红线、蓝线、紫线以及绿线的顺序看):
- 通过 SSH 认证的方式登录远程服务器,并访问和编辑远程文件
- 在本地编辑远程文件时,会实时发送 diff 序列给 lsp-bridge 服务端,服务端会根据 diff 序列在服务端重建文件的最新内容并调用 LSP Server 进行语法补全计算
lsp-bridge
在服务端计算好 LSP 补全菜单项后,发送补全数据到本机,再由本机lsp-bridge
进行补全菜单渲染
详细流程:
- Emacs 运行
lsp-bridge-open-remote-file
命令, 输入远程服务器的 IP 地址和路径 - 本地 lsp-bridge Python 模块通过 paramiko 库和远程服务器建立 SSH 安全加密隧道
- 安全隧道建立后, 本地 lsp-bridge Python 模块通过 socket 向远程服务器发送
open_file
的请求 - 远程 lsp-bridge 接受到请求以后, 读取远程服务器的文件, 返回文件内容给本地 lsp-bridge, 本地 lsp-bridge Python 模块再通过本地 lsp-bridge Elisp 模块调用 Emacs 函数创建 Buffer
- 监控 Emacs 才创建的远程文件 Buffer, 根据
before-change-functions
和after-change-functions
两个 hook, 得到每次编文本的 diff text - 本地 lsp-bridge Python 模块得到 diff text 以后, 按照顺序分别发送给远程 lsp-bridge 的文件同步模块和 LSP 模块, 文件同步模块根据 diff text 序列, 在远程服务器重建文件的最新内容(这样不管文件有多大, 文件的内容都可以实时同步, 因为 diff text 非常小)。 同时 LSP 模块负责根据 diff text 转换成 LSP 协议的 textDocument/didChange 请求发送给远程的 LSP 服务器, 也会发送文件搜索请求到其他搜索补全后端。 有些语言需要获取文件的内容时, 远端 LSP&Search 模块会从 FileSync 模块的内存中直接读取最新 Buffer 的内容, 而不会从开发者电脑绕一圈(这样也会避免在两边传送大型文件内容卡死 Emacs)
- 当远程 lsp-bridge LSP 模块得到 LSP 服务器或其他搜索后端的返回响应后, 再通过 socket 发送给本地 lsp-bridge, 经过 Python 模块和 Elisp 模块, 最终发送给 Emacs
- Emacs 得到补全数据后, 结合 lsp-bridge 的 ACM 模块(异步补全菜单), 在本地弹出补全菜单
- 其中紫线的部分比较复杂, 本地 lsp-bridge 和远端 lsp-bridge 建立链接后, 远端 lsp-bridge 需要获取 Emacs 配置时, 会从远端到本地建立一个 RPC, 从本地 Emacs 读取配置后再返回远端的 lsp-bridge, 为了保证 RPC 的数据不会受影响, 整体设计上, lsp-bridge 在安全隧道基础上建立了三个独立的通信频道, 分别处理文件同步、 LSP 请求及返回、 Elisp 远程 RPC 通道
- 整个架构的所有消息请求和数据处理代码都精巧的设计在子线程中, 保证永远不会卡住 Emacs
关键技术:
- 高性能文件同步: 文件改变后,发送 text diff 给远程服务器,这样不管 Emacs 打开多大的文件,在日常编程中,发送到服务器的数据都非常小,服务端可以实时根据 text diff 序列重建文件的最新版本,并随时保存最新版本文件内容到服务器 (而 tramp 这种传统插件的原理是即使改动了几个字符也在两端做大量同步计算,所以 tramp 性能非常差)
- 安全加密隧道: 利用 paramiko 先建立 SSH 的加密隧道保证双向通讯安全, 这样两端通讯方法只用在隧道内建立 socket 请求就好了, 通信逻辑和认证逻辑解耦
- 服务端不用安装 Emacs: 基于两端的 socket 通讯, 服务端需要调用 Elisp 函数时, 服务端反向通过 socket 调用开发者本机的 Emacs 来计算 Elisp 代码并返回结果给服务端
设计目标:
- 用户只用在服务器安装 lsp-bridge 和 LSP Server 就行了,不需要在服务端安装编辑插件
- Emacs 插件和配置全部在本地
- lsp-bridge 会自动探测所处的环境,选择用本地还是服务端的 lsp-bridge 代码去计算补全菜单
最后
lsp-bridge 配合我的其他插件 (EAF、blink-search、mark-macro、color-rg 等), 使得 Emacs 已经从全键盘操作效率、代码补全、远程代码补全、自由宏、搜索和图形应用协作等各维度完全超越 VSCode 的能力了,还能融合 Elisp、Python、JavaScript、C++、 TypeScript、Rust、Node、wasm 等技术协作编程。
到今天为止,lsp-bridge 已经提交了 1571 个补丁,修复 579 个 issue,代码贡献者超过 84 位,感谢各位 Emacser 的帮助、测试和无限的爱心!