textual-debugger (the package) provides tdb (the command-line tool and module),
a full-featured terminal-based Python debugger.
tdb is built with textual
and debugpy (the Debug Adapter Protocol engine behind
VS Code's Python debugger). It provides a rich interactive interface for stepping through code,
inspecting variables, managing breakpoints, and evaluating expressions in complex Python programs.
MIT License. Copyright 2026 by Al Danial.
tdb:
-
supports debugging of synchronous, asynchronous, multi-threaded, and multi-process Python code. It specifically supports modules
asyncio(with a built-in async task inspector and task wait graph)threading(with a thread inspector)multiprocessing/concurrent.futures(with automatic child process attachment and a process inspector)
-
supports remote attachment to debugpy-enabled Python programs
-
includes a JSON-RPC server mode that enables programmatic debug control, making it suitable for automated, headless debugging workflows and AI-assisted debugging
-
can spawn the debuggee in an external terminal to enable debugging TUI applications built with
textual,prompt-toolkit,urwid,curses,rich, and so on -
comes with a post-mortem exception hook that can be installed in Python programs to have
tdbpop open automatically at the first uncaught exception -
can be entirely keyboard-driven making it suitable for operation in non-graphical environments (mouse support is available in graphical environments)
Thank you:
-
Will McGugan for the amazing
textualmodule.tdbwould be a pale shadow of itself had I used any other TUI framework. Fantastic work, Will. -
Microsoft for the Debug Adapter Protocol (DAP) and releasing its implementation in
debugpyand the Python Debugger extension for Visual Studio Code as open source. -
Anthropic, for providing access to Claude Code through the Claude for Open Source program.
tdbwas made almost entirely with Claude Code.
Videos:
- tdb basics views, keybindings, breakpoints, stepping, variable modification, call stack
- asyncio tasks inspect asyncio tasks and their wait graph; code mod to allow pause
- threads and processes inspect variables and call stacks in multiple threads and processes
- external terminal run the debuggee in a separate terminal--ideal for debugging TUI applications
pip install textual-debuggeror (better):
uv pip install textual-debuggeror run it without installing:
uvx --from textual-debugger tdb my_program.py
# show comprehensive documentation in a terminal-based Markdown viewer
tdb --doc
# debug a script (stops at first line by default)
tdb my_program.py
# debug with arguments
tdb my_program.py arg1 arg2
# add breakpoints at lines 20 and 35 of `my_program.py` and line 14
# of `module.py` (when -k is given, --no-stop-on-entry is set and the
# program runs to the first breakpoint)
tdb -k 20 -k 35 -k module.py:14 my_program.py arg1 arg2
# use a specific virtualenv
tdb --python /path/to/venv/bin/python my_program.py
# step into, or stop at tracebacks in library code
tdb --no-just-my-code --python /path/to/venv/bin/python my_program.py
# run until first breakpoint or exit
tdb --no-stop-on-entry my_program.py
# run the debuggee in an external terminal
tdb --terminal xterm my_program.py
# attach to a remote Python program that has a debugpy server on port 5678
# (source code is automatically downloaded from the remote host)
tdb -r remotehost:5678
# attach to a remote Python program that has a debugpy server on port 5678
# and set a breakpoint where tdb and the remote program have the same
# source code layout
tdb -r remotehost:5678 -k my_program.py:42
# attach to a remote Python program that has a debugpy server on port 5678
# and set a breakpoint where code on the local host is at a different location
# than code on the remote host
tdb -r remotehost:5678 --local-root /my/code/dir --remote-root /app -k my_program.py:42
# separate tdb arguments from debuggee arguments with `--`
tdb --python /path/to/venv/bin/python -- my_program.py -k 17 --max 23.3Alternatively, use the module entry point:
python -m tdb my_program.py┌─ Header ──────────────────────────────────────────────┐
├─ Menu Bar (File / Configure / Help)───────────────────┤
│ │ │
│ Code View │ Console View (stdout) │
│ (source + breakpoints) ├───────────────────────────┤
│ │ Variable View (tree) │
│ ├───────────────────────────┤
│ │ Stack View (call stack) │
├─ Status Bar ──────────────────────────────────────────┤
│ │ │
│ Evaluate Console (REPL) │ Breakpoint View (table) │
│ │ │
├─ Footer (keybindings) ────────────────────────────────┤
└───────────────────────────────────────────────────────┘
The status bar shows the current execution state (running, paused, breakpoint hit) and location. The footer shows the most relevant keybindings for the current mode.
The Code View shows syntax-highlighted Python source with line numbers. A cursor line (blue) tracks your position; the current execution line is highlighted in gold.
View focus shortcuts (global):
| Key | View |
|---|---|
Ctrl+C |
Code View |
Ctrl+O |
Console View |
Ctrl+E |
Evaluate Console |
Ctrl+V |
Variable View |
Ctrl+S |
Stack View |
Ctrl+B |
Breakpoint View |
Menu-bar shortcuts (global):
Alt+<first-letter> opens the corresponding tab in the menu bar.
| Key | Menu |
|---|---|
Alt+F |
File (open a different script to debug) |
Alt+C |
Configure (Color Theme, Keybindings, Step Mode) |
Alt+T |
Threads |
Alt+P |
Processes |
Alt+A |
Async Tasks |
Alt+H |
Help (Documentation, About) |
Navigation (vim-style by default):
By default the Code View is in Debug mode. Hit Escape to switch to Navigate mode
In Navigate mode, you can move around the file with the following keys:
| Key | Action |
|---|---|
j / k |
Move cursor down / up |
5j, 10k |
Move N lines down / up with count prefix |
G |
Go to end of file (with count: 42G jumps to line 42) |
[ / ] |
Jump to previous / next paragraph boundary |
/ |
Search forward |
? |
Search backward |
n / N |
Next / previous search result |
PageUp / PageDown |
Scroll by page |
Switch from Navigate back to Debug mode with Escape.
Note: Many terminals send the byte sequence
ESC+fforAlt+F, which Textual's ANSI parser rewrites toCtrl+Right(the readline "forward-word" convention).tdbbinds both soAlt+Fworks as expected.
Keybindings for stepping, continuing, pausing, and stack navigation match those for gdb/pdb, with some aliases and extras thrown in for convenience.
| Key | Action |
|---|---|
n |
Step over (next statement) |
s |
Step into function call |
o / f / r |
Step out of current function (also aliased as "finish" and "return") |
c |
Continue execution |
p |
Pause a running program |
t |
Run to cursor position |
u / d |
Navigate stack up (caller) / down (callee) |
j / k |
Move cursor down / up (with count: 5j, 10k) |
G |
Go to last line (with count: 42G jumps to line 42) |
e |
Re-display the last error (traceback) |
R |
Restart the debug session |
q q |
Quit |
Ctrl+q |
Quit |
Note:
f("finish") andr("return") are both aliases for step-out. DAP's only "exit-a-function" primitive isstepOut, which runs the rest of the current function normally and stops at the return point. A true gdb-style immediate-return (skipping remaining code in the function without executing side effects) is not supported by DAP/debugpy.
Step granularity (statement vs. line): by default, n (step over) and s (step into)
treat a multi-line source statement as a single step. For example, stepping over
results = await asyncio.gather(
fetch(1, 2),
fetch(2, 1),
fetch(3, 3),
)
print(results) # next stop lands here, not on each sub-line abovelands on print(results), not on each interior sub-line of the gather call. Switch to
Line mode (Configure > Step Mode) to get debugpy's native per-line behavior, which
stops on each physical line--useful for inspecting how a complex expression is built up.
The choice is saved to ~/.config/tdb/config.json.
Click the gutter in the Code View to toggle a breakpoint, or press b in Debug mode.
Breakpoint indicators:
- Red dot: active breakpoint
- Yellow dot: conditional breakpoint
- Blue dot: disabled breakpoint
Conditional breakpoints: Double-click a breakpoint to open the condition editor.
Set a Python expression (e.g., x > 10) and/or a hit count (pause on the Nth hit).
Breakpoint View actions:
D: Disable / enable all breakpointsC: Clear all breakpoints
Breakpoints persist across session restarts.
The Variable View shows a tree of scopes (Locals, Globals) with all variables in the current frame. Expand nodes to drill into complex objects. Children are loaded lazily on demand. Variable values can be changed in the Evaluate Console.
Double-click a variable, or highlight the variable with the text cursor in
the Variables View and press Enter
to display that variable in a modal. This simplifies inspection of
large or deeply nested data structures.
The Stack View shows the full call stack. Click a frame to navigate to its source location and inspect its variables.
A read-evaluate-print loop (REPL) at the bottom left permits interactive evaluation of expressions in the current scope:
>>> len(items)
42
>>> sorted(data, key=lambda x: x.priority)[:3]
[Item(priority=1), Item(priority=2), Item(priority=3)]
- Up/Down arrows cycle through expression history
- Tab triggers DAP-based completion
- Trailing
?shows help (signature + docstring):
>>> os.path.join?
(a, *p) : Join two or more pathname components...
Variable values set here are reflected in the running code.
The Console View captures stdout (normal text) and stderr (red text) from the debuggee in real time.
If your program prints a lot, or prompts for input, or uses colors or
terminal control codes, run the program in an external terminal
with --terminal for a better experience.
The --terminal switch requires a graphical environment and a compatible
terminal emulator.
When the debuggee raises an unhandled exception, tdb:
- Shows a modal with the full traceback
- Navigates the Code View to the crash line
- Populates the Stack View with the exception's call stack
- Lets you press
Rto restart orEscapeto dismiss
Note: after dismissing the traceback modal, you can display it again by hitting
ewhen focus is in the Code View.
You can have tdb pop open automatically when any Python program crashes without the
need to launch through tdb up front. Install the hook once at the top of your program:
import sys
import tdb
sys.excepthook = tdb.exception_hookWhen an uncaught exception reaches the hook, tdb:
- Prints the standard Python traceback to stderr (so your scrollback still has a record)
- Snapshots every frame in the traceback. This includes locals, plus one level of
recursion into containers (
dict,list,tuple,set) and objects with__dict__ - Launches the TUI in post-mortem mode, inheriting the current terminal
In post-mortem mode you can:
- Navigate the call stack (
u/dor the Stack View) and see each frame's locals - Expand nested containers and object attributes in the Variables View
- Read the full traceback (including chained
cause/contextexceptions) in the Console View - Jump around the source with the full Code View (syntax highlighting, goto-line, etc.)
Stepping, continue, breakpoints, restart, and the Evaluate View are disabled. The original
interpreter is gone since the view is a frozen snapshot. Press q to exit.
The hook is a no-op when stdin/stdout aren't a tty (e.g. when your program is piped or
run from cron), so it is safe to leave installed in production code. Snapshots are
written to a temp file that is deleted as soon as tdb exits.
Snapshot depth / breadth is capped (5 levels, 50 children per container) to keep the capture cheap even for pathological object graphs; cycles are handled via identity memoization.
tdb has an improved implemenation of the standard breakpoint() function (or equivalently,
pdb.set_trace()) used to pause at a specific line to inspect, then
continuing--use tdb.breakpoint():
import tdb
def compute(n):
total = sum(range(n))
tdb.breakpoint() # pause here and drop into tdb
return totalOr hook it into the builtin breakpoint() function for the whole program:
PYTHONBREAKPOINT=tdb.breakpoint python myscript.pyWhen the call is reached, tdb starts an in-process debugpy server on a loopback port,
spawns python -m tdb -r <port> as a subprocess so the TUI takes over the terminal,
and pauses the calling thread at the line that called tdb.breakpoint() (the hook
auto-steps out of its own helper so you land in your own frame, not inside
breakpoint_hook.py). Stepping (n/s/o), continue, and setting/removing breakpoints
all work normally; quitting tdb (Ctrl+q) detaches without killing the program, and
debugpy auto-resumes any threads still paused.
This differs from tdb.exception_hook in one way:
- Requires
debugpyas a runtime dependency for the debuggee (only imported when the hook actually fires).
Unlike the exception hook (which works on a frozen snapshot), the breakpoint hook leaves
the interpreter live: variable inspection reads real objects, and stepping/continue
drive the user's program forward.
As with exception_hook, the call is a no-op when stdin/stdout aren't a tty, so it's
safe to leave in code paths that sometimes run headless.
Quitting tdb while paused in a tdb.breakpoint() session detaches the debugger and
lets the program continue running normally.
This behavior matches hitting c while in a conventional (that is, the Python
standard library's) breakpoint() session.
If you want to kill the program instead, use Ctrl+c in the terminal running the debuggee.
For programs using asyncio, the menu bar shows an Async Tasks (N) label with the count of
active tasks (updated each time the program stops). Click it to open a full-screen modal:
- Left pane: list of all tasks with name, state (pending/done/cancelled), awaiting
primitive (
Lock.acquire,Queue.get,asyncio.sleep, …), and coroutine - Right pane: detail view with full stack trace and an expandable variable tree (same as the main Variables View) for the selected task
- Press
gto switch the right pane to the wait graph which is a tree showing each blocked task, the asyncio primitive it's parked on, and the task(s) holding that primitive. Cycles (deadlocks) are highlighted in red both in the task table and as a "Deadlock cycles" section at the top of the graph. Selecting a node in the tree highlights the corresponding task in the table. - Navigate with arrow keys; press
rto refresh,Escapeto close
Note: Async task relationships may be directed acyclic graphs (DAGs) rather than trees but I don't know of a way to visualize DAGs in textual.
RPC equivalents:
# List all tasks
curl -s -X POST http://127.0.0.1:8150/rpc \
-H 'Content-Type: application/json' \
-d '{"action":"list_tasks","params":[]}'
# Inspect a specific task by name
curl -s -X POST http://127.0.0.1:8150/rpc \
-H 'Content-Type: application/json' \
-d '{"action":"inspect_task","params":["Task-1"]}'
# Show wait graph and any deadlock cycles
curl -s -X POST http://127.0.0.1:8150/rpc \
-H 'Content-Type: application/json' \
-d '{"action":"wait_graph","params":[]}'The menu bar shows a Threads (N) label when the program has 2 or more threads. Click it to open a modal with:
- Left pane: list of threads with ID and name (current thread shown in bold)
- Right pane: full stack trace and expandable variable tree for the selected thread's top frame
- Navigate with arrow keys; press
rto refresh,Escapeto close
RPC equivalents:
# List all threads (* marks current)
curl -s -X POST http://127.0.0.1:8150/rpc \
-H 'Content-Type: application/json' \
-d '{"action":"list_threads","params":[]}'
# Inspect a specific thread by ID
curl -s -X POST http://127.0.0.1:8150/rpc \
-H 'Content-Type: application/json' \
-d '{"action":"inspect_thread","params":[1]}'For programs using multiprocessing, the menu bar shows a Processes (N) label when there
are 2 or more child processes. Click it to open a modal with:
- Left pane: list of child processes with PID, name, and status (alive/exited)
- Right pane: process details, full stack trace, and expandable variable tree for the selected process
tdb automatically attaches to child processes spawned via multiprocessing.Process, multiprocessing.Pool,
or concurrent.futures.ProcessPoolExecutor. Breakpoints set in the parent are propagated to all
child processes. When any process hits a breakpoint, all other processes are paused. Pressing p
pauses all processes; c continues all.
Stepping in multi-process programs: step commands (n, s, o, f, r) apply only to the
process whose stack is currently shown in the Code View (the one that hit the breakpoint).
Other processes remain paused throughout the step. To step in a different process, open
the Processes tab and select it first. The Code View then switches focus to that process,
and subsequent step commands operate on it.
RPC equivalents:
# List all child processes
curl -s -X POST http://127.0.0.1:8150/rpc \
-H 'Content-Type: application/json' \
-d '{"action":"list_processes","params":[]}'
# Inspect a specific process by name or PID
curl -s -X POST http://127.0.0.1:8150/rpc \
-H 'Content-Type: application/json' \
-d '{"action":"inspect_process","params":["ForkPoolWorker-1"]}'Remote attachment is useful in situations where you can't launch the debuggee directly
with tdb, for example, if it is launched from another program or runs in an environment
where you can't install tdb. Two requirements must still be met though:
- the
debugpypackage must be installed in the debuggee's Python environment - you need write access to the debuggee's code to add the following code at the point where you want to attach the debugger:
# In the target program:
import debugpy
debugpy.listen(("0.0.0.0", 5678))
print("Waiting for tdb to attach on port 5678...")
debugpy.wait_for_client() # optional: pause until debugger connects
print("tdb is attached!")When the debuggee runs and hits the debugpy.wait_for_client() line, it starts a
debugpy server listening on port 5678.
Attach tdb to it with the -r switch, specifying the host and port.
If the debuggee is on the same machine, you can omit the host or use localhost.
This example assumes the debuggee runs on 192.168.1.10 and listens on port 5678:
# Attach from tdb:
tdb -r 5678 # to localhost
tdb -r 192.168.1.10:5678
# With breakpoints:
tdb -r 5678 -k my_program.py:42All debugging features (breakpoints, stepping, variable inspection, threads, processes, async tasks) work in remote attach mode. The Code View automatically navigates to the source file when the program stops.
Mapping remote paths to local copies (--local-root / --remote-root): when the
debuggee lives on another machine, or in a container, or simply in a different directory
on the same machine, the source paths it reports (and the paths it expects breakpoints
to refer to) won't match anything on the tdb host. To bridge that gap, give tdb one
or more --local-root / --remote-root pairs. Each --local-root points at a local
directory containing a copy of the code; each --remote-root is the corresponding path
on the debuggee. The two flags must be supplied in equal numbers and are paired in CLI
order via zip(), so the first --local-root matches the first --remote-root, the
second matches the second, and so on. debugpy then translates paths bidirectionally:
breakpoints set on a local file land on the matching remote file, and source paths
returned in stopped-events / stack-traces are rewritten back to the local copy so the
Code View loads directly from disk (no DAP source round-trip needed).
These flags are required whenever you want to set a -k breakpoint against a remote
debuggee whose code lives at a different path than your local copy. For example, if the
remote runs program.py at /path/to/code/program.py and your local copy is at
/local/project/code/program.py, set a breakpoint at line 321 with:
tdb -r RHOST:15678 \
--local-root /local/project/code \
--remote-root /path/to/code \
-k program.py:321With --local-root set, a relative -k FILE:LINE is resolved by searching each
--local-root directory in CLI order (first match wins); absolute paths still work as
before. Multiple pairs can be supplied to mirror multiple source trees (e.g. an
application directory and a shared library directory) in one invocation.
Some Python programs, notably text user interfaces, use terminal control
codes and require direct access to the terminal to function properly.
Such programs can be debugged with tdb by having it launch the debuggee in
a separate terminal:
tdb --terminal xterm my_tui_app.pyThe debuggee runs in a separate window of the specified terminal. Supported choices:
xterm, konsole, gnome-terminal, ghostty, kitty, iterm2, warp,
wezterm, terminator. The selected terminal must be on PATH. Debugging
proceeds as usual in the terminal where tdb was invoked.
This feature only works in graphical environments where external terminals are available.
tdb --keybindings vim my_program.py # default
tdb --keybindings emacs my_program.py
tdb --keybindings default my_program.pyThe keybinding choice is saved to ~/.config/tdb/config.json and remembered for subsequent
runs. View the full keybinding reference from the menu: Configure > Keybindings.
tdb includes a built-in debug server for programmatic control which is useful for scripted
debugging, CI pipelines, or AI-assisted debugging workflows.
python -m tdb --headless my_program.py &The server listens on http://127.0.0.1:8150/rpc (change with --server-port).
tdb --server my_program.pyBoth the interactive TUI and the JSON-RPC server run simultaneously.
Send POST requests with {"action": "...", "params": [...]}. Responses return
{"timestamp": "...", "success": true/false, "value": "..."}.
# Check status
curl -s -X POST http://127.0.0.1:8150/rpc \
-H 'Content-Type: application/json' \
-d '{"action":"status","params":[]}'
# Set a breakpoint
curl -s -X POST http://127.0.0.1:8150/rpc \
-H 'Content-Type: application/json' \
-d '{"action":"set_breakpoint","params":["/abs/path/to/file.py:42"]}'
# Continue execution
curl -s -X POST http://127.0.0.1:8150/rpc \
-H 'Content-Type: application/json' \
-d '{"action":"continue","params":[]}'
# Inspect variables
curl -s -X POST http://127.0.0.1:8150/rpc \
-H 'Content-Type: application/json' \
-d '{"action":"inspect","params":["x", "len(items)", "type(result)"]}'
# Shut down
curl -s -X POST http://127.0.0.1:8150/rpc \
-H 'Content-Type: application/json' \
-d '{"action":"quit","params":[]}'| Action | Params | Description |
|---|---|---|
help |
[] |
List all actions |
status |
[] |
Current state with location |
set_breakpoint |
["file:line"] or ["file:line", "condition", "hit_condition"] |
Set a breakpoint |
remove_breakpoint |
["file:line"] |
Remove a breakpoint |
list_breakpoints |
[] |
Show all breakpoints |
continue |
[] |
Resume execution |
next |
[] |
Step over |
step_in |
[] |
Step into |
step_out |
[] |
Step out |
pause |
[] |
Pause execution |
inspect |
["expr1", "expr2", ...] |
Evaluate multiple expressions |
evaluate |
["expression"] |
Evaluate a single expression |
stack_up |
[] |
Move up the call stack |
stack_down |
[] |
Move down the call stack |
get_stack_trace |
[] |
Full call stack |
get_output |
[] |
Drain buffered stdout/stderr |
get_source |
["file_path"] |
Read a source file |
list_threads |
[] |
List all threads |
inspect_thread |
[thread_id] |
Inspect a specific thread |
list_processes |
[] |
List child processes (multiprocessing) |
inspect_process |
["name_or_pid"] |
Inspect a specific child process |
list_tasks |
[] |
List all asyncio tasks |
inspect_task |
["task_name"] |
Inspect a specific asyncio task |
wait_graph |
[] |
Show wait graph + any deadlock cycles |
restart |
[] |
Restart session (preserves breakpoints) |
quit |
[] |
Shut down |
Subscribe to real-time debug events:
curl -N http://127.0.0.1:8150/eventsEvents: initialized, stopped, continued, terminated, exited, output.
Each is JSON with event, data, and timestamp fields.
usage: tdb [-h] [-v] [-r [HOST:]PORT] [--cwd CWD] [--no-stop-on-entry]
[--no-just-my-code] [--no-subprocess] [--python PYTHON] [--pv]
[--keybindings {default,vim,emacs}]
[--terminal {xterm,konsole,gnome-terminal,ghostty,kitty,iterm2,warp,wezterm,terminator}]
[--local-root PATH] [--remote-root PATH]
[--server] [--headless] [-k FILE:LINE|LINE] [--server-port SERVER_PORT] [-d] [--doc-text]
[program] [args ...]
| Flag | Description |
|---|---|
-r HOST:PORT |
Attach to a remote debugpy server |
--local-root PATH |
Local directory containing a copy of remote code (repeat to mirror multiple trees). Pair with --remote-root; counts must match. Required when -k sets a breakpoint against a remote debuggee whose code lives at a different path. |
--remote-root PATH |
Remote directory matched to --local-root (same CLI position via zip()). |
-k, `--breakpoint FILE:LINE |
LINE` |
--no-stop-on-entry |
Do not pause at the first line (default: stop on entry; automatic when -k is given) |
--cwd DIR |
Working directory for the debuggee |
--python PATH |
Python interpreter for the debuggee |
--pv |
Shorthand for --python .venv/bin/python |
--no-just-my-code |
Step into stdlib/site-packages code instead of skipping it |
| (default: skipped). On uncaught exceptions, the crash modal always shows the full traceback | |
| including library frames, regardless of this flag. | |
--no-subprocess |
Disable debugpy's subprocess tracking (use when debugging tdb itself) |
--terminal TERM |
Run debuggee in the named external terminal: xterm, konsole, |
gnome-terminal, ghostty, kitty, iterm2, warp, wezterm, or terminator |
|
--keybindings SCHEME |
default, vim, or emacs (saved to config) |
--server |
Enable JSON-RPC server alongside TUI |
--headless |
JSON-RPC server only, no TUI |
--server-port PORT |
Server port (default: 8150) |
On UNIX-like systems (Linux, macOS, FreeBSD, etc.),
tdb stores configuration and breakpoints in ~/.config/tdb/.
On Windows, it uses %APPDATA%\tdb\.
| File | Contents |
|---|---|
config.json |
User preferences (keybinding scheme, color theme, step mode) |
breakpoints.json |
Breakpoints from previous sessions, keyed by project directory |
Breakpoints are saved on exit and restored when debugging a program in the same directory. Each project's breakpoints are independent.
Step mode (step_mode in config.json) controls how n (step over) and s (step
into) handle multi-line source statements:
| Value | Behavior |
|---|---|
"statement" (default) |
A multi-line statement (e.g. a gather(...) call spanning five lines) is one step. The debugger keeps issuing DAP steps until execution leaves the statement, then stops on the next logical line. |
"line" |
debugpy's native per-line behavior (stops on every physical line, including each interior sub-line of a multi-line expression) |
Change it from the menu (Configure > Step Mode); the choice is saved immediately and applies to all future sessions. Breakpoint hits, exceptions, and pauses always interrupt a statement step, so a breakpoint set on a sub-line of a multi-line expression still fires as expected.
- textual : TUI framework
- debugpy : Debug Adapter Protocol implementation
- pygments : Syntax highlighting
- FastAPI + uvicorn : JSON-RPC server
MIT
This command
tdb --terminal gnome-terminal --python /path/to/venv/matplotlib/bin/python3 examples/double_pendulum.py
either ignores breakpoints or crashes after showing the first frame.
The --python argument must point to an installation with matplotlib.



