25
20

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

OAuth2、API v2でTwitter🐦にツイートできるようにしてみた(GAS)

Last updated at Posted at 2023-03-01

はじめに

Twitterの投稿をAPI v2とOAuth2をGoogle Apps Scriptで利用したときの対応をまとめています。
各々の記事は見つかったのですが両方使ってツイートする記事は見つからなかったので実施時の対応を共有します。

参考サイト

準備

アクセスキーの発行

Twitterのアクセスキーを発行します。このサイトを参考に「2. Developer アカウント登録(Essential利用)」を実施します。

スクリプト プロパティへアクセスキーを登録

アクセスキー発行で発行されたCLIENT_ID CLIENT_SECRETをPropaties Serviceへ登録する。

image.png

GAS(Google Apps Script)のTwitter側の設定

GAS(Google Apps Script)のIDの確認

GASでプロジェクトの設定からスクリプトIDをメモしておく。

Twitter側へCallback URLを入力

[GAS(Google Apps Script)のIDの確認](#GAS(Google Apps Script)のIDの確認)でメモしたスクリプトIDをここへアクセスし、認証設定にGAS(Google Apps Script)のURLを設定する

https://script.google.com/macros/d/{スクリプトID}/usercallback

image.png

実装

GASでOAuthを利用するためのライブラリ追加

GASプロジェクトでライブラリの追加でスクリプトIDに「1B7FSrk5Zi6L1rSxxTDgDEUsPzlukDsi4KGuTMorsTQHhGBzBkMun4iDF」入力して検索
IDが「OAuth2」であることを確認して追加しておく

Twitter 認証部分

こちらを参考に認証部分を実装
※ツイートするには .setScopetweet.writeが必要です。

function getService() {
  pkceChallengeVerifier();
  const userProps = PropertiesService.getUserProperties();
  const scriptProps = PropertiesService.getScriptProperties();
  const clientId = scriptProps.getProperty('CLIENT_ID');
  const clientSecret = scriptProps.getProperty('CLIENT_SECRET');

  return OAuth2.createService('twitter')
    .setAuthorizationBaseUrl('https://twitter.com/i/oauth2/authorize')
    .setTokenUrl('https://api.twitter.com/2/oauth2/token?code_verifier=' + userProps.getProperty("code_verifier"))
    .setClientId(clientId)
    .setClientSecret(clientSecret)
    .setCallbackFunction('authCallback')
    .setPropertyStore(userProps)
    .setScope('users.read tweet.read tweet.write offline.access')
    .setParam('response_type', 'code')
    .setParam('code_challenge_method', 'S256')
    .setParam('code_challenge', userProps.getProperty("code_challenge"))
    .setTokenHeaders({
      'Authorization': 'Basic ' + Utilities.base64Encode(clientId + ':' + clientSecret),
      'Content-Type': 'application/x-www-form-urlencoded'
    })
}

function authCallback(request) {
  const service = getService();
  const authorized = service.handleCallback(request);
  if (authorized) {
    return HtmlService.createHtmlOutput('Success!');
  } else {
    return HtmlService.createHtmlOutput('Denied.');
  }
}

function pkceChallengeVerifier() {
  var userProps = PropertiesService.getUserProperties();
  if (!userProps.getProperty("code_verifier")) {
    var verifier = "";
    var possible = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789-._~";

    for (var i = 0; i < 128; i++) {
      verifier += possible.charAt(Math.floor(Math.random() * possible.length));
    }

    var sha256Hash = Utilities.computeDigest(Utilities.DigestAlgorithm.SHA_256, verifier)

    var challenge = Utilities.base64Encode(sha256Hash)
      .replace(/\+/g, '-')
      .replace(/\//g, '_')
      .replace(/=+$/, '')
    userProps.setProperty("code_verifier", verifier)
    userProps.setProperty("code_challenge", challenge)
  }
}

function logRedirectUri() {
  var service = getService();
  Logger.log(service.getRedirectUri());
}

初回認証用のメソッド

初回認証用のメソッドで実行してOpen the following URL and re-run the script:につづくURLにアクセスし認証を実施する。
Successが表示されればOK!

function main() {
  const service = getService();
  if (service.hasAccess()) {
    Logger.log("Already authorized");
  } else {
    const authorizationUrl = service.getAuthorizationUrl();
    Logger.log('Open the following URL and re-run the script: %s', authorizationUrl);
  }
}

ツイート用メソッド

こちらを参考にツイート部分の実装をする。

const endpoint2 = "https://api.twitter.com/2/tweets";

function postTwitter(){
  let twitterText = "Test tweet!!!"; // ツイートしたいメッセージを入力
  let service = getService();
  if (service.hasAccess()) {   
    let message = {
      text: twitterText
    }

    const headers = {
      Authorization: 'Bearer ' + service.getAccessToken()
    }
    const response = UrlFetchApp.fetch(endpoint2, {
      method: "post",
      headers,
      muteHttpExceptions: true,
      payload: JSON.stringify(message),
      contentType: "application/json"
    });

    const result = JSON.parse(response.getContentText());

    Logger.log(JSON.stringify(result, null, 2));
    
  } else {
    Logger.log("Not Authorized");
    return null;
  }
}

実行

GASでpostTwitterを指定して実行するとツイートできます!

image.png

image.png

まとめ

OAuth2、API v2を利用してTwitterにツイートを実施することができました。

注意事項

  • Twitter APIの有償化アナウンスがされていますがこの記事の時点では1500回までの利用であれば無償で利用出来そうです。
  • ですが今後どうなるかわからないです。
  • APIの申請から発行までが時間が掛かりそう。
    ‐ Twitter APIの審査は厳しくなっているようです。

25
20
6

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
25
20

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?