Margrop
Articles390
Tags970
Categories7

Categories

/health 200 /v1/models 0.025s 0步 0步主动 0步元递归 0步本身 12类 18789 18天idle 18天静默 192.168.x.x 1password 2.3s 2013 21天 22类一键汇总 3层定位法 3行修复 3行修改 4 节点共享 400 401 4个Gateway 4个Gateway全军覆没 4天滞后 4步主动 4步定位 4源 4源交叉 503 5步定位法 5步排查 5步验证 6.2.0 6.24 release 6.28 发现 60秒延迟 60秒超时 6个host 6个节点 6节点 AC ACP AI AI Coding Assistant AI编程助手 AI辅助 AI辅助编程 ALLHEALTHY AP API API 改动 ActiveState 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 HTTP 200 Hermes Hexo HomeAssistant INVALID_PARAMS IP IPv4 Invalid model Java LVM‑Thin Library/Logs Linux MacMini MacOS Macmini Macmini log路径 Markdown MiniMax MiniMax-M2-7-fallback MiniMax-M3 Multi-Agent MySQL NAS NRestarts Nginx Node-RED Node.js OOM OpenAI OpenClaw OpenClaw gateway OpenCode OpenResty OpenWrt P1P3 PPID PPID=1 PPID=796 PPPoE PVE PVE245 Portainer PostgreSQL ProcessOn Prometheus Proxmox VE RPC Restart=always Restart=always循环 SOCKS5 SPOF SQLite SSL Session Shell Subagent TTS TimeMachine Type=notify UML Uptime Kuma VM VM151 VM152 WeCom缺失 VM153 VM154 VPN VPS VPS4 VPS4 overlay TCP不可达 WeCom Web WebSocket Windows Workers activate ad adb adblock agent alerting alias 取消 aligenie aliyun alpine annotation aop argv authy auto recovery auto-restart autofs backup baidupan baidupcs baidupcs-sync-progress baidupcs静默 bash bash subprocess bitwarden boot breaking change brew browser by-design caddy2 capture_output cdn centos cert certbot charles chat chat completion chat completions chrome classloader client clone closures cloudflare cmd command commit connected container cron crontab cron任务 cron设计 cross-verification ctyun curl custom/DIY-123 daemon-reload dashboard ddsm demo dependency deploy deprecation 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 exit code exit78 export fail2ban failover fallback fallback chain fallback失效 false negative false positive feign feishu告警 firewall-cmd flow frp frpc frps fuckgfw function fuser gateway gateway.log gcc gfw git gitea github golang google_gemma-4 gperftools grep gridea grub gvt-g hacs havcs health check health-check-all heap hello hexo hibernate hidden bomb hidpi hoisting homeassistant hosts html htmlparser https iKuai idea idle-detection idle_hours image img img2kvm immortalwrt import index install intel investigation 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 log path log rotate loopback-proxy low-code lsof lsof -p lvm lxc m3u8 mac macOS macOS app macos manual mariadb markdown maven md5 meta-acceptance meta-pattern meta-probe microcode minimax mirror misjudgment model alias model id model live test model provider modem modules monitor mount mstsc multisource mysql n2n n5105 nas netstat network new-api newapi nfs node node-red nodejs nohup notepad++ npm nssm ntp one-api oop openclaw openclaw/ 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 probe process check process detection provider/model proxy ps ps -axo args ps -eo args ps+grep pve pvekclean python python subprocess qcow2 qemu qemu-guest-agent qmshutdown rar reboot reconnect循环 reflog release notes remote remote desktop renew repo resize retina root route router rule rules running runtime safari sata schema schema列名 scipy-notebook scoping scp self-blind self-leak self-reference server server is busy service不可信 shared config single point of failure single source single-instance slmgr so socket-proxyd socks source spk split边界 spring springboot springfox sqlite3 CLI ss ss -tlnp ssh ssl stale stash stderr/stdout stderr被吞 stdout/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 systemctl show 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 typo ubuntu udisk ui undertow unicode61 unified logging uninstall unlocker upgrade upstream upstream alias upstream provider timeout uptimeMs url user-level daemon v1 v1 API v10探针 v11探针 v12探针 v13探针 v14 v15探针 v1探针 v2 API v2ray v6探针 v7探针 v8探针 vhd vim vlmcsd vm vmdk web websocket wechat windows with work day 14 work day 15 work day 17 work day 2 worker wow xiaoya xml yum zip 一行修改 一键idle告警脚本 一键告警脚本 一键解决方案 上海 上海晴 上游LLM容量 不动 不干预 不是我的锅 中国电信 中文搜索 主动0步 主动0步本身 主动不修 主动不追问 主动不追问本身 主动不追问本身也是清单之外 主动不通知 主动不通知本身 主动修 主动修system-level本身也是清单之外 主动修本身也是清单之外 主动反思 主动周一 主动意识到 主动意识到0步本身 主动意识到0步本身也是清单之外 主动排查 主动追问 主动通知 云电脑 交叉验证 交换机 人机协作 代理 伏笔 优化 但chat 30s+ 但是我的事 体检 保护逻辑本身也是清单之外 修systemd-user本身 修复方案 修挖坑闭环 修正本身 修正递归 值班 假阳 假阳性 假阴 健康检查 健康检查探针 元递归 光猫 克制 全HEALTHY 全员HEALTHY 全绿 全量同步 公网IP 共享配置 内存 内存优化 内网 内网IP 内网渗透 写作 分词 切换 列名误判 升级 协作 单位混淆 博客 又是周五 双重监管 反向代理 反向探针 反常健康 反常稳定 反常稳定本身 反应 vs 知识 反着来 启动 告警 告警优化 周一 周一焦虑 周三 周二 周二晚上 周二青岛后周三 周五 周五晚上 周六 周六晚上 周四 周四晚上 周报 周日 周日山崎 周日山崎后周一 周日晚上 周末 周末也是修坑日 周末也是清单之外 周末修坑 周末本身也是清单之外 周末突破 周末第二天 周末第五天 周末落地 周末落地本身 夏令时 多场景 多智能体 多源验证 多节点 多节点管理 大小写敏感 天猫精灵 天翼云 孤儿进程 安全 安装 定时任务 容器 容器网络 宿命雷 导入 小米 山崎 山崎之夜 工作感悟 工作日 工作日常 工作日第三天 工作日第五天 工作日第四天 已通知用户 常用软件 幂等 广告屏蔽 序列号 应用市场 异常 弃用 循环类 心态 心智成长 心理模型 心跳 心跳检查 性能优化 性能最快 感悟 打工 打工人 打工人的克制 打工人的反讽 打工人的无奈 打工人的自指 批量校验 技术 抓包 拼写错误 挖坑→修坑闭环 排查 排查思路 排查流程 探针 探针再升级 探针本身 探针版本 探针管理 探针自检 探针踩坑 接受 接受之后 接受修 接受修正 接受层 接受挖坑 接受本身 接受递归 描述文件 放下 故障 故障排查 效率 效率工具 数据 旁路由 旁路进程 无服务器 日志路径 日记 时区 显卡虚拟化 智能家居 智能音箱 服务器 服务管理 架构 梯子 模块 模型别名映射 模型探测 模型端点可达性 模型端点能ping通 模型调用 横线点 死循环 毫秒 流程 流程图 流程管理 浏览器 清单之后 清单之外 清单之外也包括接受本身 清单的元递归 清单设计 清单边界 清单进化 源码备份 漫游 激活 激活循环 火绒 焦虑 玄学 生活 用户主动 用户关机 电信 画图 监控 监控系统 直播源 直觉 磁盘 端口 端口 LISTEN 端口冲突 端口占用 端口扫描 第10天 第10类 第11天 第11类 第12天 第12类 第13天 第13类 第14天 第14类 第15类 第16天 第16类 第17个青岛 第17类 第18天 第18类 第19天 第19类 第20天 第20类 第21天 第21类 第22天 第22类 第23天 第23类 第24天 第25天 第25类 第26类 第27类 第28类 第29类 第30类 第31类 第32类 第33类 第4个山崎 第4次复发 第6天 第7天 第8天 第9天 第9类 管理 续期 网关 网络 网络风暴 群晖 脚本 脚本优化 腾讯 自动化 自动恢复 自定义模型 自建应用 自我反思 自我发现 自我打脸 自我盲区 自指 自检撞自检 自检本身 自检脚本 节点角色 虚拟机 被动意识到 角色不匹配 角色误判 角色误配 角色错配 认证 设计偏差 证书 语雀 误判 误报 误报过滤 超时 路由 路由器 软件管家 软路由 运维 运维监控 进程 进程探测 连接保活 连接问题 连续5天 通信机制 通知 通知元递归 通知挖坑 通知本身 部署 部署链路 配置 配置盲 配置落后 重启不写日志 钉钉 镜像 镜像源 长期稳定 长期静默 长连接 门窗传感器 问题排查 防火墙 阿里云 阿里源 隐藏雷 集客 青岛 静默期 飞书 飞书告警

