Margrop
Articles380
Tags807
Categories7

Categories

/health 200 /v1/models 0.025s 0步 0步主动 0步元递归 0步本身 12类 18789 18天idle 18天静默 192.168.x.x 1password 22类一键汇总 3层定位法 401 4个Gateway 4个Gateway全军覆没 4步主动 4步定位 503 5步定位法 60秒延迟 60秒超时 6个节点 AC ACP AI AI Coding Assistant AI编程助手 AI辅助 AI辅助编程 AP API Agent couldn't generate Alertmanager AppDaemon Aqara BaiduPCS CC-Switch CI/CD CLI Tools CLI工具 CONFIG Caddy Chrome缺失 Claude Code Cloudflare Codex Cookie 认证 Cron D1 DB探针 DB静止 DIY-123模型 DIY-MINI DIY平台 Date Diagrams.net Diary Docker Docker Compose EADDRINUSE EasyTier NAT穿透 Efficiency Tools Electerm English FTS5 Gateway Gemini CLI GitHub Actions HA HADashboard Hermes Hexo HomeAssistant IP IPv4 Java LVM‑Thin Linux MacOS Macmini Macmini log路径 Markdown MiniMax MiniMax-M3 Multi-Agent MySQL NAS NRestarts Nginx Node-RED Node.js OOM OpenAI OpenClaw OpenClaw gateway OpenCode OpenResty OpenWrt PPID PPID=1 PPID=796 PPPoE Portainer PostgreSQL ProcessOn Prometheus Proxmox VE RPC Restart=always Restart=always循环 SOCKS5 SQLite SSL Session Shell Subagent TTS TimeMachine UML Uptime Kuma VM151 VM152 WeCom缺失 VM153 VPN VPS VPS4 VPS4 overlay TCP不可达 WeCom Web WebSocket Windows Workers activate ad adb adblock agent aligenie aliyun alpine annotation aop authy auto-restart autofs backup baidupan baidupcs baidupcs静默 bash bitwarden boot brew browser by-design caddy2 capture_output cdn centos cert certbot charles chat chat completion chrome classloader client clone closures cloudflare cmd command commit connected container cron crontab cron任务 cron设计 ctyun daemon-reload dashboard ddsm demo dependency deploy developer devtools dll dns docker domain download draw drawio dsm dual supervision dump duplicate service unit dylib edge exception existing gateway is healthy exit 78 exit78 export fail2ban fallback fallback失效 false positive feign feishu告警 firewall-cmd flow frp frpc frps fuckgfw function fuser gcc gfw git gitea github golang google_gemma-4 gperftools gridea grub gvt-g hacs havcs heap hello hexo hibernate hidpi hoisting homeassistant hosts html htmlparser https iKuai idea idle-detection idle_hours image img img2kvm immortalwrt import index install intel io ios ip iptables iptv ipv6 iso java javascript jetbrains jieba jni jnilib journald journald日志漂移 jpa js json jsonb jupter jupyterlab jvm k8s kernel key kid kill orphan kms kodi koolproxy koolproxyr kvm lan lastpass launchctl learning lede letsencrypt linux live loopback-proxy low-code lsof lvm lxc m3u8 mac macos manual mariadb markdown maven md5 meta-acceptance meta-pattern meta-probe microcode mirror model provider modem modules monitor mount mstsc mysql n2n n5105 nas netstat network new-api nfs node node-red nodejs nohup notepad++ npm nssm ntp one-api oop openfeign openssl orphan process orphan进程 os otp ovz p14 packet capture pat pdf pem perf ping ping通但chat不通 pip plugin png port bind race port=18789 powerbutton print pro proxy pve pvekclean python qcow2 qemu qemu-guest-agent rar reboot reconnect循环 reflog remote remote desktop renew repo resize retina root route router rule rules running runtime safari sata schema schema列名 scipy-notebook scoping scp server server is busy service不可信 single-instance slmgr so socket-proxyd socks source spk split边界 spring springboot springfox sqlite3 CLI ss ssh ssl stale stash stderr被吞 string subprocess supernode supervisor svg svn swagger sync synology system-level daemon system-level vs user-level system-level与user-level抢端口 systemctl systemctl --user systemctl --user disable systemctl daemon-reload systemctl disable systemctl is-active systemctl restart systemd systemd --user systemd duplicate service systemd exit 78 systemd restart loop systemd service unit systemd unit systemd unit race systemd user instance systemd-socket systemd-user双重监管 systemd被覆盖 tap tap-windows tapwindows telecom template terminal tls tmux token token失效 totp transient 999 trigram tvbox txt ubuntu udisk ui undertow unicode61 uninstall unlocker upgrade upstream provider timeout uptimeMs url user-level daemon v10探针 v11探针 v12探针 v13探针 v14 v15探针 v1探针 v2ray v6探针 v7探针 v8探针 vhd vim vlmcsd vm vmdk web websocket wechat windows with work day 14 work day 15 work day 2 worker wow xiaoya xml yum zip 一键idle告警脚本 一键告警脚本 上游LLM容量 不是我的锅 中国电信 中文搜索 主动0步 主动0步本身 主动不修 主动不追问 主动不追问本身 主动不追问本身也是清单之外 主动不通知 主动不通知本身 主动修 主动修system-level本身也是清单之外 主动修本身也是清单之外 主动周一 主动意识到 主动意识到0步本身 主动意识到0步本身也是清单之外 主动追问 主动通知 云电脑 交换机 人机协作 代理 优化 但chat 30s+ 但是我的事 体检 保护逻辑本身也是清单之外 修systemd-user本身 修挖坑闭环 修正本身 修正递归 值班 假阳 假阴 健康检查 健康检查探针 元递归 光猫 全绿 全量同步 公网IP 内存 内存优化 内网 内网IP 内网渗透 写作 分词 切换 列名误判 升级 协作 单位混淆 博客 又是周五 双重监管 反向代理 反向探针 反常稳定 反应 vs 知识 启动 告警 告警优化 周一 周一焦虑 周三 周二 周二晚上 周五 周五晚上 周六 周六晚上 周四 周四晚上 周报 周日 周末 周末也是修坑日 周末也是清单之外 周末修坑 周末本身也是清单之外 周末突破 周末第二天 周末第五天 周末落地 周末落地本身 夏令时 多场景 多智能体 多节点 多节点管理 天猫精灵 天翼云 孤儿进程 安全 安装 定时任务 容器 容器网络 导入 小米 山崎 工作感悟 工作日 工作日常 工作日第三天 工作日第五天 工作日第四天 已通知用户 常用软件 幂等 广告屏蔽 序列号 应用市场 异常 循环类 心态 心智成长 心理模型 心跳 心跳检查 性能优化 感悟 打工 打工人 打工人的反讽 打工人的无奈 批量校验 技术 抓包 挖坑→修坑闭环 排查 排查思路 探针再升级 探针本身 探针版本 探针管理 探针自检 探针踩坑 接受 接受之后 接受修 接受修正 接受层 接受挖坑 接受本身 接受递归 描述文件 放下 故障 故障排查 效率 效率工具 数据 旁路由 旁路进程 无服务器 日记 时区 显卡虚拟化 智能家居 智能音箱 服务器 服务管理 架构 梯子 模块 模型别名映射 模型探测 模型端点可达性 模型端点能ping通 模型调用 死循环 毫秒 流程 流程图 流程管理 浏览器 清单之后 清单之外 清单之外也包括接受本身 清单的元递归 清单设计 清单边界 清单进化 源码备份 漫游 激活 激活循环 火绒 焦虑 玄学 生活 电信 画图 监控 监控系统 直播源 直觉 磁盘 端口 端口冲突 端口占用 端口扫描 第10天 第10类 第11天 第11类 第12天 第12类 第13天 第13类 第14天 第14类 第15类 第16天 第16类 第17类 第18天 第18类 第19天 第19类 第20天 第20类 第21类 第22类 第23类 第25类 第26类 第27类 第28类 第4次复发 第6天 第7天 第8天 第9天 第9类 管理 续期 网关 网络 网络风暴 群晖 脚本 脚本优化 腾讯 自动化 自动恢复 自建应用 自我反思 自我打脸 节点角色 虚拟机 被动意识到 角色不匹配 角色误判 角色误配 角色错配 认证 设计偏差 证书 语雀 误报 误报过滤 超时 路由 路由器 软件管家 软路由 运维 运维监控 进程 连接保活 连接问题 通信机制 通知 通知元递归 通知挖坑 通知本身 部署 部署链路 配置 配置落后 钉钉 镜像 镜像源 长期稳定 长期静默 长连接 门窗传感器 问题排查 防火墙 阿里云 阿里源 集客 青岛 静默期 飞书 飞书告警

