Skip to content

ChatKit actions

Actions are a way for the ChatKit SDK frontend to trigger a streaming response without the user submitting a message. They can also be used to trigger side-effects outside ChatKit SDK.

Triggering actions

In response to user interaction with widgets

Actions can be triggered by attaching an ActionConfig to any widget node that supports it. For example, you can respond to click events on Buttons. When a user clicks on this button, the action will be sent to your server where you can update the widget, run inference, stream new thread items, etc.

Button(
    label="Example",
    onClickAction=ActionConfig(
      type="example",
      payload={"id": 123},
    )
)

Actions can also be sent imperatively by your frontend with sendAction(). This is probably most useful when you need ChatKit to respond to interaction happening outside ChatKit, but it can also be used to chain actions when you need to respond on both the client and the server (more on that below).

await chatKit.sendAction({
  type: "example",
  payload: { id: 123 },
});

Handling actions

On the server

By default, actions are sent to your server. You can handle actions on your server by implementing the action method on ChatKitServer.

from collections.abc import AsyncIterator
from datetime import datetime
from typing import Any

from chatkit.actions import Action
from chatkit.server import ChatKitServer
from chatkit.types import (
    HiddenContextItem,
    ThreadItemDoneEvent,
    ThreadMetadata,
    ThreadStreamEvent,
    WidgetItem,
)

RequestContext = dict[str, Any]


class MyChatKitServer(ChatKitServer[RequestContext]):
    async def action(
        self,
        thread: ThreadMetadata,
        action: Action[str, Any],
        sender: WidgetItem | None,
        context: RequestContext,
    ) -> AsyncIterator[ThreadStreamEvent]:
        if action.type == "example":
            await do_thing(action.payload['id'])

            # often you'll want to add a HiddenContextItem so the model
            # can see that the user did something
            hidden = HiddenContextItem(
                id=self.store.generate_item_id("message", thread, context),
                thread_id=thread.id,
                created_at=datetime.now(),
                content=["<USER_ACTION>The user did a thing</USER_ACTION>"],
            )
            await self.store.add_thread_item(thread.id, hidden, context)

            # then you might want to run inference to stream a response
            # back to the user.
            async for e in self.generate(context, thread):
                yield e

        if action.type == "another.example"
          # ...

NOTE: As with any client/server interaction, actions and their payloads are sent by the client and should be treated as untrusted data.

Client

Sometimes you’ll want to handle actions in your client integration. To do that you need to specify that the action should be sent to your client-side action handler by adding handler="client to the ActionConfig.

Button(
    label="Example",
    onClickAction=ActionConfig(
      type="example",
      payload={"id": 123},
      handler="client"
    )
)

Then, when the action is triggered, it will then be passed to a callback that you provide when instantiating ChatKit.

async function handleWidgetAction(action: {type: string, Record<string, unknown>}) {
  if (action.type === "example") {
    const res = await doSomething(action)

    // You can fire off actions to your server from here as well.
    // e.g. if you want to stream new thread items or update a widget.
    await chatKit.sendAction({
      type: "example_complete",
      payload: res
    })
  }
}

chatKit.setOptions({
  // other options...
  widgets: { onAction: handleWidgetAction }
})

Strongly typed actions

By default Action and ActionConfig are not strongly typed. However, we do expose a create helper on Action making it easy to generate ActionConfigs from a set of strongly-typed actions.

class ExamplePayload(BaseModel)
    id: int

ExampleAction = Action[Literal["example"], ExamplePayload]
OtherAction = Action[Literal["other"], None]

AppAction = Annotated[
  ExampleAction
  | OtherAction,
  Field(discriminator="type"),
]

ActionAdapter: TypeAdapter[AppAction] = TypeAdapter(AppAction)

def parse_app_action(action: Action[str, Any]): AppAction
  return ActionAdapter.validate_python(action)

# Usage in a widget
# Action provides a create helper which makes it easy to generate
# ActionConfigs from strongly typed actions.
Button(
    label="Example",
    onClickAction=ExampleAction.create(ExamplePayload(id=123))
)

# usage in action handler
class MyChatKitServer(ChatKitServer[RequestContext])
    async def action(
        self,
        thread: ThreadMetadata,
        action: Action[str, Any],
        sender: WidgetItem | None,
        context: RequestContext,
    ) -> AsyncIterator[Event]:
        # add custom error handling if needed
        app_action = parse_app_action(action)
        if (app_action.type == "example"):
            await do_thing(app_action.payload.id)

Use widgets and actions to create custom forms

When widget nodes that take user input are mounted inside a Form, the values from those fields will be included in the payload of all actions that originate from within the Form.

Form values are keyed in the payload by their name e.g.

  • Select(name="title")action.payload.title
  • Select(name="todo.title")action.payload.todo.title
Form(
  direction="col",
  onSubmitAction=ActionConfig(
      type="update_todo",
      payload={"id": todo.id}
  ),
  children=[
    Title(value="Edit Todo"),

    Text(value="Title", color="secondary", size="sm"),
    Text(
      value=todo.title,
      editable=EditableProps(name="title", required=True),
    )

    Text(value="Description", color="secondary", size="sm"),
    Text(
      value=todo.description,
      editable=EditableProps(name="description"),
    ),

    Button(label="Save", submit=true)
  ]
)

class MyChatKitServer(ChatKitServer[RequestContext])
    async def action(
        self,
        thread: ThreadMetadata,
        action: Action[str, Any],
        sender: WidgetItem | None,
        context: RequestContext,
    ) -> AsyncIterator[Event]:
        if (action.type == "update_todo"):
          id = action.payload['id']
          # Any action that originates from within the Form will
          # include title and description
          title = action.payload['title']
          description = action.payload['description']

            # ...

Validation

Form uses basic native form validation; enforcing required and pattern on fields where they are configured and blocking submission when the form has any invalid field.

We may add new validation modes with better UX, more expressive validation, custom error display, etc in the future. Until then, widgets are not a great medium for complex forms with tricky validation. If you have this need, a better pattern would be to use client side action handling to trigger a modal, show a custom form there, then pass the result back into ChatKit with sendAction.

Treating Card as a Form

You can pass asForm=True to Card and it will behave as a Form, running validation and passing collected fields to the Card’s confirm action.

Payload key collisions

If there is a naming collision with some other existing pre-defined key on your payload, the form value will be ignored. This is probably a bug, so we’ll emit an error event when we see this.

Customize how actions interact with loading states in widgets

Use ActionConfig.loadingBehavior to control how actions trigger different loading states in a widget.

Button(
    label="This make take a while...",
    onClickAction=ActionConfig(
      type="long_running_action_that_should_block_other_ui_interactions",
      loadingBehavior="container"
    )
)
Value Behavior
auto The action will adapt to how it’s being used. (default)
self The action triggers loading state on the widget node that the action was bound to.
container The action triggers loading state on the entire widget container. This causes the widget to fade out slightly and become inert.
none No loading state

Using auto behavior

Generally, we recommend using auto, which is the default. auto triggers loading states based on where the action is bound, for example:

  • Button.onClickActionself
  • Select.onChangeActionnone
  • Card.confirm.actioncontainer