Skip to content
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
28 commits
Select commit Hold shift + click to select a range
bac7bfa
gh-141645: Add a TUI mode to the new tachyon profiler
pablogsal Nov 16, 2025
757ce0a
use sampling rate in progress bar
pablogsal Nov 17, 2025
85bada1
Fix failed samples calculation
pablogsal Nov 17, 2025
4a7df35
Add GIL flags
pablogsal Nov 17, 2025
d45f012
fix sample rate
pablogsal Nov 17, 2025
b1b9614
Update Lib/profiling/sampling/live_collector.py
pablogsal Nov 17, 2025
af9dd77
Update Lib/profiling/sampling/live_collector.py
pablogsal Nov 17, 2025
0e70587
Update Lib/profiling/sampling/sample.py
pablogsal Nov 17, 2025
7a43706
refactor live mode
pablogsal Nov 18, 2025
f597f8e
Implement color management and and S
pablogsal Nov 18, 2025
9f12a5c
Remove exception arg
pablogsal Nov 18, 2025
e2ff9fc
Fix tests
pablogsal Nov 18, 2025
6b21c9a
Remove pass
pablogsal Nov 18, 2025
4473aa9
Small refactor
pablogsal Nov 18, 2025
20c4d27
Split tests
pablogsal Nov 18, 2025
142b116
Implement per-thread switcher
pablogsal Nov 18, 2025
1543ce5
Supress stdin to not leave things hanging
pablogsal Nov 18, 2025
00dc8e7
Simplify
pablogsal Nov 18, 2025
5035156
Fix HZ update
pablogsal Nov 18, 2025
b69ebdd
Merge remote-tracking branch 'upstream/main' into live-tui
pablogsal Nov 18, 2025
1034708
Reorder tests
pablogsal Nov 18, 2025
1864e17
Fix tests
pablogsal Nov 18, 2025
c160d8d
Implement color diffing
pablogsal Nov 18, 2025
985680c
Change default sorting
pablogsal Nov 18, 2025
b5550f5
Fix tests
pablogsal Nov 18, 2025
5ef23e3
Check for curses
pablogsal Nov 18, 2025
99b19e0
UX improvements
pablogsal Nov 18, 2025
a51766a
Address feedback and ensuring the TUI works at the end
pablogsal Nov 20, 2025
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
19 changes: 9 additions & 10 deletions Lib/profiling/sampling/collector.py
Original file line number Diff line number Diff line change
@@ -1,20 +1,19 @@
from abc import ABC, abstractmethod

# Thread status flags
try:
from _remote_debugging import THREAD_STATUS_HAS_GIL, THREAD_STATUS_ON_CPU, THREAD_STATUS_UNKNOWN, THREAD_STATUS_GIL_REQUESTED
except ImportError:
# Fallback for tests or when module is not available
THREAD_STATUS_HAS_GIL = (1 << 0)
THREAD_STATUS_ON_CPU = (1 << 1)
THREAD_STATUS_UNKNOWN = (1 << 2)
THREAD_STATUS_GIL_REQUESTED = (1 << 3)
from .constants import (
THREAD_STATUS_HAS_GIL,
THREAD_STATUS_ON_CPU,
THREAD_STATUS_UNKNOWN,
THREAD_STATUS_GIL_REQUESTED,
)

class Collector(ABC):
@abstractmethod
def collect(self, stack_frames):
"""Collect profiling data from stack frames."""

def collect_failed_sample(self):
"""Collect data about a failed sample attempt."""

@abstractmethod
def export(self, filename):
"""Export collected data to a file."""
Expand Down
30 changes: 30 additions & 0 deletions Lib/profiling/sampling/constants.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,30 @@
"""Constants for the sampling profiler."""

# Profiling mode constants
PROFILING_MODE_WALL = 0
PROFILING_MODE_CPU = 1
PROFILING_MODE_GIL = 2
PROFILING_MODE_ALL = 3 # Combines GIL + CPU checks

# Sort mode constants
SORT_MODE_NSAMPLES = 0
SORT_MODE_TOTTIME = 1
SORT_MODE_CUMTIME = 2
SORT_MODE_SAMPLE_PCT = 3
SORT_MODE_CUMUL_PCT = 4
SORT_MODE_NSAMPLES_CUMUL = 5

