有目的(di)地(de)瞎折腾——为了温暖的被窝而实现远程开机

2021-01-09 19:23:25 +08:00
 SilencerL

0x0

这个冬天真的太 TM 的冷了,冷到我想一天 24 小时都呆在床上哪儿也不去。本来在这个美好的周六是可以实现这个同样美好的愿望,但是一大早同事来的电话击碎了我的梦想——线上项目出了点问题需要排查一下问题。虽然不情愿,但是寄人篱下给人打工哪能不低头(……),只好抄起枕边的 Surface 看一下到底是啥情况。粗略看了一下发现应该只是配置问题,去后台修改配置项就可以解决。

正当庆幸不是什么大问题改完就能继续 V 坡(本地方言睡觉)时我突然悲剧的发现:登录后台的账号密码入口在内网的一个服务上,公司的 VPN 只装在距离我床大概两米的台式机上,平时只要远程连过去就可以,但是台式机昨晚上被我手贱关机了……如果要开机势必要爬出温暖的被窝。就是在这么一个 2021 年冬日的早晨,从被窝到书桌,两米的距离好似咫尺天涯,就像鱼儿与飞鸟一个在天上一个在深海;又好似我与喜欢的人,永远不会有交会的轨迹(……)。我大概尝试了 25565 种体位,不管怎么探出手抑或用拖鞋辅助,都没法实现人在床上摸到电脑开机键。万般无奈下只好跳出被窝—冲去开机—杀回被窝(天知道这么冷为什么我还要裸睡)。

0x1

经过这么一番折腾也没啥心思接着睡觉了,为了生活的美好和温暖的被窝,我决定思考一下如何实现远程开机。正常情况下是可以借助 WOL ( Wake On LAN )来实现,但是前段时间网线坏了,所以一直都是用 WiFi 联网,然而大部分无线网卡是不支持 WOL 的。既然目前 WOL 这条路走不通,那就从硬件方面下手。

主板控制电脑开机是通过 Power SW( Power Switch )的两个引脚开合状态来控制的,通常情况下这两个引脚的状态是 ON(常开),当这两个引脚的闭合(短接)时间超过一定阈值主板则进行开关机操作,当时间更长一点则进行强制关机的操作。所以我们只要想办法远程控制这两个引脚的开合就可以实现远程开机和远程强制关机(当远程操作的时候电脑宕机了就会发现远程强制关机多么重要……),同时顺便把 WOL 给支持进去。

0x2

了解了主板开机的原理后开始考虑下如何实现目标。大概的上网搜索了一下信息,发现了一个 Blinker 平台提供了一大坨非常方便入手的 SDK 和文档之类的,配合廉价的 ESP 8266 模块可以快速搭建物联网设备。这样我们可以通过 ESP 8266 + 继电器 + Blinker 平台并稍微动手接一下电路构建一个远程开关机模块。

需要的物料:

需要的软件(点击可查看或下载)

Arduino IDE 配置

需要的工具

其他

需要先在点灯的 App 中参照下图添加设备:

0x3

获取到 Secret Key 并且添加功能按钮结束后参照如下代码(代码写的烂请多见谅;其中包括硬件启动和 WOL ):

#define BLINKER_WIFI
// 不加密通信,减小固件体积,我提供的固件编译后较小可以去掉这一行
#define BLINKER_WITHOUT_SSL
// 继电器为 ESP 8266 的 GPIO 0 控制
#define RELAY 0

#include <Blinker.h>

const char *auth = "{申请的 Secret Key}";
const char *ssid = "{2.4G WiFi 名称}";
const char *pwd = "{WiFi 密码}";

WiFiUDP UDP;
char wolBuffer[102];
// 目标主机 MAC 地址
const char mac[6] = {0x00, 0x00, 0x00, 0x00, 0x00, 0x00};

// 绑定 Button
BlinkerButton PowerButton("btnTogglePower");
BlinkerButton HardShutButton("btnHardShut");
BlinkerButton WakeOnLANButton("btnWakeOnLAN");

void setup() {
  Serial.begin(115200);

  BLINKER_DEBUG.stream(Serial);

  // 定义 GPIO 0 为输出端
  pinMode(RELAY, OUTPUT);
  // 默认 GPIO 0 高电平,继电器模块为低电平激活
  digitalWrite(RELAY, HIGH);

  Blinker.begin(auth, ssid, pwd);

  // 绑定事件
  PowerButton.attach(OnPowerButtonTap);
  HardShutButton.attach(OnHardShutButtonTap);
  WakeOnLANButton.attach(OnWakeOnLANButtonTap);
  
  UDP.begin(9);
  WOLPacketInit();
}

