Enums and Pattern Matching
03/28/2025
word count:1356
estimated reading time:1 minute
Rust’s module system includes:
- Packages - A Cargo feature that lets you build, test and share crates
- Crates - A tree of modules that produces a library or executable.
- Modules and use - Let you control the organization, scope, and privacy of paths.
- Paths - A way of naming an item, such as a struct, function, or module.
Packages and Crates
A crate is the smallest amount of code that the Rust compiler considers at a time. Crates can contain modules, and the modules may be defined in other files that get compiled with the crate.
A crate can come in one of two forms: a binary crate or a library crate.
Binary
crates are programs you can compile to an executable that you can run, such as a command line program or a server. Each must have a function called main
that defines what happens when the executable runs.
Library
crates don’t have a main
function, and they don’t compile to an executable. Instead they define the functionality to be shared with multiple projects.
A package
is a bundle of one or more crates that provides a set of functionality. A package contains Cargo.toml
file that describes how to build the crates.
How modules, paths, use
and pub
work
- Start from the root - When compiling a crate the compiler first looks in the crate root file, usually
src/lib.rs
for a library crate orsrc/main.rs
for code to compile. - Declaring modules - In the root file you can declare new modules, like so -
mod new_module;
. the compiler will look for the module’s code in these places:- Inline, inside the curly braces:
mod new_module{}
src/new_module.rs
src/new_module/mod.rs
- Inline, inside the curly braces:
- Declaring submodules - In any file other than the crate root you can declare submodules. For example, you might declare
mod sub_module;
insrc/new_module.rs
, the compiler will look for the submodules code in the parent module directory, in these places:- Inline, inside the curly braces:
mod sub_module{}
src/new_module/sub_module.rs
src/new_module/sub_module/mod.rs
- Inline, inside the curly braces:
- Paths to code in modules - Once a module is part of your crate, you can refer to code in that module from anywhere else in that same crate, as long as the privacy rules allow, using the path to the code. For example:
crate::new_module::sub_module::Test
- Private vs Public - Code within a module is private from its parent modules by default. To make a module public, declare it with
pub mod
instead ofmod
. To make items within a public module public as well usepub
before their declarations use
keyword - Within a scope the use keyword creates shortcuts to items to reduce repetition of long paths. Adduse
to thecrate::new_module::sub_module::Test
line and you can use the Test in that scope.
Example of a Module Structure
// In src/lib.rs
pub mod front_of_house {
pub mod hosting {
pub fn add_to_waitlist() {}
pub fn seat_at_table() {}
}
pub mod serving {
pub fn take_order() {}
pub fn serve_order() {}
pub fn take_payment() {}
}
}
pub fn eat_at_restaurant() {
// Absolute path
crate::front_of_house::hosting::add_to_waitlist();
// Relative path
front_of_house::hosting::add_to_waitlist();
}
A small exercise to understand Rust module system and everything mentioned above
The super
keyword
Using super
allows to reference an item that we know is in the parent module, which can make rearranging the module tree easier when the module is closely related to the parent but the parent might be moved elsewhere in the module tree someday.
fn deliver_order() {}
mod back_of_house {
fn fix_incorrect_order() {
cook_order();
super::deliver_order();
}
fn cook_order() {}
}
Using super
here allows us to access a function that is scoped within the parent module, which in the above case is the crate, the root.
If, any time in the future, we decide to move this code to another module, this won’t break anything, since super
still allows us to access the root’s function.
Making Structs and Enums Public
We can also use the pub
keyword to make structs and enums public. But there are a few details to the usage of pub with structs and enums. If we use pub
before a struct definition, we make the struct public but its fields will still be private. We need to specify it for each field we want to make public:
mod back_of_house {
pub struct Breakfast {
pub toast: String,
seasonal_fruit: String,
}
impl Breakfast {
pub fn summer(toast: &str) -> Breakfast {
Breakfast {
toast: String::from(toast),
seasonal_fruit: String::from("peaches"),
}
}
}
}
pub fn eat_at_restaurant() {
// Order a breakfast in the summer with Rye toast
let mut meal = back_of_house::Breakfast::summer("Rye");
// Change our mind about what bread we'd like
meal.toast = String::from("Wheat");
println!("I'd like {} toast please", meal.toast);
// The next line won't compile if we uncomment it; we're not allowed
// to see or modify the seasonal fruit that comes with the meal
// meal.seasonal_fruit = String::from("blueberries");
}
Because the toast
field is public, we can write and read to the toast
field using dot notation: meal.toast = ...
Notice the comments below the println!
line, if we uncomment the last line we would get an error, since seasonal_fruit
is private.
In contrast if we make an enum
public, all of its variants are then public. We only need the pub
before the enum
keyword.
The default of enum is public
Bringing paths into scope using use
keyword
use crate::front_of_house::hosting;
.
.
.
hosting::add_to_waitlist();
Adding the path after the use
keyword, brings the hosting
module into scope, so we can use its children without specifying the entire path.
Note that use only creates this shortcut for the particular scope in which it occurs
Idiomatic Use Paths
The idiomatic way to bring functions into scope with use
is to bring the parent module into scope, not the function directly. This makes it clear that the function isn’t locally defined:
// Preferred - bring module into scope
use crate::front_of_house::hosting;
// Then use the function
hosting::add_to_waitlist();
// Less clear where the function comes from
use crate::front_of_house::hosting::add_to_waitlist;
add_to_waitlist();
For structs, enums, and other items, it’s idiomatic to specify the full path:
use std::collections::HashMap;
let mut map = HashMap::new();
The as
Keyword
You can use the as
keyword to provide a new name for a type when bringing it into scope:
use std::fmt::Result;
use std::io::Result as IoResult;
fn function1() -> Result {
// --snip--
Ok(())
}
fn function2() -> IoResult<()> {
// --snip--
Ok(())
}
Re-exporting Names with pub use
When we bring a name into scope with the use
keyword, the name is private to our scope. To enable code that calls our code to refer to that name as if it had been defined in that code’s scope, we can use pub use
:
// In a library crate (lib.rs)
mod front_of_house {
pub mod hosting {
pub fn add_to_waitlist() {}
}
}
// Re-export the hosting module
pub use crate::front_of_house::hosting;
// External code can now use it as:
// restaurant::hosting::add_to_waitlist();
Using External Packages
To use an external package, add it to your Cargo.toml
file:
[dependencies]
rand = "0.8.5"
Then bring it into scope in your code:
use rand::Rng;
fn main() {
let secret_number = rand::thread_rng().gen_range(1..=100);
}
Organizing Modules in Separate Files
In larger projects, you often want to split your code into multiple files.
For example, with a module structure:
src/
├── main.rs
├── front_of_house.rs
└── front_of_house/
├── hosting.rs
└── serving.rs
Your code would look like:
// src/main.rs
mod front_of_house;
use crate::front_of_house::hosting;
fn main() {
hosting::add_to_waitlist();
}
// src/front_of_house.rs
pub mod hosting;
pub mod serving;
// src/front_of_house/hosting.rs
pub fn add_to_waitlist() {}
pub fn seat_at_table() {}
// src/front_of_house/serving.rs
pub fn take_order() {}
pub fn serve_order() {}
pub fn take_payment() {}
Working with Workspaces
For large projects consisting of multiple related packages, Rust provides workspaces. A workspace is a set of packages that share the same Cargo.lock
and output directory.
Here’s how to create a workspace:
# In a file named Cargo.toml in your workspace root
[workspace]
members = [
"package1",
"package2",
"package3/package3-utils",
]
Then you can create each package in its own directory:
workspace/
├── Cargo.toml
├── package1/
│ ├── Cargo.toml
│ └── src/
│ └── main.rs
├── package2/
│ ├── Cargo.toml
│ └── src/
│ └── lib.rs
└── package3/
└── package3-utils/
├── Cargo.toml
└── src/
└── lib.rs
To depend on a package in the same workspace:
# In package1/Cargo.toml
[dependencies]
package2 = { path = "../package2" }
To build all packages in the workspace:
cargo build --workspace
Benefits of Workspaces
- Shared Dependencies: All packages in a workspace share one
Cargo.lock
file, ensuring consistent dependency versions. - Efficient Builds: Cargo optimizes builds by sharing build artifacts among packages.
- Easy Cross-References: Packages can easily depend on each other using relative paths.
- Coordinated Tests: You can test all packages in a workspace with a single command.