PHPでBlueskyのプロフィールを更新する

(1/1)
今回は、PHPで Bluesky APIを利用し、自分のプロフィールを更新するプログラムを作ってみる。更新対象は下記の項目である。
  • 表示名
  • プロフィール情報
  • アバター画像
  • バナー画像
  • プロフィール固定メッセージ
各々の項目は独立して省略することが可能で(パラメータにNULLを代入したときに省略とみなす)、省略時には現在登録されている情報を更新せずに残すことにする。Blusky Webアプリのプロフィール変更UIは、Twitter(現・X)に比べて貧弱だ。ここでは、画像ファイルを画面に表示されている画像にドラッグ&ドロップすることで更新したり、ボタンをクリックしてファイルダイアログを表示して選択する 2つの方法を実装する。ドラッグ&ドロップもしくは選択した画像ファイルは、サーバ通信せずに、JavaScriptだけで画面表示を更新する。
(2025年11月21日)PHP8.5対応:curl_close,imagedestroyを実行しないようにした.
(2025年8月17日)画像に余計な空白が入らないようにするため一部仕様変更.
(2025年8月14日).pahooEnv導入

目次

サンプル・プログラムの実行例

PHPでBlueskyのプロフィールを更新

サンプル・プログラム

圧縮ファイルの内容
updateProfilesBluesky.phpサンプル・プログラム本体
.pahooEnvクラウドサービスを利用するためのアカウント情報などを記入する .env ファイル。
使い方は「各種クラウド連携サービス(WebAPI)の登録方法」を参照。include_path が通ったディレクトリに配置すること。
pahooInputData.phpデータ入力に関わる関数群。
使い方は「数値入力とバリデーション」「文字入力とバリデーション」などを参照。include_path が通ったディレクトリに配置すること。
pahooBlueskyAPI.phpBluesky APIに関わるクラス pahooBlueskyAPI。
使い方は「PHPでPHPでBlueskyに投稿する」などを参照。include_path が通ったディレクトリに配置すること。
pahooScraping.phpスクレイピング処理に関わるクラス pahooScraping。
スクレイピング処理に関わるクラスの使い方は「PHPでDOMDocumentを使ってスクレイピング」を参照。include_path が通ったディレクトリに配置すること。
updateProfilesBluesky.php 更新履歴
バージョン 更新日 内容
1.6.0 2025/08/14 .pahooEnv導入
1.0.0 2025/07/19 初版
pahooBlueskyAPI.php 更新履歴
バージョン 更新日 内容
2.7.1 2025/11/22 PHP8.5対応:curl_close,imagedestroyを実行しないようにした
2.7.0 2025/08/17 reductImage,uploadBlob仕様変更←画像に余計な空白が入らないようにするため
2.6.0 2025/08/14 .pahooEnv 導入
2.5.1 2025/08/10 uploadBlob() -- bug-fix
2.5.0 2025/08/02 getOGPInformation() -- og:imageがないページに対応
pahooScraping.php 更新履歴
バージョン 更新日 内容
1.2.1 2024/10/31 __construct() 文字化け対策
1.2.0 2024/09/29 getValueFistrXPath() 属性値でない指定に対応
1.1.0 2023/10/15 getValueFistrXPath() 追加
1.0.1 2023/09/29 __construct() bug-fix
1.0.0 2023/09/18 初版
pahooInputData.php 更新履歴
バージョン 更新日 内容
2.0.1 2025/08/11 getParam() bug-fix
2.0.0 2025/08/11 pahooLoadEnv() 追加
1.9.0 2025/07/26 getParam() 引数に$trim追加
1.8.1 2025/03/15 validRegexPattern() debug
1.8.0 2024/11/12 validRegexPattern() 追加

準備:PHP の https対応

クラウド連携や相手先サイトのデータを読み込むのに https通信を使うため、PHPに OpenSSLモジュールが組み込まれている必要がある。関数  phpinfo  を使って、下図のように表示されればOKだ。
OpenSSL - PHP
そうでない場合は、次の手順に従ってOpenSSLを有効化し、PHPを再起動させる必要がある。

Windowsでは、"php.ini" の下記の行を有効化する。
extension=php_openssl.dll
Linuxでは --with-openssl=/usr オプションを付けて再ビルドする。→OpenSSLインストール手順

これで準備は完了だ。

準備:pahooInputData 関数群

PHPのバージョンや入力データのバリデーションなど、汎用的に使う関数群を収めたファイル "pahooInputData.php" が同梱されているが、include_path が通ったディレクトリに配置してほしい。他のプログラムでも "pahooInputData.php" を利用するが、常に最新のファイルを1つ配置すればよい。

また、各種クラウドサービスに登録したときに取得するアカウント情報、アプリケーションパスワードなどを登録した .pahooEnv ファイルから読み込む関数 pahooLoadEnv を備えている。こちらについては、「各種クラウド連携サービス(WebAPI)の登録方法」をご覧いただきたい。

解説:pahooBlueskyAPIクラス

