1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137
|
// Copyright 2024 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "content/browser/devtools/request_body_collector.h"
#include "base/containers/extend.h"
#include "base/memory/raw_ref.h"
#include "base/numerics/safe_conversions.h"
#include "mojo/public/cpp/system/data_pipe.h"
#include "mojo/public/cpp/system/data_pipe_drainer.h"
#include "net/base/net_errors.h"
#include "services/network/public/cpp/data_element.h"
#include "services/network/public/cpp/resource_request_body.h"
#include "services/network/public/mojom/data_pipe_getter.mojom.h"
namespace content {
class RequestBodyCollector::BodyReader : public mojo::DataPipeDrainer::Client {
public:
BodyReader(
RequestBodyCollector& collector,
mojo::PendingRemote<network::mojom::DataPipeGetter> data_pipe_getter)
: collector_(collector), data_pipe_getter_(std::move(data_pipe_getter)) {
data_pipe_getter_.set_disconnect_handler(
base::BindOnce(&BodyReader::OnFailure, base::Unretained(this)));
mojo::ScopedDataPipeProducerHandle pipe_producer;
MojoResult result =
CreateDataPipe(/*options=*/nullptr, pipe_producer, pipe_consumer_);
CHECK_EQ(MOJO_RESULT_OK, result);
data_pipe_getter_->Read(
std::move(pipe_producer),
base::BindOnce(&BodyReader::OnReadStarted, base::Unretained(this)));
}
~BodyReader() override = default;
private:
// mojo::DataPipeDrainer::Client overrides
void OnDataAvailable(base::span<const uint8_t> data) override {
CHECK_NE(expected_size_, 0ul);
base::Extend(bytes_, data);
}
void OnDataComplete() override {
BodyEntry entry = expected_size_ == bytes_.size()
? BodyEntry(std::move(bytes_))
: base::unexpected("Unexpected end of data");
collector_->OnReaderComplete(this, std::move(entry));
}
void OnFailure() {
collector_->OnReaderComplete(this, base::unexpected("Error reading blob"));
// `this` is invalid at this point.
}
void OnReadStarted(int32_t status, uint64_t size) {
if (status != net::OK) {
OnFailure();
// `this` is invalid at this point.
return;
}
expected_size_ = base::checked_cast<size_t>(size);
bytes_.reserve(expected_size_);
pipe_drainer_.emplace(this, std::move(pipe_consumer_));
}
const raw_ref<RequestBodyCollector> collector_;
mojo::Remote<network::mojom::DataPipeGetter> data_pipe_getter_;
size_t expected_size_ = 0;
mojo::ScopedDataPipeConsumerHandle pipe_consumer_;
std::optional<mojo::DataPipeDrainer> pipe_drainer_;
std::vector<uint8_t> bytes_;
};
// static
std::unique_ptr<RequestBodyCollector> RequestBodyCollector::Collect(
const network::ResourceRequestBody& request_body,
CompletionCallback callback) {
std::vector<BodyEntry> bodies;
const auto& elements = *request_body.elements();
bodies.resize(elements.size());
ReadersMap readers;
std::unique_ptr<RequestBodyCollector> collector(new RequestBodyCollector());
for (size_t i = 0; i < elements.size(); ++i) {
const network::DataElement& element = elements[i];
switch (element.type()) {
case network::DataElement::Tag::kBytes:
bodies[i] = element.As<network::DataElementBytes>().bytes();
break;
case network::DataElement::Tag::kDataPipe: {
mojo::PendingRemote<network::mojom::DataPipeGetter> data_pipe_getter =
element.As<network::DataElementDataPipe>().CloneDataPipeGetter();
readers.insert(
std::make_pair(std::make_unique<BodyReader>(
*collector, std::move(data_pipe_getter)),
i));
break;
}
case network::DataElement::Tag::kFile:
case network::DataElement::Tag::kChunkedDataPipe:
bodies[i] = base::unexpected("Unsupported entry");
break;
}
}
if (readers.empty()) {
std::move(callback).Run(std::move(bodies));
return nullptr;
}
collector->callback_ = std::move(callback);
collector->bodies_ = std::move(bodies);
collector->readers_ = std::move(readers);
return collector;
}
RequestBodyCollector::~RequestBodyCollector() = default;
RequestBodyCollector::RequestBodyCollector() = default;
void RequestBodyCollector::OnReaderComplete(BodyReader* reader,
BodyEntry entry) {
auto it = readers_.find(reader);
CHECK(it != readers_.end());
bodies_[it->second] = std::move(entry);
readers_.erase(it);
if (!readers_.empty()) {
return;
}
std::move(callback_).Run(bodies_);
// `this` may be invalid at this point due to callback invocation above.
}
} // namespace content
|