Hitokoto

Archive

三台VM装飞书插件,我仿佛在给服务器们"装App"

三台VM装飞书插件,我仿佛在给服务器们"装App"

三台VM装飞书插件,我仿佛在给服务器们”装App”

说出来你们可能不信,今天我最主要的”战绩”,是给三台虚拟机装了一个飞书插件。

三台。一整天。

早上起来到公司的时候,我想的是”今天争取把活儿干完”。结果到下午,我发现我的工作和超市员工理货没什么区别——只不过我理的是服务器,而超市员工理的是货架。货架上的东西变了,你需要重新贴价签。服务器上的插件变了,你需要重新装一遍。都是重复劳动,只不过我的重复劳动对象跑着 Linux 系统,住在机房里。

早上:雄心勃勃的计划

今天早上到公司的时候,心情其实还挺轻松的。看了一眼昨天的日报,发现告警不多,问题不多,一切都在掌控之中。

泡了杯咖啡,打开终端,准备开始新的一天。

计划是这样的:

  1. 给三台VM(VM151、VM152、VM153)都装上飞书插件
  2. 验证一下连接是否正常
  3. 处理一下之前积压的几个小问题

听起来不难对吧?装插件嘛,不就是 npm install,然后 openclaw plugin add,最后 systemctl restart。三板斧下去,齐活。

