| Index: net/filter/gzip_stream_source.h | 
| diff --git a/net/filter/gzip_stream_source.h b/net/filter/gzip_stream_source.h | 
| new file mode 100644 | 
| index 0000000000000000000000000000000000000000..09f488f81bcf2d4e6cadc6b2715c5b04e3bec071 | 
| --- /dev/null | 
| +++ b/net/filter/gzip_stream_source.h | 
| @@ -0,0 +1,109 @@ | 
| +// 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. | 
| + | 
| +#ifndef NET_FILTER_GZIP_STREAM_SOURCE_H_ | 
| +#define NET_FILTER_GZIP_STREAM_SOURCE_H_ | 
| + | 
| +#include <memory> | 
| + | 
| +#include "base/macros.h" | 
| +#include "base/memory/ref_counted.h" | 
| +#include "net/base/io_buffer.h" | 
| +#include "net/base/net_export.h" | 
| +#include "net/filter/filter_stream_source.h" | 
| +#include "net/filter/gzip_header.h" | 
| + | 
| +typedef struct z_stream_s z_stream; | 
| + | 
| +namespace net { | 
| + | 
| +class NET_EXPORT_PRIVATE GzipStreamSource : public FilterStreamSource { | 
| + public: | 
| +  enum GzipStreamSourceMode { | 
| +    GZIP_STREAM_SOURCE_DEFLATE, | 
| +    GZIP_STREAM_SOURCE_GZIP, | 
| +    // Same as GZIP_STREAM_SOURCE_GZIP, but allow fallback to plain text. | 
| +    GZIP_STREAM_SOURCE_GZIP_WITH_FALLBACK, | 
| +  }; | 
| + | 
| +  ~GzipStreamSource() override; | 
| + | 
| +  // Creates a GzipStreamSource. Return nullptr if initialization fails. | 
| +  static std::unique_ptr<GzipStreamSource> Create( | 
| +      std::unique_ptr<StreamSource> previous, | 
| +      GzipStreamSourceMode mode); | 
| + | 
| + private: | 
| +  GzipStreamSource(std::unique_ptr<StreamSource> previous, | 
| +                   GzipStreamSourceMode mode); | 
| +  bool Init(); | 
| +  // StreamSource implementation | 
| +  std::string GetTypeAsString() const override; | 
| +  int FilterData(IOBuffer* output_buffer, | 
| +                 size_t output_buffer_size, | 
| +                 DrainableIOBuffer* input_buffer) override; | 
| + | 
| +  // Synchronous decompressor. This function consumes bytes from |input_buffer| | 
| +  // and decompresses them into |output_buffer| until it consumes all of | 
| +  // |input_buffer|. This decompressor will decompress a zlib stream (either | 
| +  // gzip or deflate) until the zlib reaches EOF, then will pass any further | 
| +  // input through untouched. | 
| +  int Decompress(IOBuffer* output_buffer, | 
| +                 size_t output_buffer_size, | 
| +                 DrainableIOBuffer* input_buffer); | 
| + | 
| +  // Returns how many bytes are drained from |input_buffer|. | 
| +  size_t Passthrough(char* output_buffer, | 
| +                     size_t output_buffer_size, | 
| +                     DrainableIOBuffer* input_buffer); | 
| + | 
| +  // Inserts a zlib header to the data stream before calling zlib inflate. | 
| +  // This is used to work around server bugs. | 
| +  // The function returns true on success and false otherwise. | 
| +  bool InsertZlibHeader(); | 
| + | 
| +  // Returns whether an invalid GZip header has been observed. This method | 
| +  // returns true as soon as an invalid GZip header is observed; it returns | 
| +  // false if a complete, valid header has been observed, or not enough bytes | 
| +  // have been seen to be sure. This method pulls its input from |buffer_|. | 
| +  bool IsGzipHeaderInvalid(DrainableIOBuffer* input_buffer); | 
| + | 
| +  // Returns whether this stream looks like it could be plain text (ie, not | 
| +  // actually gzipped). Right now this uses an extremely simple heuristic; see | 
| +  // the source for details. This method checks the input in |buffer_|, but does | 
| +  // not drain from it. | 
| +  bool ShouldFallbackToPlain(DrainableIOBuffer* input_buffer); | 
| + | 
| +  // Skips gzip footer bytes if necessary. Might drain input from |buffer_| if | 
| +  // there are still footer bytes to read. | 
| +  void SkipGzipFooterIfNeeded(DrainableIOBuffer* input_buffer); | 
| + | 
| +  scoped_refptr<IOBuffer> pending_read_buffer_; | 
| + | 
| +  // Indicates the type of content decoding the stream source is performing. | 
| +  // This variable is set only once by Init. | 
| +  GzipStreamSourceMode mode_; | 
| +  // The control block of zlib which actually does the decoding. | 
| +  // This data structure is initialized by Init and updated only by | 
| +  // Decompress, with InsertZlibHeader being the exception as a workaround. | 
| +  std::unique_ptr<z_stream> zlib_stream_; | 
| +  // Whether end of stream has been reached. | 
| +  bool zlib_eof_; | 
| +  // A flag used by Decompress to record whether we've successfully added | 
| +  // a zlib header to this stream. | 
| +  bool zlib_header_added_; | 
| +  // Used to parse the gzip header in gzip stream. | 
| +  // It is used when the decoding mode is GZIP_STREAM_SOURCE_GZIP. | 
| +  GZipHeader gzip_header_; | 
| +  // Whether gzip header should be checked for validity. | 
| +  bool should_check_gzip_header_; | 
| +  // Tracks how many bytes of gzip footer are yet to be filtered. | 
| +  size_t gzip_footer_bytes_left_; | 
| + | 
| +  DISALLOW_COPY_AND_ASSIGN(GzipStreamSource); | 
| +}; | 
| + | 
| +}  // namespace net | 
| + | 
| +#endif  // NET_FILTER_GZIP_STREAM_SOURCE_H__ | 
|  |