2022-05-24 16:39:54 +02:00
|
|
|
#
|
2023-11-21 21:29:58 +01:00
|
|
|
# This file is licensed under the Affero General Public License (AGPL) version 3.
|
|
|
|
#
|
2024-01-23 12:26:48 +01:00
|
|
|
# Copyright 2022 The Matrix.org Foundation C.I.C.
|
2023-11-21 21:29:58 +01:00
|
|
|
# Copyright (C) 2023 New Vector, Ltd
|
|
|
|
#
|
|
|
|
# This program is free software: you can redistribute it and/or modify
|
|
|
|
# it under the terms of the GNU Affero General Public License as
|
|
|
|
# published by the Free Software Foundation, either version 3 of the
|
|
|
|
# License, or (at your option) any later version.
|
|
|
|
#
|
|
|
|
# See the GNU Affero General Public License for more details:
|
|
|
|
# <https://www.gnu.org/licenses/agpl-3.0.html>.
|
|
|
|
#
|
|
|
|
# Originally licensed under the Apache License, Version 2.0:
|
|
|
|
# <http://www.apache.org/licenses/LICENSE-2.0>.
|
|
|
|
#
|
|
|
|
# [This file includes modifications made by New Vector Limited]
|
2022-05-24 16:39:54 +02:00
|
|
|
#
|
|
|
|
#
|
2023-08-03 20:43:51 +02:00
|
|
|
from typing import Any, List, Mapping, Optional, Sequence, Union
|
2022-05-24 16:39:54 +02:00
|
|
|
from unittest.mock import Mock
|
|
|
|
|
|
|
|
from twisted.test.proto_helpers import MemoryReactor
|
|
|
|
|
|
|
|
from synapse.appservice import ApplicationService
|
|
|
|
from synapse.server import HomeServer
|
|
|
|
from synapse.types import JsonDict
|
|
|
|
from synapse.util import Clock
|
|
|
|
|
|
|
|
from tests import unittest
|
2023-08-03 20:43:51 +02:00
|
|
|
from tests.unittest import override_config
|
2022-05-24 16:39:54 +02:00
|
|
|
|
|
|
|
PROTOCOL = "myproto"
|
|
|
|
TOKEN = "myastoken"
|
|
|
|
URL = "http://mytestservice"
|
|
|
|
|
|
|
|
|
|
|
|
class ApplicationServiceApiTestCase(unittest.HomeserverTestCase):
|
2023-02-06 13:49:06 +01:00
|
|
|
def prepare(self, reactor: MemoryReactor, clock: Clock, hs: HomeServer) -> None:
|
2022-05-24 16:39:54 +02:00
|
|
|
self.api = hs.get_application_service_api()
|
|
|
|
self.service = ApplicationService(
|
|
|
|
id="unique_identifier",
|
|
|
|
sender="@as:test",
|
|
|
|
url=URL,
|
|
|
|
token="unused",
|
|
|
|
hs_token=TOKEN,
|
|
|
|
)
|
|
|
|
|
2023-08-03 20:43:51 +02:00
|
|
|
def test_query_3pe_authenticates_token_via_header(self) -> None:
|
2022-05-24 16:39:54 +02:00
|
|
|
"""
|
|
|
|
Tests that 3pe queries to the appservice are authenticated
|
|
|
|
with the appservice's token.
|
|
|
|
"""
|
|
|
|
|
|
|
|
SUCCESS_RESULT_USER = [
|
|
|
|
{
|
|
|
|
"protocol": PROTOCOL,
|
|
|
|
"userid": "@a:user",
|
|
|
|
"fields": {
|
|
|
|
"more": "fields",
|
|
|
|
},
|
|
|
|
}
|
|
|
|
]
|
|
|
|
SUCCESS_RESULT_LOCATION = [
|
|
|
|
{
|
|
|
|
"protocol": PROTOCOL,
|
|
|
|
"alias": "#a:room",
|
|
|
|
"fields": {
|
|
|
|
"more": "fields",
|
|
|
|
},
|
|
|
|
}
|
|
|
|
]
|
|
|
|
|
2023-04-03 19:20:32 +02:00
|
|
|
URL_USER = f"{URL}/_matrix/app/v1/thirdparty/user/{PROTOCOL}"
|
|
|
|
URL_LOCATION = f"{URL}/_matrix/app/v1/thirdparty/location/{PROTOCOL}"
|
2022-05-24 16:39:54 +02:00
|
|
|
|
|
|
|
self.request_url = None
|
|
|
|
|
2022-10-04 13:06:41 +02:00
|
|
|
async def get_json(
|
2022-10-26 15:00:01 +02:00
|
|
|
url: str,
|
|
|
|
args: Mapping[Any, Any],
|
|
|
|
headers: Mapping[Union[str, bytes], Sequence[Union[str, bytes]]],
|
2022-10-04 13:06:41 +02:00
|
|
|
) -> List[JsonDict]:
|
2023-08-03 20:43:51 +02:00
|
|
|
# Ensure the access token is passed as a header.
|
2023-09-06 21:19:17 +02:00
|
|
|
if not headers or not headers.get(b"Authorization"):
|
2022-05-24 16:39:54 +02:00
|
|
|
raise RuntimeError("Access token not provided")
|
2023-08-03 20:43:51 +02:00
|
|
|
# ... and not as a query param
|
|
|
|
if b"access_token" in args:
|
|
|
|
raise RuntimeError(
|
|
|
|
"Access token should not be passed as a query param."
|
|
|
|
)
|
2022-05-24 16:39:54 +02:00
|
|
|
|
2023-09-06 21:19:17 +02:00
|
|
|
self.assertEqual(
|
|
|
|
headers.get(b"Authorization"), [f"Bearer {TOKEN}".encode()]
|
|
|
|
)
|
2023-08-03 20:43:51 +02:00
|
|
|
self.request_url = url
|
|
|
|
if url == URL_USER:
|
|
|
|
return SUCCESS_RESULT_USER
|
|
|
|
elif url == URL_LOCATION:
|
|
|
|
return SUCCESS_RESULT_LOCATION
|
|
|
|
else:
|
|
|
|
raise RuntimeError(
|
|
|
|
"URL provided was invalid. This should never be seen."
|
|
|
|
)
|
|
|
|
|
|
|
|
# We assign to a method, which mypy doesn't like.
|
2023-08-29 16:38:56 +02:00
|
|
|
self.api.get_json = Mock(side_effect=get_json) # type: ignore[method-assign]
|
2023-08-03 20:43:51 +02:00
|
|
|
|
|
|
|
result = self.get_success(
|
|
|
|
self.api.query_3pe(self.service, "user", PROTOCOL, {b"some": [b"field"]})
|
|
|
|
)
|
|
|
|
self.assertEqual(self.request_url, URL_USER)
|
|
|
|
self.assertEqual(result, SUCCESS_RESULT_USER)
|
|
|
|
result = self.get_success(
|
|
|
|
self.api.query_3pe(
|
|
|
|
self.service, "location", PROTOCOL, {b"some": [b"field"]}
|
|
|
|
)
|
|
|
|
)
|
|
|
|
self.assertEqual(self.request_url, URL_LOCATION)
|
|
|
|
self.assertEqual(result, SUCCESS_RESULT_LOCATION)
|
|
|
|
|
|
|
|
@override_config({"use_appservice_legacy_authorization": True})
|
|
|
|
def test_query_3pe_authenticates_token_via_param(self) -> None:
|
|
|
|
"""
|
|
|
|
Tests that 3pe queries to the appservice are authenticated
|
|
|
|
with the appservice's token.
|
|
|
|
"""
|
|
|
|
|
|
|
|
SUCCESS_RESULT_USER = [
|
|
|
|
{
|
|
|
|
"protocol": PROTOCOL,
|
|
|
|
"userid": "@a:user",
|
|
|
|
"fields": {
|
|
|
|
"more": "fields",
|
|
|
|
},
|
|
|
|
}
|
|
|
|
]
|
|
|
|
SUCCESS_RESULT_LOCATION = [
|
|
|
|
{
|
|
|
|
"protocol": PROTOCOL,
|
|
|
|
"alias": "#a:room",
|
|
|
|
"fields": {
|
|
|
|
"more": "fields",
|
|
|
|
},
|
|
|
|
}
|
|
|
|
]
|
|
|
|
|
|
|
|
URL_USER = f"{URL}/_matrix/app/v1/thirdparty/user/{PROTOCOL}"
|
|
|
|
URL_LOCATION = f"{URL}/_matrix/app/v1/thirdparty/location/{PROTOCOL}"
|
|
|
|
|
|
|
|
self.request_url = None
|
|
|
|
|
|
|
|
async def get_json(
|
|
|
|
url: str,
|
|
|
|
args: Mapping[Any, Any],
|
|
|
|
headers: Optional[
|
|
|
|
Mapping[Union[str, bytes], Sequence[Union[str, bytes]]]
|
|
|
|
] = None,
|
|
|
|
) -> List[JsonDict]:
|
|
|
|
# Ensure the access token is passed as a both a query param and in the headers.
|
|
|
|
if not args.get(b"access_token"):
|
|
|
|
raise RuntimeError("Access token should be provided in query params.")
|
2023-09-06 21:19:17 +02:00
|
|
|
if not headers or not headers.get(b"Authorization"):
|
2023-08-03 20:43:51 +02:00
|
|
|
raise RuntimeError("Access token should be provided in auth headers.")
|
|
|
|
|
2022-05-24 16:39:54 +02:00
|
|
|
self.assertEqual(args.get(b"access_token"), TOKEN)
|
2023-09-06 21:19:17 +02:00
|
|
|
self.assertEqual(
|
|
|
|
headers.get(b"Authorization"), [f"Bearer {TOKEN}".encode()]
|
|
|
|
)
|
2022-05-24 16:39:54 +02:00
|
|
|
self.request_url = url
|
|
|
|
if url == URL_USER:
|
|
|
|
return SUCCESS_RESULT_USER
|
|
|
|
elif url == URL_LOCATION:
|
|
|
|
return SUCCESS_RESULT_LOCATION
|
|
|
|
else:
|
|
|
|
raise RuntimeError(
|
|
|
|
"URL provided was invalid. This should never be seen."
|
|
|
|
)
|
|
|
|
|
|
|
|
# We assign to a method, which mypy doesn't like.
|
2023-08-29 16:38:56 +02:00
|
|
|
self.api.get_json = Mock(side_effect=get_json) # type: ignore[method-assign]
|
2022-05-24 16:39:54 +02:00
|
|
|
|
|
|
|
result = self.get_success(
|
|
|
|
self.api.query_3pe(self.service, "user", PROTOCOL, {b"some": [b"field"]})
|
|
|
|
)
|
|
|
|
self.assertEqual(self.request_url, URL_USER)
|
|
|
|
self.assertEqual(result, SUCCESS_RESULT_USER)
|
|
|
|
result = self.get_success(
|
|
|
|
self.api.query_3pe(
|
|
|
|
self.service, "location", PROTOCOL, {b"some": [b"field"]}
|
|
|
|
)
|
|
|
|
)
|
|
|
|
self.assertEqual(self.request_url, URL_LOCATION)
|
|
|
|
self.assertEqual(result, SUCCESS_RESULT_LOCATION)
|
2023-03-28 20:26:27 +02:00
|
|
|
|
|
|
|
def test_claim_keys(self) -> None:
|
|
|
|
"""
|
|
|
|
Tests that the /keys/claim response is properly parsed for missing
|
|
|
|
keys.
|
|
|
|
"""
|
|
|
|
|
|
|
|
RESPONSE: JsonDict = {
|
|
|
|
"@alice:example.org": {
|
|
|
|
"DEVICE_1": {
|
|
|
|
"signed_curve25519:AAAAHg": {
|
|
|
|
# We don't really care about the content of the keys,
|
|
|
|
# they get passed back transparently.
|
|
|
|
},
|
|
|
|
"signed_curve25519:BBBBHg": {},
|
|
|
|
},
|
|
|
|
"DEVICE_2": {"signed_curve25519:CCCCHg": {}},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
async def post_json_get_json(
|
|
|
|
uri: str,
|
|
|
|
post_json: Any,
|
|
|
|
headers: Mapping[Union[str, bytes], Sequence[Union[str, bytes]]],
|
|
|
|
) -> JsonDict:
|
|
|
|
# Ensure the access token is passed as both a header and query arg.
|
2023-09-06 21:19:17 +02:00
|
|
|
if not headers.get(b"Authorization"):
|
2023-03-28 20:26:27 +02:00
|
|
|
raise RuntimeError("Access token not provided")
|
|
|
|
|
2023-09-06 21:19:17 +02:00
|
|
|
self.assertEqual(
|
|
|
|
headers.get(b"Authorization"), [f"Bearer {TOKEN}".encode()]
|
|
|
|
)
|
2023-03-28 20:26:27 +02:00
|
|
|
return RESPONSE
|
|
|
|
|
|
|
|
# We assign to a method, which mypy doesn't like.
|
2023-08-29 16:38:56 +02:00
|
|
|
self.api.post_json_get_json = Mock(side_effect=post_json_get_json) # type: ignore[method-assign]
|
2023-03-28 20:26:27 +02:00
|
|
|
|
|
|
|
MISSING_KEYS = [
|
|
|
|
# Known user, known device, missing algorithm.
|
2023-04-27 18:57:46 +02:00
|
|
|
("@alice:example.org", "DEVICE_2", "xyz", 1),
|
2023-03-28 20:26:27 +02:00
|
|
|
# Known user, missing device.
|
2023-04-27 18:57:46 +02:00
|
|
|
("@alice:example.org", "DEVICE_3", "signed_curve25519", 1),
|
2023-03-28 20:26:27 +02:00
|
|
|
# Unknown user.
|
2023-04-27 18:57:46 +02:00
|
|
|
("@bob:example.org", "DEVICE_4", "signed_curve25519", 1),
|
2023-03-28 20:26:27 +02:00
|
|
|
]
|
|
|
|
|
|
|
|
claimed_keys, missing = self.get_success(
|
|
|
|
self.api.claim_client_keys(
|
|
|
|
self.service,
|
|
|
|
[
|
|
|
|
# Found devices
|
2023-04-27 18:57:46 +02:00
|
|
|
("@alice:example.org", "DEVICE_1", "signed_curve25519", 1),
|
|
|
|
("@alice:example.org", "DEVICE_2", "signed_curve25519", 1),
|
2023-03-28 20:26:27 +02:00
|
|
|
]
|
|
|
|
+ MISSING_KEYS,
|
|
|
|
)
|
|
|
|
)
|
|
|
|
|
|
|
|
self.assertEqual(claimed_keys, RESPONSE)
|
|
|
|
self.assertEqual(missing, MISSING_KEYS)
|