1 // Copyright (c) 2009, Google Inc.
2 // All rights reserved.
3 //
4 // Redistribution and use in source and binary forms, with or without
5 // modification, are permitted provided that the following conditions are
6 // met:
7 //
8 // * Redistributions of source code must retain the above copyright
9 // notice, this list of conditions and the following disclaimer.
10 // * Redistributions in binary form must reproduce the above
11 // copyright notice, this list of conditions and the following disclaimer
12 // in the documentation and/or other materials provided with the
13 // distribution.
14 // * Neither the name of Google Inc. nor the names of its
15 // contributors may be used to endorse or promote products derived from
16 // this software without specific prior written permission.
17 //
18 // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
19 // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
20 // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
21 // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
22 // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
23 // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
24 // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
25 // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
26 // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
27 // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
28 // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
29
30 #include <dlfcn.h>
31
32 #include <iostream>
33 #include <string>
34
35 #include "common/linux/libcurl_wrapper.h"
36 #include "common/using_std_string.h"
37
38 namespace google_breakpad {
LibcurlWrapper()39 LibcurlWrapper::LibcurlWrapper()
40 : init_ok_(false),
41 curl_lib_(nullptr),
42 last_curl_error_(""),
43 curl_(nullptr),
44 formpost_(nullptr),
45 lastptr_(nullptr),
46 headerlist_(nullptr) {}
47
~LibcurlWrapper()48 LibcurlWrapper::~LibcurlWrapper() {
49 if (init_ok_) {
50 (*easy_cleanup_)(curl_);
51 dlclose(curl_lib_);
52 }
53 }
54
SetProxy(const string & proxy_host,const string & proxy_userpwd)55 bool LibcurlWrapper::SetProxy(const string& proxy_host,
56 const string& proxy_userpwd) {
57 if (!CheckInit()) return false;
58
59 // Set proxy information if necessary.
60 if (!proxy_host.empty()) {
61 (*easy_setopt_)(curl_, CURLOPT_PROXY, proxy_host.c_str());
62 } else {
63 std::cout << "SetProxy called with empty proxy host.";
64 return false;
65 }
66 if (!proxy_userpwd.empty()) {
67 (*easy_setopt_)(curl_, CURLOPT_PROXYUSERPWD, proxy_userpwd.c_str());
68 } else {
69 std::cout << "SetProxy called with empty proxy username/password.";
70 return false;
71 }
72 std::cout << "Set proxy host to " << proxy_host;
73 return true;
74 }
75
AddFile(const string & upload_file_path,const string & basename)76 bool LibcurlWrapper::AddFile(const string& upload_file_path,
77 const string& basename) {
78 if (!CheckInit()) return false;
79
80 std::cout << "Adding " << upload_file_path << " to form upload.";
81 // Add form file.
82 (*formadd_)(&formpost_, &lastptr_,
83 CURLFORM_COPYNAME, basename.c_str(),
84 CURLFORM_FILE, upload_file_path.c_str(),
85 CURLFORM_END);
86
87 return true;
88 }
89
90 // Callback to get the response data from server.
WriteCallback(void * ptr,size_t size,size_t nmemb,void * userp)91 static size_t WriteCallback(void *ptr, size_t size,
92 size_t nmemb, void *userp) {
93 if (!userp)
94 return 0;
95
96 string *response = reinterpret_cast<string *>(userp);
97 size_t real_size = size * nmemb;
98 response->append(reinterpret_cast<char *>(ptr), real_size);
99 return real_size;
100 }
101
SendRequest(const string & url,const std::map<string,string> & parameters,long * http_status_code,string * http_header_data,string * http_response_data)102 bool LibcurlWrapper::SendRequest(const string& url,
103 const std::map<string, string>& parameters,
104 long* http_status_code,
105 string* http_header_data,
106 string* http_response_data) {
107 if (!CheckInit()) return false;
108
109 std::map<string, string>::const_iterator iter = parameters.begin();
110 for (; iter != parameters.end(); ++iter)
111 (*formadd_)(&formpost_, &lastptr_,
112 CURLFORM_COPYNAME, iter->first.c_str(),
113 CURLFORM_COPYCONTENTS, iter->second.c_str(),
114 CURLFORM_END);
115
116 (*easy_setopt_)(curl_, CURLOPT_HTTPPOST, formpost_);
117
118 return SendRequestInner(url, http_status_code, http_header_data,
119 http_response_data);
120 }
121
SendGetRequest(const string & url,long * http_status_code,string * http_header_data,string * http_response_data)122 bool LibcurlWrapper::SendGetRequest(const string& url,
123 long* http_status_code,
124 string* http_header_data,
125 string* http_response_data) {
126 if (!CheckInit()) return false;
127
128 (*easy_setopt_)(curl_, CURLOPT_HTTPGET, 1L);
129
130 return SendRequestInner(url, http_status_code, http_header_data,
131 http_response_data);
132 }
133
SendPutRequest(const string & url,const string & path,long * http_status_code,string * http_header_data,string * http_response_data)134 bool LibcurlWrapper::SendPutRequest(const string& url,
135 const string& path,
136 long* http_status_code,
137 string* http_header_data,
138 string* http_response_data) {
139 if (!CheckInit()) return false;
140
141 FILE* file = fopen(path.c_str(), "rb");
142 (*easy_setopt_)(curl_, CURLOPT_UPLOAD, 1L);
143 (*easy_setopt_)(curl_, CURLOPT_PUT, 1L);
144 (*easy_setopt_)(curl_, CURLOPT_READDATA, file);
145
146 bool success = SendRequestInner(url, http_status_code, http_header_data,
147 http_response_data);
148
149 fclose(file);
150 return success;
151 }
152
SendSimplePostRequest(const string & url,const string & body,const string & content_type,long * http_status_code,string * http_header_data,string * http_response_data)153 bool LibcurlWrapper::SendSimplePostRequest(const string& url,
154 const string& body,
155 const string& content_type,
156 long* http_status_code,
157 string* http_header_data,
158 string* http_response_data) {
159 if (!CheckInit()) return false;
160
161 (*easy_setopt_)(curl_, CURLOPT_POSTFIELDSIZE, body.size());
162 (*easy_setopt_)(curl_, CURLOPT_COPYPOSTFIELDS, body.c_str());
163
164 if (!content_type.empty()) {
165 string content_type_header = "Content-Type: " + content_type;
166 headerlist_ = (*slist_append_)(
167 headerlist_,
168 content_type_header.c_str());
169 }
170
171 return SendRequestInner(url, http_status_code, http_header_data,
172 http_response_data);
173 }
174
Init()175 bool LibcurlWrapper::Init() {
176 // First check to see if libcurl was statically linked:
177 curl_lib_ = dlopen(nullptr, RTLD_NOW);
178 if (curl_lib_ &&
179 (!dlsym(curl_lib_, "curl_easy_init") ||
180 !dlsym(curl_lib_, "curl_easy_setopt"))) {
181 // Not statically linked, try again below.
182 dlerror(); // Clear dlerror before attempting to open libraries.
183 dlclose(curl_lib_);
184 curl_lib_ = nullptr;
185 }
186 if (!curl_lib_) {
187 curl_lib_ = dlopen("libcurl.so", RTLD_NOW);
188 }
189 if (!curl_lib_) {
190 curl_lib_ = dlopen("libcurl.so.4", RTLD_NOW);
191 }
192 if (!curl_lib_) {
193 curl_lib_ = dlopen("libcurl.so.3", RTLD_NOW);
194 }
195 if (!curl_lib_) {
196 std::cout << "Could not find libcurl via dlopen";
197 return false;
198 }
199
200 if (!SetFunctionPointers()) {
201 std::cout << "Could not find function pointers";
202 return false;
203 }
204
205 curl_ = (*easy_init_)();
206
207 last_curl_error_ = "No Error";
208
209 if (!curl_) {
210 dlclose(curl_lib_);
211 std::cout << "Curl initialization failed";
212 return false;
213 }
214
215 init_ok_ = true;
216 return true;
217 }
218
219 #define SET_AND_CHECK_FUNCTION_POINTER(var, function_name, type) \
220 var = reinterpret_cast<type>(dlsym(curl_lib_, function_name)); \
221 if (!var) { \
222 std::cout << "Could not find libcurl function " << function_name; \
223 init_ok_ = false; \
224 return false; \
225 }
226
SetFunctionPointers()227 bool LibcurlWrapper::SetFunctionPointers() {
228
229 SET_AND_CHECK_FUNCTION_POINTER(easy_init_,
230 "curl_easy_init",
231 CURL*(*)());
232
233 SET_AND_CHECK_FUNCTION_POINTER(easy_setopt_,
234 "curl_easy_setopt",
235 CURLcode(*)(CURL*, CURLoption, ...));
236
237 SET_AND_CHECK_FUNCTION_POINTER(formadd_, "curl_formadd",
238 CURLFORMcode(*)(curl_httppost**, curl_httppost**, ...));
239
240 SET_AND_CHECK_FUNCTION_POINTER(slist_append_, "curl_slist_append",
241 curl_slist*(*)(curl_slist*, const char*));
242
243 SET_AND_CHECK_FUNCTION_POINTER(easy_perform_,
244 "curl_easy_perform",
245 CURLcode(*)(CURL*));
246
247 SET_AND_CHECK_FUNCTION_POINTER(easy_cleanup_,
248 "curl_easy_cleanup",
249 void(*)(CURL*));
250
251 SET_AND_CHECK_FUNCTION_POINTER(easy_getinfo_,
252 "curl_easy_getinfo",
253 CURLcode(*)(CURL *, CURLINFO info, ...));
254
255 SET_AND_CHECK_FUNCTION_POINTER(easy_reset_,
256 "curl_easy_reset",
257 void(*)(CURL*));
258
259 SET_AND_CHECK_FUNCTION_POINTER(slist_free_all_,
260 "curl_slist_free_all",
261 void(*)(curl_slist*));
262
263 SET_AND_CHECK_FUNCTION_POINTER(formfree_,
264 "curl_formfree",
265 void(*)(curl_httppost*));
266 return true;
267 }
268
SendRequestInner(const string & url,long * http_status_code,string * http_header_data,string * http_response_data)269 bool LibcurlWrapper::SendRequestInner(const string& url,
270 long* http_status_code,
271 string* http_header_data,
272 string* http_response_data) {
273 string url_copy(url);
274 (*easy_setopt_)(curl_, CURLOPT_URL, url_copy.c_str());
275
276 // Disable 100-continue header.
277 char buf[] = "Expect:";
278 headerlist_ = (*slist_append_)(headerlist_, buf);
279 (*easy_setopt_)(curl_, CURLOPT_HTTPHEADER, headerlist_);
280
281 if (http_response_data != nullptr) {
282 http_response_data->clear();
283 (*easy_setopt_)(curl_, CURLOPT_WRITEFUNCTION, WriteCallback);
284 (*easy_setopt_)(curl_, CURLOPT_WRITEDATA,
285 reinterpret_cast<void*>(http_response_data));
286 }
287 if (http_header_data != nullptr) {
288 http_header_data->clear();
289 (*easy_setopt_)(curl_, CURLOPT_HEADERFUNCTION, WriteCallback);
290 (*easy_setopt_)(curl_, CURLOPT_HEADERDATA,
291 reinterpret_cast<void*>(http_header_data));
292 }
293 CURLcode err_code = CURLE_OK;
294 err_code = (*easy_perform_)(curl_);
295 easy_strerror_ = reinterpret_cast<const char* (*)(CURLcode)>
296 (dlsym(curl_lib_, "curl_easy_strerror"));
297
298 if (http_status_code != nullptr) {
299 (*easy_getinfo_)(curl_, CURLINFO_RESPONSE_CODE, http_status_code);
300 }
301
302 #ifndef NDEBUG
303 if (err_code != CURLE_OK)
304 fprintf(stderr, "Failed to send http request to %s, error: %s\n",
305 url.c_str(),
306 (*easy_strerror_)(err_code));
307 #endif
308
309 Reset();
310
311 return err_code == CURLE_OK;
312 }
313
Reset()314 void LibcurlWrapper::Reset() {
315 if (headerlist_ != nullptr) {
316 (*slist_free_all_)(headerlist_);
317 headerlist_ = nullptr;
318 }
319
320 if (formpost_ != nullptr) {
321 (*formfree_)(formpost_);
322 formpost_ = nullptr;
323 }
324
325 (*easy_reset_)(curl_);
326 }
327
CheckInit()328 bool LibcurlWrapper::CheckInit() {
329 if (!init_ok_) {
330 std::cout << "LibcurlWrapper: You must call Init(), and have it return "
331 "'true' before invoking any other methods.\n";
332 return false;
333 }
334
335 return true;
336 }
337
338 } // namespace google_breakpad
339