ClayMetaObject.cs 9.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Dynamic;
  4. using System.Linq;
  5. using System.Linq.Expressions;
  6. using System.Reflection;
  7. using Masuit.Tools.Dynamics.Implementation;
  8. namespace Masuit.Tools.Dynamics;
  9. internal class ClayMetaObject : DynamicMetaObject
  10. {
  11. private static readonly MethodInfo ClayBehaviorInvokeMember = typeof(IClayBehavior).GetMethod("InvokeMember");
  12. private static readonly MethodInfo ClayBehaviorGetMember = typeof(IClayBehavior).GetMethod("GetMember");
  13. private static readonly MethodInfo ClayBehaviorSetMember = typeof(IClayBehavior).GetMethod("SetMember");
  14. private static readonly MethodInfo ClayBehaviorGetIndex = typeof(IClayBehavior).GetMethod("GetIndex");
  15. private static readonly MethodInfo ClayBehaviorSetIndex = typeof(IClayBehavior).GetMethod("SetIndex");
  16. private static readonly MethodInfo ClayBehaviorBinaryOperation = typeof(IClayBehavior).GetMethod("BinaryOperation");
  17. private static readonly MethodInfo ClayBehaviorConvert = typeof(IClayBehavior).GetMethod("Convert");
  18. private static readonly MethodInfo ClayBehaviorInvokeMemberMissing = typeof(IClayBehavior).GetMethod("InvokeMemberMissing");
  19. private static readonly MethodInfo ClayBehaviorGetMemberMissing = typeof(IClayBehavior).GetMethod("GetMemberMissing");
  20. private static readonly MethodInfo ClayBehaviorSetMemberMissing = typeof(IClayBehavior).GetMethod("SetMemberMissing");
  21. private static readonly MethodInfo ClayBehaviorConvertMissing = typeof(IClayBehavior).GetMethod("ConvertMissing");
  22. public ClayMetaObject(object value, Expression expression) : base(expression, BindingRestrictions.Empty, value)
  23. {
  24. }
  25. public ClayMetaObject(object value, Expression expression, Func<Expression, Expression> getClayBehavior) : this(value, expression)
  26. {
  27. _getClayBehavior = getClayBehavior;
  28. }
  29. private Expression GetLimitedSelf()
  30. {
  31. if (Expression.Type == LimitType || Expression.Type.IsEquivalentTo(LimitType))
  32. {
  33. return Expression;
  34. }
  35. return Expression.Convert(Expression, LimitType);
  36. }
  37. private readonly Func<Expression, Expression> _getClayBehavior = expr => Expression.Property(Expression.Convert(expr, typeof(IClayBehaviorProvider)), "Behavior");
  38. protected virtual Expression GetClayBehavior()
  39. {
  40. return _getClayBehavior(Expression);
  41. }
  42. public override DynamicMetaObject BindGetMember(GetMemberBinder binder)
  43. {
  44. var binderDefault = binder.FallbackGetMember(this);
  45. var missingLambda = Expression.Lambda(Expression.Call(GetClayBehavior(), ClayBehaviorGetMemberMissing, Expression.Lambda(binderDefault.Expression), GetLimitedSelf(), Expression.Constant(binder.Name, typeof(string))));
  46. var call = Expression.Call(GetClayBehavior(), ClayBehaviorGetMember, missingLambda, GetLimitedSelf(), Expression.Constant(binder.Name, typeof(string)));
  47. var dynamicSuggestion = new DynamicMetaObject(call, BindingRestrictions.GetTypeRestriction(Expression, LimitType).Merge(binderDefault.Restrictions));
  48. return binder.FallbackGetMember(this, dynamicSuggestion);
  49. }
  50. /// <inheritdoc />
  51. public override DynamicMetaObject BindSetMember(SetMemberBinder binder, DynamicMetaObject value)
  52. {
  53. var binderDefault = binder.FallbackSetMember(this, value);
  54. var missingLambda = Expression.Lambda(Expression.Call(GetClayBehavior(), ClayBehaviorSetMemberMissing, Expression.Lambda(binderDefault.Expression), GetLimitedSelf(), Expression.Constant(binder.Name, typeof(string)), Expression.Convert(value.Expression, typeof(object))));
  55. var call = Expression.Call(GetClayBehavior(), ClayBehaviorSetMember, missingLambda, GetLimitedSelf(), Expression.Constant(binder.Name, typeof(string)), Expression.Convert(value.Expression, typeof(object)));
  56. var dynamicSuggestion = new DynamicMetaObject(call, BindingRestrictions.GetTypeRestriction(Expression, LimitType).Merge(binderDefault.Restrictions));
  57. return binder.FallbackSetMember(this, value, dynamicSuggestion);
  58. }
  59. /// <inheritdoc />
  60. public override DynamicMetaObject BindInvokeMember(InvokeMemberBinder binder, DynamicMetaObject[] args)
  61. {
  62. var argValues = Expression.NewArrayInit(typeof(object), args.Select(x => Expression.Convert(x.Expression, typeof(Object))));
  63. var argNames = Expression.Constant(binder.CallInfo.ArgumentNames, typeof(IEnumerable<string>));
  64. var argNamedEnumerable = Expression.Call(typeof(Arguments).GetMethod("From"), argValues, argNames);
  65. var binderDefault = binder.FallbackInvokeMember(this, args);
  66. var missingLambda = Expression.Lambda(Expression.Call(GetClayBehavior(), ClayBehaviorInvokeMemberMissing, Expression.Lambda(binderDefault.Expression), GetLimitedSelf(), Expression.Constant(binder.Name, typeof(string)), argNamedEnumerable));
  67. var call = Expression.Call(GetClayBehavior(), ClayBehaviorInvokeMember, missingLambda, GetLimitedSelf(), Expression.Constant(binder.Name, typeof(string)), argNamedEnumerable);
  68. var dynamicSuggestion = new DynamicMetaObject(call, BindingRestrictions.GetTypeRestriction(Expression, LimitType).Merge(binderDefault.Restrictions));
  69. return binder.FallbackInvokeMember(this, args, dynamicSuggestion);
  70. }
  71. /// <inheritdoc />
  72. public override DynamicMetaObject BindConvert(ConvertBinder binder)
  73. {
  74. var binderDefault = binder.FallbackConvert(this);
  75. var missingLambda = Expression.Lambda(Expression.Call(GetClayBehavior(), ClayBehaviorConvertMissing, Expression.Lambda(Expression.Convert(binderDefault.Expression, typeof(object))), GetLimitedSelf(), Expression.Constant(binder.Type, typeof(Type)), Expression.Constant(binder.Explicit, typeof(bool))));
  76. var call = Expression.Call(GetClayBehavior(), ClayBehaviorConvert, missingLambda, GetLimitedSelf(), Expression.Constant(binder.Type, typeof(Type)), Expression.Constant(binder.Explicit, typeof(bool)));
  77. var convertedCall = Expression.Convert(call, binder.ReturnType);
  78. return new DynamicMetaObject(convertedCall, BindingRestrictions.GetTypeRestriction(Expression, LimitType).Merge(binderDefault.Restrictions));
  79. }
  80. /// <inheritdoc />
  81. public override DynamicMetaObject BindDeleteMember(DeleteMemberBinder binder)
  82. {
  83. throw new NotImplementedException();
  84. }
  85. /// <inheritdoc />
  86. public override DynamicMetaObject BindGetIndex(GetIndexBinder binder, DynamicMetaObject[] indexes)
  87. {
  88. var a2 = Expression.NewArrayInit(typeof(string), indexes.Select(x => Expression.Convert(x.Expression, typeof(string))));
  89. var binderFallback = binder.FallbackGetIndex(this, indexes);
  90. var call = Expression.Call(GetClayBehavior(), ClayBehaviorGetIndex, Expression.Lambda(binderFallback.Expression), GetLimitedSelf(), a2);
  91. return new DynamicMetaObject(call, BindingRestrictions.GetTypeRestriction(Expression, LimitType).Merge(binderFallback.Restrictions));
  92. }
  93. /// <inheritdoc />
  94. public override DynamicMetaObject BindSetIndex(SetIndexBinder binder, DynamicMetaObject[] indexes, DynamicMetaObject value)
  95. {
  96. var a2 = Expression.NewArrayInit(typeof(string), indexes.Select(x => Expression.Convert(x.Expression, typeof(string))));
  97. var binderFallback = binder.FallbackSetIndex(this, indexes, value);
  98. var call = Expression.Call(GetClayBehavior(), ClayBehaviorSetIndex, Expression.Lambda(binderFallback.Expression), GetLimitedSelf(), a2, Expression.Convert(value.Expression, typeof(object)));
  99. return new DynamicMetaObject(call, BindingRestrictions.GetTypeRestriction(Expression, LimitType).Merge(binderFallback.Restrictions));
  100. }
  101. /// <inheritdoc />
  102. public override DynamicMetaObject BindDeleteIndex(DeleteIndexBinder binder, DynamicMetaObject[] indexes)
  103. {
  104. throw new NotImplementedException();
  105. }
  106. /// <inheritdoc />
  107. public override DynamicMetaObject BindInvoke(InvokeBinder binder, DynamicMetaObject[] args)
  108. {
  109. var argValues = Expression.NewArrayInit(typeof(object), args.Select(x => Expression.Convert(x.Expression, typeof(Object))));
  110. var argNames = Expression.Constant(binder.CallInfo.ArgumentNames, typeof(IEnumerable<string>));
  111. var argNamedEnumerable = Expression.Call(typeof(Arguments).GetMethod("From"), argValues, argNames);
  112. var binderFallback = binder.FallbackInvoke(this, args);
  113. var call = Expression.Call(GetClayBehavior(), ClayBehaviorInvokeMember, Expression.Lambda(binderFallback.Expression), GetLimitedSelf(), Expression.Constant(null, typeof(string)), argNamedEnumerable);
  114. return new DynamicMetaObject(call, BindingRestrictions.GetTypeRestriction(Expression, LimitType).Merge(binderFallback.Restrictions));
  115. }
  116. /// <inheritdoc />
  117. public override DynamicMetaObject BindCreateInstance(CreateInstanceBinder binder, DynamicMetaObject[] args)
  118. {
  119. throw new NotImplementedException();
  120. }
  121. /// <inheritdoc />
  122. public override DynamicMetaObject BindUnaryOperation(UnaryOperationBinder binder)
  123. {
  124. throw new NotImplementedException();
  125. }
  126. /// <inheritdoc />
  127. public override DynamicMetaObject BindBinaryOperation(BinaryOperationBinder binder, DynamicMetaObject arg)
  128. {
  129. var binderFallback = binder.FallbackBinaryOperation(this, arg);
  130. var call = Expression.Call(GetClayBehavior(), ClayBehaviorBinaryOperation, Expression.Lambda(binderFallback.Expression), GetLimitedSelf(), Expression.Constant(binder.Operation, typeof(ExpressionType)), Expression.Convert(arg.Expression, typeof(object)));
  131. return new DynamicMetaObject(call, BindingRestrictions.GetTypeRestriction(Expression, LimitType));
  132. }
  133. /// <inheritdoc />
  134. public override IEnumerable<string> GetDynamicMemberNames()
  135. {
  136. throw new NotImplementedException();
  137. }
  138. }