void loop() {
  Blinker.run();
}

// 生成 WOL 数据包
void WOLPacketInit(){
  int i, j;
  
  for(i = 0; i < 6; i++){
    wolBuffer[i] = 0xFF;  
  }
  
  for(i = 0; i < 16; i++){
    for(j = 0; j < 6; j++){
      wolBuffer[6 + i * 6 + j] = mac[j];  
    }
  }
}

void SendWOLPacket(){
  // 向指定广播地址播报 WOL 数据包
  // Broadcast 可以通过 ifconfig 查看,WOL 端口是 7 或 9 可以自己尝试
  UDP.beginPacket("{Broadcast (广播地址)}", 9);
  UDP.write(wolBuffer);
  UDP.endPacket();
}

void OnPowerButtonTap(const String & state) {
  digitalWrite(RELAY, LOW);
  // 200ms 短按开机
  Blinker.delay(200);
  digitalWrite(RELAY, HIGH);
}

void OnHardShutButtonTap(const String & state) {
  digitalWrite(RELAY, LOW);
  // 5000ms 长按强制关机
  Blinker.delay(5000);
  digitalWrite(RELAY, HIGH);
}

void OnWakeOnLANButtonTap(const String & state){
  SendWOLPacket();
}

将上述代码通过 CH340C 烧录到 ESP 8266 中,等待烧录成功

可以提前打开 工具 - 串口监视器 查看,出现下面这些信息即为烧录成功,可以在点灯 App 上点击按钮操作一下看一下是否正常(在代码里相应位置写一下日志输出,我没写……)。

0x4

软件和烧录部分准备结束,下面要把模块接入到机箱里面。首先一个问题是模块的供电从哪里来,第一想到的是改一个 USB 接口然后通过外接电源供电,但是这样就要额外拖一条线在外面总感觉很蠢。突然想起来电脑的电源肯定是有持续供电的,否则一些电脑的关机后 USB 口持续供电功能怎么来的。查阅了一下 24 PIN 电源针脚定义,发现的确是有持续供电的针脚( PIN 9,5V SB ),刚好满足继电器模块的工作电压范围( DC 5V - 12V )。

所以我们只需将供电正极接在 PIN 9,负极接在随便一个 GND (接地)针脚即可(距离 PIN 9 最近的 GND 是 PIN 7 ),如图所示:

继电器模块提供了 NOCOMNC 三个接口,其中 NO 是常开端(电路非闭合),NC 是常闭端(电路闭合),COM 是公共接口。所以我们只需要用到 NO 和 COM 两个接口,平常是常开状态,根据上文的代码当 ESP 8266 收到对应指令后会将 GPIO 0 口输出低电平进而控制继电器闭合,模拟开机按键的操作,GPIO 0 口输出低电平的时间就是按下电源按键的时间,短按开机长按强制关机。

按照要求接好线:

然后将 Power SW 线接回主板控制针脚,主机上电试一下:

(此处假装有动图或者视频)

可以看到手机上点击电源后伴随着继电器一声清脆的“咔哒”声,电脑成功开机~ 点击硬关机后电脑也成功强制关机,WOL 功能因为没有网线所以暂时没法测试,不过大差不差应该是没问题的。

0x5

最后要考虑的问题是,怎么把这个模块稳妥的放在机箱内。毕竟有很多裸露的触点机箱内又有金属部分很容易误触导致短路,轻则模块 工作异常,重则主板 GG 。

经过慎重的考虑……我决定发挥垃圾佬的精神,用一个地上捡的自封袋把模块装起来(绝缘),然后用扎线带绑在内部的电源线上(固定),这样最低成本解决两个问题简直美哉~

0x6

至此,这套远程开机模块已经可以完美的初步运行,接下来可以考虑进行一系列的优化:

做完之后算了一下成本,ESP 8266 + 继电器模块大概 16 元左右,杜邦线原来用剩下的大概几毛,烧录器 10 块左右可以用到把我送走。其实这价钱可以买一个成品的远程开机设备了,但是毕竟自己瞎折腾才算是 Geeker 嘛。享受一下自己从查资料到动手最后实现目标功能的过程才是最大的乐趣。

