2024-03-24
705
之前将ESP8266接入巴法云,使用一段时间后,发现不够稳定,干脆直接在内网搭建一个HomeAssistant实现灯的控制.这里使用docker搭建:homeassistat(Home Assistant 2021.12.8):version: '3'services: homeassistant: privileged: true image: 'homeassistant/home-assistant:stable' container_name: 'homeassistant' environment: - TZ=Asia/Shanghai network_mode: host volumes: - /home/file/docker/homeassistant/config:/config restart: unless-stopped启动完成后访问服务器8123端口完成初始化.还需要安装一个mqtt服务端,这里同样使用docker搭建mosquitto:version: "3.0"services: mosquitto: container_name: 'mosquitto' image: 'eclipse-mosquitto:2.0.18' ports: - 1883:1883 - 9001:9001 volumes: - /home/file/docker/mosquitto/config:/mosquitto/config - /home/file/docker/mosquitto/data:/mosquitto/data - /home/file/docker/mosquitto/log:/mosquitto/logmosquitto.confpersistence true#persistence_location /mosquitto/data#log_dest file /mosquitto/log/mosquitto.loglistener 9001port 1883allow_anonymous true启动mqtt服务器,然后来到HomeAssistant管理页面,左侧配置-设备与服务,右下角添加集成,找到MQTT,输入刚刚搭建的mqtt服务器.烧录开发板,跟前一篇文章的巴法云代码一样,这里再贴一下,设备接线参考ESP8266通过巴法云接入米家我这边使用的WiFiManager版本为0.16.0,发现使用最新版2.0.17有无法连接MQTT服务器的问题#include ESP8266WiFi.h#include ESP8266WebServer.h#include WiFiManager.h#include Ticker.h //#include PubSubClient.h#include "PubSubClient.h" //默认,加载MQTT库文件 WiFiClient espClient; //wifi客户端模式PubSubClient client(espClient);Ticker ticker; //定时器对象#define ID_MQTT "esp8266-01" //客户端id#define D1 5 //D1#define D2 4 //D2const char* topic = "light001"; //主题名字,与巴法云控制台的主题一致const int B_led = D1; //引脚值,此引脚用于连接继电器的信号端const int led_switch = D2; //引脚值,此引脚用于连接继电器的信号端//**************************************************//const char* mqtt_server = "192.168.4.200"; //默认,MQTT服务器const int mqtt_server_port = 1883; //默认,MQTT服务器端口int count; // Ticker计数用的变量int state = 0; //灯的状态// 建立WiFiManager对象WiFiManager wifiManager; //灯光函数及引脚定义void turnOnLed();void turnOffLed();//计数void tickerCount(){ count++;}// 连接MQTT服务器void reconnect() { Serial.print("Attempting MQTT connection..."); // Attempt to connect if (client.connect(ID_MQTT)) { Serial.println("connected"); Serial.print("subscribe:"); Serial.println(topic); //订阅主题,如果需要订阅多个主题,可发送多条订阅指令client.subscribe(topic2);client.subscribe(topic3); client.subscribe(topic); } else { Serial.print("failed, rc="); Serial.println(client.state()); Serial.println(" try again in 1 seconds"); // Wait 5 seconds before retrying delay(1000); }//}}// 收到信息后的回调函数void receiveCallback(char* topic, byte* payload, unsigned int length) { Serial.print("来了Topic:"); Serial.println(topic); String msg = ""; for (int i = 0; i length; i++) { msg += (char)payload[i]; } Serial.print("Msg:"); Serial.println(msg); if (msg == "on") {//如果接收字符on,亮灯 turnOnLed();//开灯函数 } else if (msg == "off") {//如果接收字符off,关灯 turnOffLed();//关灯函数 } msg = ""; }//开灯void turnOnLed() { Serial.println("turn on light"); digitalWrite(B_led, HIGH); state = 1;}//关灯void turnOffLed() { Serial.println("turn off light"); digitalWrite(B_led, LOW); state = 0;}// 发布信息void pubMQTTmsg(){ // 建立发布主题 //巴法云个性设置,推送消息时:主题名后加/set推送消息,表示向所有订阅这个主题的设备们推送消息, //假如推送者自己也订阅了这个主题,消息不会被推送给它自己,以防止自己推送的消息被自己接收。 String topicString = "light001/set" ; char publishTopic[topicString.length() + 1]; //转换成字符数组 strcpy(publishTopic, topicString.c_str()); // 建立发布信息,当前D1引脚状态 String messageString; if(digitalRead(B_led)){ messageString = "on"; } else { messageString = "off"; } char publishMsg[messageString.length() + 1]; //转换成字符数组 strcpy(publishMsg, messageString.c_str()); // 实现ESP8266向主题发布信息,并在串口监视器显示出来 if(client.publish(publishTopic, publishMsg)){ Serial.println("Publish Topic:"); Serial.println(publishTopic); Serial.println("Publish message:"); Serial.println(publishMsg); } else { Serial.println("Message Publish Failed."); }}//程序入口void setup() { pinMode(B_led, OUTPUT); //设置引脚为输出模式 pinMode(led_switch, OUTPUT); // 开关引脚为输入模式 digitalWrite(B_led, LOW);//默认引脚上电低电平 Serial.begin(9600); //设置波特率9600 //********************自动配置网络************************ // wifiManager.setTimeout(20); wifiManager.setConfigPortalTimeout(120); // 自动连接WiFi。以下语句的参数是连接ESP8266时的WiFi名称// wifiManager.autoConnect("AutoConnectAP"); // 如果您希望该WiFi添加密码,可以使用以下语句: wifiManager.autoConnect("8266","xxxxx"); //重置wifi设置 // wifiManager.resetSettings(); // WiFi连接成功后将通过串口监视器输出连接成功信息 Serial.println(""); Serial.print("ESP8266 Connected to "); Serial.println(WiFi.SSID()); // WiFi名称 Serial.print("IP address:\t"); Serial.println(WiFi.localIP()); // IP //******************************************************* client.setServer(mqtt_server, mqtt_server_port);//设置mqtt服务器 client.setCallback(receiveCallback); //mqtt消息处理 ticker.attach(1, tickerCount); // Ticker定时对象 Serial.println("setup结束");}//循环执行 void loop() {// Serial.println("循环"); int val = digitalRead(B_led); int ctl_val = digitalRead(led_switch); if(ctl_val == 1){//按键按下 delay(400); state = !state; digitalWrite(B_led, state);// digitalWrite(led_switch, LOW); digitalWrite(led_switch, LOW); int ctl_val = digitalRead(led_switch); Serial.print("开2关值为 :\t"); Serial.println(ctl_val); } if (client.connected()) { client.loop(); if (count = 10){ // 每隔10秒钟发布一次信息 Serial.println("心跳报"); pubMQTTmsg(); count = 0; } }else{ reconnect(); }}烧录完成后,现在为HomeAssistant添加该设备在homeassistant的configuration.yaml中最后添加一行light: !include light01.yaml随后在configuration.yaml同级目录添加light01.yaml,内容为- platform: mqtt name: "卧室灯" state_topic: "light001/set" command_topic: "light001" payload_on: "on" payload_off: "off" optimistic: false qos: 0 retain: true完成后重启homeassistant即可连接HomeKit配置-设备与服务中添加HomeKit,然后开放端口在通知中可以看到绑定二维码,使用iPhone的家庭APP扫码绑定用户名-高级模式参考文章ESP8266通过MQTT接入Home Assistant实践ESP8266通过MQTT接入Home Assistant
继续阅读»
2024-03-06
794
效果演示视频(apple设备无法播放):用到的硬件:ESP8266-NodeMCU开发板、继电器模块、常开门禁开关、杜邦线若干。在巴法云建立帐号并添加MQTT设备云,将私钥复制到下方代码ID_MQTT变量中米家添加设备:我的 - 其他平台设备 - 巴法,按提示绑定巴法云帐号使用Arduino将以下程序烧录至esp8266开发板:/** 智能语言控制控制,支持同时天猫、小爱、小度、google Assistent控制* 也同时支持web控制、小程序控制、app控制,定时控制等* 项目示例:通过发送on或off控制开关* 官网:bemfa.com*/#include ESP8266WiFi.h //默认,加载WIFI头文件#include DNSServer.h#include ESP8266WebServer.h#include WiFiManager.h#include Ticker.h //#include PubSubClient.h#include "PubSubClient.h" //默认,加载MQTT库文件 WiFiClient espClient; //wifi客户端模式PubSubClient client(espClient);Ticker ticker; //定时器对象//********************需要修改的部分*******************//#define ID_MQTT "xxx" //用户私钥,控制台获取#define D1 5 //D1#define D2 4 //D2const char* topic = "light002"; //主题名字,与巴法云控制台的主题一致const int B_led = D1; //引脚值,此引脚用于连接继电器的信号端const int led_switch = D2; //引脚值,此引脚用于连接继电器的信号端//**************************************************//const char* mqtt_server = "bemfa.com"; //默认,MQTT服务器const int mqtt_server_port = 9501; //默认,MQTT服务器端口int count; // Ticker计数用的变量int state = 0; //灯的状态// 建立WiFiManager对象WiFiManager wifiManager; //灯光函数及引脚定义void turnOnLed();void turnOffLed();//计数void tickerCount(){ count++;}// 连接MQTT服务器void reconnect() { Serial.print("Attempting MQTT connection..."); // Attempt to connect if (client.connect(ID_MQTT)) { Serial.println("connected"); Serial.print("subscribe:"); Serial.println(topic); //订阅主题,如果需要订阅多个主题,可发送多条订阅指令client.subscribe(topic2);client.subscribe(topic3); client.subscribe(topic); } else { Serial.print("failed, rc="); Serial.println(client.state()); Serial.println(" try again in 1 seconds"); // Wait 5 seconds before retrying delay(1000); }}// 收到信息后的回调函数void receiveCallback(char* topic, byte* payload, unsigned int length) { Serial.print("Topic:"); Serial.println(topic); String msg = ""; for (int i = 0; i length; i++) { msg += (char)payload[i]; } Serial.print("Msg:"); Serial.println(msg); if (msg == "on") {//如果接收字符on,亮灯 turnOnLed();//开灯函数 } else if (msg == "off") {//如果接收字符off,关灯 turnOffLed();//关灯函数 } msg = ""; }//开灯void turnOnLed() { Serial.println("turn on light"); digitalWrite(B_led, HIGH); state = 1;}//关灯void turnOffLed() { Serial.println("turn off light"); digitalWrite(B_led, LOW); state = 0;}// 发布信息void pubMQTTmsg(){ // 建立发布主题 //巴法云个性设置,推送消息时:主题名后加/set推送消息,表示向所有订阅这个主题的设备们推送消息, //假如推送者自己也订阅了这个主题,消息不会被推送给它自己,以防止自己推送的消息被自己接收。 String topicString = "light002/set" ; char publishTopic[topicString.length() + 1]; //转换成字符数组 strcpy(publishTopic, topicString.c_str()); // 建立发布信息,当前D1引脚状态 String messageString; if(digitalRead(B_led)){ messageString = "on"; } else { messageString = "off"; } char publishMsg[messageString.length() + 1]; //转换成字符数组 strcpy(publishMsg, messageString.c_str()); // 实现ESP8266向主题发布信息,并在串口监视器显示出来 if(client.publish(publishTopic, publishMsg)){ Serial.println("Publish Topic:"); Serial.println(publishTopic); Serial.println("Publish message:"); Serial.println(publishMsg); } else { Serial.println("Message Publish Failed."); }}//程序入口void setup() { pinMode(B_led, OUTPUT); //设置引脚为输出模式 pinMode(led_switch, OUTPUT); // 开关引脚为输入模式 digitalWrite(B_led, LOW);//默认引脚上电低电平 Serial.begin(9600); //设置波特率9600 //********************自动配置网络************************ // wifiManager.setTimeout(20); wifiManager.setConfigPortalTimeout(120); // 自动连接WiFi。以下语句的参数是连接ESP8266时的WiFi名称// wifiManager.autoConnect("AutoConnectAP"); // 如果您希望该WiFi添加密码,可以使用以下语句: wifiManager.autoConnect("8266"); //重置wifi设置 // wifiManager.resetSettings(); // WiFi连接成功后将通过串口监视器输出连接成功信息 Serial.println(""); Serial.print("ESP8266 Connected to "); Serial.println(WiFi.SSID()); // WiFi名称 Serial.print("IP address:\t"); Serial.println(WiFi.localIP()); // IP //******************************************************* client.setServer(mqtt_server, mqtt_server_port);//设置mqtt服务器 client.setCallback(receiveCallback); //mqtt消息处理 ticker.attach(1, tickerCount); // Ticker定时对象 Serial.println("setup结束");}//循环执行 void loop() { int val = digitalRead(B_led); int ctl_val = digitalRead(led_switch); if(ctl_val == 1){//按键按下 delay(400); state = !state; digitalWrite(B_led, state); digitalWrite(led_switch, LOW); int ctl_val = digitalRead(led_switch); Serial.print("开2关值为 :\t"); Serial.println(ctl_val); } if (client.connected()) { client.loop(); if (count = 10){ // 每隔10秒钟发布一次信息 pubMQTTmsg(); count = 0; } }else{// mqtt链接断开,并且已连接上AP,则重连mqtt if(wifiManager.getWLStatusString().equals("WL_CONNECTED")){ reconnect(); } }}继电器及门禁开关分别使用D1、D2引脚作为信号触发,D1在收到开灯指令时输出高电平触发继电器动作,从而点亮灯泡D2连接门禁开关,门禁开关另一端连接开发板3.3V引脚,在按下开关时,将D2引脚拉高,作为手动控制灯开关:按下并松开开关,灯将点亮或熄灭开发板断网时仍可通过门禁开关控制灯光烧录完成后开发板将自动重启,首次未配网时,开发板将启用ssid为8266的接入点,使用无线设备连接该热点,进入网络认证页面完成网络配置.接线图如下:[图片]两条橙色线分别接入3.3v及D2引脚绿色线接5v引脚给继电器供电蓝色线接GND引脚给继电器供电灰色线接D1引脚作为继电器信号线继电器设置为高电平触发模式继电器COM接灯开关内的220V火线继电器NO接灯的控制线开发板上电或重启时,继电器为断开状态,COM口与NO为开路状态,灯不点亮发送开灯指令时,继电器吸合,COM口与NO为短路状态,灯会被点亮使用小爱语音(提前在米家绑定巴法云)控制开关或者手动发送mqtt指令on、off可以完成控制附上硬件购买截图[图片][图片][图片]更新:通过siri控制这里使用php编写一段发布mqtt消息的代码:?php/** * Created by PhpStorm. * User: ZXQ * Date: 2024/3/9 * Time: 23:43 */require('./phpMQTT.php');function console_log($data){ echo("scriptconsole.log(".json_encode($data).")/script");}$server = 'bemfa.com';$port = 9501;$clientId = 'xx';$username = null;$password = null;$clean_session = false;if (($_SERVER['REQUEST_METHOD']==='GET')){ $action = $_GET['action']; if ($action=='on'){ $mqtt = new Bluerhinos\phpMQTT($server, $port, $clientId); if ($mqtt-connect(true, NULL, $username, $password)) { //public用法:publish($topic主题, $content内容, $qos = 0, $retain = false): void $mqtt-publish('light002/set', 'on', 0, false); $mqtt-close(); } else { echo "Time out!\n"; } } if ($action=='off'){ $mqtt = new Bluerhinos\phpMQTT($server, $port, $clientId); if ($mqtt-connect(true, NULL, $username, $password)) { //public用法:publish($topic主题, $content内容, $qos = 0, $retain = false): void $mqtt-publish('light002/set', 'off', 0, false); $mqtt-close(); } else { echo "Time out!\n"; } } echo json_encode("success");}phpMQTT.php来自phpMQTT.php部署完成后,在IOS中添加两条快捷指令,分别为开灯和关灯,操作选择"从URL获取内容",输入开关灯对应URL,然后直接对Siri说开灯或者关灯就可以了,虽说这种方式有点low,但也算实现了吧...[图片]
继续阅读»
2024-01-19
730
淘了一台单网口J4125设备,安装了ESXI虚拟服务器系统,在上面运行了Windows、linux等系统实现家庭服务搭建。ESXI上的虚拟机都会分配一个局域网地址供外部访问,由于某个服务需要使用第二条网络(第二条宽带),单网口无法实现,于是购买Netgear GS108T通过vlan技术扩展设备的网口,该设备有8个网口。网口规划:1口:原网络入口2口:第二条网络入口3口:汇聚口,接入ESXI主机,同时承载两条网络数据,后续在ESXI中使用vlan分离两条网络的数据4-5口:原网络的lan口,相当于在实现ESXI服务器功能的前提下,扩展出了原来网络两个lan口备用6-7口:第二条网络的lan口,类似4-5口作用。8口:交换机管理口,为了4567口能正常访问到汇聚到3口中的虚拟机设备,需要将他们的pvid设为对应的vlanid交换机配置:创建了11、12两个vlan,分别对应原有网络及第二条网络[图片]各vlan的标记设置[图片][图片][图片]这样设置完成后,插在第4-5口的设备(及原网络的扩展lan口)无法访问ESXI中的虚拟机,但能访问这条线路来之前的所有设备(例如光猫管理页面)因此需要修改4-5口的pvid为11,这样从4-5口设备进来的数据,会被打上vlan11标签,可以从1口去除vlan标签后流出,也可带上vlan标签进入3口进行后续流转6-7口则将pvid设置为12(这里是关键点,因为默认所有口的pvid都为1,所有数据从端口进来都会被打上vlan1的标签,无法流转到3口中,造成无法访问虚拟机)在网上看到过类似问题:查看,我之前使用光猫加openwrt路由器实现iptv单线复用也遇到过类似情况,当时理解不够深刻[图片]在此交换机就设置完成了,只有通过8口才能进入交换机管理页面,猜测也可通过修改交换机vlanid为11或12实现在局域网内管理交换机,没有尝试过。ESXI设置:由于ESXI主机只有一个网口,因此我之前用的都是默认的VMNetwork端口组,现在需要新建一个端口组用于分离第二条网络的数据修改VM Network的vlanid为11,新建vlanid为12的端口组同时需要修改ManagementNetwork的vlanid为你常用的id,因为现在入口数据包含了多vlanid的数据,我这边后续只能通过在vlanid11的设备访问ESXI管理页面(交换机的3-4口以及上级光猫下的设备)[图片]这样就完成了配置,在需要第二条网络的虚拟机配置中添加对应网络vlanid的端口组即可
继续阅读»
2023-11-28
1383
外网通过移动的ipv6地址访问内网设备网络环境:移动宽带、光猫拨号、openwrt19.071.首先在光猫管理界面关闭ipv6防火墙(超级密码登陆)[图片]2.openwrt安装软件包:opkg update# opkg install luci-app-socat (官方库中好像没有,自行本地安装)opkg install ddns-scripts_cloudflare.com-v4opkg install luci-app-ddns3.ddns-cloudflare配置[图片]按图上设置,其中lookup hostname填写你的域名domain填写cloudflare中需要更新记录的域名,注意若非顶级域名需要使用@代替.,例如test.zhangxiaoqiang.top需要写成test@zhangxiaoqiang.top高级设置中:[图片]为防止获取到内网ipv6地址,需要使用第三方url方式获取目前的公网ip,ip address source 选择url方式,url填写一个可以获取ipv6的api,默认的比较慢。我的openwrt作为旁路由使用,不需要为其他设备分配ipv6地址,为防止ddns获取到ipv6前缀,我将获取前缀关闭了:Interfaces » WAN6中将Request IPv6-prefix of length 设置为disabled,也可以在上图中修改url方式为使用script方式获取地址,具体script自行编写。配置完成后回到ddns主页,在记录所在的pid列的start按钮启动ddns。4.socat端口转发原生端口转发无法支持ipv6入站请求,故使用该插件完成请提前在cloudflare中添加AAAA记录安装完成后应用入口位于network-socat,配置参考:[图片]更新:不知为何我的wan6会同时获取到/64和/128的公网地址[图片]其中/128的无法通过外网访问,这里改为script方式获取ip,官方文档:detecting_wan_ip_with_script如图修改[图片]/root/script/getIpv6.ship addr show eth0.2 | grep "inet6.*/64.*global" | awk '{print $2}' | awk -F"/" '{print $1}'请按实际情况修改eth0.2以及实际获取到可用的ipv6地址后缀:/64以下是命令解析ip addr show eth0.2 | grep "inet6.*/64.*global"筛选出ipaddr命令结果中ipv6地址那一行,结果为 inet6 xxxx:fe5d:c297/64 scope global dynamicawk '{print $2}'将结果以空格方式分割获取第二个内容,即xxxx:fe5d:c297/64awk -F"/" '{print $1}'再将上述结果根据斜杠 / 分割,打印分割后的第一个字段,最终获取实际ip地址xxxx:fe5d:c297注意wan口所在网口的所有interface都不能设为bridge(例如wan和wan6),否则会导致dhcpv6无法获取ipv6地址:Daemon.err odhcp6c: Failed to send DHCPV6 message to ff02::1:2 (Permission denied)参考文章:关闭光猫ipv6防火墙OpenWRT使用cloudflare DDNS通过socat,使OpenWrt实现ipv6端口转发
继续阅读»