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

Associate a request_name with each verify request, for logging #5353

Merged
merged 1 commit into from
Jun 6, 2019
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 2 additions & 0 deletions changelog.d/5353.misc
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
Various improvements to debug logging.

112 changes: 68 additions & 44 deletions synapse/crypto/keyring.py
Original file line number Diff line number Diff line change
Expand Up @@ -60,9 +60,9 @@


@attr.s(slots=True, cmp=False)
class VerifyKeyRequest(object):
class VerifyJsonRequest(object):
"""
A request for a verify key to verify a JSON object.
A request to verify a JSON object.

Attributes:
server_name(str): The name of the server to verify against.
Expand All @@ -85,11 +85,15 @@ class VerifyKeyRequest(object):
"""

server_name = attr.ib()
key_ids = attr.ib()
json_object = attr.ib()
minimum_valid_until_ts = attr.ib()
request_name = attr.ib()
key_ids = attr.ib(init=False)
key_ready = attr.ib(default=attr.Factory(defer.Deferred))

def __attrs_post_init__(self):
self.key_ids = signature_ids(self.json_object, self.server_name)


class KeyLookupError(ValueError):
pass
Expand All @@ -114,7 +118,9 @@ def __init__(self, hs, key_fetchers=None):
# These are regular, logcontext-agnostic Deferreds.
self.key_downloads = {}

def verify_json_for_server(self, server_name, json_object, validity_time):
def verify_json_for_server(
self, server_name, json_object, validity_time, request_name
):
"""Verify that a JSON object has been signed by a given server

Args:
Expand All @@ -125,63 +131,86 @@ def verify_json_for_server(self, server_name, json_object, validity_time):
validity_time (int): timestamp at which we require the signing key to
be valid. (0 implies we don't care)

request_name (str): an identifier for this json object (eg, an event id)
for logging.

Returns:
Deferred[None]: completes if the the object was correctly signed, otherwise
errbacks with an error
"""
req = server_name, json_object, validity_time

return logcontext.make_deferred_yieldable(
self.verify_json_objects_for_server((req,))[0]
)
req = VerifyJsonRequest(server_name, json_object, validity_time, request_name)
requests = (req,)
return logcontext.make_deferred_yieldable(self._verify_objects(requests)[0])

def verify_json_objects_for_server(self, server_and_json):
"""Bulk verifies signatures of json objects, bulk fetching keys as
necessary.

Args:
server_and_json (iterable[Tuple[str, dict, int]):
Iterable of triplets of (server_name, json_object, validity_time)
validity_time is a timestamp at which the signing key must be valid.
server_and_json (iterable[Tuple[str, dict, int, str]):
Iterable of (server_name, json_object, validity_time, request_name)
tuples.

validity_time is a timestamp at which the signing key must be
valid.

request_name is an identifier for this json object (eg, an event id)
for logging.

Returns:
List<Deferred[None]>: for each input triplet, a deferred indicating success
or failure to verify each json object's signature for the given
server_name. The deferreds run their callbacks in the sentinel
logcontext.
"""
# a list of VerifyKeyRequests
verify_requests = []
return self._verify_objects(
VerifyJsonRequest(server_name, json_object, validity_time, request_name)
for server_name, json_object, validity_time, request_name in server_and_json
)

def _verify_objects(self, verify_requests):
"""Does the work of verify_json_[objects_]for_server


Args:
verify_requests (iterable[VerifyJsonRequest]):
Iterable of verification requests.

Returns:
List<Deferred[None]>: for each input item, a deferred indicating success
or failure to verify each json object's signature for the given
server_name. The deferreds run their callbacks in the sentinel
logcontext.
"""
# a list of VerifyJsonRequests which are awaiting a key lookup
key_lookups = []
handle = preserve_fn(_handle_key_deferred)

def process(server_name, json_object, validity_time):
def process(verify_request):
"""Process an entry in the request list

Given a (server_name, json_object, validity_time) triplet from the request
list, adds a key request to verify_requests, and returns a deferred which
Adds a key request to key_lookups, and returns a deferred which
will complete or fail (in the sentinel context) when verification completes.
"""
key_ids = signature_ids(json_object, server_name)

if not key_ids:
if not verify_request.key_ids:
return defer.fail(
SynapseError(
400, "Not signed by %s" % (server_name,), Codes.UNAUTHORIZED
400,
"Not signed by %s" % (verify_request.server_name,),
Codes.UNAUTHORIZED,
)
)

