Skip to content

Commit

Permalink
[serve] Shutdown http proxy state (cherry-pick) (ray-project#35446)
Browse files Browse the repository at this point in the history
* [serve] Shutdown http proxy state (ray-project#35395)

Shutdown http proxy state so that it won't run anything in its update loop once a shutdown signal is received.

Signed-off-by: Cindy Zhang <cindyzyx9@gmail.com>

* [serve] Remove print statement + fix lint (ray-project#35439)

Signed-off-by: Cindy Zhang <cindyzyx9@gmail.com>

---------

Signed-off-by: Cindy Zhang <cindyzyx9@gmail.com>
  • Loading branch information
zcin authored May 18, 2023
1 parent 8fbae46 commit f1f4d69
Show file tree
Hide file tree
Showing 2 changed files with 49 additions and 2 deletions.
12 changes: 10 additions & 2 deletions python/ray/serve/_private/http_state.py
Original file line number Diff line number Diff line change
Expand Up @@ -41,6 +41,7 @@ def __init__(
self._status = HTTPProxyStatus.STARTING
self._health_check_obj_ref = None
self._last_health_check_time: float = 0
self._shutting_down = False

self._actor_details = HTTPProxyDetails(
node_id=node_id,
Expand Down Expand Up @@ -78,6 +79,9 @@ def update_actor_details(self, **kwargs) -> None:
self._actor_details = HTTPProxyDetails(**details_kwargs)

def update(self):
if self._shutting_down:
return

if self._status == HTTPProxyStatus.STARTING:
finished, _ = ray.wait([self._ready_obj_ref], timeout=0)
if finished:
Expand Down Expand Up @@ -128,6 +132,10 @@ def update(self):
self._health_check_obj_ref = self._actor_handle.check_health.remote()
self._last_health_check_time = time.time()

def shutdown(self):
self._shutting_down = True
ray.kill(self.actor_handle, no_restart=True)


class HTTPState:
"""Manages all state for HTTP proxies in the system.
Expand Down Expand Up @@ -164,8 +172,8 @@ def __init__(
self._start_proxies_if_needed()

def shutdown(self) -> None:
for proxy in self.get_http_proxy_handles().values():
ray.kill(proxy, no_restart=True)
for proxy_state in self._proxy_states.values():
proxy_state.shutdown()

def get_config(self):
return self._config
Expand Down
39 changes: 39 additions & 0 deletions python/ray/serve/tests/test_http_state.py
Original file line number Diff line number Diff line change
@@ -1,9 +1,11 @@
import json
from unittest.mock import patch
import asyncio

import pytest

import ray
from ray.experimental.state.api import list_actors
from ray._private.test_utils import SignalActor, wait_for_condition
from ray.serve.config import DeploymentMode, HTTPOptions
from ray.serve._private.common import HTTPProxyStatus
Expand Down Expand Up @@ -132,6 +134,43 @@ def check_proxy(status):
ray.shutdown()


def test_http_proxy_shutdown():
ray.init()

@ray.remote(num_cpus=0)
class MockHTTPProxyActor:
async def ready(self):
return json.dumps(["mock_worker_id", "mock_log_file_path"])

async def check_health(self):
await asyncio.sleep(100)

proxy = MockHTTPProxyActor.options(lifetime="detached").remote()
state = HTTPProxyState(proxy, "alice", "mock_node_id", "mock_node_ip")
assert state.status == HTTPProxyStatus.STARTING

def check_proxy(status):
state.update()
return state.status == status

# Proxy actor is ready, so status should transition STARTING -> HEALTHY
wait_for_condition(check_proxy, status=HTTPProxyStatus.HEALTHY, timeout=2)

# Confirm that a new health check has been started
state.update()
assert state._health_check_obj_ref

# Shutdown the http proxy state. Wait for the http proxy actor to be killed
state.shutdown()
wait_for_condition(lambda: len(list_actors(filters=[("state", "=", "ALIVE")])) == 0)

# Make sure that the state doesn't try to check on the status of the dead actor
state.update()
assert state.status == HTTPProxyStatus.HEALTHY

ray.shutdown()


if __name__ == "__main__":
import sys

Expand Down

0 comments on commit f1f4d69

Please sign in to comment.