ASP.NET Web API faz negociação de conteúdo por padrão - retornará XML ou JSON ou outro tipo com base no Accept
cabeçalho. Eu não preciso / quero isso, há uma maneira (como um atributo ou algo assim) para dizer à API Web para sempre retornar JSON?
ASP.NET Web API faz negociação de conteúdo por padrão - retornará XML ou JSON ou outro tipo com base no Accept
cabeçalho. Eu não preciso / quero isso, há uma maneira (como um atributo ou algo assim) para dizer à API Web para sempre retornar JSON?
Respostas:
Compatível apenas com JSON na API da Web ASP.NET - A MANEIRA CORRETA
Substitua IContentNegotiator por JsonContentNegotiator:
var jsonFormatter = new JsonMediaTypeFormatter();
//optional: set serializer settings here
config.Services.Replace(typeof(IContentNegotiator), new JsonContentNegotiator(jsonFormatter));
Implementação JsonContentNegotiator:
public class JsonContentNegotiator : IContentNegotiator
{
private readonly JsonMediaTypeFormatter _jsonFormatter;
public JsonContentNegotiator(JsonMediaTypeFormatter formatter)
{
_jsonFormatter = formatter;
}
public ContentNegotiationResult Negotiate(
Type type,
HttpRequestMessage request,
IEnumerable<MediaTypeFormatter> formatters)
{
return new ContentNegotiationResult(
_jsonFormatter,
new MediaTypeHeaderValue("application/json"));
}
}
Accept
XML obterá JSON e não 406 ?
Accept
cabeçalho.
GlobalConfiguration...Clear()
realmente funciona.
Limpe todos os formatadores e adicione o formatador Json de volta.
GlobalConfiguration.Configuration.Formatters.Clear();
GlobalConfiguration.Configuration.Formatters.Add(new JsonMediaTypeFormatter());
EDITAR
Eu adicionei para Global.asax
dentro Application_Start()
.
Philip W teve a resposta certa, mas para maior clareza e uma solução de trabalho completa, edite seu arquivo Global.asax.cs para ficar assim: (Observe que tive que adicionar a referência System.Net.Http.Formatting ao arquivo gerado no estoque)
using System;
using System.Collections.Generic;
using System.Linq;
using System.Net.Http.Formatting;
using System.Web;
using System.Web.Http;
using System.Web.Mvc;
using System.Web.Optimization;
using System.Web.Routing;
namespace BoomInteractive.TrainerCentral.Server {
// Note: For instructions on enabling IIS6 or IIS7 classic mode,
// visit http://go.microsoft.com/?LinkId=9394801
public class WebApiApplication : System.Web.HttpApplication {
protected void Application_Start() {
AreaRegistration.RegisterAllAreas();
WebApiConfig.Register(GlobalConfiguration.Configuration);
FilterConfig.RegisterGlobalFilters(GlobalFilters.Filters);
RouteConfig.RegisterRoutes(RouteTable.Routes);
BundleConfig.RegisterBundles(BundleTable.Bundles);
//Force JSON responses on all requests
GlobalConfiguration.Configuration.Formatters.Clear();
GlobalConfiguration.Configuration.Formatters.Add(new JsonMediaTypeFormatter());
}
}
}
GlobalConfiguration.Configuration.Formatters.XmlFormatter.SupportedMediaTypes.Clear();
Isso limpa o formatador XML e, portanto, padroniza para o formato JSON.
Inspirado pela excelente resposta de Dmitry Pavlov, eu alterei um pouco para que pudesse inserir qualquer formatador que desejasse aplicar.
Crédito para Dmitry.
/// <summary>
/// A ContentNegotiator implementation that does not negotiate. Inspired by the film Taken.
/// </summary>
internal sealed class LiamNeesonContentNegotiator : IContentNegotiator
{
private readonly MediaTypeFormatter _formatter;
private readonly string _mimeTypeId;
public LiamNeesonContentNegotiator(MediaTypeFormatter formatter, string mimeTypeId)
{
if (formatter == null)
throw new ArgumentNullException("formatter");
if (String.IsNullOrWhiteSpace(mimeTypeId))
throw new ArgumentException("Mime type identifier string is null or whitespace.");
_formatter = formatter;
_mimeTypeId = mimeTypeId.Trim();
}
public ContentNegotiationResult Negotiate(Type type, HttpRequestMessage request, IEnumerable<MediaTypeFormatter> formatters)
{
return new ContentNegotiationResult(_formatter, new MediaTypeHeaderValue(_mimeTypeId));
}
}
Se você quiser fazer isso apenas para um método, declare seu método como retornando em HttpResponseMessage
vez de IEnumerable<Whatever>
e faça:
public HttpResponseMessage GetAllWhatever()
{
return Request.CreateResponse(HttpStatusCode.OK, new List<Whatever>(), Configuration.Formatters.JsonFormatter);
}
este código é uma dor para o teste de unidade, mas também é possível assim:
sut = new WhateverController() { Configuration = new HttpConfiguration() };
sut.Configuration.Formatters.Add(new Mock<JsonMediaTypeFormatter>().Object);
sut.Request = new HttpRequestMessage();
Este tem cabeçalhos corretos definidos. Parece um pouco mais elegante.
public JsonResult<string> TestMethod()
{
return Json("your string or object");
}
para aqueles que usam OWIN
GlobalConfiguration.Configuration.Formatters.Clear();
GlobalConfiguration.Configuration.Formatters.Add(new JsonMediaTypeFormatter());
torna-se (em Startup.cs):
public void Configuration(IAppBuilder app)
{
OwinConfiguration = new HttpConfiguration();
ConfigureOAuth(app);
OwinConfiguration.Formatters.Clear();
OwinConfiguration.Formatters.Add(new DynamicJsonMediaTypeFormatter());
[...]
}
GlobalConfiguration.Configuration.Formatters