Skip to content

Latest commit

 

History

History
909 lines (599 loc) · 19.7 KB

Lesson-8.md

File metadata and controls

909 lines (599 loc) · 19.7 KB

ACS 4390 - Subscriptions

Review

Quick there's a couple of problems with this code you need to fix them! Remember that Kaiju battle schema? Someone started writing an implementation for it. But there are a few problems can you fix them?

Read the code and try and solve as many problems as you can find. You can do this by eye just reading the code here, or you can run the code and read the error.

const express = require('express')
const { graphqlHTTP } = require('express-graphql')
const { buildSchema } = require('graphql')

// Create a schema
const schema = buildSchema(`
type Kaiju {
	name: String!
	power: Int!
}

type City {
	name: String!
	population: Int!
}

type Battle {
	fighter1: Kaiju!
	fighter2: Kaiju!
	city: City!
}

type Query {
	getKaiju(id: Int!): Kaiju!
	allKaiju: [Kaiju!]!
	getCity(id: Int!): City!
	allCities: [City!]!
	getBattle(fighter1: String!, fighter2: String!, arena: String!): Battle! 
}`)

const fighters = [
	{ name: 'Godzilla', power: 72 },
	{ name: 'Mothra', power: 89 },
	{ name: 'Kong', power: 68 },
	{ name: 'Predator', power: 98 }
]

const cities = [
	{ name: 'Tokyo', population: 37260000 },
	{ name: 'Shanghai', population: 234800000 },
	{ name: 'Såo Paulo', population: 208800000 },
	{ name: 'New York City', population: 18650000 } 
]

// Define a resolver
const root = {
	getKaiju: ({id}) => {
		return fighters[id]
	},
	allKaiju: () => {
		return fighters 
	},
	getCity: ({id}) => {
		return cities[id]
	},
	allCities: () => {
		return cities
	},
	getBattle: ({ fighter1, fighter2 ,arena }) => {
		return { fighter1, fighter2, city }
	}
}
// Create an express app
const app = express()

// Define a route for GraphQL
app.use('/graphql', graphqlHTTP({
	schema,
	rootValue: root,
	graphiql: true
}))
 
// Start this app
const port = 4000
app.listen(port, () => {
	console.log(`Running on port: ${port}`)
})

Learning Objectives

  • Use Apollo Server
  • Create GraphQL Schema
  • Use GraphQL Subscriptions
  • Use advanced features of the Graphiql browser

Subscriptions with GraphQL

Subscriptions are described in the GraphQL docs but the implementation is left up to developers. Usually, these would be implemented with a websocket.

There are several GraphQL libraries to choose from:

  • 🚂 express-graphql
  • 🏆 apollo-server-express
  • 🧘‍♂️ graphql-yoga

This example uses 🏆 Apollo Server. This seems to be the most advanced GraphQL server available at this time.

