Nop平台以数据模型为基础,自动生成实体定义、SQL表定义、GraphQL类型、前端页面等。以部门表Department为例,缺省情况下我们会生成一个GraphQL类型Department, 并为主外键关联生成对应的属性,例如parent和children。如果增加了connection标签,我们还会为关联对象生成分页获取所对应的属性, 例如usersConnection通过类似Relay Cursor Connection的方式来分页返回属于指定部门的用户。 缺省情况下业务对象会自动继承CrudBizModel,所以它会自动生成GraphQL入口操作.
关于connection的具体介绍,参见connection.md
extend type Query{
Department__get(id:String!): Department
Department__batchGet(ids:[String!]): [Department]
Department__findPage(query:QueryBeanInput): PageBean_Department
...
}
extend type Mutation{
Department__save(data: DepartmentInput): Department
Department__delete(id:String!): Boolean
...
}
Nop平台内置了一个自动化的后台管理软件生产线,它的输入是用户需求(以Excel文档的形式表达),输出是可运行的应用系统,主要通过系统化的增量式代码生成方案来实现生产线的运转。这其中,GraphQL Schema是根据Meta元数据定义和BizModel业务模型定义自动生成的一种中间产物,我们并不会手工编写GraphQL类型定义,在编写业务代码的过程中也不需要具有任何GraphQL相关的知识,不需要实现GraphQL特有的DataFetcher、DataLoader等接口。具体的技术细节在“差量流水线”一节中会有更详细的介绍。另外可以参考以下文章:
NopGraphQL引擎在初始化的时候会利用IoC容器的动态扫描能力发现所有标记了@BizModel
注解的bean,并把它们按照BizObjName配置进行归类合并。例如
@BizModel("NopAuthUser")
public class NopAuthUserBizModel extends CrudBizModel<NopAuthUser>{
@BizMutation
public void changeSelfPassword(@Name("oldPassword") String oldPassword,
@Name("newPassword") String newPassword) {
...
}
}
@BizModel("NopAuthUser")
public class NopAuthUserBizModelEx{
@BizMutation
public void otherOperation(){
...
}
@BizMutation
@Priority(NORMAL_PRIORITY-100)
public void changeSelfPassword@Name("oldPassword") String oldPassword,
@Name("newPassword") String newPassword) {
...
}
}
NopAuthUserBizModel和NopAuthUserBizModelEx的BizObjectName都是NopAuthUser,它们的方法会叠加在一起共同生成NopAuthUser业务对象上的方法。当出现同名的函数时,会按照@Priority
优先级配置选择优先级更高的实现。如果优先级相同且函数名相同,则会抛出异常。
NopGraphQL引擎在构造BizObject的时候还会检查xbiz扩展模型,我们可以通过在NopAuthUser.xbiz模型文件中增加方法来扩展BizObject,这个模型文件可以在线更新,更新后会即时起效,无需重新初始化GraphQL类型定义。xbiz文件中定义的方法优先级最高,它会自动覆盖JavaBean中定义的业务方法。
如果把对象名相同的BizModel看作是对象的一个切片,则NopGraphQL引擎相当于是在系统初始化的时候动态收集这些对象切片,然后像docker镜像一样把它们叠加在一起,构成完整的对象定义。在运行时,最上层的xbiz切片可以被动态修改,并覆盖下层切片的功能。
BizModel切片的概念有些类似于游戏开发领域中的Entity Component System (ECS)模式,只是它累加的是动态行为而不是局部状态。
与Gather对偶的能力是Scatter:我们经常需要做一些全局规则的抽象,需要将某些公共知识自动推送到不同的业务对象中。NopGraphQL主要通过AOP机制和元编程机制来实现信息的分发:
-
公共的机制可以作为AOP拦截器作用于符合条件的业务方法上
-
xbiz文件中可以通过XLang中通用的x:gen-extends元编程机制动态生成方法定义。也可以使用外部的CodeGenerator来生成代码。
在一般的业务开发中,CRUD(Create/Read/Update/Delete)操作往往是不同的业务对象中相似度最高的部分,因此有必要对它们进行统一抽象。NopGraphQL使用设计模式中的模板方法(Template Method)模式提供了通用的CRUD实现:CrudBizModel。具体使用方法是从CrudBizModel类继承,然后可以通过实现defaultPrepareSave/afterEntityChange等函数补充定制逻辑。参见代码
CrudBizModel采用的是元数据驱动的实现方式,它会读取xmeta配置文件中的内容,内置实现了数据验证、自动初始化、级联删除、逻辑删除、数据权限等多种常见需求,所以一般情况下只需要调整xmeta和xbiz配置文件,并不需要编写定制逻辑。
-
数据验证:类似于GraphQL的输出选择,NopGraphQL可以对输入字段进行选择性验证和转换,这体现了输入和输出的对偶性。
validatedData = new ObjMetaBasedValidator(bizObjManager,bizObjName,objMeta,context,checkWriteAuth) .validateForSave(input,inputSelection)
-
自动初始化:在meta中可以配置字段的autoExpr表达式,更新或者修改的时候可以根据该配置自动初始化字段值。autoExpr表达式可以根据数据模型中的domain配置自动生成。
-
自动转换:根据meta中配置transformIn表达式,对输入的属性值进行适配转换。transformIn表达式可以根据数据模型中的domain配置自动生成。
-
级联删除:标记为cascade-delete的子表数据会随着主表数据的删除一并删除,而且会执行子表对应的BizObject业务对象上的定义的删除逻辑。
-
逻辑删除:如果启用delFlag逻辑删除标记字段,则底层的ORM引擎会自动将删除调用转换为修改delFlag的操作,并且对所有查询都自动应用delFlag=0的过滤条件,除非明确在SQL对象上设置disableLogicalDelete属性。
-
数据权限:所有读取到的实体记录都会自动验证是否满足数据权限要求。
CrudBizModel对于复杂查询提供了三个标准接口
PageBean<OrmEntity> findPage(QueryBean query, FieldSelectionBean selection);
List<OrmEntity> findList(QueryBean query);
OrmEntity findFirst(QueryBean query);
-
findPage会根据查询条件返回分页查询结果,分页逻辑可以采用cursor+next page的方式,也可以采用传统的offset+limit的方式。selection对应于前端调用时传入的返回字段集合。如果没有要求返回total总页数,则findPage内部会跳过总页数查询,如果没有要求返回items数据列表,则实际会调整真正的分页查询本身。
-
findList根据查询条件返回列表数据,如果没有设置分页大小,则按照meta上的配置选择maxPageSize条记录。
-
findFirst返回满足条件的第一条记录。
QueryBean类似于Hibernate中的Criteria查询对象,支持复杂的and/or嵌套查询条件以及排序条件。QueryBean可以由前台直接构造,在送到dao中真正执行之前它会经历如下处理过程:
-
验证查询条件中只包含标记为queryable的字段,且查询算符在每个字段的allowFilterOp集合中,缺省只允许按照相等条件进行查询。例如配置用户名支持模糊查询
<!-- 支持按照相等或者模糊匹配的方式进行查询,缺省前端生成的控件为模糊查询 --> <prop name="userName" allowFilterOp="eq,contains" xui:defaultFilterOp="contains"/>
-
追加数据权限过滤条件,例如过滤只能查看管理单位是本单位的数据。
-
增加按主键字段排序的排序条件。分页查询时如果不进行排序,则因为数据库并发执行的原因,返回的结果集合可能是随机的。所以所有分页查询原则上都应该具有排序条件,确保排序后的分页顺序一致。
QueryBean利用底层的NopOrm引擎的能力,可以很自然的支持关联对象查询,例如
<eq name="manager.dept.type" value="1" />
表示按照 manager.dept.type = 1条件进行过滤,自动根据manager_id
关联对应的部门表。
如果底层的ORM引擎不支持关联查询,也可以自行编写一个QueryTransformer接口来对QueryBean进行变换,例如将上面的等于判断转换为一个子查询
o.manager_id in (select user.id from User user, Dept dept
where user.dept_id = dept.id and dept.type = 1)
在前端,为了通过以表单方式构造复杂查询条件,我们做了如下约定:
字段名格式为: filter_{propName}__{filterOp}
例如 filter_userName__contains
表示按照contains运算符对userName字段进行过滤。对于filterOp为eq(等于条件)的情况,可以省略filterOp的部分,例如 filter_userId等价于filter_userId__eq
注意:过滤条件的值如果为空,则会忽略该字段条件。如果一定要按照空值进行查询,则可以使用__null
来表示null,使用__empty
来表示空字符串。
GraphQL中定义的操作名是全局名称,例如 query{ getUser(id:3){ id, userName}}
查询中用到的getUser方法需要在整个模型中具有唯一性,这一要求对于复用代码来说是不利的。
NopGraphQL中实现CRUD时只需要继承CrudBizModel基类,对外暴露的GraphQL操作名由对象名和方法名拼接而成。
class CrudBizModel<T>{
@BizQuery
@GraphQLReturn(bizObjName="THIS_OBJ")
public T get(@Name("id")String id){
....
}
}
@BizModel("NopAuthUser")
class NopAuthUserBizModel extends CrudBizModel<NopAuthUser>{
}
上面的示例中,NopGraphQL引擎会自动生成一个query操作NopAuthUser_get
,并且它的返回类型为THIS_OBJ
,这意味着它会被替换为当前对象所对应的BizObjName,即NopAuthUser。
注意到,采用这种实现方案,我们可以针对同一个实现类提供不同的GraphQL类型。例如
@BizModel("NopAuthUser_admin")
public NopAuthUserAdminBizModel extends CrudBizModel<NopAuthUser>{
}
同样是从CrudBizModel<NopAuthUser>
继承,但是因为BizModel注解中提供的bizObjName为NopAuthUser_admin
,则get方法返回的字段集合可以有别于普通的NopAuthUser,对后台调用的权限要求也可能不一样。
也就是说,对象上的方法名是一个局部名称,它的语义是相对于this指针而定义的。在不具备全部知识的情况下,我们可以基于相对知识编制相当复杂的逻辑,然后注入不同的this指针,就可以改变整个一组调用的具体含义。这实际上是面向对象最基本的设计原理。
面向对象技术创造了一个特殊的名---this指针,它是一种约定了的固化了的局部名称。使用this指针使得我们区分了领域(domain)的内外。在domain外对象可以有各种称谓,而domain内我们直接通过this直接指代当前对象。
代码本身只是一种形式表达,它的具体含义需要一个诠释的过程才能确定。基于对象指针的调用形式直接导向了诠释的多样化:只要注入不同的this指针,就可以提供不同的诠释。
在前台的实现中,我们使用了类似的策略:前台脚本根据方法名的后缀自动判断方法签名,例如所有以_findPage
为后缀的方法它的缺省签名都是
XXX_findPage(query:QueryBeanInput):PageBean_XXX
使用传统的Web框架在编写业务代码的时候总是不可避免的会用到框架特有的一些环境对象,例如HttpServletRequest或者SpringMVC中的ModelAndView等。这些对象都和框架特定的运行时环境强相关,使得我们的代码与某个运行时环境绑定,难以应用到多种使用场景中。最明显的,一个为在线API调用编制的服务函数,一般无法直接作为消息队列的消费者来使用。我们必须抽象出一个额外的层次:Service层,然后在Service层的基础上分别包装为Controller和MessageConsumer,让它们负责响应Web请求和消息队列。
NopGraphQL在实现业务方法时,采用的是一种框架无关的非侵入式设计,它扩展了服务方法的使用场景,简化了服务层的编写。具体来说,NopGraphQL引入了少量注解,使用POJO对象来作为输入输出对象,自动将业务方法翻译为GraphQL引擎所需的DataFetcher和DataLoader。例如
@BizModel("MyEntity")
class MyBizModel{
@BizQuery
public MyEntity get(@Name("id")String id){
return ...
}
@BizLoader
public String extProp(@ContextSource MyEntity entity){
...
}
@BizLoader(forType=OtherEntity.class)
public String otherProp(@ContextSource OtherEntity entity){
...
}
@BizLoader("someProp")
public CompletionStage<List<SomeObject>> batchLoadSomePropAsync(
@ContextSource List<MyEntity> entities){
...
}
}
-
@BizQuery
表示本方法将被映射为GraphQL中的query调用,@BizMutation
将被映射为GraphQL中的mutation调用。 -
@BizLoader
为GraphQL类型的属性提供fetcher和loader定义。注意,为了保证概念的简单性,NopGraphQL要求所有属性都必须在xmeta文件中声明,BizModel中仅是为已定义的属性提供定制的加载器。 -
如果返回值类型为CompletionStage,则表示该方法异步执行
-
如果标注了
@BizLoader
注解的方法的ContextSource参数为 List类型,则表示它对应GraphQL的DataLoader实现,支持批量加载。
基于NopGraphQL引擎编写的服务方法,可以看作具有如下函数签名
ApiResponse<Object> service(ApiRequest<Map> request);
class ApiRequest<T>{
Map<String,Object> headers;
FieldSelectionBean selection;
T data;
}
服务方法都是接收一个POJO的request对象,返回一个POJO的response对象。因为输入和输出都是简单对象,所以可以无需编码,只需要简单配置,就可以做到
-
把GraphQL服务方法发布为消息队列的消费者,它从一个topic接收request对象,向另一个topic发送返回消息,如果header中标注了one-way,则忽略返回消息。
-
将GraphQL服务方法发布为RPC服务函数
-
从批处理文件中读取Request对象,依次调用服务方法,批量提交,失败重试,然后把返回的Response消息写入到输出文件中。
GraphQL引擎可以运行在REST服务之上,提供所谓federation的功能,将多个REST服务组合为一个统一的GraphQL端点。那么反过来是不是也可以将底层的GraphQL服务方法拆解开来,暴露为一个个独立的REST资源?
NopGraphQL借助lazy字段的概念,对GraphQL类型定义Eager加载的属性集合,通过规范化的方式将GraphQL模型中的方法转化为REST服务。具体REST链接格式如下
/r/{operationName}?@selection=a,b,c{d,e}
-
通过request body来传参数
-
/r/{operationName}为服务链接,通过可选的
@selection
参数来指定对返回结果的字段选择。如果不指定,则后台会自动返回所有没有标记为lazy的属性。代码生成的时候,关联表的数据缺省会被标记为lazy,因此它们在缺省情况下不会包含在REST调用的返回结果中。
如果是query请求,则可以通过GET方法来进行调用,此时可以通过URL参数来传递调用参数。例如
GET /r/NopAuthUser_get?id=3
等价于执行 NopAuthUser_get(id:3)。
Nop平台的前端框架在百度AMIS框架的基础上,对GraphQL调用做了进一步的简化。在前端,我们现在可以使用如下url格式来发起GraphQL调用,
api: {
url: '@query:NopAuthUser__get/id,userName?id=$id'
}
上面的url链接使用了所谓的前缀引导语法,底层的ajaxFetch函数会识别@query:
前缀,并把它转化为graphql请求
query($id:String){
NopAuthUser_get(id:$id){
id, userName
}
}
ajaxFetch识别的graphql url的格式为
(@query|@mutation):{operationName}/{selection}?参数名=参数值
当我们需要为表单或者表格编写加载函数时,如果字段比较多,则手工编写graphql请求很容易出现字段遗漏。因为Nop平台的前端代码也是自动生成的,所以我们可以利用编译期信息自动生成graphql请求,使得我们恰好只选择表单或者表格中用到的数据。具体做法是引入编译期的变量formSelection, pageSelection等。例如
@query:NopAuthUser_get/{@formSelection}?id=$id
{@formSelection}
表示选择当前表单中用到的所有字段。
GraphQL是一种强类型的框架,它要求所有数据都有明确的类型定义,这在某些动态场景中使用时并不方便。例如有的时候我们可能需要把一个扩展集合返回到前端。
NopGraphQL引入了一个特殊的Scalar类型: Map,可以利用它来描述那些动态数据结构。例如
type QueryBean{
filter: Map
orderBy: [OrderFieldBean]
}
对于单位树、菜单树这样的树形结构的获取,NopGraphQL通过Directive机制提供了一个扩展语法,可以直接表达递归拉取数据,例如
NopAuthDept_findList{
value: id,
label: displayName
children @TreeChildren(max=5)
}
@TreeChild(max=5)
表示按照本层的结构最多嵌套5层。