ActivityPub 笔记
ActivityPub 协议是一个去中心化的社交网络协议,其基于 ActivityStreams 2.0 数据格式。
在 ActivityPub 协议里,一个用户在服务器上的角色为“参与者(actor)”。用户在 不同的服务器上的角色为不同的“参与者”。每一名“参与者”有:
- 一个收件箱:用于接收消息
- 一个发件箱:用于发送消息
例如,我在实例 A 上注册成为
251@example.com
,而在另外一个实例 B 上注册成为251@mastodon.art
,虽然都是我一个人注册的,但因为在不同的实例上注册,故为不同的参与者。我分别在实例 A 与实例 B 都有一个收件箱和发件箱。
而这收件箱和发件箱其实就是在 ActivityPub 参与者的 ActivityStreams 里的一个 URL。
{"@context": "https://www.w3.org/ns/activitystreams",
"type": "Person",
"id": "https://example.com/251/",
"name": "Kiryuu Sakuya",
"preferredUsername": "misaka00251",
"summary": "FAQ is love",
"inbox": "https://example.com/251/inbox/",
"outbox": "https://example.com/251/outbox/",
"followers": "https://example.com/251/followers/",
"following": "https://example.com/251/following/",
"liked": "https://example.com/251/liked/"}
正如前文说过,ActivityPub 基于 ActivityStreams 数据格式。而后者基本上概括了所有社交网络需要的活动,例如上面例子里的“粉丝”、“正在关注”、“喜欢的内容”。就算没有概括,它使用一种基于 JSON 的格式 JSON-LD。其目的既是为了可读,又方便机器处理,甚至如果你要加东西进去也很简单。
假设我现在要和 nong 哥互相隔空喊话,ActivityPub 如何帮我们做到这个?W3C 规范里有一张图:
简单来说就是:
- 用户可以使用 POST 操作来将消息发送至别人的收件箱(服务端对服 务端,仅在接入网络的情况下)
- 用户可以使用 GET 操作来从自己的收件箱获取消息(客户端对服务 端,类似于读取你的社交网络数据流)
- 用户可以使用 POST 操作来向发件箱发送消息(客户端对服务端)
- 用户可以使用 GET 操作在对方允许的情况下来向好友的发件箱获取 消息(客户端对服务端或者服务端对服务端)
我们在前文说过了,我在实例 A 上作为参与者,有一个收件箱和发件箱。那么,我首先要发送一个 ActivityStreams 对象(object):
{"@context": "https://www.w3.org/ns/activitystreams",
"type": "Note",
"to": ["https://example.org/anong/"],
"attributedTo": "https://example.com/251/",
"content": "nong 哥,柯文哲說要競選總統了,妳怎麼看:https://youtu.be/072tU1tamd0"}
能看得出来,这个对象(object)是要给 nong 哥的。接着,我把这个内容(object)发送(POST)到我的发件箱。
因为这是并不是一个活动(non-activity,其 type 为 note)对象(object),实例 A 意识到这是一个新被创建的对象(object),并创建一个活动(activity,其 type 为 create):
W3C 规定 ActivityStreams 有八大核心 type,以及用于社交网络而延伸的三组 type,总共四组。 – 活动内容(Activity objects),例如 Follow 或者 Announce。一个活动是参与者所采取的举措(action)。 – 参与者内容(Actor objects),例如 Person 或者 Group。参与者通常进行创造内容以及关注等活动。 – 对象类型(Object types),例如 Note 或者 Video。这些对象类型通常代表着某种内容。 – 集合内容(Collection objects)。这些是其他对象的分组或者列表,例如某个人的粉丝列表或者一串公共帖子的列表。
虽然 create 是活动内容里面的,但 note 不属于,这么理解即可。
{"@context": "https://www.w3.org/ns/activitystreams",
"type": "Create",
"id": "https://example.com/251/posts/9f905da5-ce8e-4b9e-a76a-8ef4b1dacc98",
"to": ["https://example.org/anong/"],
"actor": "https://example.com/251/",
"object": {"type": "Note",
"id": "https://example.com/251/posts/9f905da5-ce8e-4b9e-a76a-8ef4b1dacc98",
"attributedTo": "https://example.com/251/",
"to": ["https://example.org/anong/"],
"content": "nong 哥,柯文哲說要競選總統了,妳怎麼看:https://youtu.be/072tU1tamd0"}}
然后,实例 A 就通过 ActivityStreams 查看 nong 哥 的”参与者”(actor object),找到他的收件箱,然后发送(POST)创建的这个对象(object)给他。
技术上来说,这分了两个步骤,一个是客户端到服务器的通讯,一个是服务器与服务器之间的通讯(组成联盟)。这个例子里我们都用了,所以我们可以把整个过程抽象的理解成是我的发件箱 –> nong 哥的发件箱。
过了一会,我想看他收到了什么新消息,于是他的客户端开始获取(GET)操作,除了其他人发的沙雕图片和视频以外,还有这样一条消息:
{"@context": "https://www.w3.org/ns/activitystreams",
"type": "Create",
"id": "https://example.org/anong/p/114514",
"to": ["https://example.com/251/"],
"actor": "https://example.org/anong/",
"object": {"type": "Note",
"id": "https://example.org/anong/p/114513",
"attributedTo": "https://chatty.example/ben/",
"to": ["https://example.com/251/"],
"inReplyTo": "https://example.com/251/posts/9f905da5-ce8e-4b9e-a76a-8ef4b1dacc98",
"content": "<p>幹 還敢釣R</p>"}}
我收到消息之后,喜欢了这个 post:
{"@context": "https://www.w3.org/ns/activitystreams",
"type": "Like",
"id": "https://example.com/251/posts/e6dc43cb-f95e-4a26-8e41-ea9a5e77d0f8",
"to": ["https://example.org/anong/"],
"actor": "https://example.com/251/",
"object": "https://example.org/anong/p/114514"}
当然,我把这个对象(object)发送(POST)到我了的发件箱。因为这是一个活动(activity, 其 type 为 Like),实例 A 就不需要重新包装,可以直接发送出去。
Like 是属于 Activity Types 里面的。
最后,我给我的粉丝们发送了一条公共消息。因为是公共消息,所以不仅粉丝们可以收到,任何人原则上也可以看到。
{"@context": "https://www.w3.org/ns/activitystreams",
"type": "Create",
"id": "https://example.com/251/posts/30c68a4e-f3a8-483a-ae39-2618fcfc31fb",
"to": ["https://example.com/251/followers/",
"https://www.w3.org/ns/activitystreams#Public"],
"actor": "https://social.example/alyssa/",
"object": {"type": "Note",
"id": "https://example.com/251/posts/30c68a4e-f3a8-483a-ae39-2618fcfc31fb",
"attributedTo": "https://example.com/251/",
"to": ["https://example.com/251/followers/",
"https://www.w3.org/ns/activitystreams#Public"],
"content": "還是被發現了,哭哭"}}
以上就是一个很简单的例子了。我们通常说,某人(Somebody)对某个对象(object)做了什么(something);在 ActivityPub 里则是说是某个参与者(actor)在对一个对象(object)执行一个活动(activity)。
参考资料
- https://www.w3.org/TR/activitypub/
- https://www.w3.org/TR/activitystreams-core/
- https://www.w3.org/TR/activitystreams-vocabulary/
如果喜欢本文,欢迎点击下方的「鼓掌」按钮!
如果上面没有加载出任何东西,可以点击这里。
如果喜欢本文,欢迎点击下方的「鼓掌」按钮!
如果上面没有加载出任何东西,可以点击这里。