Skip to main content

Tremolo is a stream-oriented, asynchronous, programmable HTTP server written in pure Python. It can also serve as an ASGI server.

Project description

Tremolo

Coverage Quality Gate Status

Tremolo is a stream-oriented, asynchronous, programmable HTTP server written in pure Python. It can also serve as an ASGI server for other ASGI web frameworks.

Being built with a stream in mind, Tremolo tends to use yield instead of return in route handlers.

@app.route('/hello')
async def hello_world(**server):
    yield b'Hello '
    yield b'World!'

But that's not the only pattern. You can go straight to the Basics to start building your web application with Tremolo.

Features

Tremolo is only suitable for those who value minimalism and stability over features.

With only 3k lines of code, with no dependencies other than the Python Standard Library, it gives you:

All built-in in a single, portable folder module tremolo, in a very compact way like a Swiss Army knife.

Installation

python3 -m pip install --upgrade tremolo

Example

Here is a complete hello world example in case you missed the usual return.

from tremolo import Application

app = Application()

@app.route('/hello')
async def hello_world(**server):
    return 'Hello World!', 'latin-1'


if __name__ == '__main__':
    app.run('0.0.0.0', 8000, debug=True)

Well, latin-1 on the right side is not required. The default is utf-8.

You can save it as hello.py and just run it with python3 hello.py. And your first hello world page with Tremolo will be at http://localhost:8000/hello.

ASGI Server

Tremolo is an HTTP Server framework. You can build abstractions on top of it, say an ASGI server.

In fact, Tremolo already has ASGI server (plus WebSocket) implementation. So you can immediately use existing ASGI applications / frameworks, on top of Tremolo (ASGI server).

For example, If a minimal ASGI application with the name example.py:

async def app(scope, receive, send):
    assert scope['type'] == 'http'

    await send({
        'type': 'http.response.start',
        'status': 200,
        'headers': [
            (b'content-type', b'text/plain')
        ]
    })
    await send({
        'type': 'http.response.body',
        'body': b'Hello, World!'
    })

Then you can run as follows:

python3 -m tremolo --debug --bind 127.0.0.1:8000 example:app

To see more available options:

python3 -m tremolo --help

It's also possible to run the ASGI server programmatically (example with uvloop):

python3 example_uvloop.py

Testing

Just run python3 alltests.py for all tests. Or individual test_*.py in the tests/ folder, for example python3 tests/test_cli.py.

If you also want measurements with coverage:

coverage run alltests.py
coverage combine
coverage report
coverage html # to generate html reports

Benchmarking

The first thing to note is that Tremolo is a pure Python server framework.

As a pure Python server framework, it is hard to find a comparison. Because most servers/frameworks today are full of steroids like httptools, uvloop, Rust, etc.

You can try comparing with Uvicorn with the following option (disabling steroids to be fair):

uvicorn --loop asyncio --http h11 --log-level error example:app

vs

python3 -m tremolo --log-level ERROR example:app

You will find that Tremolo is reasonably fast.

If it's not, it could be due to --upload-rate or --download-rate limits, which take effect when the payload is slightly larger. Despite causing benchmarks to show poor results, it prevents any single client from monopolizing bandwidth, ensuring responsiveness under heavy load.

However, it should be noted that bottlenecks often occur on the application side. Which means that in real-world usage, throughput reflects more on the application than the server.

Misc.

Tremolo utilizes SO_REUSEPORT (Linux 3.9+) to load balance worker processes.

app.run('0.0.0.0', 8000, worker_num=2)

Tremolo can also listen to multiple ports in case you are using an external load balancer like Nginx / HAProxy.

app.listen(8001)
app.listen(8002)

app.run('0.0.0.0', 8000)

You can even get higher concurrency with PyPy or uvloop:

python3 -m tremolo --loop uvloop --log-level ERROR example:app

See: --loop

License

MIT License

Project details


Release history Release notifications | RSS feed

This version

0.5.0

Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

tremolo-0.5.0.tar.gz (41.4 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

tremolo-0.5.0-py3-none-any.whl (49.0 kB view details)

Uploaded Python 3

File details

Details for the file tremolo-0.5.0.tar.gz.

File metadata

  • Download URL: tremolo-0.5.0.tar.gz
  • Upload date:
  • Size: 41.4 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.7

File hashes

Hashes for tremolo-0.5.0.tar.gz
Algorithm Hash digest
SHA256 34576db064b0ea73aeb4ad9fcbcfe7102eb406d842605658d20556e22b76ca28
MD5 e373553c1a843d1374d895ce5f3b4e9d
BLAKE2b-256 02d2aebcb1246de8750373cbb2bbaaec75c0b0cb4def4a5526e309d4d81d354a

See more details on using hashes here.

Provenance

The following attestation bundles were made for tremolo-0.5.0.tar.gz:

Publisher: build_and_release.yml on nggit/tremolo

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file tremolo-0.5.0-py3-none-any.whl.

File metadata

  • Download URL: tremolo-0.5.0-py3-none-any.whl
  • Upload date:
  • Size: 49.0 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.7

File hashes

Hashes for tremolo-0.5.0-py3-none-any.whl
Algorithm Hash digest
SHA256 479bfad43c263ff3856de2fd8db8de8f8d62c3b61d0319b0e3c83ffdd1a32a8c
MD5 15cd1d5ad559d34a46636fc05bc92dfa
BLAKE2b-256 46dee51e7554ebb88f787d55c7b0f005d99b7d539d7950ffe5d01ee6e57d7fbb

See more details on using hashes here.

Provenance

The following attestation bundles were made for tremolo-0.5.0-py3-none-any.whl:

Publisher: build_and_release.yml on nggit/tremolo

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page