Реализация поддержки ETag в ASP.NET MVC4 WebAPI

В последней бета-версии ASP.NET MVC4, как бы вы поддерживали условную поддержку GET через ETag? ActionFilter должен иметь возможность завершить запрос на создание ETag для возвращаемого ресурса, чтобы сравнить его с заголовком If-None-Match в запросе. И затем, независимо от того, был ли входящий ETag в заголовке If-None-Match таким же, как сгенерированный ETag, добавьте сгенерированный ETag в заголовок ответа ETag. Но с ASP.NET MVC4 я понятия не имею, с чего начать. Какие-либо предложения?


person Bullines    schedule 28.04.2012    source источник


Ответы (4)


Лично я не фанат «магии фреймворка» и предпочитаю старый добрый код в веб-методах, иначе мы получим что-то более похожее на WCF, юк.

Итак, в вашем веб-методе Get вручную создайте ответ следующим образом:

var response = this.Request.CreateResponse(HttpStatusCode.OK, obj);
string hash = obj.ModifiedDate.GetHashCode().ToString();

response.Headers.ETag =
    new EntityTagHeaderValue(String.Concat("\"", hash, "\""), true);

return response;

Обратите внимание, что ETag, полученный из хэш-кода временной метки, является чисто иллюстрацией слабой системы маркировки объектов. Он также показывает необходимые дополнительные кавычки.

person Luke Puplett    schedule 04.07.2012

Есть ETagMessageHandler в WebApiContrib который делает то, что вам нужно.


ОБНОВИТЬ

Я реализовал кэширование на стороне сервера RFC 2616 в WebApiContrib. Найдите CachingHandler. Подробнее здесь.


Больше обновлений

Это будет активно развиваться и расширяться в рамках CacheCow. Это будет включать как клиентские, так и серверные компоненты. пакеты NuGet будут скоро опубликованы уже опубликованы.

Обработчик CachingHandler WebApiContrib по-прежнему будет поддерживаться, поэтому сообщайте мне о любых ошибках или проблемах.

person Aliostad    schedule 04.05.2012

Ответ Люка Пуплетта привел меня на правильный путь (+1), но обратите внимание, что вы также должны прочитать ETag на стороне сервера, чтобы избежать отправки всех данных с каждым запросом:

string hash = obj.ModifiedDate.GetHashCode().ToString();
var etag = new EntityTagHeaderValue(String.Concat("\"", hash, "\""), true);

if (Request.Headers.IfNoneMatch.Any(h => h.Equals(etag)))
{
    return new HttpResponseMessage(HttpStatusCode.NotModified);
}

var response = this.Request.CreateResponse(HttpStatusCode.OK, obj);
response.Headers.ETag = etag;
return response;

Также было бы неплохо соблюдать заголовок If-Modified-Since. См. RFC 2616.

person Greg McCoy    schedule 14.12.2015

Кажется, это то, что вы ищете (см. раздел «Поддержка ETags»):

http://blogs.msdn.com/b/webdev/archive/2014/03/13/начало-работы-с-asp-net-web-api-2-2-for-odata-v4-0.aspx

Если ваша модель хранится глубже в домене и вы не можете применить атрибут [ConcurrencyCheck], вы можете сделать это с помощью ODataModelBuilder:

ODataModelBuilder builder = new ODataConventionModelBuilder();
var myEntity = builder.EntitySet<MyEntity>("MyEntities"); 
myEntity.EntityType.Property(l => l.Version).ConcurrencyToken = true;

это заставит добавить свойство «@odata.etag» в тело ответа.

person Stas Natalenko    schedule 27.09.2014