1
2

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

Youtubeチャンネルを毎日1つSlackにPOSTするGASを組んだのでメモ

Last updated at Posted at 2021-12-12

概要

Youtubeチャンネルって学習コンテンツとして最近は色々優秀で定期で見ていたりするんですが、いちいちチェックしたりとか前から1つずつ遡ったりとかがすごくめんどくさいんですよね。

Feed登録したりとかして、新着がきたらお知らせるすることができるようにできたりはしてるんですが、1日1つ流れてくる仕組みが作れたら学習にとてもいいな〜と思い組んでみました。

やったこと

  • Youtubeチャンネルの動画一覧を取得してスプレッドシートに一覧を作成する
  • 作成した動画一覧から、「まだみてない動画」を順に指定のSlack channelに投稿する

こちら、作成にあたって以下のスクリプトを参考・引用させていただいております。ありがとうございます。

事前に必要なもの

  • Googleのdeveloperアカウント
  • Slackのワークスペース

準備(KEYとかIDとかURLとかの取得)

Youtube

Google Cloud Platform(準備①)

YoutubeのAPIを利用するにあたって、GCPのAPI認証キーの発行が必要になります。
こちらから発行しましょう。

スクリーンショット 2021-12-12 16.57.16.png

YoutubeチャンネルのID(準備②)

取得したいYoutubeチャンネルのIDが必要なのでとってきます。ここだろうな、と思ってとってきたらいけました。(記事用にキャプチャした時にたまたま気づいたんですが、ここって独自の文字列にできるんですね、てことは変わることもあるのか。。。)

スクリーンショット 2021-12-12 16.48.51.png

SlackのWebhookURL(準備③)

Slackのアプリ管理からIncoming Webhookを追加して、URLを作成します。以下URLから、POSTしたいSlackのワークスペースを選択してアプリ管理を開きます。

Incoming Webhookはアプリ検索で出てくるので、選択して追加。

スクリーンショット 2021-12-12 17.02.33.png

スクリーンショット 2021-12-12 17.04.29.png

POSTしたいSlack channelを選択して、追加。

スクリーンショット 2021-12-12 17.05.39.png

WebhookURLが生成される

スクリーンショット 2021-12-12 17.09.40.png

GASのScriptを貼り付け

スプレッドシートの準備(準備④)

「youtube」「(動画一覧を貼り付けるシート)」というシートのあるSpreadSheetありきのScriptになりますのでご注意。

script

/**
 * 独自メニューの追加
 */
function onOpen(){
  var myMenu = [ //メニュー配列
    {name: "更新開始", functionName: "update_videoLists"},
  ];
  SpreadsheetApp.getActiveSpreadsheet().addMenu("【動画一覧更新】",myMenu); 
}

/**
 * チャンネル動画アップデート
 */
function update_videoLists() {
  recordVideoLists("(YoutubeチャンネルのID)※準備②", "(動画一覧シートの名前)※準備④");
}

/**
 * 動画一覧アップデート
 */
var recordVideoLists = function(channelId, sheetName) {

  // 既に登録されている一覧を取得
  const ss = SpreadsheetApp.getActiveSpreadsheet();
  const sheet = ss.getSheetByName(sheetName);
  sheet.activate();
  const range = sheet.getRange(2, 1, sheet.getLastRow() - 1, sheet.getLastColumn());
  const rangeValues = range.getValues();

  var result = [];
  var lists = getVideoLists(channelId);

  // Youtubeから取得した動画一覧
  for(var i = 0; i < lists.length; i++) {
    var r = lists[i];

    // 既に登録済みの動画はinsertせずcontinue
    let alreadyFlg = false;
    for (let i = 0; i < rangeValues.length; i++) {
      if (r.videoId === rangeValues[i][4]) {
        alreadyFlg = true;
      }
    }
    if (alreadyFlg === true) {
      continue;
    }

    // 未登録の動画はinsert
    var date = new Date();
    var pub_date = new Date(r.publishedAt);
    var record = ['=ROW()-2',date, pub_date, r.title, r.videoId, 'https://www.youtube.com/watch?v=' + r.videoId, '0'];
    result.push(record);
  }
  insertRecords(result, sheetName, 3); // 3列目からデータを挿入する
}

/**
 * チャンネル情報取得して返す
 */
var getChannelInfo = function(id) {
  var key = '(Google Cloud Platform のAPIKEY)※準備①'; 
  var url = "https://www.googleapis.com/youtube/v3/channels?part=statistics,snippet&id=" + id +"&key=" + key;
  var response = UrlFetchApp.fetch(url);
  var json = JSON.parse(response.getContentText());
  var item = json.items[0];
  var channel_id = item.id;
  var statistics = item.statistics;
  var viewCount = statistics.viewCount;
  var subscriberCount = statistics.subscriberCount;
  var videoCount = statistics.videoCount;
  var snippet = item.snippet;
  var title = snippet.title;
  var description = snippet.description;
  var customUrl = snippet.customUrl;
  var publishedAt = snippet.publishedAt;

  var res = {channel_id: channel_id, viewCount: viewCount, subscriberCount: subscriberCount, videoCount: videoCount, 
    title: title, description:description, customUrl:customUrl, publishedAt: publishedAt};
  return res
}

/**
 * シートを作成して動画一覧を記載
 */
