// Copyright 2015 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "net/url_request/report_sender.h" #include #include "net/base/elements_upload_data_stream.h" #include "net/base/load_flags.h" #include "net/base/request_priority.h" #include "net/base/upload_bytes_element_reader.h" #include "net/http/http_status_code.h" #include "net/url_request/url_request_context.h" #include "net/url_request/url_request_status.h" namespace { const void* const kUserDataKey = &kUserDataKey; class CallbackInfo : public base::SupportsUserData::Data { public: CallbackInfo(const net::ReportSender::SuccessCallback& success_callback, const net::ReportSender::ErrorCallback& error_callback) : success_callback_(success_callback), error_callback_(error_callback) {} ~CallbackInfo() override = default; const net::ReportSender::SuccessCallback& success_callback() const { return success_callback_; } const net::ReportSender::ErrorCallback& error_callback() const { return error_callback_; } private: net::ReportSender::SuccessCallback success_callback_; net::ReportSender::ErrorCallback error_callback_; }; } // namespace namespace net { const int ReportSender::kLoadFlags = LOAD_BYPASS_CACHE | LOAD_DISABLE_CACHE; ReportSender::ReportSender(URLRequestContext* request_context, net::NetworkTrafficAnnotationTag traffic_annotation) : request_context_(request_context), traffic_annotation_(traffic_annotation) {} ReportSender::~ReportSender() = default; void ReportSender::Send(const GURL& report_uri, base::StringPiece content_type, base::StringPiece report, const SuccessCallback& success_callback, const ErrorCallback& error_callback) { DCHECK(!content_type.empty()); std::unique_ptr url_request = request_context_->CreateRequest( report_uri, DEFAULT_PRIORITY, this, traffic_annotation_); url_request->SetUserData( &kUserDataKey, std::make_unique(success_callback, error_callback)); url_request->SetLoadFlags(kLoadFlags); url_request->set_allow_credentials(false); HttpRequestHeaders extra_headers; extra_headers.SetHeader(HttpRequestHeaders::kContentType, content_type); url_request->SetExtraRequestHeaders(extra_headers); url_request->set_method("POST"); std::vector report_data(report.begin(), report.end()); std::unique_ptr reader( new UploadOwnedBytesElementReader(&report_data)); url_request->set_upload( ElementsUploadDataStream::CreateWithReader(std::move(reader), 0)); URLRequest* raw_url_request = url_request.get(); inflight_requests_[raw_url_request] = std::move(url_request); raw_url_request->Start(); } void ReportSender::OnResponseStarted(URLRequest* request, int net_error) { DCHECK_NE(ERR_IO_PENDING, net_error); CallbackInfo* callback_info = static_cast(request->GetUserData(&kUserDataKey)); DCHECK(callback_info); if (net_error != OK) { DVLOG(1) << "Failed to send report for " << request->url().host(); if (!callback_info->error_callback().is_null()) callback_info->error_callback().Run(request->url(), net_error, -1); } else if (request->GetResponseCode() != net::HTTP_OK) { if (!callback_info->error_callback().is_null()) callback_info->error_callback().Run(request->url(), OK, request->GetResponseCode()); } else { if (!callback_info->success_callback().is_null()) callback_info->success_callback().Run(); } CHECK_GT(inflight_requests_.erase(request), 0u); } void ReportSender::OnReadCompleted(URLRequest* request, int bytes_read) { NOTREACHED(); } } // namespace net