Bluesky に投稿したりプログラムで操作するAPIについては、公式リファレンスが詳しい。APIを利用するには、事前に、あなたのアカウントから利用登録を行い、アプリパスワードを取得する必要がある。その手順は「Bluesky API - 各種クラウド連携サービス(WebAPI)の登録方法」をご覧いただきたい。
Bluesky APIを利用するメソッドはクラス "pahooBlueskyAPI.php" に分離している。また、このクラスからクラス "pahooScraping.php" を呼び出すので、2つのクラス・ファイルを include_path の通ったディレクトリに配置すること。

解説:セッション開始

BlueskyAPI を利用するには、まずセッションを開き、アクセストークン accessJwt を取得する。使用するエンドポイントは com.atproto.server.createSession だ。
com.atproto.server.createSession
URL
https://{PDSドメイン}/xrpc/com.atproto.server.createSession
リクエスト・データ(http) header Content-Type "application/json" post identifier ハンドル名 password アプリケーション・パスワード
レスポンス・データ(json) accessJwt アクセストークン refreshJwt リフレッシュトークン handle ハンドル名 did did
アクセストークン accessJwt は、後述するメッセージや画像の投稿で使用する。寿命は1~2時間だ。
アクセストークン refreshJwt は、アクセストークンの再発行や、セッションの終了・破棄に用いることができ、寿命は数十日と長い。リフレッシュトークン refreshJwt をストレージに保存しておき、次回はアクセストークンを再発行するというのが BlueskyAPI の望ましい運用方法と思われるが、リフレッシュトークン refreshJwt だけでアクセストークン accessJwt を再発行できてしまうので、流出するとたいへん危険である。

今回つくるプログラムは、単発でメッセージや画像を投稿するものなので、リフレッシュトークン refreshJwt は使わず、プログラム起動時にアクセストークン accessJwt を取得するようにする。

pahooBlueskyAPI.php

  15: // スクレイピング処理に関わるクラス:include_pathが通ったディレクトリに配置
  16: require_once('pahooScraping.php');
  17: 
  18: // Bluesky API クラス =======================================================
  19: class pahooBlueskyAPI {
  20:     public $pds;            // PDSドメイン
  21:     public $webapi  ;       // 直前に呼び出したWebAPI URL
  22:     public $errmsg;         // エラーメッセージ
  23:     public $accessJwt;      // accessJwt
  24:     public $refreshJwt;     // refreshJwt
  25: 
  26:     const INTERNAL_ENCODING = 'UTF-8';  // 内部エンコーディング
  27:     const MAX_MESSAGE_LEN = 300;        // 投稿可能なメッセージ文字数
  28:     const URL_LEN = 23;                 // メッセージ中のURL文字数(相当)
  29:     const MAX_IMAGE_WIDTH  = 1200;      // 投稿可能な最大画像幅(ピクセル)
  30:     const MAX_IMAGE_HEIGHT = 675;       // 投稿可能な最大画像高さ(ピクセル)
  31:                                         // これより大きいときは自動縮小する
  32:     // トークンを保存するファイル名
  33:     // 秘匿性を保つことができ、かつ、PHPプログラムから読み書き可能であること
  34:     const FILENAME_TOKEN = './.token';
  35: 
  36:     // -- 以下のデータは .env ファイルに記述可能
  37:     // Bluesky API アプリパスワード
  38:     // https://bsky.app/
  39:     public $BLUESKY_HANDLE   = '';      // ハンドル名
  40:     public $BLUESKY_PASSWORD = '';      // アプリケーション・パスワード

BlueskyAPI を利用するユーザー定義クラス pahooBlueskyAPIをつくる。
上述の手順で取得したアプリケーション・パスワードをプロパティ変数 $BLUESKY_PASSWORD に、あなたのハンドル名を $BLUESKY_HANDLE に代入する。
投稿可能な最大文字数は定数 MAX_MESSAGE_LEN として用意した。現在の仕様では300文字だ。

pahooBlueskyAPI.php

  42: /**
  43:  * コンストラクタ
  44:  * もしAPIエラーが出る場合には,新規セッションにしてみる.
  45:  * @param   string $pds PDSドメイン
  46:  * @param   bool $newSession 新規セッションにするかどうか(TRUE:新規,デフォルトはFALSE)
  47:  * @return  なし
  48: */
  49: function __construct($pds, $newSession=FALSE) {
  50:     if (isset($_ENV['PAHOO_BLUESKY_HANDLE'])) {
  51:         $this->BLUESKY_HANDLE   = $_ENV['PAHOO_BLUESKY_HANDLE'];
  52:     }
  53:     if (isset($_ENV['PAHOO_BLUESKY_PASSWORD'])) {
  54:         $this->BLUESKY_PASSWORD = $_ENV['PAHOO_BLUESKY_PASSWORD'];
  55:     }
  56: 
  57:     // プロパティを初期化する.
  58:     $this->pds        = $pds;
  59:     $this->webapi     = '';
  60:     $this->errmsg     = '';
  61:     $this->accessJwt  = '';
  62:     $this->refreshJwt = '';
  63: 
  64:     // 新規セッションを開始する.
  65:     if ($newSession) {
  66:         $this->createSession();
  67:     }
  68: }

コンストラクタの引数は PDSドメインで、変数 $pds に保管し、API呼び出し時に参照できるようにした。

pahooBlueskyAPI.php

 360: /**
 361:  * 新規セッションを開始する.
 362:  * @param   なし
 363:  * @return  bool TRUE:成功/FALSE:失敗
 364: */
 365: function createSession() {
 366:     // エラーメッセージ・クリア
 367:     $this->clearerror();
 368: 
 369:     // リクエストURL
 370:     $requestURL = 'https://' . $this->pds . '/xrpc/com.atproto.server.createSession';
 371:     $this->webapi = $requestURL;
 372:     $ch = curl_init($requestURL);
 373:     // cURLを使ったリクエスト
 374:     curl_setopt($ch, CURLOPT_CONNECTTIMEOUT, 10);
 375:     curl_setopt($ch, CURLOPT_HTTPHEADER, ['Content-Type: application/json']);
 376:     curl_setopt($ch, CURLOPT_RETURNTRANSFER, TRUE);
 377:     curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, FALSE); // サーバ証明書検証をスキップ
 378:     curl_setopt($ch, CURLOPT_SSL_VERIFYHOST, FALSE); //  〃
 379:     curl_setopt($ch, CURLOPT_POST, TRUE);
 380:     curl_setopt($ch, CURLOPT_POSTFIELDS, json_encode([
 381:         'identifier' => $this->BLUESKY_HANDLE,
 382:         'password'   => $this->BLUESKY_PASSWORD,
 383:     ]));
 384: 
 385:     // レスポンス処理
 386:     $response = curl_exec($ch);
 387: //  var_dump('*createSession');
 388: //  var_dump($response);
 389:     $httpStatusCode = curl_getinfo($ch, CURLINFO_RESPONSE_CODE);
 390:     if ($httpStatusCode !200) {
 391:         $this->seterror('セッションを開始できません');
 392:         return FALSE;
 393:     }
 394:     if (PHP_VERSION_ID < 80500) {
 395:         curl_close($ch);
 396:     }
 397:     $items = json_decode($response, TRUE);
 398: 
 399:     // エラーチェックとリターン
 400:     if (isset($items['accessJwt']) && isset($items['refreshJwt'])) {
 401:         $this->accessJwt = (string)$items['accessJwt'];
 402:         $this->refreshJwt = (string)$items['refreshJwt'];
 403:         // トークンをファイルに保存する
 404:         $contents = $this->accessJwt . "\n" . $this->refreshJwt;
 405:         file_put_contents(self::FILENAME_TOKEN, $contents);
 406:         return TRUE;
 407:     } else if (isset($items['error'])) {
 408:         $this->seterror($items['message']);
 409:         return FALSE;
 410:     } else {
 411:         $this->seterror('セッションを開始できません');
 412:         return FALSE;
 413:     }
 414: }

