Skip to content

这篇文章介绍一下如何搭建一个简单的graphql服务,graphql的特性以及如何在react中使用graphql,graphql给项目开发所带来的益处和它所适应的场景不多说了,可以看下infoq上的这篇文章了解下,这里主要介绍下如何搭建和使用。
首先初始化项目react-graphql

bash
mkdir react-graphql
cd react-graphql
npm init

安装基于koa的后端依赖

bash
npm install koa koa-bodyparser koa-router apollo-server-koa graphql graphql-tools

说一下各个依赖包的用处

  • koa 基于nodejs使用中间件模型的网络服务框架
  • koa-bodyparser 解析请求体数据,form,json等格式数据
  • koa-router 用于配置路由规则,将不同路由请求映射到对应的处理逻辑
  • apollo-server-koa apollo是graphql技术发展及各种解决方案的项目,apollo-server-koa 提供了graphql开发调试的图形化工具,并结合koa对请求路由地址和请求数据进行解析
  • graphql graphql的核心,如何处理类型定义和resolve的执行方式就是由它处理的
  • graphql-tools 用于解析schema定义
    graphql-tools这里再多说一下,便于读者理解,如果简单地使用graphql去定义类型和resolve函数也是可行的,它的方式如下
js
import {
  graphql,
  GraphQLSchema,
  GraphQLObjectType,
  GraphQLString
} from 'graphql';

new GraphQLSchema({
  query: new GraphQLObjectType({
    name: 'RootQuery',       // RootQuery 根节点查询类型
    fields: {                    // 查询包含的字段
      hello: {                   // RootQueryType 查询会返回一个 hello 字段
        type: GraphQLString,     // hello 字段是一个String类型
        resolve() {              // RootQueryType 查询的解析器函数
          return 'world';        // 函数返回值为 world 字符串
        }
      }
    }
  })
});

这里就能看出其类型定义方式比较繁琐,并且和代码耦合度较高,graphql-tools 的意义就在于它会解析一种结构定义语言schema,将其解析为上述方式,这样可以将定义文件分离开来,对组织一个项目大有用处,看看用graphql-tools我们来怎么写

js
// scheme定义可以写在独立文件中
const schema = `
  schema {
    query: RootQuery
  }

  type RootQuery {
    hello: String
  }
`;

makeExecutableSchema({
  typeDefs: [
    schema,
  ],
  resolvers: {
    hello() {
        return 'world'
    }
  },
});

现在开始我们的graphql服务搭建,在项目目录中,创建server目录,并创建index.js文件,引入依赖

js
// server/index.js

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

const typeDefs = gql`
    schema {
        query: Query
    }
    
    type Query {
        hello: String
    }
`; // 定义类型

const resolvers = {
    Query: {
        hello() {
            return 'world'
        }
    }
}; // resolve 定义

const app = new Koa();

const server = new ApolloServer({
    typeDefs,
    resolvers,
    playground: true, // 开启开发UI调试工具
});

server.applyMiddleware({ app });

const port = 8000;
app.listen({ port }, () => console.log(`graphql server start at http://localhost:${port}${server.graphqlPath}`));
// 开启服务,server.graphqlPath 默认为 /graphql

package.json中添加启动命令

json
 "scripts": {
   "start": "node server/index.js"
 }

启动服务

bash
npm start

访问服务http://localhost:8000/graphql 可以看到如下UI界面,这是graphql的UI调试工具
Playground---http-localhost-8000-graphql
我们简单的graphql服务就搭建好了,在实际开发中schema和resolve会越来越多并且要进行模块分割,现在我们将schema和resolve提取出来
在server目录下创建schema和reslover文件,并添加index.js文件

js
// schema/index.js

const fs = require('fs');
const path = require('path');
const { gql } = require('apollo-server-koa');

// 包含所有的类型定义
const typeDefs = [];

// 同步读取当前目录下所有 .graphql 文件
const dirs = fs.readdirSync(__dirname);

dirs.forEach((dir) => {
    const filePath = path.join(__dirname, dir);
    if (
        fs.statSync(filePath).isFile && 
        filePath.endsWith('.graphql') // 读取.graphql文件
    ) {

        const content = fs.readFileSync(
            filePath, 
            { 
                encoding: 'utf-8' 
            }
        );

        typeDefs.push(gql`${content}`); // gql字符串模板标签函数会解析schame定义语法
    }

});

// 导出类型定义
module.exports = typeDefs;
js
// resolver/index.js

const fs = require('fs');
const path = require('path');
const _ = require('lodash'); // npm install lodash 添加依赖

// 包含所有的resolve
const resolvers = {};

// 同步读取当前目录下所有 .js 文件
const dirs = fs.readdirSync(__dirname);

dirs.forEach((dir) => {
    const filePath = path.join(__dirname, dir);
    if (
        fs.statSync(filePath).isFile &&
        filePath.endsWith('.js') &&
        !filePath.endsWith('index.js') // 不包含此文件
    ) {

        const resolver = require(filePath);

        _.merge(resolvers, resolver); // 合并所有的resolver到reslovers中
    }

});

// 导出resolvers
module.exports = resolvers;

然后修改server/index.js引入schema和relover

js
// server/index.js

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

// 加载所有的 schema
const typeDefs = require('./schema');

// 加载所有的 resolver
const resolvers = require('./resolver');

const app = new Koa();

const server = new ApolloServer({
    typeDefs,
    resolvers,
    playground: true, // 开启开发UI调试工具
});

server.applyMiddleware({ app });

const port = 8000;
app.listen({ port }, () => console.log(`graphql server start at http://localhost:${port}${server.graphqlPath}`));
// 开启服务,server.graphqlPath 默认为 /graphql

