You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

165 lines
6.3 KiB

  1. #region License
  2. // Copyright (c) 2007 James Newton-King
  3. //
  4. // Permission is hereby granted, free of charge, to any person
  5. // obtaining a copy of this software and associated documentation
  6. // files (the "Software"), to deal in the Software without
  7. // restriction, including without limitation the rights to use,
  8. // copy, modify, merge, publish, distribute, sublicense, and/or sell
  9. // copies of the Software, and to permit persons to whom the
  10. // Software is furnished to do so, subject to the following
  11. // conditions:
  12. //
  13. // The above copyright notice and this permission notice shall be
  14. // included in all copies or substantial portions of the Software.
  15. //
  16. // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
  17. // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES
  18. // OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
  19. // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
  20. // HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
  21. // WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
  22. // FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
  23. // OTHER DEALINGS IN THE SOFTWARE.
  24. #endregion
  25. using Newtonsoft.Json.Serialization;
  26. using System;
  27. using System.Collections.Generic;
  28. using System.Reflection;
  29. using System.Globalization;
  30. #if NET20
  31. using Newtonsoft.Json.Utilities.LinqBridge;
  32. #else
  33. using System.Linq;
  34. #endif
  35. namespace Newtonsoft.Json.Utilities
  36. {
  37. internal class ReflectionMember
  38. {
  39. public Type MemberType { get; set; }
  40. public Func<object, object> Getter { get; set; }
  41. public Action<object, object> Setter { get; set; }
  42. }
  43. internal class ReflectionObject
  44. {
  45. public ObjectConstructor<object> Creator { get; }
  46. public IDictionary<string, ReflectionMember> Members { get; }
  47. private ReflectionObject(ObjectConstructor<object> creator)
  48. {
  49. Members = new Dictionary<string, ReflectionMember>();
  50. Creator = creator;
  51. }
  52. public object GetValue(object target, string member)
  53. {
  54. Func<object, object> getter = Members[member].Getter;
  55. return getter(target);
  56. }
  57. public void SetValue(object target, string member, object value)
  58. {
  59. Action<object, object> setter = Members[member].Setter;
  60. setter(target, value);
  61. }
  62. public Type GetType(string member)
  63. {
  64. return Members[member].MemberType;
  65. }
  66. public static ReflectionObject Create(Type t, params string[] memberNames)
  67. {
  68. return Create(t, null, memberNames);
  69. }
  70. public static ReflectionObject Create(Type t, MethodBase creator, params string[] memberNames)
  71. {
  72. ReflectionDelegateFactory delegateFactory = JsonTypeReflector.ReflectionDelegateFactory;
  73. ObjectConstructor<object> creatorConstructor = null;
  74. if (creator != null)
  75. {
  76. creatorConstructor = delegateFactory.CreateParameterizedConstructor(creator);
  77. }
  78. else
  79. {
  80. if (ReflectionUtils.HasDefaultConstructor(t, false))
  81. {
  82. Func<object> ctor = delegateFactory.CreateDefaultConstructor<object>(t);
  83. creatorConstructor = args => ctor();
  84. }
  85. }
  86. ReflectionObject d = new ReflectionObject(creatorConstructor);
  87. foreach (string memberName in memberNames)
  88. {
  89. MemberInfo[] members = t.GetMember(memberName, BindingFlags.Instance | BindingFlags.Public);
  90. if (members.Length != 1)
  91. {
  92. throw new ArgumentException("Expected a single member with the name '{0}'.".FormatWith(CultureInfo.InvariantCulture, memberName));
  93. }
  94. MemberInfo member = members.Single();
  95. ReflectionMember reflectionMember = new ReflectionMember();
  96. switch (member.MemberType())
  97. {
  98. case MemberTypes.Field:
  99. case MemberTypes.Property:
  100. if (ReflectionUtils.CanReadMemberValue(member, false))
  101. {
  102. reflectionMember.Getter = delegateFactory.CreateGet<object>(member);
  103. }
  104. if (ReflectionUtils.CanSetMemberValue(member, false, false))
  105. {
  106. reflectionMember.Setter = delegateFactory.CreateSet<object>(member);
  107. }
  108. break;
  109. case MemberTypes.Method:
  110. MethodInfo method = (MethodInfo)member;
  111. if (method.IsPublic)
  112. {
  113. ParameterInfo[] parameters = method.GetParameters();
  114. if (parameters.Length == 0 && method.ReturnType != typeof(void))
  115. {
  116. MethodCall<object, object> call = delegateFactory.CreateMethodCall<object>(method);
  117. reflectionMember.Getter = target => call(target);
  118. }
  119. else if (parameters.Length == 1 && method.ReturnType == typeof(void))
  120. {
  121. MethodCall<object, object> call = delegateFactory.CreateMethodCall<object>(method);
  122. reflectionMember.Setter = (target, arg) => call(target, arg);
  123. }
  124. }
  125. break;
  126. default:
  127. throw new ArgumentException("Unexpected member type '{0}' for member '{1}'.".FormatWith(CultureInfo.InvariantCulture, member.MemberType(), member.Name));
  128. }
  129. if (ReflectionUtils.CanReadMemberValue(member, false))
  130. {
  131. reflectionMember.Getter = delegateFactory.CreateGet<object>(member);
  132. }
  133. if (ReflectionUtils.CanSetMemberValue(member, false, false))
  134. {
  135. reflectionMember.Setter = delegateFactory.CreateSet<object>(member);
  136. }
  137. reflectionMember.MemberType = ReflectionUtils.GetMemberUnderlyingType(member);
  138. d.Members[memberName] = reflectionMember;
  139. }
  140. return d;
  141. }
  142. }
  143. }