-
Notifications
You must be signed in to change notification settings - Fork 384
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
MSC4076: Let E2EE clients calculate app badge counts themselves (disable_badge_count) #4076
Open
ara4n
wants to merge
11
commits into
main
Choose a base branch
from
matthew/msc4076
base: main
Could not load branches
Branch not found: {{ refName }}
Loading
Could not load tags
Nothing to show
Loading
Are you sure you want to change the base?
Some commits from the old base branch may be removed from the timeline,
and old review comments may become outdated.
Open
Changes from 10 commits
Commits
Show all changes
11 commits
Select commit
Hold shift + click to select a range
a3cabb2
MSC4076: Let E2EE clients calculate app & encrypted-room badge counts…
ara4n 83e681f
fix footnotes
ara4n 8a35bae
note to fix existing spec bugs
ara4n fceab8d
s/-/_/
ara4n 27877b5
typo
ara4n aa51372
fix MSC name
ara4n 9d68fb2
more pros & cons
ara4n 7a4bbd3
fix thinko on clear_notifs
ara4n 8bf28cb
split out 'clear-notifs' into a separate MSC
ara4n af2b7ae
xref MSC4234
ara4n 1b84443
withhold the `counts` block in /notify, from manu
ara4n File filter
Filter by extension
Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,106 @@ | ||
# MSC4076: Let E2EE clients calculate app badge counts themselves (disable_badge_count) | ||
|
||
## Problem | ||
|
||
Some push notification implementations (e.g. APNS) let the server update the notification badge count shown on an app | ||
directly from the server, without the client running any code[^1]. In the early days of Matrix before we landed E2EE | ||
this was a desirable thing to do, because: 1) the server knew how many pushes it had sent to the client 2) the client | ||
displayed all pushes 3) the client typically didn't run code on receiving push, but just relied on the OS to display | ||
the push. | ||
|
||
However, since E2EE landed, this breaks down for E2EE rooms because only the client can reliably know whether a given | ||
message should trigger a push notification or not - given only the client can see the event type and event contents | ||
of E2EE messages, and so decide whether to notify based on the event type being visible - or based on mention keywords | ||
within the message contents. | ||
|
||
The workaround since we launched E2EE has been to assume that all events in E2EE rooms are visible, and so if an E2EE | ||
room is configured to notify for all messages, unrenderable events will still trigger a push notification and increment | ||
the `unread_notifications` count for that room and consider that room as unread when calculating the app badge count | ||
passed to the push gateway. As a result, invisible events like VoIP signalling will trigger an app badge count. | ||
|
||
Meanwhile, clients now display notifications on both Android & iOS by running local code on receiving a push, rather | ||
than displaying it verbatim from the server. This means the push rules can be executed locally and notifications should | ||
only be displayed if the event actually is visible and matched the push rules. Moreover, the clients can calculate and | ||
track their own accurate `unread_notification` count for E2EE rooms (while still using the `unread_notification` count | ||
in the /sync response for non-E2EE rooms if they want), and from that calculate an accurate app badge count | ||
clientside. | ||
|
||
The problem is that the server still tries to override the app badge count whenever it sends a push to the client though, | ||
which at best will be wrong (either understeering due to missing E2EE mentions, or oversteering due to alerting about | ||
rooms with invisible events) - and at worst will race and fight[^2] with the app badge count maintained locally by the | ||
app, if any. | ||
|
||
Therefore we need a way for sophisticated E2EE clients to tell the server to stop overriding their app badge count. | ||
|
||
Also, the server currently tracks which events the user has read, and will send a 'null' push to the client to update | ||
the app badge count once a given pushed event has been read. But given the server doesn't reliably know what events | ||
cause pushes in E2EE rooms, we will need a new mechanism to reduce the badge count as the user reads rooms (or parts | ||
them, or reconfigures their push settings). However, this can be considered as a separate MSC: | ||
[MSC4234](https://github.com/matrix-org/matrix-spec-proposals/pull/4234) | ||
|
||
Finally, a separate but related problem is that historically the description of the `unread` count passed to push | ||
gateways in the Push API is currently wrong[^3]: it says it's "The number of unread messages a user has across all of | ||
the rooms they are a member of." but in practice it's "The number of rooms the user has with unread push | ||
notifications" (although synapse has a config option to change that, confusingly). So we should fix that too. | ||
|
||
## Solution | ||
|
||
When a pusher is registered for a given device via `POST /_matrix/client/v3/pushers/set`, we add a new boolean field | ||
to the request called `disable_badge_count` which defaults to `false` if absent. | ||
|
||
If `disable_badge_count` is set to true, then the pusher will not specify `unread` or `missed_calls` in the | ||
ara4n marked this conversation as resolved.
Show resolved
Hide resolved
|
||
`POST /_matrix/push/v1/notify` request to the target push gateway, and so not override the client's app badge. | ||
|
||
Solving the problem of prodding other clients to update their badge counts when a user reads messages in a given room has been | ||
split out into a separate MSC: [MSC4234](https://github.com/matrix-org/matrix-spec-proposals/pull/4234). | ||
|
||
While we're here, we also fix the spec to describe the behaviour of the `unread` count field in `POST /_matrix/push/v1/notify` | ||
correctly. We also remove the `missed_calls` field at all, as nothing has ever implemented it, as far as I know - I | ||
think it was originally included for questionable futureproofness and never hooked up ("ooh, if we are tracking unread | ||
message notifs, let's also track unread call notifs")[^4]. | ||
|
||
## Potential issues | ||
|
||
1. It's not clear that the push extension will be able to reliably calculate badge counts, as they run with very | ||
constrained resources, and the client will need to have spidered all unread messages in all E2EE rooms to get a correct | ||
number. So if the client has been offline for a while and comes back, we have a problem if the push service has given | ||
up pushing the client, and potential notifs get dropped. Perhaps we can detect when this happens (e.g. seqnums on push | ||
notifs?) and warn the user to launch the app to catch up on any notifs they may have missed? | ||
|
||
2. Ideally we should also omit `unread_notifications` and `unread_thread_notifications` in /sync responses entirely for | ||
E2EE rooms for that device, given they will be subtly wrong and encourage the client not to correctly calculate them | ||
themselves. However, this is hygiene rather than functionality and can be a different MSC for an `e2ee_badge_aware` | ||
room_filter or similar. | ||
|
||
## Alternatives | ||
|
||
We could track app badge count on the server, but let it be set by your clients instead - e.g. by a dedicated API like | ||
`POST /_matrix/client/v3/pushers/set_badge { badge: 123 }`, which would in turn push it to all clients so they are in | ||
sync. This would avoid each client individually trying to figure out when to reduce the badge count in its Push | ||
Extension - instead the client sending the read receipts would do it for them. But on the other hand, it would not | ||
work with per-device push rules. And if the local client happens to be stale on spidering for old push events, the | ||
incorrect badge count would leak confusingly to the other clients which may be more accurate themselves. | ||
|
||
In terms of configuring pushers as `disable_badge_count`: we could also do this via a per-device underride push rule with | ||
a custom `action` of `disable_badge_count`. However, this isn't really a push action: it shouldn't be changeable by other | ||
push rules, for instance. Instead it's config data at the point the pusher is created, hence putting it there. | ||
|
||
## Security considerations | ||
|
||
None | ||
|
||
## Unstable prefix | ||
|
||
Until this MSC is merged: | ||
* `disable_badge_count` should be prefixed as `org.matrix.msc4076.disable_badge_count` when setting pushers | ||
|
||
## Dependencies | ||
|
||
None. | ||
|
||
## Footnotes | ||
|
||
[^1]: https://developer.apple.com/documentation/usernotifications/setting_up_a_remote_notification_server/generating_a_remote_notification#2943362 | ||
[^2]: https://github.com/vector-im/element-x-ios/issues/2066 | ||
[^3]: https://github.com/matrix-org/matrix-spec/issues/644 | ||
[^4]: https://github.com/matrix-org/synapse/blame/bdc21e72820e148941bbecb36200d51ca340748d/synapse/push/httppusher.py |
Add this suggestion to a batch that can be applied as a single commit.
This suggestion is invalid because no changes were made to the code.
Suggestions cannot be applied while the pull request is closed.
Suggestions cannot be applied while viewing a subset of changes.
Only one suggestion per line can be applied in a batch.
Add this suggestion to a batch that can be applied as a single commit.
Applying suggestions on deleted lines is not supported.
You must change the existing code in this line in order to create a valid suggestion.
Outdated suggestions cannot be applied.
This suggestion has been applied or marked resolved.
Suggestions cannot be applied from pending reviews.
Suggestions cannot be applied on multi-line comments.
Suggestions cannot be applied while the pull request is queued to merge.
Suggestion cannot be applied right now. Please check back later.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This document doesn't answer what feels like a fair question to have: Why do the sophisticated E2EE clients not just ignore the count?
My presumption is that this is done at the OS-level and the app does not get the choice?
So the next point might be: is this MSC really superior to performing this filtering in the application-specific push gateway (Sygnal) like many other such platform-specific quirks, which already have configuration options?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
yes, it's done at the OS level, so the app doesn't get the choice.
I'm not aware of a standard way to configure push gateways for clients in general to say "please don't try to override the app badge count" - wouldn't doing it in a generic way end up being precisely the same as this MSC?