149、【OS】【Nuttx】【周边】效果呈现方案解析:VSCode 打开外部链接(二)
【声明】本博客所有内容均为个人业余时间创作,所述技术案例均来自公开开源项目(如Github,Apache基金会),不涉及任何企业机密或未公开技术,如有侵权请联系删除
背景
上篇 blog
【OS】【Nuttx】【周边】效果呈现方案解析:VSCode 打开外部链接(一)
分析了 strace 日志中最后 browser.sh 和 code 命令的关系(之间并无联系,主要是通过 Node.js 执行脚本 server-cli.js,code 作为参数传入),但 browser.sh 脚本和 code 命令很像,差别只有一个 --openExternal
选项
VSCode 打开外部链接
上篇 blog 提到了 readlink -f
和 dirname
命令,其中 readlink -f
命令会输出 browser.sh 脚本的绝对路径
另外,从 browser.sh
路径来看,dirname
命令会剪掉后面三个元素,那么得到的是 ../server
段的绝对路径
可以看到,VSCode 自带的 Node.js 正在这个 server 路径下
回到 strace 日志这里
可以看到,260 行其实并不是执行 code 命令,而是 browser.sh 脚本的展开,相当于通过 node
解释器执行 server-cli.js
脚本
接着来看下最后一个关键的日志
284 行的这条 connect
日志
2715227 connect(18, {sa_family=AF_UNIX, sun_path="/run/user/1000/vscode-ipc-d5a87263-00c8-44d4-84fc-93a13038ef44.sock"}, 110) = 0
- 首先,这里的
connect(...)
是一个系统调用,用于客户端连接一个 socket - 第一个参数
18
是个文件描述符,是进程内部通过 socket() 创建的 socket 句柄,指向后面的 socket 路径 AF_UNIX
表示这是一个本地进程间通信(Unix Domain Socket),不走网络,从 Nuttx 自带的socket.h
头文件也能看出来
这里的AF_
前缀表示 Address Families,可以看到其指向PF_UNIX
宏,PF_UNIX
宏定义如下
PF_
前缀表示 Protocol Families,可以看到这里PF_UNIX
和PF_LOCAL
宏都是一个意思,本地通信sun_path=.../vscode-ipc-*.sock
是 socket 路径,位于内存文件系统中,用于进程间通信,前面的18
是这个 socket 路径的句柄,也可以理解为 ID,可以类比 C 语言里面指针和所指向内存的关系= 0
表示 socket 连接成功
所以整体而言,当前 PID 为 2715227 的进程尝试连接一个本地的 Unix Socket,并成功连接
注意,Socket(套接字)是计算机网络编程中的一个核心概念,是一种通信的端点,可以理解为网络通信的插座或接口,但是这里涉及的不是网络套接字 Socket,而是UNIX 域套接字(Unix Domain Socket),也叫本地套接字(Local Socket),用于同一台机器上不同进程之间的高效通信
这里 VSCode 用的就是这种进程间通信机制(Inter-Process Communication),VSCode 由多个进程组成,比如主进程,渲染进程,扩展宿主等,这些进程通过这种本地 Socket 文件进行高效安全的通信,这点通过其 socket 文件命名 vscode-ipc-*.sock
就能看出来
大致流程如下:
VSCode-Server 的主进程(Main Server Process)在监听 /run/user/1000/vscode-ipc-*.sock
,主服务进程由 VSCode Remote SSH 扩展首次连接时启动,而 Node.js
解释器运行的 code --openExternal
命令,本质是个客户端,需要向服务端主进程报告:有程序(这里是 browser.sh)要打开一个外部 URL
VSCode 主进程收到消息后,解析命令,将外部请求,通过 SSH 隧道,发回给本地 VSCode 客户端
ok,先分析到这儿,下篇 blog 继续