然后我们以用户发帖为例子,定义schema和reslove,并介绍graphql中的一些特性,如结构定义,常用类型,枚举,自定义类型,定义指令,mock数据,自定义请求上下文等是如何使用的
首先我们先实现创建用户,发帖,查看所有帖子这几个功能
在server/schema中创建文件schame.graphql文件,vscode有对应.graphql文件的语法提示插件

graphql
# server/schema/schema.graphql

# #号用于写注释

schema { # schema关键字用于定义schema入口
    query: Query # query是规定的入口字段,这里表示所有的查询都是查询 Query 类型下的字段
}

type Query { # type 关键字用于定义结构
    node: Node # 因为我们要将Post和User分离出去,由于结构定义不允许空结构,Node在这里只是作为占位
}

type Node {
    id: ID # ID 类型,ID类型代表唯一性,值可以是数字或字符串
}

再创建user.graphql和post.graphql

graphql
# server/schema/user.graphql

type User {             # 定义User结构
    id: ID              # ID 类型,原始类型
    name: String        # String 字符串类型,原始类型
    age: Int            # Int 整数类型,原始类型
    available: Boolean  # Boolean 类型,值为 true/false,原始类型
    money: Float        # Float 浮点类型,原始类型
    gender: Gender      # Gender 性别枚举类型
    birthday: Date      # Date 自定义日期类型
}                       # 标量类型可参见 http://spec.graphql.cn/#sec-Scalars-

enum Gender {           # enum 用于定义枚举类型
    FEMALE              # 枚举值,若没有对应的resolve值,将会是字符串
    MALE
}

extend type Query {     # extend 扩展 Query 结构,相当于 type Query { user(id: ID!): User }
    user(id: ID!): User # user表示查询字段,id: ID! 表示查询时的输入值,!表示必须由此输入值,返回 User结构的数据,可以将这种写法当作函数来理解,user函数名,id 输入值,User 返回类型
}
graphql
# server/schema/post.graphql

type Post {
    id: ID!
    title: String
    content: String
    userId: ID
    user: User
}

extend type Query {
    post(id: ID!): Post
}

注意要想查询出自定义结构,就必须将自定义结构与根节点Query结构关联起来,graphql是一层套一层的方式,最终组织成一个大的结构,例如我们现在应用的结构是这样子的

graphql
# 伪代码
schema {
    query: {
        user: User {
            ...
        }
        post: Post {
            ...
            User {
                ...
            }
        }
    }
}

到这里,如果直接运行服务,会报Date类型未找到的错误,因此我们要手动实现自定义类型Date
首先在schema.graphql最后一行声明自定义类型

graphql
# server/schema/schema.graphql

# #号用于写注释

schema { # schema关键字用于定义schema入口
    query: Query # query是规定的入口字段,这里表示所有的查询都是查询 Query 类型下的字段
}

type Query { # type 关键字用于定义结构
    node: Node # 因为我们要将Post和User分离出去,由于结构定义不允许空结构,Node在这里只是作为占位
}

type Node {
    id: ID # ID 类型,ID类型代表唯一性,值可以是数字或字符串
}

scalar Date # 自定义Date类型

创建server/scalar.js文件,并添加如下内容

js
// server/scalar.js

const { GraphQLScalarType } = require('graphql');

const resolvers = {
    Date: new GraphQLScalarType({
        name: 'Date',
        description: '日期类型',
        parseValue(value) {
            // 这个函数用于转换客户端传过来的json值
            // json值是类似这样传过来的 { query: "user(date: $date): User", variables: { $date: "2017-07-30" }}
            return new Date(value);
        },
        parseLiteral(ast) { 
            // 这个函数用于转换客户端传过来的字面量值
            // 字面量值是类似这样传过来的 { query: "user(date: '2017-07-30'): User" }
            return new Date(ast.value);
        },
        serialize(value) {
            // 发送给客户端时将Date类型的值转换成可序列化的字符串值
            return value.toISOString();
        },
    }),
};

module.exports = resolvers;

最后一步,加载我们的自定义类型

js
// server/index.js

...
// 加载所有的 resolver
const resolvers = require('./resolver');

// 加载自定义类型
// 自定义类型,其本质也是个resolve
const scalar = require('./scalar.js');
// 将自定义类型合并到resolver中
_.merge(resolvers, scalar);

const app = new Koa();
...

现在我们的自定义Date类型有了,可以运行服务,但此时还没有定义resolve,因此查询将不会获取到数据,在resolve未实现之前,我们可以先用mock数据来验证我们的查询,apollo项目提供了强大又易用的mock集成,原始类型都已经实现了mock,因此我们只需要实现自定义类型Date的mock,创建server/mock.js文件并添加如下内容

js
// server/mock.js

module.exports = {
    Date() {
        return new Date()
    }
}

将mock引入到index.js文件中

js
// server/index.js

...
_.merge(resolvers, scalar);

// mock 数据
const mocks = require('./mock');

const app = new Koa();

const server = new ApolloServer({
    typeDefs,
    resolvers,
    mocks,            // 配置mock数据
    playground: true, // 开启开发UI调试工具
});

server.applyMiddleware({ app });
...

现在我们可以启动服务,并能查询出mock数据了
23802718-5bslasedf3qrls
自此,我们实现了一个简单的graphql服务,将schema,resolve定义进行分离,实现了自定义类型和mock数据,可以写查询语句来验证我们定义的schema结构了,后续我们会继续完成resolver,自定义指令来实现鉴权,扩展请求上下文加入保存数据的功能,等我们的graphql服务都搭建完成,再来实现react与graphql的集成