graphql tools_声明式GraphQL:编写更少的代码,并使用graphql-tools完成更多工作

graphql tools

I’ve been working with GraphQL for a few months now, but only recently began using Apollo’s graphql-tools library. After learning a few idioms, I am able to mock up a functional API quickly. This is largely due to its low-code, declarative approach to type definitions.

我已经使用GraphQL已有几个月了,但是直到最近才开始使用Apollo的graphql-tools库。 学习了一些习惯用法后,我便能够快速模拟出功能性的API。 这主要是由于其对类型定义的低代码声明性方法。

从他们的例子开始 (Starting with their example)

Apollo has an interactive LaunchPad web site, like the ones covered in my Swagger series. There are several example schemas you can use, and for this article I will use their Post and Authors schema. You can download or fork the code.

阿波罗(Apollo)有一个交互式LaunchPad网站,就像我的Swagger系列文章中介绍的网站一样。 您可以使用几种示例模式,在本文中,我将使用其Post and Authors模式 。 您可以下载或分叉代码。

I will be rearranging the project folders. For this post I’ll download and store it in Github, so I can branch and modify the code through each step. Along the way, I’ll link the branches to this post.

我将重新排列项目文件夹。 对于这篇文章,我将其下载并存储在Github中,因此我可以在每个步骤中分支和修改代码。 一路上,我将分支链接到该帖子。

基础 (The basics)

  • declaring schema types

    声明架构类型

In the Launchpad, you’ll see a typeDefs template literal:

在启动板中,您将看到typeDefs模板文字:

const typeDefs = `type Author {id: Int!firstName: StringlastName: Stringposts: [Post] # the list of Posts by this author}type Post {id: Int!title: Stringauthor: Authorvotes: Int}# the schema allows the following query:type Query {posts: [Post]author(id: Int!): Author}# this schema allows the following mutation:type Mutation {upvotePost (postId: Int!): Post}
`;

There are two entities defined, Author and Post. In addition, there are two “magic” types: Query and Mutation. The Query type defines the root accessors. In this case, there’s an accessor to fetch all Posts, and another to fetch a single Author by ID.

定义了两个实体AuthorPost 。 此外,还有两种“魔术” 类型QueryMutation 。 查询类型定义根accessors 。 在这种情况下,有一个访问器来获取所有Posts ,另一个访问器是通过ID获取单个Author

Note there is no way to directly query for a list of authors or for a single post. It is possible to add such queries later.

请注意,无法直接查询作者列表或单个帖子。 以后可以添加此类查询。

  • declaring resolvers

    宣布解析器

Resolvers provide the necessary logic to support the schema. They are written as a JavaScript object with keys that match the types defined in the schema. The resolver shown below operates against static data, which I’ll cover in a moment.

解析程序提供了支持架构的必要逻辑。 它们被编写为具有与模式中定义的类型相匹配的键JavaScript对象。 下面显示的resolver针对静态数据进行操作,我将在稍后介绍。

const resolvers = {Query: {posts: () => posts,author: (_, { id }) => find(authors, { id: id }),},Mutation: {upvotePost: (_, { postId }) => {const post = find(posts, { id: postId });if (!post) {throw new Error(`Couldn't find post with id ${postId}`);}post.votes += 1;return post;},},Author: {posts: (author) => filter(posts, { authorId: author.id }),},Post: {author: (post) => find(authors, { id: post.authorId }),},
};

To link schema and resolver together, we’ll create an executable schema instance:

要将schemaresolver链接在一起,我们将创建一个可执行架构实例:

export const schema = makeExecutableSchema({typeDefs,resolvers,
});
  • the data source

    数据源

For this simple example, the data comes from two arrays of objects defined as constants: authors and posts:

对于此简单示例,数据来自定义为常量的两个对象数组: authors posts

const authors = [{ id: 1, firstName: 'Tom', lastName: 'Coleman' },{ id: 2, firstName: 'Sashko', lastName: 'Stubailo' },{ id: 3, firstName: 'Mikhail', lastName: 'Novikov' },
];const posts = [{ id: 1, authorId: 1, title: 'Introduction to GraphQL', votes: 2 },{ id: 2, authorId: 2, title: 'Welcome to Meteor', votes: 3 },{ id: 3, authorId: 2, title: 'Advanced GraphQL', votes: 1 },{ id: 4, authorId: 3, title: 'Launchpad is Cool', votes: 7 },
];
  • the server

    服务器