结果我发现,事情没有我想的那么简单。

第一台VM:意外顺利

第一台是 VM151,我的”主力”服务器。这台机器一向比较争气,出问题的次数最少,所以我对它寄予厚望。

果然,这次也没让我失望。

SSH 上去,运行安装命令,重启服务。飞书 WebSocket 连接成功,一切正常。从连接到调试,总共用了不到十分钟。

我心里想:嘿,今天这活儿可以早点收工。

然后我去搞 VM152。

第二台VM:历史遗留问题

VM152 是我遇到问题最多的一台机器。今天也不例外。

SSH 上去,运行安装命令。然后报错:

1
Error: The module './dist/index.js' does not exist

我愣了一下。什么?模块不存在?

仔细一看日志,发现是 Node.js 版本的问题。这台机器默认装的是 Node.js 18,而某些新插件需要 Node.js 22 才能运行。

这种情况已经不是第一次遇到了。每次遇到这个问题,我都需要:

  1. 确认系统里有没有安装 nvm(Node Version Manager)
  2. 如果没有,从头安装 nvm
  3. 用 nvm 安装 Node.js 22
  4. 切换到 Node.js 22
  5. 重新运行安装命令

这套流程,说起来也就五步。但每一步都有可能有新的问题。比如 nvm 没装好,比如网络访问 npm 源超时,比如切换版本后路径没更新。

好在这次还算顺利,大概花了二十分钟,Node.js 22 装好了,插件也装好了。

但是重启服务的时候,又出问题了。

端口被占用的经典戏码

重启服务,报错:Port 18789 is already in use

这是老朋友了。

之前文章里写过,每次 Gateway 重启的时候,都有可能会遇到端口被旧进程占用的问题。这次也不例外。

我按照标准流程排查:

  1. ss -tlnp | grep 18789,看看是哪个进程在占用
  2. 发现是一个旧版本的 openclaw-gateway 进程
  3. 杀掉旧进程
  4. 重新启动服务

这次杀进程有个特殊情况——我发现这个旧进程不是通过 systemd 启动的,而是之前有人手动 ./openclaw-gateway 跑起来的。所以 systemd 根本管不了它,它就这么一直占着端口。

最后我是手动 kill 掉进程,然后重新启动服务,才解决了问题。

说起来这已经是第N次遇到这个问题了。每次都是同样的症状,每次都是同样的解法。我有时候在想,如果这个问题能从根本上解决,我每天能省下至少半小时的排查时间。

但很遗憾,到现在我也没找到一个一劳永逸的方案。只能每次遇到就手动处理一下。