其他好玩的


EnD

Worte with ❤ by Wood.

6613 次点击
所在节点    分享创造
39 条回复
ruokw
2021-01-09 19:40:04 +08:00
强还是你强,归根到底还是吃的太饱,有余力。 doge
SilencerL
2021-01-09 19:45:36 +08:00
@ruokw
如果天气没这么冷大概我真的懒得拆机箱搞这个 /doge
z7356995
2021-01-09 19:48:47 +08:00
曾经也和你一样吃得太饱,现在到了能用钱解决的,绝不浪费时间了
DAPTX4869
2021-01-09 20:05:46 +08:00
大佬牛逼...
菜鸡还是选择小米智能插座吧...
Devin
2021-01-09 20:06:04 +08:00
谢谢分享,很有意思
还好我有撑衣杆
Osk
2021-01-09 20:12:52 +08:00
我选择修好网线 /doge/
shenyi97
2021-01-09 20:14:36 +08:00
挺好的,但我觉得最好用电工胶带缠几圈固定一下。
mingzhi
2021-01-09 20:35:11 +08:00
福州人?
lloovve
2021-01-09 20:47:39 +08:00
淘宝 25 搞定,开机模块,我最近这种 esp8266 做了一个电热毯恒温控制器
wangxiaoaer
2021-01-09 20:53:58 +08:00
软文不错。

顺便问下,原理是什么?那个继电设备持续联网,连接到 blinker 的服务器?联网怎么设置的。
idblife
2021-01-09 20:57:32 +08:00
找个女朋友吧
SilencerL
2021-01-09 21:01:15 +08:00
@Devin #5
谁会在床上放撑衣杆啊我摔!!


@mingzhi #8
是的哈哈哈


@wangxiaoaer #10
虽然的确不是软文……但是经你提醒我应该去应聘下软文写手哈哈哈。原理大概是 ESP 8266 通过 MQTT 协议与服务器建立“长连接”,配网信息是硬编码在固件里的,没去研究怎么通过点灯 App 自定义配网信息


@DAPTX4869 #4
主要是智能插座只能通过设置“通电自动开机”,没法实现更多自定义操作(比如异常关机开机或强制关机操作)
yanfany
2021-01-09 21:44:29 +08:00
挺好的,应该是这个月为止在 v 站看到最有趣的帖子了(狗头 ,需求痛点也戳到我了,南方人的冬天真的一言难尽,能缩在被窝是真的幸福,主要解决方案也设计的很好,让我重新想起本科硬件课程的几个实验带来的那种成就感和乐趣,可惜现在只会在坑里炼丹 hh
zzzhen
2021-01-09 21:51:41 +08:00
有意思
niubee1
2021-01-09 23:08:14 +08:00
其实,你在床头放一根衣架叉子,就 OK 了
Mountain
2021-01-10 00:21:35 +08:00
哈哈哈哈哈哈哈我也选择修网线
SilencerL
2021-01-10 00:27:28 +08:00
@yanfany #13
哈哈哈哈哈哈感谢支持 祝 2021 炼丹愉快


@Osk #6
@Mountain #16
口享!主要是 WOL 在有些时候会失灵,而且出了局域网就受限;之前如果我在公司想开家里的电脑,需要先通过 FCN 连到家里主路由的网络,然后 SSH 连接主路由 -> 在主路由上 SSH 到二级路由 -> 在二级路由发送 WOL Packet 。救国途径过于曲线……
Elethom
2021-01-10 01:44:21 +08:00
这是家宴吃太饱了? 🐶
wtks1
2021-01-10 02:50:22 +08:00
呃,我觉得修好网线可能更快一点...
DoctorCat
2021-01-10 03:45:40 +08:00
说不支持 WOL 的网卡,那是有多老啊 😂

这是一个专为移动设备优化的页面(即为了让你能够在 Google 搜索结果里秒开这个页面),如果你希望参与 V2EX 社区的讨论,你可以继续到 V2EX 上打开本讨论主题的完整版本。

https://www.v2ex.com/t/743387

V2EX 是创意工作者们的社区,是一个分享自己正在做的有趣事物、交流想法,可以遇见新朋友甚至新机会的地方。

V2EX is a community of developers, designers and creative people.

© 2021 V2EX