mirror of
https://github.com/typesense/typesense.git
synced 2025-05-21 14:12:27 +08:00
138 lines
4.6 KiB
C++
138 lines
4.6 KiB
C++
#include "http_client.h"
|
|
#include "file_utils.h"
|
|
#include "logger.h"
|
|
#include <vector>
|
|
#include <json.hpp>
|
|
|
|
std::string HttpClient::api_key = "";
|
|
std::string HttpClient::ca_cert_path = "";
|
|
|
|
long HttpClient::post_response(const std::string &url, const std::string &body, std::string &response,
|
|
std::map<std::string, std::string>& res_headers, long timeout_ms) {
|
|
CURL *curl = init_curl(url, response);
|
|
if(curl == nullptr) {
|
|
return 500;
|
|
}
|
|
|
|
curl_easy_setopt(curl, CURLOPT_POSTFIELDS, body.c_str());
|
|
return perform_curl(curl, res_headers);
|
|
}
|
|
|
|
long HttpClient::put_response(const std::string &url, const std::string &body, std::string &response,
|
|
std::map<std::string, std::string>& res_headers, long timeout_ms) {
|
|
CURL *curl = init_curl(url, response);
|
|
if(curl == nullptr) {
|
|
return 500;
|
|
}
|
|
|
|
curl_easy_setopt(curl, CURLOPT_CUSTOMREQUEST, "PUT");
|
|
curl_easy_setopt(curl, CURLOPT_POSTFIELDS, body.c_str());
|
|
return perform_curl(curl, res_headers);
|
|
}
|
|
|
|
long HttpClient::delete_response(const std::string &url, std::string &response,
|
|
std::map<std::string, std::string>& res_headers, long timeout_ms) {
|
|
CURL *curl = init_curl(url, response);
|
|
if(curl == nullptr) {
|
|
return 500;
|
|
}
|
|
|
|
curl_easy_setopt(curl, CURLOPT_CUSTOMREQUEST, "DELETE");
|
|
return perform_curl(curl, res_headers);
|
|
}
|
|
|
|
long HttpClient::get_response(const std::string &url, std::string &response,
|
|
std::map<std::string, std::string>& res_headers, long timeout_ms) {
|
|
CURL *curl = init_curl(url, response);
|
|
if(curl == nullptr) {
|
|
return 500;
|
|
}
|
|
|
|
return perform_curl(curl, res_headers);
|
|
}
|
|
|
|
void HttpClient::init(const std::string &api_key) {
|
|
HttpClient::api_key = api_key;
|
|
|
|
// try to locate ca cert file (from: https://serverfault.com/a/722646/117601)
|
|
std::vector<std::string> locations = {
|
|
"/etc/ssl/certs/ca-certificates.crt", // Debian/Ubuntu/Gentoo etc.
|
|
"/etc/pki/tls/certs/ca-bundle.crt", // Fedora/RHEL 6
|
|
"/etc/ssl/ca-bundle.pem", // OpenSUSE
|
|
"/etc/pki/tls/cacert.pem", // OpenELEC
|
|
"/etc/pki/ca-trust/extracted/pem/tls-ca-bundle.pem", // CentOS/RHEL 7
|
|
"/usr/local/etc/openssl/cert.pem", // OSX
|
|
};
|
|
|
|
HttpClient::ca_cert_path = "";
|
|
|
|
for(const std::string & location: locations) {
|
|
if(file_exists(location)) {
|
|
HttpClient::ca_cert_path = location;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
long HttpClient::perform_curl(CURL *curl, std::map<std::string, std::string>& res_headers) {
|
|
struct curl_slist *chunk = nullptr;
|
|
std::string api_key_header = std::string("x-typesense-api-key: ") + HttpClient::api_key;
|
|
chunk = curl_slist_append(chunk, api_key_header.c_str());
|
|
curl_easy_setopt(curl, CURLOPT_HTTPHEADER, chunk);
|
|
CURLcode res = curl_easy_perform(curl);
|
|
|
|
if (res != CURLE_OK) {
|
|
LOG(ERROR) << "CURL failed: " << curl_easy_strerror(res);
|
|
curl_easy_cleanup(curl);
|
|
return 500;
|
|
}
|
|
|
|
long http_code = 500;
|
|
curl_easy_getinfo (curl, CURLINFO_RESPONSE_CODE, &http_code);
|
|
|
|
extract_response_headers(curl, res_headers);
|
|
curl_easy_cleanup(curl);
|
|
|
|
return http_code == 0 ? 500 : http_code;
|
|
}
|
|
|
|
void HttpClient::extract_response_headers(CURL* curl, std::map<std::string, std::string> &res_headers) {
|
|
char* content_type;
|
|
curl_easy_getinfo (curl, CURLINFO_CONTENT_TYPE, &content_type);
|
|
res_headers.emplace("content-type", content_type);
|
|
}
|
|
|
|
CURL *HttpClient::init_curl(const std::string& url, std::string& response) {
|
|
CURL *curl = curl_easy_init();
|
|
|
|
if(curl == nullptr) {
|
|
nlohmann::json res;
|
|
res["message"] = "Failed to initialize HTTP client.";
|
|
response = res.dump();
|
|
return nullptr;
|
|
}
|
|
|
|
if(!ca_cert_path.empty()) {
|
|
curl_easy_setopt(curl, CURLOPT_CAINFO, ca_cert_path.c_str());
|
|
} else {
|
|
LOG(WARNING) << "Unable to locate system SSL certificates.";
|
|
}
|
|
|
|
curl_easy_setopt(curl, CURLOPT_URL, url.c_str());
|
|
curl_easy_setopt(curl, CURLOPT_CONNECTTIMEOUT_MS, 4000);
|
|
|
|
// to allow self-signed certs
|
|
curl_easy_setopt(curl, CURLOPT_SSL_VERIFYPEER, 0L);
|
|
curl_easy_setopt(curl, CURLOPT_SSL_VERIFYHOST, 0L);
|
|
|
|
curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, HttpClient::curl_write);
|
|
curl_easy_setopt(curl, CURLOPT_WRITEDATA, &response);
|
|
|
|
return curl;
|
|
}
|
|
|
|
size_t HttpClient::curl_write(void *contents, size_t size, size_t nmemb, std::string *s) {
|
|
s->append((char*)contents, size*nmemb);
|
|
return size*nmemb;
|
|
}
|