3
Help us understand the problem. What are the problem?

More than 1 year has passed since last update.

posted at

updated at

毎日の健康調査をする仕組みを BOT で作ってみたよ

社員やメンバーの毎日の健康調査を記録できる仕組みを BOT で開発してみました!ヾ(´∀`)ノ

経緯としてはね、こんなにもコロコロちゃんが流行したのでね。

「今朝の体温は何度だ?」「咳とか鼻水は出てないか?」( ゚Д゚)

と、リーダーのみなさんがメンバーに聞いているわけですよ。
んで、それを Excel にカチカチ記録して、一覧表にして人事部とかに「うちのチームはみな健康です!」って出したりするわけですよ。

た い へ ん !( ゚Д゚)

ってなわけで、BOT に任せちゃおうよ!って思ったわけです。

どんな仕組み?

  • LINEWORKS BOT に話しかけると健康調査が始まる
  • 答えた内容は GoogleSpreadSheet に自動で記録されていく

簡単!(*‘∀‘)

利用する技術

  • Google Apps Script (GAS)
  • Google SpreadSheet
  • LINEWORKS API

全部、無料の範囲内でできるよ!ヾ(´∀`)ノ 最高かよ

GAS や LINEWORKS BOT の登録方法がわからないよ!って方は、下記記事で紹介していますのでよろしければ読んでくださいませませ。
GoogleAppsScript で LINEWORKS のチャット BOT を作る

実際の画面はこんな感じ

ss1

ss2

ss3

ss4

ss5

ss6

ss7

ss8

