diff --git a/distributed/deploy/cluster.py b/distributed/deploy/cluster.py index 13e81f5f82..d2d0da82ae 100644 --- a/distributed/deploy/cluster.py +++ b/distributed/deploy/cluster.py @@ -427,10 +427,13 @@ def update(): cluster_repr_interval = parse_timedelta( dask.config.get("distributed.deploy.cluster-repr-interval", default="ms") ) - pc = PeriodicCallback(update, cluster_repr_interval * 1000) - self.periodic_callbacks["cluster-repr"] = pc - pc.start() + def install(): + pc = PeriodicCallback(update, cluster_repr_interval * 1000) + self.periodic_callbacks["cluster-repr"] = pc + pc.start() + + self.loop.add_callback(install) return tab def _repr_html_(self, cluster_status=None): diff --git a/distributed/deploy/tests/test_local.py b/distributed/deploy/tests/test_local.py index 85c1f21b30..781dc29a13 100644 --- a/distributed/deploy/tests/test_local.py +++ b/distributed/deploy/tests/test_local.py @@ -582,6 +582,34 @@ def test_ipywidgets(loop): assert isinstance(box, ipywidgets.Widget) +def test_ipywidgets_loop(loop): + """ + Previously cluster._ipython_display_ attached the PeriodicCallback to the + currently running loop, See https://github.com/dask/distributed/pull/6444 + """ + ipywidgets = pytest.importorskip("ipywidgets") + + async def get_ioloop(cluster): + return cluster.periodic_callbacks["cluster-repr"].io_loop + + async def amain(): + # running synchronous code in an async context to setup a + # IOLoop.current() that's different from cluster.loop + with LocalCluster( + n_workers=0, + silence_logs=False, + loop=loop, + dashboard_address=":0", + processes=False, + ) as cluster: + cluster._ipython_display_() + assert cluster.sync(get_ioloop, cluster) is loop + box = cluster._cached_widget + assert isinstance(box, ipywidgets.Widget) + + asyncio.run(amain()) + + def test_no_ipywidgets(loop, monkeypatch): from unittest.mock import MagicMock