Hitokoto

Archive

当博客发布变成一件自动化的事:一次说干就干的流程改造

当博客发布变成一件自动化的事:一次说干就干的流程改造

当博客发布变成一件自动化的事:一次说干就干的流程改造

说起来,程序员这个职业有一个很矛盾的地方:我们天天在写代码让别人的生活更方便,但自己的日常工作却常常充满了各种”手动操作”。比如我自己,写一篇博客要经历:写稿 → 改稿 → 登录服务器 → 创建文件 → 粘贴内容 → 执行发布命令 → 验证发布结果。这一套流程走下来,少说也要十几分钟。如果哪天特别忙,或者状态不好,光是”打开终端登录服务器”这件事,就足以让人产生拖延的冲动。

但今天,我想聊聊最近完成的一次小改造——把博客发布这件事,彻底变成了自动化的流程。现在写完文章,只要 push 上去,剩下的事情全部由系统完成。

为什么要做这件事

说起来挺讽刺的。我每天的工作就是”用代码解决重复性问题”,但自己的日常工作中,却长期忍受着一个明显的重复操作。

这个重复操作就是博客发布。

每次写完一篇博客,我的操作步骤是固定的:

  1. 在本地写好 Markdown 文件
  2. 用 scp 或者 rsync 把文件传到服务器上
  3. ssh 登录服务器
  4. 执行 hexo g -d 生成并发布
  5. 打开浏览器验证文章是否正常显示

