2

我正在使用 Web API 将 Hot Chocolate(GraphQL) 添加到现有的 ASP.Net Core 项目中,并重用 Web API 使用的模型。其中一个模型具有 IDictionary<string, object> 属性,该属性通过 Web API 序列化为动态 JSON。

该模型

public class Theme
{
    ...
    public IDictionary<string, object> Styles { get; set; }
    ...
}

使用 Web API 可以序列化为

{
  ...
  styles: {
    header: {
      color: "#fff", 
      background: "rgba(255, 255, 255, 0)", 
      boxShadow: "", position: "fixed"},
      ...
    },
    borderRadius: "4px",
    ...
  }
  ...
}

使用 Hot Chocolate,我正在重用模型并在启动时添加 GraphQL

services.AddGraphQL(sp => SchemaBuilder.New()
    .AddServices(sp)
    .AddQueryType(d => d.Name("Query"))
    ...
    .AddType<Theme>()
    .Create());

生成的模式变为

type Theme {
  ...
  styles: [KeyValuePairOfStringAndObject!]
  ...
}

并且只能检索到密钥

{
  themes(...) {
    edges {
      node {
        name
        styles{
          key
        }
      }
    }
  }
}

有以下回应

{
  "themes": {
    "edges": [
      {
        "node": {
          ...
          "styles": [
            {
              "key": "header"
            },
            {
              "key": "borderRadius"
            },
            ...
          ]
        }
      }
    ]
  }
}

我想设置 Hot Chocolate,以便编写一个 GraphQL 查询,该查询提供与 Web API 相同的动态结果。

更新

源是一个 JSON 字符串,它被反序列化为 Dictionary<string, object>

var jsonString = "{\"header\":{\"color\":\"#fff\"},\"borderRadius\":\"4px\"}";

theme.Styles = JsonConvert.DeserializeObject<Dictionary<string, object>>(jsonString);

问题是生成的字典没有递归序列化为 Dictionary<string, object>。

4

1 回答 1

6

Hot Chocolate 具有适用于动态数据结构的 AnyType。但是默认情况下 AnyType 不是从类型中隐式推断的。

你在这里基本上有两个选择。使用属性在您的类型上声明类型(或对流式 API 执行相同操作)。

纯代码优先:

public class Theme
{
    ...
    [GraphQLType(typeof(AnyType))]
    public IDictionary<string, object> Styles { get; set; }
    ...
}

具有模式类型的代码优先:

public class ThemeType : ObjectType<Theme> 
{
    protected override void Configure(IObjectTypeDescriptor<Theme> descriptor)
    {
        descriptor.Field(t => t.Styles).Type<AnyType>();
    }
}

或者,您可以说在您的架构中所有IDictionary<string, object>类型都是任何类型,在这种情况下您不需要显式定义类型:

services.AddGraphQL(sp => SchemaBuilder.New()
    .AddServices(sp)
    .AddQueryType(d => d.Name("Query"))
    ...
    .AddType<Theme>()
    .BindClrType<IDictionary<string, object>, AnyType>()
    .Create());

我希望这有帮助。

于 2020-06-23T11:28:57.493 回答