Most of the APIs request an endpoint to access a predefined data structure. If you want to access other resources, it is necessary to request another endpoint, it makes the process kind of tricky.
- We only define a single endpoint (for example http://example/graphql).
- Since this is a query language, all actions are done through a POST.
Solution
GraphQL allow us to retrieve only the data we need by using a query language for web APIs.
Recipe
CRUD GraphQL API with Nodejs, Express & MongoDB
- Create a new directory to store the project, run npm init to configure the new project
- Run npm install to create our package.json file
- Create a server.js file (entry point for our server)
-
Create the src folder and the below required folders and files:
- Create src/schema directory and an index.js file (will contain the business logic)
- Create a src/resolvers directory and an index.js file.
- Create an src/models directory and a post.js which holds what a post should look like.
├── src │ ├── schema │ │ └── index.js │ ├── resolvers │ │ └── index.js │ └── models │ └── post.js ├── package-lock.json ├── package.json └── server.js
-
Install the dependencies
Using npm
# npm $ npm install --save express express-graphql graphql body-parser
Using yarn
# yarn $ yarn add --save express express-graphql graphql body-parser
You can also install "nodemon" locally to avoid having to restart your server with each change
$npm install --save-dev nodemon
Also you need to update your "package.json" for using "nodemon".
"scripts": { "start": "nodemon server.js"}
-
Edit the schema, it will allow us to define our GraphQL objects and list the different actions available through our API.
// ./src/schema/index.js const { buildSchema } = require('graphql'); module.exports = buildSchema(` """ A Post refers to available attributes for a Post """ type Post { _id: ID! body: String! createdAt: String! } input PostType { body: String! } type RootQuery { posts: [Post!] post(_id: String!): Post! } type Mutation { createPost(post:PostType): Post, deletePost(_id: String): Post, updatePost(_id: String, body: String): String } schema { query: RootQuery mutation: Mutation } `);
We set up two queries, one to fetch all posts, and the other one to retrieve a post by id.
To be able to retrieve posts, we must first have them in our database. That implies we’ll have to create them. We achieve this with the help of a GraphQl mutation. It’s a query that (creates/updates/deletes).
- The "createPost" mutation is used to make a new post in this case. It takes an item of type PostInput and creates a post from it.
- The "deletePost" mutation to delete a post by id.
- The "updatePost" mutation is made to update a post, It takes two params _id and the new body.
-
Create Moongoose Model, install the mongoose package:
npm install mongoose
Then edit the file "src/model/post.js" in order to add the model.
//./src/models/post.js const mongoose = require("mongoose") const Schema = mongoose.Schema const postSchema = new Schema( { body: { type: String, required: true, }, }, { timestamps: true } ) module.exports = mongoose.model("Post", postSchema)
-
Edit the resolver file, it is a set of functions that generates a GraphQL query response. It’s a GraphQL query handler, the name of every query or mutation must exactly match the name of the resolver function.
Add this code below in the "src/resolvers/index.js"
//./src/resolvers/index.js const Post = require("../models/post") module.exports = { posts: async () => { try { const postsFetched = await Post.find() return postsFetched.map(post => { return { ...post._doc, _id: post.id, createdAt: new Date(post._doc.createdAt).toISOString(), } }) } catch (error) { throw error } }, post: async (_id) => { try { const postFetched = await Post.findById(_id); return { ...postFetched._doc, _id: postFetched.id, createdAt: new Date(postFetched._doc.createdAt).toISOString(), } } catch (error) { throw error } }, createPost: async args => { try { const { body } = args.post const post = new Post({ body, }) const newPost= await post.save() return { ...newPost._doc, _id: newPost.id } } catch (error) { throw error } }, deletePost: async (id) => { try { const deletedPost = await Post.findByIdAndDelete(id); return { ...deletedPost._doc, _id: deletedPost.id, createdAt: new Date(deletedPost._doc.createdAt).toISOString(), } } catch (error) { throw error } }, updatePost: async args => { try { const { _id, body } = args const updatedPost = await Post.findByIdAndUpdate(_id, { body: body }); return `Post ${updatedPost.id} updated Successfully!!!` } catch (error) { throw error } }, }
Based on the "src/shcema/index.js", we created a query named posts that returns an array of posts. The posts method sends the request to MongoDB using the model created with mongoose.
The mutation described before in our Schema will be handled by the second resolver function createPost. It takes the post object as an input and uses it to create a new post based on the Post model.
And to save it in MongoDB, we just need to utilize another mongoose helper, the save() function, and return the newly generated post as anticipated in the Schema.We now have a schema and resolvers for our API, which is almost everything we need to go on to the next phase and construct a server and endpoint.
-
Create MongoDB
Go to https://cloud.mongodb.com/ and create you own cluster, as recommendation download https://www.mongodb.com/es/products/compass to connect to the DB.
Save the "Connection String" from your MongoDB.
-
Create a new file nodemon.json in the root of your project folder, that file will be used to store our environment variables. In case that you are not using nodemon , create an .env file
{ "env": { "MONGO_DB": "your_Connection String" } }
-
Create the Server & connect MongoDB
Add this to "server.js" file
const express = require("express"); const { graphqlHTTP } = require("express-graphql"); const graphQlSchema = require("./src/schema"); const graphQlResolvers = require("./src/resolvers"); const mongoose = require("mongoose"); const app = express(); app.use( "/graphql", graphqlHTTP({ schema: graphQlSchema, rootValue: graphQlResolvers, graphiql: true, }) ); const uri = `${process.env.MONGO_DB}?retryWrites=true&w=majority`; const options = { useNewUrlParser: true, useUnifiedTopology: true }; mongoose .connect(uri, options) .then(() => app.listen(4000, console.log("Server is listening on 4000"))) .catch((error) => { throw error; });
Imported the schema and resolvers that were previously created. graphqlHTTPis needed to use them. It’s a express-graphql method that expects some options, in this case, are the schema and the resolvers. I also enabled graphiql, a useful tool for query testing.
The last part of the code is to establish the connection to MongoDB
-
Run the project, use "npm start" at the console, Then open at your browser: "http://localhost:4000/graphql".
We can now try our queries directly from the interface.
You can run this code to create the first post:
mutation { createPost(post: {body:"My first post"}){ body, createdAt } }
Feel free to visit this repository containing all what this blog contains: https://github.com/ICCHA-Technologies/GraphQL-api
Top comments (0)