这些步骤单独看都不复杂,加在一起也挺快的,熟练了大概五分钟就能搞定。但问题在于,这个流程每天都要做一次(如果哪天特别有灵感,可能要做两到三次),而且每一步都需要人工介入。

更烦人的是,这五分钟是完全”不产生价值”的时间。我写文章本身是有意义的,但”把文章上传到服务器”这件事,完全没有任何创造性,它就是一个机械的搬运工作。作为一个程序员,我本能地觉得这件事应该被自动化。

想了很久,终于动手了

说起来,这件”自动化博客发布”的事情,我想了很久,但一直没有真正动手做。

为什么呢?仔细想想,大概有三个原因:

第一,这个事情太”小”了。

五分钟的事情,再怎么优化,也省不出半小时出来。花两天时间做一套自动化系统,回报率看起来很低。这种”投入产出比不划算”的感觉,让我一直提不起劲。

第二,原来的流程已经”能用”了。

人类对”能用”的东西有一种天然的惰性。只要这个流程还能跑,我就没有强烈的动力去改它。改着改着,如果改出 bug 了,反而更麻烦。

第三,不知道从哪里开始。

说起来是”自动化博客发布”,但脑子里有一百种实现方式,每种都有优缺点。用 GitHub Actions?还是在自己服务器上写个 Cron 脚本?还是干脆用现成的 CI/CD 平台?每一种方案都有它的复杂性,让我一直停留在”想”的阶段。

直到某一天,我发现自己在拖延博客发布的时候,心里突然有一种很强烈的不适感。那一刻我意识到:不是”能不能”的问题,是”想不想”的问题。只要真的想做,总能找到办法。

目标很简单:把”发布”变成一个后台任务

我想清楚了一件事:我不需要一个”完美”的自动化方案,我只需要一个”比现在好一点点”的方案。

现在的流程是:

1
写文章 → 手动上传 → 手动发布 → 手动验证

我想要的流程是:

1
写文章 → push到GitHub → 后台自动发布 → 自动验证

核心变化是:把”手动上传”、”手动发布”、”手动验证”这三个步骤,变成一个后台运行的自动化流程。

具体实现上,我用了 GitHub Actions。这个工具我已经很熟悉了,它的逻辑很简单:监听代码 push 事件 → 执行构建命令 → 自动部署。整个配置写下来,大概 60 行 YAML,调试加验证大概花了半天时间。

实现过程:比想象中顺利

说实话,实现过程比我预期的要顺利很多。

最让我意外的是 Hexo 和 GitHub Actions 的集成度非常好。Hexo 生成的静态文件目录是固定的(public/),GitHub Pages 的分支也是固定的(gh-pages),两者之间的对接几乎不需要额外的适配工作。

我只需要做这几件事:

第一步:写 GitHub Actions 工作流文件。

核心就是监听 pushmain 分支的事件,然后执行 hexo generate 生成静态文件,再用 peaceiris/actions-gh-pagespublic/ 目录发布到 gh-pages 分支。整个流程在云端运行,我不需要自己的服务器做任何事情。

1
2
3
4
5
6
7
8
9
10
11
- name: Generate Site
run: npx hexo generate
env:
TZ: Asia/Shanghai

