2020-05-13 08:38:22 -04:00
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
# Copyright 2020 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.
|
|
|
|
|
2020-09-29 17:26:28 -04:00
|
|
|
from synapse.metrics import GaugeBucketCollector
|
2020-05-13 08:38:22 -04:00
|
|
|
from synapse.metrics.background_process_metrics import run_as_background_process
|
|
|
|
from synapse.storage._base import SQLBaseStore
|
2020-08-05 16:38:57 -04:00
|
|
|
from synapse.storage.database import DatabasePool
|
|
|
|
from synapse.storage.databases.main.event_push_actions import (
|
2020-05-13 08:38:22 -04:00
|
|
|
EventPushActionsWorkerStore,
|
|
|
|
)
|
|
|
|
|
2020-09-29 17:26:28 -04:00
|
|
|
# Collect metrics on the number of forward extremities that exist.
|
|
|
|
_extremities_collecter = GaugeBucketCollector(
|
|
|
|
"synapse_forward_extremities",
|
|
|
|
"Number of rooms on the server with the given number of forward extremities"
|
|
|
|
" or fewer",
|
|
|
|
buckets=[1, 2, 3, 5, 7, 10, 15, 20, 50, 100, 200, 500],
|
|
|
|
)
|
|
|
|
|
2020-09-29 17:30:00 -04:00
|
|
|
# we also expose metrics on the "number of excess extremity events", which is
|
|
|
|
# (E-1)*N, where E is the number of extremities and N is the number of state
|
|
|
|
# events in the room. This is an approximation to the number of state events
|
|
|
|
# we could remove from state resolution by reducing the graph to a single
|
|
|
|
# forward extremity.
|
|
|
|
_excess_state_events_collecter = GaugeBucketCollector(
|
|
|
|
"synapse_excess_extremity_events",
|
|
|
|
"Number of rooms on the server with the given number of excess extremity "
|
|
|
|
"events, or fewer",
|
|
|
|
buckets=[0] + [1 << n for n in range(12)],
|
|
|
|
)
|
|
|
|
|
2020-05-13 08:38:22 -04:00
|
|
|
|
|
|
|
class ServerMetricsStore(EventPushActionsWorkerStore, SQLBaseStore):
|
|
|
|
"""Functions to pull various metrics from the DB, for e.g. phone home
|
|
|
|
stats and prometheus metrics.
|
|
|
|
"""
|
|
|
|
|
2020-08-05 16:38:57 -04:00
|
|
|
def __init__(self, database: DatabasePool, db_conn, hs):
|
2020-05-13 08:38:22 -04:00
|
|
|
super().__init__(database, db_conn, hs)
|
|
|
|
|
|
|
|
# Read the extrems every 60 minutes
|
|
|
|
def read_forward_extremities():
|
|
|
|
# run as a background process to make sure that the database transactions
|
|
|
|
# have a logcontext to report to
|
|
|
|
return run_as_background_process(
|
|
|
|
"read_forward_extremities", self._read_forward_extremities
|
|
|
|
)
|
|
|
|
|
|
|
|
hs.get_clock().looping_call(read_forward_extremities, 60 * 60 * 1000)
|
|
|
|
|
|
|
|
async def _read_forward_extremities(self):
|
|
|
|
def fetch(txn):
|
|
|
|
txn.execute(
|
|
|
|
"""
|
2020-09-29 17:30:00 -04:00
|
|
|
SELECT t1.c, t2.c
|
|
|
|
FROM (
|
|
|
|
SELECT room_id, COUNT(*) c FROM event_forward_extremities
|
|
|
|
GROUP BY room_id
|
|
|
|
) t1 LEFT JOIN (
|
|
|
|
SELECT room_id, COUNT(*) c FROM current_state_events
|
|
|
|
GROUP BY room_id
|
|
|
|
) t2 ON t1.room_id = t2.room_id
|
2020-05-13 08:38:22 -04:00
|
|
|
"""
|
|
|
|
)
|
|
|
|
return txn.fetchall()
|
|
|
|
|
2020-08-05 16:38:57 -04:00
|
|
|
res = await self.db_pool.runInteraction("read_forward_extremities", fetch)
|
2020-09-29 17:30:00 -04:00
|
|
|
|
2020-09-29 17:26:28 -04:00
|
|
|
_extremities_collecter.update_data(x[0] for x in res)
|
2020-05-13 08:38:22 -04:00
|
|
|
|
2020-09-29 17:30:00 -04:00
|
|
|
_excess_state_events_collecter.update_data(
|
|
|
|
(x[0] - 1) * x[1] for x in res if x[1]
|
|
|
|
)
|
|
|
|
|
2020-08-11 17:21:20 -04:00
|
|
|
async def count_daily_messages(self):
|
2020-05-13 08:38:22 -04:00
|
|
|
"""
|
|
|
|
Returns an estimate of the number of messages sent in the last day.
|
|
|
|
|
|
|
|
If it has been significantly less or more than one day since the last
|
|
|
|
call to this function, it will return None.
|
|
|
|
"""
|
|
|
|
|
|
|
|
def _count_messages(txn):
|
|
|
|
sql = """
|
|
|
|
SELECT COALESCE(COUNT(*), 0) FROM events
|
|
|
|
WHERE type = 'm.room.message'
|
|
|
|
AND stream_ordering > ?
|
|
|
|
"""
|
|
|
|
txn.execute(sql, (self.stream_ordering_day_ago,))
|
|
|
|
(count,) = txn.fetchone()
|
|
|
|
return count
|
|
|
|
|
2020-08-11 17:21:20 -04:00
|
|
|
return await self.db_pool.runInteraction("count_messages", _count_messages)
|
2020-05-13 08:38:22 -04:00
|
|
|
|
2020-08-11 17:21:20 -04:00
|
|
|
async def count_daily_sent_messages(self):
|
2020-05-13 08:38:22 -04:00
|
|
|
def _count_messages(txn):
|
|
|
|
# This is good enough as if you have silly characters in your own
|
|
|
|
# hostname then thats your own fault.
|
|
|
|
like_clause = "%:" + self.hs.hostname
|
|
|
|
|
|
|
|
sql = """
|
|
|
|
SELECT COALESCE(COUNT(*), 0) FROM events
|
|
|
|
WHERE type = 'm.room.message'
|
|
|
|
AND sender LIKE ?
|
|
|
|
AND stream_ordering > ?
|
|
|
|
"""
|
|
|
|
|
|
|
|
txn.execute(sql, (like_clause, self.stream_ordering_day_ago))
|
|
|
|
(count,) = txn.fetchone()
|
|
|
|
return count
|
|
|
|
|
2020-08-11 17:21:20 -04:00
|
|
|
return await self.db_pool.runInteraction(
|
2020-08-05 16:38:57 -04:00
|
|
|
"count_daily_sent_messages", _count_messages
|
|
|
|
)
|
2020-05-13 08:38:22 -04:00
|
|
|
|
2020-08-11 17:21:20 -04:00
|
|
|
async def count_daily_active_rooms(self):
|
2020-05-13 08:38:22 -04:00
|
|
|
def _count(txn):
|
|
|
|
sql = """
|
|
|
|
SELECT COALESCE(COUNT(DISTINCT room_id), 0) FROM events
|
|
|
|
WHERE type = 'm.room.message'
|
|
|
|
AND stream_ordering > ?
|
|
|
|
"""
|
|
|
|
txn.execute(sql, (self.stream_ordering_day_ago,))
|
|
|
|
(count,) = txn.fetchone()
|
|
|
|
return count
|
|
|
|
|
2020-08-11 17:21:20 -04:00
|
|
|
return await self.db_pool.runInteraction("count_daily_active_rooms", _count)
|