在本教程中,您将学习如何使用简单的mutation来修改服务器上的数据并保持客户端上的状态同步。 具体来说,我们将创建一个将项目添加到列表的mutation。
本文是关于GraphQL + React的教程系列的一部分。 以下是其他部分:
Part 3 —Mutations (您正在阅读的这一章)
在本教程中,我们将执行以下操作:
1.将我们的前端连接到服务器 2.在服务器上定义GraphQL变化 3.从客户端上的React组件调用GraphQL变体
更新客户端上的状态以确保它与服务器同步每个步骤都非常简单,因此完成整个教程只需要25分钟。如果你还没有完成第1部分和第2部分,你可以先做或者直接跳到这里,并从GitHub仓库中查看本教程的开始状态:
git clone https://github.com/apollographql/graphql-tutorial.gitcd graphql-tutorialgit fetchgit checkout t3-start
我建议你查看t3-start分支,即使你已经完成了第1和第2部分,因为我们已经对App.css和App.js进行了一些修改,以改善应用的布局和文件夹结构。
要检查是否有效,让我们启动GraphQL服务器:
cd servernpm install && npm start
# ...# GraphQL Server is now running on http://localhost:4000/graphiql
我们还要在一个单独的控制台中启动为我们的前端软件包提供服务的dev服务器:
# assuming that you're in the graphql-tutorial directory
cd clientnpm install && npm start
# ...# The app is running at:## http://localhost:3000/
如果它有效,你已经准备好写下你的第一个 mutation!
1. 将前端连接到服务器
在上一篇教程中,我们构建了我们的服务器,但是我们没有将它连接到我们的前端。 为此,我们只需要在服务器和客户端上进行两个较小的更改。
由于服务器在端口400o上运行,客户端从端口3000上运行,因此我们需要在服务器上启用 CORS。 在本教程中,我们将使用Express / Connect的cors包:
# in the server directory (not the client!!)
npm install --save cors
现在我们只需在server/server.js中导入cors并修改以下行以允许来自我们前端源的跨源请求:
// ...import cors from 'cors';
// ... const server = express();server.use('*', cors({ origin: 'http://localhost:3000' }));
在前端,我们需要从react-apollo导入createNetworkInterface,然后用连接到服务器的mockNetworkInterface替换它:
// client/src/App.js
import { ApolloClient, ApolloProvider, createNetworkInterface, // <-- this line is new!} from 'react-apollo';
您可以删除用于创建mockNetworkInterface(包括导入)的所有代码,并将其替换为以下代码:
const networkInterface = createNetworkInterface({ uri: 'http://localhost:4000/graphql',});
const client = new ApolloClient({ networkInterface,});
您的客户端现在已连接到服务器,您应该看到以下信息 localhost:3000:
2. 在服务器上定义GraphQL mutation
现在客户端连接到服务器,我们可以开始真正的任务 - 编写一个mutation,将channel添加到我们的channel列表中。
首先,我们将通过将其添加到server / src / schema.js中来定义模式中的mutation:
const typeDefs = type Channel { id: ID! # "!" denotes a required field name: String}type Query { channels: [Channel] # "[]" means this is a list of channels}
# The mutation root type, used to define all mutations.type Mutation { # A mutation to add a new channel to the list of channels addChannel(name: String!): Channel};
我们刚刚添加的新Mutation类型定义了一个单一mutation - addChannel - 它只接受一个参数,即新channel的名称。 mutation返回一个channel对象,然后我们可以选择字段,就像查询一样。 以下是一个有效的mutation示例:
# an example mutation call:mutation { addChannel(name: "basketball"){ id name }}
当然,除非我们为它定义一个解析器函数,否则这个mutation不会做任何事情。 我们的解析函数存在于server/src/resolvers.js中,所以让我们来看看并为我们的新addChannel mutation添加一个解析函数。 解析函数必须将提供的名称作为参数,并在将其添加到现有列表之前为新channel生成一个ID。
让我们更改server/src/resolvers.js以添加新的nextId变量并为Mutation.addChanel定义解析器:
const channels = /* ... */let nextId = 3;
export const resolvers = { Query: { channels: () => { return channels; }, }, Mutation: { addChannel: (root, args) => { const newChannel = { id: nextId++, name: args.name }; channels.push(newChannel);return newChannel; }, },};
正如你所看到的,解析器只是推动一个新的channel到channels,递增nextId并返回新创建的channel。 如果你前往 localhost:4000/graphiql, 你应该能够从上面运行示例mutation并获得响应。
如果您在localhost:3000上重新加载客户端应用程序,您现在应该看到“basketball”channel出现。
Note: channels目前仅存储在内存中,因此每次重新启动服务器时,新添加的channel都将消失。 我们将在未来的教程中将事物与持久存储挂钩。
3. 从React组件调用 mutation
现在我们已经验证了mutation在服务器上运行,让我们编写必要的代码从客户端调用它。 首先,我们将在src/components/AddChannel.js中创建一个输入组件:
import React from 'react';
const AddChannel = () => { const handleKeyUp = (evt) => { if (evt.keyCode === 13) { console.log(evt.target.value); evt.target.value = ''; } };
return ( );};
export default AddChannel;
我们的AddChannel组件非常简单。 到目前为止,它只包含一个文本输入元素和一个handleKeyUp函数,该函数将输入文本输出到控制台,并在用户返回时清除输入字段(代码13)。
让我们将其导入到src/components/ChannelsListWithData.js中,并将其放置在我们的channel列表之前:
import AddChannel from './AddChannel';
// ...
const ChannelsList = ({ data: {loading, error, channels }}) => { if (loading) { return Loading ...; } if (error) { return {error.message}; }
return ( // <-- This is the new line. { channels.map( ch => ({ch.name}) )} );};
如果有效,您现在应该在用户界面中看到“新channel”输入:
为了使我们的输入组件调用GraphQL mutation,我们必须将它与来自react-apollo的GraphQL高阶组件(HOC)连接起来 (就像我们在第一篇教程中所做的一样). 对于mutation,graphql HOC传递一个mutate prop,我们将调用它来执行mutation。
import React from 'react';import { gql, graphql } from 'react-apollo';
const AddChannel = ({ mutate }) => { const handleKeyUp = (evt) => { if (evt.keyCode === 13) { evt.persist(); mutate({ variables: { name: evt.target.value } }) .then( res => { evt.target.value = ''; }); } };
return ( );};
const addChannelMutation = gql mutation addChannel($name: String!) { addChannel(name: $name) { id name } };
const AddChannelWithMutation = graphql( addChannelMutation)(AddChannel);
export default AddChannelWithMutation;
让我们看看它是否工作并在输入框中输入内容。 在你进入后文字消失了吗? 如果确实如此,mutation是成功的,您应该在重新加载页面后看到该项目。 当然,不得不重新加载页面并不是一个很好的用户交互,因此在下一节中,我们将了解为什么Apollo无法知道mutation意味着什么,以及我们需要拿新项目做什么来使列表重新呈现。
4. 更新客户端状态使用mutation
我们的channel列表没有自动重新呈现的原因是,Apollo无法知道我们刚刚调用的mutation与呈现我们列表的频道查询有任何关系。 只有服务器知道,但它没有办法通知我们的客户 (为此我们需要一个订阅, 我们将在未来的教程中介绍).
要在突变后更新客户端状态,我们有三种选择:
重新读取选项是迄今为止最简单的一种,它是让应用程序快速运行的好方法,所以我们现在就会这样做。
为了告诉Apollo Client我们想要在我们的mutation完成后重新获取channel,我们通过调用中的refetchQueries选项将它传递给mutate。 我们将从ChannelsListWithData.js中导出并将其导入AddChannel.js中,而不是再次写入频道查询。
// AddChannel.js
// ...import { channelsListQuery } from './ChannelsListWithData';
// ...
mutate({ variables: { name: evt.target.value }, refetchQueries: [ { query: channelsListQuery }], // <-- new })
// ...
现在,如果我们添加一个新频道,列表应该立即刷新!
就是这样; 你已经实现了我们的第一个GraphQL mutation!
当然,这只会在您进行变更后更新UI。 如果mutation是由另一个客户端发起的,那么直到您做出自己的mutation并从服务器重新获取列表,您才会发现。 大多数情况下,这不是问题,但对于实时应用程序,Apollo有一个很好的技巧,可以向程序员几乎毫不费力地透明地向所有客户端传播更新:轮询查询。
为了打开它,只需将src / components / ChannelsListWithData.js中的channelsListQuery传递给pollInterval选项即可:
export default graphql(channelsListQuery, { options: { pollInterval: 5000 },})(ChannelsList);
有了这个简单的变化,Apollo会每5秒重新运行一次查询,并且您的用户界面将更新最新的channel列表。 您可以通过打开一个新的浏览器窗口并在其中添加一个channel来进行测试。 新的channel应该在一小段延迟后出现在另一个窗口中。
恭喜,您已经在GraphQL + React教程的第三步中结束了! 您已经为GraphQL模式添加了一个mutation,为其编写了一个解析器,并从React组件中调用了mutation,并确保通过重新调取和轮询来更新UI。 结合本教程系列的第1和第2部分,您现在已经熟悉了使用Apollo编写完整的React + GraphQL应用程序的所有基本知识。
要了解如何使您的mutation更加高效并且显然更快,请继续下一部分,您将了解良好的的用户界面和存储更新!
Part 4: 具有良好的用户界面和客户端存储更新的GraphQL mutation
如果您喜欢本教程并想继续学习Apollo和GraphQL,请务必点击下面的“关注”按钮,并在Twitter上关注我们 [@apollographql_](https://twitter.com/apollographql) and @helferjs_._