Skip to content

Ousret/niquests-stats

Repository files navigation

Performance Measurements

Niquests is an advanced HTTP client with, on its core urllib3.future. It is capable of serving a wide range of features, and is actually leading the field in terms of features.

In this document, we will inspect and compare the performance of Niquests against well known clients.

Introduction

Client Constraints
aiohttp async, http/1 only, +c llhttp parser
httpx async, http/2
requests sync, http/1 only
niquests async, http/2 +hpack rust

in the given matrix of client, aiohttp is the only one who benefit from built llhttp native C extension to provide a nearly unchallengeable speed. The only way to fairly compare with aiohttp is by bringing the cores of requests, httpx and niquests so that the stack/complexity execution and features served are comparable.

we will be as fair as possible across all clients by providing the closest alike execution.

Established Scenario

The shared pool connection MUST not exceed 10 connections. The internals are free to scale at will in given limits.

Spawn 10 tasks, each task share the same "pool" of connection, and each task fetch 100 responses. The responses are collected through the main task. Each sub-task have a time counter along with the main one.

We include, in spent time, the construction of the ssl context with the system CAs. The response count must be exactly 1000, and each response must have its (bytes) content loaded.

Each script must run exactly 60 times, and data must be aggregated afterward. The average delays are kept. A reasonable delay of 60 seconds must be respected between each script to avoid penalizing the performance due to CPU throttling.

We must ensure that we eliminate:

  • Network latency by either running the server locally or in a home network (wired).
  • It's recommended that, you must have at least 8c/16t available on the server/client side(s).
  • Constrain the TLS version and cipher to ensure OpenSSL alignments with all clients.
  • You must ensure the lowest amount of "variance by external factor" like but not limited by controlling thoroughly things like P/E cores and assign process to bound cores if necessary.

The delay must be computed after contextmanager exit (aka. pool shutdown). Of course, don't do anything else during the benchmarks, nothing.

Prerequisites

Tools you must have in your environment:

  • mkcert
  • docker
  • compose plugin v2

The server will be handled by both:

  • Traefik
  • gohttpbin (x6 replicas)

Main execution script

$ mkcert httpbin.local -cert-file ./certs/httpbin.local.pem -key-file ./certs/httpbin.local.key
$ echo "127.0.0.1   httpbin.local" | sudo tee -a /etc/hosts
$ docker compose up -d
$ pip install -r requirements.txt
$ python aio_httpx.py
$ python aio_niquests.py
$ python tpe_requests.py
$ python aio_httpx_core.py
$ python aio_niquests_core.py
$ python tpe_requests_core.py
$ python aio_aiohttp.py

Results

The scenario is executed on a mid-range laptop, running Fedora 38 with Python 3.11. CPU: 12th Gen Intel® Core™ i7, 32 GiB of DDR4.

High-level APIs

Client Average Delay to Complete Estimated Throughput
requests 987 ms ~1013 req/s
httpx 720 ms ~1389 req/s
niquests 340 ms ~2941 req/s

Simplified APIs

Client Average Delay to Complete Estimated Throughput
requests core 643 ms ~1555 req/s
httpx core 490 ms ~2000 req/s
aiohttp 210 ms ~4762 req/s
niquests core 160 ms ~6200 req/s

Did you give up on HTTP/2 due to performance concerns? Think again! Do you realize that you can get 2.66 times faster with the same CPU if you ever switched to Niquests from Requests?

Yet, Niquests is not going to settle with these numbers, we're constantly thinking of innovative ways to speed things up. Nevertheless, it performs the best overall, with offered level of features.

The actual delays heavily depends on your CPU capabilities and your interpreter version. The factors should remain the same.

More computational power should give you higher advantages when using Niquests thanks to how it does multiplexing. In theory, if I were able to put my hand on a more powerful CPU, the gap between Niquests and others should be greater.