2
0

BindingPatternAssignmentExpression.cs 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292
  1. using System.Collections.Generic;
  2. using Esprima.Ast;
  3. using Jint.Native;
  4. using Jint.Native.Array;
  5. using Jint.Native.Function;
  6. using Jint.Native.Iterator;
  7. using Jint.Runtime.Environments;
  8. namespace Jint.Runtime.Interpreter.Expressions
  9. {
  10. internal sealed class BindingPatternAssignmentExpression : JintExpression
  11. {
  12. private readonly BindingPattern _pattern;
  13. private readonly JintExpression _right;
  14. public BindingPatternAssignmentExpression(
  15. Engine engine,
  16. AssignmentExpression expression) : base(engine, expression)
  17. {
  18. _pattern = (BindingPattern) expression.Left;
  19. _right = Build(engine, expression.Right);
  20. }
  21. protected override object EvaluateInternal()
  22. {
  23. var rightValue = _right.GetValue();
  24. ProcessPatterns(_engine, _pattern, rightValue);
  25. return JsValue.Undefined;
  26. }
  27. internal static void ProcessPatterns(Engine engine, BindingPattern pattern, JsValue argument)
  28. {
  29. if (pattern is ArrayPattern ap)
  30. {
  31. HandleArrayPattern(engine, ap, argument);
  32. }
  33. else if (pattern is ObjectPattern op)
  34. {
  35. HandleObjectPattern(engine, op, argument);
  36. }
  37. }
  38. private static void HandleArrayPattern(Engine engine, ArrayPattern pattern, JsValue argument)
  39. {
  40. static bool ConsumeFromIterator(IIterator it, out JsValue value, out bool done)
  41. {
  42. var item = it.Next();
  43. value = JsValue.Undefined;
  44. done = false;
  45. if (item.TryGetValue(CommonProperties.Done, out var d) && d.AsBoolean())
  46. {
  47. done = true;
  48. return false;
  49. }
  50. if (!item.TryGetValue(CommonProperties.Value, out value))
  51. {
  52. return false;
  53. }
  54. return true;
  55. }
  56. var obj = TypeConverter.ToObject(engine, argument);
  57. ArrayOperations arrayOperations = null;
  58. IIterator iterator = null;
  59. if (obj.IsArrayLike)
  60. {
  61. arrayOperations = ArrayOperations.For(obj);
  62. }
  63. else
  64. {
  65. if (!obj.TryGetIterator(engine, out iterator))
  66. {
  67. ExceptionHelper.ThrowTypeError(engine);
  68. return;
  69. }
  70. }
  71. bool iterationEnded = false;
  72. for (uint i = 0; i < pattern.Elements.Count; i++)
  73. {
  74. var left = pattern.Elements[(int) i];
  75. if (left is Identifier identifier)
  76. {
  77. JsValue value;
  78. if (arrayOperations != null)
  79. {
  80. arrayOperations.TryGetValue(i, out value);
  81. }
  82. else
  83. {
  84. if (!ConsumeFromIterator(iterator, out value, out iterationEnded))
  85. {
  86. break;
  87. }
  88. }
  89. AssignToIdentifier(engine, identifier.Name, value);
  90. }
  91. else if (left is BindingPattern bindingPattern)
  92. {
  93. JsValue value;
  94. if (arrayOperations != null)
  95. {
  96. arrayOperations.TryGetValue(i, out value);
  97. }
  98. else
  99. {
  100. value = iterator.Next();
  101. }
  102. ProcessPatterns(engine, bindingPattern, value);
  103. }
  104. else if (left is IArrayPatternElement arrayPatternElement)
  105. {
  106. if (arrayPatternElement is RestElement restElement)
  107. {
  108. ArrayInstance array;
  109. if (arrayOperations != null)
  110. {
  111. var length = arrayOperations.GetLength();
  112. array = engine.Array.ConstructFast(length - i);
  113. for (uint j = i; j < length; ++j)
  114. {
  115. arrayOperations.TryGetValue(j, out var indexValue);
  116. array.SetIndexValue(j - i, indexValue, updateLength: false);
  117. }
  118. }
  119. else
  120. {
  121. array = engine.Array.ConstructFast(0);
  122. var protocol = new ArrayConstructor.ArrayProtocol(engine, obj, array, iterator, null);
  123. protocol.Execute();
  124. }
  125. if (restElement.Argument is Identifier leftIdentifier)
  126. {
  127. AssignToIdentifier(engine, leftIdentifier.Name, array);
  128. }
  129. else if (restElement.Argument is BindingPattern bp)
  130. {
  131. ProcessPatterns(engine, bp, array);
  132. }
  133. }
  134. else if (arrayPatternElement is AssignmentPattern assignmentPattern)
  135. {
  136. JsValue value;
  137. if (arrayOperations != null)
  138. {
  139. arrayOperations.TryGetValue(i, out value);
  140. }
  141. else
  142. {
  143. ConsumeFromIterator(iterator, out value, out iterationEnded);
  144. }
  145. if (value.IsUndefined()
  146. && assignmentPattern.Right is Expression expression)
  147. {
  148. var jintExpression = Build(engine, expression);
  149. value = jintExpression.GetValue();
  150. }
  151. if (assignmentPattern.Left is Identifier leftIdentifier)
  152. {
  153. if (assignmentPattern.Right.IsFunctionWithName())
  154. {
  155. ((FunctionInstance) value).SetFunctionName(new JsString(leftIdentifier.Name));
  156. }
  157. AssignToIdentifier(engine, leftIdentifier.Name, value);
  158. }
  159. else if (assignmentPattern.Left is BindingPattern bp)
  160. {
  161. ProcessPatterns(engine, bp, value);
  162. }
  163. }
  164. else
  165. {
  166. ExceptionHelper.ThrowArgumentOutOfRangeException("pattern", "Unable to determine how to handle array pattern element");
  167. break;
  168. }
  169. }
  170. }
  171. if (!iterationEnded)
  172. {
  173. iterator?.Return();
  174. }
  175. }
  176. private static void HandleObjectPattern(Engine engine, ObjectPattern pattern, JsValue argument)
  177. {
  178. var processedProperties = pattern.Properties.Count > 0 && pattern.Properties[pattern.Properties.Count - 1] is RestElement
  179. ? new HashSet<string>()
  180. : null;
  181. var source = TypeConverter.ToObject(engine, argument);
  182. for (var i = 0; i < pattern.Properties.Count; i++)
  183. {
  184. if (pattern.Properties[i] is Property p)
  185. {
  186. JsValue sourceKey;
  187. var identifier = p.Key as Identifier;
  188. if (identifier == null)
  189. {
  190. var keyExpression = Build(engine, p.Key);
  191. sourceKey = TypeConverter.ToPropertyKey(keyExpression.GetValue());
  192. }
  193. else
  194. {
  195. sourceKey = identifier.Name;
  196. }
  197. processedProperties?.Add(sourceKey.AsStringWithoutTypeCheck());
  198. source.TryGetValue(sourceKey, out var value);
  199. if (p.Value is AssignmentPattern assignmentPattern)
  200. {
  201. if (value.IsUndefined() && assignmentPattern.Right is Expression expression)
  202. {
  203. var jintExpression = Build(engine, expression);
  204. value = jintExpression.GetValue();
  205. }
  206. if (assignmentPattern.Left is BindingPattern bp)
  207. {
  208. ProcessPatterns(engine, bp, value);
  209. continue;
  210. }
  211. var target = assignmentPattern.Left as Identifier ?? identifier;
  212. if (assignmentPattern.Right.IsFunctionWithName())
  213. {
  214. ((FunctionInstance) value).SetFunctionName(target.Name);
  215. }
  216. AssignToIdentifier(engine, target.Name, value);
  217. }
  218. else if (p.Value is BindingPattern bindingPattern)
  219. {
  220. ProcessPatterns(engine, bindingPattern, value);
  221. }
  222. else
  223. {
  224. var target = p.Value as Identifier ?? identifier;
  225. AssignToIdentifier(engine, target.Name, value);
  226. }
  227. }
  228. else
  229. {
  230. var restElement = (RestElement) pattern.Properties[i];
  231. if (restElement.Argument is Identifier leftIdentifier)
  232. {
  233. var rest = source.CreateRestObject(processedProperties);
  234. AssignToIdentifier(engine, leftIdentifier.Name, rest);
  235. }
  236. else if (restElement.Argument is BindingPattern bp)
  237. {
  238. ProcessPatterns(engine, bp, argument);
  239. }
  240. }
  241. }
  242. }
  243. private static void AssignToIdentifier(
  244. Engine engine,
  245. string name,
  246. JsValue rval)
  247. {
  248. var env = engine.ExecutionContext.LexicalEnvironment;
  249. var strict = StrictModeScope.IsStrictModeCode;
  250. if (LexicalEnvironment.TryGetIdentifierEnvironmentWithBindingValue(
  251. env,
  252. name,
  253. strict,
  254. out var environmentRecord,
  255. out _))
  256. {
  257. environmentRecord.SetMutableBinding(name, rval, strict);
  258. }
  259. else
  260. {
  261. env._record.CreateMutableBinding(name, rval);
  262. }
  263. }
  264. }
  265. }