ClayMetaObject.cs 9.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155
  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(object value, Expression expression) : DynamicMetaObject(expression, BindingRestrictions.Empty, value)
  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, Func<Expression, Expression> getClayBehavior) : this(value, expression)
  23. {
  24. _getClayBehavior = getClayBehavior;
  25. }
  26. private Expression GetLimitedSelf()
  27. {
  28. if (Expression.Type == LimitType || Expression.Type.IsEquivalentTo(LimitType))
  29. {
  30. return Expression;
  31. }
  32. return Expression.Convert(Expression, LimitType);
  33. }
  34. private readonly Func<Expression, Expression> _getClayBehavior = expr => Expression.Property(Expression.Convert(expr, typeof(IClayBehaviorProvider)), "Behavior");
  35. protected virtual Expression GetClayBehavior()
  36. {
  37. return _getClayBehavior(Expression);
  38. }
  39. public override DynamicMetaObject BindGetMember(GetMemberBinder binder)
  40. {
  41. var binderDefault = binder.FallbackGetMember(this);
  42. var missingLambda = Expression.Lambda(Expression.Call(GetClayBehavior(), ClayBehaviorGetMemberMissing, Expression.Lambda(binderDefault.Expression), GetLimitedSelf(), Expression.Constant(binder.Name, typeof(string))));
  43. var call = Expression.Call(GetClayBehavior(), ClayBehaviorGetMember, missingLambda, GetLimitedSelf(), Expression.Constant(binder.Name, typeof(string)));
  44. var dynamicSuggestion = new DynamicMetaObject(call, BindingRestrictions.GetTypeRestriction(Expression, LimitType).Merge(binderDefault.Restrictions));
  45. return binder.FallbackGetMember(this, dynamicSuggestion);
  46. }
  47. /// <inheritdoc />
  48. public override DynamicMetaObject BindSetMember(SetMemberBinder binder, DynamicMetaObject value)
  49. {
  50. var binderDefault = binder.FallbackSetMember(this, value);
  51. 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))));
  52. var call = Expression.Call(GetClayBehavior(), ClayBehaviorSetMember, missingLambda, GetLimitedSelf(), Expression.Constant(binder.Name, typeof(string)), Expression.Convert(value.Expression, typeof(object)));
  53. var dynamicSuggestion = new DynamicMetaObject(call, BindingRestrictions.GetTypeRestriction(Expression, LimitType).Merge(binderDefault.Restrictions));
  54. return binder.FallbackSetMember(this, value, dynamicSuggestion);
  55. }
  56. /// <inheritdoc />
  57. public override DynamicMetaObject BindInvokeMember(InvokeMemberBinder binder, DynamicMetaObject[] args)
  58. {
  59. var argValues = Expression.NewArrayInit(typeof(object), args.Select(x => Expression.Convert(x.Expression, typeof(Object))));
  60. var argNames = Expression.Constant(binder.CallInfo.ArgumentNames, typeof(IEnumerable<string>));
  61. var argNamedEnumerable = Expression.Call(typeof(Arguments).GetMethod("From"), argValues, argNames);
  62. var binderDefault = binder.FallbackInvokeMember(this, args);
  63. var missingLambda = Expression.Lambda(Expression.Call(GetClayBehavior(), ClayBehaviorInvokeMemberMissing, Expression.Lambda(binderDefault.Expression), GetLimitedSelf(), Expression.Constant(binder.Name, typeof(string)), argNamedEnumerable));
  64. var call = Expression.Call(GetClayBehavior(), ClayBehaviorInvokeMember, missingLambda, GetLimitedSelf(), Expression.Constant(binder.Name, typeof(string)), argNamedEnumerable);
  65. var dynamicSuggestion = new DynamicMetaObject(call, BindingRestrictions.GetTypeRestriction(Expression, LimitType).Merge(binderDefault.Restrictions));
  66. return binder.FallbackInvokeMember(this, args, dynamicSuggestion);
  67. }
  68. /// <inheritdoc />
  69. public override DynamicMetaObject BindConvert(ConvertBinder binder)
  70. {
  71. var binderDefault = binder.FallbackConvert(this);
  72. 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))));
  73. var call = Expression.Call(GetClayBehavior(), ClayBehaviorConvert, missingLambda, GetLimitedSelf(), Expression.Constant(binder.Type, typeof(Type)), Expression.Constant(binder.Explicit, typeof(bool)));
  74. var convertedCall = Expression.Convert(call, binder.ReturnType);
  75. return new DynamicMetaObject(convertedCall, BindingRestrictions.GetTypeRestriction(Expression, LimitType).Merge(binderDefault.Restrictions));
  76. }
  77. /// <inheritdoc />
  78. public override DynamicMetaObject BindDeleteMember(DeleteMemberBinder binder)
  79. {
  80. throw new NotImplementedException();
  81. }
  82. /// <inheritdoc />
  83. public override DynamicMetaObject BindGetIndex(GetIndexBinder binder, DynamicMetaObject[] indexes)
  84. {
  85. var a2 = Expression.NewArrayInit(typeof(string), indexes.Select(x => Expression.Convert(x.Expression, typeof(string))));
  86. var binderFallback = binder.FallbackGetIndex(this, indexes);
  87. var call = Expression.Call(GetClayBehavior(), ClayBehaviorGetIndex, Expression.Lambda(binderFallback.Expression), GetLimitedSelf(), a2);
  88. return new DynamicMetaObject(call, BindingRestrictions.GetTypeRestriction(Expression, LimitType).Merge(binderFallback.Restrictions));
  89. }
  90. /// <inheritdoc />
  91. public override DynamicMetaObject BindSetIndex(SetIndexBinder binder, DynamicMetaObject[] indexes, DynamicMetaObject value)
  92. {
  93. var a2 = Expression.NewArrayInit(typeof(string), indexes.Select(x => Expression.Convert(x.Expression, typeof(string))));
  94. var binderFallback = binder.FallbackSetIndex(this, indexes, value);
  95. var call = Expression.Call(GetClayBehavior(), ClayBehaviorSetIndex, Expression.Lambda(binderFallback.Expression), GetLimitedSelf(), a2, Expression.Convert(value.Expression, typeof(object)));
  96. return new DynamicMetaObject(call, BindingRestrictions.GetTypeRestriction(Expression, LimitType).Merge(binderFallback.Restrictions));
  97. }
  98. /// <inheritdoc />
  99. public override DynamicMetaObject BindDeleteIndex(DeleteIndexBinder binder, DynamicMetaObject[] indexes)
  100. {
  101. throw new NotImplementedException();
  102. }
  103. /// <inheritdoc />
  104. public override DynamicMetaObject BindInvoke(InvokeBinder binder, DynamicMetaObject[] args)
  105. {
  106. var argValues = Expression.NewArrayInit(typeof(object), args.Select(x => Expression.Convert(x.Expression, typeof(Object))));
  107. var argNames = Expression.Constant(binder.CallInfo.ArgumentNames, typeof(IEnumerable<string>));
  108. var argNamedEnumerable = Expression.Call(typeof(Arguments).GetMethod("From"), argValues, argNames);
  109. var binderFallback = binder.FallbackInvoke(this, args);
  110. var call = Expression.Call(GetClayBehavior(), ClayBehaviorInvokeMember, Expression.Lambda(binderFallback.Expression), GetLimitedSelf(), Expression.Constant(null, typeof(string)), argNamedEnumerable);
  111. return new DynamicMetaObject(call, BindingRestrictions.GetTypeRestriction(Expression, LimitType).Merge(binderFallback.Restrictions));
  112. }
  113. /// <inheritdoc />
  114. public override DynamicMetaObject BindCreateInstance(CreateInstanceBinder binder, DynamicMetaObject[] args)
  115. {
  116. throw new NotImplementedException();
  117. }
  118. /// <inheritdoc />
  119. public override DynamicMetaObject BindUnaryOperation(UnaryOperationBinder binder)
  120. {
  121. throw new NotImplementedException();
  122. }
  123. /// <inheritdoc />
  124. public override DynamicMetaObject BindBinaryOperation(BinaryOperationBinder binder, DynamicMetaObject arg)
  125. {
  126. var binderFallback = binder.FallbackBinaryOperation(this, arg);
  127. var call = Expression.Call(GetClayBehavior(), ClayBehaviorBinaryOperation, Expression.Lambda(binderFallback.Expression), GetLimitedSelf(), Expression.Constant(binder.Operation, typeof(ExpressionType)), Expression.Convert(arg.Expression, typeof(object)));
  128. return new DynamicMetaObject(call, BindingRestrictions.GetTypeRestriction(Expression, LimitType));
  129. }
  130. /// <inheritdoc />
  131. public override IEnumerable<string> GetDynamicMemberNames()
  132. {
  133. throw new NotImplementedException();
  134. }
  135. }