# Thread status flags
try:
from _remote_debugging import (
THREAD_STATUS_HAS_GIL,
THREAD_STATUS_ON_CPU,
THREAD_STATUS_UNKNOWN,
THREAD_STATUS_GIL_REQUESTED,
)
except ImportError:
# Fallback for tests or when module is not available
THREAD_STATUS_HAS_GIL = (1 << 0)
THREAD_STATUS_ON_CPU = (1 << 1)
THREAD_STATUS_UNKNOWN = (1 << 2)
THREAD_STATUS_GIL_REQUESTED = (1 << 3)
200 changes: 200 additions & 0 deletions Lib/profiling/sampling/live_collector/__init__.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,200 @@
"""Live profiling collector that displays top-like statistics using curses.

┌─────────────────────────────┐
│ Target Python Process │
│ (being profiled) │
└──────────────┬──────────────┘
│ Stack sampling at
│ configured interval
│ (e.g., 10000µs)
┌─────────────────────────────┐
│ LiveStatsCollector │
│ ┌───────────────────────┐ │
│ │ collect() │ │ Aggregates samples
│ │ - Iterates frames │ │ into statistics
│ │ - Updates counters │ │
│ └───────────┬───────────┘ │
│ │ │
│ ▼ │
│ ┌───────────────────────┐ │
│ │ Data Storage │ │
│ │ - result dict │ │ Tracks per-function:
│ │ - direct_calls │ │ • Direct samples
│ │ - cumulative_calls │ │ • Cumulative samples
│ └───────────┬───────────┘ │ • Derived time stats
│ │ │
│ ▼ │
│ ┌───────────────────────┐ │
│ │ Display Update │ │
│ │ (10Hz by default) │ │ Rate-limited refresh
│ └───────────┬───────────┘ │
└──────────────┼──────────────┘
┌─────────────────────────────┐
│ DisplayInterface │
│ (Abstract layer) │
└──────────────┬──────────────┘
┌───────┴────────┐
│ │
┌──────────▼────────┐ ┌───▼──────────┐
│ CursesDisplay │ │ MockDisplay │
│ - Real terminal │ │ - Testing │
│ - ncurses backend │ │ - No UI │
└─────────┬─────────┘ └──────────────┘
┌─────────────────────────────────────┐
│ Widget-Based Rendering │
│ ┌─────────────────────────────────┐ │
│ │ HeaderWidget │ │
│ │ • PID, uptime, time, interval │ │
│ │ • Sample stats & progress bar │ │
│ │ • Efficiency bar │ │
│ │ • Thread status & GC stats │ │
│ │ • Function summary │ │
│ │ • Top 3 hottest functions │ │
│ ├─────────────────────────────────┤ │
│ │ TableWidget │ │
│ │ • Column headers (sortable) │ │ Interactive display
│ │ • Stats rows (scrolling) │ │ with keyboard controls:
│ │ - nsamples % time │ │ s: sort, p: pause
│ │ - function file:line │ │ r: reset, /: filter
│ ├─────────────────────────────────┤ │ q: quit, h: help
│ │ FooterWidget │ │
│ │ • Legend and status │ │
│ │ • Filter input prompt │ │
│ └─────────────────────────────────┘ │
└─────────────────────────────────────┘

Architecture:

The live collector is organized into four layers. The data collection layer
(LiveStatsCollector) aggregates stack samples into per-function statistics without
any knowledge of how they will be presented. The display abstraction layer
(DisplayInterface) defines rendering operations without coupling to curses or any
specific UI framework. The widget layer (Widget, HeaderWidget, TableWidget,
FooterWidget, HelpWidget, ProgressBarWidget) encapsulates individual UI components
with their own rendering logic, promoting modularity and reusability. The
presentation layer (CursesDisplay/MockDisplay) implements the actual rendering for
terminal output and testing.

The system runs two independent update loops. The sampling loop is driven by the
profiler at the configured interval (e.g., 10000µs) and continuously collects
stack frames and updates statistics. The display loop runs at a fixed refresh rate
(default 10Hz) and updates the terminal independently of sampling frequency. This
separation allows high-frequency sampling without overwhelming the terminal with
constant redraws.

Statistics are computed incrementally as samples arrive. The collector maintains
running counters (direct calls and cumulative calls) in a dictionary keyed by
function location. Derived metrics like time estimates and percentages are computed
on-demand during display updates rather than being stored, which minimizes memory
overhead as the number of tracked functions grows.

User input is processed asynchronously during display updates using non-blocking I/O.
This allows interactive controls (sorting, filtering, pausing) without interrupting
the data collection pipeline. The collector maintains mode flags (paused,
filter_input_mode) that affect what gets displayed but not what gets collected.

"""

