Nature Remoで温度計を作ろう!その2

Nature Remo活用シリーズ

0.開封編 1.ダイキンのエアコン 2.ダイキンのエアコン、その2
3.アイリスオーヤマのサーキュレーター 4.温度計 5.温度計、その2

さて前回からの続きです。今回はコードを書いて温度計を作ります。

Perlで書いてcronで定期実行

今回作りたいプログラムはNature RemoのAPIから取得した温度と湿度の値をブラウザで表示するという簡単なプログラムです。となると特定のURLにアクセスしたタイミングでAPIコールすればいいだけですが、折角プログラムを書いて情報を取得するのであれば、定期的に実行してデータを貯め、あとから見返せるようにしておくほうが面白いかなと思いました。もしかしたら温度と湿度の変化から何か発見があるかもしれません。また結果を予めファイルに落としておけば、温度計アプリの表示も早くなります。という訳でプログラムは定期実行、温度表示は結果を書き込まれたHTMLファイルを表示するという形にしました。

実行環境はさくらのレンタルサーバ

定期実行となるとサーバ的なマシンが必要になります。本サイトでもお世話になっているさくらのレンタルサーバ(スタンダード)はPerlも使えてcronも使えますので、この環境を使わない手はありません。言語は使い慣れているPerlで書きます。

Nature RemoのAPIから温度計を作るPerlプログラム

前置きが長くなりましたが、完成したコードは以下のとおりです。ここに大公開!

#!/usr/bin/perl

$| = 1;

#モジュールを指定
use lib '/home/XXXX/cpan/lib/perl5/site_perl/5.8';

#実行時刻の取得
($sec, $min, $hour, $mday, $mon, $year,$wday, $yday, $isdst) = localtime;
$yyyy = $year + 1900;
$month = $mon + 1;

#ファイル指定
$log = '/home/XXXX/remo-api/remo-api-result.csv';
$tmp = '/home/XXXX/remo-api/tmp.json';
$html = '/home/XXXX/remo-api/kion.html';

#アクセストークンを指定
$apikey = "XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX";

#コマンドと引数を指定
$curl = "/usr/local/bin/curl";
$curlopt = "-X GET";
$url = "https://api.nature.global/1/devices";
$header1 = "-H \"accept: application/json\"";
$header2 = "-H \"Authorization: Bearer $apikey\"";
$jq = "/home/XXXX/bin/jq";

#APIアクセスを実行
$remoapi = `$curl $curlopt $url $header1 $header2`;

if ($remoapi != "") {
	exit:
	} else {
	open(TMP, ">$tmp");
	flock(TMP, 2);
	print(TMP "$remoapi");
	close(TMP);
	}

#センサー値の取り出し
$name = `$jq -r .[].name $tmp`;
$firm = `$jq -r .[].firmware_version $tmp`;
$te = `$jq .[].newest_events.te.val $tmp`;
$hu = `$jq .[].newest_events.hu.val $tmp`;
$il = `$jq .[].newest_events.il.val $tmp`;
$mo = `$jq .[].newest_events.mo.val $tmp`;

chomp($name);
chomp($firm);
chomp($te);
chomp($hu);
chomp($il);
chomp($mo);

#HTMLファイル生成
open(FILE, ">$html");
flock(FILE, 2);
print(FILE "<!doctype html>\n");
print(FILE "<html lang=\"ja\">\n");
print(FILE "<head>\n");
print(FILE "<meta charset=\"UTF-8\">\n");
print(FILE "<title>kion</title>\n");
print(FILE "<style>\n");
print(FILE "body { background:#000000; color:#ffffff}\n");
print(FILE "span.font1 {font-size: 100%; }\n");
print(FILE "span.font2 {font-size: 300%; }\n");
print(FILE "</style>\n");
print(FILE "</head>\n");
print(FILE "<body>\n");
print(FILE "<span class=\"font1\">ただいまの $name の気温と湿度は</span><br>\n");
print(FILE "<span class=\"font2\"> $te \℃ / $hu \%</span><br>\n");
print(FILE "</body>\n");
print(FILE "</html>\n");
close(FILE);

