mirror of
https://github.com/chromium/crashpad.git
synced 2024-12-27 15:32:10 +08:00
Use zlib to gzip-compress uploads
This adds zlib to Crashpad. By default in standalone Crashpad builds, the system zlib will be used where available. A copy of Chromium’s zlib (currently a slightly patched 1.2.11) is checked out via DEPS into third_party for use on Windows, which does not have a system zlib. zlib is used to produce gzip streams for HTTP upload request bodies sent by crashpad_handler by default. The Content-Encoding: gzip header is set for these compressed request bodies. Compression can be disabled for upload to servers without corresponding decompression support by starting crashpad_handler with the --no-upload-gzip option. Most minidumps compress quite well with zlib. A size reduction of 90% is not uncommon. BUG=crashpad:157 TEST=crashpad_util_test GzipHTTPBodyStream.*:HTTPTransport.* Change-Id: I99b86db3952c3685cd78f5dc858a60b54399c513 Reviewed-on: https://chromium-review.googlesource.com/438585 Reviewed-by: Robert Sesek <rsesek@chromium.org>
This commit is contained in:
parent
d98a4de718
commit
0c322ecc3f
1
.gitignore
vendored
1
.gitignore
vendored
@ -14,5 +14,6 @@
|
|||||||
/third_party/gyp/gyp
|
/third_party/gyp/gyp
|
||||||
/third_party/llvm
|
/third_party/llvm
|
||||||
/third_party/mini_chromium/mini_chromium
|
/third_party/mini_chromium/mini_chromium
|
||||||
|
/third_party/zlib/zlib
|
||||||
/xcodebuild
|
/xcodebuild
|
||||||
tags
|
tags
|
||||||
|
3
DEPS
3
DEPS
@ -39,6 +39,9 @@ deps = {
|
|||||||
'crashpad/third_party/mini_chromium/mini_chromium':
|
'crashpad/third_party/mini_chromium/mini_chromium':
|
||||||
Var('chromium_git') + '/chromium/mini_chromium@' +
|
Var('chromium_git') + '/chromium/mini_chromium@' +
|
||||||
'f65519e442d23498937251e680a3b113927613b0',
|
'f65519e442d23498937251e680a3b113927613b0',
|
||||||
|
'crashpad/third_party/zlib/zlib':
|
||||||
|
Var('chromium_git') + '/chromium/src/third_party/zlib@' +
|
||||||
|
'13dc246a58e4b72104d35f9b1809af95221ebda7',
|
||||||
}
|
}
|
||||||
|
|
||||||
hooks = [
|
hooks = [
|
||||||
|
@ -139,7 +139,8 @@ class CallRecordUploadAttempt {
|
|||||||
|
|
||||||
CrashReportUploadThread::CrashReportUploadThread(CrashReportDatabase* database,
|
CrashReportUploadThread::CrashReportUploadThread(CrashReportDatabase* database,
|
||||||
const std::string& url,
|
const std::string& url,
|
||||||
bool rate_limit)
|
bool rate_limit,
|
||||||
|
bool upload_gzip)
|
||||||
: url_(url),
|
: url_(url),
|
||||||
// Check for pending reports every 15 minutes, even in the absence of a
|
// Check for pending reports every 15 minutes, even in the absence of a
|
||||||
// signal from the handler thread. This allows for failed uploads to be
|
// signal from the handler thread. This allows for failed uploads to be
|
||||||
@ -147,7 +148,8 @@ CrashReportUploadThread::CrashReportUploadThread(CrashReportDatabase* database,
|
|||||||
// processes to be recognized.
|
// processes to be recognized.
|
||||||
thread_(15 * 60, this),
|
thread_(15 * 60, this),
|
||||||
database_(database),
|
database_(database),
|
||||||
rate_limit_(rate_limit) {
|
rate_limit_(rate_limit),
|
||||||
|
upload_gzip_(upload_gzip) {
|
||||||
}
|
}
|
||||||
|
|
||||||
CrashReportUploadThread::~CrashReportUploadThread() {
|
CrashReportUploadThread::~CrashReportUploadThread() {
|
||||||
@ -308,6 +310,7 @@ CrashReportUploadThread::UploadResult CrashReportUploadThread::UploadReport(
|
|||||||
}
|
}
|
||||||
|
|
||||||
HTTPMultipartBuilder http_multipart_builder;
|
HTTPMultipartBuilder http_multipart_builder;
|
||||||
|
http_multipart_builder.SetGzipEnabled(upload_gzip_);
|
||||||
|
|
||||||
const char kMinidumpKey[] = "upload_file_minidump";
|
const char kMinidumpKey[] = "upload_file_minidump";
|
||||||
|
|
||||||
@ -332,9 +335,11 @@ CrashReportUploadThread::UploadResult CrashReportUploadThread::UploadReport(
|
|||||||
|
|
||||||
std::unique_ptr<HTTPTransport> http_transport(HTTPTransport::Create());
|
std::unique_ptr<HTTPTransport> http_transport(HTTPTransport::Create());
|
||||||
http_transport->SetURL(url_);
|
http_transport->SetURL(url_);
|
||||||
HTTPHeaders::value_type content_type =
|
HTTPHeaders content_headers;
|
||||||
http_multipart_builder.GetContentType();
|
http_multipart_builder.PopulateContentHeaders(&content_headers);
|
||||||
http_transport->SetHeader(content_type.first, content_type.second);
|
for (const auto& content_header : content_headers) {
|
||||||
|
http_transport->SetHeader(content_header.first, content_header.second);
|
||||||
|
}
|
||||||
http_transport->SetBodyStream(http_multipart_builder.GetBodyStream());
|
http_transport->SetBodyStream(http_multipart_builder.GetBodyStream());
|
||||||
// TODO(mark): The timeout should be configurable by the client.
|
// TODO(mark): The timeout should be configurable by the client.
|
||||||
http_transport->SetTimeout(60.0); // 1 minute.
|
http_transport->SetTimeout(60.0); // 1 minute.
|
||||||
|
@ -45,9 +45,11 @@ class CrashReportUploadThread : public WorkerThread::Delegate {
|
|||||||
//! \param[in] url The URL of the server to upload crash reports to.
|
//! \param[in] url The URL of the server to upload crash reports to.
|
||||||
//! \param[in] rate_limit Whether uploads should be throttled to a (currently
|
//! \param[in] rate_limit Whether uploads should be throttled to a (currently
|
||||||
//! hardcoded) rate.
|
//! hardcoded) rate.
|
||||||
|
//! \param[in] upload_gzip Whether uploads should use `gzip` compression.
|
||||||
CrashReportUploadThread(CrashReportDatabase* database,
|
CrashReportUploadThread(CrashReportDatabase* database,
|
||||||
const std::string& url,
|
const std::string& url,
|
||||||
bool rate_limit);
|
bool rate_limit,
|
||||||
|
bool upload_gzip);
|
||||||
~CrashReportUploadThread();
|
~CrashReportUploadThread();
|
||||||
|
|
||||||
//! \brief Starts a dedicated upload thread, which executes ThreadMain().
|
//! \brief Starts a dedicated upload thread, which executes ThreadMain().
|
||||||
@ -139,6 +141,7 @@ class CrashReportUploadThread : public WorkerThread::Delegate {
|
|||||||
WorkerThread thread_;
|
WorkerThread thread_;
|
||||||
CrashReportDatabase* database_; // weak
|
CrashReportDatabase* database_; // weak
|
||||||
bool rate_limit_;
|
bool rate_limit_;
|
||||||
|
bool upload_gzip_;
|
||||||
|
|
||||||
DISALLOW_COPY_AND_ASSIGN(CrashReportUploadThread);
|
DISALLOW_COPY_AND_ASSIGN(CrashReportUploadThread);
|
||||||
};
|
};
|
||||||
|
@ -144,6 +144,14 @@ establish the Crashpad client environment before running a program.
|
|||||||
throttled to one per hour. Using this option disables that behavior, and
|
throttled to one per hour. Using this option disables that behavior, and
|
||||||
Crashpad will attempt to upload all captured reports.
|
Crashpad will attempt to upload all captured reports.
|
||||||
|
|
||||||
|
* **--no-upload-gzip**
|
||||||
|
|
||||||
|
Do not use `gzip` compression for uploaded crash reports. Normally, the
|
||||||
|
entire request body is compressed into a `gzip` stream and transmitted with
|
||||||
|
`Content-Encoding: gzip`. This option disables compression, and is intended
|
||||||
|
for use with collection servers that don’t accept uploads compressed in this
|
||||||
|
way.
|
||||||
|
|
||||||
* **--pipe-name**=_PIPE_
|
* **--pipe-name**=_PIPE_
|
||||||
|
|
||||||
Listen on the given pipe name for connections from clients. _PIPE_ must be of
|
Listen on the given pipe name for connections from clients. _PIPE_ must be of
|
||||||
|
@ -93,6 +93,7 @@ void Usage(const base::FilePath& me) {
|
|||||||
#endif // OS_MACOSX
|
#endif // OS_MACOSX
|
||||||
" --metrics-dir=DIR store metrics files in DIR (only in Chromium)\n"
|
" --metrics-dir=DIR store metrics files in DIR (only in Chromium)\n"
|
||||||
" --no-rate-limit don't rate limit crash uploads\n"
|
" --no-rate-limit don't rate limit crash uploads\n"
|
||||||
|
" --no-upload-gzip don't use gzip compression when uploading\n"
|
||||||
#if defined(OS_MACOSX)
|
#if defined(OS_MACOSX)
|
||||||
" --reset-own-crash-exception-port-to-system-default\n"
|
" --reset-own-crash-exception-port-to-system-default\n"
|
||||||
" reset the server's exception handler to default\n"
|
" reset the server's exception handler to default\n"
|
||||||
@ -291,6 +292,7 @@ int HandlerMain(int argc, char* argv[]) {
|
|||||||
#endif // OS_MACOSX
|
#endif // OS_MACOSX
|
||||||
kOptionMetrics,
|
kOptionMetrics,
|
||||||
kOptionNoRateLimit,
|
kOptionNoRateLimit,
|
||||||
|
kOptionNoUploadGzip,
|
||||||
#if defined(OS_MACOSX)
|
#if defined(OS_MACOSX)
|
||||||
kOptionResetOwnCrashExceptionPortToSystemDefault,
|
kOptionResetOwnCrashExceptionPortToSystemDefault,
|
||||||
#elif defined(OS_WIN)
|
#elif defined(OS_WIN)
|
||||||
@ -317,11 +319,13 @@ int HandlerMain(int argc, char* argv[]) {
|
|||||||
InitialClientData initial_client_data;
|
InitialClientData initial_client_data;
|
||||||
#endif // OS_MACOSX
|
#endif // OS_MACOSX
|
||||||
bool rate_limit;
|
bool rate_limit;
|
||||||
|
bool upload_gzip;
|
||||||
} options = {};
|
} options = {};
|
||||||
#if defined(OS_MACOSX)
|
#if defined(OS_MACOSX)
|
||||||
options.handshake_fd = -1;
|
options.handshake_fd = -1;
|
||||||
#endif
|
#endif
|
||||||
options.rate_limit = true;
|
options.rate_limit = true;
|
||||||
|
options.upload_gzip = true;
|
||||||
|
|
||||||
const option long_options[] = {
|
const option long_options[] = {
|
||||||
{"annotation", required_argument, nullptr, kOptionAnnotation},
|
{"annotation", required_argument, nullptr, kOptionAnnotation},
|
||||||
@ -340,6 +344,7 @@ int HandlerMain(int argc, char* argv[]) {
|
|||||||
#endif // OS_MACOSX
|
#endif // OS_MACOSX
|
||||||
{"metrics-dir", required_argument, nullptr, kOptionMetrics},
|
{"metrics-dir", required_argument, nullptr, kOptionMetrics},
|
||||||
{"no-rate-limit", no_argument, nullptr, kOptionNoRateLimit},
|
{"no-rate-limit", no_argument, nullptr, kOptionNoRateLimit},
|
||||||
|
{"no-upload-gzip", no_argument, nullptr, kOptionNoUploadGzip},
|
||||||
#if defined(OS_MACOSX)
|
#if defined(OS_MACOSX)
|
||||||
{"reset-own-crash-exception-port-to-system-default",
|
{"reset-own-crash-exception-port-to-system-default",
|
||||||
no_argument,
|
no_argument,
|
||||||
@ -407,6 +412,10 @@ int HandlerMain(int argc, char* argv[]) {
|
|||||||
options.rate_limit = false;
|
options.rate_limit = false;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
case kOptionNoUploadGzip: {
|
||||||
|
options.upload_gzip = false;
|
||||||
|
break;
|
||||||
|
}
|
||||||
#if defined(OS_MACOSX)
|
#if defined(OS_MACOSX)
|
||||||
case kOptionResetOwnCrashExceptionPortToSystemDefault: {
|
case kOptionResetOwnCrashExceptionPortToSystemDefault: {
|
||||||
options.reset_own_crash_exception_port_to_system_default = true;
|
options.reset_own_crash_exception_port_to_system_default = true;
|
||||||
@ -555,7 +564,7 @@ int HandlerMain(int argc, char* argv[]) {
|
|||||||
// configurable database setting to control upload limiting.
|
// configurable database setting to control upload limiting.
|
||||||
// See https://crashpad.chromium.org/bug/23.
|
// See https://crashpad.chromium.org/bug/23.
|
||||||
CrashReportUploadThread upload_thread(
|
CrashReportUploadThread upload_thread(
|
||||||
database.get(), options.url, options.rate_limit);
|
database.get(), options.url, options.rate_limit, options.upload_gzip);
|
||||||
upload_thread.Start();
|
upload_thread.Start();
|
||||||
|
|
||||||
PruneCrashReportThread prune_thread(database.get(),
|
PruneCrashReportThread prune_thread(database.get(),
|
||||||
|
18
third_party/zlib/README.crashpad
vendored
Normal file
18
third_party/zlib/README.crashpad
vendored
Normal file
@ -0,0 +1,18 @@
|
|||||||
|
Name: zlib
|
||||||
|
Short Name: zlib
|
||||||
|
URL: http://zlib.net/
|
||||||
|
Revision: See zlib/README.chromium
|
||||||
|
License: zlib
|
||||||
|
License File: zlib/LICENSE
|
||||||
|
Security Critical: yes
|
||||||
|
|
||||||
|
Description:
|
||||||
|
“A massively spiffy yet delicately unobtrusive compression library.”
|
||||||
|
|
||||||
|
zlib is a free, general-purpose, legally unencumbered lossless data-compression
|
||||||
|
library. zlib implements the “deflate” compression algorithm described by RFC
|
||||||
|
1951, which combines the LZ77 (Lempel-Ziv) algorithm with Huffman coding. zlib
|
||||||
|
also implements the zlib (RFC 1950) and gzip (RFC 1952) wrapper formats.
|
||||||
|
|
||||||
|
Local Modifications:
|
||||||
|
See zlib/README.chromium.
|
137
third_party/zlib/zlib.gyp
vendored
Normal file
137
third_party/zlib/zlib.gyp
vendored
Normal file
@ -0,0 +1,137 @@
|
|||||||
|
# Copyright 2017 The Crashpad Authors. All rights reserved.
|
||||||
|
#
|
||||||
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
# you may not use this file except in compliance with the License.
|
||||||
|
# You may obtain a copy of the License at
|
||||||
|
#
|
||||||
|
# http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
#
|
||||||
|
# Unless required by applicable law or agreed to in writing, software
|
||||||
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
# See the License for the specific language governing permissions and
|
||||||
|
# limitations under the License.
|
||||||
|
|
||||||
|
{
|
||||||
|
'variables': {
|
||||||
|
'conditions': [
|
||||||
|
# Use the system zlib by default where available, as it is on most
|
||||||
|
# platforms. Windows does not have a system zlib, so use “embedded” which
|
||||||
|
# directs the build to use the source code in the zlib subdirectory.
|
||||||
|
['OS!="win"', {
|
||||||
|
'zlib_source%': 'system',
|
||||||
|
}, {
|
||||||
|
'zlib_source%': 'embedded',
|
||||||
|
}],
|
||||||
|
],
|
||||||
|
},
|
||||||
|
'targets': [
|
||||||
|
{
|
||||||
|
'target_name': 'zlib',
|
||||||
|
'conditions': [
|
||||||
|
['zlib_source=="system"', {
|
||||||
|
'type': 'none',
|
||||||
|
'direct_dependent_settings': {
|
||||||
|
'defines': [
|
||||||
|
'CRASHPAD_ZLIB_SOURCE_SYSTEM',
|
||||||
|
],
|
||||||
|
},
|
||||||
|
'link_settings': {
|
||||||
|
'conditions': [
|
||||||
|
['OS=="mac"', {
|
||||||
|
'libraries': [
|
||||||
|
'$(SDKROOT)/usr/lib/libz.dylib',
|
||||||
|
],
|
||||||
|
}, {
|
||||||
|
'libraries': [
|
||||||
|
'-lz',
|
||||||
|
],
|
||||||
|
}],
|
||||||
|
],
|
||||||
|
},
|
||||||
|
}],
|
||||||
|
['zlib_source=="embedded"', {
|
||||||
|
'type': 'static_library',
|
||||||
|
'include_dirs': [
|
||||||
|
'zlib',
|
||||||
|
],
|
||||||
|
'defines': [
|
||||||
|
'CRASHPAD_ZLIB_SOURCE_EMBEDDED',
|
||||||
|
'HAVE_STDARG_H',
|
||||||
|
],
|
||||||
|
'direct_dependent_settings': {
|
||||||
|
'include_dirs': [
|
||||||
|
'zlib',
|
||||||
|
],
|
||||||
|
'defines': [
|
||||||
|
'CRASHPAD_ZLIB_SOURCE_EMBEDDED',
|
||||||
|
],
|
||||||
|
},
|
||||||
|
'sources': [
|
||||||
|
'zlib/adler32.c',
|
||||||
|
'zlib/compress.c',
|
||||||
|
'zlib/crc32.c',
|
||||||
|
'zlib/crc32.h',
|
||||||
|
'zlib/crc_folding.c',
|
||||||
|
'zlib/deflate.c',
|
||||||
|
'zlib/deflate.h',
|
||||||
|
'zlib/fill_window_sse.c',
|
||||||
|
'zlib/gzclose.c',
|
||||||
|
'zlib/gzguts.h',
|
||||||
|
'zlib/gzlib.c',
|
||||||
|
'zlib/gzread.c',
|
||||||
|
'zlib/gzwrite.c',
|
||||||
|
'zlib/infback.c',
|
||||||
|
'zlib/inffast.c',
|
||||||
|
'zlib/inffast.h',
|
||||||
|
'zlib/inffixed.h',
|
||||||
|
'zlib/inflate.c',
|
||||||
|
'zlib/inflate.h',
|
||||||
|
'zlib/inftrees.c',
|
||||||
|
'zlib/inftrees.h',
|
||||||
|
'zlib/names.h',
|
||||||
|
'zlib/simd_stub.c',
|
||||||
|
'zlib/trees.c',
|
||||||
|
'zlib/trees.h',
|
||||||
|
'zlib/uncompr.c',
|
||||||
|
'zlib/x86.c',
|
||||||
|
'zlib/x86.h',
|
||||||
|
'zlib/zconf.h',
|
||||||
|
'zlib/zlib.h',
|
||||||
|
'zlib/zutil.c',
|
||||||
|
'zlib/zutil.h',
|
||||||
|
'zlib_crashpad.h',
|
||||||
|
],
|
||||||
|
'conditions': [
|
||||||
|
['target_arch=="x86" or target_arch=="amd64"', {
|
||||||
|
'sources!': [
|
||||||
|
'zlib/simd_stub.c',
|
||||||
|
],
|
||||||
|
}, {
|
||||||
|
'sources!': [
|
||||||
|
'zlib/crc_folding.c',
|
||||||
|
'zlib/fill_window_sse.c',
|
||||||
|
'zlib/x86.c',
|
||||||
|
'zlib/x86.h',
|
||||||
|
],
|
||||||
|
}],
|
||||||
|
['OS!="win"', {
|
||||||
|
'defines': [
|
||||||
|
'HAVE_HIDDEN',
|
||||||
|
'HAVE_UNISTD_H',
|
||||||
|
],
|
||||||
|
}, {
|
||||||
|
'msvs_disabled_warnings': [
|
||||||
|
4131, # uses old-style declarator
|
||||||
|
4244, # conversion from 't1' to 't2', possible loss of data
|
||||||
|
4245, # conversion from 't1' to 't2', signed/unsigned mismatch
|
||||||
|
4267, # conversion from 'size_t' to 't', possible loss of data
|
||||||
|
4324, # structure was padded due to alignment specifier
|
||||||
|
],
|
||||||
|
}],
|
||||||
|
],
|
||||||
|
}],
|
||||||
|
],
|
||||||
|
},
|
||||||
|
],
|
||||||
|
}
|
32
third_party/zlib/zlib_crashpad.h
vendored
Normal file
32
third_party/zlib/zlib_crashpad.h
vendored
Normal file
@ -0,0 +1,32 @@
|
|||||||
|
// Copyright 2017 The Crashpad Authors. All rights reserved.
|
||||||
|
//
|
||||||
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
// you may not use this file except in compliance with the License.
|
||||||
|
// You may obtain a copy of the License at
|
||||||
|
//
|
||||||
|
// http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
//
|
||||||
|
// Unless required by applicable law or agreed to in writing, software
|
||||||
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
// See the License for the specific language governing permissions and
|
||||||
|
// limitations under the License.
|
||||||
|
|
||||||
|
#ifndef CRASHPAD_THIRD_PARTY_ZLIB_ZLIB_CRASHPAD_H_
|
||||||
|
#define CRASHPAD_THIRD_PARTY_ZLIB_ZLIB_CRASHPAD_H_
|
||||||
|
|
||||||
|
// #include this file instead of the system version of <zlib.h> or equivalent
|
||||||
|
// available at any other location in the source tree. It will #include the
|
||||||
|
// proper <zlib.h> depending on how the build has been configured.
|
||||||
|
|
||||||
|
#if defined(CRASHPAD_ZLIB_SOURCE_SYSTEM)
|
||||||
|
#include <zlib.h>
|
||||||
|
#elif defined(CRASHPAD_ZLIB_SOURCE_EMBEDDED)
|
||||||
|
#include "third_party/zlib/zlib/zlib.h"
|
||||||
|
#elif defined(CRASHPAD_ZLIB_SOURCE_CHROMIUM)
|
||||||
|
#include "third_party/zlib/zlib.h"
|
||||||
|
#else
|
||||||
|
#error Unknown zlib source
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif // CRASHPAD_THIRD_PARTY_ZLIB_ZLIB_CRASHPAD_H_
|
37
util/misc/zlib.cc
Normal file
37
util/misc/zlib.cc
Normal file
@ -0,0 +1,37 @@
|
|||||||
|
// Copyright 2017 The Crashpad Authors. All rights reserved.
|
||||||
|
//
|
||||||
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
// you may not use this file except in compliance with the License.
|
||||||
|
// You may obtain a copy of the License at
|
||||||
|
//
|
||||||
|
// http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
//
|
||||||
|
// Unless required by applicable law or agreed to in writing, software
|
||||||
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
// See the License for the specific language governing permissions and
|
||||||
|
// limitations under the License.
|
||||||
|
|
||||||
|
#include "util/misc/zlib.h"
|
||||||
|
|
||||||
|
#include "base/logging.h"
|
||||||
|
#include "base/strings/stringprintf.h"
|
||||||
|
#include "third_party/zlib/zlib_crashpad.h"
|
||||||
|
|
||||||
|
namespace crashpad {
|
||||||
|
|
||||||
|
int ZlibWindowBitsWithGzipWrapper(int window_bits) {
|
||||||
|
// See the documentation for deflateInit2() and inflateInit2() in <zlib.h>. 0
|
||||||
|
// is only valid during decompression.
|
||||||
|
|
||||||
|
DCHECK(window_bits == 0 || (window_bits >= 8 && window_bits <= 15))
|
||||||
|
<< window_bits;
|
||||||
|
|
||||||
|
return 16 + window_bits;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::string ZlibErrorString(int zr) {
|
||||||
|
return base::StringPrintf("%s (%d)", zError(zr), zr);
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace crashpad
|
42
util/misc/zlib.h
Normal file
42
util/misc/zlib.h
Normal file
@ -0,0 +1,42 @@
|
|||||||
|
// Copyright 2017 The Crashpad Authors. All rights reserved.
|
||||||
|
//
|
||||||
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
// you may not use this file except in compliance with the License.
|
||||||
|
// You may obtain a copy of the License at
|
||||||
|
//
|
||||||
|
// http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
//
|
||||||
|
// Unless required by applicable law or agreed to in writing, software
|
||||||
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
// See the License for the specific language governing permissions and
|
||||||
|
// limitations under the License.
|
||||||
|
|
||||||
|
#ifndef CRASHPAD_UTIL_MISC_ZLIB_H_
|
||||||
|
#define CRASHPAD_UTIL_MISC_ZLIB_H_
|
||||||
|
|
||||||
|
#include <string>
|
||||||
|
|
||||||
|
namespace crashpad {
|
||||||
|
|
||||||
|
//! \brief Obtain a \a window_bits parameter to pass to `deflateInit2()` or
|
||||||
|
//! `inflateInit2()` that specifies a `gzip` wrapper instead of the default
|
||||||
|
//! zlib wrapper.
|
||||||
|
//!
|
||||||
|
//! \param[in] A \a window_bits value that only specifies the base-2 logarithm
|
||||||
|
//! of the deflate sliding window size.
|
||||||
|
//!
|
||||||
|
//! \return \a window_bits adjusted to specify a `gzip` wrapper, to be passed to
|
||||||
|
//! `deflateInit2()` or `inflateInit2()`.
|
||||||
|
int ZlibWindowBitsWithGzipWrapper(int window_bits);
|
||||||
|
|
||||||
|
//! \brief Formats a string for an error received from the zlib library.
|
||||||
|
//!
|
||||||
|
//! \param[in] zr A zlib result code, such as `Z_STREAM_ERROR`.
|
||||||
|
//!
|
||||||
|
//! \return A formatted string.
|
||||||
|
std::string ZlibErrorString(int zr);
|
||||||
|
|
||||||
|
} // namespace crashpad
|
||||||
|
|
||||||
|
#endif // CRASHPAD_UTIL_MISC_ZLIB_H_
|
126
util/net/http_body_gzip.cc
Normal file
126
util/net/http_body_gzip.cc
Normal file
@ -0,0 +1,126 @@
|
|||||||
|
// Copyright 2017 The Crashpad Authors. All rights reserved.
|
||||||
|
//
|
||||||
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
// you may not use this file except in compliance with the License.
|
||||||
|
// You may obtain a copy of the License at
|
||||||
|
//
|
||||||
|
// http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
//
|
||||||
|
// Unless required by applicable law or agreed to in writing, software
|
||||||
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
// See the License for the specific language governing permissions and
|
||||||
|
// limitations under the License.
|
||||||
|
|
||||||
|
#include "util/net/http_body_gzip.h"
|
||||||
|
|
||||||
|
#include <utility>
|
||||||
|
|
||||||
|
#include "base/logging.h"
|
||||||
|
#include "base/numerics/safe_conversions.h"
|
||||||
|
#include "third_party/zlib/zlib_crashpad.h"
|
||||||
|
#include "util/misc/zlib.h"
|
||||||
|
|
||||||
|
namespace crashpad {
|
||||||
|
|
||||||
|
GzipHTTPBodyStream::GzipHTTPBodyStream(std::unique_ptr<HTTPBodyStream> source)
|
||||||
|
: input_(),
|
||||||
|
source_(std::move(source)),
|
||||||
|
z_stream_(new z_stream()),
|
||||||
|
state_(State::kUninitialized) {}
|
||||||
|
|
||||||
|
GzipHTTPBodyStream::~GzipHTTPBodyStream() {
|
||||||
|
DCHECK(state_ == State::kUninitialized ||
|
||||||
|
state_ == State::kFinished ||
|
||||||
|
state_ == State::kError);
|
||||||
|
}
|
||||||
|
|
||||||
|
FileOperationResult GzipHTTPBodyStream::GetBytesBuffer(uint8_t* buffer,
|
||||||
|
size_t max_len) {
|
||||||
|
if (state_ == State::kError) {
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (state_ == State::kFinished) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (state_ == State::kUninitialized) {
|
||||||
|
z_stream_->zalloc = Z_NULL;
|
||||||
|
z_stream_->zfree = Z_NULL;
|
||||||
|
z_stream_->opaque = Z_NULL;
|
||||||
|
|
||||||
|
// The default values for zlib’s internal MAX_WBITS and DEF_MEM_LEVEL. These
|
||||||
|
// are the values that deflateInit() would use, but they’re not exported
|
||||||
|
// from zlib. deflateInit2() is used instead of deflateInit() to get the
|
||||||
|
// gzip wrapper.
|
||||||
|
const int kZlibMaxWindowBits = 15;
|
||||||
|
const int kZlibDefaultMemoryLevel = 8;
|
||||||
|
|
||||||
|
int zr = deflateInit2(z_stream_.get(),
|
||||||
|
Z_DEFAULT_COMPRESSION,
|
||||||
|
Z_DEFLATED,
|
||||||
|
ZlibWindowBitsWithGzipWrapper(kZlibMaxWindowBits),
|
||||||
|
kZlibDefaultMemoryLevel,
|
||||||
|
Z_DEFAULT_STRATEGY);
|
||||||
|
if (zr != Z_OK) {
|
||||||
|
LOG(ERROR) << "deflateInit2: " << ZlibErrorString(zr);
|
||||||
|
state_ = State::kError;
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
state_ = State::kOperating;
|
||||||
|
}
|
||||||
|
|
||||||
|
z_stream_->next_out = buffer;
|
||||||
|
z_stream_->avail_out = base::saturated_cast<uInt>(max_len);
|
||||||
|
|
||||||
|
while (state_ != State::kFinished && z_stream_->avail_out > 0) {
|
||||||
|
if (state_ != State::kInputEOF && z_stream_->avail_in == 0) {
|
||||||
|
FileOperationResult input_bytes =
|
||||||
|
source_->GetBytesBuffer(input_, sizeof(input_));
|
||||||
|
if (input_bytes == -1) {
|
||||||
|
Done(State::kError);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (input_bytes == 0) {
|
||||||
|
state_ = State::kInputEOF;
|
||||||
|
}
|
||||||
|
|
||||||
|
z_stream_->next_in = input_;
|
||||||
|
z_stream_->avail_in = base::checked_cast<uInt>(input_bytes);
|
||||||
|
}
|
||||||
|
|
||||||
|
int zr = deflate(z_stream_.get(),
|
||||||
|
state_ == State::kInputEOF ? Z_FINISH : Z_NO_FLUSH);
|
||||||
|
if (state_ == State::kInputEOF && zr == Z_STREAM_END) {
|
||||||
|
Done(State::kFinished);
|
||||||
|
if (state_ == State::kError) {
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
} else if (zr != Z_OK) {
|
||||||
|
LOG(ERROR) << "deflate: " << ZlibErrorString(zr);
|
||||||
|
Done(State::kError);
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
DCHECK_LE(z_stream_->avail_out, max_len);
|
||||||
|
return max_len - z_stream_->avail_out;
|
||||||
|
}
|
||||||
|
|
||||||
|
void GzipHTTPBodyStream::Done(State state) {
|
||||||
|
DCHECK(state_ == State::kOperating || state_ == State::kInputEOF) << state_;
|
||||||
|
DCHECK(state == State::kFinished || state == State::kError) << state;
|
||||||
|
|
||||||
|
int zr = deflateEnd(z_stream_.get());
|
||||||
|
if (zr != Z_OK) {
|
||||||
|
LOG(ERROR) << "deflateEnd: " << ZlibErrorString(zr);
|
||||||
|
state_ = State::kError;
|
||||||
|
} else {
|
||||||
|
state_ = state;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace crashpad
|
67
util/net/http_body_gzip.h
Normal file
67
util/net/http_body_gzip.h
Normal file
@ -0,0 +1,67 @@
|
|||||||
|
// Copyright 2017 The Crashpad Authors. All rights reserved.
|
||||||
|
//
|
||||||
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
// you may not use this file except in compliance with the License.
|
||||||
|
// You may obtain a copy of the License at
|
||||||
|
//
|
||||||
|
// http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
//
|
||||||
|
// Unless required by applicable law or agreed to in writing, software
|
||||||
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
// See the License for the specific language governing permissions and
|
||||||
|
// limitations under the License.
|
||||||
|
|
||||||
|
#ifndef CRASHPAD_UTIL_NET_HTTP_BODY_GZIP_H_
|
||||||
|
#define CRASHPAD_UTIL_NET_HTTP_BODY_GZIP_H_
|
||||||
|
|
||||||
|
#include <stdint.h>
|
||||||
|
#include <sys/types.h>
|
||||||
|
|
||||||
|
#include <memory>
|
||||||
|
|
||||||
|
#include "base/macros.h"
|
||||||
|
#include "util/file/file_io.h"
|
||||||
|
#include "util/net/http_body.h"
|
||||||
|
|
||||||
|
extern "C" {
|
||||||
|
typedef struct z_stream_s z_stream;
|
||||||
|
} // extern "C"
|
||||||
|
|
||||||
|
namespace crashpad {
|
||||||
|
|
||||||
|
//! \brief An implementation of HTTPBodyStream that `gzip`-compresses another
|
||||||
|
//! HTTPBodyStream.
|
||||||
|
class GzipHTTPBodyStream : public HTTPBodyStream {
|
||||||
|
public:
|
||||||
|
explicit GzipHTTPBodyStream(std::unique_ptr<HTTPBodyStream> source);
|
||||||
|
|
||||||
|
~GzipHTTPBodyStream() override;
|
||||||
|
|
||||||
|
// HTTPBodyStream:
|
||||||
|
FileOperationResult GetBytesBuffer(uint8_t* buffer, size_t max_len) override;
|
||||||
|
|
||||||
|
private:
|
||||||
|
enum State : int {
|
||||||
|
kUninitialized,
|
||||||
|
kOperating,
|
||||||
|
kInputEOF,
|
||||||
|
kFinished,
|
||||||
|
kError,
|
||||||
|
};
|
||||||
|
|
||||||
|
// Calls deflateEnd() and transitions state_ to state. If deflateEnd() fails,
|
||||||
|
// logs a message and transitions state_ to State::kError.
|
||||||
|
void Done(State state);
|
||||||
|
|
||||||
|
uint8_t input_[4096];
|
||||||
|
std::unique_ptr<HTTPBodyStream> source_;
|
||||||
|
std::unique_ptr<z_stream> z_stream_;
|
||||||
|
State state_;
|
||||||
|
|
||||||
|
DISALLOW_COPY_AND_ASSIGN(GzipHTTPBodyStream);
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace crashpad
|
||||||
|
|
||||||
|
#endif // CRASHPAD_UTIL_NET_HTTP_BODY_GZIP_H_
|
178
util/net/http_body_gzip_test.cc
Normal file
178
util/net/http_body_gzip_test.cc
Normal file
@ -0,0 +1,178 @@
|
|||||||
|
// Copyright 2017 The Crashpad Authors. All rights reserved.
|
||||||
|
//
|
||||||
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
// you may not use this file except in compliance with the License.
|
||||||
|
// You may obtain a copy of the License at
|
||||||
|
//
|
||||||
|
// http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
//
|
||||||
|
// Unless required by applicable law or agreed to in writing, software
|
||||||
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
// See the License for the specific language governing permissions and
|
||||||
|
// limitations under the License.
|
||||||
|
|
||||||
|
#include "util/net/http_body_gzip.h"
|
||||||
|
|
||||||
|
#include <string.h>
|
||||||
|
|
||||||
|
#include <algorithm>
|
||||||
|
#include <memory>
|
||||||
|
#include <string>
|
||||||
|
#include <utility>
|
||||||
|
|
||||||
|
#include "base/macros.h"
|
||||||
|
#include "base/rand_util.h"
|
||||||
|
#include "base/numerics/safe_conversions.h"
|
||||||
|
#include "gtest/gtest.h"
|
||||||
|
#include "third_party/zlib/zlib_crashpad.h"
|
||||||
|
#include "util/misc/zlib.h"
|
||||||
|
#include "util/net/http_body.h"
|
||||||
|
|
||||||
|
namespace crashpad {
|
||||||
|
namespace test {
|
||||||
|
namespace {
|
||||||
|
|
||||||
|
class ScopedZlibInflateStream {
|
||||||
|
public:
|
||||||
|
explicit ScopedZlibInflateStream(z_stream* zlib) : zlib_(zlib) {}
|
||||||
|
~ScopedZlibInflateStream() {
|
||||||
|
int zr = inflateEnd(zlib_);
|
||||||
|
EXPECT_EQ(Z_OK, zr) << "inflateEnd: " << ZlibErrorString(zr);
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
z_stream* zlib_; // weak
|
||||||
|
DISALLOW_COPY_AND_ASSIGN(ScopedZlibInflateStream);
|
||||||
|
};
|
||||||
|
|
||||||
|
void GzipInflate(const std::string& compressed,
|
||||||
|
std::string* decompressed,
|
||||||
|
size_t buf_size) {
|
||||||
|
decompressed->clear();
|
||||||
|
|
||||||
|
// There’s got to be at least a small buffer.
|
||||||
|
buf_size = std::max(buf_size, static_cast<size_t>(1));
|
||||||
|
|
||||||
|
std::unique_ptr<uint8_t[]> buf(new uint8_t[buf_size]);
|
||||||
|
z_stream zlib = {};
|
||||||
|
zlib.zalloc = Z_NULL;
|
||||||
|
zlib.zfree = Z_NULL;
|
||||||
|
zlib.opaque = Z_NULL;
|
||||||
|
zlib.next_in = reinterpret_cast<Bytef*>(const_cast<char*>(&compressed[0]));
|
||||||
|
zlib.avail_in = base::checked_cast<uInt>(compressed.size());
|
||||||
|
zlib.next_out = buf.get();
|
||||||
|
zlib.avail_out = base::checked_cast<uInt>(buf_size);
|
||||||
|
|
||||||
|
int zr = inflateInit2(&zlib, ZlibWindowBitsWithGzipWrapper(0));
|
||||||
|
ASSERT_EQ(Z_OK, zr) << "inflateInit2: " << ZlibErrorString(zr);
|
||||||
|
ScopedZlibInflateStream zlib_inflate(&zlib);
|
||||||
|
|
||||||
|
zr = inflate(&zlib, Z_FINISH);
|
||||||
|
ASSERT_EQ(Z_STREAM_END, zr) << "inflate: " << ZlibErrorString(zr);
|
||||||
|
|
||||||
|
ASSERT_LE(zlib.avail_out, buf_size);
|
||||||
|
decompressed->assign(reinterpret_cast<char*>(buf.get()),
|
||||||
|
buf_size - zlib.avail_out);
|
||||||
|
}
|
||||||
|
|
||||||
|
void TestGzipDeflateInflate(const std::string& string) {
|
||||||
|
std::unique_ptr<HTTPBodyStream> string_stream(
|
||||||
|
new StringHTTPBodyStream(string));
|
||||||
|
GzipHTTPBodyStream gzip_stream(std::move(string_stream));
|
||||||
|
|
||||||
|
// The minimum size of a gzip wrapper per RFC 1952: a 10-byte header and an
|
||||||
|
// 8-byte trailer.
|
||||||
|
const size_t kGzipHeaderSize = 18;
|
||||||
|
|
||||||
|
// Per http://www.zlib.net/zlib_tech.html, in the worst case, zlib will store
|
||||||
|
// uncompressed data as-is, at an overhead of 5 bytes per 16384-byte block.
|
||||||
|
// Zero-length input will “compress” to a 2-byte zlib stream. Add the overhead
|
||||||
|
// of the gzip wrapper, assuming no optional fields are present.
|
||||||
|
size_t buf_size =
|
||||||
|
string.size() + kGzipHeaderSize +
|
||||||
|
(string.empty() ? 2 : (((string.size() + 16383) / 16384) * 5));
|
||||||
|
std::unique_ptr<uint8_t[]> buf(new uint8_t[buf_size]);
|
||||||
|
FileOperationResult compressed_bytes =
|
||||||
|
gzip_stream.GetBytesBuffer(buf.get(), buf_size);
|
||||||
|
ASSERT_NE(compressed_bytes, -1);
|
||||||
|
ASSERT_LE(static_cast<size_t>(compressed_bytes), buf_size);
|
||||||
|
|
||||||
|
// Make sure that the stream is really at EOF.
|
||||||
|
uint8_t eof_buf[16];
|
||||||
|
ASSERT_EQ(0, gzip_stream.GetBytesBuffer(eof_buf, sizeof(eof_buf)));
|
||||||
|
|
||||||
|
std::string compressed(reinterpret_cast<char*>(buf.get()), compressed_bytes);
|
||||||
|
|
||||||
|
ASSERT_GE(compressed.size(), kGzipHeaderSize);
|
||||||
|
EXPECT_EQ('\37', compressed[0]);
|
||||||
|
EXPECT_EQ('\213', compressed[1]);
|
||||||
|
EXPECT_EQ(Z_DEFLATED, compressed[2]);
|
||||||
|
|
||||||
|
std::string decompressed;
|
||||||
|
ASSERT_NO_FATAL_FAILURE(
|
||||||
|
GzipInflate(compressed, &decompressed, string.size()));
|
||||||
|
|
||||||
|
EXPECT_EQ(string, decompressed);
|
||||||
|
|
||||||
|
// In block mode, compression should be identical.
|
||||||
|
string_stream.reset(new StringHTTPBodyStream(string));
|
||||||
|
GzipHTTPBodyStream block_gzip_stream(std::move(string_stream));
|
||||||
|
uint8_t block_buf[4096];
|
||||||
|
std::string block_compressed;
|
||||||
|
FileOperationResult block_compressed_bytes;
|
||||||
|
while ((block_compressed_bytes = block_gzip_stream.GetBytesBuffer(
|
||||||
|
block_buf, sizeof(block_buf))) > 0) {
|
||||||
|
block_compressed.append(reinterpret_cast<char*>(block_buf),
|
||||||
|
block_compressed_bytes);
|
||||||
|
}
|
||||||
|
ASSERT_EQ(0, block_compressed_bytes);
|
||||||
|
EXPECT_EQ(compressed, block_compressed);
|
||||||
|
}
|
||||||
|
|
||||||
|
std::string MakeString(size_t size) {
|
||||||
|
std::string string;
|
||||||
|
for (size_t i = 0; i < size; ++i) {
|
||||||
|
string.append(1, (i % 256) ^ ((i >> 8) % 256));
|
||||||
|
}
|
||||||
|
return string;
|
||||||
|
}
|
||||||
|
|
||||||
|
constexpr size_t kFourKBytes = 4096;
|
||||||
|
constexpr size_t kManyBytes = 375017;
|
||||||
|
|
||||||
|
TEST(GzipHTTPBodyStream, Empty) {
|
||||||
|
TestGzipDeflateInflate(std::string());
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(GzipHTTPBodyStream, OneByte) {
|
||||||
|
TestGzipDeflateInflate(std::string("Z"));
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(GzipHTTPBodyStream, FourKBytes_NUL) {
|
||||||
|
TestGzipDeflateInflate(std::string(kFourKBytes, '\0'));
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(GzipHTTPBodyStream, ManyBytes_NUL) {
|
||||||
|
TestGzipDeflateInflate(std::string(kManyBytes, '\0'));
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(GzipHTTPBodyStream, FourKBytes_Deterministic) {
|
||||||
|
TestGzipDeflateInflate(MakeString(kFourKBytes));
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(GzipHTTPBodyStream, ManyBytes_Deterministic) {
|
||||||
|
TestGzipDeflateInflate(MakeString(kManyBytes));
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(GzipHTTPBodyStream, FourKBytes_Random) {
|
||||||
|
TestGzipDeflateInflate(base::RandBytesAsString(kFourKBytes));
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(GzipHTTPBodyStream, ManyBytes_Random) {
|
||||||
|
TestGzipDeflateInflate(base::RandBytesAsString(kManyBytes));
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace
|
||||||
|
} // namespace test
|
||||||
|
} // namespace crashpad
|
@ -17,7 +17,7 @@
|
|||||||
namespace crashpad {
|
namespace crashpad {
|
||||||
|
|
||||||
const char kContentType[] = "Content-Type";
|
const char kContentType[] = "Content-Type";
|
||||||
|
|
||||||
const char kContentLength[] = "Content-Length";
|
const char kContentLength[] = "Content-Length";
|
||||||
|
const char kContentEncoding[] = "Content-Encoding";
|
||||||
|
|
||||||
} // namespace crashpad
|
} // namespace crashpad
|
||||||
|
@ -29,6 +29,9 @@ extern const char kContentType[];
|
|||||||
//! \brief The header name `"Content-Length"`.
|
//! \brief The header name `"Content-Length"`.
|
||||||
extern const char kContentLength[];
|
extern const char kContentLength[];
|
||||||
|
|
||||||
|
//! \brief The header name `"Content-Encoding"`.
|
||||||
|
extern const char kContentEncoding[];
|
||||||
|
|
||||||
} // namespace crashpad
|
} // namespace crashpad
|
||||||
|
|
||||||
#endif // CRASHPAD_UTIL_NET_HTTP_HEADERS_H_
|
#endif // CRASHPAD_UTIL_NET_HTTP_HEADERS_H_
|
||||||
|
@ -23,6 +23,7 @@
|
|||||||
#include "base/rand_util.h"
|
#include "base/rand_util.h"
|
||||||
#include "base/strings/stringprintf.h"
|
#include "base/strings/stringprintf.h"
|
||||||
#include "util/net/http_body.h"
|
#include "util/net/http_body.h"
|
||||||
|
#include "util/net/http_body_gzip.h"
|
||||||
|
|
||||||
namespace crashpad {
|
namespace crashpad {
|
||||||
|
|
||||||
@ -116,12 +117,18 @@ void AssertSafeMIMEType(const std::string& string) {
|
|||||||
} // namespace
|
} // namespace
|
||||||
|
|
||||||
HTTPMultipartBuilder::HTTPMultipartBuilder()
|
HTTPMultipartBuilder::HTTPMultipartBuilder()
|
||||||
: boundary_(GenerateBoundaryString()), form_data_(), file_attachments_() {
|
: boundary_(GenerateBoundaryString()),
|
||||||
}
|
form_data_(),
|
||||||
|
file_attachments_(),
|
||||||
|
gzip_enabled_(false) {}
|
||||||
|
|
||||||
HTTPMultipartBuilder::~HTTPMultipartBuilder() {
|
HTTPMultipartBuilder::~HTTPMultipartBuilder() {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void HTTPMultipartBuilder::SetGzipEnabled(bool gzip_enabled) {
|
||||||
|
gzip_enabled_ = gzip_enabled;
|
||||||
|
}
|
||||||
|
|
||||||
void HTTPMultipartBuilder::SetFormData(const std::string& key,
|
void HTTPMultipartBuilder::SetFormData(const std::string& key,
|
||||||
const std::string& value) {
|
const std::string& value) {
|
||||||
EraseKey(key);
|
EraseKey(key);
|
||||||
@ -179,13 +186,24 @@ std::unique_ptr<HTTPBodyStream> HTTPMultipartBuilder::GetBodyStream() {
|
|||||||
streams.push_back(
|
streams.push_back(
|
||||||
new StringHTTPBodyStream("--" + boundary_ + "--" + kCRLF));
|
new StringHTTPBodyStream("--" + boundary_ + "--" + kCRLF));
|
||||||
|
|
||||||
return std::unique_ptr<HTTPBodyStream>(new CompositeHTTPBodyStream(streams));
|
auto composite =
|
||||||
|
std::unique_ptr<HTTPBodyStream>(new CompositeHTTPBodyStream(streams));
|
||||||
|
if (gzip_enabled_) {
|
||||||
|
return std::unique_ptr<HTTPBodyStream>(
|
||||||
|
new GzipHTTPBodyStream(std::move(composite)));
|
||||||
|
}
|
||||||
|
return composite;
|
||||||
}
|
}
|
||||||
|
|
||||||
HTTPHeaders::value_type HTTPMultipartBuilder::GetContentType() const {
|
void HTTPMultipartBuilder::PopulateContentHeaders(
|
||||||
|
HTTPHeaders* http_headers) const {
|
||||||
std::string content_type =
|
std::string content_type =
|
||||||
base::StringPrintf("multipart/form-data; boundary=%s", boundary_.c_str());
|
base::StringPrintf("multipart/form-data; boundary=%s", boundary_.c_str());
|
||||||
return std::make_pair(kContentType, content_type);
|
(*http_headers)[kContentType] = content_type;
|
||||||
|
|
||||||
|
if (gzip_enabled_) {
|
||||||
|
(*http_headers)[kContentEncoding] = "gzip";
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void HTTPMultipartBuilder::EraseKey(const std::string& key) {
|
void HTTPMultipartBuilder::EraseKey(const std::string& key) {
|
||||||
|
@ -34,6 +34,15 @@ class HTTPMultipartBuilder {
|
|||||||
HTTPMultipartBuilder();
|
HTTPMultipartBuilder();
|
||||||
~HTTPMultipartBuilder();
|
~HTTPMultipartBuilder();
|
||||||
|
|
||||||
|
//! \brief Enables or disables `gzip` compression.
|
||||||
|
//!
|
||||||
|
//! \param[in] gzip_enabled Whether to enable or disable `gzip` compression.
|
||||||
|
//!
|
||||||
|
//! When `gzip` compression is enabled, the body stream returned by
|
||||||
|
//! GetBodyStream() will be `gzip`-compressed, and the content headers set by
|
||||||
|
//! PopulateContentHeaders() will contain `Content-Encoding: gzip`.
|
||||||
|
void SetGzipEnabled(bool gzip_enabled);
|
||||||
|
|
||||||
//! \brief Sets a `Content-Disposition: form-data` key-value pair.
|
//! \brief Sets a `Content-Disposition: form-data` key-value pair.
|
||||||
//!
|
//!
|
||||||
//! \param[in] key The key of the form data, specified as the `name` in the
|
//! \param[in] key The key of the form data, specified as the `name` in the
|
||||||
@ -64,8 +73,11 @@ class HTTPMultipartBuilder {
|
|||||||
//! \return A caller-owned HTTPBodyStream object.
|
//! \return A caller-owned HTTPBodyStream object.
|
||||||
std::unique_ptr<HTTPBodyStream> GetBodyStream();
|
std::unique_ptr<HTTPBodyStream> GetBodyStream();
|
||||||
|
|
||||||
//! \brief Gets the header pair for `"Content-Type"`.
|
//! \brief Adds the appropriate content headers to \a http_headers.
|
||||||
HTTPHeaders::value_type GetContentType() const;
|
//!
|
||||||
|
//! Any headers that this method adds will replace existing headers by the
|
||||||
|
//! same name in \a http_headers.
|
||||||
|
void PopulateContentHeaders(HTTPHeaders* http_headers) const;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
struct FileAttachment {
|
struct FileAttachment {
|
||||||
@ -81,6 +93,7 @@ class HTTPMultipartBuilder {
|
|||||||
std::string boundary_;
|
std::string boundary_;
|
||||||
std::map<std::string, std::string> form_data_;
|
std::map<std::string, std::string> form_data_;
|
||||||
std::map<std::string, FileAttachment> file_attachments_;
|
std::map<std::string, FileAttachment> file_attachments_;
|
||||||
|
bool gzip_enabled_;
|
||||||
|
|
||||||
DISALLOW_COPY_AND_ASSIGN(HTTPMultipartBuilder);
|
DISALLOW_COPY_AND_ASSIGN(HTTPMultipartBuilder);
|
||||||
};
|
};
|
||||||
|
@ -71,6 +71,7 @@ TEST(HTTPMultipartBuilder, ThreeStringFields) {
|
|||||||
ASSERT_TRUE(body.get());
|
ASSERT_TRUE(body.get());
|
||||||
std::string contents = ReadStreamToString(body.get());
|
std::string contents = ReadStreamToString(body.get());
|
||||||
auto lines = SplitCRLF(contents);
|
auto lines = SplitCRLF(contents);
|
||||||
|
ASSERT_EQ(13u, lines.size());
|
||||||
auto lines_it = lines.begin();
|
auto lines_it = lines.begin();
|
||||||
|
|
||||||
// The first line is the boundary. All subsequent boundaries must match this.
|
// The first line is the boundary. All subsequent boundaries must match this.
|
||||||
@ -164,6 +165,7 @@ TEST(HTTPMultipartBuilder, OverwriteFormDataWithEscapedKey) {
|
|||||||
ASSERT_TRUE(body.get());
|
ASSERT_TRUE(body.get());
|
||||||
std::string contents = ReadStreamToString(body.get());
|
std::string contents = ReadStreamToString(body.get());
|
||||||
auto lines = SplitCRLF(contents);
|
auto lines = SplitCRLF(contents);
|
||||||
|
ASSERT_EQ(5u, lines.size());
|
||||||
auto lines_it = lines.begin();
|
auto lines_it = lines.begin();
|
||||||
|
|
||||||
const std::string& boundary = *lines_it++;
|
const std::string& boundary = *lines_it++;
|
||||||
@ -253,6 +255,7 @@ TEST(HTTPMultipartBuilder, SharedFormDataAndAttachmentKeyNamespace) {
|
|||||||
ASSERT_TRUE(body.get());
|
ASSERT_TRUE(body.get());
|
||||||
std::string contents = ReadStreamToString(body.get());
|
std::string contents = ReadStreamToString(body.get());
|
||||||
auto lines = SplitCRLF(contents);
|
auto lines = SplitCRLF(contents);
|
||||||
|
ASSERT_EQ(9u, lines.size());
|
||||||
auto lines_it = lines.begin();
|
auto lines_it = lines.begin();
|
||||||
|
|
||||||
const std::string& boundary = *lines_it++;
|
const std::string& boundary = *lines_it++;
|
||||||
|
@ -221,7 +221,21 @@ TEST(HTTPTransport, ValidFormData) {
|
|||||||
builder.SetFormData("key2", "--abcdefg123");
|
builder.SetFormData("key2", "--abcdefg123");
|
||||||
|
|
||||||
HTTPHeaders headers;
|
HTTPHeaders headers;
|
||||||
EXPECT_TRUE(headers.insert(builder.GetContentType()).second);
|
builder.PopulateContentHeaders(&headers);
|
||||||
|
|
||||||
|
HTTPTransportTestFixture test(
|
||||||
|
headers, builder.GetBodyStream(), 200, &ValidFormData);
|
||||||
|
test.Run();
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST(HTTPTransport, ValidFormData_Gzip) {
|
||||||
|
HTTPMultipartBuilder builder;
|
||||||
|
builder.SetGzipEnabled(true);
|
||||||
|
builder.SetFormData("key1", "test");
|
||||||
|
builder.SetFormData("key2", "--abcdefg123");
|
||||||
|
|
||||||
|
HTTPHeaders headers;
|
||||||
|
builder.PopulateContentHeaders(&headers);
|
||||||
|
|
||||||
HTTPTransportTestFixture test(headers, builder.GetBodyStream(), 200,
|
HTTPTransportTestFixture test(headers, builder.GetBodyStream(), 200,
|
||||||
&ValidFormData);
|
&ValidFormData);
|
||||||
|
@ -33,6 +33,7 @@ This could easily have been written in C++ instead.
|
|||||||
import BaseHTTPServer
|
import BaseHTTPServer
|
||||||
import struct
|
import struct
|
||||||
import sys
|
import sys
|
||||||
|
import zlib
|
||||||
|
|
||||||
class BufferedReadFile(object):
|
class BufferedReadFile(object):
|
||||||
"""A File-like object that stores all read contents into a buffer."""
|
"""A File-like object that stores all read contents into a buffer."""
|
||||||
@ -88,6 +89,13 @@ class RequestHandler(BaseHTTPServer.BaseHTTPRequestHandler):
|
|||||||
length = int(self.headers.get('Content-Length', -1))
|
length = int(self.headers.get('Content-Length', -1))
|
||||||
body = self.rfile.read(length)
|
body = self.rfile.read(length)
|
||||||
|
|
||||||
|
if self.headers.get('Content-Encoding', '').lower() == 'gzip':
|
||||||
|
# 15 is the value of |wbits|, which should be at the maximum possible
|
||||||
|
# value to ensure that any gzip stream can be decoded. The offset of 16
|
||||||
|
# specifies that the stream to decompress will be formatted with a gzip
|
||||||
|
# wrapper.
|
||||||
|
body = zlib.decompress(body, 16 + 15)
|
||||||
|
|
||||||
RequestHandler.raw_request += body
|
RequestHandler.raw_request += body
|
||||||
|
|
||||||
self.send_response(self.response_code)
|
self.send_response(self.response_code)
|
||||||
|
@ -23,6 +23,7 @@
|
|||||||
'dependencies': [
|
'dependencies': [
|
||||||
'../compat/compat.gyp:crashpad_compat',
|
'../compat/compat.gyp:crashpad_compat',
|
||||||
'../third_party/mini_chromium/mini_chromium.gyp:base',
|
'../third_party/mini_chromium/mini_chromium.gyp:base',
|
||||||
|
'../third_party/zlib/zlib.gyp:zlib',
|
||||||
],
|
],
|
||||||
'include_dirs': [
|
'include_dirs': [
|
||||||
'..',
|
'..',
|
||||||
@ -106,8 +107,12 @@
|
|||||||
'misc/tri_state.h',
|
'misc/tri_state.h',
|
||||||
'misc/uuid.cc',
|
'misc/uuid.cc',
|
||||||
'misc/uuid.h',
|
'misc/uuid.h',
|
||||||
|
'misc/zlib.cc',
|
||||||
|
'misc/zlib.h',
|
||||||
'net/http_body.cc',
|
'net/http_body.cc',
|
||||||
'net/http_body.h',
|
'net/http_body.h',
|
||||||
|
'net/http_body_gzip.cc',
|
||||||
|
'net/http_body_gzip.h',
|
||||||
'net/http_headers.cc',
|
'net/http_headers.cc',
|
||||||
'net/http_headers.h',
|
'net/http_headers.h',
|
||||||
'net/http_multipart_builder.cc',
|
'net/http_multipart_builder.cc',
|
||||||
|
@ -29,6 +29,7 @@
|
|||||||
'../third_party/gtest/gmock.gyp:gmock_main',
|
'../third_party/gtest/gmock.gyp:gmock_main',
|
||||||
'../third_party/gtest/gtest.gyp:gtest',
|
'../third_party/gtest/gtest.gyp:gtest',
|
||||||
'../third_party/mini_chromium/mini_chromium.gyp:base',
|
'../third_party/mini_chromium/mini_chromium.gyp:base',
|
||||||
|
'../third_party/zlib/zlib.gyp:zlib',
|
||||||
],
|
],
|
||||||
'include_dirs': [
|
'include_dirs': [
|
||||||
'..',
|
'..',
|
||||||
@ -62,6 +63,7 @@
|
|||||||
'misc/scoped_forbid_return_test.cc',
|
'misc/scoped_forbid_return_test.cc',
|
||||||
'misc/random_string_test.cc',
|
'misc/random_string_test.cc',
|
||||||
'misc/uuid_test.cc',
|
'misc/uuid_test.cc',
|
||||||
|
'net/http_body_gzip_test.cc',
|
||||||
'net/http_body_test.cc',
|
'net/http_body_test.cc',
|
||||||
'net/http_body_test_util.cc',
|
'net/http_body_test_util.cc',
|
||||||
'net/http_body_test_util.h',
|
'net/http_body_test_util.h',
|
||||||
|
Loading…
x
Reference in New Issue
Block a user