* chore: bump chromium in DEPS to 1e9f9a24aa12bea9cf194a82a7e249bd1242ec4f * chore: update patches * Make WebContents' theme color a base::Optional<SkColor> https://chromium-review.googlesource.com/c/chromium/src/+/1540022 * update autofill patch for incorrect header includes * Move Shell messages to web_test and rename to BlinkTest. https://chromium-review.googlesource.com/c/chromium/src/+/1525181 * Make PlatformNotificationServiceImpl a KeyedService. https://chromium-review.googlesource.com/c/chromium/src/+/1336150 * Move MediaPlayerId to its own file. https://chromium-review.googlesource.com/c/chromium/src/+/1547057 * Remove net/base/completion_callback.h, which is no longer used https://chromium-review.googlesource.com/c/chromium/src/+/1552821 * AW NS: support file scheme cookies https://chromium-review.googlesource.com/c/chromium/src/+/1533486 * Remove SecurityInfo and adapt remaining consumers https://chromium-review.googlesource.com/c/chromium/src/+/1509455 * Remove deprecated type-specific number to string conversion functions https://chromium-review.googlesource.com/c/chromium/src/+/1545881 * DevTools: Adding new performance histograms for launch of top 4 tools https://chromium-review.googlesource.com/c/chromium/src/+/1506388 * Update include paths for //base/hash/hash.h https://chromium-review.googlesource.com/c/chromium/src/+/1544630 * build: Disable ensure_gn_version gclient hook for mac CI checkout * update patches * use maybe version of v8::String::NewFromTwoByte * bump appveyor image version * fix mac ci hopefully * Convert enum to enum class for MenuAnchorPosition https://chromium-review.googlesource.com/c/chromium/src/+/1530508 * use maybe version of ToObject * RenderViewHost::GetProcess is no longer const * Unrefcount AuthChallengeInfo https://chromium-review.googlesource.com/c/chromium/src/+/1550631 * MenuButtonController takes Button rather than MenuButton https://chromium-review.googlesource.com/c/chromium/src/+/1500935 * add //ui/views_bridge_mac to deps to fix link error * forward declare views::Button in atom::MenuDelegate * more v8 patches * base/{=> hash}/md5.h https://chromium-review.googlesource.com/c/chromium/src/+/1535124 * gfx::{PlatformFontWin => win}::* https://chromium-review.googlesource.com/c/chromium/src/+/1534178 * fix v8 patches * [base] Rename TaskScheduler to ThreadPool https://chromium-review.googlesource.com/c/chromium/src/+/1561552 * use internal_config_base for bytecode_builtins_list_generator avoids windows link errors * FIXME: temporarily disable v8/breakpad integration * FIXME: temporarily disable prevent-will-redirect test * FIXME: disable neon on aarch64 pending crbug.com/953815 * update to account for WebCursor refactor https://chromium-review.googlesource.com/c/chromium/src/+/1562755 * enable stack dumping on appveyor * Revert "FIXME: disable neon on aarch64 pending crbug.com/953815" This reverts commit 57f082026be3d83069f2a2814684abf4dc9e7b53. * fix: remove const qualifiers to match upstream * fix: remove const qualifiers to match upstream in cc files as well * don't throw an error when testing if an object is an object * use non-deprecated Buffer constructor * Remove net::CookieSameSite::DEFAULT_MODE enum value https://chromium-review.googlesource.com/c/chromium/src/+/1567955 * depend on modded dbus-native to work around buffer deprecation https://github.com/sidorares/dbus-native/pull/262 * revert clang roll to fix arm build on linux * fixup! depend on modded dbus-native to work around buffer deprecation need more coffee * update coffee-script * robustify verify-mksnapshot w.r.t. command-line parameters * Revert "robustify verify-mksnapshot w.r.t. command-line parameters" This reverts commit a49af01411f684f6025528d604895c3696e0bc57. * fix mksnapshot by matching args * update patches * TMP: enable rdp on appveyor * Changed ContentBrowserClient::CreateQuotaPermissionContext() to return scoped_refptr. https://chromium-review.googlesource.com/c/chromium/src/+/1569376 * Make content::ResourceType an enum class. https://chromium-review.googlesource.com/c/chromium/src/+/1569345 * fixup! Make content::ResourceType an enum class. * turn off rdp * use net::CompletionRepeatingCallback instead of base::Callback<void(int)> * remove disable_ensure_gn_version_gclient_hook.patch * copy repeating callback instead of std::move * fix lint * add completion_repeating_callback.h include
		
			
				
	
	
		
			257 lines
		
	
	
	
		
			8.4 KiB
			
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			257 lines
		
	
	
	
		
			8.4 KiB
			
		
	
	
	
		
			C++
		
	
	
	
	
	
