/graphql-java-calculator

Based on directive, this project provide graphql query with the ability of orchestration, field calculation and control flow.

Primary LanguageJavaApache License 2.0Apache-2.0

graphql-java-calculator

Build and Publish Latest Release

📖 English Documentation | 📖 中文文档


基于指令机制graphql-java-calculatorgraphql查询提供了数据编排、动态计算和控制流的能力。

特性

  • 数据编排:将指定字段的获取结果作为全局可获取的上下文,为获取其他字段提供可依赖数据;
  • 动态计算:对查询结果进行排序、过滤;通过全局可获取上下文和父字段获取结果计算生成新的字段;
  • 控制流:@skip@include拓展版本,通过全局可获取上下文、字段请求参数,判断是否解析指定字段;
  • 参数转换:对字段请求参数进行转换、列表类型参数过滤、列表类型参数的元素进行转换,转换表达式可使用全局可获取上下文作为参数。

计算指令的具体使用方式参考指令说明使用示例

快速开始

1. 引入依赖

<dependency>
    <groupId>com.graphql-java-calculator</groupId>
    <artifactId>graphql-java-calculator</artifactId>
    <version>1.1</version>
</dependency>

最新版本见 Maven仓库

2. 包装执行引擎

2.1 继承AsyncDataFetcherInterface

如果项目中使用了异步DataFetcher,则使其则继承AsyncDataFetcherInterface, 并在方法实现中返回被包装的DataFetcher和使用的线程池。

2.2 创建GraphQLSource

使用配置类Config创建GraphQLSource对象,GraphQLSource包含GraphQLSchemaGraphQL, 配置类可指定脚本执行引擎、计算指令引擎使用的线程池和对象转换工具。

脚本语法使用了aviatorscript,aviator是graphql-java-calculator的默认表达行引擎, 可通过ScriptEvaluatorConfig自定义脚本执行引擎。

2.3 执行前校验

使用Validator对计算指令的使用进行语法校验、该校验包含graphql原生语法校验, 建议实现CalculatorDocumentCachedProvider缓存校验结果。

完整示例参考Example

指令说明

@fetchSource

directive @fetchSource(name: String!, sourceConvert:String) on FIELD

参数解释:

  • name:source的名称,一个查询语句中source名称必须是唯一的;
  • sourceConvert:对字段绑定DataFetcher的获取结果进行转换,所有依赖该source的指令获取的都是转换后的数据。

@fetchSource是进行数据编排的基础,该指令注解字段的DataFetcher的获取结果、可作为全局可获取上下文、在其他字段的计算指令上通过dependencySources获取。

如果@fetchSource所注解的字段在列表路径中,则将该字段的集合将作为source的值。如下查询source类型为List,元素值为用户的名称。

query fetchSourceDemo($userIds: [Int]){
    consumer{
        userInfoList(userIds: $userIds){
            userId
            name @fetchSource(name: "nameList")
        }
    }
    # ... do some thing
}

@skipBy

directive @skipBy(predicate: String!, dependencySources: String) on FIELD

参数解释:

  • predicate:判断是否跳过解析该字段的表达式,表达式参数为查询变量和其他@fetchSource;
  • dependencySources:指令表达式依赖的source,sourceName不可和查询变量。

@skipBy是graphql内置指令@skip的扩展版本,可通过表达式判断是否请求该字段,表达式参数为查询变量和其他@fetchSource。

若依赖全局可获取上下文,则表达式变量中会加上一个key为source名称、值为source的键值对。

该指令可实现类似于if(predicate){}switch(c): case predicate1: opx; case predicate2: opy;的控制流。

@includeBy

directive @includeBy(predicate: String!, dependencySources: String) on FIELD

参数解释:

  • predicate:判断是否解析该字段的表达式,表达式参数为查询变量和其他@fetchSource;
  • dependencySources:表达式依赖的source,sourceName不可和查询变量同名。

@includeBy是graphql内置指令@include的扩展版本,可通过表达式判断是否请求该字段,表达式参数为查询变量和其他@fetchSource。

若依赖全局可获取上下文,则表达式变量中会加上一个key为source名称、值为source的键值对。该指令能力同@skipBy。

@map

directive @map(mapper:String!, dependencySources:String) on FIELD

参数解释:

  • expression:计算字段值的表达式;
  • dependencySources:表达式依赖的source,sourceName如果和父节点绑定DataFetcher的获取结果key相同,则计算表达式时会覆父节点中的数据。

以父节点绑定的DataFetcher获取结果和dependencySources为参数,计算注解的字段的值。被注解字段绑定的DataFetcher不会在执行。