logger.debug(
"Verifying for %s with key_ids %s, min_validity %i",
server_name,
key_ids,
validity_time,
"Verifying %s for %s with key_ids %s, min_validity %i",
verify_request.request_name,
verify_request.server_name,
verify_request.key_ids,
verify_request.minimum_valid_until_ts,
)

# add the key request to the queue, but don't start it off yet.
verify_request = VerifyKeyRequest(
server_name, key_ids, json_object, validity_time
)
verify_requests.append(verify_request)
key_lookups.append(verify_request)

# now run _handle_key_deferred, which will wait for the key request
# to complete and then do the verification.
Expand All @@ -190,13 +219,10 @@ def process(server_name, json_object, validity_time):
# wrap it with preserve_fn (aka run_in_background)
return handle(verify_request)

results = [
process(server_name, json_object, validity_time)
for server_name, json_object, validity_time in server_and_json
]
results = [process(r) for r in verify_requests]

if verify_requests:
run_in_background(self._start_key_lookups, verify_requests)
if key_lookups:
run_in_background(self._start_key_lookups, key_lookups)

return results

Expand All @@ -207,7 +233,7 @@ def _start_key_lookups(self, verify_requests):
Once each fetch completes, verify_request.key_ready will be resolved.

Args:
verify_requests (List[VerifyKeyRequest]):
verify_requests (List[VerifyJsonRequest]):
"""

try:
Expand Down Expand Up @@ -308,7 +334,7 @@ def _get_server_verify_keys(self, verify_requests):
with a SynapseError if none of the keys are found.

Args:
verify_requests (list[VerifyKeyRequest]): list of verify requests
verify_requests (list[VerifyJsonRequest]): list of verify requests
"""

