How to Render a React App Using an Express Server in Node.js | by Yogesh Chavan | Level Up Coding
Before diving into that, let’s first understand how to serve static files using Express.js
This is very useful if you are creating a full stack app using Express.js , React.js , Node.js and any database in Nodejs . So we can run our React.js and Node.js app on the same port, avoiding the need of running two separate commands to start Node.js and React.js app.
In this article, we will explore how to render a React app using Express.js server.
Learn how to serve your entire build folder created by create-react-app using Express.js
How to Render a React App Using an Express Server in Node.js
Serve static files using Express.js:
There are some requirements where we need to access some files in browser but If you are using Express.js
in Node.js
which runs on the server side, the files are not available automatically. To get that working we need to make some code changes. So let’s do that.
Create a new folder with the name express-static-serve
and inside that folder, create another folder with the name server
and run following command from the terminal from inside the server
folder.
yarn init -y
This will create a package.json
file inside the server
folder
Install the express
and nodemon
packages by running following command
yarn add [email protected] [email protected]
nodemon
is used to restart the app when we make any change in the code
Create a new file index.js
and add the following content inside it
index.js
Add new script inside scripts
section of package.json
"start": "nodemon index.js"
package.json
Now, start the application by running yarn start
command from server
folder
If you navigate to http://localhost:5000/. you will see the message displayed in the browser
This static hardcoded message is fine, but what If we want to render the contents of HTML file which has CSS and Javascript files included instead of text — how can we do that?
For that, create a new folder with the name public
and create two folders with the name js
and css
inside it.
Add script.js
inside public/js
folder with the following content
scripts.js
Add styles.css
inside public/css
folder with the following content
styles.css
Add index.html
inside public
folder with following content
index.html
Now, inside index.js
, add an import for path
package and send the index.html
file in response for the /
route
const path = require("path");
app.get("/", (req, res) => {
res.sendFile(path.join(__dirname, "public", "index.html"));
});
Updated index.js
Now, If you check the application, you will see that button from index.html
is displayed but Javascript and CSS files are not loaded and error is displayed in the browser console
Application
So, Express.js
is not able to render the static CSS and JavaScript file from public
folder using sendFile
method.
To fix this we need to add a middleware
using express.static
method provided by Express.js
So instead of
app.get("/", (req, res) => {
res.sendFile(path.join(__dirname, "public", "index.html"));
});
add following code inside index.js
app.use(express.static("public"));
index.js
This will inform Express.js
to serve all the files from public
folder
Now, if you check the application at http://localhost:5000/, you will see that there is no error in console and CSS is correctly getting applied and clicking on button also displays the current date
Working App
If we have any other file inside the public
folder like help.html
or contact.html
, we can access that also by direct URL like http://localhost:5000/help.html or http://localhost:5000/contact.html
Create a new users.json
file inside public
folder with following content
users.json
which you can access by visiting http://localhost:5000/users.json
So by just adding express.static
method we can serve all JavaScript, CSS, HTML, and even images.
Note: We can have multiple express.static
methods in a single file each for serving different folder content.
You can find source code until this point HERE
Render React app using Express.js:
Create a new React app using create-react-app
by running following command from express-static-serve
folder
create-react-app react-app
So now, you will have two folders inside express-static-serve
folder, one with the name server
and others with the name react-app
Navigate to react-app
folder from the terminal / command prompt and run following command to install axios
library
yarn add [email protected]
Once done, delete all files from react-app/src
folder and create index.js
, App.js
and styles.css
inside src
folder
Open App.js
and add the following content
App.js
Open styles.css
and add the following content
styles.css
Open index.js
and add the following content
index.js
Now, move the server
folder from express-static-serve
to react-app
folder
So your folder structure will look like this
folder structure
Now, open package.json
from react-app
folder and change
"start": "react-scripts start",
to
"start-client": "react-scripts start",
and add new start
script
"start": "yarn run build && (cd server && yarn start)"
Here, we will generate the build
folder first and then serve that build
folder using Express.js
So your package.json
will look like this
package.json
Now, open server/index.js
and add an import for path
and another app.use
to serve the build
folder that will be created by create-react-app
soon.
const path = require("path");
app.use(express.static(path.join(__dirname, "..", "build")));
app.use(express.static("public"));
Here, in the first app.use
, as the build
folder will be created inside react-app
folder, we are creating a path for the build
folder located outside the server folder.
Note: Order of app.use
matters here. We want our React
app to be displayed on page load, so the build
folder should be loaded first as it will also contain index.html
. If we add public
folder first then index.html
from server/public
folder will be loaded as Express.js
reads the file from top to bottom and it stops rendering when it finds the first matching file.
Updated index.js
Now, navigate to react-app
folder from terminal and run yarn start
command and you will see the React app is displayed and files like help.html
, users.json
are also accessible on the same port of 5000
.
Final App
If your React App is using routing then it will not work because we’re starting server from Express.js
so the request will always go to the Express.js
server and there will not be any route for handling that.
So to fix this, add the following code after all the Express.js
routes and middlewares in server/index.js
file.
app.use((req, res, next) => {
res.sendFile(path.join(__dirname, "..", "build", "index.html"));
});
So now, when the request comes to the server for any route and route you’re trying to access does not exist on the server-side, we will be sending the index.html
file from the build
folder so your client-side route (React App) will handle that and will display the client-side route page.
Your final server/index.js
file will look like this:
You can find the complete source code for this article HERE and the live demo HERE
That’s it for today. I hope you learned something new.
Check out my recently published Mastering Redux course.
In this course, you will build 3 apps along with a food ordering app and you’ll learn:
- Basic and advanced Redux
- How to manage the complex state of array and objects
- How to use multiple reducers to manage complex redux state
- How to debug Redux application
- How to use Redux in React using react-redux library to make your app reactive.
- How to use redux-thunk library to handle async API calls and much more
and then finally we’ll build a complete food ordering app from scratch with stripe integration for accepting payments and deploy it to the production.
Don’t forget to subscribe to get my weekly newsletter with amazing tips, tricks, and articles directly in your inbox here.