#ログを出力
open(LOG, ">>$log");
flock(LOG, 2);
print(LOG "$yyyy/$month/$mday $hour:$min:$sec,name,$name,firmware_version,$firm,te,$te,hu,$hu,il,$il,mo,$mo\n");
close(LOG);

exit:

プログラム解説

折角なので要点を解説します。

#モジュールを指定
use lib '/home/XXXX/cpan/lib/perl5/site_perl/5.8';

PerlでJSONを扱えるようにするモジュール、JSON.pmがあるパスを指定します。尚、さくらのレンタルサーバではインストールされていないので、自身でインストールし、パスを指定します。

#ファイル指定
$log = '/home/XXXX/remo-api/remo-api-result.csv';
$tmp = '/home/XXXX/remo-api/tmp.json';
$html = '/home/XXXX/remo-api/kion.html';

各種ファイルを指定します。$logは定期的に取得したデータをログとして保存するCSVファイル、$tmpはjsonデータを一時的に保存しておくファイル、$htmlは温度計として表示するためのHTMLファイルです。

#アクセストークンを指定
$apikey = "XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX";

前回取得したアクセストークンを指定します。

#コマンドと引数を指定
$curl = "/usr/local/bin/curl";
$curlopt = "-X GET";
$url = "https://api.nature.global/1/devices";
$header1 = "-H \"accept: application/json\"";
$header2 = "-H \"Authorization: Bearer $apikey\"";
$jq = "/home/XXXX/bin/jq";

今回の実装は前回試したcurlコマンドとjqコマンドの組み合わせをPerlで実行させるだけです。Perlはバッククォートでコマンドを囲うと実行できる機能があるのでこれを利用します。まずは各種コマンドのパスを変数に代入します。

尚、さくらのレンタルサーバはjqコマンドは入っていないので、自身でコンパイルしてインストールする必要があります。

#APIアクセスを実行
$remoapi = `$curl $curlopt $url $header1 $header2`;

if ($remoapi != "") {
	exit:
	} else {
	open(TMP, ">$tmp");
	flock(TMP, 2);
	print(TMP "$remoapi");
	close(TMP);
	}

$remoapiにcurlコマンドの標準出力を代入します。コマンドはバッククォートで囲って実行します。$remoapiが空でなければ$tmpに一時保存します。

#センサー値の取り出し
$name = `$jq -r .[].name $tmp`;
$firm = `$jq -r .[].firmware_version $tmp`;
$te = `$jq .[].newest_events.te.val $tmp`;
$hu = `$jq .[].newest_events.hu.val $tmp`;
$il = `$jq .[].newest_events.il.val $tmp`;
$mo = `$jq .[].newest_events.mo.val $tmp`;

chomp($name);
chomp($firm);
chomp($te);
chomp($hu);
chomp($il);
chomp($mo);

jqコマンドを使って必要な値を取り出します。jqコマンドを使うと必ずお尻に改行コードが付いてしまうのでchompコマンドで叩き落とします。

#HTMLファイル生成
open(FILE, ">$html");
flock(FILE, 2);
print(FILE "<!doctype html>\n");
print(FILE "<html lang=\"ja\">\n");
print(FILE "<head>\n");
print(FILE "<meta charset=\"UTF-8\">\n");
print(FILE "<title>kion</title>\n");
print(FILE "<style>\n");
print(FILE "body { background:#000000; color:#ffffff}\n");
print(FILE "span.font1 {font-size: 100%; }\n");
print(FILE "span.font2 {font-size: 300%; }\n");
print(FILE "</style>\n");
print(FILE "</head>\n");
print(FILE "<body>\n");
print(FILE "<span class=\"font1\">ただいまの $name の気温と湿度は</span><br>\n");
print(FILE "<span class=\"font2\"> $te \℃ / $hu \%</span><br>\n");
print(FILE "</body>\n");
print(FILE "</html>\n");
close(FILE);

温度計となるHTMLファイルを出力します。

