Skip to content

Latest commit

 

History

History
426 lines (343 loc) · 77.4 KB

File metadata and controls

426 lines (343 loc) · 77.4 KB

Step 5: Mutation and Optimistic Response

So far we learned how to Query and fetch data from our GraphQL server, and in this step we will modify data using GraphQL Mutations.

The mutation we will add is follow and it will add a GitHub users to own following list.

We will first add it with a regular mutation behavior, and then we will update it to use optimistic response.

Implement Mutation

So our schema already has follow mutation declared, and we just need to implement it and call our GitHub connector:

Step 5.1: Implement Follow mutation

Changed server/src/github-connector.js
@@ -21,6 +21,17 @@
 ┊21┊21┊    return this.dataLoader.load(this.paginate(url, page, perPage));
 ┊22┊22┊  }
 ┊23┊23┊
+┊  ┊24┊  follow( login ) {
+┊  ┊25┊    return this.putToGithub(`/user/following/${login}`);
+┊  ┊26┊  }
+┊  ┊27┊
+┊  ┊28┊  putToGithub( relativeUrl ) {
+┊  ┊29┊    const url = `https://api.github.com${relativeUrl}?access_token=${this.accessToken}`;
+┊  ┊30┊
+┊  ┊31┊    const options = { method: 'PUT', headers: { 'Content-Length': 0 } };
+┊  ┊32┊    return fetch(url, options).then(() => this.dataLoader.clearAll());
+┊  ┊33┊  }
+┊  ┊34┊
 ┊24┊35┊  paginate(url, page, perPage) {
 ┊25┊36┊    let transformed = url.indexOf('?') !== -1 ? url : url + '?';
 ┊26┊37┊
Changed server/src/schema.js
@@ -29,6 +29,12 @@
 ┊29┊29┊      return githubConnector.getUserForLogin(user.login);
 ┊30┊30┊    }
 ┊31┊31┊  },
+┊  ┊32┊  Mutation: {
+┊  ┊33┊    follow(_, { login }, { githubConnector }) {
+┊  ┊34┊      return githubConnector.follow(login)
+┊  ┊35┊        .then(() => githubConnector.getUserForLogin(login))
+┊  ┊36┊    },
+┊  ┊37┊  },
 ┊32┊38┊  User: {
 ┊33┊39┊    following(user, { page, perPage }, { githubConnector }) {
 ┊34┊40┊      return githubConnector.getFollowingForLogin(user.login, page, perPage)

Add Angular Form

We will add a form with a simple <input /> tag for the GitHub username, and a simple button that triggers the actual mutation.

So let's start with adding a new Component for the form:

$ ng g component follow-user-form

And let's add it to the main HTML file:

Step 5.3: Use follow form

Changed client/src/app/app.component.html
@@ -5,5 +5,9 @@
 ┊ 5┊ 5┊  </h1>
 ┊ 6┊ 6┊  <img src="" />
 ┊ 7┊ 7┊</div>
+┊  ┊ 8┊
+┊  ┊ 9┊<h2>Follow</h2>
+┊  ┊10┊<app-follow-user-form></app-follow-user-form>
+┊  ┊11┊
 ┊ 8┊12┊<h2>Following:</h2>
 ┊ 9┊13┊<app-follow-list></app-follow-list>

Now we are going to use Angular features that related to forms, so we need to add @angular/forms:

$ yarn add @angular/forms

And import it into the NgModule:

Step 5.4: Added @angular/forms and use FormsModule in our NgModule

Changed client/src/app/app.module.ts
@@ -1,5 +1,6 @@
 ┊1┊1┊import { BrowserModule } from '@angular/platform-browser';
 ┊2┊2┊import { NgModule } from '@angular/core';
+┊ ┊3┊import { FormsModule } from '@angular/forms';
 ┊3┊4┊
 ┊4┊5┊import { AppComponent } from './app.component';
 ┊5┊6┊import { FollowListItemComponent } from './follow-list-item/follow-list-item.component';
@@ -17,7 +18,8 @@
 ┊17┊18┊  ],
 ┊18┊19┊  imports: [
 ┊19┊20┊    BrowserModule,
-┊20┊  ┊    ApolloModule.forRoot(provideClient)
+┊  ┊21┊    ApolloModule.forRoot(provideClient),
+┊  ┊22┊    FormsModule,
 ┊21┊23┊  ],
 ┊22┊24┊  providers: [],
 ┊23┊25┊  bootstrap: [AppComponent]

The implementation of the actual form is simple - it's just an <input /> tag with two-way-binding using ngModel of Angular, and a simple button that triggers an action in click:

Step 5.5: Implement ui logic for FollowUserForm

Changed client/src/app/follow-user-form/follow-user-form.component.html
@@ -1,3 +1,5 @@
-┊1┊ ┊<p>
-┊2┊ ┊  follow-user-form works!
-┊3┊ ┊</p>
+┊ ┊1┊<div>
+┊ ┊2┊  <label>GitHub Username: </label>
+┊ ┊3┊  <input [(ngModel)]="usernameToFollow" placeholder="GitHub Username">
+┊ ┊4┊  <button (click)="follow()">Follow</button>
+┊ ┊5┊</div>🚫↵
Changed client/src/app/follow-user-form/follow-user-form.component.ts
@@ -6,10 +6,14 @@
 ┊ 6┊ 6┊  styleUrls: ['./follow-user-form.component.css']
 ┊ 7┊ 7┊})
 ┊ 8┊ 8┊export class FollowUserFormComponent implements OnInit {
+┊  ┊ 9┊  private usernameToFollow: string = '';
 ┊ 9┊10┊
 ┊10┊11┊  constructor() { }
 ┊11┊12┊
 ┊12┊13┊  ngOnInit() {
 ┊13┊14┊  }
 ┊14┊15┊
+┊  ┊16┊  follow() {
+┊  ┊17┊
+┊  ┊18┊  }
 ┊15┊19┊}

