温度センサーDS18B20の測定値をブラウザで表示
温度センサーで測定したデータを、スマホやパソコンのブラウザで表示するためWebIOPiをインストールしました。WebIOPiをRaspberry Piにインストールして実行するとwebサーバが立ち上がり、ブラウザからGPIOの操作や、Raspberry Pi内のPythonプログラムを実行できるようになります。
WebIOPiのインストール
http://webiopi.trouch.com/ にアクセスし、ヘッダーメニューのDownloadをクリック
Original WebIOPi-0.7.1.tar.gz (full sources download)をクリックしてダウンロードします。ダウンロードしたファイルは、ユーザーpiのホームにある「Downloads」または「ダウンロード」フォルダにあります。WebIOPi-0.7.1.tar.gzを確認したら、ユーザーpiのホームに移動します。
次に、ターミナルを起動して次のコマンドを順番に入力します。
- tar zxf WebIOPi-0.7.1.tar.gz
- cd WebIOPi-0.1.7/
- wget https://raw.githubusercontent.com/doublebind/raspi/master/webiopi-pi2bplus.patch
- patch -p1 -i webiopi-pi2bplus.patch
- sudo ./setup.sh
完了すると、Do you want to access WebIOPi over Internet?[y/n] と聞いてきますので、「n」をタイプします。(家の中だけで使う設定です。家の外ではこのままでは使えません。)
次に、2つのコマンドを入力してください。
- wget https://raw.githubusercontent.com/neuralassembly/raspri/master/webiopi.service
- sudo mv webiopi.service /etc/systemd/system/
これで、インストールは終了です。
WebIOPiの起動と停止
ターミナルから次のコマンドを入力すると起動と停止ができます。
- sudo service webiopi start (起動)
- sudo service webiopi stop (停止)
起動しているか停止しているかを調べるのは、
- ps ax | grep webiopi
を実行してください。起動している場合は、複数行が、起動していない場合は、1行のみ表示されます。(数字は時と場合により異なります。)
WebIOPiの自動起動
WebIOPiを自動起動するときは、
- sudo systemctl enable webiopi
自動起動が不要になったときは、
- sudo systemctl disable webiopi
とターミナルから入力します。
設定ファイルの変更
ターミナルで、sudo leafpad /etc/webiopi/config と入力し、[SCRIPTS]セクションのmyscriptを次のように書き換えます。コメントアウト # も外してください。1行追加する様になると思います。(下の赤線部分)
黄色センのpasswd-fileをコメントアウトすると、ウェブアクセスしたときにパスワードは聞かれなくなります。
ファイルの配置
/usr/share/webiopi/htdocs/bme280 の中に「index.html」「javascript.js」「script.py」「styles.css」の各ファイルを保存します。jsフォルダには、「require.js」が保存してあります。
index.html
<!DOCTYPE html> <html> <head> <meta charset="utf-8"> <meta name="viewport" content="width=device-width, user-scalable=no"> <title>温度センサ(DS20B18)による温度の取得</title> <img src="" data-wp-preserve="%3Cscript%20src%3D%22js%2Frequire.js%22%3E%3C%2Fscript%3E" data-mce-resize="false" data-mce-placeholder="1" class="mce-object" width="20" height="20" alt="<script>" title="<script>" /> <img src="" data-wp-preserve="%3Cscript%3E%0A%09%09%09require(%5B%22%2Fwebiopi.js%22%2C%20%22javascript.js%22%5D%2C%20function()%7B%0A%20%20%20%20%09%09%09webiopi().ready(%20initialize_webiopi%20)%3B%0A%09%09%09%7D)%3B%0A%09%09%3C%2Fscript%3E" data-mce-resize="false" data-mce-placeholder="1" class="mce-object" width="20" height="20" alt="<script>" title="<script>" /> </head> <body> <div align="center"> <h4>温度センサ(DS20B18)により取得した温度を表示しています</h4> 温度センサー0: <input type="text" id="temp_text_0">℃<br><br> 温度センサー1: <input type="text" id="temp_text_1">℃ </div> </body> </html>
styles.css
#temp_text_0{ width: 4em; /*height:3em;*/ font-size: 2em; } #temp_text_1{ width: 4em; /*height:3em;*/ font-size: 2em; } body { background-color: #fffac2 }
javascript.js
// millisミリ秒ごとに温度を取得する関数 function getTempPeriodic(millis){ // responseに温度が格納されている var drawTemp = function(macro, args, response){ var temp = response.split(','); // response は文字型 "*.???,*.???" ','で文字列を分割 $("#temp_text_0").val(temp[0]); // temp_text_0 にtemp[0]を代入し表示 $("#temp_text_1").val(temp[1]); // temp_text_1 にtemp[1]を代入し表示 } // script.py内のgetTemp関数を実行。終了後にdrawTemp関数を呼ぶ webiopi().callMacro("getTemp", [] , drawTemp); // millisミリ秒後に自分自身を呼び出す setTimeout(function(){ getTempPeriodic(millis); }, millis); } function initialize_webiopi(){ // webiopiの準備が終わってからstyles.cssを適用する applyCustomCss('styles.css'); // 2秒ごとに温度を取得して描画 getTempPeriodic(2000); // GPIOの状態を監視しない(自分で作成した関数で温度を定期的に取得するため) webiopi().refreshGPIO(false); } function applyCustomCss(custom_css){ var head = document.getElementsByTagName('head')[0]; var style = document.createElement('link'); style.rel = "stylesheet"; style.type = 'text/css'; style.href = custom_css; head.appendChild(style); }
script.py
# -*- coding: utf-8 -*- import webiopi import os import glob import time import subprocess os.system('modprobe w1-gpio') os.system('modprobe w1-therm') webiopi.setDebug() #temp_c=0.00 base_dir = '/sys/bus/w1/devices/' #DS18B20のデータが保存されるディレクトリー #device_folder = glob.glob(base_dir + '28*')[0] device_folder = glob.glob(base_dir + '28*') #センサーごとのディレクトリーをリスト形式で取得 センサーが複数ある可能性があるため ch_length = len(device_folder) #センサーの個数を取得 temp= [0.00]*ch_length #センサーの個数分のリスト形式のtempを作成 #device_file = device_folder + '/w1_slave' #-----/sys/bus/w1/devices/28*/w1_slave の中身--------------------- # b1 01 4b 46 7f ff 0f 10 8d : crc=8d YES # b1 01 4b 46 7f ff 0f 10 8d t=27062 #----------------------------------------------------------------- def read_temp_raw(ch): #センサーごとに作成されるファイルの読み取り chは0,1,2,3,... device_file = device_folder[ch] + '/w1_slave' #センサーごとに作成されるファイル名の取得 catdata = subprocess.Popen(['cat',device_file], stdout=subprocess.PIPE, stderr=subprocess.PIPE) #Linuxのコマンド cat を使ってファイルの中身を取得 サブプロセス out,err = catdata.communicate() #サブプロセスの終了を待って標準出力と標準エラー出力を取得 out_decode = out.decode('utf-8') #バイト列を文字列に変換 lines = out_decode.split('\n') #改行で区切り、lines リスト型に代入 return lines #戻り値 リスト型 def read_temp(ch): #linesから温度を読み取る関数 chは0,1,2,3,... lines = read_temp_raw(ch) #linesの取得 while lines[0].strip()[-3:] != 'YES': #linesの最初のデータの空白を取り除き最後の3文字が'YES'でないとき time.sleep(0.2) #0.2秒待つ lines = read_temp_raw(ch) #再度linesを取得 equals_pos = lines[1].find('t=') #2行目の't='の位置を取得 if equals_pos != -1: #'t='の位置が-1でなければ 見つかった時 temp_string = lines[1][equals_pos+2:] #'t='の位置の2文字目以降の文字列を取得 temp_c = float(temp_string) / 1000.0 #浮動小数点型に変換して、1000で除算 摂氏温度に変換 #temp_f = temp_c * 9.0 / 5.0 + 32.0 #return temp_c, temp_f return temp_c #戻り値 摂氏温度 def setup(): webiopi.debug("Script with macros - Setup") #デバッグ出力を利用 def loop(): # WebIOPiが起動している間、繰り返し呼ばれる関数 #global temp_c #temp_c=read_temp(0) webiopi.sleep(1) def destroy(): # WebIOPi終了時に呼ばれる関数 webiopi.debug("Script with macros - Destroy") @webiopi.macro #Javascriptから呼ばれる関数 def getTemp(): temp_c = "{0:.3f},{1:.3f}".format(read_temp(0),+read_temp(1)) #0チャンネンルと1チャンネルを小数点以下3桁に整えて','で区切る 文字列としてtemp_cに代入 return temp_c #戻り値
最後にターミナルから、ifconfig でラズパイのIPアドレスを調べ、ブラウザから、
と入力してください。IPアドレスはifconfigで調べた数値です。ポートもデフォルトが8000です。
スマホのスクリーンショットです。