Account data callbacks
Account data callbacks allow module developers to react to changes of the account dataof local users. Account data callbacks can be registered using the module API'sregister_account_data_callbacks
method.
Callbacks
The available account data callbacks are:
on_account_data_updated
First introduced in Synapse v1.57.0
async def on_account_data_updated( user_id: str, room_id: Optional[str], account_data_type: str, content: "synapse.module_api.JsonDict",) -> None:
Called after user's account data has been updated. The module is given theMatrix ID of the user whose account data is changing, the room ID the data is associatedwith, the type associated with the change, as well as the new content. If the accountdata is not associated with a specific room, then the room ID isNone
.
This callback is triggered when new account data is added or when the data associated witha given type (and optionally room) changes. This includes deletion, since in Matrix,deleting account data consists of replacing the data associated with a given type(and optionally room) with an empty dictionary ({}
).
Note that this doesn't trigger when changing the tags associated with a room, as these areprocessed separately by Synapse.
If multiple modules implement this callback, Synapse runs them all in order.
Example
The example below is a module that implements theon_account_data_updated
callback, andsends an event to an audit room when a user changes their account data.
import jsonimport attrfrom typing import Any, Dict, Optionalfrom synapse.module_api import JsonDict, ModuleApifrom synapse.module_api.errors import ConfigError@attr.s(auto_attribs=True)class CustomAccountDataConfig: audit_room: str sender: strclass CustomAccountDataModule: def __init__(self, config: CustomAccountDataConfig, api: ModuleApi): self.api = api self.config = config self.api.register_account_data_callbacks( on_account_data_updated=self.log_new_account_data, ) @staticmethod def parse_config(config: Dict[str, Any]) -> CustomAccountDataConfig: def check_in_config(param: str): if param not in config: raise ConfigError(f"'{param}' is required") check_in_config("audit_room") check_in_config("sender") return CustomAccountDataConfig( audit_room=config["audit_room"], sender=config["sender"], ) async def log_new_account_data( self, user_id: str, room_id: Optional[str], account_data_type: str, content: JsonDict, ) -> None: content_raw = json.dumps(content) msg_content = f"{user_id} has changed their account data for type {account_data_type} to: {content_raw}" if room_id is not None: msg_content += f" (in room {room_id})" await self.api.create_and_send_event_into_room( { "room_id": self.config.audit_room, "sender": self.config.sender, "type": "m.room.message", "content": { "msgtype": "m.text", "body": msg_content } } )