Raspberry Piで、インターホンが鳴ったらプッシュ通知


家にいてインターホンが鳴っても気付かない時があったので、インターホンが鳴ったらスマホ宛にプッシュ通知を飛ばすようにしました。賃貸住宅なので勝手にインターホンをハックしたりはできないので、工事等が不要な方法を使います。

概ね下記記事と同様にRaspberry PiでGroveの光センサーを使ってインターホンの画面が付くのを検知し、SlackのIncoming Webhookを叩きました。

環境

ハードウェア面

その他、Raspberry Piセットアップ時には有線LANケーブル、有線USBキーボード・マウス、microHDMIケーブルが必要です。一度セットアップした後はローカルのWi-FiネットワークでSSHすればこれらは不要になります。

ソフトウェア面

  • Raspberry Pi debian_version 10.8
  • Python 3.7.3
  • GrovePiファームウェアバージョン 1.4.0

準備

Raspberry Piのセットアップ

本来Raspberry PiのセットアップはOSインストールメディアを焼く所から始めますが、今回OSインストール済SDカード同梱の物を購入したため省略します。Raspberry Piを起動するとOSのセットアップが始まりますがこちらも省略します。Linuxに慣れていれば特に問題は無いと思いますし、Web上にも様々な解説記事等が挙がっています。私は下記動画を見てセットアップの流れを掴んでおきました。

OSが使えるようになったら、「設定」→「Raspberry Piの設定」→「インターフェイス」から、SSHとI2Cを有効にしておきます。Web上にある記事によってはraspi-configコマンドからAdvanced Optionsを変更する、といった手順が紹介されている事がありますが、どうやらRaspberry Piのバージョン等によって異なるようです。

Raspberry Pi自体の初期設定には色々な記事がWeb上にありますが、個人的にはSSH接続設定とetckeeperインストールは実行しておいた方が良いと思います。

$ sudo apt install etckeeper

また私の場合は、プロセスを動かしっぱなしにしたりSSHが切れても作業を続行するためにscreenコマンドを準備しておきました。

GrovePi+と光センサーのセットアップ

RaspberryPiの準備ができたら、次にGrovePiのセットアップをしていきます。

  1. RaspberryPiとGrovePiを接続
  2. GrovePiのA0にlight sensorを接続
  3. (LEDを使うなら)D4にLEDを接続

下記公式ガイドに沿って、インストールしていきます。

私はこの後の工程で相当ハマったので色々なコマンドを試しましたが、たぶん$ curl -kL dexterindustries.com/update_grovepi | bash$ sudo rebootだけで良いと思います。

先程の公式ガイドにある$ i2cdetect -y 1というコマンドで04が下記のように表示されたら成功のようです。

$ sudo i2cdetect -y 1
     0  1  2  3  4  5  6  7  8  9  a  b  c  d  e  f
00:          -- 04 -- -- -- -- -- -- -- -- -- -- -- 
10: -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- 
20: -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- 
30: -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- 
40: -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- 
50: -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- 
60: -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- -- 
70: -- -- -- -- -- -- -- --   

Pythonで光センサーの値を取得

適当なディレクトリ(私の場合は~/dev/grove-light-sensor)に、Pythonのプログラムを記述していきます。

まず~/Dexter/GrovePi/Software/Python/grovepi.pyにGrovePiを扱うライブラリがありますので、コピーして持ってきます。

$ cd ~/dev/grove-light-sensor
$ cp ~/Dexter/GrovePi/Software/Python/grovepi.py ./

次に、このライブラリを使って光センサーのプログラムを書いていきます。

main.py
import grovepi
from time import sleep

while True:
    light = grovepi.analogRead(0)
    print(light)
    if light >= 100:
        print("light!")
    sleep(1)

あとは$ python3 main.pyすれば動く...はずなのですが、実際にはgrovepi.analogRead(0)の所でプログラムが止まってしまいました。Web上の情報を調べてみるとファームウェアバージョンが古い場合にこのような事象が起きることが多いようなのですが、アップデートしても直りませんでした。どうもこの記事によるとgrovepi.pyの不具合修正が必要のようでした。いろいろprintデバッグで調べた結果、私の場合は下記変更をしました。

grovepi.py
def read_identified_i2c_block(read_command_id, no_bytes):
        data = [-1]
-       while data[0] != read_command_id[0]:
+       while len(data) <= 1:
                data = read_i2c_block(no_bytes + 1)

        return data[1:]

