using Jint.Collections; using Jint.Native; using Jint.Runtime.Modules; namespace Jint.Runtime.Environments; /// /// Represents a module environment record /// https://tc39.es/ecma262/#sec-module-environment-records /// internal sealed class ModuleEnvironmentRecord : DeclarativeEnvironmentRecord { private readonly HybridDictionary _importBindings = new(); internal ModuleEnvironmentRecord(Engine engine) : base(engine, false) { } public override JsValue GetThisBinding() { return Undefined; } public void CreateImportBinding(string importName, JsModule module, string name) { _hasBindings = true; _importBindings[importName] = new IndirectBinding(module, name); } // https://tc39.es/ecma262/#sec-module-environment-records-getbindingvalue-n-s public override JsValue GetBindingValue(string name, bool strict) { if (_importBindings.TryGetValue(name, out var indirectBinding)) { return indirectBinding.Module._environment.GetBindingValue(indirectBinding.BindingName, true); } return base.GetBindingValue(name, strict); } internal override bool TryGetBinding(in BindingName name, bool strict, out Binding binding, out JsValue value) { if (_importBindings.TryGetValue(name.Key, out var indirectBinding)) { value = indirectBinding.Module._environment.GetBindingValue(indirectBinding.BindingName, true); binding = new(value, canBeDeleted: false, mutable: false, strict: true); return true; } return base.TryGetBinding(name, strict, out binding, out value); } public override bool HasThisBinding() => true; private readonly record struct IndirectBinding(JsModule Module, string BindingName); }