| 
									
										
										
										
											2019-05-03 09:48:51 +09:00
										 |  |  | // Copyright (c) 2019 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/net/node_stream_loader.h"
 | 
					
						
							| 
									
										
										
										
											2019-05-03 09:48:51 +09:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include <utility>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-24 15:58:51 -07:00
										 |  |  | #include "mojo/public/cpp/system/string_data_source.h"
 | 
					
						
							| 
									
										
										
										
											2019-09-06 14:52:54 +09:00
										 |  |  | #include "shell/common/gin_converters/callback_converter.h"
 | 
					
						
							| 
									
										
										
										
											2019-06-19 13:46:59 -07:00
										 |  |  | #include "shell/common/node_includes.h"
 | 
					
						
							| 
									
										
										
										
											2019-05-03 09:48:51 +09:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace electron { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-10 16:22:35 -08:00
										 |  |  | NodeStreamLoader::NodeStreamLoader( | 
					
						
							| 
									
										
										
										
											2019-12-13 15:13:12 -05:00
										 |  |  |     network::mojom::URLResponseHeadPtr head, | 
					
						
							| 
									
										
										
										
											2021-03-04 09:27:05 -08:00
										 |  |  |     mojo::PendingReceiver<network::mojom::URLLoader> loader, | 
					
						
							| 
									
										
										
										
											2019-12-10 16:22:35 -08:00
										 |  |  |     mojo::PendingRemote<network::mojom::URLLoaderClient> client, | 
					
						
							|  |  |  |     v8::Isolate* isolate, | 
					
						
							|  |  |  |     v8::Local<v8::Object> emitter) | 
					
						
							| 
									
										
										
										
											2021-03-04 09:27:05 -08:00
										 |  |  |     : url_loader_(this, std::move(loader)), | 
					
						
							| 
									
										
										
										
											2019-05-03 09:48:51 +09:00
										 |  |  |       client_(std::move(client)), | 
					
						
							|  |  |  |       isolate_(isolate), | 
					
						
							| 
									
										
										
										
											2021-01-26 19:16:21 +01:00
										 |  |  |       emitter_(isolate, emitter) { | 
					
						
							| 
									
										
										
										
											2021-03-04 09:27:05 -08:00
										 |  |  |   url_loader_.set_disconnect_handler( | 
					
						
							| 
									
										
										
										
											2019-05-11 15:15:01 +09:00
										 |  |  |       base::BindOnce(&NodeStreamLoader::NotifyComplete, | 
					
						
							|  |  |  |                      weak_factory_.GetWeakPtr(), net::ERR_FAILED)); | 
					
						
							| 
									
										
										
										
											2019-05-03 09:48:51 +09:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-11-18 10:47:29 -08:00
										 |  |  |   Start(std::move(head)); | 
					
						
							| 
									
										
										
										
											2019-05-03 09:48:51 +09:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | NodeStreamLoader::~NodeStreamLoader() { | 
					
						
							|  |  |  |   v8::Isolate::Scope isolate_scope(isolate_); | 
					
						
							|  |  |  |   v8::HandleScope handle_scope(isolate_); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Unsubscribe all handlers.
 | 
					
						
							|  |  |  |   for (const auto& it : handlers_) { | 
					
						
							| 
									
										
										
										
											2019-09-06 14:52:54 +09:00
										 |  |  |     v8::Local<v8::Value> args[] = {gin::StringToV8(isolate_, it.first), | 
					
						
							| 
									
										
										
										
											2019-05-03 09:48:51 +09:00
										 |  |  |                                    it.second.Get(isolate_)}; | 
					
						
							|  |  |  |     node::MakeCallback(isolate_, emitter_.Get(isolate_), "removeListener", | 
					
						
							|  |  |  |                        node::arraysize(args), args, {0, 0}); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2020-07-20 19:51:38 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |   // Destroy the stream if not already ended
 | 
					
						
							|  |  |  |   if (!ended_) { | 
					
						
							|  |  |  |     node::MakeCallback(isolate_, emitter_.Get(isolate_), "destroy", 0, nullptr, | 
					
						
							|  |  |  |                        {0, 0}); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-05-03 09:48:51 +09:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-12-13 15:13:12 -05:00
										 |  |  | void NodeStreamLoader::Start(network::mojom::URLResponseHeadPtr head) { | 
					
						
							| 
									
										
										
										
											2019-05-11 15:15:01 +09:00
										 |  |  |   mojo::ScopedDataPipeProducerHandle producer; | 
					
						
							|  |  |  |   mojo::ScopedDataPipeConsumerHandle consumer; | 
					
						
							| 
									
										
										
										
											2021-03-05 16:42:15 -08:00
										 |  |  |   MojoResult rv = mojo::CreateDataPipe(nullptr, producer, consumer); | 
					
						
							| 
									
										
										
										
											2019-05-11 15:15:01 +09:00
										 |  |  |   if (rv != MOJO_RESULT_OK) { | 
					
						
							|  |  |  |     NotifyComplete(net::ERR_INSUFFICIENT_RESOURCES); | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-05-03 09:48:51 +09:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-24 15:58:51 -07:00
										 |  |  |   producer_ = std::make_unique<mojo::DataPipeProducer>(std::move(producer)); | 
					
						
							| 
									
										
										
										
											2022-09-07 09:46:37 +02:00
										 |  |  |   client_->OnReceiveResponse(std::move(head), std::move(consumer), | 
					
						
							|  |  |  |                              absl::nullopt); | 
					
						
							| 
									
										
										
										
											2019-05-11 15:15:01 +09:00
										 |  |  | 
 | 
					
						
							|  |  |  |   auto weak = weak_factory_.GetWeakPtr(); | 
					
						
							|  |  |  |   On("end", | 
					
						
							|  |  |  |      base::BindRepeating(&NodeStreamLoader::NotifyComplete, weak, net::OK)); | 
					
						
							|  |  |  |   On("error", base::BindRepeating(&NodeStreamLoader::NotifyComplete, weak, | 
					
						
							|  |  |  |                                   net::ERR_FAILED)); | 
					
						
							| 
									
										
										
										
											2019-08-21 05:23:46 +02:00
										 |  |  |   On("readable", base::BindRepeating(&NodeStreamLoader::NotifyReadable, weak)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void NodeStreamLoader::NotifyReadable() { | 
					
						
							|  |  |  |   if (!readable_) | 
					
						
							|  |  |  |     ReadMore(); | 
					
						
							| 
									
										
										
										
											2020-06-11 11:55:59 -05:00
										 |  |  |   else if (is_reading_) | 
					
						
							|  |  |  |     has_read_waiting_ = true; | 
					
						
							| 
									
										
										
										
											2019-08-21 05:23:46 +02:00
										 |  |  |   readable_ = true; | 
					
						
							| 
									
										
										
										
											2019-05-03 09:48:51 +09:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-11 15:15:01 +09:00
										 |  |  | void NodeStreamLoader::NotifyComplete(int result) { | 
					
						
							|  |  |  |   // Wait until write finishes or fails.
 | 
					
						
							| 
									
										
										
										
											2019-08-21 05:23:46 +02:00
										 |  |  |   if (is_reading_ || is_writing_) { | 
					
						
							| 
									
										
										
										
											2019-05-11 15:15:01 +09:00
										 |  |  |     ended_ = true; | 
					
						
							|  |  |  |     result_ = result; | 
					
						
							| 
									
										
										
										
											2019-05-03 09:48:51 +09:00
										 |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-27 10:00:55 -07:00
										 |  |  |   network::URLLoaderCompletionStatus status(result); | 
					
						
							|  |  |  |   status.completion_time = base::TimeTicks::Now(); | 
					
						
							|  |  |  |   status.decoded_body_length = bytes_written_; | 
					
						
							|  |  |  |   client_->OnComplete(status); | 
					
						
							| 
									
										
										
										
											2019-05-11 15:15:01 +09:00
										 |  |  |   delete this; | 
					
						
							| 
									
										
										
										
											2019-05-03 09:48:51 +09:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-11 15:15:01 +09:00
										 |  |  | void NodeStreamLoader::ReadMore() { | 
					
						
							| 
									
										
										
										
											2019-09-25 17:38:50 -04:00
										 |  |  |   if (is_reading_) { | 
					
						
							|  |  |  |     // Calling read() can trigger the "readable" event again, making this
 | 
					
						
							|  |  |  |     // function re-entrant. If we're already reading, we don't want to start
 | 
					
						
							|  |  |  |     // a nested read, so short-circuit.
 | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-08-21 05:23:46 +02:00
										 |  |  |   is_reading_ = true; | 
					
						
							| 
									
										
										
										
											2019-09-25 17:38:50 -04:00
										 |  |  |   auto weak = weak_factory_.GetWeakPtr(); | 
					
						
							| 
									
										
										
										
											2020-03-10 18:16:58 -07:00
										 |  |  |   v8::HandleScope scope(isolate_); | 
					
						
							| 
									
										
										
										
											2019-05-11 15:15:01 +09:00
										 |  |  |   // buffer = emitter.read()
 | 
					
						
							|  |  |  |   v8::MaybeLocal<v8::Value> ret = node::MakeCallback( | 
					
						
							|  |  |  |       isolate_, emitter_.Get(isolate_), "read", 0, nullptr, {0, 0}); | 
					
						
							| 
									
										
										
										
											2019-09-25 17:38:50 -04:00
										 |  |  |   DCHECK(weak) << "We shouldn't have been destroyed when calling read()"; | 
					
						
							| 
									
										
										
										
											2019-05-03 09:48:51 +09:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-11 15:15:01 +09:00
										 |  |  |   // If there is no buffer read, wait until |readable| is emitted again.
 | 
					
						
							|  |  |  |   v8::Local<v8::Value> buffer; | 
					
						
							| 
									
										
										
										
											2019-08-21 05:23:46 +02:00
										 |  |  |   if (!ret.ToLocal(&buffer) || !node::Buffer::HasInstance(buffer)) { | 
					
						
							|  |  |  |     is_reading_ = false; | 
					
						
							| 
									
										
										
										
											2020-06-11 11:55:59 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // If 'readable' was called after 'read()', try again
 | 
					
						
							|  |  |  |     if (has_read_waiting_) { | 
					
						
							|  |  |  |       has_read_waiting_ = false; | 
					
						
							|  |  |  |       ReadMore(); | 
					
						
							|  |  |  |       return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     readable_ = false; | 
					
						
							| 
									
										
										
										
											2020-01-13 10:28:21 -08:00
										 |  |  |     if (ended_) { | 
					
						
							|  |  |  |       NotifyComplete(result_); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-05-11 15:15:01 +09:00
										 |  |  |     return; | 
					
						
							| 
									
										
										
										
											2019-08-21 05:23:46 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2019-05-11 15:15:01 +09:00
										 |  |  | 
 | 
					
						
							|  |  |  |   // Hold the buffer until the write is done.
 | 
					
						
							|  |  |  |   buffer_.Reset(isolate_, buffer); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-27 10:00:55 -07:00
										 |  |  |   bytes_written_ += node::Buffer::Length(buffer); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-13 10:25:21 -07:00
										 |  |  |   // Write buffer to mojo pipe asynchronously.
 | 
					
						
							| 
									
										
										
										
											2019-08-21 05:23:46 +02:00
										 |  |  |   is_reading_ = false; | 
					
						
							| 
									
										
										
										
											2019-05-11 15:15:01 +09:00
										 |  |  |   is_writing_ = true; | 
					
						
							| 
									
										
										
										
											2019-09-25 17:38:50 -04:00
										 |  |  |   producer_->Write(std::make_unique<mojo::StringDataSource>( | 
					
						
							|  |  |  |                        base::StringPiece(node::Buffer::Data(buffer), | 
					
						
							|  |  |  |                                          node::Buffer::Length(buffer)), | 
					
						
							|  |  |  |                        mojo::StringDataSource::AsyncWritingMode:: | 
					
						
							|  |  |  |                            STRING_STAYS_VALID_UNTIL_COMPLETION), | 
					
						
							|  |  |  |                    base::BindOnce(&NodeStreamLoader::DidWrite, weak)); | 
					
						
							| 
									
										
										
										
											2019-05-03 09:48:51 +09:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-11 15:15:01 +09:00
										 |  |  | void NodeStreamLoader::DidWrite(MojoResult result) { | 
					
						
							|  |  |  |   is_writing_ = false; | 
					
						
							|  |  |  |   // We were told to end streaming.
 | 
					
						
							|  |  |  |   if (ended_) { | 
					
						
							|  |  |  |     NotifyComplete(result_); | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-21 05:23:46 +02:00
										 |  |  |   if (result == MOJO_RESULT_OK && readable_) | 
					
						
							| 
									
										
										
										
											2019-05-11 15:15:01 +09:00
										 |  |  |     ReadMore(); | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |     NotifyComplete(net::ERR_FAILED); | 
					
						
							| 
									
										
										
										
											2019-05-03 09:48:51 +09:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-05-11 15:15:01 +09:00
										 |  |  | void NodeStreamLoader::On(const char* event, EventCallback callback) { | 
					
						
							|  |  |  |   v8::Isolate::Scope isolate_scope(isolate_); | 
					
						
							|  |  |  |   v8::HandleScope handle_scope(isolate_); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // emitter.on(event, callback)
 | 
					
						
							|  |  |  |   v8::Local<v8::Value> args[] = { | 
					
						
							| 
									
										
										
										
											2019-09-06 14:52:54 +09:00
										 |  |  |       gin::StringToV8(isolate_, event), | 
					
						
							|  |  |  |       gin_helper::CallbackToV8Leaked(isolate_, std::move(callback)), | 
					
						
							| 
									
										
										
										
											2019-05-11 15:15:01 +09:00
										 |  |  |   }; | 
					
						
							|  |  |  |   handlers_[event].Reset(isolate_, args[1]); | 
					
						
							|  |  |  |   node::MakeCallback(isolate_, emitter_.Get(isolate_), "on", | 
					
						
							|  |  |  |                      node::arraysize(args), args, {0, 0}); | 
					
						
							| 
									
										
										
										
											2021-05-27 11:48:03 -07:00
										 |  |  |   // No more code below, as this class may destruct when subscribing.
 | 
					
						
							| 
									
										
										
										
											2019-05-03 09:48:51 +09:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | }  // namespace electron
 |