@argumentTransform

directive @argumentTransform(argumentName:String!, operateType:ParamTransformType, expression:String, dependencySources:String) on FIELD

enum ParamTransformType{
    MAP
    FILTER
    LIST_MAP
}

参数解释:

  • argumentName:该指令进行转换的参数名称;
  • operateType:操作类型,包括参数整体转换MAP、列表参数过滤FILTER、列表参数元素转换LIST_MAP三种;
  • expression:计算新值、或者对参数进行过滤的表达式;
  • dependencySources:表达式依赖的source,source如果和参数变量同名、则会覆盖后者。

对字段参数进行转换、过滤,具体操作有如下三种:

  1. 参数映射(operateType = Map ):将表达式结果赋给指定的字段参数,该操作将字段上的所有变量作为表达式参数;
  2. 列表参数过滤(operateType = FILTER):过滤列表类型参数中的元素,该操作将字段上的所有变量和<"ele",元素值>作为表达式参数;
  3. 列表参数映射(operateType = LIST_MAP):使用表达式对列表参数中的每个元素进行转换,该操作将字段上的所有变量和<"ele",元素值>作为表达式参数。

若依赖全局可获取上下文,则表达式变量中会加上一个key为source名称、值为source的键值对。

@filter

directive @filter(predicate: String!) on FIELD

参数解释:

  • predicate:过滤判断表达式,结果为true的元素会被保留;

对列表进行过滤,参数为查询解析结果:当列表元素为对象类型时、表达式变量为对象对应的Map,当元素为基本类型时、表达式变量为key为ele、value为元素值。

@sortBy

directive @sortBy(comparator: String!, reversed: Boolean = false) on FIELD

参数解释:

  • comparator:按照该表达式计算结果、对列表进行排序;
  • reversed:是否进行逆序排序,默认为false。

对列表进行排序,参数为查询解析结果:当列表元素为对象类型时、表达式变量为对象对应的Map,当元素为基本类型时、表达式变量为key为ele、value为元素值。 不管reversed是否为true,表达式结果为null的元素总是排在列表最后。

使用示例

测试schema为例, 对计算指令实现数据编排、结果处理转换和控制流等的能力进行说明。

数据编排

数据编排的主要形式为请求a字段时、其请求参数为b字段的结果,或者需要b字段结果对a字段请求参数进行过滤、转换处理。

  • 获取商品信息,并通过商品列表中的sellerId获取卖家信息
query sourceInList_case01($itemIds:[Int]){
    commodity{
        itemList(itemIds: $itemIds){
            # 保存商品的卖家id,结果为 List<Integer>
            sellerId @fetchSource(name: "sellerIdList")
            name
            saleAmount
            salePrice
        }
    }

    consumer{
        userInfoList(userIds: 1)
        @argumentTransform(argumentName: "userIds", # 对参数 userIds 进行转换
                            operateType: MAP, # 操作类型为参数整体转换
                            expression: "sellerIdList", # 表达式表示使用表达式变量 sellerIdList 对参数作整体替换
                            dependencySources: ["sellerIdList"] # 依赖了全局变量 sellerIdList
        ){
            userId
            name
            age
        }
    }
}
  • 变量只有券id、查询该券绑定的商品详情。
query getItemListBindingCouponIdAndFilterUnSaleItems ( $couponId: Int) {
    marketing{
        coupon(couponId: $couponId){
            # 获取券绑定的商品id,并注册为名称为 itemIdList 的source
            bindingItemIds @fetchSource(name: "itemIdList")
        }
    }

    commodity{
        itemList(itemIds: 1)
        # 对参数 itemIds 进行映射转换,映射规则为'itemIdList'、即直接使用变量 itemIdList 进行替换,该计算依赖了名称为 itemIdList 的source
        @argumentTransform(argumentName: "itemIds", operateType: MAP,expression: "itemIdList",dependencySources: "itemIdList")
        {
            itemId
            name
            salePrice
            onSale
            # sellerId
        }
    }
}

参数转换

入参为userId,按照指定的格式拼接为 redis 的key。

query userNewInfo($userId: Int){
    consumer{
        isNewUser(redisKey: "fashion:shoes:",userId: $userId)
        # 将参数拼接为 redis 的key,fashion:shoes:{userId}
        @argumentTransform(argumentName: "redisKey",operateType: MAP ,expression: "concat(redisKey,userId)")
        {
            userId
            isNewUser
            sceneKey
        }
    }
}