remaining_requests = set(
Expand Down Expand Up @@ -357,7 +383,7 @@ def _attempt_key_fetches_with_fetcher(self, fetcher, remaining_requests):

Args:
fetcher (KeyFetcher): fetcher to use to fetch the keys
remaining_requests (set[VerifyKeyRequest]): outstanding key requests.
remaining_requests (set[VerifyJsonRequest]): outstanding key requests.
Any successfully-completed requests will be removed from the list.
"""
# dict[str, dict[str, int]]: keys to fetch.
Expand All @@ -376,7 +402,7 @@ def _attempt_key_fetches_with_fetcher(self, fetcher, remaining_requests):
# the requests.
keys_for_server[key_id] = max(
keys_for_server.get(key_id, -1),
verify_request.minimum_valid_until_ts
verify_request.minimum_valid_until_ts,
)

results = yield fetcher.get_keys(missing_keys)
Expand All @@ -386,7 +412,7 @@ def _attempt_key_fetches_with_fetcher(self, fetcher, remaining_requests):
server_name = verify_request.server_name

# see if any of the keys we got this time are sufficient to
# complete this VerifyKeyRequest.
# complete this VerifyJsonRequest.
result_keys = results.get(server_name, {})
for key_id in verify_request.key_ids:
fetch_key_result = result_keys.get(key_id)
Expand Down Expand Up @@ -454,9 +480,7 @@ def __init__(self, hs):
self.config = hs.get_config()

@defer.inlineCallbacks
def process_v2_response(
self, from_server, response_json, time_added_ms
):
def process_v2_response(self, from_server, response_json, time_added_ms):
"""Parse a 'Server Keys' structure from the result of a /key request

This is used to parse either the entirety of the response from
Expand Down Expand Up @@ -852,7 +876,7 @@ def _handle_key_deferred(verify_request):
"""Waits for the key to become available, and then performs a verification

Args:
verify_request (VerifyKeyRequest):
verify_request (VerifyJsonRequest):

Returns:
Deferred[None]
Expand Down
2 changes: 2 additions & 0 deletions synapse/federation/federation_base.py
Original file line number Diff line number Diff line change
Expand Up @@ -271,6 +271,7 @@ def _check_sigs_on_pdus(keyring, room_version, pdus):
p.sender_domain,
p.redacted_pdu_json,
p.pdu.origin_server_ts if v.enforce_key_validity else 0,
p.pdu.event_id,
)
for p in pdus_to_check_sender
]
Expand Down Expand Up @@ -306,6 +307,7 @@ def sender_err(e, pdu_to_check):
get_domain_from_id(p.pdu.event_id),
p.redacted_pdu_json,
p.pdu.origin_server_ts if v.enforce_key_validity else 0,
p.pdu.event_id,
)
for p in pdus_to_check_event_id
]
Expand Down
4 changes: 3 additions & 1 deletion synapse/federation/transport/server.py
Original file line number Diff line number Diff line change
Expand Up @@ -140,7 +140,9 @@ def authenticate_request(self, request, content):
401, "Missing Authorization headers", Codes.UNAUTHORIZED,
)

yield self.keyring.verify_json_for_server(origin, json_request, now)
yield self.keyring.verify_json_for_server(
origin, json_request, now, "Incoming request"
)

logger.info("Request from %s", origin)
request.authenticated_entity = origin
Expand Down
4 changes: 3 additions & 1 deletion synapse/groups/attestations.py
Original file line number Diff line number Diff line change
Expand Up @@ -101,7 +101,9 @@ def verify_attestation(self, attestation, group_id, user_id, server_name=None):
if valid_until_ms < now:
raise SynapseError(400, "Attestation expired")

yield self.keyring.verify_json_for_server(server_name, attestation, now)
yield self.keyring.verify_json_for_server(
server_name, attestation, now, "Group attestation"
)

def create_attestation(self, group_id, user_id):
"""Create an attestation for the group_id and user_id with default
Expand Down
18 changes: 8 additions & 10 deletions tests/crypto/test_keyring.py
Original file line number Diff line number Diff line change
Expand Up @@ -134,7 +134,7 @@ def first_lookup():
context_11.request = "11"

res_deferreds = kr.verify_json_objects_for_server(
[("server10", json1, 0), ("server11", {}, 0)]
[("server10", json1, 0, "test10"), ("server11", {}, 0, "test11")]
)

# the unsigned json should be rejected pretty quickly
Expand Down Expand Up @@ -171,7 +171,7 @@ def second_lookup():
self.http_client.post_json.return_value = defer.Deferred()

res_deferreds_2 = kr.verify_json_objects_for_server(
[("server10", json1, 0)]
[("server10", json1, 0, "test")]
)
res_deferreds_2[0].addBoth(self.check_context, None)
yield logcontext.make_deferred_yieldable(res_deferreds_2[0])
Expand Down Expand Up @@ -205,11 +205,11 @@ def test_verify_json_for_server(self):
signedjson.sign.sign_json(json1, "server9", key1)

# should fail immediately on an unsigned object
d = _verify_json_for_server(kr, "server9", {}, 0)
d = _verify_json_for_server(kr, "server9", {}, 0, "test unsigned")
self.failureResultOf(d, SynapseError)

# should suceed on a signed object
d = _verify_json_for_server(kr, "server9", json1, 500)
d = _verify_json_for_server(kr, "server9", json1, 500, "test signed")
# self.assertFalse(d.called)
self.get_success(d)

Expand Down Expand Up @@ -239,7 +239,7 @@ def get_keys(keys_to_fetch):
# the first request should succeed; the second should fail because the key
# has expired
results = kr.verify_json_objects_for_server(
[("server1", json1, 500), ("server1", json1, 1500)]
[("server1", json1, 500, "test1"), ("server1", json1, 1500, "test2")]
)
self.assertEqual(len(results), 2)
self.get_success(results[0])
Expand Down Expand Up @@ -284,7 +284,7 @@ def get_keys2(keys_to_fetch):
signedjson.sign.sign_json(json1, "server1", key1)

results = kr.verify_json_objects_for_server(
[("server1", json1, 1200), ("server1", json1, 1500)]
[("server1", json1, 1200, "test1"), ("server1", json1, 1500, "test2")]
)
self.assertEqual(len(results), 2)
self.get_success(results[0])
Expand Down Expand Up @@ -522,16 +522,14 @@ def run_in_context(f, *args, **kwargs):
defer.returnValue(rv)


def _verify_json_for_server(keyring, server_name, json_object, validity_time):
def _verify_json_for_server(kr, *args):
"""thin wrapper around verify_json_for_server which makes sure it is wrapped
with the patched defer.inlineCallbacks.
"""

@defer.inlineCallbacks
def v():
rv1 = yield keyring.verify_json_for_server(
server_name, json_object, validity_time
)
rv1 = yield kr.verify_json_for_server(*args)
defer.returnValue(rv1)

return run_in_context(v)