セッション開始メソッド createSession は、引数はなく、セッション開始に成功したかどうかを戻り値にする。

メソッドの中身は、上述のAPI仕様の通りに作った。
POSTプロトコルとして、これまでのクラウドサービス利用でも使ってきた cURL関数を利用する。

解説:セッション終了

アクセストークン accessJwt のセッションを終了するには、エンドポイント com.atproto.server.deleteSession を呼び出す。
com.atproto.server.deleteSession
URL
https://{PDSドメイン}/xrpc/com.atproto.server.deleteSession
リクエスト・データ(json) header Content-Type "application/json" post identifier "Bearer {アクセストークン}" password アプリケーション・パスワード
セッション開始時に取得したアクセストークン accessJwt を使い、このセッションをクローズする。以後、このアクセストークン accessJwt は使用できなくなる。
アクセストークン accessJwt の盗用を避ける意味で、セッションを開始したら、かならずセッション終了するようにしよう。

APIの戻り値は、httpステータスが200であれば成功、それ以外であればエラー情報が戻る。

pahooBlueskyAPI.php

 472: /**
 473:  * セッション終了する.
 474:  * @param   なし
 475:  * @return  bool TRUE:成功/FALSE:失敗
 476: */
 477: function deleteSession() {
 478:     // リクエストURL
 479:     $requestURL = 'https://' . $this->pds . '/xrpc/com.atproto.server.deleteSession';
 480:     $this->webapi = $requestURL;
 481:     $ch = curl_init($requestURL);
 482:     // cURLを使ったリクエスト
 483:     curl_setopt($ch, CURLOPT_CONNECTTIMEOUT, 10);
 484:     curl_setopt($ch, CURLOPT_HTTPHEADER, ['Authorization: Bearer ' . $this->accessJwt]);
 485:     curl_setopt($ch, CURLOPT_CUSTOMREQUEST, 'DELETE');
 486:     curl_setopt($ch, CURLOPT_RETURNTRANSFER, TRUE);
 487:     curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, FALSE); // サーバ証明書検証をスキップ
 488:     curl_setopt($ch, CURLOPT_SSL_VERIFYHOST, FALSE); //  〃
 489: 
 490:     // レスポンス処理
 491:     $response = curl_exec($ch);
 492:     if (curl_errno($ch)) {
 493:         $this->seterror('セッション終了できません' . curl_error($ch));
 494:         return FALSE;
 495:     }
 496:     if (PHP_VERSION_ID < 80500) {
 497:         curl_close($ch);
 498:     }
 499:     $this->accessJwt = '';
 500:     $this->refreshJwt = '';
 501: 
 502:     return TRUE;
 503: }

