13
8

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 5 years have passed since last update.

食べログのランキングからランチをランダムに提案してくれるSlackボット

Last updated at Posted at 2018-01-22

はじめに

本稿は下記の記事からパクリ着想を得て記載しています。

GASのスクレイピングとSlackボット作成をしたかったので、作ってみました。

できること

食べログのランキングからランダムにN個を取り出してSlackに投稿する。

完成したコード

Code.gs
//required library
//SlackApp	M3W5Ut3Q39AaIwLquryEPMwV62A3znfOO
//Parser	M1lugvAXKKtUxn_vdAG9JZleS6DrsjUUV

//Fisher–Yates shuffle(配列をシャッフルする関数)
function shuffle(array) {
  var n = array.length, t, i;

  while (n) {
    i = Math.floor(Math.random() * n--);
    t = array[n];
    array[n] = array[i];
    array[i] = t;
  }

  return array;
}

//食べログをスクレイピングしてSlackに投稿する。
function postMessage() {
  
  const prop = PropertiesService.getScriptProperties().getProperties();
  const slackApp = SlackApp.create(prop.token); //Script Propertiesに設定したトークンIDを取得
  const slackChannelId = prop.SlackChannel; //Script Propertiesに設定したチャンネルIDを取得
  const message = [ 
         '今日のおすすめランチです。',
         'ランチはここでどうでしょう?',
         'ランチはここで食べてください。'
  ];
  //食べログのURL 下記形式のURLであればそのまま使えそう。https://tabelog.com/*/rstLst/*
  //サンプルは東京駅のランチランキング
  const fetchUrl = 'https://tabelog.com/tokyo/A1302/A130201/R6586/rstLst/lunch/';

  const html = UrlFetchApp.fetch(fetchUrl).getContentText();
  //ランキングのリスト部分をパースする。
  const doc = Parser.data(html)
                    .from('<ul class="js-rstlist-info rstlist-info"')
                    .to('<li class="list-rst__rdimg">')
                    .build()
  //名前、価格etcを配列で格納。
  //現状は名前しか使っていない。できれば結合して投稿したい。
  const name = Parser.data(doc)
                    .from('<div class="list-rst__rst-name">')
                    .to('<span class="list-rst__area-genre cpy-area-genre">')
                    .iterate()
                                 
  const price = Parser.data(doc)
                    .from('<span class="c-rating__val list-rst__budget-val cpy-lunch-budget-val">')
                    .to('</span>')
                    .iterate()
  
  const rating = Parser.data(doc)
                    .from('<span class="c-rating__val c-rating__val--strong list-rst__rating-val">')
                    .to('</span>')
                    .iterate()

  const url = Parser.data(doc)
                    .from('<a class="list-rst__rst-name-target cpy-rst-name" target="_blank" href="')
                    .to('">')
                    .iterate()

  //メッセージとランキングをシャッフル
  var randomMsg = shuffle(message);
  var randomName = shuffle(name);
  
  //3つを取り出してSlackに投稿
  slackApp.postMessage(slackChannelId,randomMsg[0], {username : "ランチロシアンルーレット"});  
  for (var j=0; j<3 ;j++){
    slackApp.postMessage(slackChannelId,j+1+'. '+randomName[j].replace(/<("[^"]*"|'[^']*'|[^'">])*>/g,''), {username : "ランチロシアンルーレット"});
  }
}

コードの説明

Code.gs
//required library
//SlackApp	M3W5Ut3Q39AaIwLquryEPMwV62A3znfOO
//Parser	M1lugvAXKKtUxn_vdAG9JZleS6DrsjUUV

事前準備として、必要なライブラリをインポートします。

Code.gs
//Fisher–Yates shuffle(配列をシャッフルする関数)
function shuffle(array) {
  var n = array.length, t, i;

  while (n) {
    i = Math.floor(Math.random() * n--);
    t = array[n];
    array[n] = array[i];
    array[i] = t;
  }

  return array;
}

お店をどうやってランダムに取り出すか迷いましたが、配列をシャッフルして先頭から取り出すことにしました。
このため、シャッフルするための関数を用意しておきます。

