410 lines
13 KiB
Python
410 lines
13 KiB
Python
# Copyright 2015, 2016 OpenMarket Ltd
|
|
# Copyright 2017 New Vector Ltd
|
|
# Copyright 2019 The Matrix.org Foundation C.I.C.
|
|
#
|
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
|
# you may not use this file except in compliance with the License.
|
|
# You may obtain a copy of the License at
|
|
#
|
|
# http://www.apache.org/licenses/LICENSE-2.0
|
|
#
|
|
# Unless required by applicable law or agreed to in writing, software
|
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
# See the License for the specific language governing permissions and
|
|
# limitations under the License.
|
|
|
|
import copy
|
|
from typing import Any, Dict, List
|
|
|
|
from synapse.push.rulekinds import PRIORITY_CLASS_INVERSE_MAP, PRIORITY_CLASS_MAP
|
|
|
|
|
|
def list_with_base_rules(rawrules: List[Dict[str, Any]]) -> List[Dict[str, Any]]:
|
|
"""Combine the list of rules set by the user with the default push rules
|
|
|
|
Args:
|
|
rawrules: The rules the user has modified or set.
|
|
|
|
Returns:
|
|
A new list with the rules set by the user combined with the defaults.
|
|
"""
|
|
ruleslist = []
|
|
|
|
# Grab the base rules that the user has modified.
|
|
# The modified base rules have a priority_class of -1.
|
|
modified_base_rules = {r["rule_id"]: r for r in rawrules if r["priority_class"] < 0}
|
|
|
|
# Remove the modified base rules from the list, They'll be added back
|
|
# in the default positions in the list.
|
|
rawrules = [r for r in rawrules if r["priority_class"] >= 0]
|
|
|
|
# shove the server default rules for each kind onto the end of each
|
|
current_prio_class = list(PRIORITY_CLASS_INVERSE_MAP)[-1]
|
|
|
|
ruleslist.extend(
|
|
make_base_prepend_rules(
|
|
PRIORITY_CLASS_INVERSE_MAP[current_prio_class], modified_base_rules
|
|
)
|
|
)
|
|
|
|
for r in rawrules:
|
|
if r["priority_class"] < current_prio_class:
|
|
while r["priority_class"] < current_prio_class:
|
|
ruleslist.extend(
|
|
make_base_append_rules(
|
|
PRIORITY_CLASS_INVERSE_MAP[current_prio_class],
|
|
modified_base_rules,
|
|
)
|
|
)
|
|
current_prio_class -= 1
|
|
if current_prio_class > 0:
|
|
ruleslist.extend(
|
|
make_base_prepend_rules(
|
|
PRIORITY_CLASS_INVERSE_MAP[current_prio_class],
|
|
modified_base_rules,
|
|
)
|
|
)
|
|
|
|
ruleslist.append(r)
|
|
|
|
while current_prio_class > 0:
|
|
ruleslist.extend(
|
|
make_base_append_rules(
|
|
PRIORITY_CLASS_INVERSE_MAP[current_prio_class], modified_base_rules
|
|
)
|
|
)
|
|
current_prio_class -= 1
|
|
if current_prio_class > 0:
|
|
ruleslist.extend(
|
|
make_base_prepend_rules(
|
|
PRIORITY_CLASS_INVERSE_MAP[current_prio_class], modified_base_rules
|
|
)
|
|
)
|
|
|
|
return ruleslist
|
|
|
|
|
|
def make_base_append_rules(
|
|
kind: str, modified_base_rules: Dict[str, Dict[str, Any]]
|
|
) -> List[Dict[str, Any]]:
|
|
rules = []
|
|
|
|
if kind == "override":
|
|
rules = BASE_APPEND_OVERRIDE_RULES
|
|
elif kind == "underride":
|
|
rules = BASE_APPEND_UNDERRIDE_RULES
|
|
elif kind == "content":
|
|
rules = BASE_APPEND_CONTENT_RULES
|
|
|
|
# Copy the rules before modifying them
|
|
rules = copy.deepcopy(rules)
|
|
for r in rules:
|
|
# Only modify the actions, keep the conditions the same.
|
|
assert isinstance(r["rule_id"], str)
|
|
modified = modified_base_rules.get(r["rule_id"])
|
|
if modified:
|
|
r["actions"] = modified["actions"]
|
|
|
|
return rules
|
|
|
|
|
|
def make_base_prepend_rules(
|
|
kind: str,
|
|
modified_base_rules: Dict[str, Dict[str, Any]],
|
|
) -> List[Dict[str, Any]]:
|
|
rules = []
|
|
|
|
if kind == "override":
|
|
rules = BASE_PREPEND_OVERRIDE_RULES
|
|
|
|
# Copy the rules before modifying them
|
|
rules = copy.deepcopy(rules)
|
|
for r in rules:
|
|
# Only modify the actions, keep the conditions the same.
|
|
assert isinstance(r["rule_id"], str)
|
|
modified = modified_base_rules.get(r["rule_id"])
|
|
if modified:
|
|
r["actions"] = modified["actions"]
|
|
|
|
return rules
|
|
|
|
|
|
BASE_APPEND_CONTENT_RULES = [
|
|
{
|
|
"rule_id": "global/content/.m.rule.contains_user_name",
|
|
"conditions": [
|
|
{
|
|
"kind": "event_match",
|
|
"key": "content.body",
|
|
"pattern_type": "user_localpart",
|
|
}
|
|
],
|
|
"actions": [
|
|
"notify",
|
|
{"set_tweak": "sound", "value": "default"},
|
|
{"set_tweak": "highlight"},
|
|
],
|
|
}
|
|
]
|
|
|
|
|
|
BASE_PREPEND_OVERRIDE_RULES = [
|
|
{
|
|
"rule_id": "global/override/.m.rule.master",
|
|
"enabled": False,
|
|
"conditions": [],
|
|
"actions": ["dont_notify"],
|
|
}
|
|
]
|
|
|
|
|
|
BASE_APPEND_OVERRIDE_RULES = [
|
|
{
|
|
"rule_id": "global/override/.m.rule.suppress_notices",
|
|
"conditions": [
|
|
{
|
|
"kind": "event_match",
|
|
"key": "content.msgtype",
|
|
"pattern": "m.notice",
|
|
"_id": "_suppress_notices",
|
|
}
|
|
],
|
|
"actions": ["dont_notify"],
|
|
},
|
|
# NB. .m.rule.invite_for_me must be higher prio than .m.rule.member_event
|
|
# otherwise invites will be matched by .m.rule.member_event
|
|
{
|
|
"rule_id": "global/override/.m.rule.invite_for_me",
|
|
"conditions": [
|
|
{
|
|
"kind": "event_match",
|
|
"key": "type",
|
|
"pattern": "m.room.member",
|
|
"_id": "_member",
|
|
},
|
|
{
|
|
"kind": "event_match",
|
|
"key": "content.membership",
|
|
"pattern": "invite",
|
|
"_id": "_invite_member",
|
|
},
|
|
{"kind": "event_match", "key": "state_key", "pattern_type": "user_id"},
|
|
],
|
|
"actions": [
|
|
"notify",
|
|
{"set_tweak": "sound", "value": "default"},
|
|
{"set_tweak": "highlight", "value": False},
|
|
],
|
|
},
|
|
# Will we sometimes want to know about people joining and leaving?
|
|
# Perhaps: if so, this could be expanded upon. Seems the most usual case
|
|
# is that we don't though. We add this override rule so that even if
|
|
# the room rule is set to notify, we don't get notifications about
|
|
# join/leave/avatar/displayname events.
|
|
# See also: https://matrix.org/jira/browse/SYN-607
|
|
{
|
|
"rule_id": "global/override/.m.rule.member_event",
|
|
"conditions": [
|
|
{
|
|
"kind": "event_match",
|
|
"key": "type",
|
|
"pattern": "m.room.member",
|
|
"_id": "_member",
|
|
}
|
|
],
|
|
"actions": ["dont_notify"],
|
|
},
|
|
# This was changed from underride to override so it's closer in priority
|
|
# to the content rules where the user name highlight rule lives. This
|
|
# way a room rule is lower priority than both but a custom override rule
|
|
# is higher priority than both.
|
|
{
|
|
"rule_id": "global/override/.m.rule.contains_display_name",
|
|
"conditions": [{"kind": "contains_display_name"}],
|
|
"actions": [
|
|
"notify",
|
|
{"set_tweak": "sound", "value": "default"},
|
|
{"set_tweak": "highlight"},
|
|
],
|
|
},
|
|
{
|
|
"rule_id": "global/override/.m.rule.roomnotif",
|
|
"conditions": [
|
|
{
|
|
"kind": "event_match",
|
|
"key": "content.body",
|
|
"pattern": "@room",
|
|
"_id": "_roomnotif_content",
|
|
},
|
|
{
|
|
"kind": "sender_notification_permission",
|
|
"key": "room",
|
|
"_id": "_roomnotif_pl",
|
|
},
|
|
],
|
|
"actions": ["notify", {"set_tweak": "highlight", "value": True}],
|
|
},
|
|
{
|
|
"rule_id": "global/override/.m.rule.tombstone",
|
|
"conditions": [
|
|
{
|
|
"kind": "event_match",
|
|
"key": "type",
|
|
"pattern": "m.room.tombstone",
|
|
"_id": "_tombstone",
|
|
},
|
|
{
|
|
"kind": "event_match",
|
|
"key": "state_key",
|
|
"pattern": "",
|
|
"_id": "_tombstone_statekey",
|
|
},
|
|
],
|
|
"actions": ["notify", {"set_tweak": "highlight", "value": True}],
|
|
},
|
|
{
|
|
"rule_id": "global/override/.m.rule.reaction",
|
|
"conditions": [
|
|
{
|
|
"kind": "event_match",
|
|
"key": "type",
|
|
"pattern": "m.reaction",
|
|
"_id": "_reaction",
|
|
}
|
|
],
|
|
"actions": ["dont_notify"],
|
|
},
|
|
]
|
|
|
|
|
|
BASE_APPEND_UNDERRIDE_RULES = [
|
|
{
|
|
"rule_id": "global/underride/.m.rule.call",
|
|
"conditions": [
|
|
{
|
|
"kind": "event_match",
|
|
"key": "type",
|
|
"pattern": "m.call.invite",
|
|
"_id": "_call",
|
|
}
|
|
],
|
|
"actions": [
|
|
"notify",
|
|
{"set_tweak": "sound", "value": "ring"},
|
|
{"set_tweak": "highlight", "value": False},
|
|
],
|
|
},
|
|
# XXX: once m.direct is standardised everywhere, we should use it to detect
|
|
# a DM from the user's perspective rather than this heuristic.
|
|
{
|
|
"rule_id": "global/underride/.m.rule.room_one_to_one",
|
|
"conditions": [
|
|
{"kind": "room_member_count", "is": "2", "_id": "member_count"},
|
|
{
|
|
"kind": "event_match",
|
|
"key": "type",
|
|
"pattern": "m.room.message",
|
|
"_id": "_message",
|
|
},
|
|
],
|
|
"actions": [
|
|
"notify",
|
|
{"set_tweak": "sound", "value": "default"},
|
|
{"set_tweak": "highlight", "value": False},
|
|
],
|
|
},
|
|
# XXX: this is going to fire for events which aren't m.room.messages
|
|
# but are encrypted (e.g. m.call.*)...
|
|
{
|
|
"rule_id": "global/underride/.m.rule.encrypted_room_one_to_one",
|
|
"conditions": [
|
|
{"kind": "room_member_count", "is": "2", "_id": "member_count"},
|
|
{
|
|
"kind": "event_match",
|
|
"key": "type",
|
|
"pattern": "m.room.encrypted",
|
|
"_id": "_encrypted",
|
|
},
|
|
],
|
|
"actions": [
|
|
"notify",
|
|
{"set_tweak": "sound", "value": "default"},
|
|
{"set_tweak": "highlight", "value": False},
|
|
],
|
|
},
|
|
{
|
|
"rule_id": "global/underride/.m.rule.message",
|
|
"conditions": [
|
|
{
|
|
"kind": "event_match",
|
|
"key": "type",
|
|
"pattern": "m.room.message",
|
|
"_id": "_message",
|
|
}
|
|
],
|
|
"actions": ["notify", {"set_tweak": "highlight", "value": False}],
|
|
},
|
|
# XXX: this is going to fire for events which aren't m.room.messages
|
|
# but are encrypted (e.g. m.call.*)...
|
|
{
|
|
"rule_id": "global/underride/.m.rule.encrypted",
|
|
"conditions": [
|
|
{
|
|
"kind": "event_match",
|
|
"key": "type",
|
|
"pattern": "m.room.encrypted",
|
|
"_id": "_encrypted",
|
|
}
|
|
],
|
|
"actions": ["notify", {"set_tweak": "highlight", "value": False}],
|
|
},
|
|
{
|
|
"rule_id": "global/underride/.im.vector.jitsi",
|
|
"conditions": [
|
|
{
|
|
"kind": "event_match",
|
|
"key": "type",
|
|
"pattern": "im.vector.modular.widgets",
|
|
"_id": "_type_modular_widgets",
|
|
},
|
|
{
|
|
"kind": "event_match",
|
|
"key": "content.type",
|
|
"pattern": "jitsi",
|
|
"_id": "_content_type_jitsi",
|
|
},
|
|
{
|
|
"kind": "event_match",
|
|
"key": "state_key",
|
|
"pattern": "*",
|
|
"_id": "_is_state_event",
|
|
},
|
|
],
|
|
"actions": ["notify", {"set_tweak": "highlight", "value": False}],
|
|
},
|
|
]
|
|
|
|
|
|
BASE_RULE_IDS = set()
|
|
|
|
for r in BASE_APPEND_CONTENT_RULES:
|
|
r["priority_class"] = PRIORITY_CLASS_MAP["content"]
|
|
r["default"] = True
|
|
BASE_RULE_IDS.add(r["rule_id"])
|
|
|
|
for r in BASE_PREPEND_OVERRIDE_RULES:
|
|
r["priority_class"] = PRIORITY_CLASS_MAP["override"]
|
|
r["default"] = True
|
|
BASE_RULE_IDS.add(r["rule_id"])
|
|
|
|
for r in BASE_APPEND_OVERRIDE_RULES:
|
|
r["priority_class"] = PRIORITY_CLASS_MAP["override"]
|
|
r["default"] = True
|
|
BASE_RULE_IDS.add(r["rule_id"])
|
|
|
|
for r in BASE_APPEND_UNDERRIDE_RULES:
|
|
r["priority_class"] = PRIORITY_CLASS_MAP["underride"]
|
|
r["default"] = True
|
|
BASE_RULE_IDS.add(r["rule_id"])
|