OLD | NEW |
---|---|
(Empty) | |
1 // Copyright 2015 The Crashpad Authors. All rights reserved. | |
2 // | |
3 // Licensed under the Apache License, Version 2.0 (the "License"); | |
4 // you may not use this file except in compliance with the License. | |
5 // You may obtain a copy of the License at | |
6 // | |
7 // http://www.apache.org/licenses/LICENSE-2.0 | |
8 // | |
9 // Unless required by applicable law or agreed to in writing, software | |
10 // distributed under the License is distributed on an "AS IS" BASIS, | |
11 // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | |
12 // See the License for the specific language governing permissions and | |
13 // limitations under the License. | |
14 | |
15 #include "util/net/http_transport.h" | |
16 | |
17 #include <windows.h> | |
18 #include <winhttp.h> | |
19 | |
20 #include "base/logging.h" | |
21 #include "base/scoped_generic.h" | |
22 #include "base/strings/stringprintf.h" | |
23 #include "base/strings/utf_string_conversions.h" | |
24 #include "util/net/http_body.h" | |
25 #include "package.h" | |
26 | |
27 namespace crashpad { | |
28 | |
29 namespace { | |
30 | |
31 // PLOG doesn't work for messages from WinHTTP, so we need to use | |
32 // FORMAT_MESSAGE_FROM_HMODULE + the dll name manually here. | |
33 void LogErrorWinHttpMessage(const char* extra) { | |
34 DWORD error_code = GetLastError(); | |
35 char msgbuf[256]; | |
36 DWORD flags = FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_IGNORE_INSERTS | | |
37 FORMAT_MESSAGE_MAX_WIDTH_MASK | FORMAT_MESSAGE_FROM_HMODULE; | |
38 DWORD len = FormatMessageA(flags, | |
39 GetModuleHandle(L"winhttp.dll"), | |
40 error_code, | |
41 0, | |
42 msgbuf, | |
43 arraysize(msgbuf), | |
44 NULL); | |
45 if (len) { | |
46 LOG(ERROR) << extra << ": " << msgbuf | |
47 << base::StringPrintf(" (0x%X)", error_code); | |
48 } else { | |
49 LOG(ERROR) << base::StringPrintf( | |
50 "Error (0x%X) while retrieving error. (0x%X)", | |
51 GetLastError(), | |
52 error_code); | |
53 } | |
54 } | |
55 | |
56 struct ScopedHINTERNETTraits { | |
57 static HINTERNET InvalidValue() { | |
58 return nullptr; | |
59 } | |
60 static void Free(HINTERNET handle) { | |
61 if (handle) { | |
62 if (!WinHttpCloseHandle(handle)) { | |
63 LogErrorWinHttpMessage("WinHttpCloseHandle"); | |
64 } | |
65 } | |
66 } | |
67 }; | |
68 | |
69 using ScopedHINTERNET = base::ScopedGeneric<HINTERNET, ScopedHINTERNETTraits>; | |
70 | |
71 class HTTPTransportWin final : public HTTPTransport { | |
72 public: | |
73 HTTPTransportWin(); | |
74 ~HTTPTransportWin() override; | |
75 | |
76 bool ExecuteSynchronously() override; | |
77 | |
78 private: | |
79 DISALLOW_COPY_AND_ASSIGN(HTTPTransportWin); | |
80 }; | |
81 | |
82 HTTPTransportWin::HTTPTransportWin() : HTTPTransport() { | |
83 } | |
84 | |
85 HTTPTransportWin::~HTTPTransportWin() { | |
86 } | |
87 | |
88 bool HTTPTransportWin::ExecuteSynchronously() { | |
89 ScopedHINTERNET session( | |
90 WinHttpOpen(base::UTF8ToUTF16(PACKAGE_NAME "/" PACKAGE_VERSION).c_str(), | |
91 WINHTTP_ACCESS_TYPE_DEFAULT_PROXY, | |
92 WINHTTP_NO_PROXY_NAME, | |
93 WINHTTP_NO_PROXY_BYPASS, | |
94 0)); | |
95 if (!session.get()) { | |
96 LogErrorWinHttpMessage("WinHttpOpen"); | |
97 return false; | |
98 } | |
99 | |
100 URL_COMPONENTS url_components = {0}; | |
101 url_components.dwStructSize = sizeof(URL_COMPONENTS); | |
102 url_components.dwHostNameLength = 1; | |
103 url_components.dwUrlPathLength = 1; | |
104 url_components.dwExtraInfoLength = 1; | |
105 std::wstring url_wide(base::UTF8ToUTF16(url())); | |
106 if (!WinHttpCrackUrl( | |
107 url_wide.c_str(), 0, ICU_REJECT_USERPWD, &url_components)) { | |
108 LogErrorWinHttpMessage("WinHttpCrackUrl"); | |
109 return false; | |
110 } | |
111 std::wstring host_name(url_components.lpszHostName, | |
112 url_components.dwHostNameLength); | |
113 std::wstring url_path(url_components.lpszUrlPath, | |
114 url_components.dwUrlPathLength); | |
115 std::wstring extra_info(url_components.lpszExtraInfo, | |
116 url_components.dwExtraInfoLength); | |
117 | |
118 ScopedHINTERNET connect(WinHttpConnect( | |
119 session.get(), host_name.c_str(), url_components.nPort, 0)); | |
120 if (!connect.get()) { | |
121 LogErrorWinHttpMessage("WinHttpConnect"); | |
122 return false; | |
123 } | |
124 | |
125 ScopedHINTERNET request( | |
126 WinHttpOpenRequest(connect.get(), | |
127 base::UTF8ToUTF16(method()).c_str(), | |
128 url_path.c_str(), | |
129 nullptr, | |
130 WINHTTP_NO_REFERER, | |
131 WINHTTP_DEFAULT_ACCEPT_TYPES, | |
132 0)); | |
133 if (!request.get()) { | |
134 LogErrorWinHttpMessage("WinHttpOpenRequest"); | |
135 return false; | |
136 } | |
137 | |
138 // Add headers to the request. | |
139 for (const auto& pair : headers()) { | |
140 std::wstring header_string = | |
141 base::UTF8ToUTF16(pair.first) + L": " + base::UTF8ToUTF16(pair.second); | |
142 if (!WinHttpAddRequestHeaders(request.get(), | |
143 header_string.c_str(), | |
144 header_string.size(), | |
145 WINHTTP_ADDREQ_FLAG_ADD)) { | |
146 LogErrorWinHttpMessage("WinHttpAddRequestHeaders"); | |
147 return false; | |
148 } | |
149 } | |
150 | |
151 // We need the Content-Length up front, so buffer in memory. We should modify | |
Robert Sesek
2015/01/26 21:17:09
This would be helpful on OS X too, and then we can
scottmg
2015/01/26 21:22:34
OK, I'll take a look at doing that as a follow up.
| |
152 // the interface to not require this, and then use WinHttpWriteData after | |
153 // WinHttpSendRequest. | |
154 std::vector<uint8_t> post_data; | |
155 | |
156 // Write the body of a POST if any. | |
157 for (;;) { | |
158 uint8_t buffer[4096]; | |
159 ssize_t bytes_to_write = | |
160 body_stream()->GetBytesBuffer(buffer, sizeof(buffer)); | |
161 if (bytes_to_write == 0) | |
162 break; | |
163 post_data.insert(post_data.end(), buffer, buffer + bytes_to_write); | |
164 } | |
165 | |
166 if (!WinHttpSendRequest(request.get(), | |
167 WINHTTP_NO_ADDITIONAL_HEADERS, | |
168 0, | |
169 &post_data[0], | |
170 post_data.size(), | |
171 post_data.size(), | |
172 0)) { | |
173 LogErrorWinHttpMessage("WinHttpSendRequest"); | |
174 return false; | |
175 } | |
176 | |
177 if (!WinHttpReceiveResponse(request.get(), nullptr)) { | |
178 LogErrorWinHttpMessage("WinHttpReceiveResponse"); | |
179 return false; | |
180 } | |
181 | |
182 DWORD status_code = 0; | |
183 DWORD sizeof_status_code = sizeof(status_code); | |
184 | |
185 if (!WinHttpQueryHeaders( | |
186 request.get(), | |
187 WINHTTP_QUERY_STATUS_CODE | WINHTTP_QUERY_FLAG_NUMBER, | |
188 WINHTTP_HEADER_NAME_BY_INDEX, | |
189 &status_code, | |
190 &sizeof_status_code, | |
191 WINHTTP_NO_HEADER_INDEX)) { | |
192 LogErrorWinHttpMessage("WinHttpQueryHeaders"); | |
193 return false; | |
194 } | |
195 | |
196 if (status_code != 200) { | |
197 LOG(ERROR) << base::StringPrintf("HTTP status %d", status_code); | |
198 return false; | |
199 } | |
200 | |
201 // TODO(scottmg): Retrieve body of response if necessary with | |
202 // WinHttpQueryDataAvailable and WinHttpReadData. | |
203 | |
204 return true; | |
205 } | |
206 | |
207 } // namespace | |
208 | |
209 // static | |
210 scoped_ptr<HTTPTransport> HTTPTransport::Create() { | |
211 return scoped_ptr<HTTPTransportWin>(new HTTPTransportWin); | |
212 } | |
213 | |
214 } // namespace crashpad | |
OLD | NEW |