この修正は下記Pull Requestにしましたが、正直に言って私自身もこれは何を直しているのかあまり分かってないので自信は無いです。

こうして、下記のような感じで適当な値が取れるようになったら成功です。

$ python3 main.py
10
10
10

この値が光によって上下します。私の場合、自宅の南向き窓の近くだと太陽光が常にあたって700ぐらいの値になりました。手でセンサーを覆うと0になり、インターホンに近づけて光を付けると50〜100ぐらいになりました。

LED点滅の実装

インターホンの検知だけならば必要はありませんが、GrovePiのD4にGrove LEDを刺してから下記プログラムを実行するとLEDを光らせる事ができます。

led.py
import time
import grovepi

led = 4

def blink(sec=60):
    for _i in range(sec):
        grovepi.digitalWrite(led, 1)
        time.sleep(0.5)

        grovepi.digitalWrite(led, 0)
        time.sleep(0.5)

if __name__ == "__main__":
    blink(10)

$ python3 led.pyで、10秒間点滅すると思います。

先述のanalogRead()が動かない場合などに、こちらのプログラムでdigitalWrite()は動くかどうかの検証もできます。

Slack通知の実装

PythonからSlackへ通知を送る方法は多くの情報があるため省略します。SlackでIncoming Webhookを設定して、POSTリクエストを送ればなんでもOKです。こちらの記事の方法がライブラリ等も必要なく楽だと思います。

最終的に、光センサーを検知するメインプログラムは下記のようになりました。

import grovepi
# slack.pyというファイルにdef post_slack():で通知関数を実装した場合
from slack import post_slack
# led.pyというファイルにdef blink():でLED点滅関数を実装した場合
from led import blink

from time import sleep
from datetime import datetime

light_before = float('inf')

while True:
    sleep(1)
    light_now = grovepi.analogRead(0)
    # たまに異常値が返ってくるため、その場合はスキップ
    if light_now == 65535:
        continue
    # 前回との比較で明るさが10以上上がっていたら
    if light_now - light_before >= 10:
        post_slack("インターホンがなったと思うよ 明るさ:{}".format(light_now))
        # 連続して通知が行かないよう待つついでにLEDを光らせる。LEDが無い場合はsleep(60)
        blink(60)
    print('{},{}'.format(datetime.now(), light_now))
    light_before = light_now

プロセス監視

光センサー検知プログラムが動いているかどうかを監視し、止まっていた場合もSlackに通知するようにします。まずは下記のようなプログラムを記述します。

process_check.py
import psutil
# slack.pyというファイルにdef post_slack()で通知関数を実装した場合
from slack import post_slack

ok = False

for ps in psutil.process_iter():
    if ps.cmdline() == ["python3", "main.py"]:
        print("ok!", ps.pid)
        ok = True
        break

if ok == False:
    post_slack("プロセスが動いてないかも")

これを/etc/crontabで適当なタイミングに定期実行させておきます。

/etc/crontab
# 1時間に1回動かす例
25 *    * * *   pi /usr/bin/python3 /home/pi/dev/grove-light-sensor/process_check.py

本番環境(自宅のインターホン)へのデプロイ

自宅のインターホンの近くには通常ディスプレイやキーボード・マウスを置きづらいと思われるので、それらを外してもRaspberry Piが動かせるようWi-FiからSSH接続できることを確認しておきます。電源だけはインターホンの近くに必要になります。

私の家のインターホンは画面があるので、その一部分を光で検知できるようにセンサーをGrove Wrapperとレゴブロックで覆いました。センサーが室内の照明等の影響を受けないように覆う必要がありますが、レゴブロックがぴったりでした。

インターホン画面の近くの箱状の部分の中にLight Sensorがあります。下にあるのはRaspberry Pi本体のケースをレゴブロックでついでに作ったものです。私の自宅にはインターホンの下にカラーボックスがあるのですが、それとレゴの高さを足してもインターホンまで届かなかったので、一旦その辺にあったデカい箱の上に置いています。

動物やTNTがあるのは、レゴショップにマインクラフトのレゴブロックがあったのでついでに買ったものです。

これでインターホンを鳴らした所、無事にSlackから通知が届きました。

まとめ

Raspberry Piを使って、インターホンが鳴ったらプッシュ通知が飛ぶようにできました。

参考文献