You can serve up the executable schema through graphql_express, apollo_graphql_express, or graphql-server-express. We see that in this example.

您可以通过graphql_expressapollo_graphql_expressgraphql-server-express提供可执行模式 我们在这个例子中看到了。

The important bits are:

重要的位是:

import { graphqlExpress, graphiqlExpress } from 'graphql-server-express';
import { schema, rootValue, context } from './schema';const PORT = 3000;
const server = express();server.use('/graphql', bodyParser.json(), graphqlExpress(request => ({schema,rootValue,context: context(request.headers, process.env),
})));server.use('/graphiql', graphiqlExpress({endpointURL: '/graphql',
}));server.listen(PORT, () => {console.log(`GraphQL Server is now running on 
http://localhost:${PORT}/graphql`);console.log(`View GraphiQL at 
http://localhost:${PORT}/graphiql`);
});

Note that there are two pieces of GraphQL middleware in use:

请注意,有两个正在使用的GraphQL中间件:

  • graphqlExpress

    graphqlExpress

    the GraphQL server that handles queries and responses

    处理查询和响应的GraphQL服务器

  • graphiqlExpress

    graphiqlExpress

    the interactive GraphQL web service that allows interactive queries through an HTML UI

    交互式GraphQL Web服务,该服务允许通过HTML UI进行交互式查询

改组 (Reorganizing)

