(2025年8月17日)画像に余計な空白が入らないようにするため一部仕様変更.
(2025年8月14日).pahooEnv導入
目次
サンプル・プログラムの実行例
サンプル・プログラム
| updateProfilesBluesky.php | サンプル・プログラム本体 |
| .pahooEnv | クラウドサービスを利用するためのアカウント情報などを記入する .env ファイル。 使い方は「各種クラウド連携サービス(WebAPI)の登録方法」を参照。include_path が通ったディレクトリに配置すること。 |
| pahooInputData.php | データ入力に関わる関数群。 使い方は「数値入力とバリデーション」「文字入力とバリデーション」などを参照。include_path が通ったディレクトリに配置すること。 |
| pahooBlueskyAPI.php | Bluesky APIに関わるクラス pahooBlueskyAPI。 使い方は「PHPでPHPでBlueskyに投稿する」などを参照。include_path が通ったディレクトリに配置すること。 |
| pahooScraping.php | スクレイピング処理に関わるクラス pahooScraping。 スクレイピング処理に関わるクラスの使い方は「PHPでDOMDocumentを使ってスクレイピング」を参照。include_path が通ったディレクトリに配置すること。 |
| バージョン | 更新日 | 内容 |
|---|---|---|
| 1.6.0 | 2025/08/14 | .pahooEnv導入 |
| 1.0.0 | 2025/07/19 | 初版 |
| バージョン | 更新日 | 内容 |
|---|---|---|
| 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がないページに対応 |
| バージョン | 更新日 | 内容 |
|---|---|---|
| 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 | 初版 |
| バージョン | 更新日 | 内容 |
|---|---|---|
| 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対応
Windowsでは、"php.ini" の下記の行を有効化する。
extension=php_openssl.dllLinuxでは --with-openssl=/usr オプションを付けて再ビルドする。→OpenSSLインストール手順
これで準備は完了だ。
準備:pahooInputData 関数群
また、各種クラウドサービスに登録したときに取得するアカウント情報、アプリケーションパスワードなどを登録した .pahooEnv ファイルから読み込む関数 pahooLoadEnv を備えている。こちらについては、「各種クラウド連携サービス(WebAPI)の登録方法」をご覧いただきたい。
解説:pahooBlueskyAPIクラス
Bluesky APIを利用するメソッドはクラス "pahooBlueskyAPI.php" に分離している。また、このクラスからクラス "pahooScraping.php" を呼び出すので、2つのクラス・ファイルを include_path の通ったディレクトリに配置すること。
解説:セッション開始
| URL |
|---|
| https://{PDSドメイン}/xrpc/com.atproto.server.createSession |
アクセストークン 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 = ''; // アプリケーション・パスワード
上述の手順で取得したアプリケーション・パスワードをプロパティ変数 $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: }
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: }
メソッドの中身は、上述のAPI仕様の通りに作った。
POSTプロトコルとして、これまでのクラウドサービス利用でも使ってきた cURL関数を利用する。
解説:セッション終了
| URL |
|---|
| https://{PDSドメイン}/xrpc/com.atproto.server.deleteSession |
アクセストークン 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: }
解説:自分のプロフィール情報を取得
| URL |
|---|
| https://{PDSドメイン}/xrpc/app.bsky.actor.profile |
| フィールド名 | 要否 | 内 容 |
|---|---|---|
| repo | 必須 | ハンドル名またはdid |
| collection | 必須 | レコードコレクションのNSID "app.bsky.actor.profile" |
| rkey | 必須 | レコードキー "self" |
応答データ(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: }
解説:自分のプロフィール情報を更新
| URL |
|---|
| https://{PDSドメイン}/xrpc/com.atproto.repo.putRecord |
| フィールド名 | 要否 | 内 容 |
|---|---|---|
| repo | 必須 | ハンドル名またはdid |
| collection | 必須 | レコードコレクションのNSID "app.bsky.actor.profile" |
| 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 で取得したものと同じ構造であること |
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
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: }
画像は 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>
いずれの場合も 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: }
参考サイト
- Bluesky 公式リファレンス
- Bluesky API - 各種クラウド連携サービス(WebAPI)の登録方法
- PHPでBlueskyに投稿する:ぱふぅ家のホームページ
- PHPでBlueskyの埋め込み用HTMLを取得:ぱふぅ家のホームページ
- PHPでBlueskyの投稿を検索・一覧表示する:ぱふぅ家のホームページ

- 表示名
- プロフィール情報
- アバター画像
- バナー画像
- プロフィール固定メッセージ
各々の項目は独立して省略することが可能で(パラメータにNULLを代入したときに省略とみなす)、省略時には現在登録されている情報を更新せずに残すことにする。Blusky Webアプリのプロフィール変更UIは、Twitter(現・X)に比べて貧弱だ。ここでは、画像ファイルを画面に表示されている画像にドラッグ&ドロップすることで更新したり、ボタンをクリックしてファイルダイアログを表示して選択する 2つの方法を実装する。ドラッグ&ドロップもしくは選択した画像ファイルは、サーバ通信せずに、JavaScriptだけで画面表示を更新する。