You're using curl? Do you even lift?
liblifthttp is a C++17 HTTP client library that provides an easy to use API for both synchronous and asynchronous requests. It is built upon the super heavyweight champions libcurl and libuv libraries.
liblifthttp is licensed under the Apache 2.0 license.
- Easy to use Synchronous and Asynchronous HTTP Request APIs.
- Safe C++17 client library API, modern memory move semantics.
- Background IO thread(s) for sending and receiving Async HTTP requests.
- Request pooling for re-using HTTP requests and sharing of connection information.
See all of the examples under the examples/
directory. Below are some simple examples
to get your started on using liblifthttp with both the synchronous and asynchronous APIs.
#include <iostream>
#include <lift/lift.hpp>
int main()
{
// Synchronous requests can be created on the stack.
lift::request request{"http://www.example.com"};
// This is the blocking synchronous HTTP call.
auto response = request.perform();
std::cout << "Lift status: " << lift::to_string(response.lift_status()) << "\n";
std::cout << response << "\n"; // Will print the raw http response.
// Creating the client starts it immediately, it spawns a background thread for executing requests.
lift::client client{};
// Create the request just like we did in the sync version, but now provide a lambda for on completion.
// NOTE: that the Lambda is executed ON the Lift client background thread. If you want to handle
// on completion processing on this main thread you need to std::move() it back via a queue or inter-thread
// communication. This is imporant if any resources are shared between the threads.
// NOTE: The request is created on the heap so ownership can be passed easily via an std::unique_ptr
// to the lift::client! lift::request::make_unique() is a handy function to easily do so.
auto request_ptr = lift::request::make_unique(
"http://www.example.com",
std::chrono::seconds{10}, // Give the request 10 seconds to complete or timeout.
[](lift::request_ptr req_ptr, lift::response response) {
std::cout << "Lift status: " << lift::to_string(response.lift_status()) << "\n";
std::cout << response << "\n";
});
// Now inject the request into the client to be executed. Moving into the client is required,
// this passes ownership of the request to the client's background worker thread.
client.start_request(std::move(request_ptr));
// Block on this main thread until the lift client has completed the request, or timed out.
while (!client.empty())
{
std::this_thread::sleep_for(std::chrono::milliseconds{10});
}
return 0;
}
C++17 compiler
g++-9
clang-9
CMake
make or ninja
pthreads
libcurl-devel >= 7.59
libuv-devel
zlib-devel
openssl-devel (or equivalent curl support ssl library)
stdc++fs
Tested on:
ubuntu:20.04
fedora:31
# This will produce a static library to link against your project.
mkdir Release && cd Release
cmake -DCMAKE_BUILD_TYPE=Release ..
cmake --build .
CMake options:
Name | Default | Description |
---|---|---|
LIFT_BUILD_EXAMPLES | ON | Should the examples be built? |
LIFT_BUILD_TESTS | ON | Should the tests be built? |
LIFT_CODE_COVERAGE | OFF | Should code coverage be enabled? |
LIFT_USER_LINK_LIBRARIES | curl z uv pthread dl stdc++fs | Override lift's target link libraries. |
Note on LIFT_USER_LINK_LIBRARIES
, if override the value then all of the default link libraries/targets must be
accounted for in the override. E.g. if you are building with a custom curl target but defaults for everything else
then -DLIFT_USER_LINK_LIBRARIES="custom_curl_target z uv pthread dl stdc++fs"
would be the correct setting.
To use within your cmake project you can clone the project or use git submodules and then add_subdirectory
in the parent project's CMakeList.txt
,
assuming the lift code is in a liblifthttp/
subdirectory of the parent project:
add_subdirectory(liblifthttp)
To link to the <project_name>
then use the following:
add_executable(<project_name> main.cpp)
target_link_libraries(<project_name> PUBLIC lifthttp)
Include lift in the project's code by simply including #include <lift/lift.hpp>
as needed.
CMake can also include the project directly via a FetchContent
declaration. In your project's CMakeLists.txt
include the following code to download the git repository and make it available to link to.
cmake_minimum_required(VERSION 3.11)
# ... cmake project stuff ...
include(FetchContent)
FetchContent_Declare(
lifthttp
GIT_REPOSITORY https://github.com/jbaldwin/liblifthttp.git
GIT_TAG <TAG_OR_GIT_HASH>
)
FetchContent_MakeAvailable(lifthttp)
# ... cmake project more stuff ...
target_link_libraries(${PROJECT_NAME} PUBLIC lifthttp)
The tests are automatically run by GitHub Actions on all Pull Requests. They can also be ran locally with a default
localhost instance of nginx
. To do so the CMake option LIFT_LOCALHOST_TESTS=ON
must be set otherwise the tests
will use the hostname nginx
setup in the CI settings. After building and starting nginx
tests can be run by issuing:
# Invoke via cmake:
ctest -v
# Or invoke directly to see error messages if tests are failing:
./test/liblifthttp_tests
Note there are now proxy http requests that utilize an haproxy
instance. To run these locally you will also need
to start an instance of haproxy
.
Using the example benchmark code and a local nginx
instance serving its default welcome page. All benchmarks use keep-alive
connections. The benchmark is compared against wrk
as that is basically optimal performance since
wrk
does zero parsing of the response whereas lift
does.
Here is the CPU the benchmarks were run on:
cat /proc/cpuinfo
...
Intel(R) Core(TM) i9-9980HK CPU @ 2.40GHz
Here is how the benchmark application is called:
$ ./examples/lift_benchmark --help
Usage: ./examples/lift_benchmark<options> <url>
-c --connections HTTP Connections to use.
-t --threads Number of threads to use, connections are split
evenly between each worker thread.
-d --duration Duration of the test in seconds
-h --help Print this help usage.
Using nginx
as the webserver with the default fedora
configuration.
Connections | Threads | wrk Req/Sec | lift Req/Sec |
---|---|---|---|
1 | 1 | 31,138 | 20,785 |
100 | 1 | 148,121 | 44,393 |
100 | 2 | 220,670 | 81,335 |
100 | 3 | 241,839 | 104,747 |
100 | 4 | 275,633 | 123,481 |
100 | 8 | 249,845 | 143,911 |
File bug reports, feature requests and questions using GitHub Issues
Copyright © 2017-2020, Josh Baldwin