#ログを出力
open(LOG, ">>$log");
flock(LOG, 2);
print(LOG "$yyyy/$month/$mday $hour:$min:$sec,name,$name,firmware_version,$firm,te,$te,hu,$hu,il,$il,mo,$mo\n");
close(LOG);

最後にログとして各種値をCSVファイルとして追記していきます。

cronの実行間隔は5分に一回

プログラムができましたのでcronに登録します。Nature RemoのAPIの更新間隔は分からなかったのでとりあえず5分間隔にしました。温度計なので5分に一回は更新してほしいなという希望的観測も含まれます。

*/5 * * * * cd /home/XXXX/bin/remo-api ; /usr/bin/perl /home/XXXX/bin/remo-api/remo-api.pl 1> /dev/null

crontabにはこのように記述します。

さあ、アクセスしてみよう!Nature Remo温度計

プログラムも書いてcronの登録も終わりましたのであとは自動で温度計HTMLファイルが出力されるのを待つだけです。

温度と湿度がHTMLファイルで出力されてブラウザで表示できるようになりました!成功です!!

ちなみにファイルは kion.html (ダミー)こんな感じに生成されます。

これで部屋の温度と湿度が知りたいときは、PCを起動してブラウザを立ち上げて特定のURLにアクセスすれば分かるようになりました。(面倒くさ!核爆

と言うのは冗談で、私はオリジナルのブラウザスタートページを作っていてiframeで埋め込んで使っています。

おまけ

一ヶ月ほど動かしてログを取っていたところ2点ほど発見がありました。

APIの更新間隔は5分でよい

2019/12/6 16:10:0,name,ダイニングルーム,firmware_version,Remo/1.0.62-gabbf5bd,te,13,hu,34,il,58.2,mo,1
2019/12/6 16:15:0,name,ダイニングルーム,firmware_version,Remo/1.0.62-gabbf5bd,te,13,hu,34,il,39,mo,1
2019/12/6 16:20:0,name,ダイニングルーム,firmware_version,Remo/1.0.62-gabbf5bd,te,13,hu,34,il,28.79,mo,1
2019/12/6 16:25:0,name,ダイニングルーム,firmware_version,Remo/1.0.62-gabbf5bd,te,13,hu,34,il,18.79,mo,1
2019/12/6 16:30:0,name,ダイニングルーム,firmware_version,Remo/1.0.62-gabbf5bd,te,13,hu,34,il,8.6,mo,1

最小の更新間隔は分かりませんが、この通りilの値が変化していますので、5分間隔でもよいことが分かります。

温度が0.000001まで表示される

2019/12/5 20:45:0,name,ダイニングルーム,firmware_version,Remo/1.0.62-gabbf5bd,te,14.790001,hu,35,il,236.19,mo,1
2019/12/5 20:50:0,name,ダイニングルーム,firmware_version,Remo/1.0.62-gabbf5bd,te,14.790001,hu,35,il,236.19,mo,1
2019/12/5 20:55:0,name,ダイニングルーム,firmware_version,Remo/1.0.62-gabbf5bd,te,14.790001,hu,35,il,236.19,mo,1
2019/12/5 21:0:0,name,ダイニングルーム,firmware_version,Remo/1.0.62-gabbf5bd,te,14.790001,hu,35,il,236.19,mo,1
2019/12/5 21:5:0,name,ダイニングルーム,firmware_version,Remo/1.0.62-gabbf5bd,te,14.790001,hu,35,il,236.19,mo,1

この通り、温度のteが0.000001単位まで表示されることがあります。これは恐らく、本当の温度ではなく小数点を扱っているので表示上このようになっているものと思われます。四捨五入してもいいのですが別に困らないのでこのままにします。

おわり

Nature Remoで温度計を作ろう!

Nature Remo活用シリーズ

0.開封編 1.ダイキンのエアコン 2.ダイキンのエアコン、その2
3.アイリスオーヤマのサーキュレーター 4.温度計  

さて、めっきり寒くなってきました。空気も乾燥しています。そうなると気になるのが部屋の温度と湿度です。今、何度なのかなー、肌がカサつくけど湿度はどのくらいかなーと知りたくなりますが、生憎、私の部屋には温度計がありません。今更温度計を買うのもなんだなーと考えていたところ、ふと思い出しました。

そういえばNature Remoに温度計とかセンサーがついていたような・・・。

調べてみると、Nature Remoには温度、湿度、照度、人感センサーがついており、各センサーの値をAPI経由で取得できることがわかりました。

だったら定期的にAPIを叩いて温度と湿度をブラウザで表示できるようにしよう!

というわけで久しぶりにコードを書くことにしました。

まずはアクセストークンの取得

まずはじめにNature RemoのAPIを使えるようにするためのアクセストークンを発行します。アクセストークンを発行するためのURLはこちらです。

https://home.nature.global/

ログインをクリック

メールアドレスを入力

メールを送りましたとなるのでメールソフトを開きます。

メールに書いてあるリンクを開いて「許可する」をクリック

これでアクセストークンを発行できる状態になりました。すでにあるトークンは以前作ったIFTTT連携用ですね。

早速、アクセストークンを発行します。左下の「Generate access token」をクリックします。

新しいアクセストークンが発行されます。簡単ですね。ここで注意2点ほど。一つはアクセストークンの表示は一度きりなので必ずメモすること、2つめはアクセストークンは他人にバレないようにすることです。まあバレても部屋の温度と湿度が分かるのとエアコンとかが勝手に操作されるだけですけど・・・。

Nature RemoのAPIはcurlで叩いてJSONで取得

アクセストークンが取得できたので、プログラムを書く前に試しにAPIを叩いて情報の取得ができるか試してみましょう。Nature RemoのAPIはREST APIなのでcurlコマンドで簡単に試すことができます。

またNature RemoのAPIはマニュアルが公開されているので、必要な情報の取得や操作のためのURIはマニュアルから参照できます。

http://swagger.nature.global/

Nature Remoのセンサーの値を取得するためのコマンドは以下です。<<アクセストークン>>のところに先程取得したトークンを入れます。

curl -X GET "https://api.nature.global/1/devices" -H "accept: application/json" -H "Authorization: Bearer <<アクセストークン>>"

何やらズラズラっと出てきます。出力形式はJSONです。ちなみにこのコマンドはChromebookのLinux機能を使いました。Chromebook便利ですね!

話は逸れますが最近はREST + JSONが流行りですが、ぶっちゃけ私は苦手です。JSONって可読性は悪いし、データの抽出も一手間かかるし・・・。

まあ仕様ですので致し方ありません。jqコマンドを使って整形します。

curl -X GET "https://api.nature.global/1/devices" -H "accept: application/json" -H "Authorization: Bearer <<アクセストークン>>" | jq

可読性が良くなりました。”newest_events”以降がNature Remoの各種センターの値です。それぞれの意味はAPIマニュアルの「SensorValue」のdescriptionに以下のように書いてあります。


The reference key to SensorValue means “te” = temperature, “hu” = humidity, “il” = illumination, “mo” = movement

ということなので、

te 温度
hu 湿度
il 照度
mo 人感センサー

なのだと思います。照度の値がルクスなのかは分かりませんでした。

出力結果から私の部屋は現在、気温が18.2度で湿度が45%で明るさは10.39で人がいるということが分かります。

温度と湿度を抽出

今回の目的は温度計を作ることなので、必要な情報である温度と湿度をjqコマンドを使って更に絞り込みます。

温度だけを出す場合は以下のコマンドです。

curl -X GET "https://api.nature.global/1/devices" -H "accept: application/json" -H "Authorization: Bearer <<アクセストークン>>" | jq .[].newest_events.te.val

湿度だけを出すコマンドは以下

curl -X GET "https://api.nature.global/1/devices" -H "accept: application/json" -H "Authorization: Bearer <<アクセストークン>>" | jq .[].newest_events.hu.val

見事に目的の値を出すことに成功しました!!あとはこの値をブラウザで表示できるようHTMLを生成するだけです。

今回はここまで。次回はこれらの処理を自動で行うプログラムを作成して簡単な温度計Webアプリを作りたいと思います。

【安全第一主義】イスラエルに聖地を見に行こう! 〜 世界一厳しい?!入国編 〜

【目次】イスラエルに聖地を見に行こう!シリーズ

0. 計画編 1. 準備編 2. 事前勉強編 3. 出発編
4. 入国編      

さて、今回は入国編です。世界一厳しい?!と言われるイスラエルの入国審査からエルサレムまでのレポートです。

イスラエル上空

イスタンブール新空港を飛び立って約2時間。ついにイスラエル上空に入りました。テルアビブ北部上空を通過する模様です。

市街上空。高い建物は無さそうですが、人口は結構密集している模様です。

蛇足ですが飛行機の中でこんなエピソードが。

何の気なしに座席前に入っている雑誌をパラパラめくっていると隣に座っている恐らくイスラエル人のおじさんが「上にライトがあるから付けたらいいよ」と急に教えてくれました。何なんですかね、この外国人の他人でも急に話しかけて親切にしてくれる行為。日本人にはない習慣なのでちょっとびっくりします。一説には近くにいる他人が怪しい者ではないか確認しているとも言われますが、単に親切な人が多いなんじゃないかなとも思ったりします。「お〜、せんきゅー」とお礼を言ってライトを付けました。

ベン・グリオン国際空港に到着

さて景色を眺めたりしていたらあっという間にテルアビブ(ベン・グリオン国際空港)に到着しました。

タラップを降りて早速、人生初のヘブライ語の看板発見!記念にパシャリ。ちなみになんて書いてあるかは分かりません!(爆

特徴的な廊下。奥から下ってくる方は出国の人たちです。

「WELCOME!!」この奥が入国審査場になっています。右のヘブライ語はなんて書いてあるか分かりません。Welcomeかな?

入国審査

遂に世界で一番厳しいと言われる入国審査です。入国審査場はイスラエル人用とその他用で分かれています。それぞれ7列ぐらいあったと記憶しています。私はすごく緊張していてうまく応えられるかとか疑われて別室に連れて行かれたりしたらどうしようとか心配をしていました。が、別にやましいことは無いので正直に答えたら大丈夫かなとも考え順番を待ちます。ちなみに審査員が優しそうな女性の列に並びましたが、他の人のやり取りを見ていた限り関係無さそうです(笑

私の前の同じくアジア系の二人組が無事通過となり「Next!」と私の番になります。私の入国審査でのやり取りはこんな感じでした。

何しに来た? 観光に来ました。
何泊? 10泊です。
どこに滞在? エルサレムとテルアビブです。
どこのホテル? エルサレムはマミーラホテルでテルアビブはヒルトンテルアビブです。
バウチャー見せて (バウチャーを見せる)
OK Thank you

大体このくらいでした。質問は一つずつゆっくり聞かれ、答えるときにバッチリ目を見られます。ちなみにバウチャーは焦ってエルサレムのホテル分しか出さなかったのですが、特に何も言われませんでした。どちらかと言うと内容より受け答えの反応とか言い淀みが無いかとか嘘ついてないかとかそんな反応を見てる感じがしました。

というわけで、思ったよりあっさり入国審査を通過しました!

入国審査に合格するとこのようなIDカードが発行されます。このカードはホテルのチェックインなどの際、見せる必要があり滞在中は常に携帯する必要があります。ベリーインポータントカードです。(ホテルで言われた)

タクシーでエルサレムへ!

手荷物を受け取ってホテルで手配してもらったタクシーに乗り込みます。エルサレムへGo!

国道1号線を南下してエルサレムへ向かいます。飛行機からはよく見えませんでしたが、砂漠地帯が広がっています。

エルサレムに到着!

遂に憧れの地、エルサレムに到着しました!いやー感無量です。

今回はここまで。次回はエルサレム観光の拠点となるマミーラホテルのレビューです。