0
0

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.

【目指せ完走賞】なりかくんのDiscordAdvent Calendar 2022

Day 12

【12日目】部屋の温湿度をSwitchBotで取得してDiscordに自動で送る(Node.js)

Posted at

はじめに

こんにちは、なりかくんです。
今回は、Node.jsで部屋に置いている温湿度計(SwitchBot)でDiscordに自動で送るプログラムを作ってみようと思います。

今回使う機器

今回は、SwitchBot 温湿度計とSwitchBot ハブミニです。

SwitchBot 温湿度計は長く使っていますが全く電池を交換していないのにずっと動いていますしかなり正確に温度と湿度を見ることが出来るので重宝しています。そしてなんといってもスマホからグラフ化してみることが出来るのがワンポイントです。

SwitchBot ハブミニは、温湿度計をインターネットから操作できるようにするだけではなく、部屋の赤外線などで操作できる家電もスマホなどで外から操作できるように出来るという優れたものです。

SwitchBot API

今回は、SwitchBotにあるAPIを使って温度と湿度を取得します。なお、このAPIを使うにはハブミニが必要になります。
SwitchBot APIの詳細の仕様はGitHubに公開されていますのでそちらをお読みください。

まあ、仕様書なんて読まなくても非常に簡単に使うことが出来るのですが(笑)

API KeyとClient Secretの取得はSwitchBotのアプリから行う事が出来ます。
[プロフィール] -> [設定]で、[アプリのバージョン]を10回タップすることで開発者オプションが表示されます。
そこにAPI KeyとClient Secretが表示されます。

SwitchBotのDeviceIdを取得する。

まず、温湿度計のDeviceIdを取得する必要があります。以下のURLから取得できます。

GET https://api.switch-bot.com/v1.1/devices

以下のコードで取得できます。

switchbot.js Get DeviceId
const request = require("request");
const crypto = require("crypto");

const token = "Token";
const secret = "Client Secret";
const t = Date.now();
const nonce = "discordSwitchBot";
const data = token + t + nonce;
const signTerm = crypto.createHmac('sha256', secret)
    .update(Buffer.from(data, 'utf-8'))
    .digest();
const sign = signTerm.toString("base64");

let result = getDeviceId();
console.log(result);

async function getDeviceId () {
    let getDevices = await getRequest({
        url: "https://api.switch-bot.com/v1.1/devices",
        json: true,
        headers: {
            "Authorization": token,
            "sign": sign,
            "nonce": nonce,
            "t": t
        }
    });
    return getDevices.body;
}

function getRequest(option) {
    return new Promise((resolve, reject) => {
        request(option, function (error, response, body) {
            if (error) {
                reject(error);
            } else {
                resolve(body);
            }
        });
    });
}

これで取得すると、私の環境では以下のように出力されます。(ID等はすべてXXXXXXXXXXで隠しています。)

{
  deviceList: [
    {
      deviceId: 'XXXXXXXXXX',
      deviceName: 'なりかくんのハブミニ',
      deviceType: 'Hub Mini',
      hubDeviceId: '000000000000'
    },
    {
      deviceId: 'XXXXXXXXXX',
      deviceName: 'なりかくんの温湿度計',
      deviceType: 'Meter',
      enableCloudService: true,
      hubDeviceId: 'XXXXXXXXXX'
    }
  ],
  infraredRemoteList: [
    {
      deviceId: 'XXXXXXXXXX',
      deviceName: 'なりかくんのエアコン',
      remoteType: 'Air Conditioner',
      hubDeviceId: 'XXXXXXXXXX'
    }
  ]
}

今回は、これの温湿度計のDeviceIdを利用します。これをメモしておきましょう。

温度と湿度を取得する

次に先ほど取得したDeviceIdから温湿度計のデバイスステータスを確認して、温度と湿度を取得します。
以下のURLから取得できます。

GET https://api.switch-bot.com/v1.1/devices/<DeviceId>/status

これらをまとめて先ほど作ったプログラムと合わせるとこのようなコードを書くことが出来ます。

async function getDeviceStatus (Id) {
    let getDevices = await getRequest({
        url: `https://api.switch-bot.com/v1.1/devices/${Id}/status`,
        json: true,
        headers: {
            "Authorization": token,
            "sign": sign,
            "nonce": nonce,
            "t": t
        }
    });
    return [getDevices.body.humidity, getDevices.body.temperature];
}

これを実行すると配列として返ってきます。配列は、[getDevices.body.humidity, getDevices.body.temperature]としているので、humidityが湿度、temperatureが温度となります。
この記事を書いている時点で、以下の配列が出力されました。

[ 39, 23 ]

すべて組み合わせて1分毎に送る

では、先ほど紹介したコードをすべて組み合わせて1分毎にDiscordに送信するプログラムを作ります。
Discordへの送信は、WebHookを使って送信します。WebHookでの送信は9日目と11日目に扱っているのでそちらをご覧ください。

今回作ったコード全体はこちらになります。

switchbot.js
const request = require("request");
const crypto = require("crypto");

const token = "Token";
const secret = "Secret";
const t = Date.now();
const nonce = "discordSwitchBot";
const data = token + t + nonce;
const signTerm = crypto.createHmac('sha256', secret)
    .update(Buffer.from(data, 'utf-8'))
    .digest();
const sign = signTerm.toString("base64");

const deviceId = "DeviceId";

sendData();
setInterval(sendData, 60000);

async function sendData () {
    let getResult = await getDeviceStatus(deviceId);
    let time = parseInt(new Date().getTime()/1000);
    await getRequest({
        url: "WebHook URL",
        method: "POST",
        json: {
            content: `温度: ${getResult[1]}° , 湿度: ${getResult[0]}% (<t:${time}>)`
        }
    });
}

async function getDeviceStatus (Id) {
    let getDevices = await getRequest({
        url: `https://api.switch-bot.com/v1.1/devices/${Id}/status`,
        json: true,
        headers: {
            "Authorization": token,
            "sign": sign,
            "nonce": nonce,
            "t": t
        }
    });
    return [getDevices.body.humidity, getDevices.body.temperature];
}


async function getDeviceId () {
    let getDevices = await getRequest({
        url: "https://api.switch-bot.com/v1.1/devices",
        json: true,
        headers: {
            "Authorization": token,
            "sign": sign,
            "nonce": nonce,
            "t": t
        }
    });
    return getDevices.body;
}

function getRequest(option) {
    return new Promise((resolve, reject) => {
        request(option, function (error, response, body) {
            if (error) {
                reject(error);
            } else {
                resolve(body);
            }
        });
    });
}

これを実行してみると、以下のように毎分温度と湿度が送信されるようになりました。
image.png

最後に

SwitchBotを活用してほかにもいろいろなことが出来るような気がしてきました。
以上です、最後までお読みいただきありがとうございました。

0
0
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
0
0

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?