ObjectEnvironmentRecord.cs 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200
  1. using System.Linq;
  2. using Jint.Native;
  3. using Jint.Native.Object;
  4. using Jint.Native.Symbol;
  5. using Jint.Runtime.Descriptors;
  6. namespace Jint.Runtime.Environments
  7. {
  8. /// <summary>
  9. /// Represents an object environment record
  10. /// http://www.ecma-international.org/ecma-262/5.1/#sec-10.2.1.2
  11. /// </summary>
  12. internal sealed class ObjectEnvironmentRecord : EnvironmentRecord
  13. {
  14. internal readonly ObjectInstance _bindingObject;
  15. private readonly bool _provideThis;
  16. private readonly bool _withEnvironment;
  17. public ObjectEnvironmentRecord(
  18. Engine engine,
  19. ObjectInstance bindingObject,
  20. bool provideThis,
  21. bool withEnvironment) : base(engine)
  22. {
  23. _bindingObject = bindingObject;
  24. _provideThis = provideThis;
  25. _withEnvironment = withEnvironment;
  26. }
  27. public override bool HasBinding(string name)
  28. {
  29. var property = new JsString(name);
  30. var foundBinding = HasProperty(property);
  31. if (!foundBinding)
  32. {
  33. return false;
  34. }
  35. if (!_withEnvironment)
  36. {
  37. return true;
  38. }
  39. return !IsBlocked(name);
  40. }
  41. private bool HasProperty(JsValue property)
  42. {
  43. return _bindingObject.HasProperty(property);
  44. }
  45. internal override bool TryGetBinding(
  46. in BindingName name,
  47. bool strict,
  48. out Binding binding,
  49. out JsValue value)
  50. {
  51. // we unwrap by name
  52. binding = default;
  53. if (!HasProperty(name.StringValue))
  54. {
  55. value = default;
  56. return false;
  57. }
  58. if (_withEnvironment && IsBlocked(name.StringValue))
  59. {
  60. value = default;
  61. return false;
  62. }
  63. var desc = _bindingObject.GetProperty(name.StringValue);
  64. value = ObjectInstance.UnwrapJsValue(desc, _bindingObject);
  65. return true;
  66. }
  67. private bool IsBlocked(JsValue property)
  68. {
  69. var unscopables = _bindingObject.Get(GlobalSymbolRegistry.Unscopables);
  70. if (unscopables is ObjectInstance oi)
  71. {
  72. var blocked = TypeConverter.ToBoolean(oi.Get(property));
  73. if (blocked)
  74. {
  75. return true;
  76. }
  77. }
  78. return false;
  79. }
  80. /// <summary>
  81. /// http://www.ecma-international.org/ecma-262/6.0/#sec-object-environment-records-createmutablebinding-n-d
  82. /// </summary>
  83. public override void CreateMutableBinding(string name, bool canBeDeleted = false)
  84. {
  85. var propertyDescriptor = canBeDeleted
  86. ? new PropertyDescriptor(Undefined, PropertyFlag.ConfigurableEnumerableWritable | PropertyFlag.MutableBinding)
  87. : new PropertyDescriptor(Undefined, PropertyFlag.NonConfigurable | PropertyFlag.MutableBinding);
  88. _bindingObject.DefinePropertyOrThrow(name, propertyDescriptor);
  89. }
  90. /// <summary>
  91. /// http://www.ecma-international.org/ecma-262/6.0/#sec-object-environment-records-createmutablebinding-n-d
  92. /// </summary>
  93. internal void CreateMutableBindingAndInitialize(string name, JsValue value, bool canBeDeleted = false)
  94. {
  95. var propertyDescriptor = canBeDeleted
  96. ? new PropertyDescriptor(value, PropertyFlag.ConfigurableEnumerableWritable | PropertyFlag.MutableBinding)
  97. : new PropertyDescriptor(value, PropertyFlag.NonConfigurable | PropertyFlag.MutableBinding);
  98. _bindingObject.DefinePropertyOrThrow(name, propertyDescriptor);
  99. }
  100. /// <summary>
  101. /// http://www.ecma-international.org/ecma-262/6.0/#sec-object-environment-records-createimmutablebinding-n-s
  102. /// </summary>
  103. public override void CreateImmutableBinding(string name, bool strict = true)
  104. {
  105. ExceptionHelper.ThrowInvalidOperationException("The concrete Environment Record method CreateImmutableBinding is never used within this specification in association with Object Environment Records.");
  106. }
  107. /// <summary>
  108. /// http://www.ecma-international.org/ecma-262/6.0/#sec-object-environment-records-initializebinding-n-v
  109. /// </summary>
  110. public override void InitializeBinding(string name, JsValue value)
  111. {
  112. SetMutableBinding(name, value, false);
  113. }
  114. public override void SetMutableBinding(string name, JsValue value, bool strict)
  115. {
  116. SetMutableBinding(new BindingName(name), value, strict);
  117. }
  118. internal override void SetMutableBinding(in BindingName name, JsValue value, bool strict)
  119. {
  120. if (!_bindingObject.Set(name.StringValue, value) && strict)
  121. {
  122. ExceptionHelper.ThrowTypeError(_engine);
  123. }
  124. }
  125. public override JsValue GetBindingValue(string name, bool strict)
  126. {
  127. var desc = _bindingObject.GetProperty(name);
  128. if (strict && desc == PropertyDescriptor.Undefined)
  129. {
  130. ExceptionHelper.ThrowReferenceError(_engine, name.ToString());
  131. }
  132. return ObjectInstance.UnwrapJsValue(desc, this);
  133. }
  134. public override bool DeleteBinding(string name)
  135. {
  136. return _bindingObject.Delete(name);
  137. }
  138. public override bool HasThisBinding() => false;
  139. public override bool HasSuperBinding() => false;
  140. public override JsValue WithBaseObject() => _withEnvironment ? _bindingObject : Undefined;
  141. public override JsValue ImplicitThisValue()
  142. {
  143. if (_provideThis)
  144. {
  145. return _bindingObject;
  146. }
  147. return Undefined;
  148. }
  149. internal override string[] GetAllBindingNames()
  150. {
  151. if (!ReferenceEquals(_bindingObject, null))
  152. {
  153. return _bindingObject.GetOwnProperties().Select( x=> x.Key.ToString()).ToArray();
  154. }
  155. return System.Array.Empty<string>();
  156. }
  157. public override bool Equals(JsValue other)
  158. {
  159. return ReferenceEquals(_bindingObject, other);
  160. }
  161. public override JsValue GetThisBinding()
  162. {
  163. throw new System.NotImplementedException();
  164. }
  165. }
  166. }