Merge pull request #5353 from matrix-org/rav/verify_key_logging

Associate a request_name with each verify request, for logging
This commit is contained in:
Richard van der Hoff 2019-06-06 09:33:09 +01:00 committed by GitHub
commit 2eb47e5ee7
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
6 changed files with 86 additions and 56 deletions

2
changelog.d/5353.misc Normal file
View File

@ -0,0 +1,2 @@
Various improvements to debug logging.

View File

@ -60,9 +60,9 @@ logger = logging.getLogger(__name__)
@attr.s(slots=True, cmp=False) @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: Attributes:
server_name(str): The name of the server to verify against. server_name(str): The name of the server to verify against.
@ -85,11 +85,15 @@ class VerifyKeyRequest(object):
""" """
server_name = attr.ib() server_name = attr.ib()
key_ids = attr.ib()
json_object = attr.ib() json_object = attr.ib()
minimum_valid_until_ts = 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)) 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): class KeyLookupError(ValueError):
pass pass
@ -114,7 +118,9 @@ class Keyring(object):
# These are regular, logcontext-agnostic Deferreds. # These are regular, logcontext-agnostic Deferreds.
self.key_downloads = {} 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 """Verify that a JSON object has been signed by a given server
Args: Args:
@ -125,24 +131,31 @@ class Keyring(object):
validity_time (int): timestamp at which we require the signing key to validity_time (int): timestamp at which we require the signing key to
be valid. (0 implies we don't care) be valid. (0 implies we don't care)
request_name (str): an identifier for this json object (eg, an event id)
for logging.
Returns: Returns:
Deferred[None]: completes if the the object was correctly signed, otherwise Deferred[None]: completes if the the object was correctly signed, otherwise
errbacks with an error errbacks with an error
""" """
req = server_name, json_object, validity_time req = VerifyJsonRequest(server_name, json_object, validity_time, request_name)
requests = (req,)
return logcontext.make_deferred_yieldable( return logcontext.make_deferred_yieldable(self._verify_objects(requests)[0])
self.verify_json_objects_for_server((req,))[0]
)
def verify_json_objects_for_server(self, server_and_json): def verify_json_objects_for_server(self, server_and_json):
"""Bulk verifies signatures of json objects, bulk fetching keys as """Bulk verifies signatures of json objects, bulk fetching keys as
necessary. necessary.
Args: Args:
server_and_json (iterable[Tuple[str, dict, int]): server_and_json (iterable[Tuple[str, dict, int, str]):
Iterable of triplets of (server_name, json_object, validity_time) Iterable of (server_name, json_object, validity_time, request_name)
validity_time is a timestamp at which the signing key must be valid. 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: Returns:
List<Deferred[None]>: for each input triplet, a deferred indicating success List<Deferred[None]>: for each input triplet, a deferred indicating success
@ -150,38 +163,54 @@ class Keyring(object):
server_name. The deferreds run their callbacks in the sentinel server_name. The deferreds run their callbacks in the sentinel
logcontext. logcontext.
""" """
# a list of VerifyKeyRequests return self._verify_objects(
verify_requests = [] 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) 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 """Process an entry in the request list
Given a (server_name, json_object, validity_time) triplet from the request Adds a key request to key_lookups, and returns a deferred which
list, adds a key request to verify_requests, and returns a deferred which
will complete or fail (in the sentinel context) when verification completes. will complete or fail (in the sentinel context) when verification completes.
""" """
key_ids = signature_ids(json_object, server_name) if not verify_request.key_ids:
if not key_ids:
return defer.fail( return defer.fail(
SynapseError( SynapseError(
400, "Not signed by %s" % (server_name,), Codes.UNAUTHORIZED 400,
"Not signed by %s" % (verify_request.server_name,),
Codes.UNAUTHORIZED,
) )
) )
logger.debug( logger.debug(
"Verifying for %s with key_ids %s, min_validity %i", "Verifying %s for %s with key_ids %s, min_validity %i",
server_name, verify_request.request_name,
key_ids, verify_request.server_name,
validity_time, verify_request.key_ids,
verify_request.minimum_valid_until_ts,
) )
# add the key request to the queue, but don't start it off yet. # add the key request to the queue, but don't start it off yet.
verify_request = VerifyKeyRequest( key_lookups.append(verify_request)
server_name, key_ids, json_object, validity_time
)
verify_requests.append(verify_request)
# now run _handle_key_deferred, which will wait for the key request # now run _handle_key_deferred, which will wait for the key request
# to complete and then do the verification. # to complete and then do the verification.
@ -190,13 +219,10 @@ class Keyring(object):
# wrap it with preserve_fn (aka run_in_background) # wrap it with preserve_fn (aka run_in_background)
return handle(verify_request) return handle(verify_request)
results = [ results = [process(r) for r in verify_requests]
process(server_name, json_object, validity_time)
for server_name, json_object, validity_time in server_and_json
]
if verify_requests: if key_lookups:
run_in_background(self._start_key_lookups, verify_requests) run_in_background(self._start_key_lookups, key_lookups)
return results return results
@ -207,7 +233,7 @@ class Keyring(object):
Once each fetch completes, verify_request.key_ready will be resolved. Once each fetch completes, verify_request.key_ready will be resolved.
Args: Args:
verify_requests (List[VerifyKeyRequest]): verify_requests (List[VerifyJsonRequest]):
""" """
try: try:
@ -308,7 +334,7 @@ class Keyring(object):
with a SynapseError if none of the keys are found. with a SynapseError if none of the keys are found.
Args: Args:
verify_requests (list[VerifyKeyRequest]): list of verify requests verify_requests (list[VerifyJsonRequest]): list of verify requests
""" """
remaining_requests = set( remaining_requests = set(
@ -357,7 +383,7 @@ class Keyring(object):
Args: Args:
fetcher (KeyFetcher): fetcher to use to fetch the keys 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. Any successfully-completed requests will be removed from the list.
""" """
# dict[str, dict[str, int]]: keys to fetch. # dict[str, dict[str, int]]: keys to fetch.
@ -376,7 +402,7 @@ class Keyring(object):
# the requests. # the requests.
keys_for_server[key_id] = max( keys_for_server[key_id] = max(
keys_for_server.get(key_id, -1), 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) results = yield fetcher.get_keys(missing_keys)
@ -386,7 +412,7 @@ class Keyring(object):
server_name = verify_request.server_name server_name = verify_request.server_name
# see if any of the keys we got this time are sufficient to # 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, {}) result_keys = results.get(server_name, {})
for key_id in verify_request.key_ids: for key_id in verify_request.key_ids:
fetch_key_result = result_keys.get(key_id) fetch_key_result = result_keys.get(key_id)
@ -454,9 +480,7 @@ class BaseV2KeyFetcher(object):
self.config = hs.get_config() self.config = hs.get_config()
@defer.inlineCallbacks @defer.inlineCallbacks
def process_v2_response( def process_v2_response(self, from_server, response_json, time_added_ms):
self, from_server, response_json, time_added_ms
):
"""Parse a 'Server Keys' structure from the result of a /key request """Parse a 'Server Keys' structure from the result of a /key request
This is used to parse either the entirety of the response from This is used to parse either the entirety of the response from
@ -852,7 +876,7 @@ def _handle_key_deferred(verify_request):
"""Waits for the key to become available, and then performs a verification """Waits for the key to become available, and then performs a verification
Args: Args:
verify_request (VerifyKeyRequest): verify_request (VerifyJsonRequest):
Returns: Returns:
Deferred[None] Deferred[None]

View File

@ -271,6 +271,7 @@ def _check_sigs_on_pdus(keyring, room_version, pdus):
p.sender_domain, p.sender_domain,
p.redacted_pdu_json, p.redacted_pdu_json,
p.pdu.origin_server_ts if v.enforce_key_validity else 0, p.pdu.origin_server_ts if v.enforce_key_validity else 0,
p.pdu.event_id,
) )
for p in pdus_to_check_sender for p in pdus_to_check_sender
] ]
@ -306,6 +307,7 @@ def _check_sigs_on_pdus(keyring, room_version, pdus):
get_domain_from_id(p.pdu.event_id), get_domain_from_id(p.pdu.event_id),
p.redacted_pdu_json, p.redacted_pdu_json,
p.pdu.origin_server_ts if v.enforce_key_validity else 0, p.pdu.origin_server_ts if v.enforce_key_validity else 0,
p.pdu.event_id,
) )
for p in pdus_to_check_event_id for p in pdus_to_check_event_id
] ]

