diff --git a/zulip/zulip/__init__.py b/zulip/zulip/__init__.py index 725896e34..536102346 100644 --- a/zulip/zulip/__init__.py +++ b/zulip/zulip/__init__.py @@ -697,9 +697,12 @@ def call_endpoint( method: str = "POST", request: Optional[Dict[str, Any]] = None, longpolling: bool = False, + retry_on_rate_limit_error: bool = False, files: Optional[List[IO[Any]]] = None, timeout: Optional[float] = None, ) -> Dict[str, Any]: + secs: float + if request is None: request = dict() marshalled_request = {} @@ -707,14 +710,28 @@ def call_endpoint( if v is not None: marshalled_request[k] = v versioned_url = API_VERSTRING + (url if url is not None else "") - return self.do_api_query( - marshalled_request, - versioned_url, - method=method, - longpolling=longpolling, - files=files, - timeout=timeout, - ) + + while True: + result = self.do_api_query( + marshalled_request, + versioned_url, + method=method, + longpolling=longpolling, + files=files, + timeout=timeout, + ) + if not retry_on_rate_limit_error or result["result"] == "success": + break + elif "code" in result and result["code"] == "RATE_LIMIT_HIT": + secs = result["retry-after"] + elif "X-RateLimit-Reset" in result: + secs = float(result["X-RateLimit-Reset"]) + else: + break + logger.warning("hit API rate limit, waiting for %f seconds...", secs) + time.sleep(secs) + + return result def call_on_each_event( self,