HomeKitのデバイスを、他のコンピュータのコマンドラインから動かす方法を説明します。 Homebridge, Mosquittoを使います。
Raspberry Piの上に、Homebridge, Mosquittoが動いていて、 Appleのホーム.appのにHomebridgeが接続されていて、 パソコン(Mac)の上にMosquittoのクライアント機能がインストールされていることが前提です。 以下のページを見てください。
Homebridgeのコントロールページから、プラグインを選び、検索窓にmqttthingとタイプします。するとHomebridge Mqttthingというプラグインが現れます。
これをインストールします。 インストールが終わると設定画面が出ます。これはいつでも設定できるのですが、 とりあえず、Stateless Programmable Switchを選んでみます。 また、名前をSwitch1とし、MQTTのタイプ名も適当に入れておきます。 タイプ名は、mqtttで始まるようにしました。MQTTTプラグイン関係であることがわかりやすいかとおもいました。
これで保存して、Homebridgeを再起動します(右上方の電源スイッチのようなアイコンを押す)。 再起動後に、アクセサリのタブを見ると、押しボタンスイッチのようなアイコンが見えます。 stateless programmable switchは、テレビのリモコンみたいに押しボタン方式のスイッチです。
コンフィグボタンを押すと、Homebridgeの全体の設定が見えます。まだMQTT Thingしか入れてないのでわかりやすいです。
このうち、
"accessories": [ { "type": "statelessProgrammableSwitch", "name": "Switch1", "topics": { "getOnline": "mqttt/online", "getSwitch": "mqttt/switch" }, "accessory": "mqttthing" } ],
の部分がMQTT Thingの部分です。このプラグインのページに詳しい説明があります。
https://github.com/arachnetech/homebridge-mqttthing#readme
変更したり書き換えたりする場合は、このコンフィグファイルを編集するのが簡単で良いと思います。 テレビのリモコンのように、多数の押しボタンスイッチをまとめて定義することもできます。 LEDの調光器リモコンのように、on/off/明るさ/色合いなどのスイッチをまとめて定義するのに便利です。 これ以外にも多くの特性を持つデバイスが用意されています。
ここではstateless programmable switchを使いましたが、他にも多数のデバイスが使えます。
stateless programmable switchは、(HomeKitから見て)入力専用ですが、 Switchは入出力に対応しています。 設定して、mosquitto_subコマンドでモニターすると動きがわかります。 これを使えば、HomeKitからのon/offを受け取ってLEDを光らせるなどの仕掛けをArduinoで作れます。
HomebridgeがHomeKit環境に登録してあれば、ここで定義したSwitch1がホーム.appに現れています。
このアイコンを右クリックして、「設定」を選ぶと動作をしてできます。
1回押し、2回押し(ダブルクリック)、長押しの選択があります。ここでは1回押しで、隣のIkeaの電球、をonにする設定をしました。
今までの設定で、
でした。Switch1には、1回押し、2回押し(ダブルクリック)、長押しの3種類があります。それぞれに対応するメッセージは、1,2,Lです。 これは設定ファイルで変更することも可能ですが、1,2,Lがデフォルトです。 なので、MQTTの出版をすれば、IKEAの電球をonにできます。パソコンから、
% mosquitto_pub -h 192.168.0.134 -t mqttt/switch -m 1
とすれば良いです。他のターミナルのウィンドウから、
% mosquitto_sub -h 192.168.0.134 -t # -v
として、メッセージをモニターしておくとデバッグに役立ちます。
電球をoffにしたい場合は、2回押しや長押しに電球offを割り当てても良いです。 また、1回押しのたびに反転することもできます。 1回押しに実行されるショートカットを以下のように記述すれば良いです。
これで、
% mosquitto_pub -h 192.168.0.134 -t mqttt/switch -m 1
とするたびに、電球がon/offします。
pahoというライブラリを使いました。
https://www.eclipse.org/paho/index.php?page=clients/python/index.php
以下のプログラムでHomeKitで設定された灯を点滅させることができました。
#!/Library/Frameworks/Python.framework/Versions/3.9/bin/python3 import paho.mqtt.client as mqtt # import MQTT lib. broker_address="192.168.0.134" client = mqtt.Client() #create new instance client.connect(broker_address) #connect to broker client.publish("mqttt/switch","1") #publish
同じくArduinoからの例です。
https://github.com/knolleary/pubsubclient
にある、pubsubclientライブラリを使ってます。
上のページにあるサンプルを参考にしたプログラムが以下です。 ここでは、13番に接続したスイッチが押された時に、MQTTに1を送ります。それでHomeKitが灯をon/offしてくれるはずです。
元のサンプルは、送受信に対応していたので、subscribeの部分も残ってます。その部分は今回は使っていないです。 HomeKitからの指示でArduinoでLEDを点灯させるなどの場合には、このcallbackの関数のところに、その処理を書けば良いです。。 WiFiの設定もしているので、長いですが、多分これで動きます。
#include <WiFi.h> #include <PubSubClient.h> //input & output ping const int PUSHSW=13; //external push switch //WiFi const char SSID[] = "siiolab408_2G"; const char PASSWORD[] = "xxxxxxx"; //PubSubClient (MQTT) const char mqttbroker[] = "192.168.0.134"; //MQTT broker address const int mqttport=1883; //MQTT port const char subtopic[] = "mqttthing/sub"; //mqtt topic to subscribe これは今回使用しません const char pubtopic[] = "mqttthing/switch"; //mqtt topic to publish char clientID[] = "ESP32_xx:xx:xx:xx:xx:xx"; //MAC address is set in setup() WiFiClient wifiClient; PubSubClient mqttClient(wifiClient); void callback(char* topic, byte* payload, unsigned int length) { Serial.print("Message arrived [");Serial.print(topic);Serial.print("] "); String msg = String((char*) payload); Serial.println(msg); } //connecting MQTT void connectMQTT() { mqttClient.setServer(mqttbroker, mqttport); mqttClient.setCallback(callback); Serial.println("Attempting MQTT connection. "); // Loop until we're reconnected while (! mqttClient.connected() ) { // Attempt to connect if (mqttClient.connect(clientID)) { Serial.println("Connected. "); // Once connected, publish an announcement... mqttClient.publish(pubtopic,"Arduino is ready."); // ... and subscribe mqttClient.subscribe(subtopic); } else { Serial.print("Failed, rc="); Serial.print(mqttClient.state()); Serial.println(", try again in 5 seconds. "); // Wait 5 seconds before retrying delay(5000); } } } //connecting WiFi void connectWiFi() { WiFi.begin(SSID, PASSWORD); Serial.print("Attempting WiFi connection. SSID: "); Serial.print(SSID); while (WiFi.status() != WL_CONNECTED) { Serial.print("."); delay(1000); } Serial.print("\nWiFi connected. Assigned address is: "); Serial.println(WiFi.localIP()); } void setup() { pinMode(PUSHSW, INPUT_PULLUP); Serial.begin(115200); // ESP standard speed 115200 while (!Serial) ; // wait for serial port to connect. String wifiMACString = WiFi.macAddress(); //WiFi MAC address wifiMACString.toCharArray(&clientID[6], 18, 0); //"EPS32_xx:xx:xx:xx:xx:xx" Serial.print("\nMQTT clientID: ");Serial.println(clientID); connectWiFi(); connectMQTT(); } //last state of the PUSHSW int lastSWstate; void loop() { if( WiFi.status() == WL_DISCONNECTED) connectWiFi(); if (! mqttClient.connected()) connectMQTT(); mqttClient.loop(); int newSWstate=digitalRead(PUSHSW); if(lastLSWstate!=newLSWstate) { //update only when SW changed if(newLSWstate==LOW) mqttClient.publish(pubtopic,"1"); } lastSWstate=newSWstate; delay(2000); }