MethodInfoFunctionInstance.cs 5.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Globalization;
  4. using System.Linq;
  5. using System.Linq.Expressions;
  6. using System.Reflection;
  7. using Jint.Native;
  8. using Jint.Native.Array;
  9. using Jint.Native.Function;
  10. namespace Jint.Runtime.Interop
  11. {
  12. public sealed class MethodInfoFunctionInstance : FunctionInstance
  13. {
  14. private readonly MethodInfo[] _methods;
  15. public MethodInfoFunctionInstance(Engine engine, MethodInfo[] methods)
  16. : base(engine, null, null, false)
  17. {
  18. _methods = methods;
  19. Prototype = engine.Function.PrototypeObject;
  20. }
  21. public override JsValue Call(JsValue thisObject, JsValue[] arguments)
  22. {
  23. return Invoke(_methods, thisObject, arguments);
  24. }
  25. public JsValue Invoke(MethodInfo[] methodInfos, JsValue thisObject, JsValue[] jsArguments)
  26. {
  27. var arguments = ProcessParamsArrays(jsArguments, methodInfos);
  28. var methods = TypeConverter.FindBestMatch(Engine, methodInfos, arguments).ToList();
  29. var converter = Engine.ClrTypeConverter;
  30. foreach (var method in methods)
  31. {
  32. var parameters = new object[arguments.Length];
  33. var argumentsMatch = true;
  34. for (var i = 0; i < arguments.Length; i++)
  35. {
  36. var parameterType = method.GetParameters()[i].ParameterType;
  37. if (parameterType == typeof(JsValue))
  38. {
  39. parameters[i] = arguments[i];
  40. }
  41. else if (parameterType == typeof(JsValue[]) && arguments[i].IsArray())
  42. {
  43. // Handle specific case of F(params JsValue[])
  44. var arrayInstance = arguments[i].AsArray();
  45. var len = TypeConverter.ToInt32(arrayInstance.Get("length"));
  46. var result = new JsValue[len];
  47. for (var k = 0; k < len; k++)
  48. {
  49. var pk = TypeConverter.ToString(k);
  50. result[k] = arrayInstance.HasProperty(pk)
  51. ? arrayInstance.Get(pk)
  52. : JsValue.Undefined;
  53. }
  54. parameters[i] = result;
  55. }
  56. else
  57. {
  58. if (!converter.TryConvert(arguments[i].ToObject(), parameterType, CultureInfo.InvariantCulture, out parameters[i]))
  59. {
  60. argumentsMatch = false;
  61. break;
  62. }
  63. var lambdaExpression = parameters[i] as LambdaExpression;
  64. if (lambdaExpression != null)
  65. {
  66. parameters[i] = lambdaExpression.Compile();
  67. }
  68. }
  69. }
  70. if (!argumentsMatch)
  71. {
  72. continue;
  73. }
  74. // todo: cache method info
  75. try
  76. {
  77. return JsValue.FromObject(Engine, method.Invoke(thisObject.ToObject(), parameters.ToArray()));
  78. }
  79. catch (TargetInvocationException exception)
  80. {
  81. var meaningfulException = exception.InnerException ?? exception;
  82. var handler = Engine.Options._ClrExceptionsHandler;
  83. if (handler != null && handler(meaningfulException))
  84. {
  85. throw new JavaScriptException(Engine.Error, meaningfulException.Message);
  86. }
  87. throw meaningfulException;
  88. }
  89. }
  90. throw new JavaScriptException(Engine.TypeError, "No public methods with the specified arguments were found.");
  91. }
  92. /// <summary>
  93. /// Reduces a flat list of parameters to a params array
  94. /// </summary>
  95. private JsValue[] ProcessParamsArrays(JsValue[] jsArguments, IEnumerable<MethodInfo> methodInfos)
  96. {
  97. foreach (var methodInfo in methodInfos)
  98. {
  99. var parameters = methodInfo.GetParameters();
  100. if (!parameters.Any(p => p.HasAttribute<ParamArrayAttribute>()))
  101. continue;
  102. var nonParamsArgumentsCount = parameters.Length - 1;
  103. if (jsArguments.Length < nonParamsArgumentsCount)
  104. continue;
  105. var newArgumentsCollection = jsArguments.Take(nonParamsArgumentsCount).ToList();
  106. var argsToTransform = jsArguments.Skip(nonParamsArgumentsCount).ToList();
  107. if (argsToTransform.Count == 1 && argsToTransform.FirstOrDefault().IsArray())
  108. continue;
  109. var jsArray = Engine.Array.Construct(Arguments.Empty);
  110. Engine.Array.PrototypeObject.Push(jsArray, argsToTransform.ToArray());
  111. newArgumentsCollection.Add(jsArray.JsValue);
  112. return newArgumentsCollection.ToArray();
  113. }
  114. return jsArguments;
  115. }
  116. }
  117. }