vscode 通过 ssh 远程开发解决 node.js 版本问题

vscode 通过官方提供的 Remote-ssh 插件,让我们可以直接通过 ssh 的连接方式,打开服务器上的代码库,远程进行开发。

我相信,“高端” vscode 玩家一定尝试过该玩法,简直好处多多。

但是用久了,难免会碰到一些问题。

问题

比如,最近我老是会碰到因为 node.js 版本,导致,项目没法直接通过资源管理器中的 NPM SCRIPTS 运行起来的问题。

什么?!有前端童鞋说自己没碰到过?那么该反思下自己,是不是不喜欢用最新版本的 node.js 去构建项目了。

还是说回咱的问题吧。

之前有个项目,由于比较老了,只能用 14.xx 版本的 node.js 才能正常运行,用更高的版本的 node.js 运行,会有一堆依赖报错的问题,导致项目没法正常的跑起来。

但是,如果升级依赖吧,代价又太大。毕竟这种吃力不讨好的事情,职场老鸟都会避而远之,除非老板同意你这么做,否则没做好的话,锅可少不了。

用 vscode 远程连上服务器的项目后,选择 NPM SCRIPTS 列表中的 npm 脚本,没法正常执行,因为此时用的 node.js 版本不对。

但是,如果通过 vscode 的终端进去,如果版本刚好合适,你会发现可以直接调用类似 npm start 这种命令,直接正常运行项目。

如果不对,也不打紧,可以用 nvm 切换 node.js 版本,接下来你会发现系统中默认的 node.js 版本调整好了,通过类似 npm start 这种命令,项目就可以正常运行了。

但是,直接通过 NPM SCRIPTS 命令还是无法运行,个人推测大概是因为这里还是用的老版本的 node.js。

机缘巧合

虽然这个问题不影响开发,通过命令行去跑项目运行,也不太影响使用体验,但是总感觉有点不爽快。

毕竟发现问题,却没法解决,只能通过别的方式来规避,作为一枚富有探索精神的程序员来说,多少心有不甘。

后来也试图找过一些解决方案,换了好多关键词去 google,甚至 ChatGPT 都想方设法地问了好多次,官方 issue 也翻烂了,还是没找到合适的解决方案。

后来有一次,机缘巧合下,主机升级,重启了。

再连上去,发现用 NPM SCRIPTS ,直接就可以正常运行了,遂恍然大悟。

原来,是因为 vscode-server 每次启动运行的时候,会自动获取当前环境中的 node.js 版本,并用这个版本的 node.js 来运行服务,服务没重启当然不能更换其使用的 node.js 版本了。

验证推测

为了验证自己的推测,连到服务器上探究了一番,果不其然,跟我的猜想一致。

原来当你通过 Remote-SSH 插件连上远程的机器的时候,该插件会在远程的机器的用户文件夹下,创建一个隐藏的目录 .vscode-server,然后顺手在该目录中安装下 vscode-server 软件,然后启动一个 vscode-server 服务。本地的 vscode 其实就是不断地和机器上的 vscode-server 在通信,从而支持通过本地的 vscode 进行远程开发。

