1
4

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 1 year has passed since last update.

Google Apps ScriptでBluesky Developer APIsで投稿してみる

Last updated at Posted at 2024-02-11

はじめに

Microsoft MVPで著名なHiroさん
Bluesky APIPower Automateから投稿をする記事を公開されていらっしゃいました。

上記を参考に、Google Apps ScriptでBlueskyに投稿する方法を書いていこうかと思います。

Bluesky Developer APIs

簡単にフィードの投稿ができます。
今回はまず、投稿することを実現します。

Blueskyへの投稿は

  1. Bluesky APIを呼び出してセッションを作成、アクセストークンを取得する
  2. (1)で取得したアクセストークンを使って、Blueskyに投稿する

という二段構えの方法です。

Pythonの場合は、専用のライブラリもあるようですので、
さらに投稿は簡単です。

1. 事前準備

APIを利用するためには、

  • Blueskyのアカウント
  • APIを使うためのパスワード

上記の二点が必要です。

前述の、Hiroさんの記事に、明確に解説が書かれているため、この記事では割愛します。

  • (再掲)

2. Google Apps Script

まずはコード

blueskyにシンプルに文字を投稿する
// スクリプトプロパティからユーザーIDとアプリパスワードを取得します。
const USER_ID = PropertiesService.getScriptProperties().getProperty("USER_ID");
const APP_PASSWORD = PropertiesService.getScriptProperties().getProperty("APP_PASSWORD");

/**
 * APIを呼び出してセッションを作成し、フィードに投稿します。
 */
function bluesky_Api() {
  const post_text = "投稿するテキスト";
  const sessionToken = createSession();
  if (sessionToken) {
    postToFeed(sessionToken, post_text);
  }
}

/**
 * セッションを作成するためのAPIを呼び出します。
 * @return {string} アクセスJWTトークン
 */
function createSession() {
  const url = 'https://bsky.social/xrpc/com.atproto.server.createSession';
  const payload = {
    "identifier": USER_ID,
    "password": APP_PASSWORD
  };
  const options = {
    'method': 'post',
    'contentType': 'application/json',
    'payload': JSON.stringify(payload)
  };

  try {
    const response = UrlFetchApp.fetch(url, options);
    const content = response.getContentText();
    const json = JSON.parse(content);
    Logger.log(json.accessJwt);
    return json.accessJwt;
  } catch (error) {
    Logger.log(error.toString());
    return null;
  }
}

/**
 * フィードに投稿するためのAPIを呼び出します。
 * @param {string} accessJwt アクセスJWTトークン
 * @param {string} post_text 投稿する文字列
 */
function postToFeed(accessJwt, post_text) {
  const url = 'https://bsky.social/xrpc/com.atproto.repo.createRecord';
  const payload = {
    "repo": USER_ID,
    "collection": "app.bsky.feed.post",
    "record": {
      "text": post_text,
      "createdAt": new Date().toISOString()
    }
  };
  const options = {
    'method': 'post',
    'headers': {
      'Authorization': 'Bearer ' + accessJwt
    },
    'contentType': 'application/json',
    'payload': JSON.stringify(payload)
  };

  try {
    const response = UrlFetchApp.fetch(url, options);
    const content = response.getContentText();
    const json = JSON.parse(content);
    Logger.log(json);
  } catch (error) {
    Logger.log(error.toString());
  }
}

1. スクリプト プロパティに秘密情報を格納

まずこちらに必要な秘密情報を格納します。

image.png

取得は下記のコードで担います。

  • USER IDBlueskyのアプリパスワード
const USER_ID = PropertiesService.getScriptProperties().getProperty("USER_ID");
const APP_PASSWORD = PropertiesService.getScriptProperties().getProperty("APP_PASSWORD");

2. セッションを作成して、アクセストークンを発行する

function createSession() {
  const url = 'https://bsky.social/xrpc/com.atproto.server.createSession';
  const payload = {
    "identifier": USER_ID,
    "password": APP_PASSWORD
  };
  const options = {
    'method': 'post',
    'contentType': 'application/json',
    'payload': JSON.stringify(payload)
  };

  try {
    const response = UrlFetchApp.fetch(url, options);
    const content = response.getContentText();
    const json = JSON.parse(content);
    Logger.log(json.accessJwt);
    return json.accessJwt;
  } catch (error) {
    Logger.log(error.toString());
    return null;
  }
}

https://bsky.social/xrpc/com.atproto.server.createSessionに、HTTP要求を送ることで、
投稿をするための`アクセス トークンが取得できます。

HTTP要求の戻り値の中にあるaccessJwtが、そのトークンに該当します。

3. BlueSkyに投稿する

function postToFeed(accessJwt, post_text) {
  const url = 'https://bsky.social/xrpc/com.atproto.repo.createRecord';
  const payload = {
    "repo": USER_ID,
    "collection": "app.bsky.feed.post",
    "record": {
      "text": post_text,
      "createdAt": new Date().toISOString()
    }
  };
  const options = {
    'method': 'post',
    'headers': {
      'Authorization': 'Bearer ' + accessJwt
    },
    'contentType': 'application/json',
    'payload': JSON.stringify(payload)
  };

  try {
    const response = UrlFetchApp.fetch(url, options);
    const content = response.getContentText();
    const json = JSON.parse(content);
    Logger.log(json);
  } catch (error) {
    Logger.log(error.toString());
  }
}

accessJwtは、#2で取得した値です。
post_textは、Blueskyに投稿したい文字列を設定します。

投稿に必要なボディは下記の部分です。

const payload = {
    "repo": USER_ID, // 文字通りユーザーID
    "collection": "app.bsky.feed.post",  // フィードへの投稿はこちらで固定
    "record": {
      "text": post_text, // 投稿する文字列
      "createdAt": new Date().toISOString() // 投稿日付!?
    }
};

createdAtというプロパティで、投稿日付をコントロールします。
そもそもできるの?と思っていますが、タイムスタンプで設定の必要があるようです。

幸い現在時刻であればnew Date().toISOString()で簡単に取得できるので、
こちらで時間を設定することにしました。

いざ投稿!

下記の設定で、Google Apps Scriptを起動すると・・・

const post_text = "Google Apps ScriptでBluesky API!";

image.png

非常に気軽です。
Google スプレッドシートと合わせれば、すぐにBotも作れてしまいますね。

今回は投稿の方法のみに焦点を当てて、書いてみました。
APIの種類がとんでもなく多いので、BlueskyのAPIを試すだけで無限に時間が溶けそうです

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

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?