セッション終了メソッド deleteSession は、引数はなく、セッション開始に成功したかどうかを戻り値にする。

解説:自分のプロフィール情報を取得

解説:ユーザーのDIDを取得する - PHPでBlueskyに投稿する」で自分のプロフィール情報を取得する方法を解説したが、後述するように、今回はアバター画像やバナー画像の Blob情報を取得する必要がある。そこで、PDSリポジトリから単一レコードを取り出すエンドポイントは com.atproto.repo.getRecord を利用し、追加で必要になる時分のプロフィール情報を取得する。パラメータを GET で渡し、応答を JSON形式データを受け取る REST API である。
com.atproto.repo.getRecord
URL
https://{PDSドメイン}/xrpc/app.bsky.actor.profile
入力パラメータ
フィールド名 要否 内  容
repo 必須 ハンドル名またはdid
collection 必須 レコードコレクションのNSID
"app.bsky.actor.profile"
rkey 必須 レコードキー
"self"
APIの戻り値は、httpステータスが200であれば成功、それ以外であればエラー情報が戻る。

応答データ(JSON形式)

{
    "uri": プロフィール情報のDID,
    "cid": プロフィール情報のCID,
    "value": {
        "$type": "app.bsky.actor.profile",
        "displayName": 表示名,
        "description": プロフィール情報(UTF-8),
        "avatar": {
            "$type": "blob",
            "ref": {
                "$link": アバター画像のBlob ID(CID)
            },
            "mimeType": アバター画像のMIME Type,
            "size": アバター画像のファイルサイズ
        },
        "banner": {
            "$type": "blob",
            "ref": {
                "$link": バナー画像のBlob ID(CID)
            },
            "mimeType": バナー画像のMIME Type,
            "size": バナー画像のファイルサイズ
        },
        "pinnedPost": {
            "cid": プロフィール固定メッセージ(CID),
            "uri": プロフィール固定メッセージ(DID),
---(中略)---
        },
    }
}
---(以下略)---
}

pahooBlueskyAPI.php

1438: /**
1439:  * 自分のプロフィール情報を取得する.
1440:  * @param   なし
1441:  * @return  array メッセージ情報 / FALSE:取得失敗
1442: */
1443: function getMyProfiles() {
1444:     $userName = $this->BLUESKY_HANDLE;
1445: 
1446:     // ユーザーDIDを取得する
1447:     $userDID = $this->getDID($userName);
1448:     if ($userDID == FALSE) {
1449:         $this->seterror($url . 'はユーザーDIDを取得できません');
1450:         return FALSE;
1451:     }
1452: 
1453:     // パラメータ配列を作成する
1454:     $params = [
1455:         'repo'          => $userDID,
1456:         'collection'    => 'app.bsky.actor.profile',
1457:         'rkey'          => 'self'
1458:     ];
1459: 
1460:     // トークンを取得する.
1461:     $this->getValidToken();
1462: 
1463:     // リクエストURL
1464:     $requestURL = 'https://' . $this->pds . '/xrpc/com.atproto.repo.getRecord?' . http_build_query($params);
1465:     $this->webapi = $requestURL;
1466:     // cURLを使ったリクエスト
1467:     $ch = curl_init($requestURL);
1468:     curl_setopt($ch, CURLOPT_CONNECTTIMEOUT, 10);
1469:     curl_setopt($ch, CURLOPT_HTTPHEADER, [
1470:         'Content-Type: application/json',
1471:         'Authorization: Bearer ' . $this->accessJwt,
1472:     ]);
1473:     curl_setopt($ch, CURLOPT_RETURNTRANSFER, TRUE);
1474:     curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, FALSE); // サーバ証明書検証をスキップ
1475:     curl_setopt($ch, CURLOPT_SSL_VERIFYHOST, FALSE); //  〃
1476: 
1477:     // レスポンス処理
1478:     $response = curl_exec($ch);
1479:     $httpStatusCode = curl_getinfo($ch, CURLINFO_RESPONSE_CODE);
1480:     if (PHP_VERSION_ID < 80500) {
1481:         curl_close($ch);
1482:     }
1483:     $items = json_decode($response, TRUE);
1484:     if ($httpStatusCode !200) {
1485:         $errmsg = '自分のプロフィール情報を取得できません(http code:' . $httpStatusCode . ')';
1486:         if (isset($items['message'])) {
1487:             $errmsg .';' . $items['message'];
1488:         }
1489:         $this->seterror($errmsg);
1490:         return FALSE;
1491:     }
1492: 
1493:     return $items;
1494: }

解説:自分のプロフィール情報を更新

