GraphQL:拼接Stitching
GraphQL 既是一种用于 API 的查询语言也是一个满足你数据查询的运行时。GraphQL 对你的 API 中的数据提供了一套易于理解的完整描述,使得客户端能够准确地获得它需要的数据,而且没有任何冗余,也让 API 更容易地随着时间推移而演进,还能用于构建强大的开发者工具。
——出自 https://graphql.cn
数据是有关系的,可以利用HotChocolate.Stitching功能,把业务逻辑相关联的实体数据,从后台的两个服务中关联起来,这有点api网关的组合功能,可以把前端的一次请求,分解成后端的多次请求,并把数据组合后返回回去。
下面有这样一个例子:有个学生服务,有两个api,一个是查询全部学生,一个是按学号查询学生;另一个是成绩服务,有两个api,一个是按学号查询这个学生的全部成绩,一个是按id查询成绩。现在可以在学生实体中组合成绩,也可以在成绩实体中组合学生,这是因为学生和成绩是一个一对多的关系。
下面来看实例:
学生服务:GraphQLDemo03_Students
引入NuGet
HotChocolate.AspNetCore
HotChocolate.Data
Starup.cs
using Microsoft.AspNetCore.Builder;
using Microsoft.AspNetCore.Hosting;
using Microsoft.Extensions.DependencyInjection;
using Microsoft.Extensions.Hosting;
namespace GraphQLDemo03_Students
{
public class Startup
{
public void ConfigureServices(IServiceCollection services)
{
services
.AddSingleton<IStudentRepository, StudentRepository>()
.AddGraphQLServer()
.AddQueryType<Query>()
;
}
public void Configure(IApplicationBuilder app, IWebHostEnvironment env)
{
if (env.IsDevelopment())
{
app.UseDeveloperExceptionPage();
}
app.UseRouting();
app.UseEndpoints(endpoints =>
{
endpoints.MapGraphQL();
});
}
}
}
Query.cs
using HotChocolate;
using System.Collections.Generic;
namespace GraphQLDemo03_Students
{
public class Query
{
public IEnumerable<Student> GetStudents([Service] IStudentRepository studentRepository)
{
return studentRepository.GetStudents();
}
public Student GetStudent(string stuNo, [Service] IStudentRepository studentRepository)
{
return studentRepository.GetStudent(stuNo);
}
}
}
StudentRepository.cs
using System.Collections.Generic;
using System.Linq;
namespace GraphQLDemo03_Students
{
public interface IStudentRepository
{
IEnumerable<Student> GetStudents();
Student GetStudent(string stuNo);
}
public class StudentRepository : IStudentRepository
{
public IEnumerable<Student> GetStudents()
{
var students = new List<Student>() {
new Student("S0001","小张",20,true),
new Student("S0002","小李",19,false),
};
return students;
}
public Student GetStudent(string stuNo)
{
var students = new List<Student>() {
new Student("S0001","小张",20,true),
new Student("S0002","小李",19,false),
};
return students.SingleOrDefault(s => s.StuNo == stuNo);
}
}
public record Student(string StuNo, string Name, int Age, bool Sex);
}
成绩服务:GraphQLDemo03_Grades
引入NuGet包
HotChocolate.AspNetCore
HotChocolate.Data
Startup.cs
using Microsoft.AspNetCore.Builder;
using Microsoft.AspNetCore.Hosting;
using Microsoft.Extensions.DependencyInjection;
using Microsoft.Extensions.Hosting;
namespace GraphQLDemo03_Grades
{
public class Startup
{
public void ConfigureServices(IServiceCollection services)
{
services
.AddSingleton<IGradeRepository, GradeRepository>()
.AddGraphQLServer()
.AddQueryType<Query>()
;
}
public void Configure(IApplicationBuilder app, IWebHostEnvironment env)
{
if (env.IsDevelopment())
{
app.UseDeveloperExceptionPage();
}
app.UseRouting();
app.UseEndpoints(endpoints =>
{
endpoints.MapGraphQL();
});
}
}
}
Query.cs
using HotChocolate;
using System.Collections.Generic;
namespace GraphQLDemo03_Grades
{
public class Query
{
public IEnumerable<Grade> GetGrades([Service] IGradeRepository gradeRepository, string stuNo)
{
return gradeRepository.GetGrades(stuNo);
}
public Grade GetGrade([Service] IGradeRepository gradeRepository, int id)
{
return gradeRepository.GetGrade(id);
}
}
}
GradeRepository.cs
using System.Collections.Generic;
using System.Linq;
namespace GraphQLDemo03_Grades
{
public interface IGradeRepository
{
IEnumerable<Grade> GetGrades(string stuNo);
Grade GetGrade(int id);
}
public class GradeRepository : IGradeRepository
{
public IEnumerable<Grade> GetGrades(string stuNo)
{
var grades = new List<Grade>(){
new Grade(1,"S0001",100,"语文"),
new Grade(2,"S0001",99,"数学"),
new Grade(3,"S0002",98,"语文"),
new Grade(4,"S0002",97,"数学")
};
return grades.Where(s => s.stuNo == stuNo);
}
public Grade GetGrade(int id)
{
var grades = new List<Grade>(){
new Grade(1,"S0001",100,"语文"),
new Grade(2,"S0001",99,"数学"),
new Grade(3,"S0002",98,"语文"),
new Grade(4,"S0002",97,"数学")
};
return grades.SingleOrDefault(s => s.ID == id);
}
}
public record Grade(int ID, string stuNo, float score, string subject);
}
最重要的是组合服务,即网关服务:GraphoQLDemo03_gateway
引入NuGet包
HotChocolate.AspNetCore
HotChocolate.Data
HotChocolate.Stitching
using Microsoft.AspNetCore.Builder;
using Microsoft.AspNetCore.Hosting;
using Microsoft.Extensions.DependencyInjection;
using Microsoft.Extensions.Hosting;
using System;
namespace GraphQLDemo03_gateway
{
public class Startup
{
const string Students = "students";
const string Grades = "grades";
public void ConfigureServices(IServiceCollection services)
{
services.AddHttpClient(Students, c => c.BaseAddress = new Uri("http://localhost:7000/graphql"));
services.AddHttpClient(Grades, c => c.BaseAddress = new Uri("http://localhost:9000/graphql"));
services
.AddGraphQLServer()
.AddRemoteSchema(Students, ignoreRootTypes: true)
.AddRemoteSchema(Grades, ignoreRootTypes: true)
.AddTypeExtensionsFromString("type Query { }")
.AddTypeExtensionsFromFile("StudentStitching.graphql")
.AddTypeExtensionsFromFile("GradeStitching.graphql")
.AddTypeExtensionsFromFile("StudentExtendStitching.graphql")
.AddTypeExtensionsFromFile("GradeExtendStitching.graphql")
;
}
public void Configure(IApplicationBuilder app, IWebHostEnvironment env)
{
if (env.IsDevelopment())
{
app.UseDeveloperExceptionPage();
}
app.UseRouting();
app.UseEndpoints(endpoints =>
{
endpoints.MapGraphQL();
});
}
}
}
学生的Query类型
extend type Query{
getonestudent(stuNo:String): Student @delegate(schema: "students", path:"student(stuNo:$arguments:stuNo)")
getstudents: [Student] @delegate(schema: "students", path:"students")
}
学生Student上扩展的属性grades集合
extend type Student{
grades: [Grade] @delegate(schema: "grades", path:"grades(stuNo: $fields:stuNo)")
}
成绩的Query类型
extend type Query{
getGrades(stuNo:String): [Grade] @delegate(schema: "grades", path:"grades(stuNo:$arguments:stuNo)")
getGrade(id:Int!): Grade @delegate(schema: "grades", path:"grade(id:$arguments:id)")
}
成绩实体上扩展的学生student实体属性
extend type Grade{
student: Student @delegate(schema: "students", path:"student(stuNo: $fields:stuNo)")
}
网关项目中通过AddHttpClient把子服务地址添加进来,再通过AddRemoteSchema把子项的架构引入进来,通过AddTypeExtensionsFromFile添加.graphql定义文件,实现类型的定义和拼接,使应用达到灵活性。