summary refs log tree commit diff
path: root/synapse/storage/event_push_actions.py
blob: 6a212c630befe09caff329cf0e3b69047cc39052 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
# -*- coding: utf-8 -*-
# Copyright 2015 OpenMarket Ltd
#
# 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.

from ._base import SQLBaseStore
from twisted.internet import defer
from synapse.util.caches.descriptors import cachedInlineCallbacks

import logging
import ujson as json

logger = logging.getLogger(__name__)


class EventPushActionsStore(SQLBaseStore):
    @defer.inlineCallbacks
    def set_push_actions_for_event_and_users(self, event, tuples):
        """
        :param event: the event set actions for
        :param tuples: list of tuples of (user_id, profile_tag, actions)
        """
        values = []
        for uid, profile_tag, actions in tuples:
            values.append({
                'room_id': event.room_id,
                'event_id': event.event_id,
                'user_id': uid,
                'profile_tag': profile_tag,
                'actions': json.dumps(actions)
            })

        def f(txn):
            for uid, _, __ in tuples:
                txn.call_after(
                    self.get_unread_event_push_actions_by_room_for_user.invalidate_many,
                    (event.room_id, uid)
                )
            return self._simple_insert_many_txn(txn, "event_push_actions", values)

        yield self.runInteraction(
            "set_actions_for_event_and_users",
            f,
        )

    @cachedInlineCallbacks(num_args=3, lru=True)
    def get_unread_event_push_actions_by_room_for_user(
            self, room_id, user_id, last_read_event_id
    ):
        def _get_unread_event_push_actions_by_room(txn):
            sql = (
                "SELECT stream_ordering, topological_ordering"
                " FROM events"
                " WHERE room_id = ? AND event_id = ?"
            )
            txn.execute(
                sql, (room_id, last_read_event_id)
            )
            results = txn.fetchall()
            if len(results) == 0:
                return []

            stream_ordering = results[0][0]
            topological_ordering = results[0][1]

            sql = (
                "SELECT ea.event_id, ea.actions"
                " FROM event_push_actions ea, events e"
                " WHERE ea.room_id = e.room_id"
                " AND ea.event_id = e.event_id"
                " AND ea.user_id = ?"
                " AND ea.room_id = ?"
                " AND ("
                "       e.topological_ordering > ?"
                "       OR (e.topological_ordering = ? AND e.stream_ordering > ?)"
                ")"
            )
            txn.execute(sql, (
                user_id, room_id,
                topological_ordering, topological_ordering, stream_ordering
            )
            )
            return [
                {"event_id": row[0], "actions": json.loads(row[1])}
                for row in txn.fetchall()
            ]

        ret = yield self.runInteraction(
            "get_unread_event_push_actions_by_room",
            _get_unread_event_push_actions_by_room
        )
        defer.returnValue(ret)

    @defer.inlineCallbacks
    def remove_push_actions_for_event_id(self, room_id, event_id):
        def f(txn):
            # Sad that we have to blow away the cache for the whole room here
            txn.call_after(
                self.get_unread_event_push_actions_by_room_for_user.invalidate_many,
                (room_id,)
            )
            txn.execute(
                "DELETE FROM event_push_actions WHERE room_id = ? AND event_id = ?",
                (room_id, event_id)
            )
        yield self.runInteraction(
            "remove_push_actions_for_event_id",
            f
        )