Skills rust-best-practices
install
source · Clone the upstream repo
git clone https://github.com/apollographql/skills
Claude Code · Install into ~/.claude/skills/
T=$(mktemp -d) && git clone --depth=1 https://github.com/apollographql/skills "$T" && mkdir -p ~/.claude/skills && cp -r "$T/skills/rust-best-practices" ~/.claude/skills/apollographql-skills-rust-best-practices && rm -rf "$T"
manifest:
skills/rust-best-practices/SKILL.mdsource content
Rust Best Practices
Apply these guidelines when writing or reviewing Rust code. Based on Apollo GraphQL's Rust Best Practices Handbook.
Best Practices Reference
Before reviewing, familiarize yourself with Apollo's Rust best practices. Read ALL relevant chapters in the same turn in parallel. Reference these files when providing feedback:
- Chapter 1 - Coding Styles and Idioms: Borrowing vs cloning, Copy trait, Option/Result handling, iterators, comments
- Chapter 2 - Clippy and Linting: Clippy configuration, important lints, workspace lint setup
- Chapter 3 - Performance Mindset: Profiling, avoiding redundant clones, stack vs heap, zero-cost abstractions
- Chapter 4 - Error Handling: Result vs panic, thiserror vs anyhow, error hierarchies
- Chapter 5 - Automated Testing: Test naming, one assertion per test, snapshot testing
- Chapter 6 - Generics and Dispatch: Static vs dynamic dispatch, trait objects
- Chapter 7 - Type State Pattern: Compile-time state safety, when to use it
- Chapter 8 - Comments vs Documentation: When to comment, doc comments, rustdoc
- Chapter 9 - Understanding Pointers: Thread safety, Send/Sync, pointer types
Quick Reference
Borrowing & Ownership
- Prefer
over&T
unless ownership transfer is required.clone() - Use
over&str
,String
over&[T]
in function parametersVec<T> - Small
types (≤24 bytes) can be passed by valueCopy - Use
when ownership is ambiguousCow<'_, T>
Error Handling
- Return
for fallible operations; avoidResult<T, E>
in productionpanic! - Never use
/unwrap()
outside testsexpect() - Use
for library errors,thiserror
for binaries onlyanyhow - Prefer
operator over match chains for error propagation?
Performance
- Always benchmark with
flag--release - Run
for performance hintscargo clippy -- -D clippy::perf - Avoid cloning in loops; use
instead of.iter()
for Copy types.into_iter() - Prefer iterators over manual loops; avoid intermediate
calls.collect()
Linting
Run regularly:
cargo clippy --all-targets --all-features --locked -- -D warnings
Key lints to watch:
- unnecessary cloningredundant_clone
- oversized variants (consider boxing)large_enum_variant
- premature collectionneedless_collect
Use
#[expect(clippy::lint)] over #[allow(...)] with justification comment.
Testing
- Name tests descriptively:
process_should_return_error_when_input_empty() - One assertion per test when possible
- Use doc tests (
) for public API examples/// - Consider
for snapshot testing generated outputcargo insta
Generics & Dispatch
- Prefer generics (static dispatch) for performance-critical code
- Use
only when heterogeneous collections are neededdyn Trait - Box at API boundaries, not internally
Type State Pattern
Encode valid states in the type system to catch invalid operations at compile time:
struct Connection<State> { /* ... */ _state: PhantomData<State> } struct Disconnected; struct Connected; impl Connection<Connected> { fn send(&self, data: &[u8]) { /* only connected can send */ } }
Documentation
comments explain why (safety, workarounds, design rationale)//
doc comments explain what and how for public APIs///- Every
needs a linked issue:TODO// TODO(#42): ... - Enable
for libraries#![deny(missing_docs)]