matrix-doc/proposals/3912-relation-based-redacti...

7.5 KiB

MSC3912: Redaction of related events

There are cases where, when redacting an event, it would make sense to also redact all events that relate to it in a certain way. A few examples of this are:

  • Edits: when redacting a message that has been edited, also redacting its edits would ensure that none of the information in the original message leaks through them.
  • Live location sharing (from MSC3489): the user streaming its live location might want to delete their location history once they have stopped sharing it.
  • Moderation: a room's moderator might want to delete an entire thread from a room, a poll with all its answers, etc.

Currently, the only way to do this is for the client to keep track of each individual event and redact them itself. This is cumbersome because it might require clients to work around their own limits on how much history is stored for a given room, and might cause them to hit rate limits when attempting to send redactions, which would translate into general slowness for the user (since their other actions would also be prevented by rate limiters as a result).

Proposal

We add a new optional with_rel_types property to the body of the PUT /_matrix/client/v3/rooms/{roomId}/redact/{eventId}/{txnId} requests. This property is a list of relation types. If an event relates to the event specified in the {eventId} parameter with a relation type specified in this with_rel_types property, the server also redacts it on behalf of the user performing the redaction. The response body of this endpoint does not change.

The client may not be aware of the relation types associated with an event (for example in the case of moderation). That's why we introduce a catch-all "*" value, which if found in the list means "any relation type".

For example, let's consider the following message:

{
    "type": "m.room.message",
    "content": {
        "body": "Hello",
        "msgtype": "m.text"
    },
    "event_id": "$a"
}

And its edit:

{
    "type": "m.room.message",
    "content": {
        "body": "* Hello world!",
        "msgtype": "m.text",
        "m.new_content": {
            "body": "Hello world!",
            "msgtype": "m.text"
        },
        "m.relates_to": {
            "rel_type": "m.replace",
            "event_id": "$a"
        }
    },
    "event_id": "$b"
}

Then the request:

PUT /_matrix/client/v3/rooms/!someroom:example.com/redact/$a/foo

{
    "with_rel_types": ["m.replace"]
}

Causes events $a and $b to both get redacted.

A server may wish to leverage MSC2244 (mass redactions) to redact all events targeted by the redaction request if supports it, however this is not a requirement.

If the with_rel_types property is absent from the request body, only the event referred to by the {eventId} parameter ($a in the example above) is redacted. Same goes if with_rel_types is an empty list.

If an event that matches the redaction criteria (i.e. relates to the event that's being redacted with one of the relation types specified in the request) cannot be redacted by the user sending the redaction request (because of insufficient permissions), it is ignored. Clients may attempt to detect cases where a user would not be able to redact some events matching the redaction criteria (e.g. matching m.reference relations when redacting a m.poll.start event, matching m.thread relations, etc., with an insufficient power level to delete other users' events) and display a warning to the user in this case.

If an event that matches the redaction criteria comes down federation after redaction requests has completed, the server must attempt to redact it on behalf of the redaction request's sender.

Since the response format to PUT /_matrix/client/v3/rooms/{roomId}/redact/{eventId}/{txnId} requests does not change as a result of this proposal, servers may respond to the request once the event referenced to by the {eventId} parameter is redacted, and redact events that relate to it in the background.

This is important to note that we do not consider parent events. In the previous example, a redaction of $b would not cause $a to be redacted. The following request:

PUT /_matrix/client/v3/rooms/!someroom:example.com/redact/$b/foo

{
    "with_rel_types": ["m.replace"]
}

Causes only event $b to get redacted.

The redaction will be limited to the events which meet the validity requirements of the selected relationship types, for example these requirements for edited events. In case of the previous example, let's consider this additional event $c:

{
    "type": "m.room.message",
    "content": {
        "body": "* Hello world2!",
        "msgtype": "m.text",
        "m.new_content": {
            "body": "Hello world2!",
            "msgtype": "m.text"
        },
        "m.relates_to": {
            "rel_type": "m.replace",
            "event_id": "$b"
        }
    },
    "event_id": "$c"
}

This edit event is invalid because we must not edit an edit (We were supposed to refer to the original event: $a) Then the request:

PUT /_matrix/client/v3/rooms/!someroom:example.com/redact/$a/foo

{
    "with_rel_types": ["m.replace"]
}

Causes events $a and $b to get redacted, but no $c

Unstable feature in /versions

While this MSC is unstable, homeservers which support this MSC should indicate it by adding org.matrix.msc3912 flag in the unstable_features array of the response to GET /_matrix/client/versions requests. Once this MSC becomes a part of a spec version, clients should rely on the presence of the spec version, that supports the MSC, in versions on /versions, to determine support.

Potential issues

Since all the redactions are sent on behalf of the user who sent the original PUT /_matrix/client/v3/rooms/{roomId}/redact/{eventId}/{txnId} request, if this user leaves the room the server is no longer able to send redactions on their behalf. However, the author considers this a small edge case which is unlikely to happen in practice, and can be worked around by another user (with permissions to delete the new events) sending another PUT /_matrix/client/v3/rooms/{roomId}/redact/{eventId}/{txnId}.

Alternatives

The server could respond to the client with some information about whether some events could not be redacted due to insufficient permissions. However, this could lead to confusing user experience, wherein the server responds by telling the client all events could be redacted, but another event comes down federation shortly afterwards that cannot be redacted.

An alternative proposal would be specifying a static list of relation types to consider when redacting a message. However, this would not work well with more generic ones such as m.reference, and would restrict clients in the UX they offer to users (e.g. a client might want to show a checkbox to a user offering them to redact a whole thread when redacting the thread's root, if this user is a room moderator). This could still be helpful for some relation types (e.g. m.replaces) but clients are likely better positioned to make this decision given they have more control over feature UX.

Unstable prefixes

Until this proposal is stabilised, org.matrix.msc3912.with_relations should be used instead of with_rel_types.