View File

@ -140,7 +140,9 @@ class Authenticator(object):
401, "Missing Authorization headers", Codes.UNAUTHORIZED, 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) logger.info("Request from %s", origin)
request.authenticated_entity = origin request.authenticated_entity = origin

View File

@ -101,7 +101,9 @@ class GroupAttestationSigning(object):
if valid_until_ms < now: if valid_until_ms < now:
raise SynapseError(400, "Attestation expired") 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): def create_attestation(self, group_id, user_id):
"""Create an attestation for the group_id and user_id with default """Create an attestation for the group_id and user_id with default

View File

@ -134,7 +134,7 @@ class KeyringTestCase(unittest.HomeserverTestCase):
context_11.request = "11" context_11.request = "11"
res_deferreds = kr.verify_json_objects_for_server( 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 # the unsigned json should be rejected pretty quickly
@ -171,7 +171,7 @@ class KeyringTestCase(unittest.HomeserverTestCase):
self.http_client.post_json.return_value = defer.Deferred() self.http_client.post_json.return_value = defer.Deferred()
res_deferreds_2 = kr.verify_json_objects_for_server( 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) res_deferreds_2[0].addBoth(self.check_context, None)
yield logcontext.make_deferred_yieldable(res_deferreds_2[0]) yield logcontext.make_deferred_yieldable(res_deferreds_2[0])
@ -205,11 +205,11 @@ class KeyringTestCase(unittest.HomeserverTestCase):
signedjson.sign.sign_json(json1, "server9", key1) signedjson.sign.sign_json(json1, "server9", key1)
# should fail immediately on an unsigned object # 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) self.failureResultOf(d, SynapseError)
# should suceed on a signed object # 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.assertFalse(d.called)
self.get_success(d) self.get_success(d)
@ -239,7 +239,7 @@ class KeyringTestCase(unittest.HomeserverTestCase):
# the first request should succeed; the second should fail because the key # the first request should succeed; the second should fail because the key
# has expired # has expired
results = kr.verify_json_objects_for_server( 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.assertEqual(len(results), 2)
self.get_success(results[0]) self.get_success(results[0])
@ -284,7 +284,7 @@ class KeyringTestCase(unittest.HomeserverTestCase):
signedjson.sign.sign_json(json1, "server1", key1) signedjson.sign.sign_json(json1, "server1", key1)
results = kr.verify_json_objects_for_server( 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.assertEqual(len(results), 2)
self.get_success(results[0]) self.get_success(results[0])
@ -522,16 +522,14 @@ def run_in_context(f, *args, **kwargs):
defer.returnValue(rv) 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 """thin wrapper around verify_json_for_server which makes sure it is wrapped
with the patched defer.inlineCallbacks. with the patched defer.inlineCallbacks.
""" """
@defer.inlineCallbacks @defer.inlineCallbacks
def v(): def v():
rv1 = yield keyring.verify_json_for_server( rv1 = yield kr.verify_json_for_server(*args)
server_name, json_object, validity_time
)
defer.returnValue(rv1) defer.returnValue(rv1)
return run_in_context(v) return run_in_context(v)