# Re-export all public classes and constants for backward compatibility
from .collector import LiveStatsCollector
from .display import DisplayInterface, CursesDisplay, MockDisplay
from .widgets import (
Widget,
ProgressBarWidget,
HeaderWidget,
TableWidget,
FooterWidget,
HelpWidget,
)
from .constants import (
MICROSECONDS_PER_SECOND,
DISPLAY_UPDATE_HZ,
DISPLAY_UPDATE_INTERVAL,
MIN_TERMINAL_WIDTH,
MIN_TERMINAL_HEIGHT,
WIDTH_THRESHOLD_SAMPLE_PCT,
WIDTH_THRESHOLD_TOTTIME,
WIDTH_THRESHOLD_CUMUL_PCT,
WIDTH_THRESHOLD_CUMTIME,
HEADER_LINES,
FOOTER_LINES,
SAFETY_MARGIN,
TOP_FUNCTIONS_DISPLAY_COUNT,
COL_WIDTH_NSAMPLES,
COL_SPACING,
COL_WIDTH_SAMPLE_PCT,
COL_WIDTH_TIME,
MIN_FUNC_NAME_WIDTH,
MAX_FUNC_NAME_WIDTH,
MIN_AVAILABLE_SPACE,
MIN_BAR_WIDTH,
MAX_SAMPLE_RATE_BAR_WIDTH,
MAX_EFFICIENCY_BAR_WIDTH,
MIN_SAMPLE_RATE_FOR_SCALING,
FINISHED_BANNER_EXTRA_LINES,
COLOR_PAIR_HEADER_BG,
COLOR_PAIR_CYAN,
COLOR_PAIR_YELLOW,
COLOR_PAIR_GREEN,
COLOR_PAIR_MAGENTA,
COLOR_PAIR_RED,
COLOR_PAIR_SORTED_HEADER,
DEFAULT_SORT_BY,
DEFAULT_DISPLAY_LIMIT,
)

__all__ = [
# Main collector
"LiveStatsCollector",
# Display interfaces
"DisplayInterface",
"CursesDisplay",
"MockDisplay",
# Widgets
"Widget",
"ProgressBarWidget",
"HeaderWidget",
"TableWidget",
"FooterWidget",
"HelpWidget",
# Constants
"MICROSECONDS_PER_SECOND",
"DISPLAY_UPDATE_HZ",
"DISPLAY_UPDATE_INTERVAL",
"MIN_TERMINAL_WIDTH",
"MIN_TERMINAL_HEIGHT",
"WIDTH_THRESHOLD_SAMPLE_PCT",
"WIDTH_THRESHOLD_TOTTIME",
"WIDTH_THRESHOLD_CUMUL_PCT",
"WIDTH_THRESHOLD_CUMTIME",
"HEADER_LINES",
"FOOTER_LINES",
"SAFETY_MARGIN",
"TOP_FUNCTIONS_DISPLAY_COUNT",
"COL_WIDTH_NSAMPLES",
"COL_SPACING",
"COL_WIDTH_SAMPLE_PCT",
"COL_WIDTH_TIME",
"MIN_FUNC_NAME_WIDTH",
"MAX_FUNC_NAME_WIDTH",
"MIN_AVAILABLE_SPACE",
"MIN_BAR_WIDTH",
"MAX_SAMPLE_RATE_BAR_WIDTH",
"MAX_EFFICIENCY_BAR_WIDTH",
"MIN_SAMPLE_RATE_FOR_SCALING",
"FINISHED_BANNER_EXTRA_LINES",
"COLOR_PAIR_HEADER_BG",
"COLOR_PAIR_CYAN",
"COLOR_PAIR_YELLOW",
"COLOR_PAIR_GREEN",
"COLOR_PAIR_MAGENTA",
"COLOR_PAIR_RED",
"COLOR_PAIR_SORTED_HEADER",
"DEFAULT_SORT_BY",
"DEFAULT_DISPLAY_LIMIT",
]
Loading
Loading