Skills axum-code-review
Reviews axum web framework code for routing patterns, extractor usage, middleware, state management, and error handling. Use when reviewing Rust code that uses axum, tower, or hyper for HTTP services. Covers axum 0.7+ patterns including State, Path, Query, Json extractors.
install
source · Clone the upstream repo
git clone https://github.com/openclaw/skills
Claude Code · Install into ~/.claude/skills/
T=$(mktemp -d) && git clone --depth=1 https://github.com/openclaw/skills "$T" && mkdir -p ~/.claude/skills && cp -r "$T/skills/anderskev/axum-code-review" ~/.claude/skills/clawdbot-skills-axum-code-review && rm -rf "$T"
manifest:
skills/anderskev/axum-code-review/SKILL.mdsource content
Axum Code Review
Review Workflow
- Check Cargo.toml — Note axum version (0.6 vs 0.7+ have different patterns), Rust edition (2021 vs 2024), tower, tower-http features. Edition 2024 changes RPIT lifetime capture in handler return types and removes the need for
in custom extractors.async-trait - Check routing — Route organization, method routing, nested routers
- Check extractors — Order matters (body extractors must be last), correct types
- Check state — Shared state via
, not global mutable stateState<T> - Check error handling —
implementations, error typesIntoResponse
Output Format
Report findings as:
[FILE:LINE] ISSUE_TITLE Severity: Critical | Major | Minor | Informational Description of the issue and why it matters.
Quick Reference
| Issue Type | Reference |
|---|---|
| Route definitions, nesting, method routing | references/routing.md |
| State, Path, Query, Json, body extractors | references/extractors.md |
| Tower middleware, layers, error handling | references/middleware.md |
Review Checklist
Routing
- Routes organized by domain (nested routers for
,/api/users
)/api/orders - Fallback handlers defined for 404s
- Method routing explicit (
,.get()
, not.post()
with manual method matching).route() - No route conflicts (overlapping paths with different extractors)
Extractors
- Body-consuming extractors (
,Json
,Form
) are the LAST parameterBytes -
requiresState<T>
— typicallyT: Clone
or directArc<AppState>
deriveClone -
parameter types match the route definitionPath<T> -
fields areQuery<T>
for optional query params withOption#[serde(default)] - Custom extractors implement
(not body) orFromRequestParts
(body)FromRequest - Edition 2024: Custom extractors use native
in trait impls (noasync fn
needed for#[async_trait]
/FromRequest
)FromRequestParts
State Management
- Application state shared via
, not global mutable staticsState<T> - Database pool in state (not created per-request)
- State contains only shared resources (pool, config, channels), not request-specific data
-
derived or manually implemented on state typeClone - Edition 2024: Shared static state uses
from std (notLazyLock
oronce_cell::sync::Lazy
)lazy_static!
Error Handling
- Handler errors implement
for proper HTTP error codesIntoResponse - Internal errors don't leak to clients (no raw error messages in 500 responses)
- Error responses use consistent format (JSON error body with code/message)
-
pattern used for handlersResult<impl IntoResponse, AppError> - Edition 2024: Handler return types
capture all in-scope lifetimes by default; use-> impl IntoResponse
to opt out of capturing request lifetimes when returning owned data+ use<>
Middleware
- Tower layers applied in correct order (outer runs first on request, last on response)
-
used for common concerns (CORS, compression, tracing, timeout)tower-http - Request-scoped data passed via extensions, not global state
- Middleware errors don't panic — they return error responses
- Edition 2024: Middleware using
can migrate to native#[async_trait]
in trait implsasync fn
Severity Calibration
Critical
- Body extractor not last in handler parameters (silently consumes body, later extractors fail)
- SQL injection via path/query parameters passed directly to queries
- Internal error details leaked to clients (stack traces, database errors)
- Missing authentication middleware on protected routes
Major
- Global mutable state instead of
(race conditions)State<T> - Missing error type conversion (raw
returned to client)sqlx::Error - Missing request timeout (handlers can hang indefinitely)
- Route conflicts causing unexpected 405s
- Edition 2024:
still used forasync-trait
/FromRequest
when native async fn worksFromRequestParts
Minor
- Manual route method matching instead of
,.get().post() - Missing fallback handler (default 404 is plain text, not JSON)
- Middleware applied per-route when it should be global (or vice versa)
- Missing
for request loggingtower-http::trace - Edition 2024:
oronce_cell::sync::Lazy
used wherelazy_static!
worksstd::sync::LazyLock
Informational
- Suggestions to use
layers for common concernstower-http - Router organization improvements
- Suggestions to add OpenAPI documentation via
orutoipaaide
Valid Patterns (Do NOT Flag)
on handlers — Debugging aid that improves compile error messages#[axum::debug_handler]
for middleware-injected data — Valid pattern for request-scoped valuesExtension<T>- Returning
from handlers — More flexible than concrete typesimpl IntoResponse
per module, merged in main — Standard organization patternRouter::new()
for layer composition — Tower pattern, not over-engineeringServiceBuilder
withaxum::serve
— Standard axum 0.7+ server setupTcpListener- Native
inasync fn
/FromRequest
impls —FromRequestParts
crate no longer needed (stable since Rust 1.75)async-trait
on handler return types — Edition 2024 precise capture syntax for RPIT+ use<'a>
for shared static state — Replacesstd::sync::LazyLock
/once_cell
(stable since Rust 1.80)lazy_static
Before Submitting Findings
Load and follow
beagle-rust:review-verification-protocol before reporting any issue.