From 6c83d7b67b3eea2edfb18e7f645261673488e71c Mon Sep 17 00:00:00 2001 From: Andrew Godwin Date: Fri, 15 Sep 2023 10:21:33 -0600 Subject: [PATCH] Fix #642: Race condition searching for unseen users --- users/models/identity.py | 68 +++++++++++++++++++++------------------- 1 file changed, 35 insertions(+), 33 deletions(-) diff --git a/users/models/identity.py b/users/models/identity.py index 41c2f4272..bebaa31eb 100644 --- a/users/models/identity.py +++ b/users/models/identity.py @@ -394,39 +394,41 @@ def by_username_and_domain( domain = domain.domain else: domain = domain.lower() - try: - if local: - return cls.objects.get( - username__iexact=username, - domain_id=domain, - local=True, - ) - else: - return cls.objects.get( - username__iexact=username, - domain_id=domain, - ) - except cls.DoesNotExist: - if fetch and not local: - actor_uri, handle = cls.fetch_webfinger(f"{username}@{domain}") - if handle is None: - return None - # See if this actually does match an existing actor - try: - return cls.objects.get(actor_uri=actor_uri) - except cls.DoesNotExist: - pass - # OK, make one - username, domain = handle.split("@") - if not domain_instance: - domain_instance = Domain.get_remote_domain(domain) - return cls.objects.create( - actor_uri=actor_uri, - username=username, - domain_id=domain_instance, - local=False, - ) - return None + + with transaction.atomic(): + try: + if local: + return cls.objects.get( + username__iexact=username, + domain_id=domain, + local=True, + ) + else: + return cls.objects.get( + username__iexact=username, + domain_id=domain, + ) + except cls.DoesNotExist: + if fetch and not local: + actor_uri, handle = cls.fetch_webfinger(f"{username}@{domain}") + if handle is None: + return None + # See if this actually does match an existing actor + try: + return cls.objects.get(actor_uri=actor_uri) + except cls.DoesNotExist: + pass + # OK, make one + username, domain = handle.split("@") + if not domain_instance: + domain_instance = Domain.get_remote_domain(domain) + return cls.objects.create( + actor_uri=actor_uri, + username=username, + domain_id=domain_instance, + local=False, + ) + return None @classmethod def by_actor_uri(cls, uri, create=False, transient=False) -> "Identity":