Chat app with React & Rust
Learn how to write a Rust chat application with React on the frontend.
Source code can be found here.
With Rust going from strength to strength within the web development space, it is clear why many developers and big names are starting to take notice. As one example of this, Meta has recently recommended Rust as a server-side language. If that won’t make people sit up and look, then it’s hard to know what will - Rust’s current offering easily stands on par with most other languages that you could use in a back-end API or microservice, and it will only get better with time.
Let’s explore Rust in everyday usage by creating a Typescript React app and combining it with a Rust API that uses WebSockets. While node.js is quick to set up, doesn’t require context switching and is easy to use if you already have Javascript knowledge from learning it for writing front-end web apps, you don’t need to have a high level of knowledge in Rust to get started writing competent web services that can easily carry out whatever you need.
Initial setup
We will be using Vite to scaffold our project, as it’s a quick and fast bundler for starting up your development environment quickly as well as being less opinionated than create-react-app. Let’s get into it:
npm create vite@latest wschat-react-rust --template react-ts
This should now scaffold a project within a subfolder of your current working directory called wschat-react-rust
.
For our CSS, we’ll be using TailwindCSS. Tailwind is a utility-first CSS library that allows you to be able to quickly and easily scaffold smaller projects without having to constantly fight media queries by providing utility classes with a mobile-first approach (as a side note: this isn’t necessarily better or worse than regular CSS - this is just how I like to do it!). You can find out how to install it here. It’s quick, easy, and very easy to configure.
Before we start, make sure you delete all of the HTML from the App menu (make sure you return an empty div!), make sure any unnecessary imports are removed and ensure that Tailwind is in your main CSS file.
Here are the contents of my CSS file if you’d like to use my CSS styles exactly:
Getting Started
Before we do anything, let’s quickly scaffold our page so that we have something that we can look at (we’ll be putting this in the main App component but feel free to put this on a page component):
If you’ve already used Typescript, this component should be simple to understand. For the uninitiated however: the only changes that are there at the moment in comparison to a pure JavaScript project is that we’ve declared a new type (“Message”) which we’ll be using later on, and we’ve also had to declare specific types for our state setters as well as e.target.value
. This is important as TypeScript needs to know what type our events are, otherwise it’ll complain and refuse to compile.
That’s pretty much it for the main component! We need a modal that can get a name, and then we just need to set up WebSocket functionality. Let’s create our modal:
When we initially load up our webpage, we want this modal to appear before the user enters the chatroom as we need the user to set a name, which means we should make it so that the modal is initially visible, but once the user has confirmed a name (and is in the chat), we should hide the modal. Like before, the only real difference here in comparison to Javascript is we’ve declared types for our props as Typescript needs to know what to parse them as - otherwise, it won’t compile.
Now we can simply proceed to import the modal into our page component like so (don’t forget to pass props and use React fragments if required!):
Now that the main design of the app is done, let’s think about how we can implement a WebSocket connection. To start with, we can open a WebSocket connection at a URL by simply writing the following:
This opens a WebSocket connection at localhost:8000/ws
. Not particularly useful at the moment because we currently don’t have anything we can connect it to, but we’ll need this for testing later on.
Now that we’ve opened a WebSocket connection, we can add a method for when the connection opens, when it closes, and when we receive a message - like so:
Although we’ve told our program that we want to create a message entry when we receive a message, we don’t have a create_message
function at the moment. This function will simply consist of creating a new HTML element, appending some classes and creating the text that will go inside the container div (and appending it to the container), and then appending our message itself to the chatbox as well as scrolling down to the bottom.
Now our front end is pretty much done!
Setting up Rust
Getting started with Rust is very easy. You can install it on Linux or WSL (Windows Subsystems for Linux) by using the following command:
If you’re on Windows and don’t have WSL, you can find the install page for Rust here.
However you install it, you’ll also get Rust’s package manager called Cargo, which is like NPM for Rust. Cargo allows you to install Rust’s packages which are called “crates”.
For the back end part, because we’ll be serving the web server through Shuttle, we will need to install their CLI which we can do with the following command:
You can also use cargo-binstall
to install cargo-shuttle:
The installation may take a while depending on your Internet connection, so feel free to grab a drink while you wait. You will also want to log onto their website here through GitHub and make sure you have your API key as you will need to log in on the CLI with your key before you can make any projects.
Once the installation is done, you can start a Shuttle project with the following command below (run this in your React project at the packages.json level):
This will prompt you to input a project name - once you’ve inputted a project name, this will scaffold a project for you that uses Axum, which is a Rust web framework that is easy to build on with simple syntax. The project will be built in a folder within the current working directory with the name you chose. For this article, we will simply refer to the folder as “API” for clarity.
Once the project has been created, you’ll want to go into your Cargo.toml
and make sure it looks like the following:
This will set up our project with all of the required dependencies for our project so we can simply just import them in as required.
As it would be ideal for us to have our front and back end running at the same time, there is an npm package we can use called concurrently which we can install at the packages.json
level like so:
Now we can write an npm script to run both our front and back ends in one command! Let’s look at what that would look like:
Running this npm command while at the packages.json level simply starts up your React app and also launches your Rust project so you can work on both at the same time.
Getting Started (with Rust)
To get started, let’s create all the values we need for the server to work.
Let’s quickly dissect what these types actually mean. If you’d like to read more about what an arc is you can do so here, but in short: it’s a smart pointer that can be cloned and holds a value. In this case, we’re using it to hold a reader-writer lock (“RwLock”), which is typically used when you want the value inside to be read across multiple threads at the same time, but you want exclusive thread access for write operations (ie, can’t mutate the value in any way from another thread). In short: it’s like having a box of stuff that lets you know what’s inside when you look at it, but to change the contents you have to open the box itself (and only one person can open it at a time!).
AtomicUsize is used for user IDs as we will want the value to be shared safely across threads. You can read more about Atomic values here. We will also want our messages to be able to be serialized and deserialized from JSON - hence, the derive macro provided to us by the serde crate.
Let’s quickly make up our main function so that we have a working route that we can test with our front end:
Now at the moment we have our main application loop and a router, but as you may have noticed, ws_handler
doesn’t actually exist in our code at the moment. This is what we will be writing next, and it can be simply written as so:
This function simply receives a connection, upgrades the connection into a WebSocket connection and initiates the socket handling to be able to receive and send messages.
Now let’s implement the handle_socket
function, as it currently doesn’t actually exist:
As you may have noticed in this function, we spawn a thread to await messages and send them back. We’ll need a method of safely transporting messages across the thread we’ve created, which is why an Arc with a reader-writer lock is used.
If you use shuttle run
to locally run this project and send “Hello” to the WebSocket connection from a front-end web app, on your terminal with the Rust project deployment should return Some("Hello")
, which means we’ve successfully received a WebSocket message. Now we just need to figure out how we’re going to send it back!
Let’s create a function that will broadcast messages along every connected WebSocket:
This function will basically check that the message type matches and if it does, iterate through every single user connection and read it. Nothing too crazy here.
Let’s have a look at enriching the results:
This function adds the user ID to the incoming message. If the result is not a WebSocket message, return whatever the result was.
Now we will incorporate both of these methods into the respective section in our handle_socket
function, like so:
Now if you send a message from your front-end web app to your web server, you should receive a message in your React app from the server with the message, username and user ID! We are finally done building the bare bones of the app, but there are some other things before we’re finished that you will probably want to consider.
Now while the app is now technically a minimum viable product, there’s a couple of other things we need to sort out, like compiling React assets into our Rust project and making sure that we have a way to manually disconnect users if they’re being abusive or breaking the rules of the chat.
Admin Routing
Before we get started however, let’s quickly update our main function so that we can quickly pull in our environment variables and use static assets:
Let’s dissect our new additions quickly. shuttle_runtime::Secrets
allows us to set environmental variables using a Secrets.toml file, much like how you’d typically use a .env file to be able to use environment variables locally and in production.
Now that our setup for this section is out of the way, let’s cover the admin route first as that’ll mean we can make sure our WebSocket service is complete before we compile any assets. Let’s make a function that will take a user ID and manually disconnect them using the disconnect function we already have:
Now we can easily set up an admin router within the router function that will allow us to disconnect people manually, given a user ID and an authentication secret which you should write like so:
Now that we have this route, we can actually embed it into our main router using the “nest” method. This method is actually great for us, as it means we can put together several different groups of similar routes and functions to create one router. Let’s have a look at what this should look like on the router we’re returning:
As you can see here, our admin route should actually be <base-url>/ws/admin
, as dictated by the nest function. Now if we want to disconnect a user with the user ID of 5 manually for example, we would have to make a post request to <base-url>/ws/admin/disconnect/5
with a Bearer authentication header - as an example, if you’ve set your secret as “keyboard cat”, you need to enter an authentication header of Bearer keyboard cat
.
At this point, your router function should look like this (if not, you have likely missed a step somewhere):
Integrating Front & Back Endpoints
Now we can start integrating our front and back end together. Let’s set up our npm deploy scripts so that we can build our assets into our Rust folder:
Our build command basically tells npm that we want it to compile all of the Typescript files, and then build the compiled assets.
Now in your vite.config.ts
file you’ll want to have your defineConfig look like so:
This tells npm exactly where we want our compiled assets to be built and whether or not we should empty the target directory before building assets (outside of the regular directory, this is normally false by default so we need to set it to true).
Now if we run npm run build
, it should build our assets in the API folder in a subdirectory called static
. We can serve this directory to our users on the Rust project, which is great for us as it means we can simply use one deployment instead of having to manage two different deployments. shuttle_static_folder::StaticFolder
has a default value of “static”, so we don’t need to set the folder name manually.
Before we move on, let’s re-write the WebSocket URL so that it will dynamically match whatever the URL of our hosted project will be, instead of a fixed string. Let’s change our WebSocket connection in the React front-end like so:
Now that we’ve changed the connection URL, our compiled assets in our Rust folder will work on the local Rust run without us having to use our local Vite deployment! However, if we try to run our front end by itself, it won’t connect as this connection string relies on using the local Rust project - but you can change the connection string as required.
Now we can update our main function and router together so that the router will use tower_http
’s ServeDir
function for serving static files:
Finishing Up
Now we’re pretty much done and ready to deploy! We can call our deploy script at the packages.json
level by setting up an npm script like below:
Now if we run npm run deploy
, it should build all of our assets into the required folder and then attempt to deploy to Shuttle - assuming there are no issues, it should deploy successfully!
If you would like to change the name of your folder while keeping the deployment name the same, you can do so by simply creating a file called Shuttle.toml
at the Cargo.toml
level and creating a variable for the name
key like in a .env file (so for example if I wanted to call my project keyboard-cat
, I’d type “name=‘keyboard-cat’” into the file).
If you need to check the status of your Shuttle project at any time, you can do so by using shuttle status
at the Cargo.toml
file, or you can add the --name
flag followed by the project’s name to use it from any directory.
Was this page helpful?