- 📘 Day 28 - Game Development with Rust 🎮
- 👋 Welcome
- 🎯 Why Rust for Game Development?
- 🎮 Introduction to Game Development
- 🛠️ Setting Up a Rust Game Development Project
- 📦 Game Development Libraries and Frameworks
- 🖥️ Creating a Simple Game with Rust
- ⚙️ Handling Game Physics and Input
- 🧑🤝🧑 Multiplayer Game Development
- 🎲 Your First Rust Game: A Simple Pong Clone
- 🚀 Advanced Concepts in Rust Game Development
- ⚙️ 1. Using ECS for Complex Games
- 🏎️ 2. Optimizing Game Performance
- 🌐 3. Networking in Multiplayer Games
- 🚀 Hands-On Challenge
- 💻 Exercises - Day 28
- 🎥 Helpful Video References
- 📚 Further Reading
- 📝 Day 28 Summary
Welcome to Day 28 of the 30 Days of Rust Challenge! 🎉
Today, we will dive deep into game development with Rust. Rust’s performance, safety, and concurrency model make it an excellent choice for building both 2D and 3D games. By the end of this lesson, you’ll have the knowledge to get started on creating your own game in Rust.
By the end of today, you’ll:
- Understand why Rust is a powerful choice for game development.
- Explore popular game development frameworks like Bevy and ggez.
- Build your first game in Rust—a simple Pong clone.
Rust’s key features make it a compelling choice for game developers:
Feature | Benefits for Game Development |
---|---|
Memory Safety | Prevents crashes and undefined behavior during gameplay. |
High Performance | Zero-cost abstractions and direct hardware access for smooth rendering. |
Concurrency | Makes it easy to build multi-threaded games for better performance. |
Modern Ecosystem | Libraries like Bevy, ggez, and nphysics simplify development. |
Rust combines the performance of C++ with the safety and modern tooling of higher-level languages, making it ideal for real-time applications like games.
Game development involves designing, coding, and deploying games. It requires working with different components such as:
- Rendering: Drawing graphics to the screen (2D/3D).
- Physics: Simulating movement, collisions, and other forces.
- Audio: Adding sound effects and music.
- User Input: Responding to keyboard, mouse, or gamepad input.
- Game Logic: Creating the rules and mechanics of the game.
Rust provides high performance, which is crucial for real-time applications like games, and it also offers excellent concurrency for handling tasks like multi-threading and real-time updates.
-
Install Rust: If you haven’t already, install the latest Rust version with:
rustup update
-
Create a New Project:
cargo new rust_game --bin cd rust_game
-
Add a Game Framework:
For this tutorial, we’ll use ggez to build our first game:[dependencies] ggez = "0.7"
-
Add Dependencies: For game development, you’ll likely need libraries for graphics, input handling, and audio. These will be included in the
Cargo.toml
file.
Rust is a modern systems programming language that offers high performance, memory safety, and concurrency—ideal for building games. Its growing ecosystem includes several libraries and frameworks that make game development accessible for developers of all experience levels.
Here are the top tools and frameworks for game development in Rust:
Framework | Description | Use Case |
---|---|---|
Bevy | A modern game engine with ECS (Entity-Component-System) architecture. | 2D/3D games. |
ggez | A lightweight library for 2D game development. | 2D arcade games. |
Amethyst | A highly customizable game engine with ECS and rendering pipelines. | Complex 2D/3D games. |
Piston | A modular game engine for 2D and GUI applications. | 2D games, tools. |
macroquad | A simple and fast library for 2D/3D games with WebAssembly support. | Browser games. |
- Use Bevy for large-scale, modern games with ECS.
- Use ggez or macroquad for small to medium 2D games.
Rust provides an extensive range of libraries and frameworks tailored for game development. Let’s explore some of the most popular ones:
Piston is a modular, lightweight game engine that supports 2D graphics and GUI applications.
- Modular design, allowing you to pick and choose components.
- Support for 2D graphics, events, and input handling.
- Easy integration with other Rust libraries.
- Developing lightweight 2D games.
- Building GUI-based applications.
Here’s how to set up a simple game loop in Piston:
extern crate piston_window;
use piston_window::*;
fn main() {
let mut window: PistonWindow = WindowSettings::new("Piston Game", [640, 480])
.exit_on_esc(true)
.build()
.unwrap();
while let Some(event) = window.next() {
window.draw_2d(&event, |_context, graphics, _device| {
clear([0.0, 0.0, 0.0, 1.0], graphics);
});
}
}
Bevy is a modern and flexible game engine that uses Entity-Component-System (ECS) architecture.
- ECS-based design for scalable and efficient games.
- Built-in support for 2D and 3D rendering.
- Hot-reloading assets for rapid development.
- Building large-scale games with complex systems.
- Developing 3D or multiplayer games.
Here’s a simple Bevy setup:
use bevy::prelude::*;
fn main() {
App::new()
.add_plugins(DefaultPlugins)
.add_startup_system(setup)
.run();
}
fn setup(mut commands: Commands) {
commands.spawn(Camera2dBundle::default());
}
Amethyst is a highly customizable game engine designed for complex 2D and 3D games.
- Advanced ECS architecture for clean and efficient code.
- Support for animations, physics, and audio.
- Extensive documentation and tutorials.
- Building 3D games with physics and animations.
- Creating simulation-based applications.
ggez is a lightweight library focused on making 2D game development fun and easy.
- Simple and intuitive API for rapid development.
- Support for rendering, audio, and input handling.
- Ideal for small to medium 2D games.
- Creating retro-style arcade games.
- Prototyping 2D gameplay concepts.
Here’s how to set up a basic ggez project:
use ggez::{Context, ContextBuilder, GameResult};
use ggez::event::{self, EventHandler};
struct MyGame;
impl EventHandler for MyGame {
fn update(&mut self, _ctx: &mut Context) -> GameResult<()> {
Ok(())
}
fn draw(&mut self, ctx: &mut Context) -> GameResult<()> {
ggez::graphics::clear(ctx, ggez::graphics::Color::BLACK);
ggez::graphics::present(ctx)?;
Ok(())
}
}
fn main() -> GameResult {
let (mut ctx, mut event_loop) = ContextBuilder::new("game_name", "author")
.build()?;
let mut game = MyGame;
event::run(&mut ctx, &mut event_loop, &mut game)
}
Building a game in Rust involves:
- Setting up a game loop: A loop that continuously updates and renders the game.
- Rendering graphics: Using a library like ggez or Bevy.
- Handling input and game logic: Detecting player actions and updating game state.
Let’s create a simple 2D game using ggez, a beginner-friendly framework.
[dependencies]
ggez = "0.6"
src/main.rs
:
use ggez::{Context, GameResult};
use ggez::event::{self, EventHandler};
use ggez::graphics::{self, Color, DrawMode, Rect};
struct MainState;
impl MainState {
fn new() -> MainState {
MainState
}
}
impl EventHandler for MainState {
fn update(&mut self, _ctx: &mut Context) -> GameResult {
Ok(())
}
fn draw(&mut self, ctx: &mut Context) -> GameResult {
graphics::clear(ctx, Color::WHITE);
let rect = Rect::new(100.0, 100.0, 50.0, 50.0);
graphics::set_color(ctx, Color::new(1.0, 0.0, 0.0, 1.0))?;
graphics::rectangle(ctx, DrawMode::fill(), rect)?;
graphics::present(ctx)?;
Ok(())
}
}
fn main() -> GameResult {
let (mut ctx, mut event_loop) = ggez::ContextBuilder::new("simple_game", "ggez")
.build()?;
let state = MainState::new();
event::run(&mut ctx, &mut event_loop, state)
}
This simple game creates a red square on a white background. You can expand this further by adding more game objects, handling user input, and implementing game logic.
Game physics and input handling are essential for creating interactive and immersive gameplay.
Physics adds realism to your game by simulating the behavior of objects. In Rust, libraries like nphysics and rapier simplify the implementation of physics systems.
- Support for collision detection and response.
- Rigid body and soft body dynamics.
For example, Rapier is a fast, 2D/3D physics engine for games. You can use it to handle rigid bodies, collisions, and forces.
use rapier2d::prelude::*;
fn main() {
let gravity = vector![0.0, -9.81];
let mut physics_pipeline = PhysicsPipeline::new();
let integration_parameters = IntegrationParameters::default();
let mut rigid_body_set = RigidBodySet::new();
let mut collider_set = ColliderSet::new();
let ground = RigidBodyBuilder::fixed().translation(0.0, -10.0).build();
rigid_body_set.insert(ground);
}
Handling player input involves capturing events such as keyboard and mouse actions and mapping them to gameplay mechanics.
- Detect key presses and mouse movements.
- Map inputs to actions for responsive gameplay.
use ggez::input::keyboard::{is_key_pressed, KeyCode};
use ggez::Context;
fn update(&mut self, ctx: &mut Context) {
if is_key_pressed(ctx, KeyCode::Up) {
self.player_position.y -= 5.0;
}
if is_key_pressed(ctx, KeyCode::Down) {
self.player_position.y += 5.0;
}
}
To handle user input (e.g., keyboard, mouse, or gamepad), most game engines like ggez and Bevy provide built-in methods.
Example for handling keyboard input with ggez:
fn update(&mut self, ctx: &mut Context) -> GameResult {
if ggez::input::keyboard::is_key_pressed(ctx, ggez::event::KeyCode::W) {
println!("W key is pressed");
}
Ok(())
}
You can handle different keys to control your player’s movement or game actions.
Multiplayer game development involves creating games where multiple players can interact with each other in real-time. Rust’s performance and memory safety make it an ideal language for building networked games, whether they are peer-to-peer or client-server based.
There are several components to consider when creating multiplayer games in Rust:
- Networking Basics: Understanding protocols (TCP/UDP, WebSockets), connection management, and data transmission is crucial.
- Synchronization: Ensuring that game state is consistent across multiple clients and that player actions are properly reflected in real-time.
- Latency and Lag Compensation: Handling the delay between players' actions and the server’s response in a way that feels seamless.
- Server Architecture: Deciding whether to use a central server to control the game logic or allow each player to handle their own game state.
Let’s break down the key elements and how to approach them in Rust:
In multiplayer games, players need to exchange data over the internet or a local network. This requires an understanding of network communication protocols. The two most common protocols are:
-
TCP (Transmission Control Protocol):
TCP ensures reliable, ordered, and error-free delivery of data between clients and servers. It’s ideal for games that require high consistency, like turn-based games, or games that need to track player stats in real-time. -
UDP (User Datagram Protocol):
UDP is faster than TCP but doesn’t guarantee reliability or order. It’s used in fast-paced games where speed is essential, and occasional data loss is acceptable, like in first-person shooters or real-time strategy games. -
WebSockets:
WebSockets enable real-time, full-duplex communication between the client and server over a single, long-lived connection. It's an ideal solution for browser-based multiplayer games.
To build multiplayer games in Rust, you will need libraries that handle networking. Here are some popular options:
- Tokio is an asynchronous runtime for Rust that makes it easier to write concurrent code, such as networked applications. It supports both TCP and UDP communication, and you can use it to build both client and server applications.
Add to Cargo.toml
:
[dependencies]
tokio = { version = "1", features = ["full"] }
- TCP Server Example with Tokio:
use tokio::net::TcpListener; use tokio::prelude::*; #[tokio::main] async fn main() -> Result<(), Box<dyn std::error::Error>> { let listener = TcpListener::bind("127.0.0.1:8080").await?; println!("Server is running on 127.0.0.1:8080"); loop { let (mut socket, _) = listener.accept().await?; tokio::spawn(async move { let mut buffer = [0; 1024]; if let Ok(n) = socket.read(&mut buffer).await { println!("Received: {:?}", &buffer[..n]); } }); } }
- Serde is a powerful library used for serializing and deserializing Rust data structures. It is essential when transferring complex data between clients and servers in multiplayer games.
Add to Cargo.toml
:
[dependencies]
serde = { version = "1.0", features = ["derive"] }
serde_json = "1.0"
- Using Serde for Serialization:
use serde::{Serialize, Deserialize}; use serde_json::{to_string, from_str}; #[derive(Serialize, Deserialize)] struct GameState { player_position: (f32, f32), score: u32, } let state = GameState { player_position: (100.0, 200.0), score: 42, }; // Serialize to JSON string let json_state = to_string(&state).unwrap(); // Deserialize from JSON string let deserialized: GameState = from_str(&json_state).unwrap();
- Tokio-tungstenite is a WebSocket library for Rust. It enables full-duplex communication between a server and clients, which is critical for real-time games.
Add to Cargo.toml
:
[dependencies]
tokio-tungstenite = "0.15"
- WebSocket Client Example:
use tokio_tungstenite::connect_async; use futures_util::{SinkExt, StreamExt}; #[tokio::main] async fn main() { let (mut ws_stream, _) = connect_async("ws://localhost:8080").await.unwrap(); // Send a message ws_stream.send(tokio_tungstenite::tungstenite::protocol::Message::Text("Hello!".to_string())).await.unwrap(); // Receive a message if let Some(msg) = ws_stream.next().await { match msg { Ok(tokio_tungstenite::tungstenite::protocol::Message::Text(text)) => { println!("Received: {}", text); } _ => {} } } }
In multiplayer games, maintaining synchronization between multiple clients is crucial. Without proper synchronization, players may see inconsistent or outdated game states. There are two common approaches to handle this:
-
Client-Server Model:
In this model, the server is the authority for the game state. Clients send actions to the server, which updates the game state and sends updates back to all clients. This ensures consistency but can be challenging when dealing with latency. -
Peer-to-Peer Model:
In a peer-to-peer model, all players share the game state. Each client communicates directly with the others, without a central server. This can reduce server costs, but the challenge is dealing with synchronization, especially when one peer loses connection.
In fast-paced multiplayer games, there’s often a delay between a player's action and the server's response. This is called latency, and it can cause the game to feel sluggish.
To mitigate latency issues, you can use techniques like:
- Client-side prediction: Clients predict the outcome of their actions before receiving confirmation from the server. For example, when a player moves, the client immediately shows the movement, even though the server hasn't confirmed it yet.
- Server reconciliation: After sending an action to the server, the client compares the server’s response with the prediction and adjusts accordingly.
- Lag compensation: Using techniques like interpolation and extrapolation, the game can predict the positions of distant players and adjust based on the data received from the server.
For multiplayer games, you need to choose how to structure the game’s server architecture. Common architectures include:
-
Dedicated Servers:
A dedicated server runs the game logic and handles all client connections. This ensures authoritative control over the game state and is suitable for large-scale multiplayer games. -
Peer-to-Peer:
Each player in a peer-to-peer network acts as both a client and a server. Players directly communicate with each other to share the game state. Peer-to-peer is less expensive but can be less reliable and harder to synchronize. -
Matchmaking Servers:
Many games use matchmaking servers to pair players together and establish direct connections. These servers help facilitate connections without directly handling the game state.
Here’s an overview of how to build a simple client-server multiplayer game in Rust using Tokio and Serde:
-
Server:
The server handles the game state and communicates with all connected clients. It listens for incoming messages (player actions) and broadcasts the updated state to all players. -
Client:
The client sends player actions (e.g., movement or attacks) to the server and listens for game state updates to display the correct information. -
Game Logic:
The server maintains the game state (e.g., player positions, scores, etc.) and updates it based on the messages it receives from clients. It then broadcasts the updated state to all players.
Let’s build a basic Pong game in Rust using the ggez library.
Create a game loop to handle rendering and logic updates:
use ggez::{Context, ContextBuilder, GameResult};
use ggez::event::{self, EventHandler};
struct PongGame;
impl EventHandler for PongGame {
fn update(&mut self, _ctx: &mut Context) -> GameResult<()> {
Ok(())
}
fn draw(&mut self, ctx: &mut Context) -> GameResult<()> {
ggez::graphics::clear(ctx, ggez::graphics::Color::BLACK);
ggez::graphics::present(ctx)?;
Ok(())
}
}
fn main() -> GameResult {
let (mut ctx, mut event_loop) = ContextBuilder::new("pong", "Author")
.build()
.expect("Failed to create context");
let mut game = PongGame;
event::run(&mut ctx, &mut event_loop, &mut game)
}
Add paddles and a ball to the screen:
fn draw(&mut self, ctx: &mut Context) -> GameResult<()> {
use ggez::graphics;
ggez::graphics::clear(ctx, graphics::Color::BLACK);
let paddle = graphics::Rect::new(20.0, 100.0, 10.0, 50.0);
let ball = graphics::Rect::new(100.0, 100.0, 10.0, 10.0);
let paddle_mesh = graphics::Mesh::new_rectangle(ctx, graphics::DrawMode::fill(), paddle, graphics::Color::WHITE)?;
let ball_mesh = graphics::Mesh::new_rectangle(ctx, graphics::DrawMode::fill(), ball, graphics::Color::WHITE)?;
graphics::draw(ctx, &paddle_mesh, graphics::DrawParam::default())?;
graphics::draw(ctx, &ball_mesh, graphics::DrawParam::default())?;
graphics::present(ctx)?;
Ok(())
}
Move the paddle using the keyboard:
use ggez::input::keyboard::{KeyCode, is_key_pressed};
fn update(&mut self, ctx: &mut Context) -> GameResult<()> {
if is_key_pressed(ctx, KeyCode::Up) {
self.paddle_position.y -= 5.0;
}
if is_key_pressed(ctx, KeyCode::Down)
{
self.paddle_position.y += 5.0;
}
Ok(())
}
Include collision detection and ball movement:
fn update(&mut self, ctx: &mut Context) -> GameResult<()> {
// Ball movement
self.ball_position.x += self.ball_velocity.x;
self.ball_position.y += self.ball_velocity.y;
// Collision with walls
if self.ball_position.y <= 0.0 || self.ball_position.y >= SCREEN_HEIGHT {
self.ball_velocity.y = -self.ball_velocity.y;
}
// Collision with paddles
if self.ball_position.collides_with(&self.paddle_rect) {
self.ball_velocity.x = -self.ball_velocity.x;
}
Ok(())
}
When building more complex games, understanding advanced techniques and leveraging Rust’s strengths can make a huge difference in performance, scalability, and interactivity. Let’s dive into these advanced concepts:
ECS (Entity-Component-System) is an architectural pattern commonly used in game development. Rust's focus on safety and performance aligns perfectly with ECS principles.
- Entity: Represents a unique object in the game world (e.g., a player, enemy, or bullet).
- Component: Contains data that defines the entity’s behavior or attributes (e.g., position, velocity, health).
- System: Handles the logic or operations applied to entities with specific components (e.g., moving entities with velocity components).
- 🚀 Scalability: Easily manage thousands of game objects without bloated code.
- 🔧 Flexibility: Modify components or add new features without impacting other systems.
- ⚡ Performance: Efficiently manage memory and CPU usage through data-oriented design.
Frameworks like Bevy and Amethyst have robust ECS implementations.
Example of ECS in Bevy:
use bevy::prelude::*;
fn main() {
App::build()
.add_plugins(DefaultPlugins)
.add_startup_system(spawn_entities.system())
.add_system(move_entities.system())
.run();
}
fn spawn_entities(mut commands: Commands) {
commands.spawn().insert(Position { x: 0.0, y: 0.0 }).insert(Velocity { x: 1.0, y: 1.0 });
}
fn move_entities(mut query: Query<(&mut Position, &Velocity)>) {
for (mut pos, vel) in query.iter_mut() {
pos.x += vel.x;
pos.y += vel.y;
}
}
struct Position { x: f32, y: f32 }
struct Velocity { x: f32, y: f32 }
Rust’s design already helps with performance, but games often need further optimization for a smooth experience.
🛠️ Strategy | 🔍 Description |
---|---|
Batch Rendering | Minimize draw calls by grouping similar objects into a single render batch. |
Spatial Partitioning | Use structures like quadtrees to quickly find objects in specific regions. |
Multithreading | Leverage Rust's threading model to parallelize computations. |
Asset Loading | Load assets asynchronously to prevent blocking the main thread. |
Memory Management | Use Rust's ownership and lifetimes to prevent memory leaks and reduce overhead. |
- 🔧 Use profiling tools like
tracy
orwgpu_profiler
to identify bottlenecks. - 🚀 Optimize physics calculations using libraries like
nphysics
orrapier
. - 🌐 For 3D games, implement frustum culling to avoid rendering objects outside the camera's view.
fn draw_batched_objects(ctx: &mut Context, objects: &[GameObject]) -> GameResult<()> {
let mut batch = ggez::graphics::spritebatch::SpriteBatch::new(my_texture);
for object in objects {
batch.add(ggez::graphics::DrawParam::new().dest(object.position));
}
ggez::graphics::draw(ctx, &batch, ggez::graphics::DrawParam::default())
}
Multiplayer games require efficient networking to handle real-time communication between players. Rust offers powerful tools for building reliable, low-latency multiplayer systems.
- Client-Server Model: Most games use this model where the server handles game logic, and clients handle rendering.
- Tick Rate: The frequency at which the game state is updated and synchronized.
- Latency Handling: Techniques like lag compensation and prediction mitigate the impact of network delays.
- 📡 Tokio: Asynchronous runtime for building scalable networked applications.
- 🌍 Quinn: Library for implementing the QUIC protocol, ideal for fast and secure communication.
- 🕹️ Laminar: A UDP-based networking library designed for real-time games.
Example of a basic server using Tokio:
use tokio::net::TcpListener;
use tokio::io::{AsyncReadExt, AsyncWriteExt};
#[tokio::main]
async fn main() -> Result<(), Box<dyn std::error::Error>> {
let listener = TcpListener::bind("127.0.0.1:8080").await?;
println!("Server running on 127.0.0.1:8080");
loop {
let (mut socket, _) = listener.accept().await?;
tokio::spawn(async move {
let mut buffer = [0; 1024];
if let Ok(n) = socket.read(&mut buffer).await {
socket.write_all(&buffer[0..n]).await.unwrap();
}
});
}
}
-
Minimize Bandwidth Usage
- Compress data packets.
- Send only delta updates (changes since the last update).
-
Handle Cheating and Security
- Validate game logic on the server side.
- Use encryption for sensitive data.
-
Implement Robust Error Handling
- Retry failed packets.
- Disconnect idle or unresponsive clients.
-
Synchronize Game State
- Use interpolation and prediction to ensure smooth gameplay despite latency.
- Create a basic 2D game where the player moves a square around the screen using arrow keys.
- Build a simple multiplayer chat application using WebSockets where players can send messages to each other in real-time.
- Implement basic collision detection for your player and other game objects.
- Add scoring and display the score on the screen.
- Create a simple game where two players control characters and interact in a shared world. The game should sync player positions across both clients in real-time using a server.
- Build a small multiplayer game using WebSockets and handle player synchronization.
- Add physics to your game using Rapier and simulate gravity and collisions.
- Implement a real-time strategy (RTS) game with player synchronization. Handle lag compensation, and make sure the game state remains consistent across clients with low latency.
- Create a Basic Game Loop:
- Use the
ggez
library to create a window and set up a simple game loop that updates and renders frames.
- Use the
- Handle Keyboard Input:
- Implement basic keyboard input to move an object (e.g., a square) on the screen using
ggez
's input handling functions.
- Implement basic keyboard input to move an object (e.g., a square) on the screen using
- Display a Score:
- Display a score that increases over time, representing the game progress, using text rendering.
- Basic Collision Detection:
- Implement collision detection between objects (e.g., a player object and obstacles) within the
ggez
framework.
- Implement collision detection between objects (e.g., a player object and obstacles) within the
- Sprite Animations:
- Create a character sprite and animate it by changing the displayed image frame over time (e.g., walking or jumping).
- Sound Effects:
- Add simple sound effects using the
ggez
library, such as a sound when the player moves or collides with an object.
- Add simple sound effects using the
- Physics Engine Integration:
- Integrate a simple physics engine (e.g.,
nphysics
) to simulate realistic gravity and movement of objects.
- Integrate a simple physics engine (e.g.,
- AI for Enemies:
- Implement a basic AI for enemy characters that can follow or chase the player.
- Multiplayer Support:
- Add multiplayer functionality using the
tokio
crate to allow two players to control characters in the same game.
- Add multiplayer functionality using the
Topic | Resource |
---|---|
Piston Game Engine | Piston Official Documentation |
Bevy Game Engine | Bevy Book |
ggez Game Development | ggez Documentation |
Rust Game Development Tutorials | Rust Game Development Tutorials |
Today, you learned how to:
- Set up a game development project in Rust.
- Work with libraries like ggez, Bevy, and Piston for game development.
- Implement basic game mechanics such as player movement, input handling, and simple graphics.
- Explore game physics and multiplayer networking.
Game development with Rust opens up a world of possibilities for creating high-performance, cross-platform games. Keep practicing by building more complex games, experimenting with different libraries, and learning about game design principles.
Stay tuned for Day 29, where we will explore Game Development with Rust in Rust! 🚀
🌟 Great job on completing Day 28! Keep practicing, and get ready for Day 29!
Thank you for joining Day 29 of the 30 Days of Rust challenge! If you found this helpful, don’t forget to star this repository, share it with your friends, and stay tuned for more exciting lessons ahead!
Stay Connected
📧 Email: Hunterdii
🐦 Twitter: @HetPate94938685
🌐 Website: Working On It(Temporary)