rust-best-practices
Originally fromapollographql/skills
Installation
SKILL.md
Rust Best Practices
Unified Rust guidelines covering coding style, ownership, error handling, async patterns, traits, testing, performance, linting, and documentation. Apply when writing or reviewing Rust code.
When to Apply
- Writing new Rust code or designing APIs
- Reviewing or refactoring existing Rust code
- Implementing async systems with Tokio
- Designing error hierarchies with thiserror/anyhow
- Choosing between borrowing, cloning, or ownership transfer
- Setting up tests, benchmarks, or snapshot testing
- Configuring clippy lints and workspace settings
- Optimizing Rust code for performance
Reference Guide
Load detailed guidance based on context. Read the relevant file when the topic arises:
| Topic | Reference | Load When |
|---|---|---|
| Coding Style | references/coding-style.md |
Naming, imports, iterators, comments, string handling, macros |
| Error Handling | references/error-handling.md |
Result, Option, ?, thiserror, anyhow, custom errors, async errors |
| Ownership & Pointers | references/ownership-and-pointers.md |
Lifetimes, borrowing, smart pointers, Pin, Cow, interior mutability |
| Traits & Generics | references/traits-and-generics.md |
Trait design, dispatch, GATs, sealed traits, type state pattern |
| Async & Concurrency | references/async-and-concurrency.md |
Tokio, channels, streams, shutdown, runtime config, async traits |
| Sync Concurrency | references/concurrency-sync.md |
Atomics, Mutex, RwLock, lock ordering, Send/Sync, memory ordering |
| Testing | references/testing.md |
Unit/integration/doc tests, snapshot, proptest, mockall, benchmarks, fuzz |
| Performance | references/performance.md |
Profiling, flamegraph, cloning, stack vs heap, iterators, allocation |
| Clippy & Linting | references/clippy-and-linting.md |
Clippy config, key lints, workspace setup, #[expect] vs #[allow] |
| Documentation | references/documentation.md |
Doc comments, rustdoc, doc lints, coverage checklist |
Quick Reference: Coding Style
- Prefer
&Tover.clone()unless ownership transfer is required - Use
&stroverString,&[T]overVec<T>in function parameters - No
get_prefix on getters:fn name()notfn get_name() - Conversion naming:
as_(cheap borrow),to_(expensive/cloning),into_(ownership transfer) - Iterator methods:
iter()/iter_mut()/into_iter() - Import ordering:
std-> external crates -> workspace crates ->super::->crate:: - Comments explain why (safety, workarounds), not what
- Use
format!over string concatenation with+ - Prefer
s.bytes()overs.chars()for ASCII-only operations - Avoid macros unless necessary; prefer functions or generics
Quick Reference: Error Handling
- Return
Result<T, E>for fallible operations; reservepanic!for unrecoverable bugs - No
unwrap()in production. Useexpect()with descriptive message only when the value is logically guaranteed. Prefer?,if let,let...elsefor all other cases - Use
thiserrorfor library/crate errors,anyhowfor binaries only - Prefer
?operator overmatchchains for error propagation - Use
_elsevariants (ok_or_else,unwrap_or_else) to prevent eager allocation - Use
inspect_errandmap_errfor logging and transforming errors assert!at function entry for invariant checking (debug builds)
Quick Reference: Ownership & Pointers
- Small
Copytypes (<=24 bytes, all fieldsCopy, no heap) pass by value - Use
Cow<'_, T>when data may or may not need ownership - Meaningful lifetime names:
'src,'ctx,'conn— not just'a - Use
try_borrow()onRefCellto avoid panics; prefer over direct.borrow_mut() - Shadowing for transformations:
let x = x.parse()?
| Pointer | When to Use |
|---|---|
Box<T> |
Single ownership, heap allocation, recursive types |
Rc<T> |
Shared ownership, single-threaded |
Arc<T> |
Shared ownership, multi-threaded |
Cell<T> / RefCell<T> |
Interior mutability, single-threaded |
Mutex<T> / RwLock<T> |
Interior mutability, multi-threaded |
Quick Reference: Traits & Generics
- Prefer generics (static dispatch) by default for zero-cost abstractions
- Use
dyn Traitonly when heterogeneous collections or plugin architectures are needed - Box at API boundaries, not internally
- Object safety: no generic methods, no
Self: Sized, methods use&self/&mut self/self - Use sealed traits to prevent external implementors
- Type state pattern encodes valid states in the type system:
struct Connection<S> { _state: PhantomData<S> }
struct Disconnected;
struct Connected;
impl Connection<Connected> { fn send(&self, data: &[u8]) { /* ... */ } }
Quick Reference: Async & Concurrency
- Async for I/O-bound work, sync for CPU-bound work
- Never hold locks across
.awaitpoints — use scoped guards - Never use
std::thread::sleepin async — usetokio::time::sleep - Never spawn unboundedly — use semaphores for limits
- Ensure
Sendbounds on spawned futures - Use
JoinSetfor managing multiple concurrent tasks - Use
CancellationToken(fromtokio_util) for graceful shutdown - Instrument with
tracing+#[instrument]for async debugging
| Channel | Use Case |
|---|---|
mpsc |
Multi-producer, single-consumer message passing |
broadcast |
Multi-producer, multi-consumer event fan-out |
oneshot |
Single value, single use (request-response) |
watch |
Latest-value-only, change notification |
- Sync channels:
crossbeam::channeloverstd::sync::mpsc - Async channels:
tokio::sync::{mpsc, broadcast, oneshot, watch} - Atomics (
AtomicBool,AtomicUsize) overMutexfor primitive types - Choose memory ordering carefully:
Relaxed/Acquire/Release/SeqCst
Quick Reference: Testing
- Name tests descriptively:
process_should_return_error_when_input_empty() - One assertion per test when possible; include formatted failure messages
- Group tests in
modblocks by unit of work - Use doc tests (
///) for public API examples; run separately withcargo test --doc - Snapshot testing:
cargo insta testthencargo insta review; redact unstable fields rstestfor parameterized tests with#[case::name]labelsproptestfor property-based testing with custom strategiesmockallwith#[automock]for mocking traitscriterionfor benchmarks withiter_batchedandBenchmarkIdcargo-fuzzwithlibfuzzer_sysfor fuzz testingcargo-tarpaulinorcargo-llvm-covfor code coveragesqlx::testfor database integration tests with automatic pool injection- Use
#[should_panic]and#[ignore]attributes where appropriate
Quick Reference: Performance
- Golden rule: don't guess, measure. Always benchmark with
--release - Run
cargo clippy -- -D clippy::perffor performance-related hints - Use
cargo flamegraphorsamply(macOS) for profiling - Avoid cloning in loops; clone at the last moment only
- Pre-allocate:
Vec::with_capacity(),String::with_capacity() - Prefer iterators over manual
forloops; avoid intermediate.collect() - Stack for small types, heap for large/recursive; use
smallvecfor large const arrays - Use
Cow<'_, T>to avoid unnecessary allocation - Prefer
s.bytes()overs.chars()for ASCII-only string operations
Quick Reference: Clippy & Linting
Run regularly:
cargo clippy --all-targets --all-features --locked -- -D warnings
| Lint | Catches |
|---|---|
redundant_clone |
Unnecessary .clone() calls |
needless_borrow |
Unnecessary & borrows |
large_enum_variant |
Oversized variants (consider Box) |
needless_collect |
Premature .collect() before iteration |
map_unwrap_or |
.map().unwrap_or() chains |
unnecessary_wraps |
Functions always returning Ok/Some |
clone_on_copy |
.clone() on Copy types |
- Use
#[expect(clippy::lint)]over#[allow(...)]—expectwarns when lint no longer applies - Add justification comment on every suppression
- Set
#![warn(clippy::all)]as workspace minimum - Configure workspace lints in
Cargo.tomlwith priority levels
Quick Reference: Documentation
//comments explain why: safety invariants, workarounds, design rationale///doc comments explain what and how for all public items//!for module-level and crate-level documentation at top oflib.rs/mod.rs- Every
TODOneeds a linked issue:// TODO(#42): description - Enable
#![deny(missing_docs)]for libraries - Include
# Examples,# Errors,# Panics,# Safetysections in doc comments
| Doc Lint | Purpose |
|---|---|
missing_docs |
Ensure all public items documented |
broken_intra_doc_links |
Catch dead cross-references |
missing_panics_doc |
Document panic conditions |
missing_errors_doc |
Document error conditions |
missing_safety_doc |
Document unsafe safety requirements |
Quick Reference: Data Types & Patterns
- Use newtypes for domain semantics:
struct Email(String) - Prefer slice patterns:
if let [first, .., last] = slice - Use arrays for fixed sizes; avoid
Vecwhen length is known at compile time - Shadowing for transformation:
let x = x.parse()? Cow<str>when data might need modification of borrowed datacontains()on strings is O(n*m) — avoid nested string iteration
Deprecated to Modern Migration
| Deprecated | Better | Since |
|---|---|---|
lazy_static! |
std::sync::OnceLock |
Rust 1.70 |
once_cell::Lazy |
std::sync::LazyLock |
Rust 1.80 |
std::sync::mpsc |
crossbeam::channel (sync) |
— |
std::sync::Mutex |
parking_lot::Mutex (recommended) |
— |
failure / error-chain |
thiserror / anyhow |
— |
try!() |
? operator |
Rust 2018 |
async-trait crate |
Native async fn in traits (1.75+, limited) |
Rust 1.75 |
Cargo.toml Essentials
Recommended dependencies:
[dependencies]
thiserror = "2"
anyhow = "1"
tokio = { version = "1", features = ["full"] }
serde = { version = "1", features = ["derive"] }
tracing = "0.1"
tracing-subscriber = "0.3"
[dev-dependencies]
rstest = "0.25"
proptest = "1"
mockall = "0.13"
criterion = { version = "0.5", features = ["html_reports"] }
insta = { version = "1", features = ["yaml"] }
Workspace lints (Cargo.toml):
[workspace.lints.clippy]
all = { level = "warn", priority = -1 }
pedantic = { level = "warn", priority = -1 }
rustfmt.toml:
reorder_imports = true
imports_granularity = "Crate"
group_imports = "StdExternalCrate"
Constraints
MUST DO
- Use ownership and borrowing for memory safety
- Handle all errors explicitly via
Result/Option— no silent failures - Use
thiserrorfor library errors,anyhowfor binaries - Minimize
unsafecode; document allunsafeblocks with safety invariants - Use the type system for compile-time guarantees
- Run
cargo clippyand fix all warnings - Use
cargo fmtfor consistent formatting - Write tests including doc tests for public APIs
- Add
///documentation with examples for all public items - Use
tracingfor observability in async code - When reviewing or writing code, suggest a testing approach using the recommended tools (
rstest,proptest,insta,mockall,criterion) — even if the user did not ask for tests
MUST NOT DO
- Use
unwrap()in production code - Create memory leaks or dangling pointers
- Use
unsafewithout documented safety invariants - Ignore clippy warnings without
#[expect(...)]and justification - Hold locks across
.awaitpoints - Use
std::thread::sleepin async context - Skip error handling or use
panic!for recoverable errors - Use
Stringwhere&strsuffices; clone unnecessarily - Spawn tasks unboundedly without semaphore limits
- Mix blocking and async code without
spawn_blocking
Weekly Installs
33
Repository
pedronauck/skillsGitHub Stars
301
First Seen
Mar 18, 2026
Security Audits
Installed on
opencode33
gemini-cli33
github-copilot33
codex33
amp33
cline33