GraphQL is query language, a middle layer that defines the API between the front end and backend. It is a way of querying the data you need in a concise, clean manner.
A web api is a protocol for interfacing between the client and the server. The current most-commonly used protocol today is REST, in which theyre a multiple types of requests such as get, put, post, and delete which contain headers and a body. This allows it to send and recieve the necessary information for the client and server to interface and send eachother the necessary information.
As the application scales though, the API needs to accomodate many requests that all do very specific things. Developers dealt with this with two ways:
-
Having many very specific routes, but makes the code difficult to maintain as the api evolves and the client and servr side both have to stay in sync.
-
Having general endpoints that can be used for multipl purposes, but send and recieve more data than necessary.
Graphql is a protocol that solves this my deisgning a query language that allows you to pprecisely describe what you wanted to do. It also provides a single endpoint in which the clinet can communicate easily what it needs to do.
The schema is a model of the data that describes the structure of the data and the relationships in between them.
For instance, if we wanted to define a student with a name, we would define it as a type.
const schema = gql`
type Teacher {
name: String!
}
`;
The exclamation mark after String
above means that it is nonullable, and the gql
tag in the beginning allows it to be parsed as graphql.
There are three types of interactions: querries, mutations, and subscirptions.
Inside of schema, along with teh type definitions
The schema consists of type definitions and a top level query type for reading the data. This defines the type of querries you can make to get data. For every query listed, there is a corresponding resolver that defines how it should be parsed.
For instance, in the example below there is a query called me which resolves to a teacher type, which has to have a name field. In the resolver, it is set to resolve the Teacher to Aria
.
const schema = gql`
type Query {
me: Teacher
}
type Teacher {
name: String!
}
`;
const resolvers = {
Query: {
me: () => {
return {
name: "Aria"
};
}
}
};
Running Instructions: npm run step1
Example code: step1.js
Query to be executed
{
me {
name
}
}
We can also have querries with arguments. For instance,
const schema = gql`
type Query {
teacher(id: ID!): Teacher
}
type Teacher {
id: ID!
name: String!
}
`;
const resolvers = {
Query: {
me: () => {
return teachers[1];
},
teacher: (parents, {id}) => {
return teachers[id];
},
},
};
In this case, the resolved for teacher can parse arguments passed into the query, which allows you to query a specific id. This allows you to query for specific stdents pased on unique identifiers.
In this case, we are also assuming the "me" is the first teacher in the list, and so we return just that one. As long as what is returned matches the Teacher type, graphql can compile and execute the query.
Running Instructions: npm run step2
Example code: step2.js
Query to be executed
{
teacher(id: "2") {
name
}
me {
name
}
}
By setting the return type as [Teacher!]
we are expeting a list of teacher types where none of the teachefs are null. The resolver takes the list of teachers and returns it in this format.
const schema = gql`
type Query {
teachers: [Teacher!]
}
type Teacher {
id: ID!
name: String!
}
`;
const resolvers = {
Query: {
teachers: () => {
return Object.values(teachers);
},
}
};
Running Instructions: npm run step3
Example code: step3.js
Query to be executed
{
teachers {
id
name
}
}
In the server definition we can give it context, and use that in the querries. In the query resolver, you can access data from four possible places (parent, args, context, info) => { ... }
. We can define the context when we set up the server, and then access values from there.
const server = new ApolloServer({
typeDefs: schema,
resolvers,
context: {
me: teachers[1],
},
});
const resolvers = {
Query: {
me: (parent, args, { me }) => {
return me;
},
},
};
Running Instructions: npm run step4
Example code: step4.js
Query to be executed
{
me {
name
id
}
}
We can write a resolver so that every teacher has an associated Student. We do this by creating a student type in which one of the fields is a Teacher. The resolver currently goes to the student, looks at the teacherId, and gets the correspodning teacher. When you run the example query, it is now able to map to a teacher object with all of it's fields.
const schema = gql`
type Query {
me: Teacher
}
type Teacher {
id: ID!
name: String!
}
type Student {
id: ID!
name: String!
teacher: Teacher!
}
`;
const resolvers = {
Student: {
teacher: student => {
return teachers[student.teacherId];
}
}
};
Running Instructions: npm run step5
Example code: step5.js
Query to be executed
{
student(id: "1") {
id
name
teacher {
id
name
}
}
}
We can also have each teacher have a list of associated students. In the resolver, it can get all the students, and filter out only the ones with a matching teacher id.
const schema = gql`
type Query {
me: Teacher
}
type Teacher {
id: ID!
name: String!
students: [Student!]
}
type Student {
id: ID!
name: String!
teacher: Teacher!
}
`;
const resolvers = {
Teacher: {
students: teacher => {
return Object.values(students).filter(
student => student.teacherId === teacher.id,
);
},
},
};
Running Instructions: npm run step6
Example code: step6.js
Query to be executed
{
teachers {
name
id
students {
name
id
teacher {
id
name
}
}
}
}
Currently we have gone over querries, which allow you to access the data. Mutations, on the other hand, allow you to modify the data. In this example, we are going to write mutations to add and delete students.
const schema = gql`
type Mutation {
addStudent(name: String!): Student!
deleteStudent(id: ID!): Boolean!
}
`;
const resolvers = {
Mutation: {
addStudent: (parent, { name }, { me }) => {
const id = uuid();
const student = {
id,
name,
teacherId: me.id
};
students[id] = student;
return student;
},
deleteStudent: (parent, { id }) => {
const { [id]: student, ...otherStudents } = students;
if (!student) {
return false;
}
students = otherStudents;
return true;
}
}
};
Running Instructions: npm run step7
Example code: step7.js
To query the student
{
me {
name
id
students {
name
id
}
}
}
To create a student
mutation m {
addStudent(name:"Hana") {
name
teacher {
name
}
}
}
To delete a student, but instead change the id to the id of the correct student you want to delete
mutation m {
deleteStudent(id:"3f69a74f-48f8-4a24-9841-e287ee367734")
}