MethodInfoFunctionInstance.cs 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125
  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 = k.ToString();
  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. return JsValue.FromObject(Engine, method.Invoke(thisObject.ToObject(), parameters.ToArray()));
  76. }
  77. throw new JavaScriptException(Engine.TypeError, "No public methods with the specified arguments were found.");
  78. }
  79. /// <summary>
  80. /// Reduces a flat list of parameters to a params array
  81. /// </summary>
  82. private JsValue[] ProcessParamsArrays(JsValue[] jsArguments, IEnumerable<MethodInfo> methodInfos)
  83. {
  84. foreach (var methodInfo in methodInfos)
  85. {
  86. var parameters = methodInfo.GetParameters();
  87. if (!parameters.Any(p => p.HasAttribute<ParamArrayAttribute>()))
  88. continue;
  89. var nonParamsArgumentsCount = parameters.Length - 1;
  90. if (jsArguments.Length < nonParamsArgumentsCount)
  91. continue;
  92. var newArgumentsCollection = jsArguments.Take(nonParamsArgumentsCount).ToList();
  93. var argsToTransform = jsArguments.Skip(nonParamsArgumentsCount).ToList();
  94. if (argsToTransform.Count == 1 && argsToTransform.FirstOrDefault().IsArray())
  95. continue;
  96. var jsArray = Engine.Array.Construct(Arguments.Empty);
  97. Engine.Array.PrototypeObject.Push(jsArray, argsToTransform.ToArray());
  98. newArgumentsCollection.Add(new JsValue(jsArray));
  99. return newArgumentsCollection.ToArray();
  100. }
  101. return jsArguments;
  102. }
  103. }
  104. }