3
3

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

More than 1 year has passed since last update.

【SwitchBot】湿度に応じて自動で除湿機を起動する

Last updated at Posted at 2023-06-18

はじめに

今年は梅雨入りの時期が例年より早く、ジメジメとした季節が続いていますね。そこで、室内の湿度を監視し、湿度が高くなったら自動で除湿機を起動し、室内の湿度を下げるような仕組みを構築しました(元々構築してあったのですが、記事にするのが遅くなりました)。流れとしては、Raspberry Pi で湿度を取得し、湿度が高くなったら、 SwitchBotAPI を利用して除湿機を起動し、湿度が低くなったら除湿機を停止します。

準備したもの

※温湿度計はどちらか一つで大丈夫です。値段的にもほとんど変わらないので(ハブミニは必要になりますが…)温湿度計のほうが楽ちんでおすすめです(デジタル表記もしてくれるので分かりやすいですね)。

環境

  • Raspberry Pi 4 Model B
    • CentOS Stream 8
      • Python 3.11.0
        • requests 2.28.2
        • Adafruit-DHT 1.4.0

導入

Raspberry Pi のセットアップ

Raspberry Pi のセットアップについてはまとめてみたので参考にしてみて下さい。

モジュールのインストール

まずは、Pythonに必要なライブラリをインストールしていきましょう。こちらは DHT22 温湿度センサーモジュールを使う場合に必要なライブラリになります。温湿度計は必要ありません。

pip install Adafruit-DHT

SwitchBot の設定

SwichBot アプリ(iOS, Android)から SwitchBot スイッチデバイスの登録をして下さい。[近くのボット] から登録ができるはずです(詳細は割愛します)。この時、名前を変更しておくと後々便利です。次にスイッチデバイスを選択し(ここでは CO2ON を選択していますが、除湿機を選択して下さい)、設定からクラウドサービスの有効化を行って下さい。
IMG_0053.jpgIMG_0054.jpg

開発者トークンの取得

SwitchBot デバイスにアクセスするために必要な開発者トークンの取得を行います。SwitchBot アプリの [プロフィール] より [設定] を開きます。[アプリバージョン] の項目を10回連続でタップすると、[開発者向けオプション] が表示されます。

IMG_0055.jpgIMG_0056.jpgIMG_0057.jpg

[開発者向けオプション] を開き [トークンを取得] をタップします。すると、トークンが表示されるのでコピーし PC に送って下さい。

IMG_0058.jpgIMG_0059.jpg

デバイス ID の取得

続いて、SwitchBot デバイスのデバイス ID を取得していきます。以下のスクリプトの "YourAccessToken" 部分を先程取得したトークンに置き換えて実行して下さい。いちいちコマンドを打つのは面倒なのでスクリプトとして保存しておいたほうが後々楽です。

getdeviceid.sh
#!/bin/sh

TOKEN="YourAccessToken"

/usr/bin/curl -s -X GET "https://api.switch-bot.com/v1.0/devices" -H "Authorization:$TOKEN"| sed 's/{\"deviceId\"/\n{\"deviceId\"/g' 

実行すると以下のようにデバイス ID が表示されます。

Result
{"statusCode":100,"body":{"deviceList":[
{"deviceId":"C02CF2","deviceName":"CO2OFF","deviceType":"Bot","enableCloudService":true,"hubDeviceId":"F144D9"},
{"deviceId":"C27E36","deviceName":"温湿度計 07","deviceType":"Meter","enableCloudService":true,"hubDeviceId":"F144D9"},
{"deviceId":"C7E384","deviceName":"カーテン D3","deviceType":"Curtain","hubDeviceId":"000000","curtainDevicesIds":["E76E3C","C7E384"],"calibrate":true,"group":true,"master":false,"openDirection":"right"},
{"deviceId":"C9DFFF","deviceName":"CO2ON","deviceType":"Bot","enableCloudService":true,"hubDeviceId":"F144D9"},
{"deviceId":"D371D0","deviceName":"除湿機","deviceType":"Bot","enableCloudService":true,"hubDeviceId":"F144D9"},
{"deviceId":"E76E3C","deviceName":"カーテン CC","deviceType":"Curtain","enableCloudService":true,"hubDeviceId":"F144D9","curtainDevicesIds":["E76E3C","C7E384"],"calibrate":true,"group":true,"master":true,"openDirection":"left"},
{"deviceId":"EE93FD","deviceName":"リモートボタン 6A","deviceType":"Remote","enableCloudService":false,"hubDeviceId":"000000"},
{"deviceId":"F144D9","deviceName":"ハブミニ 7A","deviceType":"Hub Mini","hubDeviceId":"000000"}],"infraredRemoteList":[]},"message":"success"}