For large apps, we suggest splitting your GraphQL server code into 4 components: Schema, Resolvers, Models, and Connectors, which each handle a specific part of the work. (http://dev.apollodata.com/tools/graphql-tools/)

对于大型应用程序,我们建议将GraphQL服务器代码分成4个组件:架构,解析器,模型和连接器,它们分别处理工作的特定部分。 ( http://dev.apollodata.com/tools/graphql-tools/ )

Putting each type of component in its own file makes sense. I’ll go one better and put each set of components in a its own “domain” folder.

将每种类型的组件放在其自己的文件中是有意义的。 我会做得更好,并将每组组件放在一个自己的“域”文件夹中。

为什么要域名? (Why domains?)

Domains are a convenient way to split up a large system into areas of operation. Within each domain there may be subdomains. In general, subdomains have a bounded context. Within a bounded context the entity names, properties, and processes have precise meaning.

域是将大型系统划分为多个操作区域的便捷方法。 在每个域中可能有子域。 通常,子域具有有限的上下文。 在有限的上下文中,实体名称,属性和过程具有精确的含义。

I find bounded contexts to be helpful during analysis, especially when talking to domain experts.

我发现有限的上下文在分析过程中会有所帮助,特别是在与领域专家交谈时。

The fly in the ointment is that GraphQL types occupy a single namespace, so naming conflicts can exist. More on that later.

美中不足的是,GraphQL类型仅占用一个名称空间,因此可能存在命名冲突。 以后再说。

I’ll call this domain authorposts, and put the related components in the authorposts folder. Within that, I’ll create a file each for datasource, resolvers, and schema. Let’s also toss in an index.js file to simplify importing. The original schema and server files will remain in the root folder, but the schema.js code will be skeletal. The find and filter methods imported from lodash will be removed in favor of synonymous native ES6 methods. The resulting source is here.

我将这个域称为authorposts ,并将相关组件放入authorposts folder authorposts folder authorposts folder 。 在其中,我将分别为datasourceresolvers和schema创建一个文件。 让我们也将index.js文件折腾以简化导入。 原始模式和服务器文件将保留在根文件夹中,但是schema.js代码将是骨架的。 find lodash导入的filter方法将被删除,以支持同义的本机ES6方法。 结果来源在这里 。

The main schema file has become simpler. It provides skeletal structure for further extension by schemas in our domains.

主模式文件变得更加简单。 它为我们的领域中的架构提供了进一步扩展的骨架结构。

A domain schema is imported on lines 7–8, and the base schema on lines 11–23. You’ll note there is a domain property. This is arbitrary but GraphQL, or graphql-tools, insists that one property be defined.

domain模式在第7–8行导入, base模式在第11–23行导入。 您会注意到有一个属性。 这是任意的,但是GraphQL或graphql-tools坚持要定义一个属性。

The complete schema is constructed on line 26, and an executableSchema instance is created given the schema and resolvers defined so far on lines 28–33. This is what is imported by the server.js code, which is largely unchanged from the original.

完整的架构在第26行上构建,并根据第28–33行到目前为止定义的schemaresolvers创建了一个executableSchema实例。 这就是server.js代码导入的内容,与原始代码基本没有变化。

There is a trick to splitting up a schema this way. Let’s take a look:

有这样一种技巧可以拆分模式。 让我们来看看:

The first listing, authorpostResolvers.js, is pretty much a cut’n’paste job from the original schema.js source from Apollo’s example. Yet in the authorpostSchema.js code, we extend the Query and Mutator definitions that are declared in the the base schema. If you don’t use the extend keyword, the executable schema builder will complain about two Query definitions.

第一个清单authorpostResolvers.js是Apollo示例中原始schema.js源代码中的一个“剪切”粘贴工作。 但是,在authorpostSchema.js代码中,我们扩展了在基础架构中声明的QueryMutator定义。 如果不使用extend关键字,则可执行模式构建器将抱怨两个查询定义。

继续… (Continuing…)

This is a good start for organizing several schemas, one for each domain of interest (so long as you're mindful of the global namespace for types), but a complete schema, even for a single domain, can get huge. Fortunately, you can break down each schema even further, right down to the entity level, if necessary.

这是组织多个模式的一个良好的开始,一个模式用于每个感兴趣的域(只要您注意类型的全局名称空间),但是即使是单个域,一个完整的模式也会变得庞大。 幸运的是,您可以根据需要甚至进一步细分每个架构,直至实体级别 。

Here’s a modified directory structure, and listings of the new contents:

这是修改后的目录结构,并列出了新内容:

We can achieve granularity by defining two component files, then importing them into a domain schema.

我们可以通过定义两个组件文件,然后将它们导入域模式来实现粒度。

You don’t have to do one component per file. But you do want to be sure that the schema exports those components along with the schema itself as shown on line 20 of schema.js. Otherwise you’ll likely wind up missing a dependency further down the inclusion chain.

您不必为每个文件做一个组件。 但是您确实要确保该模式将这些组件与模式本身一起导出,如schema.js的第20行所示。 否则,您很可能最终会在包含链的下方错过一个依赖项。

多个架构和解析器 (Multiple schemas and resolvers)

Adding a new schema for a new domain is simple. Create a new domain folder and add dataSource, resolvers, schema, and index.js files. You can also add an optional component folder with component type definitions.

为新域添加新架构很简单。 创建一个新的域文件夹,并添加dataSource,解析器,架构和index.js文件。 您还可以添加带有组件类型定义的可选组件文件夹。

Finally, the root schema.js file must combine the schemas and resolvers from both domains:

最后,根schema.js文件必须结合两个域中的模式和解析器:

//...
import {schema as myLittleTypoSchema,resolvers as myLittleTypeResolvers
} from './myLittleDomain';import {merge
} from 'lodash';
//...
const schema = [...baseSchema, ...authorpostsSchema, ...myLittleTypoSchema]const options = {typeDefs: schema,resolvers: merge(authorpostsResolvers, myLittleTypeResolvers)
}

Note that I had to include lodash merge here because of the need for a deep merge of the two resolvers imports.

请注意,由于必须深度合并两个解析器 ,因此我必须在此处包括lodash 合并 进口。

处理命名空间冲突 (Dealing with Namespace Collisions)

If you are on a large project, you will encounter type name collisions. You might think that Account in one domain would mean the same as Account in another. Yet even if they do mean more or less similar things, chances are the properties and relationships will be different. So technically they are not the same type.

如果您在大型项目中,则会遇到类型名称冲突。 您可能会认为一个域中的帐户与另一个域中的帐户含义相同。 然而,即使它们确实或多或少地意味着相似的事物,属性和关系也有可能会不同。 因此,从技术上讲,它们不是同一类型。

At the time of this writing, GraphQL uses a single namespace for types.

在撰写本文时,GraphQL对类型使用单个名称空间。

How to work around this? Facebook apparently uses a naming convention for their 10,000 types. As awkward as that seems, it works for them.

如何解决这个问题? Facebook显然为其10,000种类型使用命名约定 。 看起来很尴尬,但对他们有用。

The Apollo graphql-tools stack appears to catch type name duplications. So you should be good there.

Apollo graphql-tools堆栈似乎捕获类型名称重复项。 所以你在那里应该很好。

There is an ongoing discussion on whether to include namespaces in GraphQL. It isn’t a simple decision . I remember the complexities caused by the introduction of XML Namespaces 10 years ago.

关于是否在GraphQL中包括名称空间的讨论正在进行中。 这不是一个简单的决定。 我记得10年前引入XML命名空间引起的复杂性。

然后去哪儿? (Where to go from here?)

This post only scratches the surface of how one might organize a large set of GraphQL schemas. The next post will be about mocking GraphQL resolvers, and how it’s possible to mix both real and mocked values in query responses.

这篇文章只是对如何组织一大套GraphQL模式的表述。 下一篇文章将关于模拟GraphQL解析器,以及如何在查询响应中混合使用真实值和模拟值。

翻译自: https://www.freecodecamp.org/news/declarative-graphql-with-graphql-tools-cd1645f94fc/

graphql tools

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/395290.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

用idea搭建SSM项目,原来这么简单

目录 前言软件环境创建项目数据库文件配置文件pom.xmllog4j.propertiesjdbc.propertiesapplicationContext.xmlspring-mvc.xmlweb.xml运行项目编写代码前言 前阵子突发奇想,想学习下SpringMVC的源码,于是打算用idea搭建一个简易的maven版SSM案例&#xf…

浙江理工大学计算机组成原理试卷,浙江理工大学计算机组成原理设计试题.doc...

计算机组成原理课程设计报告2013/2014第二学期指导教师:许建龙 张芳班级:12计科2班姓名:学号: 计算机组成原理大型实验任务书(计算机12级1、2、3班和实验班)实验目的:深入了解计算机各种指令的执行过程,以及…

mac vagrant 虚拟机nfs挂载点

需求:在mac 上安装了虚拟机,虚拟机系统为centos6.5,现在希望讲虚拟机上点目录通过nfs共享给mac使用 下面主要描述通过nfs共享目录给mac调用的过程 过程参考网址: http://www.moqifei.com/archives/1534 (我主要参考的这…

nodejs中require的路径是一个文件夹时发生了什么

node中使用require的时候如果路径是一个文件夹时,或者特殊的情况require(..);require(.); 这是node实战这本书里说的情况,但是我在node6.9版本中发现不完全是这样,可能是后来做了修改。下面说说在6.9下require路径的过程。 这里以require(.)说…

python调用ctypes中windll中的方法超时处理_python中使用ctypes调用MinGW生成的动态链接库(dll)...

关于gcc编译dll的我就不说了,网上举例一大堆,下面以g为例。假设有一个test.cpp文件如下:extern "C" {__declspec(dllexport) double add(double x,double y);}double add(double x,double y){return xy;}在MinGW中使用g编译&#x…

惯用过程模型_惯用的Ruby:编写漂亮的代码

惯用过程模型Ruby is a beautiful programming language.Ruby是一种美丽的编程语言。 According to Ruby’s official web page, Ruby is a:根据Ruby的官方网页,Ruby是: “dynamic, open source programming language with a focus on simplicity and …

采用晶体管为基本元件的计算机发展阶段是,计算机应用基础知识点

第一章 计算机基础知识1、计算机发展阶段第一代:电子管计算机采用电子管为基本元件,设计使用机器语言或汇编语言。要用于科学和工程计算 第二代:晶体管计算机采用晶体管为基本元件,程序设计采用高级语言,出现了操作系统…

springcloud系列三 搭建服务模块

搭建服务模块为了模拟正式开发环境,只是少写了service层直接在controller里面直接引用,直接上图和代码:更为方便: 创建完成之后加入配置: pom.xml文件: <?xml version"1.0" encoding"UTF-8"?> <project xmlns"http://maven.apache.org/POM…

P1801 黑匣子_NOI导刊2010提高(06)

题目描述 Black Box是一种原始的数据库。它可以储存一个整数数组&#xff0c;还有一个特别的变量i。最开始的时候Black Box是空的&#xff0e;而i等于0。这个Black Box要处理一串命令。 命令只有两种&#xff1a; ADD(x):把x元素放进BlackBox; GET:i加1&#xff0c;然后输出Bla…

MySql模糊查询

常规like的使用限制&#xff1a; 1. like %keyword &#xff1a;索引失效&#xff0c;使用全表扫描。但可以通过翻转函数like前模糊查询建立翻转函数索引走翻转函数索引&#xff0c;不走全表扫描。 2. like keyword% &#xff1a;索引有效。 3. like %keyword% &#xff1a;索引…

python psycopg2使用_python 操作数据库:psycopg2的使用

1 conn psycopg2.connect(database"testdb", user"postgres",password"cohondob", host"127.0.0.1", port"5432")这个API打开一个连接到PostgreSQL数据库。如果成功打开数据库时&#xff0c;它返回一个连接对象。2cursor c…

软件测试人员棘手的问题,Èí¼þ²âÊԵļ¬ÊÖÎÊÌ⣺ÈçºÎ±ÜÃâÖظ´ÌύȱÏÝ...

¡¡¡¡£££©£¡££¡££££©©£¡¡¡¡¡BUG£££¢¡£££¡££¡£¡£——£…

机器学习实用指南_机器学习方法:实用指南

机器学习实用指南by Karlijn Willems通过Karlijn Willems 机器学习方法&#xff1a;实用指南 (How Machines Learn: A Practical Guide) You may have heard about machine learning from interesting applications like spam filtering, optical character recognition, and …

本地仓库settings.xml中使用阿里的仓库

背景 当前使用eclipse自带的maven碰到两个蛋疼的问题&#xff1a; maven在国内使用如果不进行FQ则会痛苦不堪如便秘。maven下载大量jar包导致某盘不够用&#xff0c;需要换大的分区。因此为了解决这个问题就介绍两个eclipse配置&#xff1a;maven本地路径配置和maven外部路径配…

day6_python之md5加密

#md5是不可逆的&#xff0c;就是没有办法解密的 Python内置哈希库对字符串进行MD5加密的方法-hashlibimport hashlib def my_md5(s,salt): #用函数&#xff0c;为了提高代码的复用率s ssalt #1.必须是字符串news str(s).encode() #2.字符串需要encode编码后&#xff0…

异步服务_微服务全链路异步化实践

1. 背景随着公司业务的发展&#xff0c;核心服务流量越来越大&#xff0c;使用到的资源也越来越多。在微服务架构体系中&#xff0c;大部分的业务是基于Java 语言实现的&#xff0c;受限于Java 的线程实现&#xff0c;一个Java 线程映射到一个kernel 线程&#xff0c;造成了高并…

win7打开计算机死机,怎么样解决Win7系统运行程序引起的死机问题

Win7系统不仅需要使用到电脑中自带的一些程序&#xff0c;同时&#xff0c;也需要在win7旗舰版电脑中有选择的自己去安装一些程序。但是经常有用户会碰到Win7电脑突然跳出运行程序未响应&#xff0c;出现电脑死机的情况&#xff0c;特别是开的浏览器窗口多的时候更是死机的频繁…

(poj)1064 Cable master 二分+精度

题目链接&#xff1a;http://poj.org/problem?id1064 DescriptionInhabitants of the Wonderland have decided to hold a regional programming contest. The Judging Committee has volunteered and has promised to organize the most honest contest ever. It was decided…

PHP中如何解决高并发

PHP中如何解决高并发 1&#xff1a;硬件方面 普通的一个p4的服务器每天最多能支持大约10万左右的IP&#xff0c;如果访问量超过10W那么需要专用的服务器才能解决&#xff0c;如果硬件不给力 软件怎么优化都是于事无补的。主要影响服务器的速度 有&#xff1a;网络-硬盘读写速度…

es6 迭代器_揭秘ES6迭代器和迭代器

es6 迭代器by Tiago Lopes Ferreira由Tiago Lopes Ferreira 揭秘ES6迭代器和迭代器 (Demystifying ES6 Iterables & Iterators) ES6 introduces a new way to interact with JavaScript data structures — iteration. Let’s demystify it.ES6引入了一种与JavaScript数据…