Blueskyでは、自分のプロフィールを更新するAPIは明示的に存在しないが、前述の自分のプロフィール情報を取得するのに使ったエンドポイント com.atproto.repo.getRecord の逆の働きをする com.atproto.repo.putRecord を使って時分のプロフィール情報を更新できる。Bluesky APIの構造は、とても直交性が高いと感じた。このエンドポイントは、パラメータを PUT で渡し、応答を JSON形式データを受け取る REST API である。
com.atproto.repo.putRecord
URL
https://{PDSドメイン}/xrpc/com.atproto.repo.putRecord
入力パラメータ
フィールド名 要否 内  容
repo 必須 ハンドル名またはdid
collection 必須 レコードコレクションのNSID
"app.bsky.actor.profile"
record 必須 レコード情報
recordの構造
フィールド名 要否 内  容
$type 必須 "app.bsky.actor.profile"
displayName 必須 表示名(UTF-8)
description 必須 プロフィール情報(UTF-8)
avatar 必須 アバター画像Blob情報
com.atproto.repo.getRecord で取得したものと同じ構造であること
banner 必須 バナー画像Blob情報
com.atproto.repo.getRecord で取得したものと同じ構造であること
pinnedPost 必須 プロフィールに固定するメッセージ情報
com.atproto.repo.getRecord で取得したものと同じ構造であること
APIの戻り値は、httpステータスが200であれば成功、それ以外であればエラー情報が戻る。
record については、すべてが必須情報となる。空文字にしたものは、削除と同じ意味を持つ。
プロフィール更新メソッドは後述するが、あからじめ com.atproto.repo.getRecord を使って現時点のプロフィール情報を取得しておき、変更しない情報(パラメータにNULLをしていしたもの)については、com.atproto.repo.getRecord で得たデータをそのまま渡すという方針にする。

応答データ(JSON形式)

--- app.bsky.actor.profile と同じ ---

pahooBlueskyAPI.php

1496: /**
1497:  * 自分のプロフィール情報を更新する.
1498:  * @param   string $dispName    表示名(UTF-8)【NULL=更新しない】
1499:  * @param   string $description プロフィール(UTF-8)【NULL=更新しない】
1500:  * @param   string $avator      アバター画像ファイル名【NULL=更新しない】
1501:  * @param   string $banner      バナー画像ファイル名【NULL=更新しない】
1502:  * @param   string $pinnedPost  プロフィール固定メッセージURL【NULL=更新しない】
1503:  * @return  array メッセージ情報 / FALSE:更新失敗
1504: */
1505: function updateProfiles($dispName=NULL, $description=NULL, $avatar=NULL, $banner=NULL, $pinnedPost=NULL) {
1506:     $height = $width = 0;
1507:     $userName = $this->BLUESKY_HANDLE;
1508: 
1509:     // ユーザーDIDを取得する
1510:     $userDID = $this->getDID($userName);
1511:     if ($userDID == FALSE)      return FALSE;
1512: 
1513:     // ユーザー・プロファイル情報を取得
1514: //  $userProfiles = $this->getProfile($userName);
1515: //  if ($userProfiles == FALSE)     return FALSE;
1516: 
1517:     // 自分のプロフィール情報を取得する.
1518:     $myProfiles = $this->getMyProfiles($userName);
1519:     if ($myProfiles == FALSE)   return FALSE;
1520: 
1521:     // 表示名
1522:     if ($dispName == NULL) {
1523:         $dispName = $myProfiles['value']['displayName'];
1524:     }
1525:     // プロフィール
1526:     if ($description == NULL) {
1527:         $description = $myProfiles['value']['description'];
1528:     }
1529:     // アバター画像
1530:     if ($avatar == NULL) {
1531:         $avatar = $myProfiles['value']['avatar'];
1532:     } else {
1533:         $tempFile = tempnam(sys_get_temp_dir(), 'bsky_avatar_');
1534:         @file_put_contents($tempFile, file_get_contents($avatar));
1535:         $avatar = $this->uploadBlob($tempFile, $width, $height);
1536:         unlink($tempFile);
1537:         if ($avatar == FALSE)   return FALSE;
1538:     }
1539:     // バナー画像
1540:     if ($banner == NULL) {
1541:         $banner = $myProfiles['value']['banner'];
1542:     } else {
1543:         $tempFile = tempnam(sys_get_temp_dir(), 'bsky_banner_');
1544:         @file_put_contents($tempFile, file_get_contents($banner));
1545:         $banner = $this->uploadBlob($tempFile, $width, $height);
1546:         unlink($tempFile);
1547:         if ($banner == FALSE)   return FALSE;
1548:     }
1549:     // プロフィール固定
1550:     if ($pinnedPost == NULL) {
1551:         $pinnedPost = $myProfiles['value']['pinnedPost'];
1552:     } else {
1553:         // プロフィールに固定するURL情報を取得する
1554:         $items = $this->getPostThread($pinnedPost);
1555:         if ($items == FALSE)    return FALSE;
1556:         $pinnedPost = $items['thread']['post'];
1557:     }
1558: 
1559:     // POSTデータ配列を作成する
1560:     $record = [
1561:         '$type'         => 'app.bsky.actor.profile',
1562:         'displayName'   => $dispName,
1563:         'description'   => $description,
1564:         'avatar'        => $avatar,
1565:         'banner'        => $banner,
1566:         'pinnedPost'    => $pinnedPost
1567:     ];
1568:     // トークンを取得する.
1569:     $this->getValidToken();
1570: 
1571:     // リクエストURL
1572:     $requestURL = 'https://' . $this->pds . '/xrpc/com.atproto.repo.putRecord';
1573:     $this->webapi = $requestURL;
1574:     // cURLを使ったリクエスト
1575:     $ch = curl_init($requestURL);
1576:     curl_setopt($ch, CURLOPT_CONNECTTIMEOUT, 10);
1577:     curl_setopt($ch, CURLOPT_HTTPHEADER, [
1578:         'Content-Type: application/json',
1579:         'Authorization: Bearer ' . $this->accessJwt,
1580:     ]);
1581:     curl_setopt($ch, CURLOPT_RETURNTRANSFER, TRUE);
1582:     curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, FALSE); // サーバ証明書検証をスキップ
1583:     curl_setopt($ch, CURLOPT_SSL_VERIFYHOST, FALSE); //  〃
1584:     curl_setopt($ch, CURLOPT_POST, TRUE);
1585:     curl_setopt($ch, CURLOPT_POSTFIELDS, json_encode([
1586:         'repo'          => $userDID,
1587:         'collection'    => 'app.bsky.actor.profile',
1588:         'rkey'          => 'self',
1589:         'record'        => $record,
1590:     ]));
1591: 
1592:     // レスポンス処理
1593:     $response = curl_exec($ch);
1594:     $httpStatusCode = curl_getinfo($ch, CURLINFO_RESPONSE_CODE);
1595:     if (PHP_VERSION_ID < 80500) {
1596:         curl_close($ch);
1597:     }
1598:     $items = json_decode($response, TRUE);
1599:     if ($httpStatusCode !200) {
1600:         $errmsg = '自分のプロフィール情報を更新できません(http code:' . $httpStatusCode . ')';
1601:         if (isset($items['message'])) {
1602:             $errmsg .';' . $items['message'];
1603:         }
1604:         $this->seterror($errmsg);
1605:         return FALSE;
1606:     }
1607: 
1608:     return $items;
1609: }

