DEV Community

Cover image for 🍝 Cooking a Deliveroo clone with Next.js (React), GraphQL, Strapi and Stripe - 🏠 Restaurants list (part 2/7)
Ryan
Ryan

Posted on • Edited on

🍝 Cooking a Deliveroo clone with Next.js (React), GraphQL, Strapi and Stripe - 🏠 Restaurants list (part 2/7)

Strapi Next.js tutorial

This tutorial is part of the « Cooking a Deliveroo clone with Next.js (React), GraphQL, Strapi and Stripe » tutorial series.

Table of contents

Note: the **source code* is available on GitHub: https://github.com/strapi/strapi-examples/tree/master/nextjs-react-strapi-deliveroo-clone-tutorial*.

🏠 Restaurants list

First of all, we need to display the list of restaurants in our web app. Of course, this list is going to be managed through our API. So, we are going to start configuring it.

Define Content Type

A Content Type, also called a model, is a type of data. A Strapi API includes, by default, the user Content Type. Right now, we need restaurants, so our new Content Type is going to be, as you already guessed, restaurant (Content Types are always singular).

Here are the required steps:

  • Navigate to the Content Type Builder (http://localhost:1337/admin/plugins/content-type-builder).
  • Click on Add Content Type.
  • Set restaurant as name.
  • Click on Add New Field and create the followings fields:
    • name with type String.
    • description with type Text with Rich Text Editor (in the Advanced Settings section of the modal, select Display as a WYSIWYG).
    • image with type Media.
  • Click on Save.

Content Type Builder

At this point, your server should have automatically restarted and a new link Restaurant appears in the left menu.

Add some entries

Well done! You created your first Content Type. The next step is to add some restaurants in your database. To do so, click on "Restaurant" in the left menu (http://localhost:1337/admin/plugins/content-manager/restaurant).

You are now in the Content Manager plugin: an auto-generated user interface which let you see and edit entries.

Let's create a restaurant:

  • Click on Add New Restaurant.
  • Give it a name, a description and drop an image.
  • Save it.

Create as many restaurants as you would like to see in your apps.

Content Manager

Allow access

Having the items in database is great. Being able to request them from an API is even better. As you already know, Strapi's mission is to create API (I have got a super secret anecdote for you: its name is coming from Boot*strap* your API 😮).

When you were creating your restaurant Content Type, Strapi created, behind the scene, a few set of files located in api/restaurant. These files include the logic to expose a fully customisable CRUD API. The find route is available at http://localhost:1337/restaurants. Try to visit this URL and you will be surprised to be blocked by a 403 forbidden error. This is actually totally normal: Strapi APIs are secured by design.

Don't worry, making this route accessible is actually super intuitive:

Important: do the same thing for the authenticated role.

Now go back to http://localhost:1337/restaurants: at this point, you should be able to see your list of restaurants.

Users Permissions

Enabling GraphQL

By default, API generated with Strapi are best on REST conventions. What if I would tell you that you could transform them into GraphQL within 10 seconds?

Well, let me prove you that.

A GraphQL plugin, which will make all the work for you, is available for Strapi. Install it with the Strapi CLI:

navigate to your backend folder where strapi is installed and install graphql using the strapi install command:

cd backend
strapi install graphql
Enter fullscreen mode Exit fullscreen mode

Or click "Marketplace" on your admin dashboard and select GraphQL

And that's it, you are done installing GraphQL.

Make sure to restart your strapi server if it does not auto restart

Important:

Downgrage graphql-js dependency

With the current implementation you will receive a GraphQL error when trying to query a MongoDB _id field:

This is currently being fixed by the graphql-js maintainers but as of time of writing has not been fixed (github issue linked below).

message: "ID cannot represent value: { _bsontype: "ObjectID", id: }"

picture

This is caused by a breaking change in the graphql-js update on how the ID's from a MongoDB are serialized: https://github.com/graphql/graphql-js/issues/1518.

Based on the comments it looks like the maintainers are working on a fix but as of the time of writing (10/12/2018) the fix is to downgrade your graphql package to: GraphQL 0.13.2.

The dependency package is in the /backend folder under the GraphQL plugin folder listed at: /backend/plugins/graphql/.

cd plugins/graphql
npm install graphql@0.13.2 --save
Enter fullscreen mode Exit fullscreen mode

Inside of your packages.json file your graphql dependency should be listed as "graphql": "0.13.2"

Until the issue is fixed if you upgrade your packages inside the server you will break the implementation.


GraphQL installation

Restart your server, go to http://localhost:1337/graphql and try this query:

{
  restaurants {
    _id
    name
  }
}
Enter fullscreen mode Exit fullscreen mode

Strapi GraphQL

Display restaurants

It looks you are going to the right direction. What if we would display these restaurants in our Next app?

Restaurants list

Install Apollo in the frontend of our application, navigate to the /frontend folder:

cd ..
cd ..
cd ..
cd frontend
yarn add react-apollo next-apollo graphql gql recompose

Enter fullscreen mode Exit fullscreen mode

To connect our application with GraphQL we will use Apollo and the next-apollo implementation to wrap our components in a withData HOC to give them access to make apollo data queries.

There are a couple differnt approaches to implementing GraphQL into a Nextjs app, the approach we will take is extracting the Apollo logic into lib file and wrapping our components with a Higher Order Component called withData to handle the GQL queries inside each respective components.

Example repo detailing the Apollo Next.js implementation:
https://github.com/adamsoffer/next-apollo-example.

Create a lib directory in the root of the project:

mkdir lib
cd lib
touch apollo.js
Enter fullscreen mode Exit fullscreen mode

Path: /frontend/lib/apollo.js

We will generate the list of Restaurants inside a RestaurantList file as:

cd ..
cd components

mkdir RestaurantList
cd RestaurantList
touch index.js
Enter fullscreen mode Exit fullscreen mode

Path: /frontend/components/RestaurantList/index.js

Now update your /pages/index.js home route to display the Restaurant list:

Path: /frontend/pages/index.js

We will need to update our _app.js file to wrap our application with the Apollo Provider that will enable GraphQL to make queries:

Path: /frontend/pages/_app.js

Now you should see the list of restaurants on the page.

Well done!

🍔 In the next section, you will learn how to display the list of dishes: https://dev.to/ryanrez/-cooking-a-deliveroo-clone-with-nextjs-react-graphql-strapi-and-stripe----dishes-list-part-37-5eh/

Top comments (11)

Collapse
 
chuanman2707 profile image
chuanman2707

I don't understand this part


RestaurantList.getInitialProps = async ({ req }) => {
const res = await fetch("https://api.github.com/repos/zeit/next.js");
const json = await res.json();
return { stars: json.stargazers_count };

why you need to fetch query from https://api.github.com/repos/zeit/next.js

Collapse
 
ryanaz profile image
Ryan

This can actually be removed, it was only in code as an example for fetching data in NextJs and has nothing to do with the application. I’ll take it out of the code base to prevent confusion

Collapse
 
hipertrix profile image
Antonio M Ferreira • Edited

Great tutorial!

I had to comment apart to run here, I'm using Postgres and I also changed _id to id.

//pages/index.js

//searchQuery
// const searchQuery = restaurants.filter(query =>
//   query.name.toLowerCase().includes(search)
// );
const searchQuery = restaurants;
Collapse
 
sankar2389 profile image
sankar

When I try this query

{

restaurants {

_id

name

}

}

in localhost:1337/graphql , I still got an error. Even though I downgraded my graphql version to 0.13.2

Collapse
 
ryanaz profile image
Ryan

Could you post the error you are seeing so I can take a look?

Collapse
 
ryanaz profile image
Ryan

I just realized my gists for the actual code didn't transfer through, you may want to take a look now and retry. Let me know if you have an error still and Ill try to help you fix it

Collapse
 
sankar2389 profile image
sankar • Edited

Attached the screenshot ibb.co/pXphg33

Thread Thread
 
ryanaz profile image
Ryan

Thanks for the screenshot, did you choose your DB as MongoDB in the Strapi setup or a different option? If you chose a different option you may need to use id instead of _id, can you try changing it to id and seeing if your able to pull results?

Thread Thread
 
sankar2389 profile image
sankar

Yes, my mistake. I got the output without downgrading graphql version itself. Thanks

Collapse
 
razvancretu profile image
RazvanCretu

I literally followed all your steps and I get this error when I run my frontend:

error - ./node_modules/@apollo/react-hooks/lib/react-hooks.esm.js
Module not found: Can't resolve 'apollo-client' in 'C:\Projects\frontend\node_modules\@apollo\react-hooks\lib'

Could someone help me?

Collapse
 
gameoverwill profile image
Wilfredo Pérez

Awesome, part 2 finished. I can't wait to start part 3.