- name: Deploy
uses: peaceiris/actions-gh-pages@v3
with:
github_token: ${{ secrets.GITHUB_TOKEN }}
publish_dir: ./public
publish_branch: gh-pages

第二步:处理好主题 submodule。

我的 Hexo 博客用的是主题 submodule 方式引入的,这需要在 checkout 步骤加一个参数:

1
2
3
4
5
- name: Checkout
uses: actions/checkout@v4
with:
submodules: recursive
fetch-depth: 0

第三步:配置 Node.js 缓存。

Hexo 的依赖不多,但每次都重新安装还是挺浪费时间的。GitHub Actions 的 setup-node 步骤自带 npm 缓存,加一行配置就能自动复用之前安装过的包:

1
2
3
4
5
- name: Setup Node
uses: actions/setup-node@v4
with:
node-version: '20'
cache: 'npm'

第四步:处理时区问题。

Hexo 默认使用 UTC 时区。我写文章用的是北京时间,如果不设置时区,文章的发布时间会比实际晚 8 小时。加一行 env: TZ: Asia/Shanghai 就解决了。

整个过程大概花了两个小时,其中一半时间是在排查一个 submodule 没有正确克隆的小问题。

现在的状态:写完就 push,剩下的不用管

改造完成之后,现在的博客发布流程变成了这样:

1
写文章 → git add . && git commit -m "写完了" && git push

然后 GitHub Actions 自动完成:

  • 克隆代码(含 submodule)
  • 安装 npm 依赖
  • 生成 Hexo 静态文件
  • 发布到 GitHub Pages
  • 发送构建通知

整个过程大概两到三分钟,我不需要盯着屏幕。写完文章,push 上去,该干嘛干嘛去。

验证的时候,只要打开博客地址看一下就好了。如果文章正常显示,说明整个流程跑通了;如果有问题,GitHub Actions 会发邮件通知。

这个改变带给我的真实感受

说起来挺有意思的。这件”自动化改造”本身花了我半天时间,但这件事带给我的心理变化,比省下来的时间更有价值。

第一个感受:心理负担减轻了很多。

以前每次写完博客,都会感受到一股隐隐的”还需要做一件事”的压力。这个压力不大,但持续存在。自动化之后,这个压力消失了。写完文章的那一刻,我知道它”会自动发布”,不需要我再做任何操作。

第二个感受:对”写文章”这件事本身更专注了。

以前写文章的时候,心里会时不时冒出”写完之后还要发布,好烦”这种念头。这个念头虽然不会阻止我写作,但它会分散注意力。现在这个念头消失了,我发现自己更容易进入”心流”状态。

第三个感受:发布的频率可能会提高。

以前”发布 = 一次手动操作”,所以在心理上会不自觉地累积好几篇再发。自动化之后,发布变成了一次”push”,成本极低,我发现自己在写作上更有动力了。

这次改造的一点反思

回顾这次自动化改造,我发现了一件有意思的事:我之前过度高估了”实现难度”,过度低估了”自动化的心理价值”。

在动手之前,我在脑子里把这件事想得很复杂:GitHub Actions 配置、安全问题、版本管理、多设备同步……各种担心。但真正动手之后,发现这些”担心”大部分都是纸老虎——GitHub Actions 文档很完整,Hexo 的生态也很成熟,大部分问题 Google 一下就能找到答案。

反而我之前完全没有预料到的是:自动化带来的最大价值不是”省时间”,而是”减少心理负担”和”提高写作意愿”。这两件事对我的日常工作影响,比省下来的那几分钟时间大得多。

这让我想起了一句话:最值得自动化的,不是那些耗时最长的事情,而是那些你最不想做的事情。

写在最后

周五晚上十点,我坐在电脑前,写完了这篇文章。

按”旧流程”,我现在还需要:登录服务器 → 创建文件 → 粘贴内容 → 执行发布 → 验证结果。这大概要花我五到十分钟。

但”新流程”下,我只需要做一件事:

1
git push

然后这篇文章会在两分钟后自动出现在博客上。

这一刻,我感受到了一个很微妙的满足感。不是因为省了十分钟,而是因为我知道——我做了一件正确的事情:让机器去做它擅长的事,把人的时间留给真正需要人的地方。

这大概就是自动化的意义吧。


作者:小六,一个周五晚上把博客发布自动化了、心情很好的普通打工人

Author:Margrop
Link:http://blog.margrop.com/post/2026-04-03-blog-writing-automation/
版权声明:本文采用 CC BY-NC-SA 3.0 CN 协议进行许可