// Copyright (c) 2014 GitHub, Inc. // Use of this source code is governed by the MIT license that can be // found in the LICENSE file. #include #include #include #include "base/files/file_util.h" #include "base/task/thread_pool.h" #include "base/threading/thread_restrictions.h" #include "base/trace_event/trace_config.h" #include "content/public/browser/tracing_controller.h" #include "shell/common/gin_converters/callback_converter.h" #include "shell/common/gin_converters/file_path_converter.h" #include "shell/common/gin_converters/value_converter.h" #include "shell/common/gin_helper/dictionary.h" #include "shell/common/gin_helper/promise.h" #include "shell/common/node_includes.h" #include "third_party/abseil-cpp/absl/types/optional.h" using content::TracingController; namespace gin { template <> struct Converter { static bool FromV8(v8::Isolate* isolate, v8::Local val, base::trace_event::TraceConfig* out) { // (alexeykuzmin): A combination of "categoryFilter" and "traceOptions" // has to be checked first because none of the fields // in the `memory_dump_config` dict below are mandatory // and we cannot check the config format. gin_helper::Dictionary options; if (ConvertFromV8(isolate, val, &options)) { std::string category_filter, trace_options; if (options.Get("categoryFilter", &category_filter) && options.Get("traceOptions", &trace_options)) { *out = base::trace_event::TraceConfig(category_filter, trace_options); return true; } } base::Value::Dict memory_dump_config; if (ConvertFromV8(isolate, val, &memory_dump_config)) { *out = base::trace_event::TraceConfig( base::Value(std::move(memory_dump_config))); return true; } return false; } }; } // namespace gin namespace { using CompletionCallback = base::OnceCallback; absl::optional CreateTemporaryFileOnIO() { base::FilePath temp_file_path; if (!base::CreateTemporaryFile(&temp_file_path)) return absl::nullopt; return absl::make_optional(std::move(temp_file_path)); } void StopTracing(gin_helper::Promise promise, absl::optional file_path) { auto resolve_or_reject = base::BindOnce( [](gin_helper::Promise promise, const base::FilePath& path, absl::optional error) { if (error) { promise.RejectWithErrorMessage(error.value()); } else { promise.Resolve(path); } }, std::move(promise), *file_path); if (file_path) { auto split_callback = base::SplitOnceCallback(std::move(resolve_or_reject)); auto endpoint = TracingController::CreateFileEndpoint( *file_path, base::BindOnce(std::move(split_callback.first), absl::nullopt)); if (!TracingController::GetInstance()->StopTracing(endpoint)) { std::move(split_callback.second) .Run(absl::make_optional( "Failed to stop tracing (was a trace in progress?)")); } } else { std::move(resolve_or_reject) .Run(absl::make_optional( "Failed to create temporary file for trace data")); } } v8::Local StopRecording(gin_helper::Arguments* args) { gin_helper::Promise promise(args->isolate()); v8::Local handle = promise.GetHandle(); base::FilePath path; if (args->GetNext(&path) && !path.empty()) { StopTracing(std::move(promise), absl::make_optional(path)); } else { // use a temporary file. base::ThreadPool::PostTaskAndReplyWithResult( FROM_HERE, {base::MayBlock(), base::TaskPriority::USER_VISIBLE}, base::BindOnce(CreateTemporaryFileOnIO), base::BindOnce(StopTracing, std::move(promise))); } return handle; } v8::Local GetCategories(v8::Isolate* isolate) { gin_helper::Promise&> promise(isolate); v8::Local handle = promise.GetHandle(); // Note: This method always succeeds. TracingController::GetInstance()->GetCategories(base::BindOnce( gin_helper::Promise&>::ResolvePromise, std::move(promise))); return handle; } v8::Local StartTracing( v8::Isolate* isolate, const base::trace_event::TraceConfig& trace_config) { gin_helper::Promise promise(isolate); v8::Local handle = promise.GetHandle(); if (!TracingController::GetInstance()->StartTracing( trace_config, base::BindOnce(gin_helper::Promise::ResolvePromise, std::move(promise)))) { // If StartTracing returns false, that means it didn't invoke its callback. // Return an already-resolved promise and abandon the previous promise (it // was std::move()d into the StartTracing callback and has been deleted by // this point). return gin_helper::Promise::ResolvedPromise(isolate); } return handle; } void OnTraceBufferUsageAvailable( gin_helper::Promise promise, float percent_full, size_t approximate_count) { gin_helper::Dictionary dict = gin::Dictionary::CreateEmpty(promise.isolate()); dict.Set("percentage", percent_full); dict.Set("value", approximate_count); promise.Resolve(dict); } v8::Local GetTraceBufferUsage(v8::Isolate* isolate) { gin_helper::Promise promise(isolate); v8::Local handle = promise.GetHandle(); // Note: This method always succeeds. TracingController::GetInstance()->GetTraceBufferUsage( base::BindOnce(&OnTraceBufferUsageAvailable, std::move(promise))); return handle; } void Initialize(v8::Local exports, v8::Local unused, v8::Local context, void* priv) { gin_helper::Dictionary dict(context->GetIsolate(), exports); dict.SetMethod("getCategories", &GetCategories); dict.SetMethod("startRecording", &StartTracing); dict.SetMethod("stopRecording", &StopRecording); dict.SetMethod("getTraceBufferUsage", &GetTraceBufferUsage); } } // namespace NODE_LINKED_BINDING_CONTEXT_AWARE(electron_browser_content_tracing, Initialize)