0
0

Go言語で画像付きのツイートを投稿する

Posted at

初めに

個人で和歌をテーマにした以下のスマホ向けのおみくじアプリを開発・運用しています。

Screenshot_home.png Screenshot_fortune_1.png Screenshot_fortune_2.png Screenshot_emperor_list.png Screenshot_emperor_detail.png

[iOS]
https://apps.apple.com/us/app/%E5%A4%A7%E5%BE%A1%E5%BF%83%E3%82%A2%E3%83%97%E3%83%AA/id1627544916

[Android]
https://play.google.com/store/apps/details?id=jp.sikisimanomiti.oomigokoro

今回、アプリで扱っている御製(歴代天皇が詠んだ和歌)をアプリを利用していない人々にも届けるために、定期的にX(旧Twitter)に自動で投稿する機能を実装しました。

構成

実装後の構成としては下図のようになりました(赤枠で囲んだ部分が今回追加した箇所になります)。

バックエンド構成図.png

上図を実現するために以下の機能を追加しました。

  • バックエンドAPI
    • DBからランダムに和歌を1首取得して歴代天皇の画像付きツイートをXへポストする機能
  • インフラ
    • Cloud Schedulerで毎日AM06:00に上記追加したAPIをコールする機能

バックエンドAPIはGo言語で実装していて、画像付きでXへポストする機能を既存のクライアントライブラリでは実現できなかったので自分でXのAPIを呼んで実装しました。備忘として実装したコードを残して

実装

①画像付きツイートを投稿する

Xのツイートを投稿するAPIのドキュメント(V1.1とV2.0)を確認すると、画像付きのツイートを投稿することはできません。また、X API platform resourcesによると、画像をアップロードするAPIはV2.0ではまだ存在しないようでV1.1のAPIを使うしかありませんが、ツイートを投稿するAPIはV2.0を利用することにします(いつ廃止になるか分からないので)。

現状、Xに画像付きのツイートを投稿するなら以下の手順で行うことになると思います。

  1. V1.1のメディアアップロードのAPIで画像を投稿
  2. 1のAPI呼び出しで取得したmedia_idを付与してツイートを投稿

上記手順で実行するGo言語のコードを以下に掲載します。

golang
import (
	"bytes"
	"context"
	"encoding/json"
	"fmt"
	"github.com/caarlos0/env"
	"github.com/dghubble/oauth1"
	"io"
	"log"
	"mime/multipart"
	"net/http"
	"os"
)

type config struct {
	ConsumerKey       string `env:"CONSUMER_KEY,required"`
	ConsumerSecret    string `env:"CONSUMER_SECRET,required"`
	AccessToken       string `env:"ACCESS_TOKEN,required"`
	AccessTokenSecret string `env:"ACCESS_TOKEN_SECRET,required"`
}

const (
	// 画像をアップロードするAPIのエンドポイント
	UploadMediaEndpoint = "https://upload.twitter.com/1.1/media/upload.json"
	// ツイート投稿用のAPIのエンドポイント
	ManageTweetEndpoint = "https://api.twitter.com/2/tweets"
)

func PostPoemToX(ctx context.Context) error {
	// 環境変数の読み込み
	var cfg config
	if err := env.Parse(&cfg); err != nil {
		log.Fatal(err)
	}

	// アップロードする画像のファイルパスを指定
	filePath := "images/sample.jpg"

	// OAuth1のクライアントを作成
	config := oauth1.NewConfig(cfg.ConsumerKey, cfg.ConsumerSecret)
	token := oauth1.NewToken(cfg.AccessToken, cfg.AccessTokenSecret)
	httpClient := config.Client(oauth1.NoContext, token)

	// ファイルを開く
	file, err := os.Open(filePath)
	if err != nil {
		log.Fatal("Error opening file:", err)
		return err
	}
	defer file.Close()

	// マルチパートフォームデータのバッファを作成
	body := &bytes.Buffer{}
	writer := multipart.NewWriter(body)

	// ファイルパートを作成
	part, err := writer.CreateFormFile("media", filePath)
	if err != nil {
		log.Fatal("Error creating form file:", err)
		return err
	}

	// ファイルの内容をコピー
	_, err = io.Copy(part, file)
	if err != nil {
		log.Fatal("Error copying file content:", err)
		return err
	}

	// 書き込みを完了
	err = writer.Close()
	if err != nil {
		log.Fatal("Error closing writer:", err)
		return err
	}

	// 画像をアップロードするリクエストを作成
	req, err := http.NewRequest("POST", UploadMediaEndpoint, body)
	if err != nil {
		log.Fatal("Error creating request:", err)
		return err
	}

	// ヘッダーを設定
	req.Header.Set("Content-Type", writer.FormDataContentType())

	// リクエストを送信
	resp, err := httpClient.Do(req)
	if err != nil {
		log.Fatal("Error sending request:", err)
		return err
	}
	defer resp.Body.Close()

	// レスポンスの内容を読み取る
	respBody, err := io.ReadAll(resp.Body)
	if err != nil {
		log.Fatal("Error reading response:", err)
		return err
	}

	// レスポンスからmedia_idを取得
	mediaID, err := extractMediaID(string(respBody))
	if err != nil {
		log.Fatal("Failed to extract media ID:", err)
		return err
	}

	// JSONにエンコード
	tweetBody, err := json.Marshal(map[string]interface{}{
		"text": "Hello, X!",
		"media": map[string]interface{}{
			"media_ids": []string{mediaID},
		},
	})
	if err != nil {
		log.Fatal("Error marshaling tweet data:", err)
		return err
	}

	// ツイートを送信するリクエストを作成
	req, err = http.NewRequest("POST", ManageTweetEndpoint, bytes.NewBuffer(tweetBody))
	if err != nil {
		log.Fatal("Error creating tweet request:", err)
		return err
	}

	// ヘッダーを設定
	req.Header.Set("Content-Type", "application/json")

	// リクエストを送信
	resp, err = httpClient.Do(req)
	if err != nil {
		log.Fatal("Error sending tweet request:", err)
		return err
	}
	defer resp.Body.Close()

	// レスポンスの内容を読み取る
	respBody, err = io.ReadAll(resp.Body)
	if err != nil {
		log.Fatal("Error reading tweet response:", err)
		return err
	}

	return nil
}

type UploadMediaResponse struct {
	MediaIDString string `json:"media_id_string"`
}

// JSONレスポンスからmedia_idを抽出
func extractMediaID(respBody string) (string, error) {
	var uploadResponse UploadMediaResponse

	// JSONレスポンスを構造体にデコード
	err := json.Unmarshal([]byte(respBody), &uploadResponse)
	if err != nil {
		return "", fmt.Errorf("failed to parse JSON response: %w", err)
	}

	// media_id_stringを返す
	return uploadResponse.MediaIDString, nil
}

②Cloud Schedulerで毎日AM06:00にAPIをコール

インフラはTerraformを使って構築しているので、毎日AM06:00にAPIをコールするCloud Schedulerを以下のTerraformのコードで構築しました。

terraform
resource "google_cloud_scheduler_job" "post-tweet-job" {
  name             = "post-tweet-job"
  description      = "post tweet job"
  schedule         = "0 6 * * *"
  time_zone        = "Asia/Tokyo"
  attempt_deadline = "320s"

  retry_config {
    retry_count = 1
  }

  http_target {
    http_method = "POST"
    uri         = "https://example.jp/api/poems"
  }
}

まとめ

APIがV1.1とV2.0とあって、できることが微妙に違うので、ちょっと迷いましたが、最終的に画像付きツイートを投稿できるようになったので良かった。

参考

0
0
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
0
0