动态生成GraphQL架构支持

Jas*_*onS 3 java graphql graphql-js graphql-java

是否可以动态创建GraphQL架构?

我们将数据存储在mongoDB中,并且可能会添加新字段.我们不希望在mongoDB文档中对此新添加的字段进行任何代码更改.

我们有什么方法可以动态生成架构吗?

模式在代码中定义,但对于java(模式为pojo),添加新属性时,必须更新并重新编译代码,然后再次存档和部署jar.有什么方法可以通过数据生成模式而不是预先定义它?

目前我们正在使用java相关项目(graphql-java,graphql-java-annotations)进行GraphQL开发.

kaq*_*qao 5

您可以使用graphql-spqr,它允许您根据服务类自动生成模式.在你的情况下,它看起来像这样:

public class Pojo {

    private Long id;
    private String name;

    // whatever Ext is, any (complex) object would work fine
   private List<Ext> exts;
}

public class Ext {
    public String something;
    public String somethingElse;
}
Run Code Online (Sandbox Code Playgroud)

据推测,您有一个包含业务逻辑的服务类:

public class PojoService {

    //this could also return List<Pojo> or whatever is applicable
    @GraphQLQuery(name = "pojo")
    public Pojo getPojo() {...}
}
Run Code Online (Sandbox Code Playgroud)

要公开此服务,您只需执行以下操作:

GraphQLSchema schema = new GraphQLSchemaGenerator()
                .withOperationsFromSingleton(new PojoService())
                .generate();
Run Code Online (Sandbox Code Playgroud)

然后,您可以触发查询,例如:

query test {
    pojo {
        id
        name 
        exts {
            something
            somethingElse
        } } }
Run Code Online (Sandbox Code Playgroud)

不需要任何奇怪的包装或自定义代码,也不需要牺牲类型安全性.适用于泛型,依赖注入或您在项目中可能拥有的任何其他爵士乐.

完全披露:我是graphql-spqr的作者.


Jas*_*onS 0

经过几天的调查。我发现用 Java 动态生成 schema 很困难(或者成本太高)。

嗯,从另一个角度来说。我认为我们可以使用Map一种折衷的方式来实现这一目标。

  • POJO/实体

    public class POJO{
        @GraphQLField
        private Long id;
    
        @GraphQLField
        private String name;
    
        // ...
    
        @GraphQLField
        private GMap exts;   
    }
    
    Run Code Online (Sandbox Code Playgroud)

    GMap是一个定制的Map(因为Map/HashMap是JDK内部类,不能作为GraphQL Schema而只能扩展)。

  • 地图

    public class GMap extends HashMap<String, String> {
    
        @GraphQLField
        public String get(@GraphQLName("key") String key) {
            return super.get(key);
        }
    }
    
    Run Code Online (Sandbox Code Playgroud)
  • 从客户端检索数据

    // query script
    query test
    {
        your_method
        {
            id
            name
            exts {
                get(key: "ext") // Add a extended attribute someday
            }
        }
    }
    
    // result 
    {
        "errors":[],
        "data":
        {
            "list":
            [
                {"id":1, name: "name1", exts: {"get": "ext1"}},
                {"id":2, name: "name2", exts: {"get": "ext2"}}
            ]
        }
    }
    
    Run Code Online (Sandbox Code Playgroud)