This document outlines the coding guidelines for contributing to zkemail.rs. Following these guidelines will help maintain a consistent and high-quality codebase.
- Tool: Use
rustfmt
to automatically format your code. Ensure that all code is formatted before committing. Runcargo fmt
to format your code according to the project's style guidelines. - Indentation: Use 4 spaces per indentation level. Do not use tabs.
- Line Length: Aim to keep lines under 100 characters, but it's not a strict rule. Use your judgment to ensure readability.
- Imports: Group imports into four sections:
extern crate
,use
,use crate
, anduse super
.- Example:
extern crate serde; use std::collections::HashMap; use std::io::{self, Read}; use crate::utils::config; use super::super::common::logger;
- Example:
- Braces: Use the Allman style for braces, where the opening brace is on the same line as the function signature.
- Example:
fn main() { // function body }
- Example:
- Comments: Use
//
for single-line comments and/* ... */
for multi-line comments. - Whitespace: Use a single space after commas and colons, and no space before commas and colons.
- Example:
let numbers = vec![1, 2, 3]; let user = User { name: "Alice", age: 30 };
- Example:
- Function Length: Aim to keep functions short and focused. If a function is too long, consider breaking it up into smaller functions.
- Code Duplication: Avoid duplicating code. If you find yourself copying and pasting code, consider refactoring it into a shared function or module.
- No warnings: Ensure that your code compiles without warnings. Fix any warnings before committing.
- Tool: Use
cargo clippy
to lint your code and catch common mistakes and improve your Rust code. Runcargo clippy
before committing your code to ensure it adheres to Rust's best practices and the project's specific requirements. - Handling Lints: Address all warnings and errors reported by
clippy
. If you must ignore a lint, use#[allow(clippy::lint_name)]
and provide a comment explaining why.
- Variables and Functions: Use
snake_case
.- Example:
let user_name = "Alice";
- Example:
- Structs and Enums: Use
PascalCase
.- Example:
struct UserAccount { ... }
- Example:
- Constants: Use
UPPER_SNAKE_CASE
.- Example:
const MAX_USERS: u32 = 100;
- Example:
- Module Names: Use
snake_case
.- Example:
mod user_account;
- Example:
- Public Items: All public functions, structs, and modules must have documentation comments using
///
.- Example:
/// Creates a new user account. /// /// # Arguments /// /// * `name` - The name of the user. /// /// # Returns /// /// A `UserAccount` struct. pub fn create_user_account(name: &str) -> UserAccount { // function body }
- Example:
- Private Items: Document private items where the intent or functionality is not immediately clear.
- Module Documentation: Each module should have a comment at the top explaining its purpose.
- Example:
//! This module contains utility functions for user management. // module contents
- Example:
- Use of
Result
andOption
:- Use
Result
for operations that can fail andOption
for values that may or may not be present. - Example:
fn find_user(id: u32) -> Option<User> { // function body } fn open_file(path: &str) -> Result<File, io::Error> { // function body }
- Use
- Custom Error Types: When appropriate, define custom error types using
enum
and implement theanyhow::Error
trait. - Error Propagation: Propagate errors using
?
where possible to simplify error handling.