ArrayInstance.cs 7.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212
  1. using System.Linq;
  2. using Jint.Native.Object;
  3. using Jint.Runtime;
  4. using Jint.Runtime.Descriptors;
  5. namespace Jint.Native.Array
  6. {
  7. public class ArrayInstance : ObjectInstance
  8. {
  9. private readonly Engine _engine;
  10. public ArrayInstance(Engine engine) : base(engine)
  11. {
  12. _engine = engine;
  13. }
  14. public override string Class
  15. {
  16. get
  17. {
  18. return "Array";
  19. }
  20. }
  21. /// Implementation from ObjectInstance official specs as the one
  22. /// in ObjectInstance is optimized for the general case and wouldn't work
  23. /// for arrays
  24. public override void Put(string propertyName, JsValue value, bool throwOnError)
  25. {
  26. if (!CanPut(propertyName))
  27. {
  28. if (throwOnError)
  29. {
  30. throw new JavaScriptException(Engine.TypeError);
  31. }
  32. return;
  33. }
  34. var ownDesc = GetOwnProperty(propertyName);
  35. if (ownDesc.IsDataDescriptor())
  36. {
  37. var valueDesc = new PropertyDescriptor(value: value, writable: null, enumerable: null, configurable: null);
  38. DefineOwnProperty(propertyName, valueDesc, throwOnError);
  39. return;
  40. }
  41. // property is an accessor or inherited
  42. var desc = GetProperty(propertyName);
  43. if (desc.IsAccessorDescriptor())
  44. {
  45. var setter = desc.Set.Value.TryCast<ICallable>();
  46. setter.Call(new JsValue(this), new[] { value });
  47. }
  48. else
  49. {
  50. var newDesc = new PropertyDescriptor(value, true, true, true);
  51. DefineOwnProperty(propertyName, newDesc, throwOnError);
  52. }
  53. }
  54. public override bool DefineOwnProperty(string propertyName, PropertyDescriptor desc, bool throwOnError)
  55. {
  56. var oldLenDesc = GetOwnProperty("length");
  57. var oldLen = TypeConverter.ToNumber(oldLenDesc.Value.Value);
  58. if (propertyName == "length")
  59. {
  60. if (!desc.Value.HasValue)
  61. {
  62. return base.DefineOwnProperty("length", desc, throwOnError);
  63. }
  64. var newLenDesc = new PropertyDescriptor(desc);
  65. uint newLen = TypeConverter.ToUint32(desc.Value.Value);
  66. if (newLen != TypeConverter.ToNumber(desc.Value.Value))
  67. {
  68. throw new JavaScriptException(_engine.RangeError);
  69. }
  70. newLenDesc.Value = newLen;
  71. if (newLen >= oldLen)
  72. {
  73. return base.DefineOwnProperty("length", newLenDesc, throwOnError);
  74. }
  75. if (!oldLenDesc.Writable.Value)
  76. {
  77. if (throwOnError)
  78. {
  79. throw new JavaScriptException(_engine.TypeError);
  80. }
  81. return false;
  82. }
  83. bool newWritable;
  84. if (!newLenDesc.Writable.HasValue || newLenDesc.Writable.Value)
  85. {
  86. newWritable = true;
  87. }
  88. else
  89. {
  90. newWritable = false;
  91. newLenDesc.Writable = true;
  92. }
  93. var succeeded = base.DefineOwnProperty("length", newLenDesc, throwOnError);
  94. if (!succeeded)
  95. {
  96. return false;
  97. }
  98. // in the case of sparse arrays, treat each concrete element instead of
  99. // iterating over all indexes
  100. if (Properties.Count < oldLen - newLen)
  101. {
  102. var keys = Properties.Keys.ToArray();
  103. foreach (var key in keys)
  104. {
  105. uint index;
  106. // is it the index of the array
  107. if (uint.TryParse(key, out index) && index >= newLen && index < oldLen)
  108. {
  109. var deleteSucceeded = Delete(key, false);
  110. if (!deleteSucceeded)
  111. {
  112. newLenDesc.Value = new JsValue(index + 1);
  113. if (!newWritable)
  114. {
  115. newLenDesc.Writable = false;
  116. }
  117. base.DefineOwnProperty("length", newLenDesc, false);
  118. if (throwOnError)
  119. {
  120. throw new JavaScriptException(_engine.TypeError);
  121. }
  122. return false;
  123. }
  124. }
  125. }
  126. }
  127. else
  128. {
  129. while (newLen < oldLen)
  130. {
  131. // algorithm as per the spec
  132. oldLen--;
  133. var deleteSucceeded = Delete(TypeConverter.ToString(oldLen), false);
  134. if (!deleteSucceeded)
  135. {
  136. newLenDesc.Value = oldLen + 1;
  137. if (!newWritable)
  138. {
  139. newLenDesc.Writable = false;
  140. }
  141. base.DefineOwnProperty("length", newLenDesc, false);
  142. if (throwOnError)
  143. {
  144. throw new JavaScriptException(_engine.TypeError);
  145. }
  146. return false;
  147. }
  148. }
  149. }
  150. if (!newWritable)
  151. {
  152. DefineOwnProperty("length", new PropertyDescriptor(value: null, writable: false, enumerable: null, configurable: null), false);
  153. }
  154. return true;
  155. }
  156. else if (IsArrayIndex(propertyName))
  157. {
  158. var index = TypeConverter.ToUint32(propertyName);
  159. if (index >= oldLen && !oldLenDesc.Writable.Value)
  160. {
  161. if (throwOnError)
  162. {
  163. throw new JavaScriptException(_engine.TypeError);
  164. }
  165. return false;
  166. }
  167. var succeeded = base.DefineOwnProperty(propertyName, desc, false);
  168. if (!succeeded)
  169. {
  170. if (throwOnError)
  171. {
  172. throw new JavaScriptException(_engine.TypeError);
  173. }
  174. return false;
  175. }
  176. if (index >= oldLen)
  177. {
  178. oldLenDesc.Value = index + 1;
  179. base.DefineOwnProperty("length", oldLenDesc, false);
  180. }
  181. return true;
  182. }
  183. return base.DefineOwnProperty(propertyName, desc, throwOnError);
  184. }
  185. public static bool IsArrayIndex(JsValue p)
  186. {
  187. return TypeConverter.ToString(TypeConverter.ToUint32(p)) == TypeConverter.ToString(p) && TypeConverter.ToUint32(p) != uint.MaxValue;
  188. }
  189. }
  190. }