var insertRecords = function(arrData, sheetName, startRow){
  let rows = arrData.length;
  if (rows > 0) {
    let cols = arrData[0].length;
    let ss = SpreadsheetApp.getActiveSpreadsheet(); //スプレッドシートを取得
    let sheet = ss.getSheetByName(sheetName); // sheetName という名前のシートを取得
    if (sheet == null) {
      sheet = mySS.insertSheet(sheetName); // 新シートの追加
      Logger.log("シートを追加しました:" + sheetName);
    }
    sheet.insertRows(startRow, rows); // 空行の作成
    sheet.getRange(startRow, 1, rows, cols).setValues(arrData); // データの書き込み
  }
}

/**
 * 動画一覧を取得
 */
var getVideoLists = function(channelId) {
  let lists = [];
  let pageToken = "";
  for(let page = 0; page < 4; page++) {
    let url = "https://www.googleapis.com/youtube/v3/search";
    url += "?part=snippet&maxResults=50&order=date&type=video";
    url += "&pageToken=" + pageToken + "&channelId=" + channelId +"&key=" + 'AIzaSyBwq7ieCfIQp5E72H5k8mJhpYOEUJ_-JcM';
    let response = UrlFetchApp.fetch(url);
    let json = JSON.parse(response.getContentText());
    if (json.kind == "youtube#searchListResponse") {
      let items = json.items;
      for(let i = 0; i < items.length; i++) {
        let item = items[i];
        if (item.kind == "youtube#searchResult") {
          let id = item.id;
          if (id.kind == "youtube#video") {
            let snippet = item.snippet;
            let video = {videoId: id.videoId, publishedAt: snippet.publishedAt, title: snippet.title};
            lists.push(video);
          }
        }
      }    
    }
    if ("nextPageToken" in json) {
      pageToken = json.nextPageToken;
    } else {
      break;
    }
  }
  return lists
}

/**
 * 動画を1つSlackにポストして投稿
 */
function todayMovie() {
  
  const ss = SpreadsheetApp.getActiveSpreadsheet();
  const sheet = ss.getSheetByName('(動画一覧のシート名)※準備④');
  sheet.activate();
  const range = sheet.getRange(3, 1, sheet.getLastRow() - 2, sheet.getLastColumn());
  const rangeValues = range.getValues();

  let alreadyFlg = false;
  for (let i = 0; i < rangeValues.length; i++) {
    if (alreadyFlg === true) {
      continue;
    }
    Logger.log(rangeValues[i][6]);
    if (rangeValues[i][6] === 0.0) {
      alreadyFlg = true;
      sheet.getRange(3 + i, 7).setValue(1);
      postSlack(rangeValues[i]);
    }
  }
}

/**
 * Slack投稿
 */
function postSlack(issue) {
  if (issue.length <= 0) {
    return;
  }
  console.log(issue.length);
  
  const postUrl  = '(SlackのWebhookURL)※準備③';
  // slackのincoming webhook用
  const username = 'botbot';  // 通知時に表示されるユーザー名
  const icon     = ':hatching_chick:'; // 通知時に表示されるアイコン
  const subject = '【本日の動画ですよ~】'; //この辺はご自由に
  const body    = '' + subject + '\n' + createPostMessage(issue) + '\n';
  const jsonData = {
    'username'  : username,
    'icon_emoji': icon,
    'text'      : body
  };
  const options = {
    'method'     : 'post',
    'contentType': 'application/json',
    'payload'    : JSON.stringify(jsonData)
  };

  UrlFetchApp.fetch(postUrl, options);
}

/**
 * 投稿メッセージ作成
 */
function createPostMessage(issues) {
  let message = '';
  message += formatDate(new Date(issues[2])) + '配信\n';;
  message += issues[3] + '\n';
  message += issues[5] + '\n';
  return message;
}

/**
 * 日付フォーマット作成
 */
function formatDate(date) {
    var format = 'YYYY-MM-DD';
    format     = format.replace(/YYYY/g, date.getFullYear());
    format     = format.replace(/MM/g, ('0' + (date.getMonth() + 1)).slice(-2));
    format     = format.replace(/DD/g, ('0' + date.getDate()).slice(-2));
    return format;
}

一応トリガーを設定

スクリーンショット 2021-12-12 17.14.37.png

一応使い方

基本はトリガに従って、毎日指定のSlack channelに流してくれるだけなので、特段触ったりメンテナンスしたりする必要はありません。

一応スプレッドシートの方には、動画一覧のシートが作成されていて、「SlackPOSTFlg」みたいな項目で投稿の有無がわかるようになっています。(なのでこの数値を0に戻すと再度投稿の対象になります)

スクリーンショット 2021-12-12 17.23.38.png

雑記

ちなみに、個人的にはもうここまででいいやってなったことなのですが、記事公開にあたって気になってる箇所は一応列挙しておきます

  • スプレッドシートのタブに手動更新用の導線をつけたものの、なんか動画が怪しい(同じ動画をもっかい取得してしまったりする)。たぶん、一覧取得・書き込みの順番を見直さないといけない。
  • それぞれの関数の組み合わせとか、引数の持たせ方でもっと拡張性あげられると思う。いろんなチャンネルの動画をPOSTしたり、複数のSlack channelにPOSTしたり。
  • 取得したYoutubeチャンネルの動画は、できたら「配信開始」の日付順で流したい。多分YoutubeAPIの取得順をいじればできると思う。

以上、おわり。

1
2
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
1
2

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?