Мне очень трудно перейти со службы данных WCF на веб-api odata v4. Я застрял в следующем вопросе:
Веб-API odata не поддерживает @odata.bind.
Я нашел следующую ссылку: https://github.com/OData/WebApi/issues/158 Кто-то предложил реализовать собственный производный класс ODataEntityDeserializer
следующим образом:
public class ExtendedODataEntityDeserializer : ODataEntityDeserializer
{
public ExtendedODataEntityDeserializer(ODataDeserializerProvider deserializerProvider) : base(deserializerProvider)
{
}
public override void ApplyNavigationProperty(
object entityResource,
ODataNavigationLinkWithItems navigationLinkWrapper,
IEdmEntityTypeReference entityType,
ODataDeserializerContext readContext)
{
base.ApplyNavigationProperty(entityResource, navigationLinkWrapper, entityType, readContext);
foreach (var childItem in navigationLinkWrapper.NestedItems)
{
var entityReferenceLink = childItem as ODataEntityReferenceLinkBase;
if (entityReferenceLink != null)
{
var navigationPropertyName = navigationLinkWrapper.NavigationLink.Name;
Uri referencedEntityUrl = entityReferenceLink.EntityReferenceLink.Url;
if (!referencedEntityUrl.IsAbsoluteUri)
{
referencedEntityUrl = new Uri(readContext.Request.RequestUri, referencedEntityUrl);
}
var linkedEntities = (Model.LinkedEntityCollection)entityResource;
linkedEntities.Add(navigationPropertyName, referencedEntityUrl);
}
}
}
}
LinkedEntityCollection — это базовый класс для моих классов сущностей, который служит словарем (реализация MyEntity : LinkedEntityCollection
тривиальна).
public class ExtendedODataDeserializerProvider : ODataDeserializerProvider
{
private static ExtendedODataDeserializerProvider _instance = null;
private ExtendedODataDeserializerProvider()
{
_instance = this;
}
public static ExtendedODataDeserializerProvider Instance
{
get { return _instance ?? new ExtendedODataDeserializerProvider(); }
}
public override ODataEdmTypeDeserializer GetEdmTypeDeserializer(Microsoft.OData.Edm.IEdmTypeReference edmType)
{
return DefaultODataDeserializerProvider.Instance.GetEdmTypeDeserializer(edmType);
}
public override ODataDeserializer GetODataDeserializer(Microsoft.OData.Edm.IEdmModel model, Type type, System.Net.Http.HttpRequestMessage request)
{
return new ExtendedODataEntityDeserializer(DefaultODataDeserializerProvider.Instance);
}
}
Регистрация в конфигурации http:
public static void Register(HttpConfiguration config)
{
ODataModelBuilder builder = new ODataConventionModelBuilder();
// registering entities: builder.EntitySet<T>("EntityName") ...
ODataBatchHandler batchHandler = new DefaultODataBatchHandler(new HttpServer(config));
batchHandler.MessageQuotas.MaxOperationsPerChangeset = 10;
batchHandler.MessageQuotas.MaxPartsPerBatch = 10;
var odataFormatters = ODataMediaTypeFormatters.Create(DefaultODataSerializerProvider.Instance, ExtendedODataDeserializerProvider.Instance);
config.Formatters.Clear();
config.Formatters.AddRange(odataFormatters);
config.MapODataServiceRoute(routeName: "Central", routePrefix: "Odata", model: builder.GetEdmModel(), batchHandler: batchHandler);
}
Однако, если вы используете этот хак, вы теряете возможность создавать ссылки между объектами, как описано в этой статье: http://www.asp.net/web-api/overview/odata-support-in-aspnet-web-api/odata-v4/entity-relations-in-odata-v4
"@odata.id" не анализируется, предполагается, что это сущность в функции ODataEntityDeserializer.Read
. Может быть, я что-то принципиально упускаю здесь, так как я новичок в веб-API (config.Formatters.Clear()
кажется немного резким). Или есть другой обходной путь для поддержки @odata.bind?
Есть ли шанс, что эта функция будет реализована в ближайшем будущем?
Обновить
У меня работает хак. Проблема заключалась в том, что я должен был вывести ExtendedODataDeserializerProvider
из DefaultODataDeserializerProvider
вместо ODataDeserializerProvider
и соответственно вернуть десериализаторы:
public override ODataDeserializer GetODataDeserializer(
Microsoft.OData.Edm.IEdmModel model,
Type type,
System.Net.Http.HttpRequestMessage request)
{
if (type == typeof(ODataActionParameters) ||
type == typeof(ODataUntypedActionParameters) ||
type == typeof(Uri))
{
return base.GetODataDeserializer(model, type, request);
}
return new ExtendedODataEntityDeserializer(DefaultODataDeserializerProvider.Instance);
}
Итак, остается вопрос: когда мы получим поддержку @odata.bind в asp.net webapi odata?