add multi-token support

This commit is contained in:
holger krekel
2024-03-27 15:03:59 +01:00
parent 312f86223c
commit 0a93c76e66
2 changed files with 63 additions and 28 deletions

View File

@@ -43,20 +43,30 @@ class Notifier:
metadata_dir = self.get_metadata_dir(mbox) metadata_dir = self.get_metadata_dir(mbox)
token_path = metadata_dir / METADATA_TOKEN_KEY token_path = metadata_dir / METADATA_TOKEN_KEY
write_path = token_path.with_suffix(".tmp") write_path = token_path.with_suffix(".tmp")
write_path.write_text(token) tokens = []
if token_path.exists():
tokens = token_path.read_text().split() + [token]
if token not in tokens:
tokens.append(token)
write_path.write_text(" ".join(tokens))
write_path.rename(token_path) write_path.rename(token_path)
def del_token(self, mbox): def del_token(self, mbox, token):
metadata_dir = self.get_metadata_dir(mbox) tokens = self.get_tokens(mbox)
if metadata_dir is not None: if token in tokens:
metadata_dir.joinpath(METADATA_TOKEN_KEY).unlink(missing_ok=True) tokens.remove(token)
token_path = self.get_metadata_dir(mbox) / METADATA_TOKEN_KEY
write_path = token_path.with_suffix(".tmp")
write_path.write_text(" ".join(tokens))
write_path.rename(token_path)
def get_token(self, mbox): def get_tokens(self, mbox):
metadata_dir = self.get_metadata_dir(mbox) metadata_dir = self.get_metadata_dir(mbox)
if metadata_dir is not None: if metadata_dir is not None:
token_path = metadata_dir / METADATA_TOKEN_KEY token_path = metadata_dir / METADATA_TOKEN_KEY
if token_path.exists(): if token_path.exists():
return token_path.read_text() return token_path.read_text().split()
return []
def new_message_for_mbox(self, mbox): def new_message_for_mbox(self, mbox):
self.to_notify_queue.put(mbox) self.to_notify_queue.put(mbox)
@@ -68,8 +78,7 @@ class Notifier:
def thread_run_one(self, requests_session): def thread_run_one(self, requests_session):
mbox = self.to_notify_queue.get() mbox = self.to_notify_queue.get()
token = self.get_token(mbox) for token in self.get_tokens(mbox):
if token:
response = requests_session.post( response = requests_session.post(
"https://notifications.delta.chat/notify", "https://notifications.delta.chat/notify",
data=token, data=token,
@@ -78,7 +87,7 @@ class Notifier:
if response.status_code == 410: if response.status_code == 410:
# 410 Gone status code # 410 Gone status code
# means the token is no longer valid. # means the token is no longer valid.
self.del_token(mbox) self.del_token(mbox, token)
def handle_dovecot_protocol(rfile, wfile, notifier): def handle_dovecot_protocol(rfile, wfile, notifier):
@@ -109,7 +118,8 @@ def handle_dovecot_request(msg, transactions, notifier):
keyname = keyparts[2] keyname = keyparts[2]
mbox = parts[1] mbox = parts[1]
if keyname == METADATA_TOKEN_KEY: if keyname == METADATA_TOKEN_KEY:
return f"O{notifier.get_token(mbox)}\n" res = " ".join(notifier.get_tokens(mbox))
return f"O{res}\n"
logging.warning("lookup ignored: %r", msg) logging.warning("lookup ignored: %r", msg)
return "N\n" return "N\n"
elif short_command == DICTPROXY_ITERATE_CHAR: elif short_command == DICTPROXY_ITERATE_CHAR:

View File

@@ -22,20 +22,20 @@ def test_notifier_persistence(tmp_path):
notifier1 = Notifier(vmail_dir) notifier1 = Notifier(vmail_dir)
notifier2 = Notifier(vmail_dir) notifier2 = Notifier(vmail_dir)
assert notifier1.get_token("user1@example.org") is None assert not notifier1.get_tokens("user1@example.org")
assert notifier2.get_token("user1@example.org") is None assert not notifier2.get_tokens("user1@example.org")
notifier1.set_token("user1@example.org", "01234") notifier1.set_token("user1@example.org", "01234")
notifier1.set_token("user3@example.org", "456") notifier1.set_token("user3@example.org", "456")
assert notifier2.get_token("user1@example.org") == "01234" assert notifier2.get_tokens("user1@example.org") == ["01234"]
assert notifier2.get_token("user3@example.org") == "456" assert notifier2.get_tokens("user3@example.org") == ["456"]
notifier2.del_token("user1@example.org") notifier2.del_token("user1@example.org", "01234")
assert notifier1.get_token("user1@example.org") is None assert not notifier1.get_tokens("user1@example.org")
def test_notifier_delete_without_set(notifier): def test_notifier_delete_without_set(notifier):
notifier.del_token("user@example.org") notifier.del_token("user@example.org", "123")
assert not notifier.get_token("user@example.org") assert not notifier.get_tokens("user@example.org")
def test_handle_dovecot_request_lookup_fails(notifier): def test_handle_dovecot_request_lookup_fails(notifier):
@@ -50,20 +50,20 @@ def test_handle_dovecot_request_happy_path(notifier):
tx = "1111" tx = "1111"
msg = f"B{tx}\tuser@example.org" msg = f"B{tx}\tuser@example.org"
res = handle_dovecot_request(msg, transactions, notifier) res = handle_dovecot_request(msg, transactions, notifier)
assert not res and notifier.get_token("user@example.org") is None assert not res and not notifier.get_tokens("user@example.org")
assert transactions == {tx: dict(mbox="user@example.org", res="O\n")} assert transactions == {tx: dict(mbox="user@example.org", res="O\n")}
msg = f"S{tx}\tpriv/guid00/devicetoken\t01234" msg = f"S{tx}\tpriv/guid00/devicetoken\t01234"
res = handle_dovecot_request(msg, transactions, notifier) res = handle_dovecot_request(msg, transactions, notifier)
assert not res assert not res
assert len(transactions) == 1 assert len(transactions) == 1
assert notifier.get_token("user@example.org") == "01234" assert notifier.get_tokens("user@example.org") == ["01234"]
msg = f"C{tx}" msg = f"C{tx}"
res = handle_dovecot_request(msg, transactions, notifier) res = handle_dovecot_request(msg, transactions, notifier)
assert res == "O\n" assert res == "O\n"
assert len(transactions) == 0 assert len(transactions) == 0
assert notifier.get_token("user@example.org") == "01234" assert notifier.get_tokens("user@example.org") == ["01234"]
# trigger notification for incoming message # trigger notification for incoming message
assert ( assert (
@@ -92,7 +92,7 @@ def test_handle_dovecot_protocol_set_devicetoken(notifier):
wfile = io.BytesIO() wfile = io.BytesIO()
handle_dovecot_protocol(rfile, wfile, notifier) handle_dovecot_protocol(rfile, wfile, notifier)
assert wfile.getvalue() == b"O\n" assert wfile.getvalue() == b"O\n"
assert notifier.get_token("user@example.org") == "01234" assert notifier.get_tokens("user@example.org") == ["01234"]
def test_handle_dovecot_protocol_set_get_devicetoken(notifier): def test_handle_dovecot_protocol_set_get_devicetoken(notifier):
@@ -108,7 +108,7 @@ def test_handle_dovecot_protocol_set_get_devicetoken(notifier):
) )
wfile = io.BytesIO() wfile = io.BytesIO()
handle_dovecot_protocol(rfile, wfile, notifier) handle_dovecot_protocol(rfile, wfile, notifier)
assert notifier.get_token("user@example.org") == "01234" assert notifier.get_tokens("user@example.org") == ["01234"]
assert wfile.getvalue() == b"O\n" assert wfile.getvalue() == b"O\n"
rfile = io.BytesIO( rfile = io.BytesIO(
@@ -168,7 +168,29 @@ def test_notifier_thread_run(notifier):
notifier.thread_run_one(ReqMock()) notifier.thread_run_one(ReqMock())
url, data, timeout = requests[0] url, data, timeout = requests[0]
assert data == "01234" assert data == "01234"
assert notifier.get_token("user@example.org") == "01234" assert notifier.get_tokens("user@example.org") == ["01234"]
def test_multi_device_notifier(notifier):
requests = []
class ReqMock:
def post(self, url, data, timeout):
requests.append((url, data, timeout))
class Result:
status_code = 200
return Result()
notifier.set_token("user@example.org", "01234")
notifier.set_token("user@example.org", "56789")
notifier.new_message_for_mbox("user@example.org")
notifier.thread_run_one(ReqMock())
url, data, timeout = requests[0]
assert data == "01234"
url, data, timeout = requests[1]
assert data == "56789"
def test_notifier_thread_run_gone_removes_token(notifier): def test_notifier_thread_run_gone_removes_token(notifier):
@@ -179,14 +201,17 @@ def test_notifier_thread_run_gone_removes_token(notifier):
requests.append((url, data, timeout)) requests.append((url, data, timeout))
class Result: class Result:
status_code = 410 status_code = 410 if data == "01234" else 200
return Result() return Result()
notifier.set_token("user@example.org", "01234") notifier.set_token("user@example.org", "01234")
notifier.new_message_for_mbox("user@example.org") notifier.new_message_for_mbox("user@example.org")
assert notifier.get_token("user@example.org") == "01234" assert notifier.get_tokens("user@example.org") == ["01234"]
notifier.set_token("user@example.org", "45678")
notifier.thread_run_one(ReqMock()) notifier.thread_run_one(ReqMock())
url, data, timeout = requests[0] url, data, timeout = requests[0]
assert data == "01234" assert data == "01234"
assert notifier.get_token("user@example.org") is None url, data, timeout = requests[1]
assert data == "45678"
assert notifier.get_tokens("user@example.org") == ["45678"]