有目的(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.

7069 次点击
所在节点    分享创造
39 条回复
nuk
2021-01-10 04:53:48 +08:00
控制信号开关这种,来个 mos 管就行了啊,比继电器小多了,还少接两根线
carlclone
2021-01-10 09:16:29 +08:00
智能插座通电开机就好了。。一星期前才搞,成本 19 元
SilencerL
2021-01-10 13:04:48 +08:00
@DoctorCat
……可以列举几个常见的支持 WOL 的无线网卡吗
ysc3839
2021-01-10 13:31:36 +08:00
@SilencerL #17 看你的描述,路由器是配置成了多层 NAT,配置正确的话两个路由器和电脑都是在同一个局域网的,直接在主路由上发送 WOL 就能唤醒。
SilencerL
2021-01-10 15:01:25 +08:00
@ysc3839
因为装修布线限制,目前是主路由带两个二级路由( A 和 B ),然后主路由配置静态路由让两个二级路由在一定程度上互通。
我的描述可能有误,二级路由 A 有 FCN 服务,所以我是从外部通过 FCN 连接到路由 A 的网络,然后 SSH 到路由 B 再发送 WOL 包。虽然路由 A 和路由 B 通过静态路由可以互相访问了但是 WOL 包过不去。
本来可以把两个二级路由当交换机使用,让所有设备都在主路由网断下,但是这样两个二级路由的一些特殊服务就没法用了。。就很蛋疼,只能这么先用着
hhyygg
2021-01-10 21:58:02 +08:00
intel ax200 可以 wowlan😂楼主能搞出一套东西,太强了
Kiriya
2021-01-10 21:59:51 +08:00
你们搞的太复杂了,遥控开机了解下,开机板串联接电源按钮上
SilencerL
2021-01-10 22:30:17 +08:00
@hhyygg #26
WoWLAN 只能有限的唤醒睡眠状态吧,无法和 WOL 一样打开关机状态的电脑好像?
lili1397
2021-01-10 23:05:31 +08:00
创意不错。今年流行远程办公,公司的电脑有时候需要远程开机,做好了还是能用的上的。
这个启动装置用树莓派来做,硬件会少很多。
jerfoxu
2021-01-11 09:21:38 +08:00
牛逼呀,特别适合钉钉打卡的
itianjing
2021-01-11 17:14:17 +08:00
我目前的方案是 BIOS 设置通电开机,然后电脑电源插在智能插座上,需要开机的时候吼一声小爱同学(大爱同学,小赖同学……昨天晚上发现主要唤醒词是 4 个字同时第二个字以“ai”结尾都能唤醒,所以以后就叫它大爱同学了)就好了。
SilencerL
2021-01-11 17:28:06 +08:00
@itianjing #31 对对对 我也发现小爱同学的唤醒逻辑是四个字中后面三个字匹配的哈哈哈哈哈哈哈
pytth
2021-01-14 09:45:10 +08:00
太麻烦了,可以拆开机箱,网上买一个远程开机的针脚(十几块),再买一个 Gosund 智能插座连上 wifi (官方价 39 元),接入米家 app,用手机开机即可,电脑通电就开机。
yutou527
2021-01-14 10:34:23 +08:00
远程门禁三年前我已经实践成功了😂
SilencerL
2021-01-14 10:42:19 +08:00
@pytth #33
我感觉点外卖挺方便的, 但是为啥还有人下厨自己做饭.


@yutou527 #34
我们这门禁是 POE 供电, 无信号的时候只有两个 1.5V 左右的点容给低负载的芯片供电, 当主动按下视频按钮或楼下按了门禁才会有 5V 的供电从 POE 交换机过来, 所以外部芯片找不到地方供电.. 拉明线又太蠢.
目前初步准备先用 POE 分离器从 POE 线里面分离出信号和电源, 再想下一步怎么搞了...
yutou527
2021-01-14 11:35:46 +08:00
我当时用的独立供电的树莓派,拉了超长的线到门禁,没想过在门禁偷电,随便搞搞。😂
zacharyjia
2021-01-17 11:12:06 +08:00
其实应该是 written (逃
jsgbsj
2021-01-17 18:57:21 +08:00
真羡慕你还有精力去倒腾
coloz
2021-08-05 09:36:59 +08:00
才看到,嘻嘻

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

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

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

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

© 2021 V2EX