diff --git a/atom/browser/net/url_request_fetch_job.cc b/atom/browser/net/url_request_fetch_job.cc index b6aadc89ec62..388b9f9550b3 100644 --- a/atom/browser/net/url_request_fetch_job.cc +++ b/atom/browser/net/url_request_fetch_job.cc @@ -59,7 +59,7 @@ class ResponsePiper : public net::URLFetcherResponseWriter { job_->HeadersCompleted(); first_write_ = false; } - return job_->DataAvailable(buffer, num_bytes); + return job_->DataAvailable(buffer, num_bytes, callback); } int Finish(const net::CompletionCallback& callback) override { return net::OK; @@ -166,21 +166,23 @@ void URLRequestFetchJob::HeadersCompleted() { NotifyHeadersComplete(); } -int URLRequestFetchJob::DataAvailable(net::IOBuffer* buffer, int num_bytes) { - // Do nothing if pending_buffer_ is empty, i.e. there's no ReadRawData() - // operation waiting for IO completion. - if (!pending_buffer_.get()) +int URLRequestFetchJob::DataAvailable(net::IOBuffer* buffer, + int num_bytes, + const net::CompletionCallback& callback) { + // Store the data in a drainable IO buffer if pending_buffer_ is empty, i.e. + // there's no ReadRawData() operation waiting for IO completion. + if (!pending_buffer_.get()) { + response_piper_callback_ = callback; + drainable_buffer_ = new net::DrainableIOBuffer(buffer, num_bytes); return net::ERR_IO_PENDING; + } // pending_buffer_ is set to the IOBuffer instance provided to ReadRawData() // by URLRequestJob. int bytes_read = std::min(num_bytes, pending_buffer_size_); memcpy(pending_buffer_->data(), buffer->data(), bytes_read); - // Clear the buffers before notifying the read is complete, so that it is - // safe for the observer to read. - pending_buffer_ = nullptr; - pending_buffer_size_ = 0; + ClearPendingBuffer(); ReadRawDataComplete(bytes_read); return bytes_read; @@ -191,6 +193,35 @@ void URLRequestFetchJob::Kill() { fetcher_.reset(); } +void URLRequestFetchJob::StartReading() { + if (drainable_buffer_.get()) { + auto num_bytes = drainable_buffer_->BytesRemaining(); + if (num_bytes <= 0 && !response_piper_callback_.is_null()) { + int size = drainable_buffer_->BytesConsumed(); + drainable_buffer_ = nullptr; + response_piper_callback_.Run(size); + response_piper_callback_.Reset(); + return; + } + + int bytes_read = std::min(num_bytes, pending_buffer_size_); + memcpy(pending_buffer_->data(), drainable_buffer_->data(), bytes_read); + + drainable_buffer_->DidConsume(bytes_read); + + ClearPendingBuffer(); + + ReadRawDataComplete(bytes_read); + } +} + +void URLRequestFetchJob::ClearPendingBuffer() { + // Clear the buffers before notifying the read is complete, so that it is + // safe for the observer to read. + pending_buffer_ = nullptr; + pending_buffer_size_ = 0; +} + int URLRequestFetchJob::ReadRawData(net::IOBuffer* dest, int dest_size) { if (GetResponseCode() == 204) { request()->set_received_response_content_length(prefilter_bytes_read()); @@ -198,6 +229,9 @@ int URLRequestFetchJob::ReadRawData(net::IOBuffer* dest, int dest_size) { } pending_buffer_ = dest; pending_buffer_size_ = dest_size; + // Read from drainable_buffer_ if available, i.e. response data became + // available before ReadRawData was called. + StartReading(); return net::ERR_IO_PENDING; } @@ -229,8 +263,8 @@ void URLRequestFetchJob::OnURLFetchComplete(const net::URLFetcher* source) { return; } - pending_buffer_ = nullptr; - pending_buffer_size_ = 0; + ClearPendingBuffer(); + if (fetcher_->GetStatus().is_success()) ReadRawDataComplete(0); else diff --git a/atom/browser/net/url_request_fetch_job.h b/atom/browser/net/url_request_fetch_job.h index 65d73f27ca94..da1c1678e58a 100644 --- a/atom/browser/net/url_request_fetch_job.h +++ b/atom/browser/net/url_request_fetch_job.h @@ -25,9 +25,14 @@ class URLRequestFetchJob : public JsAsker, // Called by response writer. void HeadersCompleted(); - int DataAvailable(net::IOBuffer* buffer, int num_bytes); + int DataAvailable(net::IOBuffer* buffer, + int num_bytes, + const net::CompletionCallback& callback); protected: + void StartReading(); + void ClearPendingBuffer(); + // JsAsker: void BeforeStartInUI(v8::Isolate*, v8::Local) override; void StartAsync(std::unique_ptr options) override; @@ -46,8 +51,10 @@ class URLRequestFetchJob : public JsAsker, scoped_refptr url_request_context_getter_; std::unique_ptr fetcher_; scoped_refptr pending_buffer_; + scoped_refptr drainable_buffer_; int pending_buffer_size_; std::unique_ptr response_info_; + net::CompletionCallback response_piper_callback_; DISALLOW_COPY_AND_ASSIGN(URLRequestFetchJob); }; diff --git a/spec/api-protocol-spec.js b/spec/api-protocol-spec.js index 2a1d158c3dfd..1b46d586a41a 100644 --- a/spec/api-protocol-spec.js +++ b/spec/api-protocol-spec.js @@ -3,8 +3,7 @@ const http = require('http') const path = require('path') const qs = require('querystring') const remote = require('electron').remote -const BrowserWindow = remote.require('electron').BrowserWindow -const protocol = remote.require('electron').protocol +const {BrowserWindow, protocol, webContents} = remote describe('protocol module', function () { var protocolName = 'sp' @@ -511,6 +510,42 @@ describe('protocol module', function () { }) }) }) + + it('loads resource for webContents', function (done) { + var contents = null + var server = http.createServer(function (req, res) { + if (req.url == '/serverRedirect') { + res.statusCode = 301 + res.setHeader('Location', 'http://' + req.rawHeaders[1]) + res.end() + } else { + res.end(text) + } + }) + server.listen(0, '127.0.0.1', function () { + var port = server.address().port + var url = `${protocolName}://fake-host` + var redirectURL = `http://127.0.0.1:${port}/serverRedirect` + var handler = function (request, callback) { + callback({ + url: redirectURL + }) + } + protocol.registerHttpProtocol(protocolName, handler, function (error) { + if (error) { + return done(error) + } + contents = webContents.create({}) + contents.on('did-finish-load', function () { + assert.equal(contents.getURL(), url) + server.close() + contents.destroy() + done() + }) + contents.loadURL(url) + }) + }) + }) }) describe('protocol.isProtocolHandled', function () {