こちらの記事は Google Cloud Platform Advent Calendar 2017 の 4日目への投稿になります。
みなさん、毎日GCP使ってますか?
わたしは最近あまりさわれていません。。こんにちは。chidakiyoです。
先日 GCPUG にて、 GCPでNoAdmした話 という内容で発表させていただきました。
資料の中で詳細はQiitaで書きます、と言っていたのでその記事になります。
前提
エンジニアという仕事をしていると、毎日何かしらのデータの確認など定点観測を行っていると思います。
毎日決まったSQLを投げてコツコツ確認するということを頑張れる人もいると思いますが、私は技術は自分がラクをするために使いたい派なので、毎日朝起きたら見たい情報がきれいなグラフになってSlack投稿されている状態が嬉しいです。
今回は、サービス運営にあたって、BigQueryへデータが蓄積されているのでそのデータを元にSlackにグラフを自動的に投稿するという仕組みについて書きます。
汎用性
要点はGoogle SpreadSheetにグラフが作成できればそれを定期的にSlackに投げれるというものなので、
- 日々追いたいデータがある非エンジニア(株価とか、競合他社の価格調査など)
- 口座の残高の推移(データを取れる方法があればできるかも)
そういうものが欲しい人もチャレンジしてみてもいいかもしれません。
データソースはBigQueryに限らずに利用できると思います。
構成
構成は以下のようになります。
おおまかな流れとしては
- AppsScript(1)がSQLを発行してデータを取得してくる
- Google SpreadSheetが取得したデータを用いてグラフを作成する (この部分は特にプログラミングは必要なく、普段使っているように表を元にグラフを作っておけば新しいデータを取得した際にグラフが更新される)
- AppsScript(2)がSlackへグラフを画像として送る。
![スクリーンショット 2017-12-04 9.11.16.png](https://qiita-user-contents.imgix.net/https%3A%2F%2Fqiita-image-store.s3.amazonaws.com%2F0%2F28681%2Ff3865f85-0321-eb40-04fd-8387c9ae498e.png?ixlib=rb-4.0.0&auto=format&gif-q=60&q=75&s=9be36c5d9e72c1d69dc6cf47a4982155)
手順1 (データ取得側を作る)
Google SpreadSheetの新規ファイルを作成する
既存のファイルを利用してもOKです。
![スクリーンショット 2017-12-04 9.16.30.png](https://qiita-user-contents.imgix.net/https%3A%2F%2Fqiita-image-store.s3.amazonaws.com%2F0%2F28681%2Fa2143ebd-e097-17c6-8905-628344f693b9.png?ixlib=rb-4.0.0&auto=format&gif-q=60&q=75&s=976c8678aeb126507edc92cb72982b5d)
AppScriptを作成する
メニューの 「ツール」 → 「スクリプトエディタ」 からAppScriptを開きます。
![スクリーンショット 2017-12-04 9.17.17.png](https://qiita-user-contents.imgix.net/https%3A%2F%2Fqiita-image-store.s3.amazonaws.com%2F0%2F28681%2F1b7dbdad-6bca-c38c-ed28-b2f27d135ef6.png?ixlib=rb-4.0.0&auto=format&gif-q=60&q=75&s=a54216fbbc191b011005d22d746dd6b2)
スクリプト(1)を貼ります
こちらのスクリプトは、定期的に実行し、BigQueryから取得したデータをSpreadSheetに入力するという役割になります。
一番最初のprojectIdはBigQueryのあるプロジェクトのIDを入力します。
/* 実行するProject ID */
var projectId = 'gas-bq-sample';
/* シートの全クエリを実行 */
function runAllQueries() {
var spreadsheet = SpreadsheetApp.getActiveSpreadsheet();
var queries_sheet = spreadsheet.getSheetByName("Queries");
for (var i = 2; i <= queries_sheet.getLastRow(); i++) {
var sheet_name = queries_sheet.getRange(i, 1).getValue();
var sheet = spreadsheet.getSheetByName(sheet_name);
if(sheet == null) {
sheet = spreadsheet.insertSheet();
sheet.setName(sheet_name);
}
runQuery(
queries_sheet.getRange(i, 2).getValue(),
sheet,
queries_sheet.getRange(i, 3).getValue().toString(),
queries_sheet.getRange(i, 4).getValue().toString()
);
}
}
/* クエリを実行 */
function runQuery(query, sheet, title, query_type) {
var request = {
query: query
};
if (query_type == 'legacy') {
request.useLegacySql = true
} else {
request.useLegacySql = false
}
var queryResults = BigQuery.Jobs.query(request, projectId);
var jobId = queryResults.jobReference.jobId;
// Check on status of the Query Job.
var sleepTimeMs = 500;
while (!queryResults.jobComplete) {
Utilities.sleep(sleepTimeMs);
sleepTimeMs *= 2;
queryResults = BigQuery.Jobs.getQueryResults(projectId, jobId);
}
// Get all the rows of results.
var rows = queryResults.rows;
while (queryResults.pageToken) {
queryResults = BigQuery.Jobs.getQueryResults(projectId, jobId, {
pageToken: queryResults.pageToken
});
rows = rows.concat(queryResults.rows);
}
if (rows) {
// clear exists data and set title
sheet.clear();
sheet.appendRow([title]);
// Append the headers.
var headers = queryResults.schema.fields.map(function(field) {
return field.name;
});
sheet.appendRow(headers);
// Append the results.
var data = new Array(rows.length);
for (var i = 0; i < rows.length; i++) {
var cols = rows[i].f;
data[i] = new Array(cols.length);
for (var j = 0; j < cols.length; j++) {
data[i][j] = cols[j].v;
if(headers[j].substr(-3)=="_at") {
data[i][j] = cols[j].v
if(data[i][j] < 10000000000000) {
data[i][j] = new Date(data[i][j]);
}
else if(data[i][j] < 10000000000000*1000) {
data[i][j] = new Date(data[i][j] / 1000);
}
}
}
}
sheet.getRange(3, 1, rows.length, headers.length).setValues(data);
} else {
Logger.log('No rows returned.');
}
};
function onOpen() {
var spreadsheet = SpreadsheetApp.getActiveSpreadsheet();
var entries = [{
name : "このシートの内容を更新",
functionName : "runCurrentSheetQuery"
},
{
name : "全シートの内容を更新",
functionName : "runAllQueries"
}];
spreadsheet.addMenu("BigQuery", entries);
};
データを出力するシートを作成する
以下のシートを作成します。
-
Queries
シート。- 実行するクエリを定義するシートです。1カラム目が結果を出力するシート名、2カラム目がSQLクエリ、4カラム目がstandard/legacyの切り替えのフラグです。(3カラム目を利用しないので注意)
クエリを作成する
BigQueryで実行するクエリを2行目以降に書いていきます。
以下の例は、GCPの課金データをexportしたテーブルに対して実行している例です。
GCPの課金Exportのv1のスキーマは自分でもまだあまり使っていないのであまり良くわかってませんが例としてざっくりしたSQLを実行しています(こちらどのようなSQLが良いかご存じの方はご指摘ください)
StandardSQLで実行するのでD列のカラムは何も書きません。(LegacySQLの場合にはLegacyと入力してください)
![スクリーンショット_2017-12-04_11_34_10.png](https://qiita-user-contents.imgix.net/https%3A%2F%2Fqiita-image-store.s3.amazonaws.com%2F0%2F28681%2F480df1be-da7a-19fb-f197-c19962b43ac3.png?ixlib=rb-4.0.0&auto=format&gif-q=60&q=75&s=a0aa4a1a3cb0fa2726e172d347a566b9)
スクリプト側でBigQueryAPI利用を有効にする
メニューの 「リソース」 → 「Googleの拡張サービス」 を選択し、
![スクリーンショット 2017-12-04 11.25.45.png](https://qiita-user-contents.imgix.net/https%3A%2F%2Fqiita-image-store.s3.amazonaws.com%2F0%2F28681%2F4961a0e9-2bee-06e1-3dc8-c99730e47277.png?ixlib=rb-4.0.0&auto=format&gif-q=60&q=75&s=978697b88e3b7ad4cc5d03e4afd0b26a)
GoogleAPIコンソールでも有効にする必要があると書いてあるので、
そのリンクを押し、遷移した先で 「APIとサービスの有効化」 をクリックする
![スクリーンショット 2017-12-04 11.27.34.png](https://qiita-user-contents.imgix.net/https%3A%2F%2Fqiita-image-store.s3.amazonaws.com%2F0%2F28681%2F92cab403-8b43-e983-8510-763e4d543b95.png?ixlib=rb-4.0.0&auto=format&gif-q=60&q=75&s=b0200a6998bac02b1cdede12a65efb46)
bigquery
と検索し、BigQueryAPIを選択する
![スクリーンショット 2017-12-04 11.28.10.png](https://qiita-user-contents.imgix.net/https%3A%2F%2Fqiita-image-store.s3.amazonaws.com%2F0%2F28681%2Ffc819675-0e5d-43a8-5bc3-0f7826d031ab.png?ixlib=rb-4.0.0&auto=format&gif-q=60&q=75&s=b74d11a44b1670f42d91a87d1c9eb867)
BigQueryAPIを有効にする
![スクリーンショット 2017-12-04 11.28.22.png](https://qiita-user-contents.imgix.net/https%3A%2F%2Fqiita-image-store.s3.amazonaws.com%2F0%2F28681%2Fe07537f6-e187-d292-bd47-281eec02f56f.png?ixlib=rb-4.0.0&auto=format&gif-q=60&q=75&s=bb9e8832133251d17a287f051599fed4)
ここまでやればひとまずBQ接続周りの設定はOK。
スクリプトを一旦手動実行する
初回実行時にAPI接続の権限などを確認する処理が必要になるので、初回の1回は手動で実行します。
![スクリーンショット_2017-12-04_11_23_00.png](https://qiita-user-contents.imgix.net/https%3A%2F%2Fqiita-image-store.s3.amazonaws.com%2F0%2F28681%2F589da151-31a5-26d8-6e70-d751c3ad7a99.png?ixlib=rb-4.0.0&auto=format&gif-q=60&q=75&s=968b16e9e22b98ea69682cf269f1975d)
初回実行すると承認が必要です。というダイアログが表示されるので「許可を確認」をクリックし、認証情報をポチポチする。
![スクリーンショット 2017-12-04 11.26.58.png](https://qiita-user-contents.imgix.net/https%3A%2F%2Fqiita-image-store.s3.amazonaws.com%2F0%2F28681%2Fd4b7c00f-7f8b-7332-f7a6-d38b00812b3f.png?ixlib=rb-4.0.0&auto=format&gif-q=60&q=75&s=1dc8e81cb8736a850b7f1fc1436279d2)
認証情報の入力が完了すると、何のエラーも出力されずに終了した状態であれば、SpreadSheet側にBigQueryでSQLを実行した結果が指定したシートに出力されているはず。
![スクリーンショット 2017-12-04 11.30.28.png](https://qiita-user-contents.imgix.net/https%3A%2F%2Fqiita-image-store.s3.amazonaws.com%2F0%2F28681%2F0f7b84de-487f-577b-a696-aa51a77a8f2e.png?ixlib=rb-4.0.0&auto=format&gif-q=60&q=75&s=9fbdde9475f7d008543c20af7e404fa2)
グラフを作成する
SpreadSheet上でグラフを作成する。
初回のみ、SQLからの出力データに対し、グラフを作成する必要があります。
次回から新しいデータが入力された際にグラフが自動的に更新され、その更新されたグラフを図としてSlackに投稿するということになります。
詳しいグラフの作成の仕方は他にいい記事がたくさんあるのでここでは割愛します。
![スクリーンショット 2017-12-04 11.32.09.png](https://qiita-user-contents.imgix.net/https%3A%2F%2Fqiita-image-store.s3.amazonaws.com%2F0%2F28681%2F11264fea-1808-350e-d3cc-4a815f89286a.png?ixlib=rb-4.0.0&auto=format&gif-q=60&q=75&s=3e1e3bc29b524ec035de48b0b045d2ef)
Tips: ここまでの部分がグラフを作成するという処理になります。BQをスクリプトで取得するという部分を別の仕組み(Adsenseから自動的にデータを取ってくる)などに切り替えることで別のデータソースを利用することも出来ます。
手順2 (グラフをSlackに送る側を作る)
スクリプト(2) を作成する
メニューの 「新規作成」 → 「スクリプトファイル」 から新規でスクリプトファイルを作成する。
![スクリーンショット 2017-12-04 11.59.33.png](https://qiita-user-contents.imgix.net/https%3A%2F%2Fqiita-image-store.s3.amazonaws.com%2F0%2F28681%2F09158161-225d-121d-383f-77f1efcb58a8.png?ixlib=rb-4.0.0&auto=format&gif-q=60&q=75&s=be641cc5c4f98e81283e7be3a2815480)
スクリプト(2) を貼ります
function notify(){
var sheetName = 'billing' // 送りたいグラフがあるシート名を入力する
var title = '◯◯のレポート' // Slackに通知する際の画像のタイトル
var index = 0 // グラフが1つだけの場合0 (複数のグラフがある場合には1などと指定する)
var chartImage = getChart(sheetName, index)
slack(chartImage, title);
}
// Slackに送る
function slack(chart, title) {
var url = 'https://slack.com/api/files.upload'; // slackのAPIのURL
var token = 'xoxb-000000000000-XXXXXXXXXXXXXXXXXXXXXXXX'; // Slackのボットのトークン
var channel = '#hoge_channel'; // チャンネルに送る時。(事前にbotのinviteが必要)
//var channel = '@foo_san'; // 個人宛に送りたい時
var payload = {
'token' : token,
'channels' : channel,
'file' : chart,
'filename' : title
};
var params = {
'method' : 'post',
'payload' : payload
};
var response = UrlFetchApp.fetch(url, params);
}
// シートからチャートを取得する
function getChart(sheetName, index) {
var ss = SpreadsheetApp.getActiveSpreadsheet();
var sheet = ss.getSheetByName(sheetName); // シート名を指定
var charts = sheet.getCharts();
// グラフが複数ある場合は調整が必要です。
var chartImage = charts[index].getBlob().getAs('image/png').setName("graph.png");
return chartImage
}
書き換える必要がある箇所は以下です
- sheetName
- title
- index (必要に応じて)
- token
- channel
スクリプトを一旦手動実行する
こちらのスクリプトも同様に外部にHTTPアクセスする際の権限の確認が必要なため、一旦手動実行します。
![スクリーンショット 2017-12-04 12.09.46.png](https://qiita-user-contents.imgix.net/https%3A%2F%2Fqiita-image-store.s3.amazonaws.com%2F0%2F28681%2F12975b89-ca5f-a8e5-bbb3-979485bab1fb.png?ixlib=rb-4.0.0&auto=format&gif-q=60&q=75&s=8c544c3d70b093b5cdc85f2a0d7f1d50)
Slackに通知が届いていることを確認する
この段階でSlackの指定したチャンネルに先程作ったグラフが送られてくるので確認する。
![スクリーンショット_2017-12-04_12_13_45.png](https://qiita-user-contents.imgix.net/https%3A%2F%2Fqiita-image-store.s3.amazonaws.com%2F0%2F28681%2F357466b6-2d79-0c05-d755-c7f78fc0ce91.png?ixlib=rb-4.0.0&auto=format&gif-q=60&q=75&s=3ced3c1a3cfb9267458ef1dcf8154415)
定期的に通知が送られるようにする
ここから定期的に処理が行われるようにする設定です。
ここまで読んできてわかるように、
- データを集める処理
- Slackにグラフを投稿する処理
はそれぞれ独立しているため、2つの設定をそれぞれ設定し、タイミングも「データ取得→Slack送信」となるように調整する必要があります。(あたりまえですね)
定期実行を設定する
メニューの 「編集」 → 「現在のプロジェクトのトリガー」 を選択し、
![スクリーンショット 2017-12-04 12.19.54.png](https://qiita-user-contents.imgix.net/https%3A%2F%2Fqiita-image-store.s3.amazonaws.com%2F0%2F28681%2F3a573af8-876c-18f0-7e41-58aeebb1712e.png?ixlib=rb-4.0.0&auto=format&gif-q=60&q=75&s=6eb6f5b119d7dec7f3f64525f86cd7c9)
2つのトリガーを設定します。
- データを集める処理は
runAllQueries
を設定 - Slackにグラフを投稿する処理は
notify
を設定します。
![スクリーンショット 2017-12-04 12.21.06.png](https://qiita-user-contents.imgix.net/https%3A%2F%2Fqiita-image-store.s3.amazonaws.com%2F0%2F28681%2F03089965-279a-fc86-0ef4-e2fdc006a4f9.png?ixlib=rb-4.0.0&auto=format&gif-q=60&q=75&s=4852e9a9f8adec0a92c4d328279c6b28)
処理時間はそれぞれ、 「午前3時〜4時」 → 「午前7時〜8時」 という形で設定しています。
まとめ
これで定期的に確認したい情報をグラフとしてSlackに投稿する仕組みができました。
データの量によりますが、かなり格安(ほぼ無料)で運用できるので、個人で集めているデータをSlackに連携するというイメージであればすぐにできると思います。
ヒントとおまけ
- Queriesのシートは縦にどんどんSQLを増やせます。
- 接続先のBQのプロジェクトが単一であればAテーブル、Bテーブルなど別々にそれぞれSQLを実行しその結果を用いてグラフを作成し通知できます。
- SQLが苦手な人は、シンプルにSQLでテーブルのデータを取ってきて、SpreadSheet上で様々な加工をした後、グラフ化して通知することも出来ます。(ピボットしたいなどという非エンジニアは多いかと)
- SpreadSheet側で大量の処理をすると「コンピュート時間が長い」とお怒りのメールが届きます。SpreadSheet上にもQuery関数と言うものがあり、セルに対してSQLを書ける機能なのですがそちらを利用するほうがセルごとに関数を入れるより処理が軽いような気がします(個人の感想)
参考
[まだBIツールで消耗してるの? ~ サーバレス・KPI分析ダッシュボードをGAS + Slackで]
http://blog.yamotty.com/entry/20161114/1479079800
[BigQueryを簡単にグラフにするGoogle Apps Script]
http://toreta.blog.jp/archives/20649904.html