尝试使用System.Web.Http.OData.Delta在ASP.NET Web API服务中实现PATCH方法,但似乎无法对类型为IEnumerable<T>
的属性应用更改。我正在使用Delta的最新Git版本(2012.2-rc-76-g8a73abe)。有没有人能够做到这一点?
考虑这种数据类型,它应该可以在对Web API服务的PATCH请求中更新:
public class Person
{
HashSet<int> _friends = new HashSet<int>();
public int Id { get; set; }
public string FirstName { get; set; }
public string LastName { get; set; }
public IEnumerable<int> Friends
{
get { return _friends; }
set
{
_friends = value != null ? new HashSet<int>(value) : new HashSet<int>();
}
}
public Person(int id, string firstName, string lastName)
{
Id = id;
FirstName = firstName;
LastName = lastName;
}
public Person()
{
}
}
此Web API方法通过Delta<Person>
实现对人员的修补:
public void Patch(int id, Delta<Person> delta)
{
var person = _persons.Single(p => p.Id == id);
delta.Patch(person);
}
如果我向服务发送带有以下JSON的PATCH请求,那么该人的Friends
属性应该更新,但不会发生:
{"Friends": [1]}
问题的关键在于如何使用此数据进行Delta更新Friends
。另请参阅discussion at CodePlex。
答案 0 :(得分:3)
问题可能是Deta会尝试将JSON的JArray
分配给您的Hashset<int>
如果你正在使用它来对抗JsonMEdiaTypeFormatter并且你内化了Delta代码(意味着你可以修改它),你必须做这样的事情(这很粗糙,但有效):
内部bool TrySetPropertyValue(string name, object value)
Delta<T>
,返回false:
if (value != null && !cacheHit.Property.PropertyType.IsPrimitive && !isGuid && !cacheHit.Property.PropertyType.IsAssignableFrom(value.GetType()))
{
return false;
}
更改为:
var valueType = value.GetType();
var propertyType = cacheHit.Property.PropertyType;
if (value != null && !propertyType.IsPrimitive && !propertyType.IsAssignableFrom(valueType))
{
var array = value as JArray;
if (array == null)
return false;
var underlyingType = propertyType.GetGenericArguments().FirstOrDefault() ??
propertyType.GetElementType();
if (underlyingType == typeof(string))
{
var a = array.ToObject<IEnumerable<string>>();
value = Activator.CreateInstance(propertyType, a);
}
else if (underlyingType == typeof(int))
{
var a = array.ToObject<IEnumerable<int>>();
value = Activator.CreateInstance(propertyType, a);
}
else
return false;
}
这只适用于int
或string
的集合,但希望能够帮助您找到一个好的方向。
例如,现在你的模型可以有:
public class Team {
public HashSet<string> PlayerIds { get; set; }
public List<int> CoachIds { get; set; }
}
您可以成功更新它们。
答案 1 :(得分:1)
您可以覆盖Delta类的TrySetPropertyValue
方法,并使用JArray类:
public sealed class DeltaWithCollectionsSupport<T> : Delta<T> where T : class
{
public override bool TrySetPropertyValue(string name, object value)
{
var propertyInfo = typeof(T).GetProperty(name);
return propertyInfo != null && value is JArray array
? base.TrySetPropertyValue(name, array.ToObject(propertyInfo.PropertyType))
: base.TrySetPropertyValue(name, value);
}
}
答案 2 :(得分:0)
如果您使用的是ODataMediaTypeFormatter,那么这应该可行。不过有几点值得一提。 1)您的收藏必须是可设置的。 2)更换整个系列。你不能删除/添加单个元素。
此外,还有问题跟踪第1项 - '670 -Delta should support non-settable collections.'