2021-01-31 11:05:37 +01:00
# Cargo build system and package manager
## Creating a project
```ps1
cargo new project_name # creates project folder and basic files
cargo new --vcs=git project_name # init project as git repo
```
2021-09-20 19:35:32 +02:00
## Building, Running & Checking a project
2021-01-31 11:05:37 +01:00
Inside the project directory:
```ps1
2021-09-20 19:35:32 +02:00
cargo build # build project and download eventual needed dependencies
cargo build --release # build project for release (build + optimisations)
2021-01-31 11:05:37 +01:00
cargo run # executes the built executable
cargo check # verifies buildability without producing an executable
```
2021-09-20 19:35:32 +02:00
## Dependencies
2021-01-31 11:05:37 +01:00
In `Cargo.toml` :
```toml
[dependencies]
crate_name = "< version_number > "
```
## Code Organization
2021-09-20 19:35:32 +02:00
Rust has a number of features that allow to manage the code's organization, including which details are exposed, which details are private, and what names are in each scope in the programs.
2021-01-31 11:05:37 +01:00
These features, sometimes collectively referred to as the module system, include:
- **Packages**: A Cargo feature that allows to build, test, and share crates
- **Crates**: A tree of modules that produces a library or executable
- **Modules** and `use` : Allow to control the organization, scope, and privacy of paths
- **Paths**: A way of naming an item, such as a struct, function, or module
### Packages & Crates
2021-08-13 19:00:10 +02:00
A crate is a binary or library. The crate root is a source file that the Rust compiler starts from and makes up the root module of he crate.
2021-01-31 11:05:37 +01:00
A package is one or more crates that provide a set of functionality. A package contains a `Cargo.toml` file that describes how to build those crates.
2021-08-13 19:00:10 +02:00
Several rules determine what a package can contain. A package must contain `zero` or `one` **library crates** (`lib` ?), and no more.
2021-09-20 19:35:32 +02:00
It can contain as many `binary` crates as you'd like, but it must contain at least one crate (either library or binary).
2021-01-31 11:05:37 +01:00
2021-08-13 19:00:10 +02:00
If a package contains `src/main.rs` and `src/lib.rs` , it has two crates: a library and a binary, both with the same name as the package.
2021-01-31 11:05:37 +01:00
A package can have multiple binary crates by placing files in the `src/bin` directory: each file will be a separate binary crate.
A crate will group related functionality together in a scope so the functionality is easy to share between multiple projects.
2021-08-13 19:00:10 +02:00
Cargo follows a convention that `src/main.rs` is the crate root of a binary crate with the same name as the package.
2021-01-31 11:05:37 +01:00
Likewise, Cargo knows that if the package directory contains `src/lib.rs` , the package contains a library crate with the same name as the package, and `src/lib.rs` is its crate root.
### Modules
Modules allow to organize code within a crate into groups for readability and easy reuse. Modules also control the privacy of items, which is whether an item can be used by outside code (*public*) or is an internal implementation detail and not available for outside use (*private*).
2021-09-20 19:35:32 +02:00
Define a module by starting with the `mod` keyword and then specify the name of the module and place curly brackets around the body of the module.
2021-01-31 11:05:37 +01:00
Inside modules, it's possible to have other modules. Modules can also hold definitions for other items, such as structs, enums, constants, traits, or functions.
### Paths
A path can take two forms:
- An **absolute path** starts from a crate root by using a crate name or a literal crate.
- A **relative path** starts from the current module and uses self, super, or an identifier in the current module.
Both absolute and relative paths are followed by one or more identifiers separated by double colons (`::` ).
```rs
module::function(); // rel path (same crate)
super::function();; // rel path starting in outer module (same crate)
crate::module::function(); // abs path (same crate)
< crate_name > ::module::function(); // abs path (other crate)
```
### Public vs Private
2021-08-13 19:00:10 +02:00
Modules aren’ t useful only for organizing the code. They also define Rust’ s privacy boundary:
the line that encapsulates the implementation details external code isn’ t allowed to know about, call, or rely on.
So, to make an item like a function or struct private, put it in a module.
2021-01-31 11:05:37 +01:00
2021-08-13 19:00:10 +02:00
The way privacy works in Rust is that all items (functions, methods, structs, enums, modules, and constants) are *private by default* .
Items in a parent module can’ t use the private items inside child modules, but items in child modules can use the items in their ancestor modules.
2021-01-31 11:05:37 +01:00
The reason is that child modules wrap and hide their implementation details, but the child modules can see the context in which they’ re defined.
```rs
mod module {
fn func() {}
}
// for a function to be accessible both module and function must be public
pub mod public_module {
pub fn public_func() {}
}
mod file_level_module; // define a module for the whole file (same name as file)
```
2021-08-13 19:00:10 +02:00
It's possible to use `pub` to designate *structs* and *enums* as public, but there are a few extra details.
2021-09-20 19:35:32 +02:00
If `pub` is used before a struct definition, this makes the struct public, but the struct's fields will still be private.
2021-08-13 19:00:10 +02:00
It's possible make each field public or not on a case-by-case basis.
2021-01-31 11:05:37 +01:00
In contrast, if an enum is made public, all of its variants are then public.
### `use`
```rs
use < crate_name > ::module; // import module (abs path, other crate)
2021-08-13 19:00:10 +02:00
use crate::module; // import module (abs path, same crate)
use self::module; // import module (rel path, same crate)
2021-01-31 11:05:37 +01:00
2021-09-20 19:35:32 +02:00
use < crate_name > ::module as alias; // import module w/ aliases
2021-01-31 11:05:37 +01:00
pub use < crate_name > ::module; // re-exporting (import and make available to others)
use < crate_name > ::module::{self, Item}; // import multiple paths
2021-07-12 16:18:53 +02:00
pub use < crate_name > ::module::*; // import all public items (Glob operator)
2021-01-31 11:05:37 +01:00
module::function(); // use func w/ shorter path
```
## Separating into multiple files
```txt
src
2021-08-13 19:00:10 +02:00
|_main.rs --> default executable file
|_lib.rs --> default library file
2021-01-31 11:05:37 +01:00
|__module
2021-08-13 19:00:10 +02:00
| |_mod.rs --> export submodules
| |_submodule.rs --> submodule
```
```rs
// main.rs
2021-09-20 19:35:32 +02:00
mod module; // declare module directory as a module
2021-08-13 19:00:10 +02:00
// mod.rs
pub mod sub_module; // declare sub_module file as a module
2021-01-31 11:05:37 +01:00
```