今回欲しいデバイス ID は"除湿機"の "D371D0" と"温湿度計"の "C27E36" になります。メモしておきましょう。これで、SwichBot デバイスを API で制御する準備ができました(温湿度計を使わずに DHT22 温湿度センサーモジュールを使う場合は温湿度計のデバイス ID は必要ありません)。

SwitchBot デバイスの操作

続いて、SwitchBotAPI を実行します。ソースコードは以下の通りです。

switchbot.py
#!/usr/bin/env python3
import requests, json
# 取得したトークンに置き換えて下さい。
ACCESS_TOKEN="YourAccessToken"
API_BASE_URL="https://api.switch-bot.com"

# Switch ON/OFF
def device_control3():
# 取得したデバイス ID に置き換えて下さい。
    DEVICEID="D371D0"
    headers = {
        # ヘッダー
        'Content-Type': 'application/json; charset: utf8',
        'Authorization': ACCESS_TOKEN
    }
    url = API_BASE_URL + "/v1.0/devices/" + DEVICEID + "/commands"
    body = {
        # 操作内容
        "command":"turnOn",
        "parameter":"default",
        "commandType":"command"
    }
    ddd = json.dumps(body)
    res = requests.post(url, data=ddd, headers=headers)
    return res

# DHT22 温湿度センサーモジュール を使う場合は必要ありません。
def device_control4() -> dict:
# 取得したデバイス ID に置き換えて下さい。
    DEVICEID="C27E36"
    headers = {
        # ヘッダー
        'Content-Type': 'application/json; charset: utf8',
        'Authorization': ACCESS_TOKEN
    }
    url = API_BASE_URL + "/v1.0/devices/" + DEVICEID + "/status"
    res = requests.get(url, headers=headers)
    #print(res.text)
    return res.json()["body"]

if __name__ == '__main__':
    # 処理呼び出し
    #device_control3()
    #device_control4()

デバイスごとに設定するパラメータ(command, parameter, commandType)については、
SwitchBotAPI の README に記載されています。参考にしてみて下さい。実行してスイッチデバイスがきちんと動けば問題ありません。

湿度の測定

SwitchBot 温湿度計の場合

先程作成した switchbot.py と同じディレクトリに以下のソースコードを保存して下さい。

chkhumidity.py
#!/usr/bin/env python3
import time
import switchbot

triggered = False
while True:
    time.sleep(5)
    res=switchbot.device_control4()
    t=res.get("temperature")
    h=res.get("humidity")
    print( "Temp= {0:0.1f} \u2103".format(t))
    print( "Humidity= {0:0.1f} %".format(h))

実行した結果、正しく湿度が表示されていれば問題ありません(ここでは同時に温度も取得しています)。

[web@localhost humidity]$ ./chkhumidity2.py 
Temp= 28.1 ℃
Humidity= 52.0 %

DHT22 温湿度センサーモジュールの場合

続いて、DHT22 温湿度センサーモジュール(以下、センサー)を使う場合の説明をしていきます。このセンサーには以下の画像のようにデータ(DAT) のやり取りをするピンと電源用のピン(VCC)、グラウンド(グランド)用のピン(GND)の3種類があります。これを正しく Raspberry Pi の GPIO に挿し込まなければ機能しません。

GPIO の概要について軽く説明しておきます。GPIO を構成するピンは、主に4つの機能に分類できます。

  • 電源ピン:電源を供給するピン。3.3V と 5V があります
  • GNDピン:グランドピン。電位の基準を決めている回路のこと
  • DNC:ユーザが使用できないピン。古いタイプの Raspberry Pi に存在しましたが、現在では I/Oとして使用できるように改良されています
  • I/O:汎用入出力ピン

Raspberry Pi 4 の GPIO ピン配置(ピンマッピング)については、以下記事にまとめてあるので参考にして下さい。

画像1.png

デバイスによっては、代替機能が利用されている場合があり、競合する可能性があります。そのため、代替機能がある GPIO ピンの利用は避けた方がいいかもしれません。また、GPIO ピンによっては接触不良なのかきちんとセンサーのデータが読み込めないことがあります。その時は GPIO ピンを差し替えたりしてみて下さい。私の場合、GPIO 22ピンを利用しようとしたのですが、データが読み込めなかったので GPIO 27ピンを利用しています。電源は 3.3V, 5V どちらでも構いません。

