Raymii.org
Quis custodiet ipsos custodes?Home | About | All pages | Cluster Status | RSS Feed
C++ set up cpp-httplib with SSL support with cMake
Published: 14-12-2020 | Author: Remy van Elst | Text only version of this article
❗ This post is over four years old. It may no longer be up to date. Opinions may have changed.
Table of Contents
For a small personal project that talks to a few JSON API's and does some data parsing I needed a header only C++ HTTP library. Header only because that is the simplest way to include it in my project, just copy and paste the file. I came across the project cpp-httplib, which fits my needs, does all the http methods, small, a few examples and it looks modern and has recent development commits.
Setup and getting it working was easy, but as soon as I tried an https url, I
got an exception (what(): 'https' scheme is not supported.
). This guide
shows you how to setup cpp-httplib for SSL support with cmake. It took me
a bit longer than I wanted to set it up correctly, so why not save you the
effort.
Recently I removed all Google Ads from this site due to their invasive tracking, as well as Google Analytics. Please, if you found this content useful, consider a small donation using any of the options below:
I'm developing an open source monitoring app called Leaf Node Monitoring, for windows, linux & android. Go check it out!
Consider sponsoring me on Github. It means the world to me if you show your appreciation and you'll help pay the server costs.
You can also sponsor me by getting a Digital Ocean VPS. With this referral link you'll get $200 credit for 60 days. Spend $25 after your credit expires and I'll get $25!
This is the example code we're compiling, as is in the readme, for HTTP:
httplib::Client cli("http://jsonplaceholder.typicode.com");
if (auto res = cli.Get("/todos/1")) {
if (res->status == 200) {
std::cout << res->body << std::endl;
}
}
Example output for HTTP:
/home/remy/CLionProjects/example1/cmake-build-debug/example1
{
"userId": 1,
"id": 1,
"title": "delectus aut autem",
"completed": false
}
For HTTPS you must read the entire article, at the bottom is the correct code and CmakeLists.txt, since just changing the URL will not work.
OpenSSL support via cmake
The README has one paragraph on SSL support, if you strip out the example, it's more like one line:
SSL support is available with CPPHTTPLIB OPENSSL SUPPORT. libssl and libcrypto should be linked.
On Ubuntu you should install the OpenSSL development libraries:
apt install libssl-dev
For the rest I assume you have your development libraries installed and cmake set up.
In your CmakeLists.txt
you should add the following to add OpenSSL:
find_package(OpenSSL REQUIRED)
if(OPENSSL_FOUND)
set(HTTPLIB_IS_USING_OPENSSL TRUE)
endif()
The cpp-httplib
library also needs a few linker flags and compile options,
which you can add to cmake with the below lines:
target_link_libraries(${PROJECT_NAME} PUBLIC
$<$<BOOL:${HTTPLIB_IS_USING_OPENSSL}>:OpenSSL::SSL>
$<$<BOOL:${HTTPLIB_IS_USING_OPENSSL}>:OpenSSL::Crypto>)
target_compile_definitions(${PROJECT_NAME} PUBLIC
$<$<BOOL:${HTTPLIB_IS_USING_OPENSSL}>:CPPHTTPLIB_OPENSSL_SUPPORT>
)
The compile definitons are a complicated way of doing -DCPPHTTPLIB_OPENSSL_SUPPORT
, based on an earlier variable.
These two blocks should be everything you need for your code to be able to link agains OpenSSL.
Hostname without the https part
This took me another few minutes to find out. Each and every request I tried to do
failed, with a httplib::Connection
error:
Nothing more, no specific SSL error, nothing. Turns out, this code doesn't work if you include the
scheme part inside the URL (https://
):
httplib::SSLClient cli("https://jsonplaceholder.typicode.com");
If you omit that part, it will work:
httplib::SSLClient cli("jsonplaceholder.typicode.com");
It doesn't help that there is no error message and that the .Error()
method
just says Connection
. What is happening here, more than just "something
wrong". Took me a few more tries and looking at the unit tests to figure out
what the intended behaviour should be.
You can use the preprocessor macro to check if you can use SSL, example below:
#ifdef CPPHTTPLIB_OPENSSL_SUPPORT
httplib::SSLClient cli(_domain);
cli.enable_server_certificate_verification(true);
#else
httplib::Client cli(_domain);
#endif
Make sure to catch any exceptions and check for nullpointers.
Conclusion & Full Code
Setting up the correct cMake flags and troubleshooting why code that should work doesn't work took me more time than I wanted, so I hope this guide saves you some time. Below you'll find the full example code.
CmakeLists.txt:
cmake_minimum_required(VERSION 3.17)
project(Example1)
set(CMAKE_CXX_STANDARD 17)
find_package(OpenSSL REQUIRED)
if(OPENSSL_FOUND)
set(HTTPLIB_IS_USING_OPENSSL TRUE)
endif()
add_executable(${PROJECT_NAME} main.cpp)
target_link_libraries(${PROJECT_NAME} PUBLIC
$<$<BOOL:${HTTPLIB_IS_USING_OPENSSL}>:OpenSSL::SSL>
$<$<BOOL:${HTTPLIB_IS_USING_OPENSSL}>:OpenSSL::Crypto>)
target_compile_definitions(${PROJECT_NAME} PUBLIC
$<$<BOOL:${HTTPLIB_IS_USING_OPENSSL}>:CPPHTTPLIB_OPENSSL_SUPPORT>
)
main.cpp
int main() {
#ifdef CPPHTTPLIB_OPENSSL_SUPPORT
httplib::SSLClient cli("jsonplaceholder.typicode.com");
cli.enable_server_certificate_verification(true);
#else
httplib::Client cli(http://jsonplaceholder.typicode.com);
#endif
if (auto res = cli.Get("/todos/1")) {
if (res->status == 200) {
std::cout << res->body << std::endl;
}
}
return 0;
}
Output:
/home/remy/CLionProjects/example1/cmake-build-debug/example1
{
"userId": 1,
"id": 1,
"title": "delectus aut autem",
"completed": false
}
Tags: c++
, cmake
, cpp
, cpp-httplib
, development
, http
, https
, openssl
, ssl
, tutorials
, web