Adding GraphQL Mutation to client-side

Now let's create a GraphQL file for our mutation:

Step 5.6: Added Follow mutation

Added client/src/app/graphql/follow.mutation.ts
@@ -0,0 +1,10 @@
+┊  ┊ 1┊import gql from 'graphql-tag';
+┊  ┊ 2┊
+┊  ┊ 3┊export const FollowMutation = gql`
+┊  ┊ 4┊  mutation follow($login: String!) {
+┊  ┊ 5┊    follow(login: $login) {
+┊  ┊ 6┊      id
+┊  ┊ 7┊      name
+┊  ┊ 8┊      login
+┊  ┊ 9┊    }
+┊  ┊10┊  }`;

We are using GraphQL variable, called $login, and we will later fill this variable with the form data.

Next, we need to implement follow() method using Apollo, so let's add it using Angular dependency injection, and use add to trigger our GraphQL mutation:

Step 5.7: Implement follow mutation

Changed client/src/app/follow-user-form/follow-user-form.component.html
@@ -2,4 +2,7 @@
 ┊2┊2┊  <label>GitHub Username: </label>
 ┊3┊3┊  <input [(ngModel)]="usernameToFollow" placeholder="GitHub Username">
 ┊4┊4┊  <button (click)="follow()">Follow</button>
+┊ ┊5┊</div>
+┊ ┊6┊<div>
+┊ ┊7┊  {{ followResultMessage }}
 ┊5┊8┊</div>🚫↵
Changed client/src/app/follow-user-form/follow-user-form.component.ts
@@ -1,4 +1,6 @@
 ┊1┊1┊import { Component, OnInit } from '@angular/core';
