From 21918234498e7bd7a13206df519b963cdbf37b39 Mon Sep 17 00:00:00 2001 From: Andrew Huang Date: Fri, 10 Jan 2025 11:52:47 -0800 Subject: [PATCH 1/7] update --- examples/reference/chat/ChatFeed.ipynb | 47 ++++++++++++++++++++++++++ panel/chat/_param.py | 23 +++++++++++++ panel/chat/feed.py | 16 ++++++--- panel/tests/chat/test_feed.py | 32 ++++++++++++++++-- 4 files changed, 111 insertions(+), 7 deletions(-) create mode 100644 panel/chat/_param.py diff --git a/examples/reference/chat/ChatFeed.ipynb b/examples/reference/chat/ChatFeed.ipynb index d754d02248..95f80b0152 100644 --- a/examples/reference/chat/ChatFeed.ipynb +++ b/examples/reference/chat/ChatFeed.ipynb @@ -50,6 +50,7 @@ "* **`header`** (Any): The header of the chat feed; commonly used for the title. Can be a string, pane, or widget.\n", "* **`callback_user`** (str): The default user name to use for the message provided by the callback.\n", "* **`callback_avatar`** (str, BytesIO, bytes, ImageBase): The default avatar to use for the entry provided by the callback. Takes precedence over `ChatMessage.default_avatars` if set; else, if None, defaults to the avatar set in `ChatMessage.default_avatars` if matching key exists. Otherwise defaults to the first character of the `callback_user`.\n", + "* **`callback_exception`** (str, Callable): How to handle exceptions raised by the callback. If \"raise\", the exception will be raised. If \"summary\", a summary will be sent to the chat feed. If \"verbose\" or \"traceback\", the full traceback will be sent to the chat feed. If \"ignore\", the exception will be ignored. If a callable is provided, the signature must contain the `exception` and `instance` arguments and it will be called with the exception.\n", "* **`help_text`** (str): If provided, initializes a chat message in the chat log using the provided help text as the message object and `help` as the user. This is useful for providing instructions, and will not be included in the `serialize` method by default.\n", "* **`placeholder_text`** (str): The text to display next to the placeholder icon.\n", "* **`placeholder_params`** (dict) Defaults to `{\"user\": \" \", \"reaction_icons\": {}, \"show_copy_icon\": False, \"show_timestamp\": False}` Params to pass to the placeholder `ChatMessage`, like `reaction_icons`, `timestamp_format`, `show_avatar`, `show_user`, `show_timestamp`.\n", @@ -433,6 +434,52 @@ "chat_feed.send(\"This will fail...\")" ] }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "Alternatively, you can provide a callable that accepts the exception and the instance as arguments to handle different exception scenarios." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": {}, + "outputs": [], + "source": [ + "import random\n", + "\n", + "\n", + "def callback(content):\n", + " if random.random() < 0.5:\n", + " raise RuntimeError(\"This is an unhandled error\")\n", + " raise ValueError(\"This is a handled error\")\n", + "\n", + "\n", + "def callback_exception_handler(exception, instance):\n", + " if isinstance(exception, ValueError):\n", + " instance.stream(\"I can handle this\", user=\"System\")\n", + " return\n", + " instance.stream(\"Fatal error occurred\", user=\"System\")\n", + "\n", + " # you can raise a new exception here if desired\n", + " # raise RuntimeError(\"Fatal error occurred\") from exception\n", + "\n", + "\n", + "chat_feed = pn.chat.ChatFeed(\n", + " callback=callback, callback_exception=callback_exception_handler\n", + ")" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": {}, + "outputs": [], + "source": [ + "chat_feed.send(\"This will sometimes fail...\")" + ] + }, { "cell_type": "markdown", "metadata": {}, diff --git a/panel/chat/_param.py b/panel/chat/_param.py new file mode 100644 index 0000000000..9f54a12e70 --- /dev/null +++ b/panel/chat/_param.py @@ -0,0 +1,23 @@ +from param import Parameter + + +class CallbackException(Parameter): + """ + A Parameter type to validate callback_exception options. Supports + "raise", "summary", "verbose", "traceback", "ignore", or a callable. + """ + + def __init__(self, default="summary", **params): + super().__init__(default=default, **params) + self._validate(default) + + def _validate(self, val): + self._validate_value(val, self.allow_None) + + def _validate_value(self, val, allow_None, valid=("raise", "summary", "verbose", "traceback", "ignore")): + if ((val is None and allow_None) or val in valid or callable(val)): + return + raise ValueError( + f"Callback exception mode {val} not recognized. " + f"Valid options are {valid} or a callable." + ) diff --git a/panel/chat/feed.py b/panel/chat/feed.py index d5bda8cabe..1ee06686f2 100644 --- a/panel/chat/feed.py +++ b/panel/chat/feed.py @@ -34,6 +34,7 @@ from ..viewable import Children from ..widgets import Widget from ..widgets.button import Button +from ._param import CallbackException from .icon import ChatReactionIcons from .message import ChatMessage from .step import ChatStep @@ -100,13 +101,16 @@ class ChatFeed(ListPanel): the previous message value `contents`, the previous `user` name, and the component `instance`.""") - callback_exception = param.Selector( - default="summary", objects=["raise", "summary", "verbose", "ignore"], doc=""" + callback_exception = CallbackException( + default="summary", doc=""" How to handle exceptions raised by the callback. If "raise", the exception will be raised. If "summary", a summary will be sent to the chat feed. - If "verbose", the full traceback will be sent to the chat feed. + If "verbose" or "traceback", the full traceback will be sent to the chat feed. If "ignore", the exception will be ignored. + If a callable is provided, the signature must contain the + `exception` and `instance` arguments and it + will be called with the exception. """) callback_user = param.String(default="Assistant", doc=""" @@ -574,13 +578,15 @@ async def _prepare_response(self, *_) -> None: self._callback_state = CallbackState.STOPPED except Exception as e: send_kwargs: dict[str, Any] = dict(user="Exception", respond=False) - if self.callback_exception == "summary": + if callable(self.callback_exception): + self.callback_exception(e, self) + elif self.callback_exception == "summary": self.send( f"Encountered `{e!r}`. " f"Set `callback_exception='verbose'` to see the full traceback.", **send_kwargs ) - elif self.callback_exception == "verbose": + elif self.callback_exception in ("verbose", "traceback"): self.send(f"```python\n{traceback.format_exc()}\n```", **send_kwargs) elif self.callback_exception == "ignore": return diff --git a/panel/tests/chat/test_feed.py b/panel/tests/chat/test_feed.py index 1ac906b1e5..e20a7fe48a 100644 --- a/panel/tests/chat/test_feed.py +++ b/panel/tests/chat/test_feed.py @@ -1076,12 +1076,13 @@ def callback(msg, user, instance): await async_wait_until(lambda: "division by zero" in chat_feed.objects[-1].object) assert chat_feed.objects[-1].user == "Exception" - async def test_callback_exception_traceback(self, chat_feed): + @pytest.mark.parametrize("callback_exception", ["traceback", "verbose"]) + def test_callback_exception_traceback(self, chat_feed, callback_exception): def callback(msg, user, instance): return 1 / 0 chat_feed.callback = callback - chat_feed.callback_exception = "verbose" + chat_feed.callback_exception = callback_exception chat_feed.send("Message", respond=True) await async_wait_until(lambda: chat_feed.objects[-1].object.startswith( "```python\nTraceback (most recent call last):" @@ -1097,6 +1098,33 @@ def callback(msg, user, instance): chat_feed.send("Message", respond=True) await async_wait_until(lambda: len(chat_feed.objects) == 1) + def test_callback_exception_raise(self, chat_feed): + def callback(msg, user, instance): + return 1 / 0 + + chat_feed.callback = callback + chat_feed.callback_exception = "raise" + with pytest.raises(ZeroDivisionError, match="division by zero"): + chat_feed.send("Message", respond=True) + wait_until(lambda: len(chat_feed.objects) == 1) + + def test_callback_exception_callable(self, chat_feed): + def callback(msg, user, instance): + raise ValueError("Expected error") + + def exception_callback(exception, instance): + instance.stream(f"The exception: {exception}") + + chat_feed.callback = callback + chat_feed.callback_exception = exception_callback + chat_feed.send("Message", respond=True) + wait_until(lambda: len(chat_feed.objects) == 2) + assert chat_feed.objects[-1].object == "The exception: Expected error" + + def test_callback_exception_invalid_option(self, chat_feed): + with pytest.raises(ValueError, match="Valid options are"): + chat_feed.callback_exception = "abc" + async def test_callback_stop_generator(self, chat_feed): def callback(msg, user, instance): yield "A" From 1d4bb63bd165e1c2d1c18aca2e0db69a94a0690b Mon Sep 17 00:00:00 2001 From: Andrew Huang Date: Tue, 17 Dec 2024 12:16:06 -0800 Subject: [PATCH 2/7] ignore typing --- panel/tests/chat/test_feed.py | 2 +- panel/tests/chat/test_interface.py | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/panel/tests/chat/test_feed.py b/panel/tests/chat/test_feed.py index e20a7fe48a..77d206389b 100644 --- a/panel/tests/chat/test_feed.py +++ b/panel/tests/chat/test_feed.py @@ -23,7 +23,7 @@ "max_width": 201, } -ChatFeed.callback_exception = "raise" +ChatFeed.callback_exception = "raise" # type: ignore @pytest.fixture diff --git a/panel/tests/chat/test_interface.py b/panel/tests/chat/test_interface.py index 9cf9b64412..dfcfdd460d 100644 --- a/panel/tests/chat/test_interface.py +++ b/panel/tests/chat/test_interface.py @@ -15,7 +15,7 @@ from panel.widgets.input import FileInput, TextAreaInput, TextInput from panel.widgets.select import RadioButtonGroup -ChatInterface.callback_exception = "raise" +ChatInterface.callback_exception = "raise" # type: ignore class TestChatInterface: From 8eae15f7d6c593ebb7860133ce95603eb837b467 Mon Sep 17 00:00:00 2001 From: Andrew Huang Date: Tue, 17 Dec 2024 12:26:21 -0800 Subject: [PATCH 3/7] add release notes --- CHANGELOG.md | 6 ++++++ doc/about/releases.md | 6 ++++++ 2 files changed, 12 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index 4bb80f5d35..7be434b58d 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,5 +1,11 @@ # Releases +## Version 1.6.0 + +### Enhancements + +- Allow `ChatFeed.callback_exception` to accept a callable and 'traceback' (alias for verbose) ([#7558](https://github.com/holoviz/panel/pull/7558)) + ## Version 1.5.5 This release fixes a regression causing .node_modules to be bundled into our released wheel and introduces a number of bug fixes and enhancements. Many thanks to @mayonnaisecolouredbenz7, @pmeier, @Italirz, @Coderambling and our maintainer team @MarcSkovMadsen, @hoxbro, @ahuang11, @thuydotm, @maximlt and @philippjfr. diff --git a/doc/about/releases.md b/doc/about/releases.md index ec6bf67459..e1dc2000e5 100644 --- a/doc/about/releases.md +++ b/doc/about/releases.md @@ -2,6 +2,12 @@ See [the HoloViz blog](https://blog.holoviz.org/#category=panel) for a visual summary of the major features added in each release. +## Version 1.6.0 + +### Enhancements + +- Allow `ChatFeed.callback_exception` to accept a callable and 'traceback' (alias for verbose) ([#7558](https://github.com/holoviz/panel/pull/7558)) + ## Version 1.5.5 This release fixes a regression causing .node_modules to be bundled into our released wheel and introduces a number of bug fixes and enhancements. Many thanks to @mayonnaisecolouredbenz7, @pmeier, @Italirz, @Coderambling and our maintainer team @MarcSkovMadsen, @hoxbro, @ahuang11, @thuydotm, @maximlt and @philippjfr. From d7955cb38f33a407787ec0d5e2bd0c05dbeaf1a8 Mon Sep 17 00:00:00 2001 From: Andrew Huang Date: Fri, 10 Jan 2025 11:53:02 -0800 Subject: [PATCH 4/7] maybe fix? --- panel/tests/chat/test_feed.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/panel/tests/chat/test_feed.py b/panel/tests/chat/test_feed.py index 77d206389b..9528e6c62b 100644 --- a/panel/tests/chat/test_feed.py +++ b/panel/tests/chat/test_feed.py @@ -1108,7 +1108,7 @@ def callback(msg, user, instance): chat_feed.send("Message", respond=True) wait_until(lambda: len(chat_feed.objects) == 1) - def test_callback_exception_callable(self, chat_feed): + async def test_callback_exception_callable(self, chat_feed): def callback(msg, user, instance): raise ValueError("Expected error") @@ -1118,7 +1118,7 @@ def exception_callback(exception, instance): chat_feed.callback = callback chat_feed.callback_exception = exception_callback chat_feed.send("Message", respond=True) - wait_until(lambda: len(chat_feed.objects) == 2) + await async_wait_until(lambda: len(chat_feed.objects) == 2) assert chat_feed.objects[-1].object == "The exception: Expected error" def test_callback_exception_invalid_option(self, chat_feed): From 9448cb53080d45d7206ebf0aba3d579d53dbbc29 Mon Sep 17 00:00:00 2001 From: Andrew Huang Date: Fri, 10 Jan 2025 11:58:57 -0800 Subject: [PATCH 5/7] add back async --- panel/tests/chat/test_feed.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/panel/tests/chat/test_feed.py b/panel/tests/chat/test_feed.py index 9528e6c62b..39c1f75d6c 100644 --- a/panel/tests/chat/test_feed.py +++ b/panel/tests/chat/test_feed.py @@ -1077,7 +1077,7 @@ def callback(msg, user, instance): assert chat_feed.objects[-1].user == "Exception" @pytest.mark.parametrize("callback_exception", ["traceback", "verbose"]) - def test_callback_exception_traceback(self, chat_feed, callback_exception): + async def test_callback_exception_traceback(self, chat_feed, callback_exception): def callback(msg, user, instance): return 1 / 0 From c5e4d9680aaa652a2ba32b9ca1e3414b8c4a095e Mon Sep 17 00:00:00 2001 From: Andrew Huang Date: Fri, 10 Jan 2025 12:09:25 -0800 Subject: [PATCH 6/7] support async --- panel/chat/feed.py | 4 +++- panel/tests/chat/test_feed.py | 16 +++++++++++++++- 2 files changed, 18 insertions(+), 2 deletions(-) diff --git a/panel/chat/feed.py b/panel/chat/feed.py index 1ee06686f2..af0ca85e7a 100644 --- a/panel/chat/feed.py +++ b/panel/chat/feed.py @@ -579,7 +579,9 @@ async def _prepare_response(self, *_) -> None: except Exception as e: send_kwargs: dict[str, Any] = dict(user="Exception", respond=False) if callable(self.callback_exception): - self.callback_exception(e, self) + result = self.callback_exception(e, self) + if isawaitable(result): + await result elif self.callback_exception == "summary": self.send( f"Encountered `{e!r}`. " diff --git a/panel/tests/chat/test_feed.py b/panel/tests/chat/test_feed.py index 39c1f75d6c..f6ae8e31db 100644 --- a/panel/tests/chat/test_feed.py +++ b/panel/tests/chat/test_feed.py @@ -1108,13 +1108,27 @@ def callback(msg, user, instance): chat_feed.send("Message", respond=True) wait_until(lambda: len(chat_feed.objects) == 1) - async def test_callback_exception_callable(self, chat_feed): + def test_callback_exception_callable(self, chat_feed): def callback(msg, user, instance): raise ValueError("Expected error") def exception_callback(exception, instance): instance.stream(f"The exception: {exception}") + chat_feed.callback = callback + chat_feed.callback_exception = exception_callback + chat_feed.send("Message", respond=True) + wait_until(lambda: len(chat_feed.objects) == 2) + assert chat_feed.objects[-1].object == "The exception: Expected error" + + async def test_callback_exception_async_callable(self, chat_feed): + async def callback(msg, user, instance): + raise ValueError("Expected error") + + async def exception_callback(exception, instance): + await asyncio.sleep(0.1) + instance.stream(f"The exception: {exception}") + chat_feed.callback = callback chat_feed.callback_exception = exception_callback chat_feed.send("Message", respond=True) From f55ea87bf526e3384adf539b1bd9ee1d0842e32e Mon Sep 17 00:00:00 2001 From: Andrew Huang Date: Fri, 10 Jan 2025 12:45:47 -0800 Subject: [PATCH 7/7] silence warning --- panel/chat/feed.py | 7 ++++--- panel/tests/chat/test_feed.py | 4 ++-- 2 files changed, 6 insertions(+), 5 deletions(-) diff --git a/panel/chat/feed.py b/panel/chat/feed.py index af0ca85e7a..4cc8cea831 100644 --- a/panel/chat/feed.py +++ b/panel/chat/feed.py @@ -579,9 +579,10 @@ async def _prepare_response(self, *_) -> None: except Exception as e: send_kwargs: dict[str, Any] = dict(user="Exception", respond=False) if callable(self.callback_exception): - result = self.callback_exception(e, self) - if isawaitable(result): - await result + if iscoroutinefunction(self.callback_exception): + await self.callback_exception(e, self) + else: + self.callback_exception(e, self) elif self.callback_exception == "summary": self.send( f"Encountered `{e!r}`. " diff --git a/panel/tests/chat/test_feed.py b/panel/tests/chat/test_feed.py index f6ae8e31db..4b338fb209 100644 --- a/panel/tests/chat/test_feed.py +++ b/panel/tests/chat/test_feed.py @@ -1108,6 +1108,7 @@ def callback(msg, user, instance): chat_feed.send("Message", respond=True) wait_until(lambda: len(chat_feed.objects) == 1) + @pytest.mark.filterwarnings("ignore::pytest.PytestUnraisableExceptionWarning") def test_callback_exception_callable(self, chat_feed): def callback(msg, user, instance): raise ValueError("Expected error") @@ -1118,7 +1119,6 @@ def exception_callback(exception, instance): chat_feed.callback = callback chat_feed.callback_exception = exception_callback chat_feed.send("Message", respond=True) - wait_until(lambda: len(chat_feed.objects) == 2) assert chat_feed.objects[-1].object == "The exception: Expected error" async def test_callback_exception_async_callable(self, chat_feed): @@ -1132,7 +1132,7 @@ async def exception_callback(exception, instance): chat_feed.callback = callback chat_feed.callback_exception = exception_callback chat_feed.send("Message", respond=True) - await async_wait_until(lambda: len(chat_feed.objects) == 2) + async_wait_until(lambda: len(chat_feed.objects) == 2) assert chat_feed.objects[-1].object == "The exception: Expected error" def test_callback_exception_invalid_option(self, chat_feed):