understanding-streamlit-architecture

Understanding Streamlit architecture

Safety Notice

This listing is imported from skills.sh public index metadata. Review upstream SKILL.md and repository scripts before running.

Copy this and send it to your AI assistant to learn

Install skill "understanding-streamlit-architecture" with this command: npx skills add streamlit/streamlit/streamlit-streamlit-understanding-streamlit-architecture

Understanding Streamlit architecture

Streamlit is a client-server application with bidirectional WebSocket communication using Protocol Buffers. Use this file as a quick mental model and navigation index; use references/backend.md , references/frontend.md , and references/communication.md for implementation details.

Concepts glossary

Concept Description Key files

Command Any function exposed in Streamlit's public API (st.* namespace). Commands can create elements/widgets, control execution flow, or configure app behavior. lib/streamlit/delta_generator.py , lib/streamlit/elements/ , lib/streamlit/commands/

Element Umbrella term for all UI components in Streamlit: widgets, containers, and display elements. Represented in Element.proto as a oneof union of ~50+ types. proto/streamlit/proto/Element.proto , lib/streamlit/elements/

Widget Interactive element (button, slider, text_input) that triggers reruns on user interaction. Value accessible via return value or st.session_state . Some elements become widgets conditionally (e.g., dataframe/chart with on_select ). lib/streamlit/elements/widgets/ , frontend/lib/src/components/widgets/

Display Element Non-interactive element (text, markdown, image, chart) that renders content without triggering reruns by itself. lib/streamlit/elements/ , frontend/lib/src/components/elements/

Container Layout block that groups elements spatially (sidebar, columns, expander, tabs, form). Represented as BlockNode in the element tree. lib/streamlit/elements/layouts.py , Block.proto

DeltaGenerator The st object; API entry point that queues UI deltas. Uses mixin pattern to compose all st.* commands. lib/streamlit/delta_generator.py

Session State Per-session dictionary (st.session_state ) persisting data across reruns. Stores widget values and user variables. lib/streamlit/runtime/state/session_state.py

Rerun Re-execution of the user script for the current page. Triggered by widget interactions, st.rerun() , file changes, or fragment timers. Rebuilds the element tree while preserving session state. lib/streamlit/runtime/scriptrunner/script_runner.py , lib/streamlit/commands/execution_control.py

Form Container (st.form ) that batches widget inputs, deferring reruns until form submission. lib/streamlit/elements/form.py , WidgetStateManager.ts

Fragment Decorator (@st.fragment ) enabling partial reruns of specific UI sections without full script re-execution. lib/streamlit/runtime/fragment.py

Caching Decorators (@st.cache_data , @st.cache_resource ) that memoize function results to avoid redundant computation. lib/streamlit/runtime/caching/

Pages Multipage app system using st.navigation() and st.Page() or auto-discovery from pages/ directory. lib/streamlit/navigation/ , lib/streamlit/runtime/pages_manager.py

Config App configuration via .streamlit/config.toml controlling server, client, and theme settings. lib/streamlit/config.py , lib/streamlit/config_option.py

Theming Customizable UI themes (Light/Dark/Custom) defined in config or via theme editor. lib/streamlit/runtime/theme_util.py , frontend/lib/src/theme/

Secrets Secure credential storage via .streamlit/secrets.toml (local) or platform settings (deployed). Accessed via st.secrets . lib/streamlit/runtime/secrets.py

Connection Database/service abstraction (st.connection ) with built-in caching and secrets integration. lib/streamlit/connections/

Custom Components User-built extensions using React/iframe. v1 (legacy): declare_component() API. v2 (current): Bidirectional components with improved state management. component-lib/ , lib/streamlit/components/v1/ , lib/streamlit/components/v2/

Static File Serving Files in static/ directory served directly via /app/static/* (when static serving is enabled). lib/streamlit/web/server/server.py , lib/streamlit/web/server/starlette/starlette_routes.py

App Testing Testing framework (AppTest ) for simulating user interactions and inspecting rendered output. lib/streamlit/testing/

Core mental model

flowchart TB subgraph Backend["Backend (Python)"] Script[User Script] DG[DeltaGenerator] Runtime[Runtime/AppSession] end

subgraph Protocol["WebSocket + Protobuf"]
    FM[ForwardMsg]
    BM[BackMsg]
end

subgraph Frontend["Frontend (React)"]
    App[App.tsx]
    Tree[AppRoot Tree]
    Renderer[ElementNodeRenderer]
    WSM[WidgetStateManager]
end

Script --> DG
DG --> FM
FM --> App
App --> Tree
Tree --> Renderer
Renderer --> WSM
WSM -->|User interaction| BM
BM --> Runtime
Runtime -->|Rerun| Script

Key insight: Script execution is rerun-driven: most widget interactions trigger reruns (full app or fragment-scoped). State persists via st.session_state and caching decorators.

Execution model

Streamlit's execution model differs from traditional web frameworks:

Rerun triggers:

  • Widget interaction: User clicks button, moves slider, etc.

  • Source code change: File watcher detects script modification

  • st.rerun() : Explicit programmatic rerun

  • Fragment timer: @st.fragment(run_every=...) periodic reruns

Execution order nuances:

  • Scripts execute top-to-bottom on every rerun

  • Callbacks first: on_change /on_click handlers run before the main script body

  • Fragments typically isolate reruns: Widget interactions inside @st.fragment usually rerun only that fragment

  • Control flow exceptions: st.stop() , st.rerun() , st.switch_page() raise exceptions to halt/redirect execution

Session isolation:

  • Each browser tab = separate AppSession with its own SessionState

  • Refreshing the page creates a new session (unless reconnecting within TTL)

  • No shared state between sessions (use external storage for multi-user state)

What persists across reruns (within a session):

  • st.session_state values

  • Cached function results (@st.cache_data , @st.cache_resource )

  • Uploaded files

  • Fragment registrations

What resets on each rerun:

  • Local variables in script

  • Widget return values (re-read from SessionState )

  • Element tree (rebuilt from scratch, then diffed)

Architecture layers

Backend (Python)

Component File Purpose

Runtime lib/streamlit/runtime/runtime.py

Singleton managing app lifecycle and sessions

AppSession lib/streamlit/runtime/app_session.py

Per-browser-tab: ScriptRunner + SessionState + ForwardMsgQueue

ScriptRunner lib/streamlit/runtime/scriptrunner/script_runner.py

Executes user scripts in separate thread

DeltaGenerator lib/streamlit/delta_generator.py

API entry point using mixin pattern

SessionState lib/streamlit/runtime/state/session_state.py

Widget values and user variables

Elements lib/streamlit/elements/

Backend implementation of st.* commands

For backend deep dive: See references/backend.md

Frontend (TypeScript/React)

Component File Purpose

App frontend/app/src/App.tsx

Central orchestrator

AppRoot frontend/lib/src/render-tree/AppRoot.ts

Immutable element tree with 4 containers

ElementNodeRenderer frontend/lib/src/components/core/Block/ElementNodeRenderer.tsx

Maps protos to React components

WidgetStateManager frontend/lib/src/WidgetStateManager.ts

Widget state, forms, query params

ConnectionManager frontend/connection/src/ConnectionManager.ts

WebSocket state machine

For frontend deep dive: See references/frontend.md

Communication (Protobuf)

Proto Purpose

ForwardMsg.proto

Server to client: deltas, session events, navigation

BackMsg.proto

Client to server: rerun requests with widget states

Element.proto

~50+ element types in oneof type union

WidgetStates.proto

Widget values: trigger_value , string_value , bool_value , etc.

Location: proto/streamlit/proto/

For protocol deep dive: See references/communication.md

Essential concepts (quick map)

  • Script rerun model: Widget interaction -> BackMsg (ClientState ) -> backend updates SessionState -> script rerun -> ForwardMsg deltas.

  • Deep dive: references/communication.md#widget-interaction-to-script-rerun

  • Delta path system: Elements are addressed by delta paths (for example [0, 2, 3] ) to support efficient tree updates.

  • Deep dive: references/communication.md#delta-ui-changes

  • active_script_hash semantics: ForwardMsg.metadata.active_script_hash scopes node ownership across multipage and fragment reruns.

  • Deep dive: references/communication.md#forwardmsg-metadata-active_script_hash

  • Element tree structure: Frontend AppRoot maintains main , sidebar , event , and bottom containers with BlockNode /ElementNode .

  • Deep dive: references/frontend.md#element-tree-frontendlibsrcrender-tree

  • Element identity semantics: Delta paths decide where a node lives; element IDs decide whether stateful elements can reconnect to prior state after remounts.

  • Deep dive: references/element-identity.md

  • Fragments (@st.fragment ): Fragment interactions usually trigger fragment-scoped reruns and update only affected regions.

  • Deep dive: references/backend.md

Element identity

  • Delta path controls where a node is placed in the render tree.

  • Element ID controls whether a stateful element can reconnect to prior state after remounts.

  • Widgets use the ID to connect frontend WidgetStateManager state, BackMsg.WidgetStates , backend SessionState , callbacks, and st.session_state .

  • Some non-widgets also use IDs for frontend-only reconstruction (for example chart view state or media autoplay guards).

  • Delta-path changes can remount elements, but a stable element.id can still let React preserve a leaf when it remains under the same rendered parent list.

Deep dive: references/element-identity.md

Key implementation patterns

  • Backend mixin composition: DeltaGenerator composes st.* API via mixins.

  • Deep dive: references/backend.md#deltagenerator-libstreamlitdelta_generatorpy

  • Frontend visitor pattern: Render-tree updates/staleness cleanup use visitors.

  • Deep dive: references/frontend.md#visitor-pattern

  • Message deduplication: ForwardMsg.hash

  • ref_hash with frontend ForwardMsgCache reduces bandwidth.
  • Deep dive: references/communication.md#message-caching

Quick reference: adding features

  • Proto definition: proto/streamlit/proto/<Element>.proto

  • Register in Element.proto: Add to oneof type

  • Backend mixin: lib/streamlit/elements/<element>.py

  • Frontend component: frontend/lib/src/components/elements/<Element>/ or frontend/lib/src/components/widgets/<Element>/ (depending on element vs widget)

  • Register in ElementNodeRenderer: Add case to switch statement

  • Compile protos: make protobuf

See wiki/new-feature-guide.md for a detailed implementation guide.

Maintaining this documentation

When making changes that impact any of the concepts documented here (Runtime, AppSession, DeltaGenerator, element tree, communication protocol, element identity, etc.), update the relevant sections in this skill's files (SKILL.md , references/backend.md , references/frontend.md , references/communication.md , references/element-identity.md ) to keep them accurate.

Source Transparency

This detail page is rendered from real SKILL.md content. Trust labels are metadata-based hints, not a safety guarantee.

Related Skills

Related by shared tags or category signals.

General

streamlit

No summary provided by upstream source.

Repository SourceNeeds Review
General

debugging-streamlit

No summary provided by upstream source.

Repository SourceNeeds Review
General

implementing-new-features

No summary provided by upstream source.

Repository SourceNeeds Review