This skill is complementary to
argent-react-native-optimization
, not a replacement for it.
2. Tool Overview
React Profiler (Hermes / React commits)
| Tool | Purpose |
|---|
| Start CPU sampling + inject React commit-capture hook. Optional: (default 100). |
| Stop recording; stores cpuProfile + commitTree in session. |
| Call if you were interrupted in the middle of the flow, never in another scenario (debugger drop, Metro reload, pause, subagent handoff, any doubt). Returns session_status: "active" | "taken_over" | "stopped" | "no_react_runtime"
. Side-effect free. |
| Run pipeline -> report with CPU-enriched hot commits, sorted by DESC. Saves raw data to disk. |
react-profiler-component-source
| AST lookup: file, line, memoization status, 50 lines of source for a component. |
| Live fiber walk: render counts + durations per component (no profiling session required). |
react-profiler-fiber-tree
| Live fiber walk: full component hierarchy as JSON. |
Drill-Down Query Tools (call after analyze)
| Tool | Purpose |
|---|
| Targeted CPU investigation: top functions, time-windowed CPU, call trees, per-component CPU. |
| Targeted commit investigation: by component, time range, commit index, or cascade tree. |
| iOS Instruments drill-down: hang stacks, function callers, thread breakdown, leak details. |
| Cross-correlated report when both React Profiler and native profiler ran in parallel. |
| List and reload previous profiling sessions from disk for re-investigation with query tools. |
For native profiling (CPU hotspots, UI hangs, memory leaks), see the
skill.
3. Agent Behavior Guidelines
Follow these rules throughout the profiling workflow:
- Start and in parallel (two tool calls in one message). Both need ; use the same UDID for both so their data can be correlated later. This gives best coverage.
- If the user only wants native profiling, use the skill workflow. Only skip if the user has already explicitly said they don't want native profiling in this session
After analysis: ask about next steps
After presenting the analysis report, always ask the user what they want to do next. Present these options:
- Investigate further — drill down into specific findings using query tools (CPU call trees, commit cascades, hang stacks, etc.) to identify root causes with confidence before making changes.
- Implement fixes — apply changes based on the current findings, then re-profile to measure whether the metric changed (improved, regressed, or stayed flat).
- Done for now — accept the report as-is.
Do NOT silently move on after the report. The report is the starting point, not the end — query tools exist specifically to let you dig deeper into anything the report flags.
During investigation: use query tools proactively
When drilling down, chain query tool calls based on what you find:
- A hot commit -> mode= to see all components -> mode= for the slowest one -> mode= for the hot function -> read the source file -> propose a fix.
- A memory leak -> mode= to identify the responsible module -> read the native source if actionable.
- A native hang -> mode= to get the native call chain -> correlate with React commit timing.
After fixes: always re-profile
When you apply a fix, always re-profile the same scenario afterward. Compare before/after metrics (commit durations, CPU time, render counts) and report honestly: did the target metric improve, stay flat, or regress? Did any
other metric get worse? If you need to reference the original data, use
to reload the pre-fix session. If the fix showed no improvement or introduced a regression, say so explicitly and reconsider the approach.
Use flows for reproducible profiling
When profiling requires a specific interaction sequence (scroll a list, navigate screens, trigger an animation),
record the interaction as a flow using the
skill before the first profiling run. Then replay the same flow for every subsequent run. This eliminates interaction variance as a confounder and makes before/after comparisons meaningful. Especially important when:
- You are about to re-profile after applying a fix (Step 8).
- The user asks you to compare multiple profiling sessions.
- The interaction path is more than 2-3 steps long.
4. Standard Profiling Workflow
Complete all steps in order — do not break mid-flow.
Step 1: Start profiling
Mind the react-native and ios-native profiler selection mentioned above when starting the session and start the tools.
Save from the response — you will need it for annotation offsets. Every subsequent profiler/query call in this session must use the same
. Before beginning, define lightweight success criteria with the user: which metric matters most (e.g.,
, specific commit duration, render count for a component) and what threshold would be meaningful. This anchors later evaluation. On success:
- if user asked you to perform the profiling, determine how to profile yourself using tools described in skill.
- if the user stated they wish to perform the interaction themselves — suggest what interaction to perform (e.g. "scroll the list", "switch tabs") and wait for their reply.
If you received information about existing profiling session being owned by another agent:
- if session is marked as "stale", you may overtake it without prompting the user for allowance
- if session is NOT "stale" - before taking action and terminating the other session, stop and ask user what you should do, explaining the situation.
Annotate every interaction
After each
or
call, record an annotation using the returned
. Compute
offsetMs = timestampMs - startedAtEpochMs
. Do this for
every interaction — including back-navigation swipes, not just the primary action. Pass all collected annotations to
in Step 3.
Step 2: Stop and collect
Call
and in parallel. Only skip
if you did not start it in Step 1. Note
and
.
If
fiber_renders_captured: 0
, warn the user — React commit data may be missing.
Step 3: Analyze
Call
with
,
,
,
, and
. The report includes metadata such as
,
, and
— check these in the returned markdown report.
If you performed interactions using
/
, pass
to mark when each action occurred. Each annotation's
must be computed as
tapTimestampMs - startedAtEpochMs
, where
is the
returned by the gesture-tap/gesture-swipe tool and
was returned by
. Do
not use
for this calculation — only server-side timestamps from the tool return values.
If dual profiling, also call
, then
you must call
for the cross-correlated view — do not skip this step when both profilers ran; the combined report surfaces correlations that individual reports miss.
The analyze report includes CPU hotspots per commit — showing exactly which JS functions ran during each slow React commit. Raw data is saved to disk automatically for later reload.
Step 4: Assess results
Analyze whether the results give you a proper image of what is wrong with the application - do not assume improvement always exists, verify results logically with reference to how react-native works. Make sure to give honest feedback and be ready to change the approach if needed.
Step 5: Present findings and ask about next steps
Present a concise summary of the key findings - present whether possibilities for improvement exist and how performing further actions could affect performance. Then follow the "After analysis" guideline — ask whether to investigate further, implement fixes (if available), or stop.
Step 6: Drill-down investigation (iterative)
Based on findings from the report, use query tools to investigate deeper:
- Slow component? -> mode= component_name= — shows what JS functions ran during that component's commits.
- Want to see the call tree? -> mode= function_name= — shows callers and callees.
- What happened during a time window? -> mode= — lists all commits in a range.
- Full commit detail? -> mode= commit_index=38 — all components, props changes, parent cascade.
- Who triggered whom? -> mode= — visual parent-child cascade.
- iOS hang details? -> mode= — native call stacks during a hang.
Repeat as needed until you identify the root cause function and file, referring to step 4 for honest evaluation. After each round of investigation, ask the user if they want to continue digging or move to fixing.
Step 7: Reload a previous session
If you profiled multiple scenarios and need to revisit earlier data:
- Call mode= to see all saved sessions with timestamps (the list now also shows Runtime / Device / Metro bundle columns to help identify the right session).
- Call mode= session_id= device_id= to reload React data. scopes the reload into the cache slot.
- Call mode= session_id= device_id= to reload native profiler data.
- Query tools now operate on the reloaded session data — pass the same you loaded with, otherwise they will miss the cache.
This is useful for before/after comparisons: profile, fix, re-profile, then reload the original session to compare metrics side by side.
Step 8: Apply fix and re-profile
If fix is present, read the source code of the identified bottleneck using
react-profiler-component-source
or the Read tool. Apply the fix, then re-profile (Step 1 -> user interaction -> Step 2 -> Step 3 -> Step 4). Report whether the target metric improved, stayed flat, or regressed. Also check whether the fix introduced regressions in other metrics (e.g., render count dropped but CPU time increased, or a different component now re-renders more). If the fix showed no net benefit or unacceptable tradeoffs, revert and reconsider.
Tip: If the interaction sequence was recorded as a flow (see "Use flows for reproducible profiling" above), replay it with
instead of manually repeating the steps. This guarantees identical interaction conditions for the comparison. If the flow fails during replay (e.g., a UI fix changed the layout), follow
skill §10 (Flow Self-Improvement) to diagnose and repair the flow before retrying the profiling cycle.
If the user stated that they do not wish for changes, present the profiling report and skip the fix but suggest it to the user.
React Compiler rule: If the analyze report indicates React Compiler is enabled, do NOT propose
/
/
unless you confirmed compiler bail-out (check
react-profiler-fiber-tree
for absent
on that component).
5. Important Caveats
- Dev mode inflation: renders are ~3x slower than production. Prioritize high — it scales to prod.
- Re-run after fixes: Always re-profile after changes. Report honestly whether the metric improved, regressed, or stayed flat — do not assume improvement.
- is informational: Components in and re-render by design.
- Strict Mode: Double-invokes renders. The pipeline halves automatically when detected.
- Debugger connection: If interrupted, started profiling also closes. Before attempting recovery, call — it tells you whether the session is , , , or , so you can decide whether to stop, restart, or reconnect first.
- Confounders to watch for:
- Live API data may differ between runs (different payload sizes, content counts), which shifts render counts and durations independently of your fix. Note when data-dependent components show variance.
- Profiler overhead inflates CPU measurements. If iOS Instruments shows , , or Hermes internals dominating the JS thread, that reflects profiler instrumentation cost — not app work. Discount those entries.
- Runs are not perfectly reproducible. Small variations (under ~10-15%) in commit duration may be noise; only treat consistent, directional changes as signal.
For standalone diagnostic tools (live render stats, fiber tree, CPU summary), see
references/diagnostic-tools.md
.