Как включить и использовать модули ES6 в движке JavaScript V8? - PullRequest
0 голосов
/ 26 августа 2018

Я использую встроенный движок V8 в своем приложении (Windows Desktop) C ++. Я понимаю, что V8 имеет поддержку модулей ES6. Как мне активировать и использовать эту функцию в моем приложении?

Я бы не ожидал, что у кого-либо будет полный проработанный пример того, как это работает, но ответ высокого уровня, указывающий мне (и будущим читателям) в правильном направлении, полностью оправдает мои надежды и стремления к этому вопросу.

1 Ответ

0 голосов
/ 27 августа 2018

В ряду реальных примеров из V8 (я действительно планировал написать некоторые в какой-то момент), я напишу один здесь.Для некоторых примеров использования в дикой природе я рекомендую реализацию Node.js , или мою собственную , обе из которых используют очень похожие макеты (написанные одними и теми же людьми).Существует также реализация в D8, отладчике CLI V8 .

Local<String> source_text = String::NewFromUtf8(
    isolate, "import 'some thing'; 1 + 1");

ScriptOrigin origin(String::NewFromUtf8("main.mjs"),      // specifier
                    Integer::New(isolate, 0),             // line offset
                    Integer::New(isolate, 0),             // column offset
                    False(isolate),                       // is cross origin
                    Local<Integer>(),                     // script id
                    Local<Value>(),                       // source map URL
                    False(isolate),                       // is opaque
                    False(isolate),                       // is WASM
                    True(isolate));                       // is ES6 module
Context::Scope context_scope(context);
ScriptCompiler::Source source(source_text, origin);
Local<Module> module;
if (!ScriptCompiler::CompileModule(isolate, &source).ToLocal(&module)) {
  // if you have a v8::TryCatch, you should check it here.
  return;
}

// You can resolve import requests ahead of time (useful for async)
for (int i = 0; i < module->GetModuleRequestsLength(); i++) {
  Local<String> specifier = module->GetModuleRequest(i); // "some thing"
}

// or you can resolve them sync in the InstantiateModule callback
module->InstantiateModule(context, [](Local<Context> context, // "main.mjs"
                                      Local<String> specifier, // "some thing"
                                      Local<Module> referrer) {
  return Local<Module>();
});

// setting this callback enables dynamic import
isolate->SetImportModuleDynamicallyCallback([](Local<Context> context,
                                               Local<ScriptOrModule> referrer,
                                               Local<String> specifier) {
  return MaybeLocal<Promise>();
});

// setting this callback enables import.meta
isolate->SetHostInitializeImportMetaObjectCallback([](Local<Context> context,
                                                      Local<Module> module,
                                                      Local<Object> meta) {
  // meta->Set(key, value); you could set import.meta.url here
});

Local<Value> result;
if (module->Evaluate(context).ToLocal(&result)) {
  String::Utf8Value utf8(isolate, result);
  printf("module eval result: %s\n", *utf8);
} else {
  // once again, if you have a v8::TryCatch, use it here.
}
...