agents/skills/aptos/oracle-analysis/SKILL.md
Trigger Pattern ORACLE flag (required) - Inject Into Breadth agents, depth-external, depth-edge-case
npx skillsauth add plamentsv/plamen oracle-analysisInstall this skill globally with one command. Works with Claude Code, Cursor, and Windsurf.
3 of 9 scanners reported clean
Some scanners were skipped, did not run, or reported a non-clean status. Review each row below.
Trigger Pattern: ORACLE flag (required) Inject Into: Breadth agents, depth-external, depth-edge-case Purpose: Analyze all oracle integrations in Aptos Move protocols for staleness, decimal errors, zero/negative prices, confidence intervals, multi-oracle aggregation, and failure modes
For every oracle the protocol consumes:
STEP PRIORITY: Steps 6 (Failure Modes) and 5c (Deviation Reference) are where HIGH/CRITICAL severity findings most commonly hide. Do NOT rush these steps. If constrained, skip conditional sections (4a-4d, 5a) before skipping 5c or 6.
Enumerate ALL oracle data sources the protocol reads:
| Oracle | Type | Module Path | Functions Called | Consumers (protocol functions) | Update Frequency | Freshness Guarantee | |--------|------|-------------|-----------------|-------------------------------|-----------------|---------------------| | {name} | Pyth / Switchboard / Custom / On-chain TWAP | {module::path} | {get_price / get_result / etc.} | {list all} | {expected} | {documented or UNKNOWN} |
Aptos oracle landscape:
pyth::price_feed module, returns Price { price: I64, conf: u64, expo: I64, publish_time: u64 }switchboard::aggregator module, returns aggregator results with mantissa and scaleTable or SmartTable for price storageFor each oracle: What decision does the protocol make based on this data? (pricing, liquidation threshold, reward rate, rebase trigger, collateral valuation, etc.)
Hardcoded stablecoin pricing: Does the protocol skip oracle lookup for any asset and hardcode its price (e.g., USDC = 1e8)? All assets require dynamic oracle pricing — stablecoins depeg.
For each oracle identified in Step 1:
| Oracle | Timestamp Checked? | Max Staleness Enforced? | Staleness Threshold | Appropriate? | |--------|-------------------|------------------------|--------------------:|-------------| | {name} | YES/NO | YES/NO | {seconds or NONE} | {analysis} |
Pyth-specific: Is price.publish_time compared against timestamp::now_seconds()? What max age is enforced?
Switchboard-specific: Is the aggregator's latest_confirmed_round.round_open_timestamp validated?
Chained feed deviation: If derived prices require multiple feeds (e.g., token_A/USD via token_A/APT + APT/USD), sum individual deviation thresholds to compute total worst-case deviation. If total exceeds LTV buffer → FINDING.
If NO staleness check: What happens when the oracle returns stale data?
For each consumer function, trace the impact of receiving data that is {freshness_guarantee x 2} old:
| Consumer Function | Data Used | If Stale By {X}: Impact | Severity | |-------------------|-----------|------------------------|----------| | {function} | {price/rate} | {specific impact} | {H/M/L} |
| Check | Code Reference | Status |
|-------|---------------|--------|
| get_price() or get_price_no_older_than() used? | {location} | {which} |
| price.publish_time freshness validated? | {location} | YES/NO |
| price.price (I64) sign checked (> 0)? | {location} | YES/NO |
| price.conf confidence interval checked? | {location} | YES/NO |
| price.expo (negative exponent) handled correctly? | {location} | YES/NO |
| Price feed ID hardcoded or configurable? | {location} | {which} |
| Check | Code Reference | Status | |-------|---------------|--------| | Aggregator authority validated? | {location} | YES/NO | | Result staleness checked? | {location} | YES/NO | | Min/max response thresholds enforced? | {location} | YES/NO | | Aggregator config (min oracle results, variance threshold) appropriate? | {location} | YES/NO |
For each oracle data flow:
| Oracle | Oracle Decimals/Exponent | Consumer Expects | Normalization Applied? | Correct? | |--------|------------------------|-----------------|----------------------|----------| | {name} | {expo or scale} | {expected by math} | YES/NO | {analysis} |
Pyth decimal handling: Pyth uses expo field (typically negative, e.g., expo = -8 means 8 decimal places). The actual price = price.price * 10^expo. Common errors:
expo as positive when it is negativeSwitchboard decimal handling: Uses mantissa and scale (or decimals). Actual value = mantissa * 10^(-scale).
MANDATORY GREP: Search all oracle consumer files for hardcoded decimal constants: 100000000, 1e8, 10_000_000, DECIMAL, PRECISION. For each hit: (1) Is this a decimal normalization constant? (2) Does it match the ACTUAL oracle's decimal format? (3) If the oracle feed changes or is swapped, does this constant break?
Decimal chain trace: For each arithmetic operation using oracle data, trace the full decimal chain: oracle_output_decimals -> normalization_step -> consumer_expected_decimals. If any step uses a hardcoded constant rather than reading decimals dynamically -> FINDING.
Common decimal mismatches on Aptos:
expo = -8 (8 decimals), but protocol assumes 18price * amount where price and amount have different decimal basesGrep ALL oracle consumer files for 10_u128|pow\(10|DECIMALS|PRECISION|100000000|normalize. For each match, fill:
| File:Line | Pattern | Hardcoded Value | Oracle's Actual Decimals | Match? | |-----------|---------|-----------------|-------------------------|--------|
If ANY row shows Match=NO or oracle decimals UNKNOWN with hardcoded constant -> FINDING (R16). Skipping this step is a Step Execution violation (x3d).
<!-- LOAD_IF: TWAP -->If protocol uses any TWAP oracle (DEX-derived, custom accumulator, etc.):
| TWAP Oracle | Window Length | Pool Liquidity | Manipulation Cost (est.) | Sufficient? | |-------------|-------------|----------------|-------------------------|-------------| | {oracle} | {seconds} | {USD value} | {estimated} | YES/NO |
Rule of thumb: TWAP window < 30 min AND pool TVL < $10M -> potentially manipulable.
| Check | Status | Impact if Wrong | |-------|--------|-----------------| | Overflow protection on cumulative price difference? | YES/NO | {impact} | | Geometric vs arithmetic mean -- correct for use case? | {which used} | {impact if wrong} | | Time-weighted vs block-weighted -- which is used? | {which} | {manipulation vector} | | Empty observation slots handled? | YES/NO | {impact} | | Aptos epoch boundaries handled? (epoch changes can affect timestamps) | YES/NO | {impact} |
During rapid price movements, TWAP lags spot price. Trace:
Check oracle behavior when history is insufficient: (1) zero snapshots, (2) single snapshot, (3) window period not yet elapsed.
| Cold-Start State | Oracle Return Value | Protocol Behavior | Exploitable? | |------------------|--------------------:|-------------------|-------------|
For each exploitable state: can attacker act during cold-start window at manipulated price? Tag: [BOUNDARY:snapshots=0], [BOUNDARY:snapshots=1]. If TWAP returns 0 or aborts during cold-start with no fallback -> FINDING (R16, minimum Medium).
<!-- END_LOAD_IF: TWAP -->For multi-oracle systems or oracle-based thresholds:
<!-- LOAD_IF: MULTI_ORACLE -->| Oracle System | Aggregation Method | Oracle Count | Agreement Required | What if Disagreement? | |---------------|-------------------|-------------|-------------------|----------------------| | {system} | Median / Mean / Weighted / First-valid | {N} | {M of N} | {fallback behavior} |
Check: What happens at exact threshold boundaries?
| Threshold | Oracle Data Used | Threshold Value | At Exact Boundary | Off-by-One? | |-----------|-----------------|----------------|-------------------|-------------| | {name} | {oracle field} | {value} | {behavior at exact value} | YES/NO |
Check > vs >=: At the exact threshold value, does the protocol behave as intended?
For each deviation check in the protocol (maxDeviation, priceDeviation, deviationThreshold, etc.):
| Parameter | Measured Against | Reference Source | Reference Manipulable? | Reference Staleable? | |-----------|-----------------|-----------------|----------------------|---------------------|
Checks:
[TRACE:deviation check: current vs {reference} -> reference source: {X} -> manipulable: {Y/N}]For each oracle, model failure scenarios:
| Failure Mode | Oracle Behavior | Protocol Response | Impact | Mitigation Present? | |-------------|-----------------|-------------------|--------|-------------------| | Zero return | Returns price = 0 | {what happens} | {impact} | YES/NO | | Abort | Call aborts (Move has no try/catch) | {what happens} | {impact} | YES/NO -- can_* check first? | | Stale (freshness exceeded) | Returns old data | {what happens} | {impact} | YES/NO -- staleness check? | | Extreme value | Returns outlier | {what happens} | {impact} | YES/NO -- bounds check? | | Negative price (Pyth I64) | Returns < 0 | {what happens} | {impact} | YES/NO -- sign check? | | Feed not initialized | Resource does not exist | {what happens} | {impact} | YES/NO -- exists<T> check? |
Aptos-specific failure note: Move does not have try/catch. Oracle call failures result in transaction abort. This means:
exists<> or similarFor each unmitigated failure mode: What is the worst-case impact? Can it lead to fund loss?
Circuit breaker check: Does the protocol have a mechanism to pause oracle-dependent operations if the oracle enters a failure state?
{CONTRACTS} -- Move modules to analyze
{ORACLE_MODULES} -- Oracle module paths (pyth::price_feed, switchboard::aggregator, custom)
{CONSUMER_FUNCTIONS} -- Functions that read oracle data
{PRICE_FEED_IDS} -- Pyth price feed identifiers or Switchboard aggregator addresses
{TOKEN_DECIMALS} -- Decimal configuration of tokens in scope
**ID**: [OR-N]
**Severity**: [based on fund impact and likelihood of oracle failure/manipulation]
**Step Execution**: checkmark1,2,3,4,5,6 | x(reasons) | ?(uncertain)
**Rules Applied**: [R1:Y, R4:Y, R10:Y, R16:Y]
**Location**: module::function:LineN
**Title**: Oracle [issue type] in [function] enables [attack/failure]
**Description**: [Specific oracle issue with data flow trace]
**Impact**: [Quantified impact under worst-case oracle scenario]
| Field | Required | Description | |-------|----------|-------------| | oracle_inventory | yes | All oracle data sources and consumers | | staleness_vectors | yes | Unmitigated staleness paths | | decimal_mismatches | yes | Decimal normalization issues | | failure_modes | yes | Oracle failure scenarios and protocol response | | finding | yes | CONFIRMED / REFUTED / CONTESTED | | evidence | yes | Code locations with line numbers | | step_execution | yes | Status for each step |
| Section | Required | Completed? | Notes | |---------|----------|------------|-------| | 1. Oracle Inventory | YES | Y/x/? | | | 2. Staleness Analysis | YES | Y/x/? | For each oracle | | 2c. Pyth-Specific Checks | IF Pyth used | Y/x(N/A)/? | | | 2d. Switchboard-Specific Checks | IF Switchboard used | Y/x(N/A)/? | | | 3. Decimal Normalization Audit | YES | Y/x/? | | | 3d. Decimal Grep Sweep | YES | Y/x/? | MANDATORY mechanical step | | 4. TWAP-Specific Analysis | IF TWAP used | Y/x(N/A)/? | | | 4d. TWAP Cold-Start Analysis | IF TWAP used | Y/x(N/A)/? | Zero/single snapshot states | | 5. Oracle Weight / Threshold Boundaries | IF multi-oracle or thresholds | Y/x(N/A)/? | | | 5c. Deviation Reference Point Audit | IF deviation checks exist | Y/x(N/A)/? | Reference manipulability | | 6. Oracle Failure Modes | YES | Y/x/? | For each oracle |
development
Prepare Solidity projects for a security audit — test coverage, test quality, NatSpec docs, code hygiene, dependency health, best-practice enforcement, deployment readiness, and project documentation checks. Generates a scored Audit Readiness Report and optionally runs static analysis. Trigger on: "prepare for audit", "audit readiness", "pre-audit check", "audit prep", "NatSpec check", or any request to review a Solidity codebase before a security review.
development
Launch the Plamen deterministic Web3 security audit pipeline
development
Run the Plamen smart-contract audit wizard in Codex
testing
Launch the Plamen deterministic L1 infrastructure audit pipeline