# 查看运行的进程
ps -ef |grep .vscode-server
root      1253     1  0 15:00 ?        00:00:00 sh /root/.vscode-server/bin/1a5daa3a0231a0fbba4f14db7ec463cf99d7768e/bin/code-server --start-server --host=127.0.0.1 --accept-server-license-terms --enable-remote-auto-shutdown --port=0 --telemetry-level all --connection-token-file /root/.vscode-server/.1a5daa3a0231a0fbba4f14db7ec463cf99d7768e.token
root      1265  1253  0 15:00 ?        00:00:18 /root/.vscode-server/bin/1a5daa3a0231a0fbba4f14db7ec463cf99d7768e/node /root/.vscode-server/bin/1a5daa3a0231a0fbba4f14db7ec463cf99d7768e/out/server-main.js --start-server --host=127.0.0.1 --accept-server-license-terms --enable-remote-auto-shutdown --port=0 --telemetry-level all --connection-token-file /root/.vscode-server/.1a5daa3a0231a0fbba4f14db7ec463cf99d7768e.token
root      1368  1265  0 15:00 ?        00:00:03 /root/.vscode-server/bin/1a5daa3a0231a0fbba4f14db7ec463cf99d7768e/node /root/.vscode-server/bin/1a5daa3a0231a0fbba4f14db7ec463cf99d7768e/out/bootstrap-fork --type=ptyHost --logsPath /root/.vscode-server/data/logs/20231204T150019
root      7379  1368  0 15:49 pts/4    00:00:00 /bin/bash --init-file /root/.vscode-server/bin/1a5daa3a0231a0fbba4f14db7ec463cf99d7768e/out/vs/workbench/contrib/terminal/browser/media/shellIntegration-bash.sh
root      8272  1265  0 15:53 ?        00:00:06 /root/.vscode-server/bin/1a5daa3a0231a0fbba4f14db7ec463cf99d7768e/node --dns-result-order=ipv4first /root/.vscode-server/bin/1a5daa3a0231a0fbba4f14db7ec463cf99d7768e/out/bootstrap-fork --type=extensionHost --transformURIs --useHostProxy=false
root      8283  1265  0 15:53 ?        00:00:00 /root/.vscode-server/bin/1a5daa3a0231a0fbba4f14db7ec463cf99d7768e/node /root/.vscode-server/bin/1a5daa3a0231a0fbba4f14db7ec463cf99d7768e/out/bootstrap-fork --type=fileWatcher
root      8356  8272  0 15:53 ?        00:00:00 /root/.vscode-server/bin/1a5daa3a0231a0fbba4f14db7ec463cf99d7768e/node /root/.vscode-server/bin/1a5daa3a0231a0fbba4f14db7ec463cf99d7768e/extensions/json-language-features/server/dist/node/jsonServerMain --node-ipc --clientProcessId=8272
root      8601  8374  0 15:53 ?        00:00:00 /root/.vscode-server/code-1a5daa3a0231a0fbba4f14db7ec463cf99d7768e command-shell --cli-data-dir /root/.vscode-server/cli --parent-process-id 8374 --on-port --require-token 420f17f0d59a
root      8753  8272  0 16:15 ?        00:00:01 /root/.vscode-server/bin/1a5daa3a0231a0fbba4f14db7ec463cf99d7768e/node --max-old-space-size=3072 /root/.vscode-server/bin/1a5daa3a0231a0fbba4f14db7ec463cf99d7768e/extensions/node_modules/typescript/lib/tsserver.js --serverMode partialSemantic --useInferredProjectPerProjectRoot --disableAutomaticTypingAcquisition --cancellationPipeName /tmp/vscode-typescript0/a6e9883aad51f1da31b8/tscancellation-5a954acdf4ef24e6466c.tmp* --locale zh-cn --noGetErrOnBackgroundUpdate --validateDefaultNpmLocation --useNodeIpc
root      8754  8272  1 16:15 ?        00:00:04 /root/.vscode-server/bin/1a5daa3a0231a0fbba4f14db7ec463cf99d7768e/node --max-old-space-size=3072 /root/.vscode-server/bin/1a5daa3a0231a0fbba4f14db7ec463cf99d7768e/extensions/node_modules/typescript/lib/tsserver.js --useInferredProjectPerProjectRoot --enableTelemetry --cancellationPipeName /tmp/vscode-typescript0/a6e9883aad51f1da31b8/tscancellation-2340bc35bd699f3648c0.tmp* --locale zh-cn --noGetErrOnBackgroundUpdate --validateDefaultNpmLocation --useNodeIpc
root      8770  8754  0 16:15 ?        00:00:00 /root/.vscode-server/bin/1a5daa3a0231a0fbba4f14db7ec463cf99d7768e/node /root/.vscode-server/bin/1a5daa3a0231a0fbba4f14db7ec463cf99d7768e/extensions/node_modules/typescript/lib/typingsInstaller.js --globalTypingsCacheLocation /root/.cache/typescript/5.2 --enableTelemetry --typesMapLocation /root/.vscode-server/bin/1a5daa3a0231a0fbba4f14db7ec463cf99d7768e/extensions/node_modules/typescript/lib/typesMap.json --validateDefaultNpmLocation
root      8797   543  0 16:19 pts/0    00:00:00 grep --color=auto .vscode-server

所以,了解了原理以后,再想解决我们碰到的问题,就很简单了。

我们把服务器上的 vscode-server 服务关掉,重开一下就好了。

不要用 vscode 指令

当然,这里其实还有个坑。

在 vscode 界面,按住 ctrl+p,然后在出现的输入框里键入 > remote-ssh 会自动列出一些可以使用的 remote-ssh 命令。

里面会有 在主机上终止 VS Code 服务器在主机上卸载 VS Code 服务器 等指令。

但是你用了就会发现一个坑,这些命令都有用,但是他在终止、卸载完,就会立马重新装一个新的 VS Code 服务器,根本不给你调整环境中 node.js 版本的时间。

所以我们成功的执行我们的操作,我们得想别的方案来实现。

在主机上手动卸载 vscode-server

我们连接上主机,通过下面的方式,即可成功卸载掉 vscode-server

# 列出所有依赖 .vscode-server 路径的进程
ps -ef |grep .vscode-server

# 杀掉所有依赖于 .vscode-server 路径的进程
ps uxa | grep .vscode-server | awk '{print $2}' | xargs kill -9

# 删除掉 .vscode-server 文件夹
rm -rf ~/.vscode-server/

当然卸载掉以后,先别急着重新用我们的 vscode 编辑器连接服务器上的项目。

先用 nvm 切换下系统中默认的 node.js 版本。

# 默认指向对应的想用的 node.js 版本
nvm alias default nodeVersin

为了验证我们设置 node.js 版本的操作是否成功,可以试着断开远程的连接,重新登录下。

如果发现没问题,即可重新在本地用 vscode 连接服务器上的项目。

当然,这地方还有个需要注意的点是,不能傻傻的使用 nvm use nodeVersion 就以为将 node.js 默认的版本切换成功了。

这个操作只针对当前 bash,退出去重新进来,就会发现,系统中默认的 node.js 版本又是之前的版本了。

反思

现在,让我们回过头来看,会发现,其实解决问题的思路并不复杂。关键是,我们要了解清楚引起问题的原因是什么。

知其然更要知其所以然,才能在碰到问题的时候,抽丝剥茧,迅速的定位到问题所在。

参考链接

下面是一些参考资料,有兴趣进一步研究的童鞋可以作为参考

  1. visual studio code - How to kill VSCode remote services on Ubuntu Host - Stack Overflow
  2. arch linux - changing nodejs version through nvm is not persistent - Super User
  3. GitHub - nvm-sh/nvm: Node Version Manager - POSIX-compliant bash script to manage multiple active node.js versions
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,504评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,434评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,089评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,378评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,472评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,506评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,519评论 3 413
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,292评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,738评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,022评论 2 329
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,194评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,873评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,536评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,162评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,413评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,075评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,080评论 2 352

推荐阅读更多精彩内容