Skip to content

Commit

Permalink
MSC4174: add support for WebPush pusher kind
Browse files Browse the repository at this point in the history
  • Loading branch information
MatMaul committed Dec 7, 2024
1 parent ecbc0b7 commit 09ba9c3
Show file tree
Hide file tree
Showing 11 changed files with 1,110 additions and 28 deletions.
2 changes: 2 additions & 0 deletions changelog.d/17987.feature
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
MSC4174: add support for WebPush pusher kind.

97 changes: 97 additions & 0 deletions docs/webpush.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,97 @@

# WebPush

## Setup & configuration

In the synapse virtualenv, generate the server key pair by running
`vapid --gen --applicationServerKey`. This will generate a `private_key.pem`
(which you'll refer to in the config file with `vapid_private_key`)
and `public_key.pem` file, and also a string labeled `Application Server Key`.

You'll copy the Application Server Key to `vapid_app_server_key` so that
web applications can fetch it through `/capabilities` and use it to subscribe
to the push manager:

```js
serviceWorkerRegistration.pushManager.subscribe({
userVisibleOnly: true,
applicationServerKey: "...",
});
```

You also need to set an e-mail address in `vapid_contact_email` in the config file,
where the push server operator can reach you in case they need to notify you
about your usage of their API.

Since for webpush, the push server endpoint is variable and comes from the browser
through the push data, you may not want to have your synapse instance connect to any
random addressable server.
You can use the global options `ip_range_blacklist` and `ip_range_allowlist` to manage that.

A default time-to-live of 15 minutes is set for webpush, but you can adjust this by setting
the `ttl: <number of seconds>` configuration option for the pusher.
If notifications can't be delivered by the push server aftet this time, they are dropped.

## Push key and expected push data

In your web application, [the push manager subscribe method](https://developer.mozilla.org/en-US/docs/Web/API/PushManager/subscribe)
will return
[a subscription](https://developer.mozilla.org/en-US/docs/Web/API/PushSubscription)
with an `endpoint` and `keys` property, the latter containing a `p256dh` and `auth`
property. The `p256dh` key is used as the push key, and the push data must contain
`endpoint` and `auth`. You can also set `default_payload` in the push data;
any properties set in it will be present in the push messages you receive,
so it can be used to pass identifiers specific to your client
(like which account the notification is for).

### events_only

As of the time of writing, all webpush-supporting browsers require you to set
`userVisibleOnly: true` when calling (`pushManager.subscribe`)
[https://developer.mozilla.org/en-US/docs/Web/API/PushManager/subscribe], to
(prevent abusing webpush to track users)[https://goo.gl/yqv4Q4] without their
knowledge. With this (mandatory) flag, the browser will show a "site has been
updated in the background" notification if no notifications are visible after
your service worker processes a `push` event. This can easily happen when synapse
sends a push message to clear the unread count, which is not specific
to an event. With `events_only: true` in the pusher data, synapse won't forward
any push message without a event id. This prevents your service worker being
forced to show a notification to push messages that clear the unread count.

### only_last_per_room

You can opt in to only receive the last notification per room by setting
`only_last_per_room: true` in the push data. Note that if the first notification
can be delivered before the second one is sent, you will still get both;
it only has an effect when notifications are queued up on the gateway.

### Multiple pushers on one origin

Also note that because you can only have one push subscription per service worker,
and hence per origin, you might create pushers for different accounts with the same
p256dh push key. To prevent the server from removing other pushers with the same
push key for your other users, you should set `append` to `true` when uploading
your pusher.

## Notification format

The notification as received by your web application will contain the following keys
(assuming non-null values were sent by the homeserver). These are the
same as specified in [the push gateway spec](https://matrix.org/docs/spec/push_gateway/r0.1.0#post-matrix-push-v1-notify),
but the sub-keys of `counts` (`unread` and `missed_calls`) are flattened into
the notification object.

```
room_id
room_name
room_alias
membership
event_id
sender
sender_display_name
user_is_target
type
content
unread
missed_calls
```
6 changes: 6 additions & 0 deletions mypy.ini
Original file line number Diff line number Diff line change
Expand Up @@ -99,3 +99,9 @@ ignore_missing_imports = True

[mypy-multipart.*]
ignore_missing_imports = True

[mypy-pywebpush.*]
ignore_missing_imports = True

[mypy-py_vapid.*]
ignore_missing_imports = True
582 changes: 579 additions & 3 deletions poetry.lock

Large diffs are not rendered by default.

6 changes: 6 additions & 0 deletions pyproject.toml
Original file line number Diff line number Diff line change
Expand Up @@ -251,6 +251,8 @@ Pympler = { version = "*", optional = true }
parameterized = { version = ">=0.7.4", optional = true }
idna = { version = ">=2.5", optional = true }
pyicu = { version = ">=2.10.2", optional = true }
pywebpush = { version = ">=2.0", optional = true }
py-vapid = { version = ">=1.9", optional = true }

[tool.poetry.extras]
# NB: Packages that should be part of `pip install matrix-synapse[all]` need to be specified
Expand All @@ -277,6 +279,7 @@ test = ["parameterized", "idna"]
# requires libicu's development headers installed on the system (e.g. libicu-dev on
# Debian-based distributions).
user-search = ["pyicu"]
webpush = ["pywebpush", "py-vapid"]

# The duplication here is awful. I hate hate hate hate hate it. However, for now I want
# to ensure you can still `pip install matrix-synapse[all]` like today. Two motivations:
Expand Down Expand Up @@ -310,6 +313,9 @@ all = [
"pympler",
# improved user search
"pyicu",
# WebPush support
"pywebpush",
"py-vapid",
# omitted:
# - test: it's useful to have this separate from dev deps in the olddeps job
# - systemd: this is a system-based requirement
Expand Down
50 changes: 50 additions & 0 deletions synapse/config/experimental.py
Original file line number Diff line number Diff line change
Expand Up @@ -38,6 +38,14 @@
except ImportError:
HAS_AUTHLIB = False

# Determine whether pywebpush is installed.
try:
import pywebpush # noqa: F401

HAS_PYWEBPUSH = True
except ImportError:
HAS_PYWEBPUSH = False

if TYPE_CHECKING:
# Only import this if we're type checking, as it might not be installed at runtime.
from authlib.jose.rfc7517 import JsonWebKey
Expand Down Expand Up @@ -256,6 +264,28 @@ class MSC3866Config:
require_approval_for_new_accounts: bool = False


@attr.s(auto_attribs=True, frozen=True, slots=True)
class MSC4174Config:
"""Configuration for MSC4174"""

enabled: bool = attr.ib(default=False, validator=attr.validators.instance_of(bool))

@enabled.validator
def _check_enabled(self, attribute: attr.Attribute, value: bool) -> None:
# Only allow enabling MSC4174 if pywebpush is installed
if value and not HAS_PYWEBPUSH:
raise ConfigError(
"MSC4174 is enabled but pywebpush is not installed. "
"Please install pywebpush to use MSC4174.",
("experimental", "msc4174", "enabled"),
)

vapid_contact_email: str = ""
vapid_private_key: str = ""
vapid_app_server_key: str = ""
ttl: int = 15 * 60


class ExperimentalConfig(Config):
"""Config section for enabling experimental features"""

Expand Down Expand Up @@ -447,3 +477,23 @@ def read_config(self, config: JsonDict, **kwargs: Any) -> None:

# MSC4076: Add `disable_badge_count`` to pusher configuration
self.msc4076_enabled: bool = experimental.get("msc4076_enabled", False)

# MSC4174: webpush push kind
raw_msc4174_config = experimental.get("msc4174", {})
self.msc4174 = MSC4174Config(**raw_msc4174_config)
if self.msc4174.enabled:
if not self.msc4174.vapid_contact_email:
raise ConfigError(
"'vapid_contact_email' must be provided when enabling WebPush support",
("experimental", "msc4174", "vapid_contact_email"),
)
if not self.msc4174.vapid_private_key:
raise ConfigError(
"'vapid_private_key' must be provided when enabling WebPush support",
("experimental", "msc4174", "vapid_private_key"),
)
if not self.msc4174.vapid_app_server_key:
raise ConfigError(
"'vapid_app_server_key' must be provided when enabling WebPush support",
("experimental", "msc4174", "vapid_app_server_key"),
)
55 changes: 30 additions & 25 deletions synapse/push/httppusher.py
Original file line number Diff line number Diff line change
Expand Up @@ -111,7 +111,6 @@ def __init__(self, hs: "HomeServer", pusher_config: PusherConfig):
self.device_display_name = pusher_config.device_display_name
self.device_id = pusher_config.device_id
self.pushkey_ts = pusher_config.ts
self.data = pusher_config.data
self.backoff_delay = HttpPusher.INITIAL_BACKOFF_SEC
self.failing_since = pusher_config.failing_since
self.timed_call: Optional[IDelayedCall] = None
Expand All @@ -123,9 +122,9 @@ def __init__(self, hs: "HomeServer", pusher_config: PusherConfig):

self.push_jitter_delay_ms = hs.config.push.push_jitter_delay_ms

self.data = pusher_config.data
if self.data is None:
if pusher_config.data is None:
raise PusherConfigException("'data' key can not be null for HTTP pusher")
self.data = pusher_config.data

# Check if badge counts should be disabled for this push gateway
self.disable_badge_count = self.hs.config.experimental.msc4076_enabled and bool(
Expand All @@ -138,26 +137,29 @@ def __init__(self, hs: "HomeServer", pusher_config: PusherConfig):
pusher_config.pushkey,
)

# Validate that there's a URL and it is of the proper form.
if "url" not in self.data:
raise PusherConfigException("'url' required in data for HTTP pusher")

url = self.data["url"]
if not isinstance(url, str):
raise PusherConfigException("'url' must be a string")
url_parts = urllib.parse.urlparse(url)
# Note that the specification also says the scheme must be HTTPS, but
# it isn't up to the homeserver to verify that.
if url_parts.path != "/_matrix/push/v1/notify":
raise PusherConfigException(
"'url' must have a path of '/_matrix/push/v1/notify'"
)
self.url = ""
if pusher_config.kind == "http":
# Validate that there's a URL and it is of the proper form.
if "url" not in self.data:
raise PusherConfigException("'url' required in data for HTTP pusher")

url = self.data["url"]
if not isinstance(url, str):
raise PusherConfigException("'url' must be a string")
url_parts = urllib.parse.urlparse(url)
# Note that the specification also says the scheme must be HTTPS, but
# it isn't up to the homeserver to verify that.
if url_parts.path != "/_matrix/push/v1/notify":
raise PusherConfigException(
"'url' must have a path of '/_matrix/push/v1/notify'"
)
self.url = url

self.data_minus_url = {}
self.data_minus_url.update(self.data)
del self.data_minus_url["url"]

self.url = url
self.http_client = hs.get_proxied_blocklisted_http_client()
self.data_minus_url = {}
self.data_minus_url.update(self.data)
del self.data_minus_url["url"]
self.badge_count_last_call: Optional[int] = None

def on_started(self, should_check_for_notifs: bool) -> None:
Expand Down Expand Up @@ -188,7 +190,10 @@ async def _update_badge(self) -> None:
)
if self.badge_count_last_call is None or self.badge_count_last_call != badge:
self.badge_count_last_call = badge
await self._send_badge(badge)
if await self.send_badge(badge):
http_badges_processed_counter.inc()
else:
http_badges_failed_counter.inc()

def on_timer(self) -> None:
self._start_processing()
Expand Down Expand Up @@ -510,7 +515,7 @@ async def dispatch_push_event(

return res

async def _send_badge(self, badge: int) -> None:
async def send_badge(self, badge: int) -> bool:
"""
Args:
badge: number of unread messages
Expand All @@ -534,9 +539,9 @@ async def _send_badge(self, badge: int) -> None:
}
try:
await self.http_client.post_json_get_json(self.url, d)
http_badges_processed_counter.inc()
return True
except Exception as e:
logger.warning(
"Failed to send badge count to %s: %s %s", self.name, type(e), e
)
http_badges_failed_counter.inc()
return False
9 changes: 9 additions & 0 deletions synapse/push/pusher.py
Original file line number Diff line number Diff line change
Expand Up @@ -22,6 +22,7 @@
import logging
from typing import TYPE_CHECKING, Callable, Dict, Optional

import synapse.config.experimental
from synapse.push import Pusher, PusherConfig
from synapse.push.emailpusher import EmailPusher
from synapse.push.httppusher import HttpPusher
Expand All @@ -42,6 +43,14 @@ def __init__(self, hs: "HomeServer"):
"http": HttpPusher
}

if (
synapse.config.experimental.HAS_PYWEBPUSH
and self.config.experimental.msc4174.enabled
):
from synapse.push.webpushpusher import WebPushPusher

self.pusher_types["webpush"] = WebPushPusher

logger.info("email enable notifs: %r", hs.config.email.email_enable_notifs)
if hs.config.email.email_enable_notifs:
self.mailers: Dict[str, Mailer] = {}
Expand Down
Loading

0 comments on commit 09ba9c3

Please sign in to comment.