// Copyright (c) 2017 GitHub, Inc.
 | 
						|
// Use of this source code is governed by the MIT license that can be
 | 
						|
// found in the LICENSE file.
 | 
						|
 | 
						|
#include "atom/browser/net/url_request_stream_job.h"
 | 
						|
 | 
						|
#include <algorithm>
 | 
						|
#include <memory>
 | 
						|
#include <ostream>
 | 
						|
#include <string>
 | 
						|
#include <utility>
 | 
						|
 | 
						|
#include "atom/common/api/event_emitter_caller.h"
 | 
						|
#include "atom/common/atom_constants.h"
 | 
						|
#include "atom/common/native_mate_converters/net_converter.h"
 | 
						|
#include "atom/common/node_includes.h"
 | 
						|
#include "base/strings/string_number_conversions.h"
 | 
						|
#include "base/strings/string_util.h"
 | 
						|
#include "base/task/post_task.h"
 | 
						|
#include "base/threading/thread_task_runner_handle.h"
 | 
						|
#include "base/time/time.h"
 | 
						|
#include "content/public/browser/browser_task_traits.h"
 | 
						|
#include "native_mate/dictionary.h"
 | 
						|
#include "net/base/net_errors.h"
 | 
						|
#include "net/filter/gzip_source_stream.h"
 | 
						|
 | 
						|
