Before you start
We will provide a new skeleton for distributed implementation using Tonic, a Rust implementation of gRPC. gRPC is a high-performance, open-source universal RPC framework.
1. Setup Protocol Buffers
Protocol Buffers (protobuf) are a language-neutral, platform-neutral extensible mechanism for serialising structured data.
gPRC uses protobuf to (de)serialise structured data.
To install protobuf, type
sudo apt install protobuf-compiler
sudo apt install protobuf-compiler
brew install protobuf
Only if you are using native Windows rather than WSL2
download protoc-xxx-win64.zip
here (xxx represents the version)
Unzip and put it somewhere like C:\Program Files\protoc-xxx-win64
Add C:\Program Files\protoc-xxx-win64\bin
to your path environment variables
To check if protobuf compiler is working, type
protoc --version
2. Tonic (gRPC) hello world example
We recommend that you go through the hello world tutorial provided by Tonic.
You can skip this example if you are already familiar with Tonic or gRPC.
3. Download the skeleton
Download the distributed skeleton for the coursework here. (the distributed branch, not the master branch)
Make sure you are cloning or downloading the distributed
branch!
If you are using git clone
, switch to the distributed
branch by typing git switch distributed
If you are using Use this template
, please check Include all branches
and switch to the distributed
branch as well.
For WSL2 users
If you are using WSL2, ensure your skeleton is located within the WSL2 file system. Specifically, your project should be located at ~/.../gol-rs-skeleton
, NOT at /mnt/.../gol-rs-skeleton
For native Windows users
Don't forget to copy SDL2.lib
and SDL2.dll
to the root inside the gol-rs-skeleton
folder.
Open the skeleton with your IDE, the file structure should look like the following. If it doesn't, please make sure you have switched to the distributed
branch!
gol-rs-skeleton/ (the folder opened by IDE)
├── controller/
│ ├── benches/
│ ├── check/
│ ├── images/
│ ├── src/
│ ├── tests/
│ ├── build.rs
│ └── Cargo.toml
├── proto/
│ └── stub.proto
├── server/
│ ├── src/
│ ├── build.rs
│ └── Cargo.toml
├── .gitignore
├── Cargo.toml
└── README.md
Open a terminal, cd to the server
folder and start the server by typing
cd server
cargo run --release
Open another terminal, cd to the controller
folder, and start the controller by typing
cd controller
cargo run --release
You will see the server print something like this:
$ cargo run --release
[gol_server] request: World { width: 3, height: 3, cell_values: [255, 255, 255, 0, 0, 0, 255, 255, 255] }
And the controller will print something like this:
$ cargo run --release
[gol_rs::gol::distributor] response: AliveCellsCount { cells_count: 6 }
4. Explain the skeleton with gRPC examples
Controller (a.k.a. client) side
Open controller/src/gol/distributor.rs
, you may find it similar to the parallel version, except we've made remote_distributor()
an async function.
In example_rpc_call()
, we connect to the server first.
let mut client = ControllerHandlerClient::connect(format!("http://{}", server_addr)).await?;
The server address server_addr
is defined in Params
struct, and the Params
struct is converted from Args (controller/src/args.rs)
struct that defines command line arguments including the default server address. We used Clap as the command line argument parser.
To pass custom server address by command line argument, type (127.0.0.1:8030 in this case)
cargo run --release -- --server_addr "127.0.0.1:8030"
After connected to the server, you will see that we create and convert a 2D world Vec<Vec<CellValue>>
to bytes Vec<u8>
.
let bytes = world.iter().flat_map(|row| row.as_bytes()).copied().collect();
Then we push the "world" (bytes) to the server using an RPC call and wait (blocking) for the result to return from the server.
// Push the world to the server and receive the response (number of alive cells) by RPC call
// the RPC call `push_world()` is defined in `proto/stub.proto`
let response = client.push_world(
tonic::Request::new(World {
width: 3,
height: 3,
cell_values: bytes,
})
).await;
We handle the response by pattern matching, and print out the result (calculation of the number of alive cells) from the server.
Note that we also assert the correctness of the calculation performed by the server.
match response {
Ok(response) => {
let msg = response.into_inner();
log::info!(target: "Distributor", "response: {:?}", msg);
assert_eq!(
msg.cells_count as usize,
world.iter().flatten().filter(|cell| cell.is_alive()).count()
);
},
Err(e) => log::error!(target: "Distributor", "Server error: {}", e),
}
Finally, we make another RPC call to notify the server (broker) to shutdown.
// Another example of closing the server by RPC call
client.shutdown_broker(tonic::Request::new(Empty { })).await?;
Server (a.k.a. broker) side
Open server/src/main.rs
, where you will find the functions we invoked from controller within the ControllerHandler.
In shutdown_broker()
, we shutdown the server by sending a signal to the shutdown_tx
channel, which is ultimately handled on line 42.
Server::builder()
.add_service(ControllerHandlerServer::new(Arc::clone(&broker)))
.serve_with_shutdown(addr, async { shutdown_rx.recv().await.unwrap() })
.await?;
In push_world()
, we extract the data from the request and save the width
and height
by read-write locks.
async fn push_world(&self, request: Request<World>) -> Result<Response<AliveCellsCount>, Status> {
let world = request.into_inner();
log::info!("request: {:?}", world);
*self.width.write().await = world.width;
*self.height.write().await = world.height;
...
}
We also convert the bytes Vec<u8>
to a 2D world Vec<Vec<CellValue>>
and save it accordingly.
*self.cell_values.write().await = world.cell_values
.chunks(world.width as usize)
.map(|row| row.iter().copied().map(CellValue::from).collect())
.collect();
You can notice width
, height
and cell_values
are wrapped with RwLock
(read-write lock).
pub struct Broker {
shutdown_tx: Sender<()>,
width: RwLock<u32>,
height: RwLock<u32>,
cell_values: RwLock<Vec<Vec<CellValue>>>,
}
Finally, we count the number of alive cells and return the result to the controller as a response.
let alive_count = self.cell_values.read().await.iter()
.flatten().filter(|cell| cell.is_alive()).count();
// Return number of alive cells as response
Ok(Response::new(AliveCellsCount{ cells_count: alive_count as u32 }))
Stub.proto
Open the proto/stub.proto
, where you will find the definition of the "Interface" for communication between the client and the server.
5. Add all together
You are free to modify (and remove the examples above) from the skeleton; however, please keep the tests under controller/tests/**.rs
, you will use them for testing your distributed implementation.
Since gRPC is a language-neutral, platform-neutral RPC framework, you can implement different distributed components in various languages. This can also be treated as an extension.
For example, you could use Rust to write the server (broker or workers), while keeping the controller in Golang, or even try other languages.
- Controller (Golang) <==gRPC==> Broker/Server (Rust) <==gRPC==> Workers (Rust)
Note that the protocols are different between native Golang RPC (which you learnt in previous labs) and the gRPC; do not mix them up.
If you decided to use this mixture pattern, you should use gRPC in Golang as well, here's an example.