参数过滤

  • 请求商品信息前,过滤掉没有指定绑定券id的商品id,券绑定的商品id来自其他字段数据。
query filterItemListByBindingCouponIdAndFilterUnSaleItems ( $couponId: Int,$itemIds: [Int]) {
    commodity{
        itemList(itemIds: $itemIds)
        @argumentTransform(argumentName: "itemIds", operateType: FILTER,dependencySources: "itemIdList",expression: "listContain(itemIdList,ele)")
        {
            itemId
            name
            salePrice
            onSale
        }
    }

    marketing{
        coupon(couponId: $couponId){
            bindingItemIds
            @fetchSource(name: "itemIdList")
        }
    }
}

控制流

控制流主要为根据条件,判断是否请求某个类型数据、或者请求哪个类型数据。

控制流通过 @skipBy进行控制 directive @skipBy(expression: String!, dependencySources: String) on FIELD

通过 @skipBy 可实现类似 switch-case的控制流,

// @fetchSource
switch(value):
    // @skipBy(value,judgeFunction)
    case(judgeFunction_1(value)): opration_1;
    case(judgeFunction_2(value)): opration_2;
    case(judgeFunction_2(value)): opration_2;

和if控制流 if(conditioin){...add return value...}

# 如果用户不在 ab实验实验组区间[0,3]内,则对其查看的页面不展示优惠券、即不请求券数据
query abUserForCouponAcquire($userId: Int, $couponId: Int,$abKey:String){

    marketing
    @skipBy(predicate: "abValue <= 3",dependencySources: "abValue")
    {
        coupon(couponId: $couponId){
            couponId
            couponText
            price
        }
    }

    toolInfo{
        abInfo(userId: $userId,abKey:$abKey) @fetchSource(name: "abValue")
    }
}

数据补全/动态计算

  • 分别查找券信息和列表商品信息;
  • 如果商品绑定了券则返回券后价和是否绑定的标识
  • 对券数据拼接描述文案;
query calculateCouponPrice_Case01 ($couponId: Int, $itemIds: [Int]){

    marketing{
        coupon(couponId: $couponId)
        @fetchSource(name: "itemCouponInfo",sourceConvert: "list2MapWithAssignedValue('coupon.bindingItemIds','coupon.price')")
        {
            base
            price
            bindingItemIds
            desc: couponText @map(mapper: "'满' + base + '减' + price")
        }
    }

    commodity{
        itemList(itemIds: $itemIds){
            itemId
            name
            salePrice
            isUsedCoupon: onSale @map(dependencySources: "itemCouponInfo",mapper: "seq.get(itemCouponInfo,itemId)!=nil")
            # 券后价
            couponPrice: salePrice @map(dependencySources: "itemCouponInfo",mapper: "salePrice - (seq.get(itemCouponInfo,itemId) == nil? 0:seq.get(itemCouponInfo,itemId)) ")
        }
    }
}

列表排序

对列表字段进行排序。例如对商品进行排序:将可用券的商品放在列表前边。

  1. 先通过 @fetchSource@map 指令标识商品是否可用指定券;
  2. 使用 @sortBy对列表进行排序;
query sortByWithSource_case01{
    commodity{
        itemList(itemIds: [9,11,10,12])
        @sortBy(comparator: "!isContainBindingItemIds")
        {
            isContainBindingItemIds:onSale @map(mapper: "listContain(bindingItemIds,itemId)",dependencySources: "bindingItemIds")
            itemId
            name
            salePrice
        }
    }

    marketing{
        coupon(couponId: 1){
            bindingItemIds @fetchSource(name: "bindingItemIds")
        }
    }
}

列表过滤

对列表进行过滤:只保留可用券的商品。

  1. 先通过 @fetchSource@map 指令标识商品是否可用指定券;
  2. 使用 @filter 过滤出可使用券的商品。
query filter_case01{

    # 查询可使用指定券的商品id
    marketing{
        coupon(couponId: 1){
            bindingItemIds @fetchSource(name: "bindingItemIds")
        }
    }

    commodity{
        itemList(itemIds: [9,11,10,12])
        # 通过 filter 过滤不可用券的商品
        @filter(predicate: "!isContainBindingItemIds")
        {
            # 通过 @map 命令标识该商品是否可使用券
            isContainBindingItemIds:onSale @map(mapper: "listContain(bindingItemIds,itemId)",dependencySources: "bindingItemIds")
            itemId
            name
            salePrice
        }
    }
}

联系反馈

欢迎在issue区对使用中遇到的bug、设计问题或者期待的新特性进行讨论。

也可通过以下方式联系作者: