InterfaceProxyBehavior.cs 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289
  1. using System;
  2. using System.Collections.Concurrent;
  3. using System.Collections.Generic;
  4. using System.Dynamic;
  5. using System.Linq;
  6. using System.Linq.Expressions;
  7. using System.Reflection;
  8. using Castle.Core.Interceptor;
  9. using Castle.DynamicProxy;
  10. using Microsoft.CSharp.RuntimeBinder;
  11. using Binder = Microsoft.CSharp.RuntimeBinder.Binder;
  12. namespace ClaySharp.Behaviors {
  13. public class InterfaceProxyBehavior : ClayBehavior {
  14. private static readonly IProxyBuilder ProxyBuilder = new DefaultProxyBuilder();
  15. static readonly MethodInfo DynamicMetaObjectProviderGetMetaObject = typeof(IDynamicMetaObjectProvider).GetMethod("GetMetaObject");
  16. public override object ConvertMissing(Func<object> proceed, object self, Type type, bool isExplicit) {
  17. if (type.IsInterface && type != typeof(IDynamicMetaObjectProvider)) {
  18. var proxyType = ProxyBuilder.CreateInterfaceProxyTypeWithoutTarget(
  19. type,
  20. new[] { typeof(IDynamicMetaObjectProvider) },
  21. ProxyGenerationOptions.Default);
  22. var interceptors = new IInterceptor[] { new Interceptor(self) };
  23. var proxy = Activator.CreateInstance(proxyType, new object[] { interceptors, self });
  24. return proxy;
  25. }
  26. return proceed();
  27. }
  28. class Interceptor : IInterceptor {
  29. public object Self { get; private set; }
  30. public Interceptor(object self) {
  31. Self = self;
  32. }
  33. public void Intercept(IInvocation invocation) {
  34. if (invocation.Method == DynamicMetaObjectProviderGetMetaObject) {
  35. var expression = (Expression)invocation.Arguments.Single();
  36. invocation.ReturnValue = new ForwardingMetaObject(
  37. expression,
  38. BindingRestrictions.Empty,
  39. invocation.Proxy,
  40. (IDynamicMetaObjectProvider)Self,
  41. exprProxy => Expression.Field(exprProxy, "__target"));
  42. return;
  43. }
  44. var invoker = BindInvoker(invocation);
  45. invoker(invocation);
  46. if (invocation.ReturnValue != null &&
  47. !invocation.Method.ReturnType.IsAssignableFrom(invocation.ReturnValue.GetType()) &&
  48. invocation.ReturnValue is IClayBehaviorProvider) {
  49. var returnValueBehavior = ((IClayBehaviorProvider) invocation.ReturnValue).Behavior;
  50. invocation.ReturnValue = returnValueBehavior.Convert(
  51. () => returnValueBehavior.ConvertMissing(
  52. () => invocation.ReturnValue,
  53. invocation.ReturnValue,
  54. invocation.Method.ReturnType,
  55. false),
  56. invocation.ReturnValue,
  57. invocation.Method.ReturnType,
  58. false);
  59. }
  60. }
  61. static readonly ConcurrentDictionary<MethodInfo, Action<IInvocation>> Invokers = new ConcurrentDictionary<MethodInfo, Action<IInvocation>>();
  62. private static Action<IInvocation> BindInvoker(IInvocation invocation) {
  63. return Invokers.GetOrAdd(invocation.Method, CompileInvoker);
  64. }
  65. private static Action<IInvocation> CompileInvoker(MethodInfo method) {
  66. var methodParameters = method.GetParameters();
  67. var invocationParameter = Expression.Parameter(typeof(IInvocation), "invocation");
  68. var targetAndArgumentInfos = Pack(
  69. CSharpArgumentInfo.Create(CSharpArgumentInfoFlags.None, null),
  70. methodParameters.Select(
  71. mp => CSharpArgumentInfo.Create(CSharpArgumentInfoFlags.NamedArgument, mp.Name)));
  72. var targetAndArguments = Pack<Expression>(
  73. Expression.Property(invocationParameter, invocationParameter.Type, "InvocationTarget"),
  74. methodParameters.Select(
  75. (mp, index) =>
  76. Expression.Convert(
  77. Expression.ArrayIndex(
  78. Expression.Property(invocationParameter, invocationParameter.Type,
  79. "Arguments"),
  80. Expression.Constant(index)), mp.ParameterType)));
  81. Expression body = null;
  82. if (method.IsSpecialName) {
  83. if (body == null && method.Name.Equals("get_Item")) {
  84. body = Expression.Dynamic(
  85. Binder.GetIndex(
  86. CSharpBinderFlags.InvokeSpecialName,
  87. typeof(object),
  88. targetAndArgumentInfos),
  89. typeof(object),
  90. targetAndArguments);
  91. }
  92. if (body == null && method.Name.Equals("set_Item")) {
  93. var targetAndArgumentInfosWithoutTheNameValue = Pack(
  94. CSharpArgumentInfo.Create(CSharpArgumentInfoFlags.None, null),
  95. methodParameters.Select(
  96. mp => mp.Name == "value" ? CSharpArgumentInfo.Create(CSharpArgumentInfoFlags.None, null) : CSharpArgumentInfo.Create(CSharpArgumentInfoFlags.NamedArgument, mp.Name)));
  97. body = Expression.Dynamic(
  98. Binder.SetIndex(
  99. CSharpBinderFlags.InvokeSpecialName,
  100. typeof(object),
  101. targetAndArgumentInfosWithoutTheNameValue),
  102. typeof(object),
  103. targetAndArguments);
  104. }
  105. if (body == null && method.Name.StartsWith("get_")) {
  106. // Build lambda containing the following call site:
  107. // (IInvocation invocation) => {
  108. // invocation.ReturnValue = (object) ((dynamic)invocation.InvocationTarget).{method.Name};
  109. // }
  110. body = Expression.Dynamic(
  111. Binder.GetMember(
  112. CSharpBinderFlags.InvokeSpecialName,
  113. method.Name.Substring("get_".Length),
  114. typeof(object),
  115. targetAndArgumentInfos),
  116. typeof(object),
  117. targetAndArguments);
  118. }
  119. if (body == null && method.Name.StartsWith("set_")) {
  120. body = Expression.Dynamic(
  121. Binder.SetMember(
  122. CSharpBinderFlags.InvokeSpecialName,
  123. method.Name.Substring("set_".Length),
  124. typeof(object),
  125. targetAndArgumentInfos),
  126. typeof(object),
  127. targetAndArguments);
  128. }
  129. }
  130. if (body == null) {
  131. // Build lambda containing the following call site:
  132. // (IInvocation invocation) => {
  133. // invocation.ReturnValue = (object) ((dynamic)invocation.InvocationTarget).{method.Name}(
  134. // {methodParameters[*].Name}: ({methodParameters[*].Type})invocation.Arguments[*],
  135. // ...);
  136. // }
  137. body = Expression.Dynamic(
  138. Binder.InvokeMember(
  139. CSharpBinderFlags.None,
  140. method.Name,
  141. null,
  142. typeof(object),
  143. targetAndArgumentInfos),
  144. typeof(object),
  145. targetAndArguments);
  146. }
  147. if (body != null && method.ReturnType != typeof(void)) {
  148. body = Expression.Assign(
  149. Expression.Property(invocationParameter, invocationParameter.Type, "ReturnValue"),
  150. Expression.Convert(body, typeof(object)));
  151. }
  152. var lambda = Expression.Lambda<Action<IInvocation>>(body, invocationParameter);
  153. return lambda.Compile();
  154. }
  155. }
  156. static IEnumerable<T> Pack<T>(T t1) {
  157. if (!Equals(t1, default(T)))
  158. yield return t1;
  159. }
  160. static IEnumerable<T> Pack<T>(T t1, IEnumerable<T> t2) {
  161. if (!Equals(t1, default(T)))
  162. yield return t1;
  163. foreach (var t in t2)
  164. yield return t;
  165. }
  166. static IEnumerable<T> Pack<T>(T t1, IEnumerable<T> t2, T t3) {
  167. if (!Equals(t1, default(T)))
  168. yield return t1;
  169. foreach (var t in t2)
  170. yield return t;
  171. if (!Equals(t3, default(T)))
  172. yield return t3;
  173. }
  174. /// <summary>
  175. /// Based on techniques discussed by Tomáš Matoušek
  176. /// at http://blog.tomasm.net/2009/11/07/forwarding-meta-object/
  177. /// </summary>
  178. public sealed class ForwardingMetaObject : DynamicMetaObject {
  179. private readonly DynamicMetaObject _metaForwardee;
  180. public ForwardingMetaObject(Expression expression, BindingRestrictions restrictions, object forwarder,
  181. IDynamicMetaObjectProvider forwardee, Func<Expression, Expression> forwardeeGetter)
  182. : base(expression, restrictions, forwarder) {
  183. // We'll use forwardee's meta-object to bind dynamic operations.
  184. _metaForwardee = forwardee.GetMetaObject(
  185. forwardeeGetter(
  186. Expression.Convert(expression, forwarder.GetType()) // [1]
  187. )
  188. );
  189. }
  190. // Restricts the target object's type to TForwarder.
  191. // The meta-object we are forwarding to assumes that it gets an instance of TForwarder (see [1]).
  192. // We need to ensure that the assumption holds.
  193. private DynamicMetaObject AddRestrictions(DynamicMetaObject result) {
  194. var restricted = new DynamicMetaObject(
  195. result.Expression,
  196. BindingRestrictions.GetTypeRestriction(Expression, Value.GetType()).Merge(result.Restrictions),
  197. _metaForwardee.Value
  198. );
  199. return restricted;
  200. }
  201. // Forward all dynamic operations or some of them as needed //
  202. public override DynamicMetaObject BindGetMember(GetMemberBinder binder) {
  203. return AddRestrictions(_metaForwardee.BindGetMember(binder));
  204. }
  205. public override DynamicMetaObject BindSetMember(SetMemberBinder binder, DynamicMetaObject value) {
  206. return AddRestrictions(_metaForwardee.BindSetMember(binder, value));
  207. }
  208. public override DynamicMetaObject BindDeleteMember(DeleteMemberBinder binder) {
  209. return AddRestrictions(_metaForwardee.BindDeleteMember(binder));
  210. }
  211. public override DynamicMetaObject BindGetIndex(GetIndexBinder binder, DynamicMetaObject[] indexes) {
  212. return AddRestrictions(_metaForwardee.BindGetIndex(binder, indexes));
  213. }
  214. public override DynamicMetaObject BindSetIndex(SetIndexBinder binder, DynamicMetaObject[] indexes, DynamicMetaObject value) {
  215. return AddRestrictions(_metaForwardee.BindSetIndex(binder, indexes, value));
  216. }
  217. public override DynamicMetaObject BindDeleteIndex(DeleteIndexBinder binder, DynamicMetaObject[] indexes) {
  218. return AddRestrictions(_metaForwardee.BindDeleteIndex(binder, indexes));
  219. }
  220. public override DynamicMetaObject BindInvokeMember(InvokeMemberBinder binder, DynamicMetaObject[] args) {
  221. return AddRestrictions(_metaForwardee.BindInvokeMember(binder, args));
  222. }
  223. public override DynamicMetaObject BindInvoke(InvokeBinder binder, DynamicMetaObject[] args) {
  224. return AddRestrictions(_metaForwardee.BindInvoke(binder, args));
  225. }
  226. public override DynamicMetaObject BindCreateInstance(CreateInstanceBinder binder, DynamicMetaObject[] args) {
  227. return AddRestrictions(_metaForwardee.BindCreateInstance(binder, args));
  228. }
  229. public override DynamicMetaObject BindUnaryOperation(UnaryOperationBinder binder) {
  230. return AddRestrictions(_metaForwardee.BindUnaryOperation(binder));
  231. }
  232. public override DynamicMetaObject BindBinaryOperation(BinaryOperationBinder binder, DynamicMetaObject arg) {
  233. return AddRestrictions(_metaForwardee.BindBinaryOperation(binder, arg));
  234. }
  235. public override DynamicMetaObject BindConvert(ConvertBinder binder) {
  236. return AddRestrictions(_metaForwardee.BindConvert(binder));
  237. }
  238. }
  239. }
  240. }