ObjectEnvironmentRecord.cs 6.7 KB

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