blob: 4d0e8ef3021afaa40796d1d368c639c30b03a7be [file] [log] [blame]
// Copyright 2016 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/tools/content_decoder_tool/content_decoder_tool.h"
#include <memory>
#include <utility>
#include "base/logging.h"
#include "base/strings/string_util.h"
#include "net/base/completion_once_callback.h"
#include "net/base/io_buffer.h"
#include "net/base/test_completion_callback.h"
#include "net/filter/brotli_source_stream.h"
#include "net/filter/gzip_source_stream.h"
#include "net/filter/source_stream.h"
namespace net {
namespace {
const int kBufferLen = 4096;
const char kDeflate[] = "deflate";
const char kGZip[] = "gzip";
const char kXGZip[] = "x-gzip";
const char kBrotli[] = "br";
class StdinSourceStream : public SourceStream {
public:
explicit StdinSourceStream(std::istream* input_stream)
: SourceStream(SourceStream::TYPE_NONE), input_stream_(input_stream) {}
~StdinSourceStream() override = default;
// SourceStream implementation.
int Read(IOBuffer* dest_buffer,
int buffer_size,
CompletionOnceCallback callback) override {
if (input_stream_->eof())
return OK;
if (input_stream_) {
input_stream_->read(dest_buffer->data(), buffer_size);
int bytes = input_stream_->gcount();
return bytes;
}
return ERR_FAILED;
}
std::string Description() const override { return ""; }
bool MayHaveMoreBytes() const override { return true; }
private:
std::istream* input_stream_;
DISALLOW_COPY_AND_ASSIGN(StdinSourceStream);
};
} // namespace
// static
bool ContentDecoderToolProcessInput(std::vector<std::string> content_encodings,
std::istream* input_stream,
std::ostream* output_stream) {
std::unique_ptr<SourceStream> upstream(
std::make_unique<StdinSourceStream>(input_stream));
for (std::vector<std::string>::const_reverse_iterator riter =
content_encodings.rbegin();
riter != content_encodings.rend(); ++riter) {
std::string content_encoding = *riter;
std::unique_ptr<SourceStream> downstream = nullptr;
if (base::LowerCaseEqualsASCII(content_encoding, kBrotli)) {
downstream = CreateBrotliSourceStream(std::move(upstream));
} else if (base::LowerCaseEqualsASCII(content_encoding, kDeflate)) {
downstream = GzipSourceStream::Create(std::move(upstream),
SourceStream::TYPE_DEFLATE);
} else if (base::LowerCaseEqualsASCII(content_encoding, kGZip) ||
base::LowerCaseEqualsASCII(content_encoding, kXGZip)) {
downstream = GzipSourceStream::Create(std::move(upstream),
SourceStream::TYPE_GZIP);
} else {
LOG(ERROR) << "Unsupported decoder '" << content_encoding << "'.";
return false;
}
if (downstream == nullptr) {
LOG(ERROR) << "Couldn't create the decoder.";
return false;
}
upstream = std::move(downstream);
}
if (!upstream) {
LOG(ERROR) << "Couldn't create the decoder.";
return false;
}
scoped_refptr<IOBuffer> read_buffer =
base::MakeRefCounted<IOBufferWithSize>(kBufferLen);
while (true) {
TestCompletionCallback callback;
int bytes_read =
upstream->Read(read_buffer.get(), kBufferLen, callback.callback());
if (bytes_read == ERR_IO_PENDING)
bytes_read = callback.WaitForResult();
if (bytes_read < 0) {
LOG(ERROR) << "Couldn't decode stdin.";
return false;
}
output_stream->write(read_buffer->data(), bytes_read);
// If EOF is read, break out the while loop.
if (bytes_read == 0)
break;
}
return true;
}
} // namespace net