v8/test/unittests/objects/modules-unittest.cc

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

1180 lines
44 KiB
C++
Raw Normal View History

// Copyright 2022 the V8 project authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
Reland "[include] Split out v8.h" This is a reland of d1b27019d3bf86360ea838c317f8505fac6d3a7e Fixes include: Adding missing file to bazel build Forward-declaring classing before friend-classing them to fix win/gcc Add missing v8-isolate.h include for vtune builds Original change's description: > [include] Split out v8.h > > This moves every single class/function out of include/v8.h into a > separate header in include/, which v8.h then includes so that > externally nothing appears to have changed. > > Every include of v8.h from inside v8 has been changed to a more > fine-grained include. > > Previously inline functions defined at the bottom of v8.h would call > private non-inline functions in the V8 class. Since that class is now > in v8-initialization.h and is rarely included (as that would create > dependency cycles), this is not possible and so those methods have been > moved out of the V8 class into the namespace v8::api_internal. > > None of the previous files in include/ now #include v8.h, which means > if embedders were relying on this transitive dependency then it will > give compile failures. > > v8-inspector.h does depend on v8-scripts.h for the time being to ensure > that Chrome continue to compile but that change will be reverted once > those transitive #includes in chrome are changed to include it directly. > > Full design: > https://docs.google.com/document/d/1rTD--I8hCAr-Rho1WTumZzFKaDpEp0IJ8ejZtk4nJdA/edit?usp=sharing > > Bug: v8:11965 > Change-Id: I53b84b29581632710edc80eb11f819c2097a2877 > Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/3097448 > Reviewed-by: Yang Guo <yangguo@chromium.org> > Reviewed-by: Camillo Bruni <cbruni@chromium.org> > Reviewed-by: Jakob Kummerow <jkummerow@chromium.org> > Reviewed-by: Leszek Swirski <leszeks@chromium.org> > Reviewed-by: Michael Lippautz <mlippautz@chromium.org> > Commit-Queue: Dan Elphick <delphick@chromium.org> > Cr-Commit-Position: refs/heads/main@{#76424} Cq-Include-Trybots: luci.v8.try:v8_linux_vtunejit Bug: v8:11965 Change-Id: I99f5d3a73bf8fe25b650adfaf9567dc4e44a09e6 Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/3113629 Reviewed-by: Leszek Swirski <leszeks@chromium.org> Reviewed-by: Camillo Bruni <cbruni@chromium.org> Reviewed-by: Michael Lippautz <mlippautz@chromium.org> Reviewed-by: Jakob Kummerow <jkummerow@chromium.org> Reviewed-by: Simon Zünd <szuend@chromium.org> Commit-Queue: Dan Elphick <delphick@chromium.org> Cr-Commit-Position: refs/heads/main@{#76460}
2021-08-23 13:01:06 +00:00
#include "include/v8-function.h"
#include "src/flags/flags.h"
#include "test/unittests/test-utils.h"
#include "testing/gtest/include/gtest/gtest.h"
namespace {
using ModuleTest = v8::TestWithContext;
using v8::Context;
[modules] Add refactored API to get ModuleRequests and expose import assertions This change refactors the v8.h API as discussed in https://docs.google.com/document/d/1yuXgNHSbTAPubT1Mg0JXp5uTrfirkvO1g5cHHCe-LmY/edit#heading=h.q0c9h4p928mn such that a v8::Module exposes module requests as a FixedArray of ModuleRequest objects, which can then be used to obtain their module specifier and source code offset. This replaces the old functions that passed back individual specifier Strings and Locations via repeated calls to getters that take an index. These are marked as deprecated. The new ModuleRequest interface includes a getter for an ImportAssertions FixedArray, which will contain the import assertions for the request if --harmony-import-assertions is set, and will be empty otherwise. One notable change here is that the APIs now return source code offsets rather than v8::Locations. The host must then call the new Module::SourceOffsetToLocation to convert these offsets into line/column numbers. This requires a bit more back-and-forth, but allows the host to defer the cost of converting from source offset to line/column numbers until an error needs to be reported, potentially skipping the work altogether. Bug: v8:10958 Change-Id: I181639737c701e467324e6c781aa4d7bdd87ae8c Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/2545577 Commit-Queue: Dan Clark <daniec@microsoft.com> Reviewed-by: Camillo Bruni <cbruni@chromium.org> Reviewed-by: Marja Hölttä <marja@chromium.org> Cr-Commit-Position: refs/heads/master@{#71387}
2020-11-24 19:13:00 +00:00
using v8::Data;
using v8::FixedArray;
using v8::HandleScope;
[modules] Add refactored API to get ModuleRequests and expose import assertions This change refactors the v8.h API as discussed in https://docs.google.com/document/d/1yuXgNHSbTAPubT1Mg0JXp5uTrfirkvO1g5cHHCe-LmY/edit#heading=h.q0c9h4p928mn such that a v8::Module exposes module requests as a FixedArray of ModuleRequest objects, which can then be used to obtain their module specifier and source code offset. This replaces the old functions that passed back individual specifier Strings and Locations via repeated calls to getters that take an index. These are marked as deprecated. The new ModuleRequest interface includes a getter for an ImportAssertions FixedArray, which will contain the import assertions for the request if --harmony-import-assertions is set, and will be empty otherwise. One notable change here is that the APIs now return source code offsets rather than v8::Locations. The host must then call the new Module::SourceOffsetToLocation to convert these offsets into line/column numbers. This requires a bit more back-and-forth, but allows the host to defer the cost of converting from source offset to line/column numbers until an error needs to be reported, potentially skipping the work altogether. Bug: v8:10958 Change-Id: I181639737c701e467324e6c781aa4d7bdd87ae8c Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/2545577 Commit-Queue: Dan Clark <daniec@microsoft.com> Reviewed-by: Camillo Bruni <cbruni@chromium.org> Reviewed-by: Marja Hölttä <marja@chromium.org> Cr-Commit-Position: refs/heads/master@{#71387}
2020-11-24 19:13:00 +00:00
using v8::Int32;
using v8::Isolate;
using v8::Local;
[modules] Add refactored API to get ModuleRequests and expose import assertions This change refactors the v8.h API as discussed in https://docs.google.com/document/d/1yuXgNHSbTAPubT1Mg0JXp5uTrfirkvO1g5cHHCe-LmY/edit#heading=h.q0c9h4p928mn such that a v8::Module exposes module requests as a FixedArray of ModuleRequest objects, which can then be used to obtain their module specifier and source code offset. This replaces the old functions that passed back individual specifier Strings and Locations via repeated calls to getters that take an index. These are marked as deprecated. The new ModuleRequest interface includes a getter for an ImportAssertions FixedArray, which will contain the import assertions for the request if --harmony-import-assertions is set, and will be empty otherwise. One notable change here is that the APIs now return source code offsets rather than v8::Locations. The host must then call the new Module::SourceOffsetToLocation to convert these offsets into line/column numbers. This requires a bit more back-and-forth, but allows the host to defer the cost of converting from source offset to line/column numbers until an error needs to be reported, potentially skipping the work altogether. Bug: v8:10958 Change-Id: I181639737c701e467324e6c781aa4d7bdd87ae8c Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/2545577 Commit-Queue: Dan Clark <daniec@microsoft.com> Reviewed-by: Camillo Bruni <cbruni@chromium.org> Reviewed-by: Marja Hölttä <marja@chromium.org> Cr-Commit-Position: refs/heads/master@{#71387}
2020-11-24 19:13:00 +00:00
using v8::Location;
using v8::MaybeLocal;
using v8::Module;
[modules] Add refactored API to get ModuleRequests and expose import assertions This change refactors the v8.h API as discussed in https://docs.google.com/document/d/1yuXgNHSbTAPubT1Mg0JXp5uTrfirkvO1g5cHHCe-LmY/edit#heading=h.q0c9h4p928mn such that a v8::Module exposes module requests as a FixedArray of ModuleRequest objects, which can then be used to obtain their module specifier and source code offset. This replaces the old functions that passed back individual specifier Strings and Locations via repeated calls to getters that take an index. These are marked as deprecated. The new ModuleRequest interface includes a getter for an ImportAssertions FixedArray, which will contain the import assertions for the request if --harmony-import-assertions is set, and will be empty otherwise. One notable change here is that the APIs now return source code offsets rather than v8::Locations. The host must then call the new Module::SourceOffsetToLocation to convert these offsets into line/column numbers. This requires a bit more back-and-forth, but allows the host to defer the cost of converting from source offset to line/column numbers until an error needs to be reported, potentially skipping the work altogether. Bug: v8:10958 Change-Id: I181639737c701e467324e6c781aa4d7bdd87ae8c Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/2545577 Commit-Queue: Dan Clark <daniec@microsoft.com> Reviewed-by: Camillo Bruni <cbruni@chromium.org> Reviewed-by: Marja Hölttä <marja@chromium.org> Cr-Commit-Position: refs/heads/master@{#71387}
2020-11-24 19:13:00 +00:00
using v8::ModuleRequest;
using v8::Promise;
using v8::ScriptCompiler;
using v8::ScriptOrigin;
using v8::String;
using v8::Value;
ScriptOrigin ModuleOrigin(Local<v8::Value> resource_name, Isolate* isolate) {
ScriptOrigin origin(isolate, resource_name, 0, 0, false, -1,
Local<v8::Value>(), false, false, true);
return origin;
}
static Local<Module> dep1;
static Local<Module> dep2;
MaybeLocal<Module> ResolveCallback(Local<Context> context,
Local<String> specifier,
Local<FixedArray> import_assertions,
Local<Module> referrer) {
CHECK_EQ(0, import_assertions->Length());
Isolate* isolate = context->GetIsolate();
if (specifier->StrictEquals(
String::NewFromUtf8(isolate, "./dep1.js").ToLocalChecked())) {
return dep1;
} else if (specifier->StrictEquals(
String::NewFromUtf8(isolate, "./dep2.js").ToLocalChecked())) {
return dep2;
} else {
isolate->ThrowException(
String::NewFromUtf8(isolate, "boom").ToLocalChecked());
return MaybeLocal<Module>();
}
}
TEST_F(ModuleTest, ModuleInstantiationFailures1) {
HandleScope scope(isolate());
v8::TryCatch try_catch(isolate());
Local<Module> module;
{
Local<String> source_text = NewString(
"import './foo.js';\n"
"export {} from './bar.js';");
ScriptOrigin origin = ModuleOrigin(NewString("file.js"), isolate());
ScriptCompiler::Source source(source_text, origin);
module = ScriptCompiler::CompileModule(isolate(), &source).ToLocalChecked();
CHECK_EQ(Module::kUninstantiated, module->GetStatus());
Local<FixedArray> module_requests = module->GetModuleRequests();
CHECK_EQ(2, module_requests->Length());
Local<ModuleRequest> module_request_0 =
module_requests->Get(context(), 0).As<ModuleRequest>();
CHECK(
NewString("./foo.js")->StrictEquals(module_request_0->GetSpecifier()));
int offset = module_request_0->GetSourceOffset();
CHECK_EQ(7, offset);
Location loc = module->SourceOffsetToLocation(offset);
CHECK_EQ(0, loc.GetLineNumber());
CHECK_EQ(7, loc.GetColumnNumber());
CHECK_EQ(0, module_request_0->GetImportAssertions()->Length());
Local<ModuleRequest> module_request_1 =
module_requests->Get(context(), 1).As<ModuleRequest>();
CHECK(
NewString("./bar.js")->StrictEquals(module_request_1->GetSpecifier()));
offset = module_request_1->GetSourceOffset();
CHECK_EQ(34, offset);
loc = module->SourceOffsetToLocation(offset);
CHECK_EQ(1, loc.GetLineNumber());
CHECK_EQ(15, loc.GetColumnNumber());
CHECK_EQ(0, module_request_1->GetImportAssertions()->Length());
}
// Instantiation should fail.
{
v8::TryCatch inner_try_catch(isolate());
CHECK(module->InstantiateModule(context(), ResolveCallback).IsNothing());
CHECK(inner_try_catch.HasCaught());
CHECK(inner_try_catch.Exception()->StrictEquals(NewString("boom")));
CHECK_EQ(Module::kUninstantiated, module->GetStatus());
}
// Start over again...
{
Local<String> source_text = NewString(
"import './dep1.js';\n"
"export {} from './bar.js';");
ScriptOrigin origin = ModuleOrigin(NewString("file.js"), isolate());
ScriptCompiler::Source source(source_text, origin);
module = ScriptCompiler::CompileModule(isolate(), &source).ToLocalChecked();
}
// dep1.js
{
Local<String> source_text = NewString("");
ScriptOrigin origin = ModuleOrigin(NewString("dep1.js"), isolate());
ScriptCompiler::Source source(source_text, origin);
dep1 = ScriptCompiler::CompileModule(isolate(), &source).ToLocalChecked();
}
// Instantiation should fail because a sub-module fails to resolve.
{
v8::TryCatch inner_try_catch(isolate());
CHECK(module->InstantiateModule(context(), ResolveCallback).IsNothing());
CHECK(inner_try_catch.HasCaught());
CHECK(inner_try_catch.Exception()->StrictEquals(NewString("boom")));
CHECK_EQ(Module::kUninstantiated, module->GetStatus());
}
CHECK(!try_catch.HasCaught());
}
static Local<Module> fooModule;
static Local<Module> barModule;
MaybeLocal<Module> ResolveCallbackWithImportAssertions(
Local<Context> context, Local<String> specifier,
Local<FixedArray> import_assertions, Local<Module> referrer) {
Isolate* isolate = context->GetIsolate();
if (specifier->StrictEquals(
String::NewFromUtf8(isolate, "./foo.js").ToLocalChecked())) {
CHECK_EQ(0, import_assertions->Length());
return fooModule;
} else if (specifier->StrictEquals(
String::NewFromUtf8(isolate, "./bar.js").ToLocalChecked())) {
CHECK_EQ(3, import_assertions->Length());
Local<String> assertion_key =
import_assertions->Get(context, 0).As<Value>().As<String>();
CHECK(String::NewFromUtf8(isolate, "a")
.ToLocalChecked()
->StrictEquals(assertion_key));
Local<String> assertion_value =
import_assertions->Get(context, 1).As<Value>().As<String>();
CHECK(String::NewFromUtf8(isolate, "b")
.ToLocalChecked()
->StrictEquals(assertion_value));
Local<Data> assertion_source_offset_object =
import_assertions->Get(context, 2);
Local<Int32> assertion_source_offset_int32 =
assertion_source_offset_object.As<Value>()
->ToInt32(context)
.ToLocalChecked();
int32_t assertion_source_offset = assertion_source_offset_int32->Value();
CHECK_EQ(65, assertion_source_offset);
Location loc = referrer->SourceOffsetToLocation(assertion_source_offset);
CHECK_EQ(1, loc.GetLineNumber());
CHECK_EQ(35, loc.GetColumnNumber());
return barModule;
} else {
isolate->ThrowException(
String::NewFromUtf8(isolate, "boom").ToLocalChecked());
return MaybeLocal<Module>();
}
}
TEST_F(ModuleTest, ModuleInstantiationWithImportAssertions) {
bool prev_import_assertions = i::v8_flags.harmony_import_assertions;
i::v8_flags.harmony_import_assertions = true;
HandleScope scope(isolate());
v8::TryCatch try_catch(isolate());
Local<Module> module;
{
Local<String> source_text = NewString(
"import './foo.js' assert { };\n"
"export {} from './bar.js' assert { a: 'b' };");
ScriptOrigin origin = ModuleOrigin(NewString("file.js"), isolate());
ScriptCompiler::Source source(source_text, origin);
module = ScriptCompiler::CompileModule(isolate(), &source).ToLocalChecked();
CHECK_EQ(Module::kUninstantiated, module->GetStatus());
Local<FixedArray> module_requests = module->GetModuleRequests();
CHECK_EQ(2, module_requests->Length());
Local<ModuleRequest> module_request_0 =
module_requests->Get(context(), 0).As<ModuleRequest>();
CHECK(
NewString("./foo.js")->StrictEquals(module_request_0->GetSpecifier()));
int offset = module_request_0->GetSourceOffset();
CHECK_EQ(7, offset);
Location loc = module->SourceOffsetToLocation(offset);
CHECK_EQ(0, loc.GetLineNumber());
CHECK_EQ(7, loc.GetColumnNumber());
CHECK_EQ(0, module_request_0->GetImportAssertions()->Length());
Local<ModuleRequest> module_request_1 =
module_requests->Get(context(), 1).As<ModuleRequest>();
CHECK(
NewString("./bar.js")->StrictEquals(module_request_1->GetSpecifier()));
offset = module_request_1->GetSourceOffset();
CHECK_EQ(45, offset);
loc = module->SourceOffsetToLocation(offset);
CHECK_EQ(1, loc.GetLineNumber());
CHECK_EQ(15, loc.GetColumnNumber());
Local<FixedArray> import_assertions_1 =
module_request_1->GetImportAssertions();
CHECK_EQ(3, import_assertions_1->Length());
Local<String> assertion_key =
import_assertions_1->Get(context(), 0).As<String>();
CHECK(NewString("a")->StrictEquals(assertion_key));
Local<String> assertion_value =
import_assertions_1->Get(context(), 1).As<String>();
CHECK(NewString("b")->StrictEquals(assertion_value));
int32_t assertion_source_offset =
import_assertions_1->Get(context(), 2).As<Int32>()->Value();
CHECK_EQ(65, assertion_source_offset);
loc = module->SourceOffsetToLocation(assertion_source_offset);
CHECK_EQ(1, loc.GetLineNumber());
CHECK_EQ(35, loc.GetColumnNumber());
}
[modules] Add refactored API to get ModuleRequests and expose import assertions This change refactors the v8.h API as discussed in https://docs.google.com/document/d/1yuXgNHSbTAPubT1Mg0JXp5uTrfirkvO1g5cHHCe-LmY/edit#heading=h.q0c9h4p928mn such that a v8::Module exposes module requests as a FixedArray of ModuleRequest objects, which can then be used to obtain their module specifier and source code offset. This replaces the old functions that passed back individual specifier Strings and Locations via repeated calls to getters that take an index. These are marked as deprecated. The new ModuleRequest interface includes a getter for an ImportAssertions FixedArray, which will contain the import assertions for the request if --harmony-import-assertions is set, and will be empty otherwise. One notable change here is that the APIs now return source code offsets rather than v8::Locations. The host must then call the new Module::SourceOffsetToLocation to convert these offsets into line/column numbers. This requires a bit more back-and-forth, but allows the host to defer the cost of converting from source offset to line/column numbers until an error needs to be reported, potentially skipping the work altogether. Bug: v8:10958 Change-Id: I181639737c701e467324e6c781aa4d7bdd87ae8c Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/2545577 Commit-Queue: Dan Clark <daniec@microsoft.com> Reviewed-by: Camillo Bruni <cbruni@chromium.org> Reviewed-by: Marja Hölttä <marja@chromium.org> Cr-Commit-Position: refs/heads/master@{#71387}
2020-11-24 19:13:00 +00:00
// foo.js
{
Local<String> source_text = NewString("Object.expando = 40");
ScriptOrigin origin = ModuleOrigin(NewString("foo.js"), isolate());
ScriptCompiler::Source source(source_text, origin);
fooModule =
ScriptCompiler::CompileModule(isolate(), &source).ToLocalChecked();
}
[modules] Add refactored API to get ModuleRequests and expose import assertions This change refactors the v8.h API as discussed in https://docs.google.com/document/d/1yuXgNHSbTAPubT1Mg0JXp5uTrfirkvO1g5cHHCe-LmY/edit#heading=h.q0c9h4p928mn such that a v8::Module exposes module requests as a FixedArray of ModuleRequest objects, which can then be used to obtain their module specifier and source code offset. This replaces the old functions that passed back individual specifier Strings and Locations via repeated calls to getters that take an index. These are marked as deprecated. The new ModuleRequest interface includes a getter for an ImportAssertions FixedArray, which will contain the import assertions for the request if --harmony-import-assertions is set, and will be empty otherwise. One notable change here is that the APIs now return source code offsets rather than v8::Locations. The host must then call the new Module::SourceOffsetToLocation to convert these offsets into line/column numbers. This requires a bit more back-and-forth, but allows the host to defer the cost of converting from source offset to line/column numbers until an error needs to be reported, potentially skipping the work altogether. Bug: v8:10958 Change-Id: I181639737c701e467324e6c781aa4d7bdd87ae8c Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/2545577 Commit-Queue: Dan Clark <daniec@microsoft.com> Reviewed-by: Camillo Bruni <cbruni@chromium.org> Reviewed-by: Marja Hölttä <marja@chromium.org> Cr-Commit-Position: refs/heads/master@{#71387}
2020-11-24 19:13:00 +00:00
// bar.js
{
Local<String> source_text = NewString("Object.expando += 2");
ScriptOrigin origin = ModuleOrigin(NewString("bar.js"), isolate());
ScriptCompiler::Source source(source_text, origin);
barModule =
ScriptCompiler::CompileModule(isolate(), &source).ToLocalChecked();
}
CHECK(
module->InstantiateModule(context(), ResolveCallbackWithImportAssertions)
.FromJust());
CHECK_EQ(Module::kInstantiated, module->GetStatus());
MaybeLocal<Value> result = module->Evaluate(context());
CHECK_EQ(Module::kEvaluated, module->GetStatus());
Local<Promise> promise = Local<Promise>::Cast(result.ToLocalChecked());
CHECK_EQ(promise->State(), v8::Promise::kFulfilled);
CHECK(promise->Result()->IsUndefined());
// TODO(v8:12781): One IsInt32 matcher be added in
// gmock-support.h, we could use IsInt32 to replace
// this.
{
Local<Value> result = RunJS("Object.expando");
CHECK(result->IsInt32());
CHECK_EQ(42, result->Int32Value(context()).FromJust());
}
CHECK(!try_catch.HasCaught());
i::v8_flags.harmony_import_assertions = prev_import_assertions;
[modules] Add refactored API to get ModuleRequests and expose import assertions This change refactors the v8.h API as discussed in https://docs.google.com/document/d/1yuXgNHSbTAPubT1Mg0JXp5uTrfirkvO1g5cHHCe-LmY/edit#heading=h.q0c9h4p928mn such that a v8::Module exposes module requests as a FixedArray of ModuleRequest objects, which can then be used to obtain their module specifier and source code offset. This replaces the old functions that passed back individual specifier Strings and Locations via repeated calls to getters that take an index. These are marked as deprecated. The new ModuleRequest interface includes a getter for an ImportAssertions FixedArray, which will contain the import assertions for the request if --harmony-import-assertions is set, and will be empty otherwise. One notable change here is that the APIs now return source code offsets rather than v8::Locations. The host must then call the new Module::SourceOffsetToLocation to convert these offsets into line/column numbers. This requires a bit more back-and-forth, but allows the host to defer the cost of converting from source offset to line/column numbers until an error needs to be reported, potentially skipping the work altogether. Bug: v8:10958 Change-Id: I181639737c701e467324e6c781aa4d7bdd87ae8c Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/2545577 Commit-Queue: Dan Clark <daniec@microsoft.com> Reviewed-by: Camillo Bruni <cbruni@chromium.org> Reviewed-by: Marja Hölttä <marja@chromium.org> Cr-Commit-Position: refs/heads/master@{#71387}
2020-11-24 19:13:00 +00:00
}
TEST_F(ModuleTest, ModuleInstantiationFailures2) {
HandleScope scope(isolate());
v8::TryCatch try_catch(isolate());
// root1.js
Local<Module> root;
{
Local<String> source_text =
NewString("import './dep1.js'; import './dep2.js'");
ScriptOrigin origin = ModuleOrigin(NewString("root1.js"), isolate());
ScriptCompiler::Source source(source_text, origin);
root = ScriptCompiler::CompileModule(isolate(), &source).ToLocalChecked();
}
// dep1.js
{
Local<String> source_text = NewString("export let x = 42");
ScriptOrigin origin = ModuleOrigin(NewString("dep1.js"), isolate());
ScriptCompiler::Source source(source_text, origin);
dep1 = ScriptCompiler::CompileModule(isolate(), &source).ToLocalChecked();
}
// dep2.js
{
Local<String> source_text = NewString("import {foo} from './dep3.js'");
ScriptOrigin origin = ModuleOrigin(NewString("dep2.js"), isolate());
ScriptCompiler::Source source(source_text, origin);
dep2 = ScriptCompiler::CompileModule(isolate(), &source).ToLocalChecked();
}
{
v8::TryCatch inner_try_catch(isolate());
CHECK(root->InstantiateModule(context(), ResolveCallback).IsNothing());
CHECK(inner_try_catch.HasCaught());
CHECK(inner_try_catch.Exception()->StrictEquals(NewString("boom")));
CHECK_EQ(Module::kUninstantiated, root->GetStatus());
CHECK_EQ(Module::kUninstantiated, dep1->GetStatus());
CHECK_EQ(Module::kUninstantiated, dep2->GetStatus());
}
// Change dep2.js
{
Local<String> source_text = NewString("import {foo} from './dep2.js'");
ScriptOrigin origin = ModuleOrigin(NewString("dep2.js"), isolate());
ScriptCompiler::Source source(source_text, origin);
dep2 = ScriptCompiler::CompileModule(isolate(), &source).ToLocalChecked();
}
{
v8::TryCatch inner_try_catch(isolate());
CHECK(root->InstantiateModule(context(), ResolveCallback).IsNothing());
CHECK(inner_try_catch.HasCaught());
CHECK(!inner_try_catch.Exception()->StrictEquals(NewString("boom")));
CHECK_EQ(Module::kUninstantiated, root->GetStatus());
CHECK_EQ(Module::kInstantiated, dep1->GetStatus());
CHECK_EQ(Module::kUninstantiated, dep2->GetStatus());
}
// Change dep2.js again
{
Local<String> source_text = NewString("import {foo} from './dep3.js'");
ScriptOrigin origin = ModuleOrigin(NewString("dep2.js"), isolate());
ScriptCompiler::Source source(source_text, origin);
dep2 = ScriptCompiler::CompileModule(isolate(), &source).ToLocalChecked();
}
{
v8::TryCatch inner_try_catch(isolate());
CHECK(root->InstantiateModule(context(), ResolveCallback).IsNothing());
CHECK(inner_try_catch.HasCaught());
CHECK(inner_try_catch.Exception()->StrictEquals(NewString("boom")));
CHECK_EQ(Module::kUninstantiated, root->GetStatus());
CHECK_EQ(Module::kInstantiated, dep1->GetStatus());
CHECK_EQ(Module::kUninstantiated, dep2->GetStatus());
}
}
static MaybeLocal<Module> CompileSpecifierAsModuleResolveCallback(
Local<Context> context, Local<String> specifier,
Local<FixedArray> import_assertions, Local<Module> referrer) {
CHECK_EQ(0, import_assertions->Length());
Isolate* isolate = context->GetIsolate();
ScriptOrigin origin = ModuleOrigin(
String::NewFromUtf8(isolate, "module.js").ToLocalChecked(), isolate);
ScriptCompiler::Source source(specifier, origin);
return ScriptCompiler::CompileModule(isolate, &source).ToLocalChecked();
}
TEST_F(ModuleTest, ModuleEvaluation) {
HandleScope scope(isolate());
v8::TryCatch try_catch(isolate());
Local<String> source_text = NewString(
"import 'Object.expando = 5';"
"import 'Object.expando *= 2';");
ScriptOrigin origin = ModuleOrigin(
String::NewFromUtf8(isolate(), "file.js").ToLocalChecked(), isolate());
ScriptCompiler::Source source(source_text, origin);
Local<Module> module =
ScriptCompiler::CompileModule(isolate(), &source).ToLocalChecked();
CHECK_EQ(Module::kUninstantiated, module->GetStatus());
CHECK(module
->InstantiateModule(context(),
CompileSpecifierAsModuleResolveCallback)
.FromJust());
CHECK_EQ(Module::kInstantiated, module->GetStatus());
MaybeLocal<Value> result = module->Evaluate(context());
CHECK_EQ(Module::kEvaluated, module->GetStatus());
Local<Promise> promise = result.ToLocalChecked().As<Promise>();
CHECK_EQ(promise->State(), v8::Promise::kFulfilled);
CHECK(promise->Result()->IsUndefined());
// TODO(v8:12781): One IsInt32 matcher be added in
// gmock-support.h, we could use IsInt32 to replace
// this.
{
Local<Value> result = RunJS("Object.expando");
CHECK(result->IsInt32());
CHECK_EQ(10, result->Int32Value(context()).FromJust());
}
CHECK(!try_catch.HasCaught());
}
TEST_F(ModuleTest, ModuleEvaluationError1) {
HandleScope scope(isolate());
v8::TryCatch try_catch(isolate());
Local<String> source_text =
NewString("Object.x = (Object.x || 0) + 1; throw 'boom';");
ScriptOrigin origin = ModuleOrigin(
String::NewFromUtf8(isolate(), "file.js").ToLocalChecked(), isolate());
ScriptCompiler::Source source(source_text, origin);
Local<Module> module =
ScriptCompiler::CompileModule(isolate(), &source).ToLocalChecked();
CHECK_EQ(Module::kUninstantiated, module->GetStatus());
CHECK(module
->InstantiateModule(context(),
CompileSpecifierAsModuleResolveCallback)
.FromJust());
CHECK_EQ(Module::kInstantiated, module->GetStatus());
{
v8::TryCatch inner_try_catch(isolate());
MaybeLocal<Value> result = module->Evaluate(context());
CHECK_EQ(Module::kErrored, module->GetStatus());
Local<Value> exception = module->GetException();
CHECK(exception->StrictEquals(NewString("boom")));
// TODO(v8:12781): One IsInt32 matcher be added in
// gmock-support.h, we could use IsInt32 to replace
// this.
{
Local<Value> result = RunJS("Object.x");
CHECK(result->IsInt32());
CHECK_EQ(1, result->Int32Value(context()).FromJust());
}
// With top level await, we do not throw and errored evaluation returns
// a rejected promise with the exception.
CHECK(!inner_try_catch.HasCaught());
Local<Promise> promise = result.ToLocalChecked().As<Promise>();
CHECK_EQ(promise->State(), v8::Promise::kRejected);
CHECK_EQ(promise->Result(), module->GetException());
}
{
v8::TryCatch inner_try_catch(isolate());
MaybeLocal<Value> result = module->Evaluate(context());
CHECK_EQ(Module::kErrored, module->GetStatus());
Local<Value> exception = module->GetException();
CHECK(exception->StrictEquals(NewString("boom")));
// TODO(v8:12781): One IsInt32 matcher be added in
// gmock-support.h, we could use IsInt32 to replace
// this.
{
Local<Value> result = RunJS("Object.x");
CHECK(result->IsInt32());
CHECK_EQ(1, result->Int32Value(context()).FromJust());
}
// With top level await, we do not throw and errored evaluation returns
// a rejected promise with the exception.
CHECK(!inner_try_catch.HasCaught());
Local<Promise> promise = result.ToLocalChecked().As<Promise>();
CHECK_EQ(promise->State(), v8::Promise::kRejected);
CHECK_EQ(promise->Result(), module->GetException());
}
CHECK(!try_catch.HasCaught());
}
static Local<Module> failure_module;
static Local<Module> dependent_module;
MaybeLocal<Module> ResolveCallbackForModuleEvaluationError2(
Local<Context> context, Local<String> specifier,
Local<FixedArray> import_assertions, Local<Module> referrer) {
CHECK_EQ(0, import_assertions->Length());
Isolate* isolate = context->GetIsolate();
if (specifier->StrictEquals(
String::NewFromUtf8(isolate, "./failure.js").ToLocalChecked())) {
return failure_module;
} else {
CHECK(specifier->StrictEquals(
String::NewFromUtf8(isolate, "./dependent.js").ToLocalChecked()));
return dependent_module;
}
}
TEST_F(ModuleTest, ModuleEvaluationError2) {
HandleScope scope(isolate());
v8::TryCatch try_catch(isolate());
Local<String> failure_text = NewString("throw 'boom';");
ScriptOrigin failure_origin =
ModuleOrigin(NewString("failure.js"), isolate());
ScriptCompiler::Source failure_source(failure_text, failure_origin);
failure_module = ScriptCompiler::CompileModule(isolate(), &failure_source)
.ToLocalChecked();
CHECK_EQ(Module::kUninstantiated, failure_module->GetStatus());
CHECK(failure_module
->InstantiateModule(context(),
ResolveCallbackForModuleEvaluationError2)
.FromJust());
CHECK_EQ(Module::kInstantiated, failure_module->GetStatus());
{
v8::TryCatch inner_try_catch(isolate());
MaybeLocal<Value> result = failure_module->Evaluate(context());
CHECK_EQ(Module::kErrored, failure_module->GetStatus());
Local<Value> exception = failure_module->GetException();
CHECK(exception->StrictEquals(NewString("boom")));
// With top level await, we do not throw and errored evaluation returns
// a rejected promise with the exception.
CHECK(!inner_try_catch.HasCaught());
Local<Promise> promise = result.ToLocalChecked().As<Promise>();
CHECK_EQ(promise->State(), v8::Promise::kRejected);
CHECK_EQ(promise->Result(), failure_module->GetException());
}
Local<String> dependent_text =
NewString("import './failure.js'; export const c = 123;");
ScriptOrigin dependent_origin =
ModuleOrigin(NewString("dependent.js"), isolate());
ScriptCompiler::Source dependent_source(dependent_text, dependent_origin);
dependent_module = ScriptCompiler::CompileModule(isolate(), &dependent_source)
.ToLocalChecked();
CHECK_EQ(Module::kUninstantiated, dependent_module->GetStatus());
CHECK(dependent_module
->InstantiateModule(context(),
ResolveCallbackForModuleEvaluationError2)
.FromJust());
CHECK_EQ(Module::kInstantiated, dependent_module->GetStatus());
{
v8::TryCatch inner_try_catch(isolate());
MaybeLocal<Value> result = dependent_module->Evaluate(context());
CHECK_EQ(Module::kErrored, dependent_module->GetStatus());
Local<Value> exception = dependent_module->GetException();
CHECK(exception->StrictEquals(NewString("boom")));
CHECK_EQ(exception, failure_module->GetException());
// With top level await, we do not throw and errored evaluation returns
// a rejected promise with the exception.
CHECK(!inner_try_catch.HasCaught());
Local<Promise> promise = result.ToLocalChecked().As<Promise>();
CHECK_EQ(promise->State(), v8::Promise::kRejected);
CHECK_EQ(promise->Result(), failure_module->GetException());
CHECK(failure_module->GetException()->StrictEquals(NewString("boom")));
}
CHECK(!try_catch.HasCaught());
}
TEST_F(ModuleTest, ModuleEvaluationCompletion1) {
HandleScope scope(isolate());
v8::TryCatch try_catch(isolate());
const char* sources[] = {
"",
"var a = 1",
"import '42'",
"export * from '42'",
"export {} from '42'",
"export {}",
"var a = 1; export {a}",
"export function foo() {}",
"export class C extends null {}",
"export let a = 1",
"export default 1",
"export default function foo() {}",
"export default function () {}",
"export default (function () {})",
"export default class C extends null {}",
"export default (class C extends null {})",
"for (var i = 0; i < 5; ++i) {}",
};
for (auto src : sources) {
Local<String> source_text = NewString(src);
ScriptOrigin origin = ModuleOrigin(NewString("file.js"), isolate());
ScriptCompiler::Source source(source_text, origin);
Local<Module> module =
ScriptCompiler::CompileModule(isolate(), &source).ToLocalChecked();
CHECK_EQ(Module::kUninstantiated, module->GetStatus());
CHECK(module
->InstantiateModule(context(),
CompileSpecifierAsModuleResolveCallback)
.FromJust());
CHECK_EQ(Module::kInstantiated, module->GetStatus());
// Evaluate twice.
Local<Value> result_1 = module->Evaluate(context()).ToLocalChecked();
CHECK_EQ(Module::kEvaluated, module->GetStatus());
Local<Value> result_2 = module->Evaluate(context()).ToLocalChecked();
CHECK_EQ(Module::kEvaluated, module->GetStatus());
Local<Promise> promise = result_1.As<Promise>();
CHECK_EQ(promise->State(), v8::Promise::kFulfilled);
CHECK(promise->Result()->IsUndefined());
// Second evaluation should return the same promise.
Local<Promise> promise_too = result_2.As<Promise>();
CHECK_EQ(promise, promise_too);
CHECK_EQ(promise_too->State(), v8::Promise::kFulfilled);
CHECK(promise_too->Result()->IsUndefined());
}
CHECK(!try_catch.HasCaught());
}
TEST_F(ModuleTest, ModuleEvaluationCompletion2) {
HandleScope scope(isolate());
v8::TryCatch try_catch(isolate());
const char* sources[] = {
"'gaga'; ",
"'gaga'; var a = 1",
"'gaga'; import '42'",
"'gaga'; export * from '42'",
"'gaga'; export {} from '42'",
"'gaga'; export {}",
"'gaga'; var a = 1; export {a}",
"'gaga'; export function foo() {}",
"'gaga'; export class C extends null {}",
"'gaga'; export let a = 1",
"'gaga'; export default 1",
"'gaga'; export default function foo() {}",
"'gaga'; export default function () {}",
"'gaga'; export default (function () {})",
"'gaga'; export default class C extends null {}",
"'gaga'; export default (class C extends null {})",
};
for (auto src : sources) {
Local<String> source_text = NewString(src);
ScriptOrigin origin = ModuleOrigin(NewString("file.js"), isolate());
ScriptCompiler::Source source(source_text, origin);
Local<Module> module =
ScriptCompiler::CompileModule(isolate(), &source).ToLocalChecked();
CHECK_EQ(Module::kUninstantiated, module->GetStatus());
CHECK(module
->InstantiateModule(context(),
CompileSpecifierAsModuleResolveCallback)
.FromJust());
CHECK_EQ(Module::kInstantiated, module->GetStatus());
Local<Value> result_1 = module->Evaluate(context()).ToLocalChecked();
CHECK_EQ(Module::kEvaluated, module->GetStatus());
Local<Value> result_2 = module->Evaluate(context()).ToLocalChecked();
CHECK_EQ(Module::kEvaluated, module->GetStatus());
Local<Promise> promise = result_1.As<Promise>();
CHECK_EQ(promise->State(), v8::Promise::kFulfilled);
CHECK(promise->Result()->IsUndefined());
// Second Evaluation should return the same promise.
Local<Promise> promise_too = result_2.As<Promise>();
CHECK_EQ(promise, promise_too);
CHECK_EQ(promise_too->State(), v8::Promise::kFulfilled);
CHECK(promise_too->Result()->IsUndefined());
}
CHECK(!try_catch.HasCaught());
}
TEST_F(ModuleTest, ModuleNamespace) {
HandleScope scope(isolate());
v8::TryCatch try_catch(isolate());
Local<v8::Object> ReferenceError =
RunJS("ReferenceError")->ToObject(context()).ToLocalChecked();
Local<String> source_text = NewString(
"import {a, b} from 'export var a = 1; export let b = 2';"
"export function geta() {return a};"
"export function getb() {return b};"
"export let radio = 3;"
"export var gaga = 4;");
ScriptOrigin origin = ModuleOrigin(NewString("file.js"), isolate());
ScriptCompiler::Source source(source_text, origin);
Local<Module> module =
ScriptCompiler::CompileModule(isolate(), &source).ToLocalChecked();
CHECK_EQ(Module::kUninstantiated, module->GetStatus());
CHECK(module
->InstantiateModule(context(),
CompileSpecifierAsModuleResolveCallback)
.FromJust());
CHECK_EQ(Module::kInstantiated, module->GetStatus());
Local<Value> ns = module->GetModuleNamespace();
CHECK_EQ(Module::kInstantiated, module->GetStatus());
Local<v8::Object> nsobj = ns->ToObject(context()).ToLocalChecked();
CHECK_EQ(nsobj->GetCreationContext().ToLocalChecked(), context());
// a, b
CHECK(nsobj->Get(context(), NewString("a")).ToLocalChecked()->IsUndefined());
CHECK(nsobj->Get(context(), NewString("b")).ToLocalChecked()->IsUndefined());
// geta
{
auto geta = nsobj->Get(context(), NewString("geta")).ToLocalChecked();
auto a = geta.As<v8::Function>()
->Call(context(), geta, 0, nullptr)
.ToLocalChecked();
CHECK(a->IsUndefined());
}
// getb
{
v8::TryCatch inner_try_catch(isolate());
auto getb = nsobj->Get(context(), NewString("getb")).ToLocalChecked();
CHECK(getb.As<v8::Function>()->Call(context(), getb, 0, nullptr).IsEmpty());
CHECK(inner_try_catch.HasCaught());
CHECK(inner_try_catch.Exception()
->InstanceOf(context(), ReferenceError)
.FromJust());
}
// radio
{
v8::TryCatch inner_try_catch(isolate());
// https://bugs.chromium.org/p/v8/issues/detail?id=7235
// CHECK(nsobj->Get(context(), NewString("radio")).IsEmpty());
CHECK(nsobj->Get(context(), NewString("radio"))
.ToLocalChecked()
->IsUndefined());
CHECK(inner_try_catch.HasCaught());
CHECK(inner_try_catch.Exception()
->InstanceOf(context(), ReferenceError)
.FromJust());
}
// gaga
{
auto gaga = nsobj->Get(context(), NewString("gaga")).ToLocalChecked();
CHECK(gaga->IsUndefined());
}
CHECK(!try_catch.HasCaught());
CHECK_EQ(Module::kInstantiated, module->GetStatus());
module->Evaluate(context()).ToLocalChecked();
CHECK_EQ(Module::kEvaluated, module->GetStatus());
// geta
{
auto geta = nsobj->Get(context(), NewString("geta")).ToLocalChecked();
auto a = geta.As<v8::Function>()
->Call(context(), geta, 0, nullptr)
.ToLocalChecked();
CHECK_EQ(1, a->Int32Value(context()).FromJust());
}
// getb
{
auto getb = nsobj->Get(context(), NewString("getb")).ToLocalChecked();
auto b = getb.As<v8::Function>()
->Call(context(), getb, 0, nullptr)
.ToLocalChecked();
CHECK_EQ(2, b->Int32Value(context()).FromJust());
}
// radio
{
auto radio = nsobj->Get(context(), NewString("radio")).ToLocalChecked();
CHECK_EQ(3, radio->Int32Value(context()).FromJust());
}
// gaga
{
auto gaga = nsobj->Get(context(), NewString("gaga")).ToLocalChecked();
CHECK_EQ(4, gaga->Int32Value(context()).FromJust());
}
CHECK(!try_catch.HasCaught());
}
TEST_F(ModuleTest, ModuleEvaluationTopLevelAwait) {
HandleScope scope(isolate());
v8::TryCatch try_catch(isolate());
const char* sources[] = {
"await 42",
"import 'await 42';",
"import '42'; import 'await 42';",
};
for (auto src : sources) {
Local<String> source_text = NewString(src);
ScriptOrigin origin = ModuleOrigin(NewString("file.js"), isolate());
ScriptCompiler::Source source(source_text, origin);
Local<Module> module =
ScriptCompiler::CompileModule(isolate(), &source).ToLocalChecked();
CHECK_EQ(Module::kUninstantiated, module->GetStatus());
CHECK(module
->InstantiateModule(context(),
CompileSpecifierAsModuleResolveCallback)
.FromJust());
CHECK_EQ(Module::kInstantiated, module->GetStatus());
Local<Promise> promise =
Local<Promise>::Cast(module->Evaluate(context()).ToLocalChecked());
CHECK_EQ(Module::kEvaluated, module->GetStatus());
CHECK_EQ(promise->State(), v8::Promise::kFulfilled);
CHECK(promise->Result()->IsUndefined());
CHECK(!try_catch.HasCaught());
}
}
TEST_F(ModuleTest, ModuleEvaluationTopLevelAwaitError) {
HandleScope scope(isolate());
const char* sources[] = {
"await 42; throw 'boom';",
"import 'await 42; throw \"boom\";';",
"import '42'; import 'await 42; throw \"boom\";';",
};
for (auto src : sources) {
v8::TryCatch try_catch(isolate());
Local<String> source_text = NewString(src);
ScriptOrigin origin = ModuleOrigin(NewString("file.js"), isolate());
ScriptCompiler::Source source(source_text, origin);
Local<Module> module =
ScriptCompiler::CompileModule(isolate(), &source).ToLocalChecked();
CHECK_EQ(Module::kUninstantiated, module->GetStatus());
CHECK(module
->InstantiateModule(context(),
CompileSpecifierAsModuleResolveCallback)
.FromJust());
CHECK_EQ(Module::kInstantiated, module->GetStatus());
Local<Promise> promise =
Local<Promise>::Cast(module->Evaluate(context()).ToLocalChecked());
CHECK_EQ(Module::kErrored, module->GetStatus());
CHECK_EQ(promise->State(), v8::Promise::kRejected);
CHECK(promise->Result()->StrictEquals(NewString("boom")));
CHECK(module->GetException()->StrictEquals(NewString("boom")));
// TODO(cbruni) I am not sure, but this might not be supposed to throw
// because it is async.
CHECK(!try_catch.HasCaught());
}
}
namespace {
struct DynamicImportData {
DynamicImportData(Isolate* isolate_, Local<Promise::Resolver> resolver_,
Local<Context> context_, bool should_resolve_)
: isolate(isolate_), should_resolve(should_resolve_) {
resolver.Reset(isolate, resolver_);
context.Reset(isolate, context_);
}
Isolate* isolate;
v8::Global<Promise::Resolver> resolver;
v8::Global<Context> context;
bool should_resolve;
};
void DoHostImportModuleDynamically(void* import_data) {
std::unique_ptr<DynamicImportData> import_data_(
static_cast<DynamicImportData*>(import_data));
Isolate* isolate(import_data_->isolate);
HandleScope handle_scope(isolate);
Local<Promise::Resolver> resolver(import_data_->resolver.Get(isolate));
Local<Context> realm(import_data_->context.Get(isolate));
Context::Scope context_scope(realm);
if (import_data_->should_resolve) {
resolver->Resolve(realm, True(isolate)).ToChecked();
} else {
resolver
->Reject(realm, String::NewFromUtf8(isolate, "boom").ToLocalChecked())
.ToChecked();
}
}
v8::MaybeLocal<v8::Promise> HostImportModuleDynamicallyCallbackResolve(
Local<Context> context, Local<Data> host_defined_options,
Local<Value> resource_name, Local<String> specifier,
Local<FixedArray> import_assertions) {
Isolate* isolate = context->GetIsolate();
Local<v8::Promise::Resolver> resolver =
v8::Promise::Resolver::New(context).ToLocalChecked();
DynamicImportData* data =
new DynamicImportData(isolate, resolver, context, true);
isolate->EnqueueMicrotask(DoHostImportModuleDynamically, data);
return resolver->GetPromise();
}
v8::MaybeLocal<v8::Promise> HostImportModuleDynamicallyCallbackReject(
Local<Context> context, Local<Data> host_defined_options,
Local<Value> resource_name, Local<String> specifier,
Local<FixedArray> import_assertions) {
Isolate* isolate = context->GetIsolate();
Local<v8::Promise::Resolver> resolver =
v8::Promise::Resolver::New(context).ToLocalChecked();
DynamicImportData* data =
new DynamicImportData(isolate, resolver, context, false);
isolate->EnqueueMicrotask(DoHostImportModuleDynamically, data);
return resolver->GetPromise();
}
} // namespace
TEST_F(ModuleTest, ModuleEvaluationTopLevelAwaitDynamicImport) {
HandleScope scope(isolate());
isolate()->SetMicrotasksPolicy(v8::MicrotasksPolicy::kExplicit);
isolate()->SetHostImportModuleDynamicallyCallback(
HostImportModuleDynamicallyCallbackResolve);
v8::TryCatch try_catch(isolate());
const char* sources[] = {
"await import('foo');",
"import 'await import(\"foo\");';",
"import '42'; import 'await import(\"foo\");';",
};
for (auto src : sources) {
Local<String> source_text = NewString(src);
ScriptOrigin origin = ModuleOrigin(NewString("file.js"), isolate());
ScriptCompiler::Source source(source_text, origin);
Local<Module> module =
ScriptCompiler::CompileModule(isolate(), &source).ToLocalChecked();
CHECK_EQ(Module::kUninstantiated, module->GetStatus());
CHECK(module
->InstantiateModule(context(),
CompileSpecifierAsModuleResolveCallback)
.FromJust());
CHECK_EQ(Module::kInstantiated, module->GetStatus());
Local<Promise> promise =
Local<Promise>::Cast(module->Evaluate(context()).ToLocalChecked());
CHECK_EQ(Module::kEvaluated, module->GetStatus());
CHECK_EQ(promise->State(), v8::Promise::kPending);
CHECK(!try_catch.HasCaught());
isolate()->PerformMicrotaskCheckpoint();
CHECK_EQ(promise->State(), v8::Promise::kFulfilled);
}
}
TEST_F(ModuleTest, ModuleEvaluationTopLevelAwaitDynamicImportError) {
HandleScope scope(isolate());
isolate()->SetMicrotasksPolicy(v8::MicrotasksPolicy::kExplicit);
isolate()->SetHostImportModuleDynamicallyCallback(
HostImportModuleDynamicallyCallbackReject);
v8::TryCatch try_catch(isolate());
const char* sources[] = {
"await import('foo');",
"import 'await import(\"foo\");';",
"import '42'; import 'await import(\"foo\");';",
};
for (auto src : sources) {
Local<String> source_text = NewString(src);
ScriptOrigin origin = ModuleOrigin(NewString("file.js"), isolate());
ScriptCompiler::Source source(source_text, origin);
Local<Module> module =
ScriptCompiler::CompileModule(isolate(), &source).ToLocalChecked();
CHECK_EQ(Module::kUninstantiated, module->GetStatus());
CHECK(module
->InstantiateModule(context(),
CompileSpecifierAsModuleResolveCallback)
.FromJust());
CHECK_EQ(Module::kInstantiated, module->GetStatus());
Local<Promise> promise =
Local<Promise>::Cast(module->Evaluate(context()).ToLocalChecked());
CHECK_EQ(Module::kEvaluated, module->GetStatus());
CHECK_EQ(promise->State(), v8::Promise::kPending);
CHECK(!try_catch.HasCaught());
isolate()->PerformMicrotaskCheckpoint();
CHECK_EQ(Module::kErrored, module->GetStatus());
CHECK_EQ(promise->State(), v8::Promise::kRejected);
CHECK(promise->Result()->StrictEquals(NewString("boom")));
CHECK(module->GetException()->StrictEquals(NewString("boom")));
CHECK(!try_catch.HasCaught());
}
}
TEST_F(ModuleTest, TerminateExecutionTopLevelAwaitSync) {
HandleScope scope(isolate());
v8::TryCatch try_catch(isolate());
context()
->Global()
->Set(context(), NewString("terminate"),
v8::Function::New(context(),
[](const v8::FunctionCallbackInfo<Value>& info) {
info.GetIsolate()->TerminateExecution();
})
.ToLocalChecked())
.ToChecked();
Local<String> source_text = NewString("terminate(); while (true) {}");
ScriptOrigin origin = ModuleOrigin(NewString("file.js"), isolate());
ScriptCompiler::Source source(source_text, origin);
Local<Module> module =
ScriptCompiler::CompileModule(isolate(), &source).ToLocalChecked();
CHECK(module
->InstantiateModule(context(),
CompileSpecifierAsModuleResolveCallback)
.FromJust());
CHECK(module->Evaluate(context()).IsEmpty());
CHECK(try_catch.HasCaught());
CHECK(try_catch.HasTerminated());
CHECK_EQ(module->GetStatus(), Module::kErrored);
CHECK_EQ(module->GetException(), v8::Null(isolate()));
}
TEST_F(ModuleTest, TerminateExecutionTopLevelAwaitAsync) {
HandleScope scope(isolate());
v8::TryCatch try_catch(isolate());
context()
->Global()
->Set(context(), NewString("terminate"),
v8::Function::New(context(),
[](const v8::FunctionCallbackInfo<Value>& info) {
info.GetIsolate()->TerminateExecution();
})
.ToLocalChecked())
.ToChecked();
Local<Promise::Resolver> eval_promise =
Promise::Resolver::New(context()).ToLocalChecked();
context()
->Global()
->Set(context(), NewString("evalPromise"), eval_promise)
.ToChecked();
Local<String> source_text =
NewString("await evalPromise; terminate(); while (true) {}");
ScriptOrigin origin = ModuleOrigin(NewString("file.js"), isolate());
ScriptCompiler::Source source(source_text, origin);
Local<Module> module =
ScriptCompiler::CompileModule(isolate(), &source).ToLocalChecked();
CHECK(module
->InstantiateModule(context(),
CompileSpecifierAsModuleResolveCallback)
.FromJust());
Local<Promise> promise =
Local<Promise>::Cast(module->Evaluate(context()).ToLocalChecked());
CHECK_EQ(module->GetStatus(), Module::kEvaluated);
CHECK_EQ(promise->State(), Promise::PromiseState::kPending);
CHECK(!try_catch.HasCaught());
CHECK(!try_catch.HasTerminated());
eval_promise->Resolve(context(), v8::Undefined(isolate())).ToChecked();
CHECK(try_catch.HasCaught());
CHECK(try_catch.HasTerminated());
CHECK_EQ(promise->State(), Promise::PromiseState::kPending);
// The termination exception doesn't trigger the module's
// catch handler, so the module isn't transitioned to kErrored.
CHECK_EQ(module->GetStatus(), Module::kEvaluated);
}
static Local<Module> async_leaf_module;
static Local<Module> sync_leaf_module;
static Local<Module> cycle_self_module;
static Local<Module> cycle_one_module;
static Local<Module> cycle_two_module;
MaybeLocal<Module> ResolveCallbackForIsGraphAsyncTopLevelAwait(
Local<Context> context, Local<String> specifier,
Local<FixedArray> import_assertions, Local<Module> referrer) {
CHECK_EQ(0, import_assertions->Length());
Isolate* isolate = context->GetIsolate();
if (specifier->StrictEquals(
String::NewFromUtf8(isolate, "./async_leaf.js").ToLocalChecked())) {
return async_leaf_module;
} else if (specifier->StrictEquals(
String::NewFromUtf8(isolate, "./sync_leaf.js")
.ToLocalChecked())) {
return sync_leaf_module;
} else if (specifier->StrictEquals(
String::NewFromUtf8(isolate, "./cycle_self.js")
.ToLocalChecked())) {
return cycle_self_module;
} else if (specifier->StrictEquals(
String::NewFromUtf8(isolate, "./cycle_one.js")
.ToLocalChecked())) {
return cycle_one_module;
} else {
CHECK(specifier->StrictEquals(
String::NewFromUtf8(isolate, "./cycle_two.js").ToLocalChecked()));
return cycle_two_module;
}
}
TEST_F(ModuleTest, IsGraphAsyncTopLevelAwait) {
HandleScope scope(isolate());
{
Local<String> source_text = NewString("await notExecuted();");
ScriptOrigin origin = ModuleOrigin(NewString("async_leaf.js"), isolate());
ScriptCompiler::Source source(source_text, origin);
async_leaf_module =
ScriptCompiler::CompileModule(isolate(), &source).ToLocalChecked();
CHECK(async_leaf_module
->InstantiateModule(context(),
ResolveCallbackForIsGraphAsyncTopLevelAwait)
.FromJust());
CHECK(async_leaf_module->IsGraphAsync());
}
{
Local<String> source_text = NewString("notExecuted();");
ScriptOrigin origin = ModuleOrigin(NewString("sync_leaf.js"), isolate());
ScriptCompiler::Source source(source_text, origin);
sync_leaf_module =
ScriptCompiler::CompileModule(isolate(), &source).ToLocalChecked();
CHECK(sync_leaf_module
->InstantiateModule(context(),
ResolveCallbackForIsGraphAsyncTopLevelAwait)
.FromJust());
CHECK(!sync_leaf_module->IsGraphAsync());
}
{
Local<String> source_text = NewString("import './async_leaf.js'");
ScriptOrigin origin = ModuleOrigin(NewString("import_async.js"), isolate());
ScriptCompiler::Source source(source_text, origin);
Local<Module> module =
ScriptCompiler::CompileModule(isolate(), &source).ToLocalChecked();
CHECK(module
->InstantiateModule(context(),
ResolveCallbackForIsGraphAsyncTopLevelAwait)
.FromJust());
CHECK(module->IsGraphAsync());
}
{
Local<String> source_text = NewString("import './sync_leaf.js'");
ScriptOrigin origin = ModuleOrigin(NewString("import_sync.js"), isolate());
ScriptCompiler::Source source(source_text, origin);
Local<Module> module =
ScriptCompiler::CompileModule(isolate(), &source).ToLocalChecked();
CHECK(module
->InstantiateModule(context(),
ResolveCallbackForIsGraphAsyncTopLevelAwait)
.FromJust());
CHECK(!module->IsGraphAsync());
}
{
Local<String> source_text = NewString(
"import './cycle_self.js'\n"
"import './async_leaf.js'");
ScriptOrigin origin = ModuleOrigin(NewString("cycle_self.js"), isolate());
ScriptCompiler::Source source(source_text, origin);
cycle_self_module =
ScriptCompiler::CompileModule(isolate(), &source).ToLocalChecked();
CHECK(cycle_self_module
->InstantiateModule(context(),
ResolveCallbackForIsGraphAsyncTopLevelAwait)
.FromJust());
CHECK(cycle_self_module->IsGraphAsync());
}
{
Local<String> source_text1 = NewString("import './cycle_two.js'");
ScriptOrigin origin1 = ModuleOrigin(NewString("cycle_one.js"), isolate());
ScriptCompiler::Source source1(source_text1, origin1);
cycle_one_module =
ScriptCompiler::CompileModule(isolate(), &source1).ToLocalChecked();
Local<String> source_text2 = NewString(
"import './cycle_one.js'\n"
"import './async_leaf.js'");
ScriptOrigin origin2 = ModuleOrigin(NewString("cycle_two.js"), isolate());
ScriptCompiler::Source source2(source_text2, origin2);
cycle_two_module =
ScriptCompiler::CompileModule(isolate(), &source2).ToLocalChecked();
CHECK(cycle_one_module
->InstantiateModule(context(),
ResolveCallbackForIsGraphAsyncTopLevelAwait)
.FromJust());
CHECK(cycle_one_module->IsGraphAsync());
CHECK(cycle_two_module->IsGraphAsync());
}
}
} // anonymous namespace