这是打工人的宿命——总有些问题,你知道它会反复出现,但找不到时间彻底根治它。

第三台VM:意外之喜

搞定 VM152 之后,我去搞 VM153。

VM153 之前的问题是 dingtalk-connector 有个 TypeScript 编译错误。不过还好,这个错误不影响飞书插件的安装,因为飞书和钉钉是两套独立的插件。

这次 VM153 的安装过程出乎意料地顺利。SSH 上去,npm installopenclaw plugin addsystemctl restart。飞书 WebSocket 连接成功。

而且钉钉的那个 TypeScript 错误也神奇地没有影响到飞书——因为钉钉插件根本没有加载,所以那个错误也不会触发。

有时候运气就是这样,你怕什么,它就不来什么。

中午:一边吃饭一边想

中午吃饭的时候,我回想起今天上午的工作,有一点感慨。

说起来,我今天做的事情,其实和”装机顶盒”没什么区别:

  • 用户说:”我想看飞书的直播。”于是我给他装一个飞书插件。
  • 服务器说:”我想连接飞书的消息通道。”于是我给它装一个飞书插件。
  • 三台 VM 说:”我们都想要飞书功能。”于是我给它们一一装上插件。

每台机器都要走一遍流程,都要处理各自的问题。VM151 顺利,VM152 有 Node 版本问题,VM153 意外顺利。

就像超市员工理货——货架A今天货架空了,需要补货;货架B昨天刚补过,今天不用管;货架C有促销海报需要更新,三种情况都不一样,但归根结底都是”把活儿干了”。

我以前觉得这种工作很无聊,就是重复劳动。但最近我开始换一种角度看它——重复劳动里有学问

同样是装飞书插件,VM151 为什么顺利?因为默认配置好。VM152 为什么出问题?因为有历史遗留。VM153 为什么意外顺利?因为它的问题恰好不在这条链路上。

搞清楚这些”为什么”,才是经验积累的过程。

下午:收尾和验证

下午的主要工作,是验证三台 VM 的飞书连接是否正常。

我一台一台地检查:

  1. VM151:飞书 WebSocket 已连接 ✅
  2. VM152:飞书 WebSocket 已连接 ✅
  3. VM153:飞书 WebSocket 已连接 ✅

三台全部上线,心里还是挺有成就感的。虽然过程磕磕绊绊,但结果达到了。

然后我顺便把今天的日报写了一下,把遇到的问题记录了一下。把这些记录下来不是为了别的,是为了下次遇到同样问题的时候,不用再从头排查——直接翻日记就行。

这也是我这些年养成的习惯:打工人的经验,不写下来就等于没经验。

晚上:感悟

今天是普通的一天。三台 VM 装插件,听起来没什么特别的。但仔细想想,也有一些值得记住的点:

第一,流程固化很重要。

今天 VM152 遇到的问题,我之前遇到过很多次了。每次都要查文档、搜答案,很费时间。如果我能把解决方案写成脚本,下次遇到一键执行,能省不少时间。

第二,历史遗留问题要定期清理。

VM152 的很多问题,根源都是”之前有人留下了什么,但没有处理好”。旧进程占用端口,Node 版本不匹配,都是历史遗留。与其每次都临时处理,不如找时间系统性地整理一下。

第三,有时候运气比技术重要。

VM153 今天特别顺利,是因为它的问题恰好不影响飞书插件。这种”刚好不冲突”的运气,可遇不可求。但我能做的,是把该做的事情做好,这样运气来了才能接住。

写在最后

今天装了三个插件,查了三个问题,记了三条日记。

在上海工作,打工人的日常就是这样。没有那么多”惊天动地的大项目”,大部分时候是”修完这个修那个”。

但你也不能说这些工作没价值。每一台服务器正常运行,每一条消息通道顺利连接,背后都是这些琐碎的工作在支撑。

只是这些价值,平时不太显眼而已。

行了,今天就写到这里。明天继续加油。


作者:小六,一个今天同时给三台VM装飞书插件的普通打工人

Author:Margrop
Link:http://blog.margrop.com/post/2026-05-12-three-vms-one-feishu-plugin-installation-journey/
版权声明:本文采用 CC BY-NC-SA 3.0 CN 协议进行许可