(I couldn't get subscriptions to work with express-graphql.)

Implementing a server with Apollo is similar to implementing the express-graphql server but with a few differences.

😎 🏆 🤔

Build a GraphQL chat with Subscriptions

Start a new npm project:

npm init -y

Start by importing dependencies. For this project you'll use:

npm install apollo-server@2 graphql

For this project you are using Apollo Server you can read more about it here:

https://www.apollographql.com/docs/

Set up your server by creating:

touch server.js

Add a start script to your package.json:

"scripts": {
	...
	"start": "nodemon server.js"
 },

Setup the server

In server.js start working on your server.

Import your dependencies:

const { ApolloServer, gql, PubSub } = require('apollo-server');

This project will use Subscriptions. To handle events associated with subscriptions you'll use the PubSub class. Make an instance of PubSub

const pubsub = new PubSub();

You'll come back to this later. We need to put some things in place first before we make use of this pubsub instance.

The listening for and publishing subscriptions will be handled with PubSub. Read more about it here:

https://www.apollographql.com/docs/apollo-server/data/subscriptions/#the-pubsub-class

Add a Schema

Add some type definitions. This is where your GraphQL schema will live for this project:

const typeDefs = gql`
	type Post {
		message: String!
		date: String!
	}

	type Query {
		posts: [Post!]!
	}

	type Mutation {
		addPost(message: String!): Post!
	}

	type Subscription {
		newPost: Post!
	}
`

This defines an object type: Post, a Query type: posts, a Mutation type: addPost, and a Subscription type: newPost.

Read more about the gql function here:

https://www.apollographql.com/docs/resources/graphql-glossary/#gql-function

Aside: JS Template Strings

The syntax here looks a little strange:

const typeDefs = gql`
	...
`

Following gql with the backquotes looks weird, what is going on?

tl;dr gql is a function and the JS allows us to follow a function with a template string and omit the parenthesis. read the notes below there is more to this feature!

Why? Using this syntax makes it include a multiline string value as a parameter to the gql function.

Read more here:

https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Template_literals#tagged_templates

Mockup some Data

Define an array of post objects. Each message will have a message and date field. This matches the Post type defined in the schema.

const data = [
	{ message: 'hello world', date: new Date() }
]

You'll expand on this in the challenges later.

Define your resolvers

Now define your resolvers. Apollo resolvers work the same but have a little different structure from the resolvers used with express-graphql.

Let's build the resolvers in stages.

Start with the root level properties:

const resolvers = {
	Query: {
		// Query types
	},
	Mutation: {
		// Mutation types
	},
	Subscription: {
		// Subscription types
	}
}

Notice the root level properties here match the GraphQL query types: Query, Mutation, and Subscription. When you define resolvers for your query types you'll match your schema to this.

For example, the schema shows our posts query:

type Query {
	posts: [Post!]!
}

Query Resolver

You'll define the resolver for this Query:

const resolvers = {
	Query: {
		posts: () => {
			return data
		}
	},
	Mutation: {
		// Mutation types
	},
	Subscription: {
		// Subscription types
	}
}

That was an easy one. Let's try that again...

Mutation Resolver

Add the addPost mutation. This will go under Mutation:

// Resolvers 
const resolvers = {
	Query: {
		...
	},
	Mutation: {
		addPost: (_, { message }) => {
			const post = { message, date: new Date() }
			data.push(post)
			pubsub.publish('NEW_POST', { newPost: post }) // Publish!
			return post
		}
	},
	Subscription: {
		// Subscription types
	}
}
const resolvers = {
	Mutation: {
		addPost: (parent, args, context, info) => {
			...
		}
	}
}

Note! The arguments for a resolver function in Apollo are different! A resolver takes 4 arguments:

  • parent
  • args
  • context
  • info

You'll be looking at the other arguments later, now let's focus on args.

We need the _ in place to get to the second argument, and we'll use the deconstruction syntax to turn the args into variables.

addPost: (_, { message }) { ... }

Here we destructure args to get message.

Note!

pubsub.publish('NEW_POST', { newPost: post })

This line will publish a post to clients who subscribed.

Clients who are subscribed to "NEW_POST" will receive

{ newPost: post }

This has to match the return type of the subscription!

// Schema
type Subscription {
	newPost: Post!
}
const resolvers = {
	Query: {
		user(parent, args, context, info) {
			return users.find(user => user.id === args.id);
		}
	}
}

Read more about Apollo resolvers here:

Read about Appollo Resolvers

Array.find()

Subscription Resolvers

Resolvers for subscriptions work a little differently from other resolvers.

Here is where you will use pubsub.

Note! The PubSub class is used for development. For production, the docs suggest using the PubSubEngine class.

A subscription uses a subscribe method. This method must return an instance of AsyncIterator.

Adding a subscription

Your schema defines newPost subscription, so you'll resolve it with a method at: Subscription.newPost.subscribe

# Schema
type Subscription {
	newPost: Post!
}

So our resolver will be:

const resolvers = {
		...
	Subscription: {
		newPost: {
			subscribe: () => pubsub.asyncIterator('NEW_POST')
		}
	}
}

Update your resolver!

This subscribe method will add a subscription for newPost.

Notice you're calling: pubsub.asyncIterator('NEW_POST') to generate the instance of AsyncIterator. The argument: 'NEW_POST' is a "tag" that will identify this subscription from other subscriptions.

Read about Appollo Subscriptions

Create and Start the Server

Create an instance of ApolloServer and include the typedefs and resolvers.

const server = new ApolloServer({ 
	typeDefs, 
	resolvers 
});

Start the server:

// The `listen` method launches a web server.
server.listen().then(({ url }) => {
	console.log(`🚀 Server ready at ${url}`);
});

Testing your work!

The next step will be to test the server and test subscriptions. You do this with Graphiql browser for now since you haven't created a client application yet.

So far the system supports the following features:

  • Returns a list of posts: posts
    • Each post has a message and date
  • Adds a new post to the list: addPost
  • Subscribes to new posts: newPost
    • This should display a post whenever a post is added

We need to test all of these features now!

Open: http://localhost:4000/graphql? In your browser.

The interface looks a little different but the function is the same.

Aside: Operation names

When you run a query you can give it an "Operation Name". This allows you to identify and describe a query with a name.

# Operation name: Luke
Query GetLuke {
	person(personID: 1) {
		name
	}
}

You must include the query type before the Operation name!

Read more about operation names: https://graphql.org/learn/queries/#operation-name

Test posts Query

Test your posts:

# Operation name: Posts
query Posts {
	posts {
		message
		date
	}
}

Click the ▶️ button. You should see a list of posts.

Test addPost Mutation

Add this new query below the existing query:

# Operation name: AddPost
mutation AddPost {
	addPost(message:"Foo Bar") {
		message
		date
	}
}

Clicking the ▶️ button should show a menu listing your named operations! Choose "AddPost".

Now click ▶️ and choose "Posts". This should list your posts and the new post should be in the list!

Debug any errors you have here at this step.

test newPost Subscription

If you can list your posts and add new posts you can register a subscription for posts. Registering a subscription will display the new post when a new post is created.

To make this work you'll need to dedicate a new tab inside the Graphiql interface. Notice at the top there are tabs. make a new one by clicking ✚.

Pro tip: double click to name your tabs!

In the new tab add a subscription query:

subscription NewPost {
	newPost {
		message
		date
	}
}

Click the ▶️ button. This should run constantly. Notice the run button now looks like a stop button. This tab is now dedicated to watching for subscriptions.

This should register a subscription that will display the message and date of the new post when it is created.

Switch the first tab and run the AddPost operation again. You can change the message for fun.

Switch to the subscription tab and if everything is work you should see the new post!

Review

Review your work. Answer these questions for your self:

  • What are your types?
  • Which resolver handles subscriptions?
  • Which resolver publishes the subscription?

Challenges!

The goal of these challenges is to make a Slack-like server. To do this you need to create channels, post messages to channels, subscribe to a channel, and subscribe to new channels.

Slack would require a user object and authentication. We are going to leave this project to scope it to the core functionality. You could add these features later if you wanted to continue working on this!

Challenge 1 - managing Channels

The first step is to look at your data and ask yourself where you would store channels? Here are a couple of options:

Option 1: Use an object. Imagine your data storage as an object.

{
	channel1: [ {}, {}, ... ],
	channel2: [ {}, {}, ... ]
}

Each property is the channel name and the value is an array of posts. Pros you can be sure that channels are unique! Cons there isn't any room for channel metadata.

Option 2: As above but you use an object for the value of each channel property.

{
	channel1: { name: 'Cats!', posts: [ {}, {}, ... ] },
	channel2: { name: 'Dogs', posts: [ {}, {}, ... ] }
}

Here you could add metadata to each channel. But your structure is more complex.

Option 3: Use an array of objects with a channel name and posts property.

[
	{ channel: 'channel1', posts: [ {}, {}, ... ] },
	{ channel: 'channel2', posts: [ {}, {}, ... ] }
]

This works but you will need to keep the channel names unique.

There are other options. You can think of your ideas. What's important now is to understand your solution since it will affect how you answer the challenges coming up.

Refactor your code to support channels. Include at least one channel in the refactored code!

Challenge 2 - Channels Query

Slack has channels and messages can be posted to a channel. To start you'll need to define a schema for a channel type.

Create a Query type for channels. It should return an array of channel names.

Challenge 3 - addChannel Mutation

Now that we can get a list of the channels it's time to make new channels.

Add a Mutation that creates a new channel.

Challenge 4 - newChannel Subscription

Now that you can make new channels you might want to be notified when a new channel was created.

Add a Subscription for newChannel.

Challenge 5 - Posts need a channel

Now that posts are stored when with a channel you'll need to supply a channel when asking for posts!

The posts Query might look like this now:

posts(channel: String!): [Post!]

Challenge 6 - Add Post to channel

All posts need to be associated with a channel. You need to make sure when a post is created that it is assigned to a channel. How you do this depends on the arrangement of your data.

Here is a suggestion. You probably want to modify the addPost Mutation to look something like:

posts(channel: String!, message: String!): Post!

Challenge 9 - New Posts Subscription

Posts are now added to a channel we probably only want to be notified when a post is added to a channel we are interested in.

Modify the newPost Subscription to support a channel.

newPost(channel: String!): Post

Challenge 8 - Test your work

Build some queries in Graphiql to test your work.

Write a query for each of the following:

  • channels - Display a list of channels
  • addChannel - adds a new channel
  • newChannel - subscribe to new channels
  • Test new channels are added by chowing the list after you add a channel
  • Test a new channel is published via subscription
  • posts - show posts for a channel
  • addPost - adds a new post to a channel
  • Test posts added to the correct channel
  • newPost - Check the subscription is showing new posts.

Final Project

Complete the challenges here. Submit them to GradeScope.

Start working on the final assignment. Choose one:

React + Apollo Tutorial

Following this tutorial, you will be building a Hackernews clone with React using a GraphQL server. This tutorial focuses on the React Front end that connects to the GraphQL backend built using Node. The backend is provided you will building the client.

Do this tutorial if you want to focus on React and client-side projects.

GraphQL Node Tutorial

In this tutorial, you will be building a Hackernews clone using Apollo server. This a Node project and focuses on building the server. You won't be building a client.

Do this project if you want to focus on server-side projects.

Your Custom project

If you have an idea for a project that uses GraphQL you can do this! It can focus on the server-side or the client-side or can be a mix of both.

Your project should include the following:

  • Uses all three Query types: Query, Mutation, and Subscription
  • Uses Apollo Server
  • If your project is client-focused it should use React. Your server can use a simple in-memory data store.
  • If your project is server-focused it should use Apollo Server. Your server should use a database or other persistent storage type.

Stretch Challenge!

Do both tutorials! The GraphQL Node tutorial builds a HackerNews clone server and the React + Apollo tutorial builds a React client for the HackerNews backend. By doing both projects you would be building the entire system!

After Class

You should decide an what you are going to do for the final project

Resources