| Index: test/inspector/inspector-test.cc
|
| diff --git a/test/inspector/inspector-test.cc b/test/inspector/inspector-test.cc
|
| new file mode 100644
|
| index 0000000000000000000000000000000000000000..58486cf493e4d8452143056b24263f7b48aea78b
|
| --- /dev/null
|
| +++ b/test/inspector/inspector-test.cc
|
| @@ -0,0 +1,226 @@
|
| +// Copyright 2016 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.
|
| +
|
| +#if !defined(_WIN32) && !defined(_WIN64)
|
| +#include <unistd.h> // NOLINT
|
| +#endif // !defined(_WIN32) && !defined(_WIN64)
|
| +
|
| +#include "include/libplatform/libplatform.h"
|
| +#include "include/v8.h"
|
| +
|
| +#include "src/base/platform/platform.h"
|
| +#include "src/flags.h"
|
| +#include "src/utils.h"
|
| +#include "src/vector.h"
|
| +
|
| +#include "test/inspector/inspector-impl.h"
|
| +#include "test/inspector/task-runner.h"
|
| +
|
| +namespace {
|
| +
|
| +void Exit() {
|
| + fflush(stdout);
|
| + fflush(stderr);
|
| + _exit(0);
|
| +}
|
| +
|
| +class UtilsExtension : public v8::Extension {
|
| + public:
|
| + UtilsExtension()
|
| + : v8::Extension("v8_inspector/utils",
|
| + "native function print(); native function quit();") {}
|
| + virtual v8::Local<v8::FunctionTemplate> GetNativeFunctionTemplate(
|
| + v8::Isolate* isolate, v8::Local<v8::String> name) {
|
| + v8::Local<v8::Context> context = isolate->GetCurrentContext();
|
| + if (name->Equals(context, v8::String::NewFromUtf8(
|
| + isolate, "print", v8::NewStringType::kNormal)
|
| + .ToLocalChecked())
|
| + .FromJust()) {
|
| + return v8::FunctionTemplate::New(isolate, UtilsExtension::Print);
|
| + } else if (name->Equals(context,
|
| + v8::String::NewFromUtf8(isolate, "quit",
|
| + v8::NewStringType::kNormal)
|
| + .ToLocalChecked())
|
| + .FromJust()) {
|
| + return v8::FunctionTemplate::New(isolate, UtilsExtension::Quit);
|
| + }
|
| + return v8::Local<v8::FunctionTemplate>();
|
| + }
|
| +
|
| + private:
|
| + static void Print(const v8::FunctionCallbackInfo<v8::Value>& args) {
|
| + for (int i = 0; i < args.Length(); i++) {
|
| + v8::HandleScope handle_scope(args.GetIsolate());
|
| + if (i != 0) {
|
| + printf(" ");
|
| + }
|
| +
|
| + // Explicitly catch potential exceptions in toString().
|
| + v8::TryCatch try_catch(args.GetIsolate());
|
| + v8::Local<v8::Value> arg = args[i];
|
| + v8::Local<v8::String> str_obj;
|
| +
|
| + if (arg->IsSymbol()) {
|
| + arg = v8::Local<v8::Symbol>::Cast(arg)->Name();
|
| + }
|
| + if (!arg->ToString(args.GetIsolate()->GetCurrentContext())
|
| + .ToLocal(&str_obj)) {
|
| + try_catch.ReThrow();
|
| + return;
|
| + }
|
| +
|
| + v8::String::Utf8Value str(str_obj);
|
| + int n =
|
| + static_cast<int>(fwrite(*str, sizeof(**str), str.length(), stdout));
|
| + if (n != str.length()) {
|
| + printf("Error in fwrite\n");
|
| + Quit(args);
|
| + }
|
| + }
|
| + printf("\n");
|
| + fflush(stdout);
|
| + }
|
| +
|
| + static void Quit(const v8::FunctionCallbackInfo<v8::Value>& args) { Exit(); }
|
| +};
|
| +
|
| +class SetTimeoutTask : public TaskRunner::Task {
|
| + public:
|
| + SetTimeoutTask(v8::Isolate* isolate, v8::Local<v8::Function> function)
|
| + : function_(isolate, function) {}
|
| + virtual ~SetTimeoutTask() {}
|
| +
|
| + bool is_inspector_task() final { return false; }
|
| +
|
| + void Run(v8::Isolate* isolate,
|
| + const v8::Global<v8::Context>& global_context) override {
|
| + v8::MicrotasksScope microtasks_scope(isolate,
|
| + v8::MicrotasksScope::kRunMicrotasks);
|
| + v8::HandleScope handle_scope(isolate);
|
| + v8::Local<v8::Context> context = global_context.Get(isolate);
|
| + v8::Context::Scope context_scope(context);
|
| +
|
| + v8::Local<v8::Function> function = function_.Get(isolate);
|
| + v8::MaybeLocal<v8::Value> result;
|
| + v8_inspector::V8Inspector* inspector =
|
| + InspectorClientImpl::InspectorFromContext(context);
|
| + if (inspector) inspector->willExecuteScript(context, function->ScriptId());
|
| + result = function->Call(context, context->Global(), 0, nullptr);
|
| + if (inspector) inspector->didExecuteScript(context);
|
| + }
|
| +
|
| + private:
|
| + v8::Global<v8::Function> function_;
|
| +};
|
| +
|
| +class SetTimeoutExtension : public v8::Extension {
|
| + public:
|
| + SetTimeoutExtension()
|
| + : v8::Extension("v8_inspector/setTimeout",
|
| + "native function setTimeout();") {}
|
| +
|
| + virtual v8::Local<v8::FunctionTemplate> GetNativeFunctionTemplate(
|
| + v8::Isolate* isolate, v8::Local<v8::String> name) {
|
| + return v8::FunctionTemplate::New(isolate, SetTimeoutExtension::SetTimeout);
|
| + }
|
| +
|
| + private:
|
| + static void SetTimeout(const v8::FunctionCallbackInfo<v8::Value>& args) {
|
| + v8::Isolate* isolate = args.GetIsolate();
|
| + if (args.Length() != 2 || !args[1]->IsNumber() || !args[0]->IsFunction() ||
|
| + args[1].As<v8::Number>()->Value() != 0.0) {
|
| + fprintf(stderr,
|
| + "Internal error: only setTimeout(function, 0) is supported.");
|
| + Exit();
|
| + }
|
| + v8::Local<v8::Context> context = args.GetIsolate()->GetCurrentContext();
|
| + TaskRunner::FromContext(context)->Append(new SetTimeoutTask(
|
| + args.GetIsolate(), v8::Local<v8::Function>::Cast(args[0])));
|
| + }
|
| +};
|
| +
|
| +v8_inspector::String16 ToString16(const v8_inspector::StringView& string) {
|
| + if (string.is8Bit())
|
| + return v8_inspector::String16(
|
| + reinterpret_cast<const char*>(string.characters8()), string.length());
|
| + return v8_inspector::String16(
|
| + reinterpret_cast<const uint16_t*>(string.characters16()),
|
| + string.length());
|
| +}
|
| +
|
| +class FrontendChannelImpl : public InspectorClientImpl::FrontendChannel {
|
| + public:
|
| + explicit FrontendChannelImpl(TaskRunner* frontend_task_runner)
|
| + : frontend_task_runner_(frontend_task_runner) {}
|
| + virtual ~FrontendChannelImpl() {}
|
| +
|
| + void SendMessageToFrontend(const v8_inspector::StringView& message) final {
|
| + v8_inspector::String16Builder script;
|
| + script.append("InspectorTest.dispatchMessage(");
|
| + script.append(ToString16(message));
|
| + script.append(")");
|
| + frontend_task_runner_->Append(new ExecuteStringTask(script.toString()));
|
| + }
|
| +
|
| + private:
|
| + TaskRunner* frontend_task_runner_;
|
| +};
|
| +
|
| +} // namespace
|
| +
|
| +int main(int argc, char* argv[]) {
|
| + v8::V8::InitializeICUDefaultLocation(argv[0]);
|
| + v8::Platform* platform = v8::platform::CreateDefaultPlatform();
|
| + v8::V8::InitializePlatform(platform);
|
| + v8::internal::FlagList::SetFlagsFromCommandLine(&argc, argv, true);
|
| + v8::V8::InitializeExternalStartupData(argv[0]);
|
| + v8::V8::Initialize();
|
| +
|
| + SetTimeoutExtension set_timeout_extension;
|
| + v8::RegisterExtension(&set_timeout_extension);
|
| + UtilsExtension utils_extension;
|
| + v8::RegisterExtension(&utils_extension);
|
| + SendMessageToBackendExtension send_message_to_backend_extension;
|
| + v8::RegisterExtension(&send_message_to_backend_extension);
|
| +
|
| + v8::base::Semaphore ready_semaphore(0);
|
| +
|
| + const char* backend_extensions[] = {"v8_inspector/setTimeout"};
|
| + v8::ExtensionConfiguration backend_configuration(
|
| + arraysize(backend_extensions), backend_extensions);
|
| + TaskRunner backend_runner(&backend_configuration, &ready_semaphore);
|
| + ready_semaphore.Wait();
|
| + SendMessageToBackendExtension::set_backend_task_runner(&backend_runner);
|
| +
|
| + const char* frontend_extensions[] = {"v8_inspector/utils",
|
| + "v8_inspector/frontend"};
|
| + v8::ExtensionConfiguration frontend_configuration(
|
| + arraysize(frontend_extensions), frontend_extensions);
|
| + TaskRunner frontend_runner(&frontend_configuration, &ready_semaphore);
|
| + ready_semaphore.Wait();
|
| +
|
| + FrontendChannelImpl frontend_channel(&frontend_runner);
|
| + InspectorClientImpl inspector_client(&backend_runner, &frontend_channel,
|
| + &ready_semaphore);
|
| + ready_semaphore.Wait();
|
| +
|
| + for (int i = 1; i < argc; ++i) {
|
| + if (argv[i][0] == '-') break;
|
| +
|
| + bool exists = false;
|
| + v8::internal::Vector<const char> chars =
|
| + v8::internal::ReadFile(argv[i], &exists, true);
|
| + if (!exists) {
|
| + fprintf(stderr, "Internal error: script file doesn't exists: %s\n",
|
| + argv[i]);
|
| + Exit();
|
| + }
|
| + v8_inspector::String16 source =
|
| + v8_inspector::String16::fromUTF8(chars.start(), chars.length());
|
| + frontend_runner.Append(new ExecuteStringTask(source));
|
| + }
|
| +
|
| + frontend_runner.Join();
|
| + return 0;
|
| +}
|
|
|