This is a work-in-progress document describing Mastodon’s proposed way of representing quote posts, users’ choices regarding whether their posts can be quoted and by whom (quote policies), and a mechanism for servers to verify compliance with such policies.
This document describes protocol considerations, which do not necessarily translate directly to User Experience considerations. For instance, the use of the approval mechanism described in this document does not imply that the user’s approval is manual.
The key words “MUST”, “MUST NOT”, “REQUIRED”, “SHALL”, “SHALL NOT”, “SHOULD”, “SHOULD NOT”, “RECOMMENDED”, “MAY”, and “OPTIONAL” in this specification are to be interpreted as described in [RFC-2119].
In the remaining of this document, “quoted object” refers to the object being quoted, “original author” to its author, and “quote post” refers to the object quoting the “quoted object”.
(This section is non-normative.)
This proposal has benefitted from significant discussions on SocialHub as well as discussions with trwnh and GoToSocial developers. In fact, the interactionPolicy
vocabulary directly comes from GoToSocial’s interaction policies which have since evolved along the current proposal.
A “quote post” is represented as an object with a quote
(https://w3id.org/fep/044f#quote
) attribute.
{
"@context": [
"https://www.w3.org/ns/activitystreams",
{
"quote": {
"@id": "https://w3id.org/fep/044f#quote",
"@type": "@id"
}
}
],
"type": "Note",
"id": "https://example.com/users/bob/statuses/1",
"attributedTo": "https://example.com/users/bob",
"to": [
"https://www.w3.org/ns/activitystreams#Public",
"https://example.com/users/alice"
],
"content": "I am quoting alice's post<span class=\"quote-inline\"><br/>RE: <a href=\"https://example.com/users/alice/statuses/1\">https://example.com/users/alice/statuses/1</a></span>",
"quote": "https://example.com/users/alice/statuses/1"
}
This example is non-normative, and the <span class=\"quote-inline\"><br/>RE: <a href=\"https://example.com/users/alice/statuses/1\">https://example.com/users/alice/statuses/1</a></span>
part of the content is an example of textual fallback, but does not otherwise carry meaning. In particular, it does not influence where the embedded quote should be displayed.
(This section is non-normative.)
While this FEP introduces https://w3id.org/fep/044f#quote
, there are competing definitions for the representation of quote posts:
_misskey_quote
(https://misskey-hub.net/ns/#_misskey_quote
)quoteUrl
(https://www.w3.org/ns/activitystreams#quoteUrl
)quoteUri
(http://fedibird.com/ns#quoteUri
)https://misskey-hub.net/ns/#_misskey_quote
rel
valueWe believe each of those to have significant drawbacks, such as re-using a namespace that has no definition for them, implying the value is an URL or URI, or using an unusual naming scheme, and none of them are linked to a control mechanism like the one defined in this FEP, hence why we introduced https://w3id.org/fep/044f#quote
.
That being said, we suggest some of them as fallback for compatibility with existing fediverse software implementations.
{
"@context": [
"https://www.w3.org/ns/activitystreams",
{
"quoteUrl": "as:quoteUrl",
"quoteUri": "http://fedibird.com/ns#quoteUri",
"_misskey_quote": "https://misskey-hub.net/ns/#_misskey_quote",
"quote": {
"@id": "https://w3id.org/fep/044f#quote",
"@type": "@id"
}
}
],
"type": "Note",
"id": "https://example.com/users/bob/statuses/1",
"attributedTo": "https://example.com/users/bob",
"to": [
"https://www.w3.org/ns/activitystreams#Public",
"https://example.com/users/alice"
],
"content": "I am quoting alice's post<span class=\"quote-inline\"><br/>RE: <a href=\"https://example.com/users/alice/statuses/1\">https://example.com/users/alice/statuses/1</a></span>",
"quote": "https://example.com/users/alice/statuses/1",
"quoteUrl": "https://example.com/users/alice/statuses/1",
"quoteUri": "https://example.com/users/alice/statuses/1",
"_misskey_quote": "https://example.com/users/alice/statuses/1",
"tag": [
{
"type": "Link",
"mediaType": "application/ld+json; profile=\"https://www.w3.org/ns/activitystreams\"",
"rel": "https://misskey-hub.net/ns#_misskey_quote",
"href": "https://example.com/users/alice/statuses/1"
}
]
}
Users may not want their posts to be quoted, or not by everyone. To allow users to express that, we re-use GoToSocial’s interaction policies and define a canQuote
sub-policy.
Each quotable object advertises an interactionPolicy
(https://gotosocial.org/ns#interactionPolicy
) with a canQuote
(https://gotosocial.org/ns#canQuote
) sub-policy.
A sub-policy is defined by two attributes:
automaticApproval
(https://gotosocial.org/ns#automaticApproval
): an array of Actor
and Collection
of Actor
objects from whom interactions are expected to be automatically approvedmanualApproval
(https://gotosocial.org/ns#manualApproval
): an array of Actor
and Collection
of Actor
objects from whom interactions are subject to manual reviewInteractions from actors that are neither in automaticApproval
nor manualApproval
are expected to never be approved.
To advertise a policy of disallowing all quotes, interactionPolicy.canQuote.automaticApproval
SHOULD contain the object author’s identifier as its single value. This is because an empty array is equivalent to a missing property under JSON-LD canonicalization.
automaticApproval
and manualApproval
SHOULD be restricted to individual actors, the special public collection https://www.w3.org/ns/activitystreams#Public
, the author’s followers
collection, and the author’s following
collection.
Note that the policy is entirely advisory. It SHOULD be used to provide user interface hints such as enabling a “Quote” button or explaining why an object cannot be quoted, but it MUST NOT be used to verify whether a quote post is valid. See later sections for the actual verification mechanism.
{
"@context": [
"https://www.w3.org/ns/activitystreams",
{
"gts": "https://gotosocial.org/ns#",
"interactionPolicy": {
"@id": "gts:interactionPolicy",
"@type": "@id"
},
"canQuote": {
"@id": "gts:canQuote",
"@type": "@id"
},
"automaticApproval": {
"@id": "gts:automaticApproval",
"@type": "@id"
}
}
],
"interactionPolicy": {
"canQuote": {
"automaticApproval": "https://example.com/users/alice/followers"
}
},
"type": "Note",
"id": "https://example.com/users/alice/statuses/1",
"attributedTo": "https://example.com/users/alice",
"to": "https://www.w3.org/ns/activitystreams#Public",
"content": "I allow my followers to quote this post"
}
In order to enforce a policy, we rely on approval stamps, a mechanism used to tell third-party servers that a quote is approved, regardless of the current state of the policy.
Quote approval stamps are objects of the type QuoteAuthorization
(https://w3id.org/fep/044f#QuoteAuthorization
), with interactingObject
(https://gotosocial.org/ns#interactingObject
), interactionTarget
(https://gotosocial.org/ns#interactionTarget
) and attributedTo
attributes.
The interactingObject
attribute MUST reference the accepted quote post, the interactionTarget
attribute MUST reference the quoted object, and the attributedTo
attribute MUST correspond to the author of the quoted object.
A QuoteAuthorization
object MUST be dereferenceable by all parties allowed to see the original post, and MAY be publicly dereferenceable. It MUST NOT embed its interactingObject
as to avoid possible information leaks. For the same reason, it MUST NOT embed its interactionTarget
object if the server is unable to verify that the party dereferencing the object has permission to see the quoted object.
When a third-party attempts to dereference the QuoteAuthorization
, the interactionTarget
MAY be inlined if the third-party has permission to access the quoted object. This is so that the third-party does not have to perform a second request to access the quoted object.
QuoteAuthorization
The following stamp can be used to prove that actor https://example.com/users/alice
has accepted https://example.org/users/bob/statuses/1
as a quote of her post https://example.com/users/alice/statuses/1
:
{
"@context": [
"https://www.w3.org/ns/activitystreams",
{
"QuoteAuthorization": "https://w3id.org/fep/044f#QuoteAuthorization",
"gts": "https://gotosocial.org/ns#",
"interactingObject": {
"@id": "gts:interactingObject",
"@type": "@id"
},
"interactionTarget": {
"@id": "gts:interactionTarget",
"@type": "@id"
}
}
],
"type": "QuoteAuthorization",
"id": "https://example.com/users/alice/stamps/1",
"attributedTo": "https://example.com/users/alice",
"interactingObject": "https://example.org/users/bob/statuses/1",
"interactionTarget": "https://example.com/users/alice/statuses/1"
}
QuoteAuthorization
To be considered valid for a particular quote post, a QuoteAuthorization
MUST satisfy the following properties:
interactingObject
is the quote post under considerationinteractionTarget
property is the quoted objectattributedTo
property is the author of its interactionTarget
QuoteAuthorization
object can be assertedQuoteAuthorization
An approval stamp can be revoked by Delete
ing the stamp.
QuoteRequest
request activityThe QuoteRequest
(https://w3id.org/fep/044f#QuoteRequest
) activity type is introduced to request approval for a quote post.
The QuoteRequest
activity uses the object
property to refer to the quoted object, and the instrument
property to refer to the quote post.
QuoteRequest
activity{
"@context": [
"https://www.w3.org/ns/activitystreams",
{
"QuoteRequest": "https://w3id.org/fep/044f#QuoteRequest",
"quote": {
"@id": "https://w3id.org/fep/044f#quote",
"@type": "@id"
}
}
],
"type": "QuoteRequest",
"id": "https://example.com/users/bob/statuses/1/quote",
"actor": "https://example.com/users/bob",
"object": "https://example.com/users/alice/statuses/1",
"instrument": {
"type": "Note",
"id": "https://example.com/users/bob/statuses/1",
"attributedTo": "https://example.com/users/bob",
"to": [
"https://www.w3.org/ns/activitystreams#Public",
"https://example.com/users/alice"
],
"content": "I am quoting alice's post<br/>RE: https://example.com/users/alice/statuses/1",
"quote": "https://example.com/users/alice/statuses/1"
}
}
When receiving a QuoteRequest
activity, the original author decides (either manually or automatically) whether the quote is acceptable. Software that automatically accepts quotes on the author’s behalf should notify the author of such quotes according to their notification settings.
The receiving end MAY inspect the instrument
of the QuoteRequest
itself to decide whether it is acceptable.
If the quote post is considered acceptable, the original author MUST reply with an Accept
activity with the QuoteRequest
activity as its object, and a QuoteAuthorization
as its result
.
If the quote post is considered unacceptable, the authority SHOULD reply with a Reject
activity with the QuoteRequest
activity as its object.
Accept
{
"@context": [
"https://www.w3.org/ns/activitystreams",
{
"QuoteRequest": "https://w3id.org/fep/044f#QuoteRequest"
}
],
"type": "Accept",
"to": "https://example.com/users/bob",
"id": "https://example.com/users/alice/activities/1234",
"actor": "https://example.com/users/alice",
"object": {
"type": "QuoteRequest",
"id": "https://example.com/users/bob/statuses/1/quote",
"actor": "https://example.com/users/bob",
"object": "https://example.com/users/alice/statuses/1",
"instrument": "https://example.org/users/bob/statuses/1"
},
"result": "https://example.com/users/alice/stamps/1"
}
Reject
{
"@context": [
"https://www.w3.org/ns/activitystreams",
{
"Quote": "https://w3id.org/fep/044f#QuoteRequest"
}
],
"type": "Reject",
"to": "https://example.com/users/bob",
"id": "https://example.com/users/alice/activities/1234",
"actor": "https://example.com/users/alice",
"object": {
"type": "QuoteRequest",
"id": "https://example.com/users/bob/statuses/1/quote",
"actor": "https://example.com/users/bob",
"object": "https://example.com/users/alice/statuses/1",
"instrument": "https://example.org/users/bob/statuses/1"
}
}
In order to get approval, the quote post author MUST send a QuoteRequest
(https://w3id.org/fep/044f#QuoteRequest
) activity to the author of the quoted object, with the quoted object as its object
property and the quote post as its instrument
.
The quote post SHOULD be inlined in the instrument
property and, if not, it SHOULD dereferenceable by the recipient at this point, as the author of the quoted object may want to inspect it to decide whether to accept the quote.
The quote post author MAY wait until they receive an Accept
or Reject
activity before sending the post’s Create
activity to its intended audience.
Doing so is possible for ActivityPub servers that implement the current proposal, and avoids having to issue an Update
soon afterwards the Create
for the quote post.
It is however not possible to implement for ActivityPub clients, which will likely need to issue a Create
before the QuoteRequest
activity.
If the author of the quote post receives a Reject
activity from the quoted object’s author to their QuoteRequest
activity, they MUST consider the quote post to be explicitly rejected.
If the implementation waits for the Accept
before issuing a Create
, this MAY translate as the inability to publish the quote post.
Otherwise, it MAY translate as a Delete
to outright remove the quote post, or an Update
to remove the quote part from the post.
If the author of the quote receives an Accept
activity, they MUST add a reference to its result
in the quoteAuthorization
(https://w3id.org/fep/044f#quoteAuthorization
) property.
Depending on whether they already sent a Create
activity to the quote post’s intended audience, they SHOULD send a Create
activity or an Update
activity with the updated quoteAuthorization
property.
Note
object{
"@context": [
"https://www.w3.org/ns/activitystreams",
{
"quote": "https://w3id.org/fep/044f#quote",
"quoteAuthorization": "https://w3id.org/fep/044f#quoteAuthorization"
}
],
"type": "Note",
"id": "https://example.com/users/bob/statuses/1",
"attributedTo": "https://example.com/users/bob",
"to": [
"https://www.w3.org/ns/activitystreams#Public",
"https://example.com/users/alice"
],
"content": "I am quoting alice's post<br/>RE: https://example.com/users/alice/statuses/1",
"quote": "https://example.com/users/alice/statuses/1",
"quoteAuthorization": "https://example.com/users/alice/stamps/1"
}
When processing a quote post from a remote actor, a recipient MUST consider them unapproved unless any of those conditions apply:
attributedTo
)quoteAuthorization
exists, can be dereferenced and is a valid QuoteAuthorization
activity for this objectThe original author may want to perform /a posteriori/ moderation of the quote posts, or block a quote poster in particular.
To do this, the original actor MUST Delete
the QuoteAuthorization
. They SHOULD send the Delete
activity to the quote post’s author and any recipient it has reasons to think has accessed the quote post.
The original author MUST NOT embed the object
nor the target
of the QuoteAuthorization
, so as to avoid potential information leakage.
Upon receiving a Delete
activity for a previously-verified QuoteAuthorization
, third-parties MUST check that the Delete
is valid and MUST subsequently consider the quote post unapproved.
Additionally, if the recipient owns the quote post, it MUST forward the Delete
to the audience of the quote post.
Because getting revocation properly forwarded depends on the good will of the revoked post’s author, it may be necessary to have other means of checking whether an approval has been revoked.
For this reason, recipients SHOULD re-check the quoteAuthorization
document when an already-known quote post is accessed for the first time in a given period of time.
This proposal has been made with great care to not require new server behavior, allowing ActivityPub clients to implement this proposal without requiring generic ActivityPub server software to implement additional logic.
In particular, this is the reason the approval stamp is a separate object rather than the Accept
itself. Indeed, nothing in the ActivityPub specification would cause a Reject
or Undo
activity to invalidate the Accept
activity itself, which means it would not be suitable as an approval stamp.
While ActivityPub does not technically forbid Accept
activities to be the target of a Delete
activity, we have found no precedent for that, and we anticipate that deleting activities might not be correctly handled across the fediverse.
For this reason, we opted to use a separate object that can be directly managed by an ActivityPub client, for instance by issuing a Create
activity ahead of sending the Accept
activity, and that can be deleted with the usual mechanism.
An alternative approach we considered is using a dedicated endpoint to check for approval of a quote. This would effectively allow externalizing approval verification to a separate mechanism, but while this would not require new server behavior, this would still require a new server component to be specified, which is why we opted for the mechanism described in this specification instead.
(This section is non-normative.)
To clients that do not implement this FEP, quote posts are seen as regular posts with no explicit relation with the quoted post. While this is preferable to having the quoted post relayed without the comment, this is still lacking significant semantic context.
Therefore, quote posts should be authored in such a way that their contents include a reference to the quoted post, e.g. by adding <span class=\"quote-inline\"><br/>RE: <a href=\"https://example.com/users/alice/statuses/1\">https://example.com/users/alice/statuses/1</a></span>
.
Using a special class like quote-inline
can be useful to hide redundant information information when the post is detected to have an attached quote.
(This section is non-normative.)
Servers that do not implementing the current FEP will still be able to quote the post without providing any dogpiling-reducing friction. There is unfortunately nothing we can do about that. However, servers which do implement the current FEP should refuse displaying those quotes, so implementing the control mechanisms of the current FEP remains worthwhile.
Effectively revoking authorized quote posts relies on the participation of the quote poster’s server to effectively reach the audience of the quote post. This means that an ill-intentioned server which obtained an authorization could deliberately refuse to forward the revocation. Still, the ability to revoke a quote post remains useful between well-intentioned servers, and opportunistic re-verification of quote approvals should also help with discovering that a quote authorization has been revoked, despite the potential lack of forwarding.
By not adding a hash or copy of the reply in the QuoteAuthorization
object, malicious actors could exploit this in a split horizon setting, sending different versions of the same activity to different actors. This is, however, already a concern in pretty much all contexts in ActivityPub, and enshrining that information in the QuoteAuthorization
object would have many drawbacks:
QuoteAuthorization
object is publicly dereferenceableNone so far.
CC0 1.0 Universal (CC0 1.0) Public Domain Dedication
To the extent possible under law, the authors of this Fediverse Enhancement Proposal have waived all copyright and related or neighboring rights to this work.