+┊ ┊2┊import { Apollo } from 'apollo-angular';
+┊ ┊3┊import { FollowMutation } from '../graphql/follow.mutation';
 ┊2┊4┊
 ┊3┊5┊@Component({
 ┊4┊6┊  selector: 'app-follow-user-form',
@@ -7,13 +9,29 @@
 ┊ 7┊ 9┊})
 ┊ 8┊10┊export class FollowUserFormComponent implements OnInit {
 ┊ 9┊11┊  private usernameToFollow: string = '';
+┊  ┊12┊  private followResultMessage: string = '';
 ┊10┊13┊
-┊11┊  ┊  constructor() { }
+┊  ┊14┊  constructor(private apollo: Apollo) {
+┊  ┊15┊  }
 ┊12┊16┊
 ┊13┊17┊  ngOnInit() {
 ┊14┊18┊  }
 ┊15┊19┊
 ┊16┊20┊  follow() {
+┊  ┊21┊    if (this.usernameToFollow === '') {
+┊  ┊22┊      return;
+┊  ┊23┊    }
+┊  ┊24┊
+┊  ┊25┊    this.apollo.mutate<any>({
+┊  ┊26┊      mutation: FollowMutation,
+┊  ┊27┊      variables: {
+┊  ┊28┊        login: this.usernameToFollow,
+┊  ┊29┊      },
+┊  ┊30┊    }).subscribe(({ data: { follow } }) => {
+┊  ┊31┊      const { name, login } = follow;
 ┊17┊32┊
+┊  ┊33┊      this.followResultMessage = `You are now following ${login}${name ? ` (${name})` : ''}!`;
+┊  ┊34┊      this.usernameToFollow = '';
+┊  ┊35┊    });
 ┊18┊36┊  }
 ┊19┊37┊}

We also created a class variable called followResultMessage and display it - this will be our temporary feedback for the action's success.

Optimistic Response

At the moment, the user's feedback after adding sending the form is just a message that says that the user is now being followed by you.

We can improve this behavior by adding optimistic response.

Optimistic response is our way to predict the result of the server, and reflect it to the client immediately, and later replace it with the actual response from the server.

This is a powerful feature that allows you to create good UI behavior and great experience.

So our goal is to replace the simple "success" message, and add the new followed user into the following list.

Apollo-client allows you to add optimisticResponse object to your Mutation definition, and we also need implement updateQueries.

updateQueries is a mechanism that allows the developer to "patch" the Apollo-client cache, and update specific GraphQL requests with data - causing every Component that use these Queries to update.

This is how we implemented updateQueries and optimisticResponse in our project:

Step 5.8: Added optimistic response

Changed client/src/app/follow-user-form/follow-user-form.component.html
@@ -2,7 +2,4 @@
 ┊2┊2┊  <label>GitHub Username: </label>
 ┊3┊3┊  <input [(ngModel)]="usernameToFollow" placeholder="GitHub Username">
 ┊4┊4┊  <button (click)="follow()">Follow</button>
-┊5┊ ┊</div>
-┊6┊ ┊<div>
-┊7┊ ┊  {{ followResultMessage }}
 ┊8┊5┊</div>🚫↵
Changed client/src/app/follow-user-form/follow-user-form.component.ts
@@ -1,5 +1,6 @@
 ┊1┊1┊import { Component, OnInit } from '@angular/core';
 ┊2┊2┊import { Apollo } from 'apollo-angular';
