2025/12/07

自作ActivityPub実装に投稿埋め込み機能を実装した

本稿はFediverse Advent Calendar 2025の7日目の記事である。僕は自作のActivityPub実装を運用している。詳細は左記のリンク先に譲るとして、簡単に説明するとRails 8とHotwireで構築された一人専用の実装系だ。内蔵クライアントを持たず、フロントエンドは閲覧にのみ対応している。

フロントエンド部分はリアルなマイクロブログとして機能させるためにLikeやRTに相当するアクティビティは可視化せず、代わりにRSSを通じて投稿を取得できる。外からは昔の質素な一言ブログのように見えるが、内部では共通の通信規格で交信を行っている分散型SNSソフトウェアと捉えてもらうと話が早い。

このようなミニマルな構成に手馴染みの良さを感じている一方、リッチな既存の実装系と比べるとやはり機能の不足は否めない。同種のソフトウェアであるMastodonやMisskeyには非常に多くの外部向け機能が備わっており、僕にとってその最たるものが投稿埋め込み機能であった。ブロガーにとってこれは、とりわけ重要なアイテムと言っても過言ではない。(以下はMastodonの例)

Post by @[email protected]
View on Mastodon

投稿埋め込み機能があれば追加の説明なしにコンテキストを挿入できるし、一度投稿した画像も使い回せる。フレームによって境界が区切られているのでアイキャッチ的にも使える。アカウントの存在が周知されてフォロワーの増加にも繋がる。まさに良いことづくめの実にすばらしい機能なのだ。

そこで、自作ActivityPub実装に本機能を実装した。自分ひとりだけが利用する実装系であり、基本的に自分以外の投稿が埋め込まれる可能性を考慮しなくても良かったので、通常のiframe利用と比べて警戒すべき点はそう多くなかったように思われる。以下、解説を行う。

設計方針

MastodonはJSON APIを介してReactで埋め込み部分をレンダリングしていたが、僕の実装系はHotwireベースなのでより簡潔にまとめられると考えた。具体的にはサーバ側でERBテンプレートをレンダリングし、バニラJSでシンプルにiframeを制御する。

 1// embed.js
 2(function() {
 3  'use strict';
 4
 5  const embeds = new Map();
 6
 7  function generateId() {
 8    return Math.random().toString(36).substr(2, 9);
 9  }
10
11  function init() {
12    document.querySelectorAll('div.letter-embed').forEach(function(container) {
13      const embedUrl = container.getAttribute('data-embed-url');
14      if (!embedUrl) return;
15
16      const id = generateId();
17      const iframe = document.createElement('iframe');
18
19      iframe.src = embedUrl;
20      iframe.width = '100%';
21      iframe.height = '400';
22      iframe.style.border = 'none';
23      iframe.style.overflow = 'hidden';
24      iframe.style.display = 'block';
25      iframe.sandbox =
26        'allow-scripts allow-same-origin allow-popups allow-popups-to-escape-sandbox';
27      iframe.setAttribute('loading', 'lazy');
28      iframe.setAttribute('scrolling', 'no');
29
30      embeds.set(id, iframe);
31
32      iframe.onload = function() {
33        iframe.contentWindow.postMessage(
34          {
35            type: 'setHeight',
36            id: id
37          },
38          '*'
39        );
40      };
41
42      container.innerHTML = '';
43      container.appendChild(iframe);
44
45      container.style.margin = '0';
46      container.style.padding = '0';
47      container.style.border = 'none';
48      container.style.background = 'none';
49      container.style.overflow = 'hidden';
50    });
51  }
52
53  window.addEventListener('message', function(e) {
54    if (e.data && e.data.type === 'setHeight' && e.data.height) {
55      embeds.forEach(function(iframe) {
56        if (iframe.contentWindow === e.source) {
57          iframe.height = e.data.height;
58        }
59      });
60    }
61  });
62
63  if (document.readyState === 'loading') {
64    document.addEventListener('DOMContentLoaded', init);
65  } else {
66    init();
67  }
68})();

また、画面デザインも特殊な装飾は一切行わずにフロントエンドのビューをほぼ再利用する。Mastodonをはじめ多くのActivityPub実装は、利用者を増やそうとするモチベーションが存在しているために華美なロゴやスタイリング、キャッチコピーなどが表示されていることが多い。

