Сериализовать IDictionary с GraphQL и Hot Choclate в динамику c JSON - PullRequest
0 голосов
/ 21 июня 2020

Я добавляю Hot Chocolate (GraphQL) в существующий проект ASP. Net Core с веб-API и повторно использую модели, которые используются веб-API. Одна из моделей имеет IDictionary свойство, которое сериализуется в динамический c JSON с помощью веб-API.

Модель

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

, которая с помощью веб-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, который дает такой же динамический c результат, что и веб-API .

ОБНОВЛЕНИЕ

Источник - строка JSON, десериализованная в словарь с

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

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

Проблема в том, что результирующий словарь не рекурсивно сериализуется в словарь .

1 Ответ

1 голос
/ 23 июня 2020

Hot Chocolate имеет AnyType для динамических c структур данных. Но AnyType по умолчанию не выводится неявно из типов.

Здесь в основном есть два варианта. Либо объявите тип своего типа с помощью атрибута (или сделайте то же самое с текучим API).

Pure Code-First:

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

Code-First с типами схемы:

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());

Надеюсь, это поможет.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...