| 
									
										
										
										
											2018-06-18 18:45:58 -07:00
										 |  |  | // Copyright (c) 2018 GitHub, Inc.
 | 
					
						
							|  |  |  | // Use of this source code is governed by the MIT license that can be
 | 
					
						
							|  |  |  | // found in the LICENSE file.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-19 13:46:59 -07:00
										 |  |  | #include "shell/browser/api/electron_api_net_log.h"
 | 
					
						
							| 
									
										
										
										
											2018-09-12 19:25:56 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-31 18:34:44 -07:00
										 |  |  | #include <string>
 | 
					
						
							| 
									
										
										
										
											2018-09-12 19:25:56 -05:00
										 |  |  | #include <utility>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-04 23:38:56 +05:30
										 |  |  | #include "base/command_line.h"
 | 
					
						
							| 
									
										
										
										
											2021-06-20 18:52:28 -07:00
										 |  |  | #include "base/files/file_path.h"
 | 
					
						
							| 
									
										
										
										
											2023-02-03 12:43:42 +01:00
										 |  |  | #include "base/functional/bind.h"
 | 
					
						
							| 
									
										
										
										
											2020-03-09 16:13:59 +00:00
										 |  |  | #include "base/task/thread_pool.h"
 | 
					
						
							| 
									
										
										
										
											2018-11-07 18:53:09 +05:30
										 |  |  | #include "chrome/browser/browser_process.h"
 | 
					
						
							| 
									
										
										
										
											2018-10-04 23:38:56 +05:30
										 |  |  | #include "components/net_log/chrome_net_log.h"
 | 
					
						
							|  |  |  | #include "content/public/browser/storage_partition.h"
 | 
					
						
							| 
									
										
										
										
											2019-06-19 14:31:55 -07:00
										 |  |  | #include "electron/electron_version.h"
 | 
					
						
							| 
									
										
										
										
											2020-03-18 16:46:05 -07:00
										 |  |  | #include "gin/object_template_builder.h"
 | 
					
						
							| 
									
										
										
										
											2021-06-20 18:52:28 -07:00
										 |  |  | #include "net/log/net_log_capture_mode.h"
 | 
					
						
							| 
									
										
										
										
											2019-06-19 13:46:59 -07:00
										 |  |  | #include "shell/browser/electron_browser_context.h"
 | 
					
						
							|  |  |  | #include "shell/browser/net/system_network_context_manager.h"
 | 
					
						
							| 
									
										
										
										
											2019-10-25 22:03:28 +09:00
										 |  |  | #include "shell/common/gin_converters/file_path_converter.h"
 | 
					
						
							|  |  |  | #include "shell/common/gin_helper/dictionary.h"
 | 
					
						
							| 
									
										
										
										
											2019-06-19 13:46:59 -07:00
										 |  |  | #include "shell/common/node_includes.h"
 | 
					
						
							| 
									
										
										
										
											2018-06-18 18:45:58 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-25 22:03:28 +09:00
										 |  |  | namespace gin { | 
					
						
							| 
									
										
										
										
											2019-07-25 16:06:39 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | template <> | 
					
						
							|  |  |  | struct Converter<net::NetLogCaptureMode> { | 
					
						
							|  |  |  |   static bool FromV8(v8::Isolate* isolate, | 
					
						
							|  |  |  |                      v8::Local<v8::Value> val, | 
					
						
							|  |  |  |                      net::NetLogCaptureMode* out) { | 
					
						
							|  |  |  |     std::string type; | 
					
						
							|  |  |  |     if (!ConvertFromV8(isolate, val, &type)) | 
					
						
							|  |  |  |       return false; | 
					
						
							|  |  |  |     if (type == "default") | 
					
						
							|  |  |  |       *out = net::NetLogCaptureMode::kDefault; | 
					
						
							|  |  |  |     else if (type == "includeSensitive") | 
					
						
							|  |  |  |       *out = net::NetLogCaptureMode::kIncludeSensitive; | 
					
						
							|  |  |  |     else if (type == "everything") | 
					
						
							|  |  |  |       *out = net::NetLogCaptureMode::kEverything; | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |       return false; | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-25 22:03:28 +09:00
										 |  |  | }  // namespace gin
 | 
					
						
							| 
									
										
										
										
											2019-07-25 16:06:39 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-18 18:45:58 -07:00
										 |  |  | namespace electron { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-23 15:31:38 -07:00
										 |  |  | namespace { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | scoped_refptr<base::SequencedTaskRunner> CreateFileTaskRunner() { | 
					
						
							|  |  |  |   // The tasks posted to this sequenced task runner do synchronous File I/O for
 | 
					
						
							|  |  |  |   // checking paths and setting permissions on files.
 | 
					
						
							|  |  |  |   //
 | 
					
						
							|  |  |  |   // These operations can be skipped on shutdown since FileNetLogObserver's API
 | 
					
						
							|  |  |  |   // doesn't require things to have completed until notified of completion.
 | 
					
						
							| 
									
										
										
										
											2020-03-09 16:13:59 +00:00
										 |  |  |   return base::ThreadPool::CreateSequencedTaskRunner( | 
					
						
							|  |  |  |       {base::MayBlock(), base::TaskPriority::USER_VISIBLE, | 
					
						
							| 
									
										
										
										
											2019-05-23 15:31:38 -07:00
										 |  |  |        base::TaskShutdownBehavior::SKIP_ON_SHUTDOWN}); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | base::File OpenFileForWriting(base::FilePath path) { | 
					
						
							|  |  |  |   return base::File(path, | 
					
						
							|  |  |  |                     base::File::FLAG_CREATE_ALWAYS | base::File::FLAG_WRITE); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-01 15:10:32 +09:00
										 |  |  | void ResolvePromiseWithNetError(gin_helper::Promise<void> promise, | 
					
						
							|  |  |  |                                 int32_t error) { | 
					
						
							| 
									
										
										
										
											2019-05-23 15:31:38 -07:00
										 |  |  |   if (error == net::OK) { | 
					
						
							|  |  |  |     promise.Resolve(); | 
					
						
							|  |  |  |   } else { | 
					
						
							|  |  |  |     promise.RejectWithErrorMessage(net::ErrorToString(error)); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | }  // namespace
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-18 18:45:58 -07:00
										 |  |  | namespace api { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-18 16:46:05 -07:00
										 |  |  | gin::WrapperInfo NetLog::kWrapperInfo = {gin::kEmbedderNativeGin}; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-04 23:38:56 +05:30
										 |  |  | NetLog::NetLog(v8::Isolate* isolate, ElectronBrowserContext* browser_context) | 
					
						
							| 
									
										
										
										
											2021-01-26 19:16:21 +01:00
										 |  |  |     : browser_context_(browser_context) { | 
					
						
							| 
									
										
										
										
											2019-05-23 15:31:38 -07:00
										 |  |  |   file_task_runner_ = CreateFileTaskRunner(); | 
					
						
							| 
									
										
										
										
											2018-06-18 18:45:58 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-23 15:31:38 -07:00
										 |  |  | NetLog::~NetLog() = default; | 
					
						
							| 
									
										
										
										
											2018-06-18 18:45:58 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-25 16:06:39 -07:00
										 |  |  | v8::Local<v8::Promise> NetLog::StartLogging(base::FilePath log_path, | 
					
						
							| 
									
										
										
										
											2020-03-18 16:46:05 -07:00
										 |  |  |                                             gin::Arguments* args) { | 
					
						
							| 
									
										
										
										
											2019-07-25 16:06:39 -07:00
										 |  |  |   if (log_path.empty()) { | 
					
						
							| 
									
										
										
										
											2020-03-18 16:46:05 -07:00
										 |  |  |     args->ThrowTypeError("The first parameter must be a valid string"); | 
					
						
							| 
									
										
										
										
											2019-05-23 15:31:38 -07:00
										 |  |  |     return v8::Local<v8::Promise>(); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-25 16:06:39 -07:00
										 |  |  |   net::NetLogCaptureMode capture_mode = net::NetLogCaptureMode::kDefault; | 
					
						
							|  |  |  |   uint64_t max_file_size = network::mojom::NetLogExporter::kUnlimitedFileSize; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-25 22:03:28 +09:00
										 |  |  |   gin_helper::Dictionary dict; | 
					
						
							| 
									
										
										
										
											2019-07-25 16:06:39 -07:00
										 |  |  |   if (args->GetNext(&dict)) { | 
					
						
							|  |  |  |     v8::Local<v8::Value> capture_mode_v8; | 
					
						
							|  |  |  |     if (dict.Get("captureMode", &capture_mode_v8)) { | 
					
						
							| 
									
										
										
										
											2019-10-25 22:03:28 +09:00
										 |  |  |       if (!gin::ConvertFromV8(args->isolate(), capture_mode_v8, | 
					
						
							|  |  |  |                               &capture_mode)) { | 
					
						
							| 
									
										
										
										
											2020-03-18 16:46:05 -07:00
										 |  |  |         args->ThrowTypeError("Invalid value for captureMode"); | 
					
						
							| 
									
										
										
										
											2019-07-25 16:06:39 -07:00
										 |  |  |         return v8::Local<v8::Promise>(); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     v8::Local<v8::Value> max_file_size_v8; | 
					
						
							|  |  |  |     if (dict.Get("maxFileSize", &max_file_size_v8)) { | 
					
						
							| 
									
										
										
										
											2019-10-25 22:03:28 +09:00
										 |  |  |       if (!gin::ConvertFromV8(args->isolate(), max_file_size_v8, | 
					
						
							|  |  |  |                               &max_file_size)) { | 
					
						
							| 
									
										
										
										
											2020-03-18 16:46:05 -07:00
										 |  |  |         args->ThrowTypeError("Invalid value for maxFileSize"); | 
					
						
							| 
									
										
										
										
											2019-07-25 16:06:39 -07:00
										 |  |  |         return v8::Local<v8::Promise>(); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-23 15:31:38 -07:00
										 |  |  |   if (net_log_exporter_) { | 
					
						
							| 
									
										
										
										
											2020-03-18 16:46:05 -07:00
										 |  |  |     args->ThrowTypeError("There is already a net log running"); | 
					
						
							| 
									
										
										
										
											2019-05-23 15:31:38 -07:00
										 |  |  |     return v8::Local<v8::Promise>(); | 
					
						
							| 
									
										
										
										
											2018-06-18 18:45:58 -07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-01 15:10:32 +09:00
										 |  |  |   pending_start_promise_ = | 
					
						
							| 
									
										
										
										
											2024-01-10 23:23:35 +01:00
										 |  |  |       std::make_optional<gin_helper::Promise<void>>(args->isolate()); | 
					
						
							| 
									
										
										
										
											2019-05-23 15:31:38 -07:00
										 |  |  |   v8::Local<v8::Promise> handle = pending_start_promise_->GetHandle(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   auto command_line_string = | 
					
						
							|  |  |  |       base::CommandLine::ForCurrentProcess()->GetCommandLineString(); | 
					
						
							| 
									
										
										
										
											2019-06-17 13:37:55 -07:00
										 |  |  |   auto channel_string = std::string("Electron " ELECTRON_VERSION); | 
					
						
							| 
									
										
										
										
											2022-06-27 15:50:08 -05:00
										 |  |  |   base::Value::Dict custom_constants = net_log::GetPlatformConstantsForNetLog( | 
					
						
							|  |  |  |       command_line_string, channel_string); | 
					
						
							| 
									
										
										
										
											2019-05-23 15:31:38 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-04 23:38:56 +05:30
										 |  |  |   auto* network_context = | 
					
						
							| 
									
										
										
										
											2021-05-03 20:13:46 -07:00
										 |  |  |       browser_context_->GetDefaultStoragePartition()->GetNetworkContext(); | 
					
						
							| 
									
										
										
										
											2018-10-04 23:38:56 +05:30
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-20 18:52:28 -07:00
										 |  |  |   network_context->CreateNetLogExporter( | 
					
						
							|  |  |  |       net_log_exporter_.BindNewPipeAndPassReceiver()); | 
					
						
							|  |  |  |   net_log_exporter_.set_disconnect_handler( | 
					
						
							|  |  |  |       base::BindOnce(&NetLog::OnConnectionError, base::Unretained(this))); | 
					
						
							| 
									
										
										
										
											2019-05-23 15:31:38 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-05 21:35:34 -05:00
										 |  |  |   file_task_runner_->PostTaskAndReplyWithResult( | 
					
						
							|  |  |  |       FROM_HERE, base::BindOnce(OpenFileForWriting, log_path), | 
					
						
							| 
									
										
										
										
											2019-05-23 15:31:38 -07:00
										 |  |  |       base::BindOnce(&NetLog::StartNetLogAfterCreateFile, | 
					
						
							|  |  |  |                      weak_ptr_factory_.GetWeakPtr(), capture_mode, | 
					
						
							|  |  |  |                      max_file_size, std::move(custom_constants))); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return handle; | 
					
						
							| 
									
										
										
										
											2018-06-18 18:45:58 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-24 10:54:32 -07:00
										 |  |  | void NetLog::StartNetLogAfterCreateFile(net::NetLogCaptureMode capture_mode, | 
					
						
							|  |  |  |                                         uint64_t max_file_size, | 
					
						
							| 
									
										
										
										
											2022-06-27 15:50:08 -05:00
										 |  |  |                                         base::Value::Dict custom_constants, | 
					
						
							| 
									
										
										
										
											2019-05-24 10:54:32 -07:00
										 |  |  |                                         base::File output_file) { | 
					
						
							| 
									
										
										
										
											2019-05-23 15:31:38 -07:00
										 |  |  |   if (!net_log_exporter_) { | 
					
						
							|  |  |  |     // Theoretically the mojo pipe could have been closed by the time we get
 | 
					
						
							|  |  |  |     // here via the connection error handler. If so, the promise has already
 | 
					
						
							|  |  |  |     // been resolved.
 | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   DCHECK(pending_start_promise_); | 
					
						
							|  |  |  |   if (!output_file.IsValid()) { | 
					
						
							|  |  |  |     std::move(*pending_start_promise_) | 
					
						
							|  |  |  |         .RejectWithErrorMessage( | 
					
						
							|  |  |  |             base::File::ErrorToString(output_file.error_details())); | 
					
						
							|  |  |  |     net_log_exporter_.reset(); | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   net_log_exporter_->Start( | 
					
						
							|  |  |  |       std::move(output_file), std::move(custom_constants), capture_mode, | 
					
						
							|  |  |  |       max_file_size, | 
					
						
							|  |  |  |       base::BindOnce(&NetLog::NetLogStarted, base::Unretained(this))); | 
					
						
							| 
									
										
										
										
											2018-06-18 18:45:58 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-23 15:31:38 -07:00
										 |  |  | void NetLog::NetLogStarted(int32_t error) { | 
					
						
							|  |  |  |   DCHECK(pending_start_promise_); | 
					
						
							|  |  |  |   ResolvePromiseWithNetError(std::move(*pending_start_promise_), error); | 
					
						
							| 
									
										
										
										
											2018-10-04 23:38:56 +05:30
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-23 15:31:38 -07:00
										 |  |  | void NetLog::OnConnectionError() { | 
					
						
							|  |  |  |   net_log_exporter_.reset(); | 
					
						
							|  |  |  |   if (pending_start_promise_) { | 
					
						
							|  |  |  |     std::move(*pending_start_promise_) | 
					
						
							|  |  |  |         .RejectWithErrorMessage("Failed to start net log exporter"); | 
					
						
							| 
									
										
										
										
											2018-10-04 23:38:56 +05:30
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-05-23 15:31:38 -07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2018-10-04 23:38:56 +05:30
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-23 15:31:38 -07:00
										 |  |  | bool NetLog::IsCurrentlyLogging() const { | 
					
						
							|  |  |  |   return !!net_log_exporter_; | 
					
						
							| 
									
										
										
										
											2018-06-18 18:45:58 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-18 16:46:05 -07:00
										 |  |  | v8::Local<v8::Promise> NetLog::StopLogging(gin::Arguments* args) { | 
					
						
							|  |  |  |   gin_helper::Promise<void> promise(args->isolate()); | 
					
						
							| 
									
										
										
										
											2019-02-21 12:32:44 +00:00
										 |  |  |   v8::Local<v8::Promise> handle = promise.GetHandle(); | 
					
						
							| 
									
										
										
										
											2018-06-18 18:45:58 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-23 15:31:38 -07:00
										 |  |  |   if (net_log_exporter_) { | 
					
						
							|  |  |  |     // Move the net_log_exporter_ into the callback to ensure that the mojo
 | 
					
						
							|  |  |  |     // pointer lives long enough to resolve the promise. Moving it into the
 | 
					
						
							|  |  |  |     // callback will cause the instance variable to become empty.
 | 
					
						
							|  |  |  |     net_log_exporter_->Stop( | 
					
						
							| 
									
										
										
										
											2022-06-27 15:50:08 -05:00
										 |  |  |         base::Value::Dict(), | 
					
						
							| 
									
										
										
										
											2019-05-23 15:31:38 -07:00
										 |  |  |         base::BindOnce( | 
					
						
							| 
									
										
										
										
											2021-06-20 18:52:28 -07:00
										 |  |  |             [](mojo::Remote<network::mojom::NetLogExporter>, | 
					
						
							| 
									
										
										
										
											2019-11-01 15:10:32 +09:00
										 |  |  |                gin_helper::Promise<void> promise, int32_t error) { | 
					
						
							| 
									
										
										
										
											2019-05-23 15:31:38 -07:00
										 |  |  |               ResolvePromiseWithNetError(std::move(promise), error); | 
					
						
							|  |  |  |             }, | 
					
						
							|  |  |  |             std::move(net_log_exporter_), std::move(promise))); | 
					
						
							| 
									
										
										
										
											2018-10-04 23:38:56 +05:30
										 |  |  |   } else { | 
					
						
							| 
									
										
										
										
											2019-05-23 15:31:38 -07:00
										 |  |  |     promise.RejectWithErrorMessage("No net log in progress"); | 
					
						
							| 
									
										
										
										
											2018-10-04 23:38:56 +05:30
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-02-19 11:48:27 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-21 12:32:44 +00:00
										 |  |  |   return handle; | 
					
						
							| 
									
										
										
										
											2018-10-04 23:38:56 +05:30
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-18 16:46:05 -07:00
										 |  |  | gin::ObjectTemplateBuilder NetLog::GetObjectTemplateBuilder( | 
					
						
							|  |  |  |     v8::Isolate* isolate) { | 
					
						
							|  |  |  |   return gin::Wrappable<NetLog>::GetObjectTemplateBuilder(isolate) | 
					
						
							|  |  |  |       .SetProperty("currentlyLogging", &NetLog::IsCurrentlyLogging) | 
					
						
							|  |  |  |       .SetMethod("startLogging", &NetLog::StartLogging) | 
					
						
							|  |  |  |       .SetMethod("stopLogging", &NetLog::StopLogging); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const char* NetLog::GetTypeName() { | 
					
						
							|  |  |  |   return "NetLog"; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-04 23:38:56 +05:30
										 |  |  | // static
 | 
					
						
							| 
									
										
										
										
											2019-10-25 22:03:28 +09:00
										 |  |  | gin::Handle<NetLog> NetLog::Create(v8::Isolate* isolate, | 
					
						
							|  |  |  |                                    ElectronBrowserContext* browser_context) { | 
					
						
							|  |  |  |   return gin::CreateHandle(isolate, new NetLog(isolate, browser_context)); | 
					
						
							| 
									
										
										
										
											2018-06-18 18:45:58 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | }  // namespace api
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | }  // namespace electron
 |