Skip to content
This repository was archived by the owner on Apr 26, 2024. It is now read-only.

Commit 7af6f9d

Browse files
anoadragon453H-Shay
authored andcommitted
Fix an invalid comparison of UserPresenceState to str (#14393)
1 parent a3b35ee commit 7af6f9d

File tree

5 files changed

+46
-8
lines changed

5 files changed

+46
-8
lines changed

changelog.d/14393.bugfix

+1
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
Fix a bug introduced in 1.58.0 where a user with presence state 'org.matrix.msc3026.busy' would mistakenly be set to 'online' when calling `/sync` or `/events` on a worker process.

synapse/handlers/presence.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -478,7 +478,7 @@ async def user_syncing(
478478
return _NullContextManager()
479479

480480
prev_state = await self.current_state_for_user(user_id)
481-
if prev_state != PresenceState.BUSY:
481+
if prev_state.state != PresenceState.BUSY:
482482
# We set state here but pass ignore_status_msg = True as we don't want to
483483
# cause the status message to be cleared.
484484
# Note that this causes last_active_ts to be incremented which is not

tests/handlers/test_presence.py

+35-6
Original file line numberDiff line numberDiff line change
@@ -15,6 +15,7 @@
1515
from typing import Optional
1616
from unittest.mock import Mock, call
1717

18+
from parameterized import parameterized
1819
from signedjson.key import generate_signing_key
1920

2021
from synapse.api.constants import EventTypes, Membership, PresenceState
@@ -37,6 +38,7 @@
3738
from synapse.types import UserID, get_domain_from_id
3839

3940
from tests import unittest
41+
from tests.replication._base import BaseMultiWorkerStreamTestCase
4042

4143

4244
class PresenceUpdateTestCase(unittest.HomeserverTestCase):
@@ -505,7 +507,7 @@ def test_last_active(self):
505507
self.assertEqual(state, new_state)
506508

507509

508-
class PresenceHandlerTestCase(unittest.HomeserverTestCase):
510+
class PresenceHandlerTestCase(BaseMultiWorkerStreamTestCase):
509511
def prepare(self, reactor, clock, hs):
510512
self.presence_handler = hs.get_presence_handler()
511513
self.clock = hs.get_clock()
@@ -716,20 +718,47 @@ def test_set_presence_from_syncing_keeps_status(self):
716718
# our status message should be the same as it was before
717719
self.assertEqual(state.status_msg, status_msg)
718720

719-
def test_set_presence_from_syncing_keeps_busy(self):
720-
"""Test that presence set by syncing doesn't affect busy status"""
721-
# while this isn't the default
722-
self.presence_handler._busy_presence_enabled = True
721+
@parameterized.expand([(False,), (True,)])
722+
@unittest.override_config(
723+
{
724+
"experimental_features": {
725+
"msc3026_enabled": True,
726+
},
727+
}
728+
)
729+
def test_set_presence_from_syncing_keeps_busy(self, test_with_workers: bool):
730+
"""Test that presence set by syncing doesn't affect busy status
723731
732+
Args:
733+
test_with_workers: If True, check the presence state of the user by calling
734+
/sync against a worker, rather than the main process.
735+
"""
724736
user_id = "@test:server"
725737
status_msg = "I'm busy!"
726738

739+
# By default, we call /sync against the main process.
740+
worker_to_sync_against = self.hs
741+
if test_with_workers:
742+
# Create a worker and use it to handle /sync traffic instead.
743+
# This is used to test that presence changes get replicated from workers
744+
# to the main process correctly.
745+
worker_to_sync_against = self.make_worker_hs(
746+
"synapse.app.generic_worker", {"worker_name": "presence_writer"}
747+
)
748+
749+
# Set presence to BUSY
727750
self._set_presencestate_with_status_msg(user_id, PresenceState.BUSY, status_msg)
728751

752+
# Perform a sync with a presence state other than busy. This should NOT change
753+
# our presence status; we only change from busy if we explicitly set it via
754+
# /presence/*.
729755
self.get_success(
730-
self.presence_handler.user_syncing(user_id, True, PresenceState.ONLINE)
756+
worker_to_sync_against.get_presence_handler().user_syncing(
757+
user_id, True, PresenceState.ONLINE
758+
)
731759
)
732760

761+
# Check against the main process that the user's presence did not change.
733762
state = self.get_success(
734763
self.presence_handler.get_state(UserID.from_string(user_id))
735764
)

tests/module_api/test_api.py

+3
Original file line numberDiff line numberDiff line change
@@ -778,8 +778,11 @@ def _test_sending_local_online_presence_to_local_user(
778778
worker process. The test users will still sync with the main process. The purpose of testing
779779
with a worker is to check whether a Synapse module running on a worker can inform other workers/
780780
the main process that they should include additional presence when a user next syncs.
781+
If this argument is True, `test_case` MUST be an instance of BaseMultiWorkerStreamTestCase.
781782
"""
782783
if test_with_workers:
784+
assert isinstance(test_case, BaseMultiWorkerStreamTestCase)
785+
783786
# Create a worker process to make module_api calls against
784787
worker_hs = test_case.make_worker_hs(
785788
"synapse.app.generic_worker", {"worker_name": "presence_writer"}

tests/replication/_base.py

+6-1
Original file line numberDiff line numberDiff line change
@@ -542,8 +542,13 @@ def handle_command(self, command, *args):
542542
self.send("OK")
543543
elif command == b"GET":
544544
self.send(None)
545+
546+
# Connection keep-alives.
547+
elif command == b"PING":
548+
self.send("PONG")
549+
545550
else:
546-
raise Exception("Unknown command")
551+
raise Exception(f"Unknown command: {command}")
547552

548553
def send(self, msg):
549554
"""Send a message back to the client."""

0 commit comments

Comments
 (0)