+┊ ┊3┊import update from 'immutability-helper';
 ┊3┊4┊import { FollowMutation } from '../graphql/follow.mutation';
 ┊4┊5┊
 ┊5┊6┊@Component({
@@ -9,7 +10,6 @@
 ┊ 9┊10┊})
 ┊10┊11┊export class FollowUserFormComponent implements OnInit {
 ┊11┊12┊  private usernameToFollow: string = '';
-┊12┊  ┊  private followResultMessage: string = '';
 ┊13┊13┊
 ┊14┊14┊  constructor(private apollo: Apollo) {
 ┊15┊15┊  }
@@ -27,10 +27,29 @@
 ┊27┊27┊      variables: {
 ┊28┊28┊        login: this.usernameToFollow,
 ┊29┊29┊      },
-┊30┊  ┊    }).subscribe(({ data: { follow } }) => {
-┊31┊  ┊      const { name, login } = follow;
+┊  ┊30┊      optimisticResponse: {
+┊  ┊31┊        __typename: 'Mutation',
+┊  ┊32┊        follow: {
+┊  ┊33┊          __typename: 'User',
+┊  ┊34┊          id: '',
+┊  ┊35┊          name: '',
+┊  ┊36┊          login: this.usernameToFollow,
+┊  ┊37┊        },
+┊  ┊38┊      },
+┊  ┊39┊      updateQueries: {
+┊  ┊40┊        Me: (prev, { mutationResult }: { mutationResult: any }) => {
+┊  ┊41┊          const result = mutationResult.data.follow;
 ┊32┊42┊
-┊33┊  ┊      this.followResultMessage = `You are now following ${login}${name ? ` (${name})` : ''}!`;
+┊  ┊43┊          return update(prev, {
+┊  ┊44┊            me: {
+┊  ┊45┊              following: {
+┊  ┊46┊                $push: [result]
+┊  ┊47┊              },
+┊  ┊48┊            },
+┊  ┊49┊          });
+┊  ┊50┊        },
+┊  ┊51┊      }
+┊  ┊52┊    }).subscribe(() => {
 ┊34┊53┊      this.usernameToFollow = '';
 ┊35┊54┊    });
 ┊36┊55┊  }

The optimisticResponse object must match and specify the exact GraphQL type that returns from the server requests, in a special field called __typename - this is how GraphQL identify each object.

So in this case, we are returning a Mutation type that contains a fields called follow (this is the mutation itself), that contains a User type with the fields. We don't have all the fields to create a full UI prediction - but we do have the login - to let's use, and let's add name as empty string.

So we know that object returned from the server, we just need to patch the cache data.

The implementation of updateQueries is an Object, there the key is the GraphQL operation name of the Query we want to patch.

The GraphQL operation name is the name that comes after the word query in you Query definition, so in this case we want to patch Me Query, because this is where the following array comes from:

query Me { /// <--- This is the GraphQL operation name
    me {
        id
        following {
            ...
        }
    }
}

Next, the callback of updateQueries will get the current cache state, and the mutation result. This callback actually called twice now - the first time for our optimistic response, and the for actual mutation result.

We take the current state, and patch it using a tool called update from the package immutability-helper, so we are taking the result object of the mutation, and $push it into the existing array of following users.

Don't forget to add immutability-helper by running:

$ yarn add immutability-helper

Next - let's do some minor UI change and display the GitHub login name instead of the GitHub name, when it's not available (because during the time between the optimistic response and the server response, we don't know the name of the user):

Step 5.10: Display login name when name is not available

Changed client/src/app/follow-list-item/follow-list-item.component.html
@@ -1,3 +1,3 @@
 ┊1┊1┊<li class="follow-list-item">
-┊2┊ ┊  {{ user.name }}
+┊ ┊2┊  {{ user.name && user.name !== '' ? user.name : user.login }}
 ┊3┊3┊</li>🚫↵

Let's add another small change to the result handler - and check if the user exists in the list before adding it, so we won't have duplicates:

Step 5.11: Don't add a users that already exists in the list

Changed client/src/app/follow-user-form/follow-user-form.component.ts
@@ -37,9 +37,13 @@
 ┊37┊37┊        },
 ┊38┊38┊      },
 ┊39┊39┊      updateQueries: {
-┊40┊  ┊        Me: (prev, { mutationResult }: { mutationResult: any }) => {
+┊  ┊40┊        Me: (prev: any, { mutationResult }: { mutationResult: any }) => {
 ┊41┊41┊          const result = mutationResult.data.follow;
 ┊42┊42┊
+┊  ┊43┊          if (prev.me.following && prev.me.following.find(followingUser => followingUser.login === result.login)) {
+┊  ┊44┊            return prev;
+┊  ┊45┊          }
+┊  ┊46┊
 ┊43┊47┊          return update(prev, {
 ┊44┊48┊            me: {
 ┊45┊49┊              following: {
< Previous Step Next Step >