From 33d5fb5f8548f29fba7049e4a31003cf2dfe9f31 Mon Sep 17 00:00:00 2001 From: Daniel Bluhm Date: Tue, 5 Sep 2023 20:26:12 -0400 Subject: [PATCH 1/2] feat: add timeout to did resolver resolve method Signed-off-by: Daniel Bluhm --- aries_cloudagent/resolver/did_resolver.py | 30 +++++++++++++++-------- 1 file changed, 20 insertions(+), 10 deletions(-) diff --git a/aries_cloudagent/resolver/did_resolver.py b/aries_cloudagent/resolver/did_resolver.py index 1ed7e55868..cc50f10e63 100644 --- a/aries_cloudagent/resolver/did_resolver.py +++ b/aries_cloudagent/resolver/did_resolver.py @@ -4,6 +4,7 @@ retrieving did's from different sources provided by the method type. """ +import asyncio from datetime import datetime from itertools import chain import logging @@ -42,9 +43,14 @@ async def _resolve( profile: Profile, did: Union[str, DID], service_accept: Optional[Sequence[Text]] = None, + *, + timeout: Optional[int] = None, ) -> Tuple[BaseDIDResolver, dict]: - """Retrieve doc and return with resolver.""" - # TODO Cache results + """Retrieve doc and return with resolver. + + This private method enables the public resolve and resolve_with_metadata + methods to share the same logic. + """ if isinstance(did, DID): did = str(did) else: @@ -52,10 +58,13 @@ async def _resolve( for resolver in await self._match_did_to_resolver(profile, did): try: LOGGER.debug("Resolving DID %s with %s", did, resolver) - document = await resolver.resolve( - profile, - did, - service_accept, + document = await asyncio.wait_for( + resolver.resolve( + profile, + did, + service_accept, + ), + timeout if timeout is not None else 30, ) return resolver, document except DIDNotFound: @@ -68,18 +77,20 @@ async def resolve( profile: Profile, did: Union[str, DID], service_accept: Optional[Sequence[Text]] = None, + *, + timeout: Optional[int] = None, ) -> dict: """Resolve a DID.""" - _, doc = await self._resolve(profile, did, service_accept) + _, doc = await self._resolve(profile, did, service_accept, timeout=timeout) return doc async def resolve_with_metadata( - self, profile: Profile, did: Union[str, DID] + self, profile: Profile, did: Union[str, DID], *, timeout: Optional[int] = None ) -> ResolutionResult: """Resolve a DID and return the ResolutionResult.""" resolution_start_time = datetime.utcnow() - resolver, doc = await self._resolve(profile, did) + resolver, doc = await self._resolve(profile, did, timeout=timeout) time_now = datetime.utcnow() duration = int((time_now - resolution_start_time).total_seconds() * 1000) @@ -120,7 +131,6 @@ async def dereference( document: Optional[BaseDIDDocument] = None, ) -> Resource: """Dereference a DID URL to its corresponding DID Doc object.""" - # TODO Use cached DID Docs when possible try: parsed = DIDUrl.parse(did_url) if not parsed.did: From 6ea611f413ea8cee9a967d37e2fba4240b57116e Mon Sep 17 00:00:00 2001 From: Daniel Bluhm Date: Wed, 6 Sep 2023 12:14:16 -0400 Subject: [PATCH 2/2] refactor: default timeout as constant Signed-off-by: Daniel Bluhm --- aries_cloudagent/resolver/did_resolver.py | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/aries_cloudagent/resolver/did_resolver.py b/aries_cloudagent/resolver/did_resolver.py index cc50f10e63..4438fa1b9a 100644 --- a/aries_cloudagent/resolver/did_resolver.py +++ b/aries_cloudagent/resolver/did_resolver.py @@ -30,6 +30,8 @@ class DIDResolver: """did resolver singleton.""" + DEFAULT_TIMEOUT = 30 + def __init__(self, resolvers: Optional[List[BaseDIDResolver]] = None): """Create DID Resolver.""" self.resolvers = resolvers or [] @@ -64,7 +66,7 @@ async def _resolve( did, service_accept, ), - timeout if timeout is not None else 30, + timeout if timeout is not None else self.DEFAULT_TIMEOUT, ) return resolver, document except DIDNotFound: