Up until now in your Eventonica project, all the data is deleted every time you refresh the page (unless you've added localStorage) and you can't have multiple users of your app share data. That's because the data you're storing is stored in your web browser. In this part of the Eventonica project, we'll create an Express API to store the data and serve it to all users of your site.
For example, in your code before, to get all the events, you might've had a function like app.getAllEvents()
. Instead, we will create an API endpoint like http://127.0.0.1:4000/events
that returns all the current events as a JSON response.
-
Before, each tab had it's own copy of events. Now they can be stored in one location so all users can see the same data and interact with it
-
Centralizing the logic allows us to add a database so the data will live on even if the server is restarted or crashes
-
Turning the logic of our system into a REST API makes it way more flexible
- you could call it using a non-JavaScript client such as Postman or an app written in a different language (like an Android app in Java). All modern languages speak HTTP.
So let's get to it!
The following directions are an adaptation of this freeCodeCamp tutorial, but changed to suit our Eventonica project.
-
In your terminal, navigate to your
eventonica/client
directory . Start it withnpm start
. -
You will need to create a second app for your Express backend. In a second terminal window, navigate to your general
Eventonica
project folder. Follow these commands to create a new project calledserver
and start it. If it asks you if you want to downloadexpress-generator
, chooseyes
.npx express-generator server cd server npm install
Note: If you see warning on your terminal like
npm WARN deprecated [email protected]: Deprecated, use jstransformer <br/> npm WARN deprecated [email protected]: Please update to at least constantinople 3.1.1 <br/>npm WARN deprecated [email protected]: Jade has been renamed to pug, please install the latest version of pug instead of jade
You need to follow this instructions
- In package.json remove Jade
- Install Pug (Jade has been renamed to pug)
npm install pug
- Run
npm install
again - In server/app.js changes
app.set('view engine', 'jade');
toapp.set('view engine', 'pug');
- In views folder rename all jade file to pug. Ex: index.jade -> index.pug
- Run
npm audit
to check for any vulnerabilities again
-
Inside the server directory, go to bin/www and change the port number from 3000 to 4000. You will be running both apps at the same time later on, so doing this will avoid issues.
// server/bin/www var port = normalizePort(process.env.PORT || '4000'); app.set('port', port);
-
In your second window, you should now be able to start
server
on port 4000 without any problems using the commandnpm start
. Open a browser window and go to http://localhost:4000/. If it's working, you should see a welcome message! -
Open
server/routes/index.js
and find following code:res.render('index', { title: 'Express' });
Change the title so it says this instead:
res.render('index', { title: 'Our express app is working properly' });
-
Stop your
server
app and restart.http://localhost:4000
should now show your new message. -
.gitignore
yournode_modules
. Push your project up to GitHub.
Since Node v14, there are two kinds of modules, CommonJS Modules (CJS) and EcmaScript Modules (ESM) . Lets convert our CJS file to ESM:
Steps to move cjs project to ESM
- Tell Node.js that all files are ES Modules by adding
"type": "module"
to the package.json. - Use the .mjs file extension for all your files. Example:
bin/www.mjs
,routes/index.mjs
,routes/users.mjs
,views/app.mjs
- Update start script in package.json
"start": "node ./bin/www.mjs"
- Replace all require()/module.export with import/export.
- for example: In
app.mjs
file changevar express = require("express");
toimport express from "express";
- at the end of the
app.mjs
file changemodule.exports = app;
toexport default app;
- for example: In
- Now start the server.
- did you get the error:
__dirname
is not defined in ES module scope? try to solve by yourself or refer this article
View github source code for eventonica/server here
-
Duplicate your
server/routes/index.mjs
file and name itserver/routes/events.mjs
. In this new file, change line 6 to say:res.render('index', { title: 'This is my events route.' });
-
In
server/app.mjs
, add this to line 25:app.use("/events", eventsRouter);
You'll need to importeventsRouter
, so add this to line 9:import usersRouter from "./routes/users.mjs";
-
Stop your
server
app and restart.http://localhost:4000/events
should now show your new message: This is my events route. You just made a new route!Note: Obviously, any other app calling
http://localhost:4000/events
would be doing it to get data, not to get a visual web page, but it's nice to have proof that things are working so far. Thanks express-generator!
-
Copy your list of mock users from your client/src/components/Users.js file to the
server/routes/users.mjs
into an array calledusers
.let mockUsers = [ { id: 1, name: 'Marlin', email: '[email protected]' }, { id: 2, name: 'Nemo', email: '[email protected]' }, { id: 3, name: 'Dory', email: '[email protected]' } ];
-
Update the endpoint so it returns
res.json({users:[your mock users here]});
// server/routes/users.mjs` router.get('/', function (req, res, next) { console.log(req.body, 'the body'); res.json({ users: mockUsers }); });
Check Review the difference between res.json
and res.send
. Typically front-end apps to expect to receive responses as a JSON.
Restart your server. Do you see the users list at http://localhost:4000/users
?
What shape should the response object be?
The example response returns { users: [array of users] }
instead of just res.json(users)
. This is because naming the user object users
is a more clear way of presenting this information. Also, this endpoint could send other information in addition to the user array. For example, it is common to have pagination in GET endpoints. So the response could eventually be something like res.json( { users:[user array], pagination: {pageSize: 10, page: 1 } });
Testing in Postman (or a similar app) is a great way to test and understand your backend services. While you can test the URLs in Chrome, or test your endpoints by calling them in React, they offer less flexibility, and/or might have less context into the endpoint's behavior and errors.
-
Testing the endpoint Open Postman, and open a new tab called "Eventonica". "GET" should be automatically selected in the dropdown. Enter
http://localhost:4000/users
into the request URL and press send. You should see the same list of users in the response below. -
Testing sending information Now the get users endpoint always returns all users. However, in the future we might want filtering.
To test this, you can console log console.log(req.body, 'the body')
before the res.json
line. In Postman, now try adding a body. Click the "body" tab and select "raw". Then select "JSON" from the dropdown. Try sending a JSON of something that the API might send. For example, it could send {"name": "nemo"}
for when you implement filtering. If you added the console log, do you see this information printed in your express server terminal? (Note that console logs will not show up in Postman).
-
Back in your frontend, open
client/src/Users.jsx
. Add this code to be the next line right afterconst Users = () => {
so that it is inside your React code block:const [users, setUsers] = useState([]); console.log('users', users); const getUsers = () => { fetch('http://localhost:4000/users') .then((res) => res.json()) .then((res) => setUsers(res.users)); }; useEffect(() => { // useEffect will run getUsers() every time this component loads, as opposed to just the first time it is rendered. getUsers(); }, []);
Check Why do we set
setUsers(res.users)
instead ofsetUsers(res)
? Remember thatres
is an object with a keyusers
, but it could also have other information. Try to keep states as specific as possible. ie if theres
response did have other information such as pagination, we want to keep theusers
state as "specific" as possible, so that it only stores the users array part of the API response. -
If you look at http://localhost:3000/ or your terminal, it will probably say that
useState
anduseEffect
are not defined. You should import these React hooks from React like this on line 1:import React, { useEffect, useState } from 'react';
-
If you visit http://localhost:3000/ and look in your User Management section.... you won't see it. But if you look in your console, your console log should be working as expected and printing
apiResponse
. You may be getting a403 error: forbidden
or aAccess-Control-Allow-Origin
message. So what's the problem? -
We need to allow cross-origin resource sharing. By default, your Express app will block "localhost:3000" because it's not using the same domain as itself, "localhost:4000". But since you're working locally, we can disable this for now.
-
In your terminal navigate to the
server
directory, stop your app, and install the CORS package:npm install --save cors
-
In
server/app.mjs
, import CORS on line 6:import cors from "cors";
-
Now on line 22 have express use CORS:
app.use(cors());
-
Restart
server
. If you refresh localhost:3000, you should see the response from your/users
route!
Now your challenge is to:
- Move your example events out of
client/src/components/Events.jsx
and intoserver/routes/events.js
and make sure it is a single array of events. - Have the events array be the response from http://localhost:4000/events, and make sure it renders in your frontend on localhost:3000
- Have your React Events component render events as HTML list items rather than plain text.
Add remaining REST API routes for users
listed in the project README. Don't worry about events
or favorites
for now... you can add those after we get our database going.
Users
now needs "add" and "delete" functionality. For example, a frontend function called addUser()
should make a POST request to http://localhost:4000/users/
and add a user by posting JSON with your API (which currently just saves to a variable since we have no DB), and the API would need a route like this:
router.post('/', function (req, res, next) {
// save request data to a variable in routes/users.js
res.send('some message about your data being saved, and a copy of that data');
});
Tip
Did you declare your users
variable with var
, const
, or let
? You can review the differences here
This could make a difference depending on how you implement your addUser
functionality - how so?
-
Commit after every successful addition - that way if you get mixed up, you have a clean save point to return to.
-
Use Postman or your preferred tool to test the API routes.
-
Try refreshing the page and using it in multiple browser windows. Your data is persisted in memory in Express, and will only be cleared when the app is restarted. When can the data be saved across sessions? The solution is to add a Postgres database in the next outline.
-
You can use body-parser middleware to parse JSON request body data.
-
If you run into a CORS error, you can use cors middleware to sidestep the error for local development.
- Try adding error handling to one or more of your Express API's. These are useful for returning errors when API calls have missing/malformed data.
See mention of CORS above.
- Check the log of your Express API server - you enabled morgan logging, right? 😇
- If the request is making it there, maybe the path is wrong
- If the request is not making it there, make sure you're calling fetch with an Accept header. If you don't, it will be handled by your React app server, which does not know about your Express routes
- Your fetch calls should look something like:
fetch('/data', { headers: { "Accept": "application/json" } })
will make be sent to the configured proxy, e.g.http://localhost:3000/data
- Example of calling API from React component using fetch
Pro Tip - the morgan middleware is nice to log all requests to your server
- Once you have your app working, it might be helpful to put all the
fetch
code for calling your API in a dedicated module, perhaps calledeventonica-api.js
and then calling it from your component