Code.gs
//食べログをスクレイピングしてSlackに投稿する。
function postMessage() {
  
  const prop = PropertiesService.getScriptProperties().getProperties();
  const slackApp = SlackApp.create(prop.token); //Script Propertiesに設定したトークンIDを取得
  const slackChannelId = prop.SlackChannel; //Script Propertiesに設定したチャンネルIDを取得
  const message = [ 
         '今日のおすすめランチです。',
         'ランチはここでどうでしょう?',
         'ランチはここで食べてください。'
  ];

Slackの投稿に必要なトークンとチャンネルのIDは、コード内に含めずに、スクリプトのプロパティに定義しておき、PropertiesService.getScriptProperties().getProperties()で取り出します。

Code.gs
  //食べログのURL 右記形式のURLであればそのまま使えそう。 https://tabelog.com/*/rstLst/*
  //サンプルは東京駅のランチランキング
  const fetchUrl = 'https://tabelog.com/tokyo/A1302/A130201/R6586/rstLst/lunch/';

食べログのURLを指定します。
https://tabelog.com/*/rstLst/*」 という形式のURLであれば、今回のコードがそのまま使えるようです。

Code.gs

  const html = UrlFetchApp.fetch(fetchUrl).getContentText();
  //ランキングのリスト部分をパースする。
  const doc = Parser.data(html)
                    .from('<ul class="js-rstlist-info rstlist-info"')
                    .to('<li class="list-rst__rdimg">')
                    .build()
  //名前、価格etcを配列で格納。
  //現状は名前しか使っていない。できれば結合して投稿したい。
  const name = Parser.data(doc)
                    .from('<div class="list-rst__rst-name">')
                    .to('<span class="list-rst__area-genre cpy-area-genre">')
                    .iterate()
                                 
  const price = Parser.data(doc)
                    .from('<span class="c-rating__val list-rst__budget-val cpy-lunch-budget-val">')
                    .to('</span>')
                    .iterate()
  
  const rating = Parser.data(doc)
                    .from('<span class="c-rating__val c-rating__val--strong list-rst__rating-val">')
                    .to('</span>')
                    .iterate()

  const url = Parser.data(doc)
                    .from('<a class="list-rst__rst-name-target cpy-rst-name" target="_blank" href="')
                    .to('">')
                    .iterate()

ライブラリParserを利用して、ページの情報を取得します。
必要な情報が含まれる箇所のHTMLタグをfrom/toで指定して、build()で文字列として取得します。名称や星の数はliterate()で配列として格納します。

なお、このコードでは、名前しか使用していません。
本当は星の数やURLも一緒に投稿したかったのですが、うまい紐づけかたが思い浮かびませんでした。(コメント貰えると嬉しいです)

Code.gs
  //メッセージとランキングをシャッフル
  var randomMsg = shuffle(message);
  var randomName = shuffle(name);

いつも同じメッセージだとつまらないので、冒頭で定義したshuffleファンクションを利用して、シャッフルしたメッセージを取り出します。
また、お店の名前はこのままだとランキングの順位で並んでいるだけなので、こちらもシャッフルします。

Code.gs
  //3つを取り出してSlackに投稿
  slackApp.postMessage(slackChannelId,randomMsg[0], {username : "ランチロシアンルーレット"});  
  for (var j=0; j<3 ;j++){
    slackApp.postMessage(slackChannelId,j+1+'. '+randomName[j].replace(/<("[^"]*"|'[^']*'|[^'">])*>/g,''), {username : "ランチロシアンルーレット"});
  }
}

ライブラリSlackAppを利用してSlackに投稿します。
先ほどシャッフルした配列randomNameから順番に3つを取り出します。

配列name(とそれをシャッフルしたrandomName)には<a>タグが含まれてしまうので、replace(/<("[^"]*"|'[^']*'|[^'">])*>/g,'')でタグを削除します。

JavaScriptでHTMLタグを削除する正規表現 - Qiita

できあがり

slackbot.png

あとがき

本文中にも書きましたが、できれば店の名前とURLや星の数を紐づけて投稿したいなと思っています。

参考リンク

13
8
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
13
8

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?