This is a text-only version of the following page on https://raymii.org: --- Title : C++ set up cpp-httplib with SSL support with cMake Author : Remy van Elst Date : 14-12-2020 URL : https://raymii.org/s/tutorials/Cpp_set_up_cpp-httplib_with_ssl_support_with_cmake.md.html Format : Markdown/HTML --- 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][1], 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. <p class="ad"> <b>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:</b><br><br> <a href="https://leafnode.nl">I'm developing an open source monitoring app called Leaf Node Monitoring, for windows, linux & android. Go check it out!</a><br><br> <a href="https://github.com/sponsors/RaymiiOrg/">Consider sponsoring me on Github. It means the world to me if you show your appreciation and you'll help pay the server costs.</a><br><br> <a href="https://www.digitalocean.com/?refcode=7435ae6b8212">You can also sponsor me by getting a Digital Ocean VPS. With this referral link you'll get $100 credit for 60 days. </a><br><br> </p> 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][2] 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: ![error][5] 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 } [1]: https://github.com/yhirose/cpp-httplib [2]: https://github.com/yhirose/cpp-httplib#openssl-support [5]: /s/inc/img/httplib-ssl.png --- License: All the text on this website is free as in freedom unless stated otherwise. This means you can use it in any way you want, you can copy it, change it the way you like and republish it, as long as you release the (modified) content under the same license to give others the same freedoms you've got and place my name and a link to this site with the article as source. This site uses Google Analytics for statistics and Google Adwords for advertisements. You are tracked and Google knows everything about you. Use an adblocker like ublock-origin if you don't want it. All the code on this website is licensed under the GNU GPL v3 license unless already licensed under a license which does not allows this form of licensing or if another license is stated on that page / in that software: This program is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program. If not, see <http://www.gnu.org/licenses/>. Just to be clear, the information on this website is for meant for educational purposes and you use it at your own risk. I do not take responsibility if you screw something up. Use common sense, do not 'rm -rf /' as root for example. If you have any questions then do not hesitate to contact me. See https://raymii.org/s/static/About.html for details.