| name | flight-profiler-watch |
| description | Display input/output args, return object and cost time of a Python method invocation in a live process. Use this to observe function behavior at runtime ā see what arguments go in, what comes back, and how long it takes. This is the most frequently used diagnostic command. |
flight-profiler-watch
Display the input/output args, return object and cost time of method invocation. This is the most commonly used command for understanding runtime behavior of a specific function.
Prerequisites: Read the flight-profiler-attach skill first for platform requirements, installation, permissions, and connection details.
When to Use
- You want to see what arguments a function receives and what it returns
- You need to measure how long a specific function call takes
- You want to filter calls by argument values or return values
- You need to catch only exception-throwing invocations
- You want to observe a method's behavior without modifying its source code
Usage
flight_profiler <pid> --cmd "watch module [class] method [options]" --no-color
Use -n to limit capture count so the command auto-exits.
Positional Arguments
module ā the module name as it would be imported in the target process. For example, if the target code does from myapp.utils import helper, then module is myapp.utils. PyFlightProfiler locates the module via importlib.import_module. If you're unsure of the module name, run a separate command to resolve it first: flight_profiler <pid> --cmd "module /absolute/path/to/file.py" --no-color, then use the returned module name here.
class (optional) ā class name, omit if the method is a module-level function
method ā target method name
Options
| Flag | Description | Default |
|---|
--expr <value> | Watch expression. Available variables: target, return_obj, *args, **kwargs (note: cost is NOT available in expr, only in filter) | args,kwargs |
-x, --expand <value> | Object tree expand level (1-4, or -1 for infinity) | 1 |
-e, --exception | Only record when method throws exception | off |
-nm, --nested-method <value> | Watch nested method with depth restrict to 1 | none |
-r, --raw | Use __str__ (equivalent to print(obj)) instead of default JSON serialization | off |
-v, --verbose | Display all nested items in target list or dict (no truncation) | off |
-n, --limits <value> | Max display count, auto-stops after reaching limit | 10 |
-f, --filter <value> | Filter expression. Available variables: target, return_obj, cost, args, kwargs | none |
Choosing the Right --expr
--expr directly controls what you see in the output. Always match the user's intent to the right expression ā the default args,kwargs only shows inputs, not outputs.
| User intent | --expr value |
|---|
| See input arguments | args,kwargs (default) |
| See return value / output | return_obj |
| See both input and output | args,kwargs,return_obj |
| See input, output, and the object itself | target,args,kwargs,return_obj |
| See a specific return field | return_obj['field_name'] |
| See the class instance (self) | target |
| See everything | target,args,kwargs,return_obj |
When the user says "input/output", "args and return", "what goes in and comes back", or similar ā always use --expr args,kwargs,return_obj, never just the default.
Output Format
Each captured invocation produces output in this format:
āāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāā
ā 2026-04-19 10:33:36:520 method=__main__.compute cost=2.380133ms is_exp=False result={
EXPR: args,kwargs
TYPE: <class 'tuple'>
VALUE: (
(25, 26),
{}
)
}
Output Fields
| Field | Description |
|---|
ā / ā | Success or exception indicator |
| Timestamp | When the method was called (yyyy-MM-dd HH:mm:ss:SSS) |
method | Fully qualified method name: module.method or module.class.method |
cost | Execution time in milliseconds (precision to microseconds) |
is_exp | True if the method threw an exception, False otherwise |
EXPR | The expression used to extract the displayed value |
TYPE | Python type of the expression result |
VALUE | The expression result, formatted as JSON (or __str__ if -r) |
EXCEPTION | (only when is_exp=True) Full traceback of the exception |
Examples
Watch a module-level function (default args,kwargs)
flight_profiler <pid> --cmd "watch __main__ compute -n 1" --no-color
Output:
āāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāā
ā 2026-04-19 10:33:36:520 method=__main__.compute cost=2.380133ms is_exp=False result={
EXPR: args,kwargs
TYPE: <class 'tuple'>
VALUE: (
(
25,
26
),
{}
)
}
Watch a module-level function ā only return value
flight_profiler <pid> --cmd "watch __main__ compute --expr return_obj -n 1" --no-color
Output:
āāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāā
ā ... method=__main__.compute cost=2.561092ms is_exp=False result={
EXPR: return_obj
TYPE: <class 'int'>
VALUE: 89
}
Watch a function with kwargs
flight_profiler <pid> --cmd "watch __main__ fetch_data -n 1" --no-color
Output ā args contains positional args, kwargs contains keyword args:
āāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāā
ā ... method=__main__.fetch_data cost=0.010014ms is_exp=False result={
EXPR: args,kwargs
TYPE: <class 'tuple'>
VALUE: (
(
"SELECT *"
),
{
"limit": 5,
"timeout": 30
}
)
}
Watch a class instance method
flight_profiler <pid> --cmd "watch __main__ UserService get_user -n 1" --no-color
Output ā for class methods, self is automatically stripped from args:
āāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāā
ā ... method=__main__.UserService.get_user cost=10.342121ms is_exp=False result={
EXPR: args,kwargs
TYPE: <class 'tuple'>
VALUE: (
(
46
),
{}
)
}
Use --expr to see target (self) and return value
flight_profiler <pid> --cmd "watch __main__ UserService get_user --expr target,return_obj -n 1" --no-color
Output ā target is the class instance (self), shown with its attributes:
āāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāā
ā ... method=__main__.UserService.get_user cost=10.831118ms is_exp=False result={
EXPR: target,return_obj
TYPE: <class 'tuple'>
VALUE: (
UserService({}),
{
"active": True,
"id": 56,
"name": "user_56"
}
)
}
Use --expr to extract a specific return field
flight_profiler <pid> --cmd "watch __main__ UserService get_user --expr return_obj['name'] -n 1" --no-color
Output:
āāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāā
ā ... method=__main__.UserService.get_user cost=12.423992ms is_exp=False result={
EXPR: return_obj['name']
TYPE: <class 'str'>
VALUE: "user_68"
}
Watch class method with expanded output
flight_profiler <pid> --cmd "watch __main__ UserService get_user --expr return_obj -x 2 -n 1" --no-color
Output:
āāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāā
ā ... method=__main__.UserService.get_user cost=12.030125ms is_exp=False result={
EXPR: return_obj
TYPE: <class 'dict'>
VALUE: {
"active": True,
"id": 81,
"name": "user_81"
}
}
Capture exceptions with -e
flight_profiler <pid> --cmd "watch __main__ UserService delete_user -e -n 1" --no-color
Output ā note ā, is_exp=True, and the EXCEPTION field:
āāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāā
ā ... method=__main__.UserService.delete_user cost=0.024080ms is_exp=True result={
EXPR: args,kwargs
TYPE: <class 'tuple'>
VALUE: (
(
58
),
{}
)
EXCEPTION: Traceback (most recent call last):
...
File "watch_demo_script.py", line 11, in delete_user
raise ValueError(f"Cannot delete user {user_id}")
ValueError: Cannot delete user 58
}
Watch with filter ā only slow calls
flight_profiler <pid> --cmd "watch __main__ compute -f cost>10 -n 3" --no-color
Watch with filter ā by argument value
flight_profiler <pid> --cmd "watch __main__ compute -f args[0]>2 -n 1" --no-color
Watch nested inner function with -nm
flight_profiler <pid> --cmd "watch __main__ UserService nested_outer -nm inner_calc -n 1" --no-color
Output ā method shows the full path including nested function name:
āāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāā
ā ... method=__main__.UserService.nested_outer.inner_calc cost=0.071049ms is_exp=False result={
EXPR: args,kwargs
TYPE: <class 'tuple'>
VALUE: (
(),
{}
)
}
Raw mode -r (use __str__ instead of JSON serialization)
flight_profiler <pid> --cmd "watch __main__ compute -r -n 1" --no-color
Output ā VALUE uses __str__ (like print(obj)) instead of default JSON serialization:
āāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāāā
ā ... method=__main__.compute cost=2.539158ms is_exp=False result={
EXPR: args,kwargs
TYPE: <class 'tuple'>
VALUE: ((82, 83), {})
}
--expr Expression Guide
The --expr option controls what data is displayed for each invocation. It accepts any valid Python expression.
Available variables:
target ā the class instance (self) for class methods; None for module-level functions
return_obj ā the method's return value (None if exception was thrown)
*args ā positional arguments (for class methods, self is already stripped)
**kwargs ā keyword arguments
Note: cost is NOT available in --expr. Use -f filter to filter by cost.
Common patterns:
--expr args,kwargs
--expr return_obj
--expr args,return_obj
--expr return_obj['key']
--expr return_obj['data']['items']
--expr target
--expr target,return_obj
--expr len(args[0])
--expr type(return_obj)
-f Filter Guide
The -f option controls which invocations are displayed. Only invocations where the filter expression evaluates to True are shown.
Available variables:
target ā the class instance (or None)
return_obj ā the return value
cost ā execution time in milliseconds
args ā positional arguments tuple
kwargs ā keyword arguments dict
Common patterns:
-f cost>10
-f cost>100
-f args[0]>100
-f args[0]=="hello"
-f args[0]["query"]=="SELECT *"
-f return_obj is not None
-f return_obj['success']==True
-f len(return_obj)>0
-f kwargs.get('debug')==True
-f cost>10 and args[0]>0
-f return_obj is not None and cost<5
Troubleshooting: No Output
If watch produces no output, the most likely reason is the target method is not being called during the observation window. Before concluding there's a problem:
- Confirm the method is on the active call path ā make sure the code path that invokes the target method is actually being triggered (e.g., send a request, trigger the workflow)
- Check the module name ā use the
module command to verify the correct module name if unsure
- Check for filter issues ā if using
-f, the filter might be too restrictive. Try without -f first
- Check
-n limits ā if limits was reached in a previous watch, the method may already be unwatched
Handling Command Output
Handling Large Objects
When an object is too large to serialize or the output is truncated/incomplete:
-
Prefer minimal extraction ā use --expr to extract only the fields you need, instead of the entire object:
--expr return_obj
--expr return_obj['status']
--expr return_obj['data']['id']
--expr type(return_obj),len(return_obj)
-
Use -v (verbose) ā if you do need the full object, add -v to disable truncation so all nested items in lists/dicts are shown completely.
Tips
- The
-f filter runs inside the target process so it can reference live objects
- The
--expr option controls what gets printed; the -f option controls which invocations get printed
cost is only available in -f filter, not in --expr
- For class methods,
self is automatically stripped from args ā use target to access the instance
- Use
-x 2 or -x 3 to expand nested objects in the output (max 4, or -1 for infinity)
-r (raw) uses __str__ (like print(obj)) instead of JSON serialization ā useful when objects have custom __str__ or you want the native Python representation
-v (verbose) shows all items in lists/dicts without truncation
Related Commands
- trace ā shows full call tree with timing (drill into sub-calls)
- reload ā hot-patch a function after editing its source file, without restarting the process. Use watch to inspect internal behavior before/after reload when you need to observe arguments, return values, or timing changes that aren't visible from normal program output
- getglobal ā inspect a specific global/static field value without watching calls
Source Files
- CLI plugin:
flight_profiler/plugins/watch/cli_plugin_watch.py
- Parser:
flight_profiler/plugins/watch/watch_parser.py
- Server plugin:
flight_profiler/plugins/watch/server_plugin_watch.py