diff --git a/libs/checkpoint-postgres/langgraph/checkpoint/postgres/aio.py b/libs/checkpoint-postgres/langgraph/checkpoint/postgres/aio.py index a5a82b185..e5db87c70 100644 --- a/libs/checkpoint-postgres/langgraph/checkpoint/postgres/aio.py +++ b/libs/checkpoint-postgres/langgraph/checkpoint/postgres/aio.py @@ -380,6 +380,18 @@ def list( Yields: Iterator[CheckpointTuple]: An iterator of matching checkpoint tuples. """ + try: + # check if we are in the main thread, only bg threads can block + # we don't check in other methods to avoid the overhead + if asyncio.get_running_loop() is self.loop: + raise asyncio.InvalidStateError( + "Synchronous calls to AsyncSqliteSaver are only allowed from a " + "different thread. From the main thread, use the async interface. " + "For example, use `checkpointer.alist(...)` or `await " + "graph.ainvoke(...)`." + ) + except RuntimeError: + pass aiter_ = self.alist(config, filter=filter, before=before, limit=limit) while True: try: @@ -410,7 +422,7 @@ def get_tuple(self, config: RunnableConfig) -> Optional[CheckpointTuple]: if asyncio.get_running_loop() is self.loop: raise asyncio.InvalidStateError( "Synchronous calls to AsyncPostgresSaver are only allowed from a " - "different thread. From the main thread, use the async interface." + "different thread. From the main thread, use the async interface. " "For example, use `await checkpointer.aget_tuple(...)` or `await " "graph.ainvoke(...)`." ) diff --git a/libs/checkpoint-sqlite/langgraph/checkpoint/sqlite/aio.py b/libs/checkpoint-sqlite/langgraph/checkpoint/sqlite/aio.py index 72fca5bea..9d19b5889 100644 --- a/libs/checkpoint-sqlite/langgraph/checkpoint/sqlite/aio.py +++ b/libs/checkpoint-sqlite/langgraph/checkpoint/sqlite/aio.py @@ -159,7 +159,7 @@ def get_tuple(self, config: RunnableConfig) -> Optional[CheckpointTuple]: if asyncio.get_running_loop() is self.loop: raise asyncio.InvalidStateError( "Synchronous calls to AsyncSqliteSaver are only allowed from a " - "different thread. From the main thread, use the async interface." + "different thread. From the main thread, use the async interface. " "For example, use `await checkpointer.aget_tuple(...)` or `await " "graph.ainvoke(...)`." ) @@ -191,6 +191,18 @@ def list( Yields: Iterator[CheckpointTuple]: An iterator of matching checkpoint tuples. """ + try: + # check if we are in the main thread, only bg threads can block + # we don't check in other methods to avoid the overhead + if asyncio.get_running_loop() is self.loop: + raise asyncio.InvalidStateError( + "Synchronous calls to AsyncSqliteSaver are only allowed from a " + "different thread. From the main thread, use the async interface. " + "For example, use `checkpointer.alist(...)` or `await " + "graph.ainvoke(...)`." + ) + except RuntimeError: + pass aiter_ = self.alist(config, filter=filter, before=before, limit=limit) while True: try: