Panel navigation during real time updates

Hi All,

I’ve created multiple real time dashboards in panel and each one works individually exactly as designed. The dashboards display real time data including pitch, roll, yaw, alt, lat, long, etc. using the full range of Holoviz/Panel indicators and widgets.

When I try and run them together using --glob navigation between apps breaks the background process and it shuts down.

Closed a session running at the / endpoint

I’ve previously created a topic 7346 and I’ve taken this example and updated it for my latest issue. Here a quick and dirty MRE.

app.py

from asyncio import sleep

import holoviews as hv
import pandas as pd
import panel as pn
from holoviews.streams import Buffer
import hvplot.pandas  # no qa

pn.extension()

data = pn.state.cache["data"]

dfstream = Buffer(pd.DataFrame(data, index=[pd.Timestamp.now()]), length=100, index=False)


def plot(data, window_seconds, alpha):
    data = data.rolling(f"{window_seconds}s").mean()
    return data.hvplot(y="y", ylim=(0, 50), alpha=alpha, color="blue", line_width=5)


window_seconds = pn.widgets.IntSlider(value=5, start=1, end=10, name="Window (secs)")
alpha = pn.widgets.FloatSlider(value=1, start=0, end=1, name="Alpha")
iplot = hv.DynamicMap(
    plot,
    streams={
        "data": dfstream.param.data,
        "window_seconds": window_seconds.param.value,
        "alpha": alpha.param.value,
    },
)

pages = {
    "Page 1": pn.Column(iplot, window_seconds, alpha),
    "Page 2": pn.Column("# Page 2", "...more bla"),
}


def show(page):
    return pages[page]


starting_page = pn.state.session_args.get("page", [b"Page 1"])[0].decode()
page = pn.widgets.RadioButtonGroup(
    value=starting_page,
    options=list(pages.keys()),
    name="Page",
    sizing_mode="fixed",
    button_type="success",
)
ishow = pn.bind(show, page=page)
pn.state.location.sync(page, {"value": "page"})

ACCENT_COLOR = "#0072B5"
DEFAULT_PARAMS = {
    "site": "Panel Multi Page App",
    "accent_base_color": ACCENT_COLOR,
    "header_background": ACCENT_COLOR,
}
pn.template.FastListTemplate(
    title="As Single Page App",
    sidebar=[page],
    main=[ishow],
    **DEFAULT_PARAMS,
).servable()


async def run():
    while True:
        await sleep(0.1)
        data = pn.state.cache["data"]
        dfstream.send(pd.DataFrame(data, index=[pd.Timestamp.now()]))


pn.state.onload(run)

app2.py

from asyncio import sleep

import holoviews as hv
import pandas as pd
import panel as pn
from holoviews.streams import Buffer
import hvplot.pandas  # no qa

pn.extension()

data = pn.state.cache["data"]

dfstream = Buffer(pd.DataFrame(data, index=[pd.Timestamp.now()]), length=100, index=False)


def plot(data, window_seconds, alpha):
    data = data.rolling(f"{window_seconds}s").mean()
    return data.hvplot(y="a", ylim=(0, 50), alpha=alpha, color="blue", line_width=5)


window_seconds = pn.widgets.IntSlider(value=5, start=1, end=10, name="Window (secs)")
alpha = pn.widgets.FloatSlider(value=1, start=0, end=1, name="Alpha")
iplot = hv.DynamicMap(
    plot,
    streams={
        "data": dfstream.param.data,
        "window_seconds": window_seconds.param.value,
        "alpha": alpha.param.value,
    },
)

pages = {
    "Page 1": pn.Column(iplot, window_seconds, alpha),
    "Page 2": pn.Column("# Page 2", "...more bla"),
}


def show(page):
    return pages[page]


starting_page = pn.state.session_args.get("page", [b"Page 1"])[0].decode()
page = pn.widgets.RadioButtonGroup(
    value=starting_page,
    options=list(pages.keys()),
    name="Page",
    sizing_mode="fixed",
    button_type="success",
)
ishow = pn.bind(show, page=page)
pn.state.location.sync(page, {"value": "page"})

ACCENT_COLOR = "#0072B5"
DEFAULT_PARAMS = {
    "site": "Panel Multi Page App",
    "accent_base_color": ACCENT_COLOR,
    "header_background": ACCENT_COLOR,
}
pn.template.FastListTemplate(
    title="As Single Page App",
    sidebar=[page],
    main=[ishow],
    **DEFAULT_PARAMS,
).servable()


async def run():
    while True:
        await sleep(0.1)
        data = pn.state.cache["data"]
        dfstream.send(pd.DataFrame(data, index=[pd.Timestamp.now()]))


pn.state.onload(run)

index.py

"""
Panel App Gallery

A Panel-based index page that serves as a gallery of Panel applications with search and pagination functionality.
This demonstrates how to use `panel serve *.py --index index.py` to create a custom Python-based index page
instead of relying on HTML/Jinja templates.

Features:

- Real-time search functionality that filters apps by name and description
- Responsive thumbnails and app metadata display
- Pagination support when there are more than APPS_PER_PAGE results
- Modern Design styling with hover effects
- Auto-discovery of Panel apps in the current directory

Usage:
    panel serve *.py --index index.py --dev --show

This app provides an interactive way to browse and search through a collection of Panel applications,
making it easy to find specific apps in larger collections. Each app displays its name, description,
and thumbnail image with direct links to launch the application.
"""

import math
from typing import List

import panel as pn
import param

# Configuration
APPS_PER_PAGE = 20
ACCENT_COLOR = "#4099da"
ACCENT_FILL_HOVER = "#0072b5"

# Initialize Panel with required extensions
pn.extension("tabulator", sizing_mode="stretch_width")


class AppInfo:
    """Data class to hold information about a Panel app"""

    def __init__(self, name: str, filename: str, description: str = "", thumbnail: str = ""):
        self.name = name
        self.filename = filename
        self.description = description or f"Panel application: {name}"
        self.thumbnail = thumbnail or self._get_default_thumbnail()

    def _get_default_thumbnail(self) -> str:
        """Generate a default thumbnail placeholder"""
        return ""

    def url(self) -> str:
        """Generate the URL to launch the app"""
        return f"./{self.filename.replace('.py', '')}"


class AppGallery(pn.viewable.Viewer):
    """
    A Panel-based gallery for browsing and searching Panel applications.

    This component provides:

    - Search functionality with real-time filtering
    - Pagination for large app collections
    - Thumbnail previews and app metadata
    - Direct links to launch applications
    """

    # Parameters for state management
    search_term = param.String(default="", doc="Current search term")
    current_page = param.Integer(default=1, bounds=(1, None), doc="Current page number")
    apps_per_page = param.Integer(
        default=APPS_PER_PAGE, bounds=(1, 50), doc="Number of apps per page"
    )

    def __init__(self, **params):
        super().__init__(**params)

        # Discover available Panel apps
        self.all_apps = self._discover_apps()

        # Create UI components
        self._create_ui()

    def _discover_apps(self) -> List[AppInfo]:
        """Discover Panel apps in the current directory"""
        apps = []

        # Add some demo apps for illustration
        demo_apps = [
            AppInfo(
                "App",
                "app.py",
                "Interactive dashboard for...",
                thumbnail=r".\assets\images\thumbnails\app.png",
            ),
            AppInfo(
                "App 2",
                "app2.py",
                "Interactive dashboard for...",
                thumbnail=r".\assets\images\thumbnails\app2.png",
            ),
        ]

        apps.extend(demo_apps)

        return sorted(apps, key=lambda x: x.name)

    def _create_ui(self):
        """Create the user interface components"""
        with pn.config.set(sizing_mode="stretch_width"):
            # Header
            self._header = pn.pane.HTML("""
            <div style="text-align: center; padding: 20px 0;">
                <h1 style="color: #2E86AB; margin-bottom: 10px;">Holoviz</h1>
                <h1 style="color: #2E86AB; margin-bottom: 5px;">Real Time Data Analytics</h1>
                <p style="color: #666; font-size: 18px;">Discover and explore applications</p>
            </div>
            """)

            # Search bar
            self._search_input = pn.widgets.TextInput.from_param(
                self.param.search_term,
                placeholder="Search apps by name or description...",
                margin=(10, 20),
                sizing_mode="stretch_width",
            )

            # Apps grid container
            self._apps_container = pn.FlexBox(
                sizing_mode="stretch_width",
                align="center",
            )

            # Pagination controls
            self._pagination = pn.Row(sizing_mode="stretch_width", styles={"margin": "0 auto"})

            # Main layout
            self._layout = pn.Column(
                self._header,
                self._search_input,
                self._apps_container,
                pn.Spacer(sizing_mode="stretch_height"),
                self._pagination,
                max_width=975,
                styles={"margin": "0 auto"},
                align="center",
                sizing_mode="stretch_both",
            )

        # Initial update will be triggered automatically by parameter reactivity
        self._update_display()

    @param.depends("search_term", watch=True)
    def _on_search_change(self):
        """Handle search term changes by resetting to first page"""
        self.current_page = 1

    def _get_filtered_apps(self) -> List[AppInfo]:
        """Get filtered apps based on search term"""
        # Filter apps based on search term
        search_term = str(self.search_term or "")
        if search_term:
            filtered_apps = [
                app
                for app in self.all_apps
                if (
                    search_term.lower() in app.name.lower()
                    or search_term.lower() in app.description.lower()
                )
            ]
        else:
            filtered_apps = self.all_apps

        return filtered_apps

    def _get_paginated_apps(self, apps: List[AppInfo]) -> List[AppInfo]:
        """Get apps for current page"""
        # Safely get parameter values with defaults
        current_page = max(1, getattr(self, "current_page", 1) or 1)
        apps_per_page = max(1, getattr(self, "apps_per_page", APPS_PER_PAGE) or APPS_PER_PAGE)

        start_idx = (current_page - 1) * apps_per_page
        end_idx = start_idx + apps_per_page
        return apps[start_idx:end_idx]

    def _create_app_card(self, app: AppInfo) -> pn.pane.HTML:
        """Create a card for a single app"""
        card_html = f"""
        <div style="
            border: 1px solid var(--neutral-stroke-rest);
            border-radius: 12px;
            padding: 20px;
            margin: 10px;
            box-shadow: 0 2px 4px rgba(0,0,0,0.1);
            transition: all 0.3s ease;
            height: 320px;
            display: flex;
            flex-direction: column;
        " onmouseover="this.style.transform='translateY(-5px)'; this.style.boxShadow='0 8px 25px rgba(0,0,0,0.15)'"
           onmouseout="this.style.transform='translateY(0px)'; this.style.boxShadow='0 2px 4px rgba(0,0,0,0.1)'">
            
            <div style="text-align: center; margin-bottom: 15px;">
                <img src="{app.thumbnail}" 
                     style="width: 100%; height: 120px; object-fit: cover; border-radius: 8px; background: #f5f5f5;" 
                     alt="App thumbnail"/>
            </div>
            
            <h3 style="color: {ACCENT_COLOR}; margin: 0 0 10px 0; font-size: 18px; text-align: center;">
                {app.name}
            </h3>
            
            <p style="color: #666; margin: 0 0 15px 0; font-size: 14px; line-height: 1.4; flex-grow: 1; overflow: hidden;">
                {app.description[:120]}{"..." if len(app.description) > 120 else ""}
            </p>
            
            <div style="text-align: center; margin-top: auto;">
                <a href="{app.url()}" 
                   style="
                       background: {ACCENT_COLOR};
                       color: white;
                       padding: 8px 16px;
                       text-decoration: none;
                       border-radius: 6px;
                       font-weight: 500;
                       transition: background-color 0.2s;
                   "
                   onmouseover="this.style.backgroundColor='{ACCENT_FILL_HOVER}'"
                   onmouseout="this.style.backgroundColor='{ACCENT_COLOR}'">
                    Open App →
                </a>
            </div>
        </div>
        """
        return pn.pane.HTML(card_html, width=300, height=320)

    def _create_pagination_controls(self, total_apps: int) -> pn.Row:
        """Create pagination controls"""
        # Safely get parameter values with defaults
        current_page = max(1, getattr(self, "current_page", 1) or 1)
        apps_per_page = max(1, getattr(self, "apps_per_page", APPS_PER_PAGE) or APPS_PER_PAGE)
        total_pages = math.ceil(total_apps / apps_per_page)

        if total_pages <= 1:
            return pn.Row()

        controls = []

        # Previous button
        if current_page > 1:
            prev_btn = pn.widgets.Button(
                name="← Previous",
                button_style="outline",
            )
            prev_btn.on_click(lambda event: self._go_to_page(current_page - 1))
            controls.append(prev_btn)

        # Page numbers
        start_page = max(1, current_page - 2)
        end_page = min(total_pages, current_page + 2)

        for page_num in range(start_page, end_page + 1):
            if page_num == current_page:
                # Current page (highlighted)
                page_btn = pn.widgets.Button(
                    name=str(page_num), button_type="primary", margin=(5, 5), width=40
                )
            else:
                # Other pages
                page_btn = pn.widgets.Button(
                    name=str(page_num), button_style="outline", margin=(5, 5), width=40
                )
                page_btn.on_click(lambda event, p=page_num: self._go_to_page(p))

            controls.append(page_btn)

        # Next button
        if current_page < total_pages:
            next_btn = pn.widgets.Button(name="Next →", button_style="outline", margin=(5, 10))
            next_btn.on_click(lambda event: self._go_to_page(current_page + 1))
            controls.append(next_btn)

        return pn.Row(*controls)

    def _go_to_page(self, page_num: int):
        """Navigate to a specific page"""
        self.current_page = page_num  # Reactivity will handle the update automatically

    @param.depends("search_term", "current_page", watch=True)
    def _update_display(self):
        """Update the apps display based on current filters and page"""
        # Get filtered apps
        filtered_apps = self._get_filtered_apps()

        # Get paginated apps
        paginated_apps = self._get_paginated_apps(filtered_apps)

        # Clear current display
        self._apps_container.clear()
        self._pagination.clear()

        if not paginated_apps:
            # No apps found
            no_apps_msg = pn.pane.HTML(
                "<div style='text-align: center; padding: 40px; color: #666;'>"
                "<h3>No apps found</h3>"
                "<p>Try adjusting your search terms.</p>"
                "</div>",
                sizing_mode="stretch_width",
            )
            self._apps_container.append(no_apps_msg)
        else:
            # Create grid of app cards
            self._apps_container[:] = [self._create_app_card(app) for app in paginated_apps]

        # Update pagination
        pagination_controls = self._create_pagination_controls(len(filtered_apps))
        if len(pagination_controls):
            self._pagination[:] = pagination_controls

    def __panel__(self):
        """Return the panel for notebook display"""
        return self._layout

    @classmethod
    def create_app(cls, **params):
        """Create a servable app instance"""
        instance = cls(**params)

        return pn.template.FastListTemplate(
            title="Office Hours",
            sidebar_width=0,  # No sidebar needed
            header_background=ACCENT_COLOR,
            main_layout=None,
            main=instance,
            favicon=r"assets\favicon-01.png",
        )


# Serve the app
if __name__ == "__main__":
    # Run with: python index.py
    AppGallery.create_app().show(
        port=5030, autoreload=True, open=True, static_dirs={"assets": "./assets"}
    )
elif pn.state.served:
    # Run with: panel serve *.py --index index.py --dev --show
    AppGallery.create_app().servable()

/tasks/setup.py

import sys
import threading
import time

import numpy as np
import panel as pn

running = True

import random


def random_dict():
    keys = ["x", "y", "z", "a", "b", "c", "d", "e", "f", "g"]
    # Generate a list of random integers corresponding to the number of keys
    random_values = [random.randint(1, 50) for _ in keys]

    return dict(zip(keys, random_values))


def worker():
    """
    Worker function to read data from network object and update the data in the Panel state cache.

    Parameters:
    None

    Returns:
    None
    """

    while running:
        time.sleep(0.001)
        pn.state.cache["data"] = random_dict()


def session_created(session_context):
    """
    Function to handle the creation of a new session.

    Parameters:
    - session_context (object): The context of the session being created.

    Returns:
    None
    """
    print(f"Created a session running at the {session_context.request.uri} endpoint")
    thread = threading.Thread(target=worker)
    thread.start()


pn.state.on_session_created(session_created)


def session_destroyed(session_context):
    """
    Stop the object and exit the program when a session is destroyed.

    Parameters:
    - session_context (object): The context of the session that was destroyed.

    Returns:
    None
    """
    print(f"Closed a session running at the {session_context.request.uri} endpoint")
    sys.exit()


pn.state.on_session_destroyed(session_destroyed)

Running the MRE using the following command and navigating between the apps via the index quickly breaks down.

python -m panel serve .\*.py --glob --setup .\tasks\setup.py --index index.py

Any help would be greatly appreciated.