解説:メイン・プログラムの初期値

updateProfilesBluesky.php

  63: // 初期値(START) =============================================================
  64: 
  65: // 表示幅(ピクセル)
  66: define('WIDTH', 600);
  67: 
  68: // アバター画像の表示幅(ピクセル)
  69: define('WIDTH_AVATAR', 64);
  70: 
  71: // 初期値(END) ===============================================================

メイン・プログラムの初期値は、「変更不可」の記載のないものは自由に変更できる。

解説:自分のプロフィール情報を取得する

updateProfilesBluesky.php

 307: /**
 308:  * 自分のプロフィール情報を取得する.
 309:  * @param   string $errmsg エラーメッセージを格納する変数
 310:  * @return  array プロフィール情報
 311:  *              ['displayName'] 表示名
 312:  *              ['description'] プロフィール情報
 313:  *              ['bannerURL']   バナー画像URL
 314:  *              ['avatarURL']   アバター画像URL
 315:  *          FALSE 取得失敗
 316: */
 317: function getMyProfiles(&$errmsg) {
 318:     $errmsg = '';
 319: 
 320:     // 自分のプロフィール情報を格納する配列
 321:     $myProfiles = array();
 322: 
 323:     // インスタンスを生成する.
 324:     $pbs = new pahooBlueskyAPI('bsky.social');
 325: 
 326:     // 自分のプロフィール情報を取得する.
 327:     $items = $pbs->getMyProfiles();
 328:     if ($items === FALSE) {
 329:         $errmsg = $pbs->geterror();
 330:         return FALSE;
 331:     }
 332:     $myProfiles['webapi'] = $pbs->webapi;
 333: 
 334:     // 表示名
 335:     $myProfiles['displayName'] = $items['value']['displayName'];
 336:     // プロフィール情報
 337:     $myProfiles['description'] = $items['value']['description'];
 338: 
 339:     // バナー画像URL
 340:     if (preg_match('/^image\/([a-z0-9]+)$/', $items['value']['banner']['mimeType'], $arr1) === 0) {
 341:         $errmsg = 'バナー画像フォーマットが不明';
 342:         return FALSE;
 343:     }
 344:     if (preg_match('/at\:\/\/(did\:plc\:[^\/]+\/)/', $items['uri'], $arr2) === 0) {
 345:         $errmsg = 'バナーATURIが不正';
 346:         return FALSE;
 347:     }
 348:     $path = 'https://cdn.bsky.app/img/banner/plain/' . $arr2[1];
 349:     $myProfiles['bannerURL'] = $path . $items['value']['banner']['ref']['$link'. '@' . $arr1[1];
 350: 
 351:     // アバター画像URL
 352:     if (preg_match('/^image\/([a-z0-9]+)$/', $items['value']['avatar']['mimeType'], $arr1) === 0) {
 353:         $errmsg = 'アバター画像フォーマットが不明';
 354:         return FALSE;
 355:     }
 356:     if (preg_match('/at\:\/\/(did\:plc\:[^\/]+\/)/', $items['uri'], $arr2) === 0) {
 357:         $errmsg = 'アバターATURIが不正';
 358:         return FALSE;
 359:     }
 360:     $path = 'https://cdn.bsky.app/img/avatar/plain/' . $arr2[1];
 361:     $myProfiles['avatarURL'] = $path . $items['value']['avatar']['ref']['$link'. '@' . $arr1[1];
 362: 
 363:     // プロフィール固定メッセージURL
 364:     if (isset($items['value']['pinnedPost'])) {
 365:         if (preg_match('/\/([^\/]+)$/', $items['value']['pinnedPost']['uri']) === 0) {
 366:             $errmsg = 'プロフィール固定メッセージURLが不正';
 367:             return FALSE;
 368:         }
 369:         $myProfiles['pinnedPostURL'] = $pbs->atruri2postURL($items['value']['pinnedPost']['uri']);
 370:     }
 371: 
 372:     // プロフィール固定メッセージ(埋め込みHTML)
 373:     $res = $pbs->getEmbedPosts($myProfiles['pinnedPostURL']);
 374:     if ($res === FALSE) {
 375:         $errmsg = $pbs->geterror();
 376:         return FALSE;
 377:     }
 378:     $myProfiles['pinnedPost'] = $res;
 379: 
 380:     // インスタンスを解放する.
 381:     $pbs = NULL;
 382: 
 383:     return $myProfiles;
 384: }

ユーザー関数 getMyProfiles は、前述のメソッド getMyProfiles を呼び出し、プロフィール情報(表示名、プロフィール情報、アバター画像、バナー画像、プロフィール固定メッセージ)を配列に格納して返す。
画像は ATURI形式で返るため、これをURLに置換して配列に代入する。
プロフィール固定メッセージも ATURI形式で返るため、これをURLに置換して配列に代入する。

解説:自分のプロフィール情報を取得する

updateProfilesBluesky.php

 117: .img-wrapper {
 118:     position: relative;
 119:     display: inline-block;
 120: }
 121: .img-wrapper img {
 122:     display: block;
 123:     width: 300px; /* 必要に応じて調整 */
 124:     height: auto;
 125: }
 126: .img-wrapper input[type="file"] {
 127:     position: absolute;
 128:     top: 0;
 129:     left: 0;
 130:     width: 100%;
 131:     height: 100%;
 132:     opacity: 0; /* 見えなくする */
 133:     cursor: pointer; /* ポインタ変更でクリック可能に */
 134:     pointer-events: all;
 135: }
 136: .overlay-label {
 137:     position: absolute;
 138:     top: 10px;
 139:     left: 10px;
 140:     background: rgba(0,0,0,0.5);
 141:     color: white;
 142:     padding: 4px 8px;
 143:     border-radius: 4px;
 144:     font-size: 14px;
 145:     pointer-events: none; /* クリック無効(下のinputが反応) */
 146: }
 147: </style>
 148: <script>
 149: 
 150: // ページロード直後の処理
 151: document.addEventListener('DOMContentLoaded', () => {
 152:     const dropBanner = document.getElementById('banner');
 153:     dropBanner.addEventListener('dragover', handleDragOver, false);
 154:     dropBanner.addEventListener('drop', handleFileSelectBanner, false);
 155:     const bannerFile = document.getElementById('bannerFile');
 156:     bannerFile.addEventListener('change', handleChangeImageBanner, false);
 157: 
 158:     const dropavatar = document.getElementById('avatar');
 159:     dropavatar.addEventListener('dragover', handleDragOver, false);
 160:     dropavatar.addEventListener('drop', handleFileSelectAvatar, false);
 161:     const avatarFile = document.getElementById('avatarFile');
 162:     avatarFile.addEventListener('change', handleChangeImageAvatar, false);
 163: });
 164: 
 165: /**
 166:  * 対象オブジェクトに画像ファイルをドラッグオーバーしたときの処理
 167:  * @param   object evt 対象オブジェクトのID
 168:  * @return  なし
 169: */
 170: function handleDragOver(evt) {
 171:     evt.stopPropagation();
 172:     evt.preventDefault();
 173:     evt.dataTransfer.dropEffect = 'copy';
 174: }
 175: 
 176: /**
 177:  * 対象オブジェクトに画像ファイルをドロップしたときの処理(バナー画像)
 178:  * @param   object evt 対象オブジェクトのID
 179:  * @return  なし
 180: */
 181: function handleFileSelectBanner(evt) {
 182:     evt.stopPropagation();
 183:     evt.preventDefault();
 184: 
 185:     var files = evt.dataTransfer.files; 
 186:     var output = [];
 187: 
 188:     document.getElementById('bannerFile').files = files;
 189: 
 190:     // 表示画像を変更する
 191:     const file = this.files[0];
 192:     if (file && file.type.startsWith('image/')) {
 193:         const reader = new FileReader();
 194:         reader.onload = function(e) {
 195:             banner.src = e.target.result;
 196:         };
 197:         reader.readAsDataURL(file);
 198:     } else {
 199:         alert('画像ファイルを選んでください');
 200:     }
 201: }
 202: 
 203: /**
 204:  * 対象オブジェクトに画像ファイルをドロップしたときの処理(アバター画像)
 205:  * @param   object evt 対象オブジェクトのID
 206:  * @return  なし
 207: */
 208: function handleFileSelectAvatar(evt) {
 209:     evt.stopPropagation();
 210:     evt.preventDefault();
 211: 
 212:     var files = evt.dataTransfer.files; 
 213:     var output = [];
 214: 
 215:     document.getElementById('avatarFile').files = files;
 216: 
 217:     // 表示画像を変更する
 218:     const file = this.files[0];
 219:     if (file && file.type.startsWith('image/')) {
 220:         const reader = new FileReader();
 221:         reader.onload = function(e) {
 222:             avatar.src = e.target.result;
 223:         };
 224:         reader.readAsDataURL(file);
 225:     } else {
 226:         alert('画像ファイルを選んでください');
 227:     }
 228: }
 229: 
 230: /**
 231:  * バナー画像を変更する
 232:  * @param   なし
 233:  * @return  なし
 234: */
 235: function handleChangeImageBanner() {
 236:     const file = this.files[0];
 237:     if (file && file.type.startsWith('image/')) {
 238:         const reader = new FileReader();
 239:         reader.onload = function(e) {
 240:             banner.src = e.target.result;
 241:         };
 242:         reader.readAsDataURL(file);
 243:     } else {
 244:         alert('画像ファイルを選んでください');
 245:     }
 246: }
 247: 
 248: /**
 249:  * アバター画像を変更する
 250:  * @param   なし
 251:  * @return  なし
 252: */
 253: function handleChangeImageAvatar() {
 254:     const file = this.files[0];
 255:     if (file && file.type.startsWith('image/')) {
 256:         const reader = new FileReader();
 257:         reader.onload = function(e) {
 258:             avatar.src = e.target.result;
 259:         };
 260:         reader.readAsDataURL(file);
 261:     } else {
 262:         alert('画像ファイルを選んでください');
 263:     }
 264: }
 265: </script>
 266: </head>

Blusky Webアプリのプロフィール変更UIは、Twitter(現・X)に比べて貧弱だ。ここでは、画像ファイルを画面に表示されている画像にドラッグ&ドロップすることで更新したり、ボタンをクリックしてファイルダイアログを表示して選択する2つの方法をJavaScriptで実装している。ドラッグ&ドロップもしくは選択した画像ファイルは、サーバ通信せずに、JavaScriptだけで画面表示を更新する。
いずれの場合も input type="file" オブジェクトに更新後のファイルを格納するようにしてある。

解説:自分のプロフィール情報を更新する

updateProfilesBluesky.php

 386: /**
 387:  * 自分のプロフィール情報を更新する.
 388:  * @param   string $errmsg エラーメッセージを格納する変数
 389:  * @return  array メッセージ情報 / FALSE:更新失敗
 390: */
 391: function updateMyProfiles(&$errmsg) {
 392:     $errmsg = '';
 393: 
 394:     // 表示名
 395:     $dispName = trim((string)getParam('displayName', TRUE, NULL));
 396: 
 397:     // プロフィール情報
 398:     $description = trim((string)getParam('description', TRUE, NULL));
 399: 
 400:     // アバター画像
 401:     if (isset($_FILES['avatarFile']) && $_FILES['avatarFile']['tmp_name'!== '') {
 402:         $avatar = $_FILES['avatarFile']['tmp_name'];
 403:     } else {
 404:         $avatar = NULL;
 405:     }
 406: 
 407:     // バナー画像
 408:     if (isset($_FILES['bannerFile']) && $_FILES['bannerFile']['tmp_name'!== '') {
 409:         $banner = $_FILES['bannerFile']['tmp_name'];
 410:     } else {
 411:         $banner = NULL;
 412:     }
 413: 
 414:     // プロフィール固定メッセージURL
 415:     $pinnedPost = trim((string)getParam('pinnedPostURL', TRUE, NULL));
 416: 
 417:     // インスタンスを生成する.
 418:     $pbs = new pahooBlueskyAPI('bsky.social');
 419: 
 420:     // 自分のプロフィール情報を更新する.
 421:     $res = $pbs->updateProfiles($dispName, $description, $avatar, $banner, $pinnedPost);
 422: 
 423:     // インスタンスを解放する.
 424:     $pbs = NULL;
 425: 
 426:     return $res;
 427: }

ユーザー関数 updateMyProfiles は、前述のメソッド getMyProfiles を呼び出し、POST渡しされたプロフィール情報(表示名、プロフィール情報、アバター画像、バナー画像、プロフィール固定メッセージ)を使って更新する。更新のない情報については、NULLを代入することで、更新しない。

参考サイト

(この項おわり)
header