I'm just getting started with MVC 4 Web API and I seem to be misunderstanding how it works.
Before Web API I had a simple MVC action method like this:
public JsonResult User()
{
return Json(new
{
firstName = "Joe",
lastName = "Jacobs",
email = "[email protected]"
});
}
That would work fine. In the new web API controller I am trying to do something similar.
public object User()
{
return new
{
firstName = "Joe",
lastName = "Jacobs",
email = "[email protected]"
}
}
This fails with a serialization error:
The 'ObjectContent`1' type failed to serialize the response body for content type 'application/xml; charset=utf-8'.
Inner exception:
Type '<>f__AnonymousType1`3[System.String,System.String,System.String]' cannot be serialized. Consider marking it with the DataContractAttribute attribute, and marking all of its members you want serialized with the DataMemberAttribute attribute. If the type is a collection, consider marking it with the CollectionDataContractAttribute. See the Microsoft .NET Framework documentation for other supported types.
What am I not understanding about returning anonymous type from the API controller?
If you look at the Fiddler (sample in here I use Firefox)
By default, request from browser will accepts application/xml, not application/json
But, you can create fake request from Fiddler by adding one header:
Accept: application/json
It will work
From the link:
The XML serializer does not support anonymous types or JObject instances. If you use these features for your JSON data, you should remove the XML formatter from the pipeline, as described later in this article.
How to remove XmlFormatter
:
var configuration = GlobalConfiguration.Configuration;
configuration.Formatters.Remove(configuration.Formatters.XmlFormatter);
You could also use the JsonMediaTypeFormatter so you do not need the JSONObject and related classes. Then you can return a dynamic type in your controller class.
public static void Register(HttpConfiguration config)
{
config.Formatters.Clear();
config.Formatters.Add(new JsonMediaTypeFormatter());
config.MapHttpAttributeRoutes();
}
public class YourController : ApiController
{
[HttpGet, Route("getstuff/{stuffId}")]
public dynamic Get(string stuffId)
{
var stuff = Model.Stuff.Get(stuffId);
return new {
success= stuff != null,
stuffId = stuff.Id,
name = stuff.Name
};
}
}
If you also want to support Jsonp you can inherit the JsonMediaTypeFormatter and create you own JsonpMediaTypeFormatter (which also can be found on stackoverflow: https://stackoverflow.com/a/12492552/1138266).
I was having a similar problem and the solution was to add a snippet to the
Global.asax.cs
file.
The snippet to add is as follows:
GlobalConfiguration.Configuration.Formatters.JsonFormatter.SerializerSettings
.ReferenceLoopHandling = Newtonsoft.Json.ReferenceLoopHandling.Ignore;
GlobalConfiguration.Configuration.Formatters
.Remove(GlobalConfiguration.Configuration.Formatters.XmlFormatter);
I also added the above lines of code to the very top of the Application_Start method which left the method looking like this:
protected void Application_Start()
{
GlobalConfiguration.Configuration.Formatters.JsonFormatter.SerializerSettings
.ReferenceLoopHandling = Newtonsoft.Json.ReferenceLoopHandling.Ignore;
GlobalConfiguration.Configuration.Formatters
.Remove(GlobalConfiguration.Configuration.Formatters.XmlFormatter);
AreaRegistration.RegisterAllAreas();
GlobalConfiguration.Configure(WebApiConfig.Register);
FilterConfig.RegisterGlobalFilters(GlobalFilters.Filters);
RouteConfig.RegisterRoutes(RouteTable.Routes);
BundleConfig.RegisterBundles(BundleTable.Bundles);
}
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With