私、美術センスが壊滅的なので、デザイン的なものには突っ込まないでね!( ノД`)シクシク…

アカウントのところはモザイクかけちゃっていますが、もちろん誰の記録かわかるようになっています!
なので、この SpreadSheet を見ればみなの健康状態が一目瞭然なのですヨ!(*‘∀‘)

コードの公開

それでは、コードをズラズラっと。

app.gs
function setOptions(){
  return {
    "apiId" : "xxxxxxxx",
    "consumerKey" : "xxxxxxxxxxxxxxxx",
    "serverId" : "xxxxxxxxxxxxxxxxxxxxxxxxxxxxx",
    "privateKey" : "-----BEGIN PRIVATE KEY-----\nxxxxxxxxxxxxxxxxxxxxxxxxx\n-----END PRIVATE KEY-----",
    "botNo" : 000000000
  };
}

function doPost(e) {
  if (e == null || e.postData == null || e.postData.contents == null) return;
  const requestJSON = e.postData.contents;
  const requestObj = JSON.parse(requestJSON);
  if (requestObj.content.text == "利用開始") return;
  const accountId = requestObj.source.accountId;

  // postback 対応
  const postback = requestObj.content.postback;
  var templateNo = 1;

  // postback message 受信時に次の temlateNo の取得および spreadsheet の操作
  if(postback) { 
    const text = requestObj.content.text;
    // spreadsheet 取得
    const ss = SpreadsheetApp.getActive();
    var sheet = ss.getSheetByName(accountId);
    switch(postback){
      case "start" :
        if(text == "はい"){
          templateNo = 2;
          if(sheet == null) sheet = ss.insertSheet(accountId);
          var date = new Date();
          sheet.getRange("A1").setValue([Utilities.formatDate( date, 'Asia/Tokyo', 'yyyy/MM/dd')]);
        } else { templateNo = null }
        break;
      case "bodyTemperature_1" :
        templateNo = 3;
        sheet.getRange("B1").setValue(text);
        break;
      case "bodyTemperature_2" :
        templateNo = 4;
        sheet.getRange("C1").setValue(text);
        break;
      case "bodyTemperature_3" :
        (text == "はい") ? templateNo = 5 : templateNo = 2;
        break;
      case "condition" :
        templateNo = 6;
        sheet.getRange("D1").setValue(text);
        break;
      case "hospital" :
        templateNo = 7;
        sheet.getRange("E1").setValue(text);
        break;
      case "medicine" :
        templateNo = 8
        sheet.getRange("F1").setValue(text);
        break;
      case "recode" :
        if(text == "はい"){
          templateNo = null;
          // キャッシュを DB に統合
          const values = sheet.getRange("A1:F1").getValues();
          const data = [values[0][0], accountId, values[0][1] + values[0][2]/10, values[0][3], values[0][4], values[0][5]];
          ss.deleteSheet(sheet);
          sheet = ss.getSheetByName("シート1");
          sheet.appendRow(data);
        } else { templateNo = 2 };
        break;
      default : break;
    }
  }

  // LINEWORKS にメッセージを送信
  if(templateNo){
    sendButtonMsg(setOptions(), accountId, setButtonTemplate(templateNo, accountId));
  } else {
    LINEWORKS.sendMsg(setOptions(), accountId, "健康調査を終了します。\nお疲れ様でした。");
  }
  return
}
function setUp() {
  const obj = setOptions();
  const uri = "https://apis.worksmobile.com/r/" + obj.apiId + "/message/v1/bot/" + obj.botNo + "/persistentmenu";
  const json = {
    "content": {
        "actions": [{
            "type": "message",
            "label": "健康調査を開始する"
        }]
    }
  };
  const options = {
    "method": "post",
    "headers": LINEWORKS.setHeaders(obj),
    "payload": JSON.stringify(json)
  };
  UrlFetchApp.fetch(uri, options); 
}

function sendButtonMsg(obj, accountId, content){
  const uri = "https://apis.worksmobile.com/r/" + obj.apiId + "/message/v1/bot/" + obj.botNo + "/message/push";
  const json = {
    "accountId": accountId,
    "content": content
  };
  const options = {
    "method": "post",
    "headers": LINEWORKS.setHeaders(obj),
    "payload": JSON.stringify(json)
  };
  UrlFetchApp.fetch(uri, options);  
}

function setButtonTemplate(templateNo, accountId){
  var contentText;
  var actions = [];
  const yn = ["はい","いいえ"];
  var ss = SpreadsheetApp.getActive();
  var sheet = ss.getSheetByName(accountId);

  switch(templateNo){
    case 1 :
      contentText = "健康調査を開始しますか?";
      for(var i = 0; i < yn.length; i++){
        actions.push({"type": "message", "label": yn[i], "postback": "start"});
      }
      break;
    case 2 :
      contentText = "今日の体温を教えてください。\n何度代でしたか?";
      for(var i = 35; i <= 41; i++){
        actions.push({"type": "message", "label": String(i), "postback": "bodyTemperature_1"});
      }
      break;
    case 3 :
      contentText = "続けて細かい体温を教えてください。\n何分でしたか?";
      for(var i = 0; i <= 9; i++){
        actions.push({"type": "message", "label": String(i), "postback": "bodyTemperature_2"});
      }
      break;
    case 4 :
      const bodyTemperature = sheet.getRange("B1").getValue() + sheet.getRange("C1").getValue()/10;
      contentText = "今日の体温は " + bodyTemperature + "℃ですね?";
      for(var i = 0; i < yn.length; i++){
        actions.push({"type": "message", "label": yn[i], "postback": "bodyTemperature_3"});
      }
      break;
    case 5 :
      contentText = "以下のような自覚症状はありますか?";
      const condition = ["","のどの痛み","くしゃみ・鼻水","倦怠感(だるさ)","息苦しさ","特になし"];
      for(var i = 0; i < condition.length; i++){
        actions.push({"type": "message", "label": condition[i], "postback": "condition"});
      }
      break;
    case 6 :
      contentText = "医療機関を受診しましたか?";
      for(var i = 0; i < yn.length; i++){
        actions.push({"type": "message", "label": yn[i], "postback": "hospital"});
      }
      break;
    case 7 :
      contentText = "薬を服用しましたか?";
      for(var i = 0; i < yn.length; i++){
        actions.push({"type": "message", "label": yn[i], "postback": "medicine"});
      }
      break;
    case 8 :
      const temperature = sheet.getRange("B1").getValue() + sheet.getRange("C1").getValue()/10;
      const symptoms = sheet.getRange("D1").getValue();
      const hospital = sheet.getRange("E1").getValue();
      const medicine = sheet.getRange("F1").getValue();
      contentText = "(本日の健康状態)\n体温:" + temperature + "\n自覚症状:" + symptoms + "\n医療機関の受診:" + hospital + "\n薬の服用:" + medicine + "\n\n 以上でよろしいですか?";
      for(var i = 0; i < yn.length; i++){
        actions.push({"type": "message", "label": yn[i], "postback": "recode"});
      }
      break;
    default : break;
  }
  return {
    "type": "button_template",
    "contentText": contentText,
    "actions": actions
  };
}

もっと綺麗に書けるのだろうけど、取り敢えず動くことを優先させました。
あと、エラー回避してません。運用しながらちょびちょび修正かけてます。

LINEWORKS ライブラリの利用

認証周りとか自前でやると面倒なので、LINEWORKS ライブラリを利用しています。
LINEWORKS ライブラリの登録方法は こちら を参考にしてください。

苦労したところ

アカウントごとに記録する際に、セッション管理に悩みました。
結局、アカウントごとに一時的に Sheet を作成して、最後の大元の Sheet に転記して削除するようにしました。

今後、拡張したい機能

エラー処理はもちろん入れないとですよねー。今のままだと不安定ですし。
セッションが残っている場合に備えて定期的に削除したり。

37.5℃以上のメンバーがいたら管理者に通知するとか。
Weekly レポートを作成して送信するとか。

色々と要望を聞きつつ、増やしていきたいですね!

おわりに

ここまでお付き合いいただきありがとうございました。

久々にサービスっぽいもの作った!楽しい!(*‘∀‘)
何かを作るのは本当に楽しいですね~♪

これ、みんなが利用できるように、ライブラリ作って初期設定を簡素化したら、使ってくれるかな?
コロコロブームが去ってもリモートワークを続けるところも多そうですし、メンバーの健康把握も課題の一つになってくるのでは・・・?

ちょっと頑張ってみようかな('ω')

ではまた!(^^)/

参考にさせていただきましたm(_ _)m

GoogleAppsScript 公式サイト
LINEWORKS Developers

Why not register and get more from Qiita?
  1. We will deliver articles that match you
    By following users and tags, you can catch up information on technical fields that you are interested in as a whole
  2. you can read useful information later efficiently
    By "stocking" the articles you like, you can search right away
Sign upLogin
3
Help us understand the problem. What are the problem?