対して、僕の実装系はごく個人的なプロジェクトで目的意識がないゆえ簡素なデザインで済む。元のフロントエンド部分と同じくLikeやRTに相当する表示領域もいらない。ページ上で本文を隔てる境界として機能しつつ、それでいて挿入される場所の世界観を壊さない。そのような形が望ましい。

実装内容

巷に蔓延る埋め込み機能には特定のサイズを超えるコンテンツ(画像や長文など)に対応しきれず、そのまま見切れたり、逆にはみ出したり、スクロールバーを露出させる代物が少なくない。破綻した境界は世界観を壊す。埋め込み機能が美しくあるには自己の領分を押し留める努力をしなければならない。

 1// embedded.html.erb
 2    (function () {
 3      let lastHeight = 0;
 4
 5      function notifyHeight() {
 6        const article = document.querySelector('article');
 7        const height = article ? article.offsetHeight + 1 : Math.max(
 8          document.documentElement.scrollHeight,
 9          document.body.scrollHeight,
10        );
11
12        if (height !== lastHeight) {
13          lastHeight = height;
14          window.parent.postMessage(
15            {
16              type: "setHeight",
17              height: height,
18            },
19            "*",
20          );
21        }
22      }
23
24      window.addEventListener("load", notifyHeight);
25
26      const observer = new MutationObserver(notifyHeight);
27      observer.observe(document.body, {
28        childList: true,
29        subtree: true,
30      });
31
32      if ("ResizeObserver" in window) {
33        const resizeObserver = new ResizeObserver(notifyHeight);
34        resizeObserver.observe(document.body);
35      }
36
37      const mediaElements = document.querySelectorAll('img, video, iframe');
38      mediaElements.forEach(function(el) {
39        el.addEventListener('load', notifyHeight);
40        el.addEventListener('error', notifyHeight);
41      });
42
43      let checkCount = 0;
44      const intervalId = setInterval(function() {
45        notifyHeight();
46        checkCount++;
47        if (checkCount >= 10) {
48          clearInterval(intervalId);
49        }
50      }, 500);
51    })();

本機能の実装では、上記の通りpostMessageで親ウインドウに高さを通知し、MutationObserverResizeObserverで動的なリサイズに対応した。画像の読み込みも監視して、全体のサイズが確定してから描写を開始する仕組みだ。特定の解像度での見切れを防止するために1pxの余白も追加している。大抵の環境ではうまく働いてくれると思う。

フロントエンド側には埋め込みコードのコピーボタンを設置してある。任意の投稿の時刻表示から詳細画面に遷移して、右下のクリップボードアイコンをクリックすると埋め込みコードを取得できる。こうした便利アイコンをあえてまとめずに横に並べていくと、そのうち昔のマッキントッシュのようなレトロ感を出せると期待している。

今のところ、皆さんは僕のSNSの投稿を任意のWebページ上に埋め込むことができる。自作ActivityPub実装の投稿が、さらに自作の機能によってどこかに埋め込まれているのを見るのはなかなかの誉れである。自分の言葉を伝播せしめている実感がある。じゃんじゃん埋め込んでもらって構わない。特定の条件下で機能しないなどのフィードバックも随時募集している。

トラブルシューティング

Cloudflare絡みでいくつか引っかかった点があったので共有したい。まず、投稿埋め込み機能を実装する都合上、Rails側でX-Frame-Optionsを許可しなければならないが、Cloudflare側でも同様のルールを作成しておかないと弾かれてしまう。Cloudflareのトップページからドメイン → ルール → 概要に進んで「レスポンスヘッダー変換ルール」から以下の要領で作成する。

次に、投稿埋め込み機能のJavaScriptがキャッシングされると更新しても古いスクリプトが動き続けてしまうので、同じく「キャッシュルール」から以下の要領で/embed.jsをキャッシュの例外に設定する。以上で問題が解決される。

まとめ

実装のハードルが高いと聞いていた投稿埋め込み機能であったが、主に自分しか使わないActivityPub実装であったこと、求める機能性やデザインが最小限であったこと、あとClaude Sonnet 4.5が頑張ってくれたことなどによって比較的簡単に実装できた。

©2011 辻谷陸王 | Fediverse | Keyoxide | RSS | 小説