The downside of this solution is that sender can insert arbitrary hidden text by creating .quote-inline
elements (unless clients require inner text to match the RE: <URL>
template exactly).
They can, but Iâm not sure how much of that is an issue. Itâs also already something that senders can do using some class names in multiple implementations. I am uneasy with mandating a exact content format here, especially since implementations may e.g. want to make that part an actual link, which would lead us to have to specify the exact HTML.
One solution I was thinking of since the original Object Links FEP appeared regarding this was to match the a
element by the Linkâs href and walking itâs parent elements until one has a textContent matching the name of the object link.
This seems like it would match quite a few use cases (crucially, both in cases where Link.name == a
.textContent and in cases where there are surrounding decorations like the RE:
prefix), and the only thing itâd require out of producers would be to wrap the text in any HTML element (even a plain span
without any classes would do)
My main concern about this is Iâm unsure about how many HTML tooling actually supports computing textContent
.
I think the stamp can apply to other scopes without becoming ambiguous. Essentially, it would specify allowed activities, including actors, objects, and targets. So your example would become something like this
{
"@context": [
"https://www.w3.org/ns/activitystreams",
{
"toot": "http://joinmastodon.org/ns#",
"Authorization": "toot:Authorization"
}
],
"type": "Authorization",
"id": "https://example.com/users/alice/stamps/1",
"activity": "Quote",
"actor": "https://example.org/users/bob",
"attributedTo": "https://example.com/users/alice",
"object": "https://example.org/users/bob/statuses/1",
"target": "https://example.com/users/alice/statuses/1"
}
But again, please donât think of this as an objection. It can be tackled as itâs own topic. And I see no reason that multiple kinds of authz tokens canât coexist, so the existence of this specialized one doesnât preclude a future generic token
So, essentially moving the QuoteAuthorization
to a generic Authorization
with an activity
property. I donât see a fundamental difference in both approaches, but I can see why youâd consider this cleaner. Iâm personally fine with this change!
I was thinking more about this and wondered what we really wanted to achieve here. If we donât require the RE: <URL>
template in the spec, then people can still insert arbitrary hidden text, it would just have to also be in the Link.name
, but in both cases this would be hidden from the user, and in both cases, inspecting the ActivityStreams object would surface it. Using the textContent
would just make it harder to implement for clients which do not have convenient HTML parsing and CSS addressing.
the example seems confusing semantically; right now, it looks like an Authorization activity performed by bob and attributed to both alice and bob.
from a data modeling perspective i donât particularly have any opposition to a base Authorization
class that can be subclassed as LikeAuthorization
, ShareAuthorization
, ReplyAuthorization
, QuoteAuthorization
, and so on as needed.
i think itâs also possible to have a more generic âlink approvalâ that can be an attribute of a generic web link? so take the approvedBy
and make it not only a property of as:Link
but also an attribute of Link headers or a/link HTML nodes? RFC 8288 - Web Linking â although itâs unclear when something is a string and when something is a URI
I would personally like some usage such as:
<a href="https://quote.example/url" class="quote">RE: https://quote.example/url</a>
as it uses the same pattern already done for mentions and tags, e.g.
<a href="https://quote.example/@alice" class="mention">@alice@quote.example</a>
<a href="https://quote.example/topic" class="hashtag">#topic</a>
Iâm not sure about how rel relations work, see RFC 8288, but one might also consider the rel attribute, i.e.
<a href="https://quote.example/url" rel="quote">RE: https://quote.example/url</a>
<a href="https://quote.example/alice" rel="mention">@alice@quote.examplel</a>
<a href="https://quote.example/topic" rel="tag">#topic</a>
However, harmonizing these things should probably be the topic of a different FEP.
We would like to create one FEP to standardise representing links to ActivityPub objects in content.
A big issue we have right now is that it is up to the receiving server or client to try to guess if a link has an AP representation or not, for example to decide if the link should be opened in-client or as a web URL, or to display them with some special UI (like Discord does when you paste a link to a message). We think that the authoring software should be the one doing this work. Object links are a good candidate for this but have a few missing things (I dont remember the details), so we are planning to try to harmonize and standardize this.
After another round of thinking it would probably be enough to specify that if a link contains the class tag
that one should look in the tag
property of the object to obtain more information. Using a class also has the advantage that one could extend it to a span, e.g. use
<span class="tag">RE: <a href="https://host.example/obj">host.example/obj</a></span>
instead of
<a href="https://host.example/obj" class="tag">RE: host.example/obj</a></span>
Examples
{
"content": "<a href='https://host.example/objId' class='tag'>host.example/objId</a>",
"tag": [{
"type": "Link",
"href": "https://host.example/objId",
"mediaType": "application/activity+json",
"name": "host.example/objId"
}]
}
to link to an ActivityPub object and
{
"content": "<a href='https://host.example/page' class='tag'>host.example/page</a>",
"tag": [{
"type": "Link",
"href": "https://host.example/page",
"mediaType": "text/html",
"name": "host.example/page"
}]
}
to link to a webpage. Similarly, one can use
{
"content": "<a href='https://host.example/alice' class='tag'>@alice</a>",
"tag": [{
"type": "Mention",
"href": "https://host.example/alice",
"name": "@alice"
}]
}
for a mention. Iâll leave adding Hashtag to the reader.
Why not link relation?
The link relation tag is probably inappropriate for this, see HTML Standard
Just wanna add to this convo that weâve been working with interaction policies in GoToSocial for a while now, and the protocol extension described here is extensible to include a canQuote
property on an interaction policy: Interaction Policy - GoToSocial Documentation
Reply, Like, and Announce controls and approvals etc are already running in GoToSocial since v0.17.0 last year. It would be pretty neato if Mastodon could reuse the interaction policy properties and approval flow for quote posts, insofar as thatâs possible. Have discussed this in the Mastodon discord with Claire as well, but just putting it here too.
We are of course not interested in breaking compatibility for the sake of breaking compatibility.
Reviewing the current proposal and GoToSocialâs current documentation, I will try to highlight the differences in the two proposals.
Advertising a quote policy
https://docs.gotosocial.org/en/latest/federation/interaction_policy/#overview
GoToSocialâs interaction policies do not define a quote policy, but following the pattern of canLike
and so on, a natural extension would be canQuote
.
The current proposal has toot:acceptsQuotesFrom
which is roughly equivalent to gts:interactionPolicy.gts:canQuote.gts:always
.
There are three differences:
- vocabulary itself, the differences here are uninteresting
- attribute nesting in GoToSocialâs case, while the current proposal has a flat top-level attribute. I still donât think there is a strong argument for nesting attributes here, and it makes things more awkward. There is a precedent for that in ActivityPub through the
endpoints
attribute on actors, though - under the current proposal, you can only declare whether quotes are likely to be accepted with no more granularity, but GoToSocial has
always
andrequiresApproval
. I am not sure whether this distinction is worthwhile, andalways
might be a misnomer there (what it says is that approval is expected to be immediate and automatic, whilerequiresApproval
says manual user approval is required)
While I have hang-ups with the last two differences, I donât think they are a show-stopper by themselves.
Approval stamps
https://docs.gotosocial.org/en/latest/federation/interaction_policy/#approval-objects
GoToSocialâs documentation defines LikeApproval
, ReplyApproval
and AnnounceApproval
that are analogous to the QuoteAuthorization
defined in this proposal. GTSâ documentation is less clear and much more lenient on the definition of the attributes, but this uses the same object
, target
and attributedTo
attributes as the current proposal.
Apart from the *Approval
naming scheme being different to the *Authorization
naming scheme, this is actually very similar (which makes sense since it has been adopted from in-progress discussion on the current proposal).
This also means that the objections against the re-use of object
and target
attributes are valid against both the current proposal and GTSâ implementation.
Quote request
The notion of quote requests is only found in the current proposal.
This makes sense, as GoToSocial does not implement quotes and did not attempt specifying them yet. But this is still a slight departure in the general approval flow, as we will come back to in the next section.
Requesting, obtaining and validating approval
The approval flow described in this proposal and that described in GoToSocialâs documentation are pretty different. Of course, GoToSocial not defining quote posts, we cannot have an exact direct comparison, but the closest to that is replies: if GoToSocial handled replies the same way that this proposal does, it would require a Reply
activity that would be used to ask the replied-to-actor for an approval stamp to attach to the reply Note
object, but this is not the case.
Instead, GoToSocial special-cases delivery with âDO NOT DISTRIBUTE THE ACTIVITY FURTHER THAN THIS AT THIS POINTâ.
This requires S2S behaviour changes that we have been careful to avoid in our current specification with this following flow:
This is probably one of the most fundamental differences between GTSâ specification and ours. We could always re-use the vocabulary and specify a different flow for Quote
and QuoteAuthorization
, but having ad-hoc flows seems ugly and error-prone. Furthermore, this would be advocating for a specification that does require S2S special-casing, which we wanted to avoid in the first place.
Server behavior considerations are discussed at length in the current proposal: Pre-FEP: Quote posts, quote policies and quote controls
Revocation of a quote post
This is completely missing from GoToSocialâs specification, but it is not fundamentally incompatible with it. This could easily be added.
Opportunistic re-verification of quote approvals
This is completely missing from GoToSocialâs specification, but it is not fundamentally incompatible with it. This could easily be added.
It seems like the biggest show stopper for mastodon adopting the same / similar thing to GtS is the fact that Reply or Like or Announce are sent only to the interactee, when trying to get approval, rather than ReplyRequest, LikeRequest, or AnnounceRequest or whatever else. Iâd be happy to make changes in GtS to use these types (or something similar) instead if it provides better compatibility with âpureâ activity pub servers that use the c2s api. Not an issue. The rest of the differences seem mostly stylistic to me I think, unless Iâm missing anything!
Edit: maybe itâs worth renaming Quote to QuoteRequest or something like that to clarify that youâre sending it to seek approval.
Edit2: I would stand behind the âalwaysâ and âapprovalRequiredâ distinction as I think it can provide useful UI benefits. V. similar to the âmanuallyApprovesFollowersâ flag for follow requests.
I strongly agree that the âalwaysâ and âapprovalRequiredâ distinction is very much useful!