動作確認のために以下のソースコードを保存し、root 権限で実行して下さい。なぜなら GPIO ピンへのアクセスには通常 root 権限が必要になるからです。ソースコード中のピン番号は、接続したピン番号に置き換えて下さい。

chkhumidity.py
#!/usr/bin/env python3
import Adafruit_DHT as DHT
import time

sensor = DHT.DHT22
# 接続したピン番号に置き換えて下さい。
pin = 27
triggered = False
while True:
    time.sleep(5)
    h,t = DHT.read_retry(sensor, pin)
    print( "Temp= {0:0.1f} \u2103".format(t))
    print( "Humidity= {0:0.1f} %".format(h))

実行した結果、正しく湿度が表示されていれば問題ありません(ここでは同時に温度も取得しています)。

[root@localhost humidity]# ./chkhumidity.py 
Temp= 29.6 ℃
Humidity= 58.8 %

本実装

スイッチデバイスが動くことが確認でき、湿度も取得することがでたので、続いて室内の湿度に応じて自動で除湿機を起動するコードの実装に移ります。SwitchBot 温湿度計の場合と DHT22 温湿度センサーモジュールの場合がありますが、それぞれ湿度の校正が異なっており、同じ数値を出してはくれません。なので、湿度のしきい値に関しては実際に運用してみて、決めて下さい。以下、ソースコードになります。

SwitchBot 温湿度計の場合

humidity.py
#!/usr/bin/env python3
import time
import switchbot

triggered = False
while True:
    res=switchbot.device_control4()
    humidity=res.get("humidity")
    #print(humidity)
    if humidity >= 62 and not triggered:
      res = switchbot.device_control3()
      triggered = True
    if humidity <= 55 and triggered:
      res = switchbot.device_control3()
      triggered = False
    time.sleep(120)

DHT22 温湿度センサーモジュールの場合

humidity.py
#!/usr/bin/env python3
import Adafruit_DHT as DHT
import time
import switchbot

sensor = DHT.DHT22
pin = 27
triggered = False
while True:
    data = DHT.read_retry(sensor, pin)
    humidity = data[0]
    temp = data[1]  
    print(humidity,temp)
    if humidity >= 62 and not triggered:
      res = switchbot.device_control3()
      triggered = True
    if humidity <= 55 and triggered:
      res = switchbot.device_control3()
      triggered = False
    time.sleep(120)

service への登録

常時稼働させておくために、サービスとして登録しておきましょう。登録するためには root 権限が必要になります。まずは、以下ディレクトリに移動してください。

[root@localhost ~]# cd /etc/systemd/system/

その後、以下のコードを humidity.service として保存して下さい。

humidity.service
[Unit]
Description=humidity

[Install]
WantedBy=multi-user.target

[Service]
User=root
# 実行するコードのディレクトリに置き換えて下さい
ExecStart=/root/humidity/humidity.py
Restart=always

最後にサーバ起動時にサービスも自動起動してくれるように設定します。

サービスの自動起動
[root@localhost system]# systemctl start humidity.service
[root@localhost system]# systemctl enable humidity.service 
Created symlink /etc/systemd/system/multi-user.target.wants/humidity.service → /etc/systemd/system/humidity.service.

以下コマンドを実行して Active: active (running) と表示されていれば問題ありません。

サービスの状態
[root@localhost system]# systemctl status humidity
● humidity.service - humidity
   Loaded: loaded (/etc/systemd/system/humidity.service; enabled; vendor preset: disabled)
   Active: active (running) since Fri 2018-06-22 20:12:09 JST; 4 years 11 months ago
 Main PID: 1072 (python3)
    Tasks: 1 (limit: 23882)
   CGroup: /system.slice/humidity.service
           └─1072 python3 /root/humidity/humidity2.py

Jun 22 20:12:09 localhost.localdomain systemd[1]: humidity.service: Service RestartSec=100ms expired, s>
Jun 22 20:12:09 localhost.localdomain systemd[1]: humidity.service: Scheduled restart job, restart coun>
Jun 22 20:12:09 localhost.localdomain systemd[1]: Stopped humidity.
Jun 22 20:12:09 localhost.localdomain systemd[1]: Started humidity.

最後に

冷房ではなく、ドライ(除湿)を使えばいいのではとも思うのですが、なんとなくドライ使うのが嫌なんですよね(なんででしょう…)。そのため、湿気が気になるので除湿機を購入し今回の自動で除湿機を起動する機構を構築しました。いかがったでしょうか。もう梅雨入りしているため、記事の投稿(2023/06/18)が遅くなってしまいましたが、皆さんの生活が快適になればと思います。

3
3
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
3
3

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?