namespace atom {
 | 
						|
 | 
						|
namespace {
 | 
						|
 | 
						|
void BeforeStartInUI(base::WeakPtr<URLRequestStreamJob> job,
 | 
						|
                     mate::Arguments* args) {
 | 
						|
  v8::Local<v8::Value> value;
 | 
						|
  int error = net::OK;
 | 
						|
  bool ended = false;
 | 
						|
  if (!args->GetNext(&value) || !value->IsObject()) {
 | 
						|
    // Invalid opts.
 | 
						|
    base::PostTaskWithTraits(
 | 
						|
        FROM_HERE, {content::BrowserThread::IO},
 | 
						|
        base::BindOnce(&URLRequestStreamJob::OnError, job, net::ERR_FAILED));
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  mate::Dictionary opts(args->isolate(), v8::Local<v8::Object>::Cast(value));
 | 
						|
  int status_code;
 | 
						|
  if (!opts.Get("statusCode", &status_code)) {
 | 
						|
    // assume HTTP OK if statusCode is not passed.
 | 
						|
    status_code = 200;
 | 
						|
  }
 | 
						|
  std::string status("HTTP/1.1 ");
 | 
						|
  status.append(base::NumberToString(status_code));
 | 
						|
  status.append(" ");
 | 
						|
  status.append(
 | 
						|
      net::GetHttpReasonPhrase(static_cast<net::HttpStatusCode>(status_code)));
 | 
						|
  status.append("\0\0", 2);
 | 
						|
  scoped_refptr<net::HttpResponseHeaders> response_headers(
 | 
						|
      new net::HttpResponseHeaders(status));
 | 
						|
 | 
						|
  if (opts.Get("headers", &value)) {
 | 
						|
    mate::Converter<net::HttpResponseHeaders*>::FromV8(args->isolate(), value,
 | 
						|
                                                       response_headers.get());
 | 
						|
  }
 | 
						|
 | 
						|
  if (!opts.Get("data", &value)) {
 | 
						|
    // Assume the opts is already a stream
 | 
						|
    value = opts.GetHandle();
 | 
						|
  } else if (value->IsNullOrUndefined()) {
 | 
						|
    // "data" was explicitly passed as null or undefined, assume the user wants
 | 
						|
    // to send an empty body.
 | 
						|
    ended = true;
 | 
						|
    base::PostTaskWithTraits(
 | 
						|
        FROM_HERE, {content::BrowserThread::IO},
 | 
						|
        base::BindOnce(&URLRequestStreamJob::StartAsync, job, nullptr,
 | 
						|
                       base::RetainedRef(response_headers), ended, error));
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  mate::Dictionary data(args->isolate(), v8::Local<v8::Object>::Cast(value));
 | 
						|
  if (!data.Get("on", &value) || !value->IsFunction() ||
 | 
						|
      !data.Get("removeListener", &value) || !value->IsFunction()) {
 | 
						|
    // If data is passed but it is not a stream, signal an error.
 | 
						|
    base::PostTaskWithTraits(
 | 
						|
        FROM_HERE, {content::BrowserThread::IO},
 | 
						|
        base::BindOnce(&URLRequestStreamJob::OnError, job, net::ERR_FAILED));
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  auto subscriber = base::MakeRefCounted<mate::StreamSubscriber>(
 | 
						|
      args->isolate(), data.GetHandle(), job,
 | 
						|
      base::ThreadTaskRunnerHandle::Get());
 | 
						|
 | 
						|
  base::PostTaskWithTraits(
 | 
						|
      FROM_HERE, {content::BrowserThread::IO},
 | 
						|
      base::BindOnce(&URLRequestStreamJob::StartAsync, job, subscriber,
 | 
						|
                     base::RetainedRef(response_headers), ended, error));
 | 
						|
}
 | 
						|
 | 
						|
}  // namespace
 | 
						|
 | 
						|
URLRequestStreamJob::URLRequestStreamJob(net::URLRequest* request,
 | 
						|
                                         net::NetworkDelegate* network_delegate)
 | 
						|
    : net::URLRequestJob(request, network_delegate),
 | 
						|
      pending_buf_(nullptr),
 | 
						|
      pending_buf_size_(0),
 | 
						|
      ended_(false),
 | 
						|
      response_headers_(nullptr),
 | 
						|
      weak_factory_(this) {}
 | 
						|
 | 
						|
URLRequestStreamJob::~URLRequestStreamJob() {
 | 
						|
  DCHECK(!subscriber_ || subscriber_->HasOneRef());
 | 
						|
}
 | 
						|
 | 
						|
void URLRequestStreamJob::Start() {
 | 
						|
  auto request_details = std::make_unique<base::DictionaryValue>();
 | 
						|
  FillRequestDetails(request_details.get(), request());
 | 
						|
  base::PostTaskWithTraits(
 | 
						|
      FROM_HERE, {content::BrowserThread::UI},
 | 
						|
      base::BindOnce(&JsAsker::AskForOptions, base::Unretained(isolate()),
 | 
						|
                     handler(), std::move(request_details),
 | 
						|
                     base::Bind(&BeforeStartInUI, weak_factory_.GetWeakPtr())));
 | 
						|
}
 | 
						|
 | 
						|
void URLRequestStreamJob::StartAsync(
 | 
						|
    scoped_refptr<mate::StreamSubscriber> subscriber,
 | 
						|
    scoped_refptr<net::HttpResponseHeaders> response_headers,
 | 
						|
    bool ended,
 | 
						|
    int error) {
 | 
						|
  if (error != net::OK) {
 | 
						|
    NotifyStartError(
 | 
						|
        net::URLRequestStatus(net::URLRequestStatus::FAILED, error));
 | 
						|
    return;
 | 
						|
  }
 | 
						|
 | 
						|
  ended_ = ended;
 | 
						|
  response_headers_ = response_headers;
 | 
						|
  subscriber_ = subscriber;
 | 
						|
  request_start_time_ = base::TimeTicks::Now();
 | 
						|
  NotifyHeadersComplete();
 | 
						|
}
 | 
						|
 | 
						|
void URLRequestStreamJob::OnData(std::vector<char>&& buffer) {  // NOLINT
 | 
						|
  DCHECK_CURRENTLY_ON(content::BrowserThread::IO);
 | 
						|
  if (write_buffer_.empty()) {
 | 
						|
    // Quick branch without copying.
 | 
						|
    write_buffer_ = std::move(buffer);
 | 
						|
  } else {
 | 
						|
    // write_buffer_ += buffer
 | 
						|
    size_t len = write_buffer_.size();
 | 
						|
    write_buffer_.resize(len + buffer.size());
 | 
						|
    std::copy(buffer.begin(), buffer.end(), write_buffer_.begin() + len);
 | 
						|
  }
 | 
						|
 | 
						|
  // Copy to output.
 | 
						|
  if (pending_buf_) {
 | 
						|
    int len = BufferCopy(&write_buffer_, pending_buf_.get(), pending_buf_size_);
 | 
						|
    write_buffer_.erase(write_buffer_.begin(), write_buffer_.begin() + len);
 | 
						|
    pending_buf_ = nullptr;
 | 
						|
    pending_buf_size_ = 0;
 | 
						|
    ReadRawDataComplete(len);
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
void URLRequestStreamJob::OnEnd() {
 | 
						|
  ended_ = true;
 | 
						|
  if (pending_buf_) {
 | 
						|
    ReadRawDataComplete(0);
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
void URLRequestStreamJob::OnError(int error) {
 | 
						|
  NotifyStartError(net::URLRequestStatus(net::URLRequestStatus::FAILED, error));
 | 
						|
}
 | 
						|
 | 
						|
int URLRequestStreamJob::ReadRawData(net::IOBuffer* dest, int dest_size) {
 | 
						|
  response_start_time_ = base::TimeTicks::Now();
 | 
						|
 | 
						|
  if (ended_ && write_buffer_.empty())
 | 
						|
    return 0;
 | 
						|
 | 
						|
  // When write_buffer_ is empty, there is no data valable yet, we have to save
 | 
						|
  // the dest buffer util DataAvailable.
 | 
						|
  if (write_buffer_.empty()) {
 | 
						|
    pending_buf_ = dest;
 | 
						|
    pending_buf_size_ = dest_size;
 | 
						|
    return net::ERR_IO_PENDING;
 | 
						|
  }
 | 
						|
 | 
						|
  // Read from the write buffer and clear them after reading.
 | 
						|
  int len = BufferCopy(&write_buffer_, dest, dest_size);
 | 
						|
  write_buffer_.erase(write_buffer_.begin(), write_buffer_.begin() + len);
 | 
						|
  return len;
 | 
						|
}
 | 
						|
 | 
						|
void URLRequestStreamJob::DoneReading() {
 | 
						|
  write_buffer_.clear();
 | 
						|
}
 | 
						|
 | 
						|
void URLRequestStreamJob::DoneReadingRedirectResponse() {
 | 
						|
  if (subscriber_) {
 | 
						|
    DCHECK(subscriber_->HasAtLeastOneRef());
 | 
						|
    subscriber_ = nullptr;
 | 
						|
  }
 | 
						|
  DoneReading();
 | 
						|
}
 | 
						|
 | 
						|
std::unique_ptr<net::SourceStream> URLRequestStreamJob::SetUpSourceStream() {
 | 
						|
  std::unique_ptr<net::SourceStream> source =
 | 
						|
      net::URLRequestJob::SetUpSourceStream();
 | 
						|
  size_t i = 0;
 | 
						|
  std::string type;
 | 
						|
  while (response_headers_->EnumerateHeader(&i, "Content-Encoding", &type)) {
 | 
						|
    if (base::LowerCaseEqualsASCII(type, "gzip") ||
 | 
						|
        base::LowerCaseEqualsASCII(type, "x-gzip")) {
 | 
						|
      return net::GzipSourceStream::Create(std::move(source),
 | 
						|
                                           net::SourceStream::TYPE_GZIP);
 | 
						|
    } else if (base::LowerCaseEqualsASCII(type, "deflate")) {
 | 
						|
      return net::GzipSourceStream::Create(std::move(source),
 | 
						|
                                           net::SourceStream::TYPE_DEFLATE);
 | 
						|
    }
 | 
						|
  }
 | 
						|
  return source;
 | 
						|
}
 | 
						|
 | 
						|
bool URLRequestStreamJob::GetMimeType(std::string* mime_type) const {
 | 
						|
  return response_headers_->GetMimeType(mime_type);
 | 
						|
}
 | 
						|
 | 
						|
int URLRequestStreamJob::GetResponseCode() const {
 | 
						|
  return response_headers_->response_code();
 | 
						|
}
 | 
						|
 | 
						|
void URLRequestStreamJob::GetResponseInfo(net::HttpResponseInfo* info) {
 | 
						|
  info->headers = response_headers_;
 | 
						|
}
 | 
						|
 | 
						|
void URLRequestStreamJob::GetLoadTimingInfo(
 | 
						|
    net::LoadTimingInfo* load_timing_info) const {
 | 
						|
  load_timing_info->send_start = request_start_time_;
 | 
						|
  load_timing_info->send_end = request_start_time_;
 | 
						|
  load_timing_info->request_start = request_start_time_;
 | 
						|
  load_timing_info->receive_headers_end = response_start_time_;
 | 
						|
}
 | 
						|
 | 
						|
void URLRequestStreamJob::Kill() {
 | 
						|
  weak_factory_.InvalidateWeakPtrs();
 | 
						|
  net::URLRequestJob::Kill();
 | 
						|
}
 | 
						|
 | 
						|
int URLRequestStreamJob::BufferCopy(std::vector<char>* source,
 | 
						|
                                    net::IOBuffer* target,
 | 
						|
                                    int target_size) {
 | 
						|
  int bytes_written = std::min(static_cast<int>(source->size()), target_size);
 | 
						|
  memcpy(target->data(), source->data(), bytes_written);
 | 
						|
  return bytes_written;
 | 
						|
}
 | 
						|
 | 
						|
}  // namespace atom
 |