JoshCap20 / areion

The fastest Python web server. A lightweight, fast, and extensible asynchronous Python web server framework.
MIT License
0 stars 1 forks source link
api framework python web web-framework

Areion Server

License Downloads Downloads

PyPi PyPi

Welcome to the Areion HTTP Server documentation. Areion is a lightweight, asynchronous HTTP server written in Python, designed for simplicity and extensibility. This documentation provides a comprehensive guide to using Areion, covering everything from getting started to advanced usage and component details.

Areion supports asynchronous operations, multithreading, routing, orchestration, customizable loggers, and template engines. The framework provides an intuitive API for building web services, with components like the Orchestrator, Router, Logger, and Engine easily swappable or extendable.

We designed Areion to have as few dependencies as possible. We created our own HTTP server on top of asyncio's sockets. While we dream of being the fastest, most preferred Python web server, we know we have a long way to go. We are still in the early stages of development, and we welcome any feedback, contributions, or suggestions. The documentation below is likely to become outdated as we continue to migrate to v2.0.0 which will feature a whole documentation site with more examples, tutorials, and guides.

Development Mode: Add the flag with_development_mode(True) to the AreionServerBuilder to enable development mode. This mode will automatically add Swagger UI and OpenAPI routes to your server. They are accessible from the routes /docs and /openapi respectively. Note: Providing comprehensive docstrings with detailed parameter descriptions and return types will enhance the accuracy of the autogenerated OpenAPI schema. This, in turn, will prepopulate the Swagger UI request body examples, facilitating more effective testing.

Table of Contents


Benchmark

We conducted performance benchmarks to compare Areion, FastAPI, and Flask, focusing on throughput and latency under high-load conditions. The goal was to evaluate each framework's ability to handle concurrent connections efficiently and provide fast response times. We used the same JSON response in all frameworks to ensure a fair comparison.

Benchmark Results

These show the results of running the benchmark test for 30 seconds with 12 threads and 400 connections on my local machine. The test was conducted using the wrk benchmarking tool. The results are summarized below, followed by detailed output for each framework.

Summary

Framework Requests/sec Avg Latency (ms) Transfer/sec Total Requests Socket Errors
Areion 47,241.97 8.46 4.42 MB 1,418,550 Read: 545
FastAPI 3,579.10 111.53 531.27 KB 107,613 Read: 419
Flask 555.98 47.45 104.79 KB 16,708 Connect: 74, Read: 36,245

Visualization

Requests per Second

Average Latency

Detailed Results

Results show the performance when keep-alive connections are used (HTTP/1.1 default). Areion still significantly outperforms when the header Connection: close is set.

Areion

Running 30s test @ http://localhost:8000/json
  12 threads and 400 connections
  Thread Stats   Avg      Stdev     Max   +/- Stdev
    Latency     8.46ms    2.06ms  47.38ms   95.91%
    Req/Sec     3.96k   430.06     5.36k    87.17%
  1,418,550 requests in 30.03s, 132.58MB read
  Socket errors: connect 0, read 545, write 0, timeout 0
Requests/sec:  47,241.97
Transfer/sec:      4.42MB

FastAPI

Running 30s test @ http://localhost:8000/json
  12 threads and 400 connections
  Thread Stats   Avg      Stdev     Max   +/- Stdev
    Latency   111.53ms   31.97ms 498.08ms   89.55%
    Req/Sec   300.08     59.85   430.00     86.17%
  107,613 requests in 30.07s, 15.60MB read
  Socket errors: connect 0, read 419, write 0, timeout 0
Requests/sec:   3,579.10
Transfer/sec:    531.27KB

Flask

Running 30s test @ http://localhost:8000/json
  12 threads and 400 connections
  Thread Stats   Avg      Stdev     Max   +/- Stdev
    Latency    47.45ms   53.33ms 556.78ms   96.24%
    Req/Sec   183.12    104.16   590.00     70.64%
  16,708 requests in 30.05s, 3.08MB read
  Socket errors: connect 74, read 36,245, write 125, timeout 0
Requests/sec:     555.98
Transfer/sec:    104.79KB

Analysis

Throughput (Requests per Second)

Areion handled approximately 13 times more requests per second than FastAPI and 85 times more than Flask.

Average Latency

Areion's average latency is about 5.6 times lower than Flask and 13 times lower than FastAPI, indicating faster response times.

Total Requests Handled

Areion processed significantly more total requests during the test duration.

Socket Errors

For handling 10x the requests of FastAPI and 85x the requests of Flask, Areion had a relatively low number of socket errors.

Getting Started

Installation

Areion can be installed via pip:

pip install areion

Developing

To get started developing this project, follow these steps:

  1. Clone the repository:
git clone https://github.com/JoshCap20/areion.git
  1. Install the dependencies (preferably in a virtual environment):
pip install -r requirements.txt
  1. Run the tests:
pytest

Quick Start Guide

Below is a simple example to get you started with Areion.


from areion import AreionServerBuilder, DefaultRouter, HttpRequest

# Initialize the router
router = DefaultRouter()

# Define a simple route
@router.route("/hello")
def hello_world(request: HttpRequest):
    return "Hello, World!"

# Build and run the server
server = AreionServerBuilder().with_router(router).build()
server.run()

Explanation:

Development Tools

Development tools are essential to increasing development ease and efficiency. With this in mind, we also don't want to slow down the server in production. To enable development tools, add the with_development_mode(True) flag to the AreionServerBuilder.

Development Tools:

Coming Soon:

Core Components

AreionServer

The AreionServer is the main class that represents the server. It manages components such as the router, orchestrator, logger, and HTTP server.

Key Features:

Do not initialize AreionServer directly unless you know what you're doing. Use AreionServerBuilder to create an instance which includes safeguards and defaults required for other components.

AreionServerBuilder

The AreionServerBuilder provides a fluent interface for constructing an AreionServer instance with the desired components and configurations.

Only a router is required for a minimal server setup.

Example:

from areion import AreionServerBuilder, DefaultRouter, DefaultLogger, DefaultOrchestrator, DefaultEngine

router = DefaultRouter()
logger = DefaultLogger()
orchestrator = DefaultOrchestrator()
engine = DefaultEngine()

server = AreionServerBuilder().with_router(router).with_logger(logger).with_orchestrator(orchestrator).with_port(8080).with_engine(engine).with_static_dir("static").build()

Router

The Router class manages URL routes and their corresponding handlers. If no methods are defined, it defaults to only accepting GET requests. You can define middleware at the route, route group, or global level.

Key Features:

FastAPI-like Route Definition Example

This default usage is similar to Flask and FastAPI, with decorators for defining routes and middleware.

Usage:

from areion import DefaultRouter

router = DefaultRouter()

# Adding a route with a dynamic segment
@router.route("/user/:id", methods=["GET"])
def get_user(request, id):
    return f"User ID: {id}"

# Adding global middleware
def log_request(handler):
    def wrapper(request, *args, **kwargs):
        print(f"Received request: {request.path}")
        return handler(request, *args, **kwargs)
    return wrapper

router.add_global_middleware(log_request)

Subrouter (Grouping Routes) Example

main.py:

from areion import DefaultRouter

from areion import (
    AreionServer,
    AreionServerBuilder,
    DefaultLogger,
    DefaultEngine,
    DefaultRouter,
    HttpResponse
)
from users import users_router

main_router = DefaultRouter()
main_router.include_router(users_router)

logger = DefaultLogger(log_file="server.log")
engine = DefaultEngine()

server: AreionServer = (
    AreionServerBuilder()
        .with_router(main_router)
        .with_engine(engine)
        .with_logger(logger)
    .build()
)

if __name__ == "__main__":
    server.run()

users.py:

from areion import DefaultRouter, HttpResponse

users_router = DefaultRouter(prefix="/users")

@users_router.route("/", methods=["GET"])
def get_all_users(request):
    return HttpResponse(status_code=200, body={"users": []}, content_type="application/json")

@users_router.route("/:user_id", methods=["GET"])
def get_user(request, user_id):
    body = request.get_parsed_body()
    if not body.get("token"):
        return HttpResponse(status_code=401, body="Unauthorized", content_type="text/plain")
    return HttpResponse(status_code=200, body={"user_id": user_id}, content_type="application/json")

Django-like Route Definition Example

You can also use a Django-like syntax for defining routes:

router = DefaultRouter()

def get_user(request, id):
    return f"User ID: {id}"

router.add_route("/user/:id", get_user, methods=["GET"])

Route with middleware:

router = DefaultRouter()

def get_user(request, id):
    return f"User ID: {id}"

def log_request(handler):
    def wrapper(request, *args, **kwargs):
        print(f"Received request: {request.path}")
        return handler(request, *args, **kwargs)
    return wrapper

router.add_route("/user/:id", get_user, methods=["GET"], middlewares=[log_request])

HttpServer

The HttpServer class handles the low-level HTTP protocol details.

Key Features:

Usage:

This class is usually managed internally by AreionServer and doesn't require direct interaction.

Default Component Implementation

Areion provides default implementations for several core components that can be used out of the box. These components are designed to work together seamlessly and provide a solid foundation for building web applications. Feel free to use these default components or create your own custom implementations according to the interfaces in areion/base.

Orchestrator

The Orchestrator class manages background tasks and scheduling. The startup and shutdown are managed by the AerionServer.

Key Features:

Usage:

from areion import DefaultOrchestrator

orchestrator = DefaultOrchestrator(max_workers=4)

def background_task():
    print("Running background task")

# Submit a task
orchestrator.submit_task(background_task)

# Schedule a cron task
orchestrator.schedule_cron_task(background_task, {'hour': '*/1'})

Logger

The Logger class provides logging capabilities. This logger is passed to multiple other components and also injected into each HttpRequest that is accesible in every route handler.

Key Features:

Usage:

from areion import DefaultLogger, AreionServerBuilder, DefaultRouter

router = DefaultRouter()
logger = DefaultLogger(log_file="server.log", log_level="INFO")

@router.route("/log", methods=["GET"])
def some_handler(request):
    logger.info("Processing request")
    return "Response"

server = AreionServerBuilder().with_router(router).with_logger(logger).build()

Engine

The Engine class handles template rendering.

Key Features:

Jinja2 Integration: Uses Jinja2 for template rendering. Template Directory: Configurable template directory.

Usage:

from areion import DefaultEngine

engine = Engine(templates_dir="templates")

# Rendering a template
def home(request):
    context = {"title": "Home"}
    return request.render_template("home.html", context)

Advanced Usage

Middleware

Middleware functions allow you to process requests and responses globally or for specific routes.

Creating Middleware:

def auth_middleware(handler):
    def wrapper(request, *args, **kwargs):
        if not request.get_header("Authorization"):
            return HttpResponse(status_code=401, body="Unauthorized")
        return handler(request, *args, **kwargs)
    return wrapper

Applying Middleware:

Global Middleware:

router.add_global_middleware(auth_middleware)

Route-specific Middleware:

@router.route("/dashboard", middlewares=[auth_middleware])
def dashboard(request):
    return "Welcome to the dashboard"

Grouping Routes

You can group routes under a common path and apply middlewares to the group.

Usage:

# Define group-specific middleware
def group_middleware(handler):
    def wrapper(request, *args, **kwargs):
        print("Group middleware executed")
        return handler(request, *args, **kwargs)
    return wrapper

# Create a route group
api_group = router.group("/api", middlewares=[group_middleware])

@api_group.route("/users")
def api_users(request):
    return "API Users"

@api_group.route("/posts")
def api_posts(request):
    return "API Posts"

Template Rendering

Use the Engine component to render templates with dynamic content.

Usage:

def profile(request):
    user_data = {"name": "John Doe", "age": 30}
    return request.render_template("profile.html", user_data)

Task Scheduling

Schedule tasks to run at specified intervals using the Orchestrator.

Usage:

def cleanup_task():
    print("Performing cleanup")

# Schedule to run every day at midnight
orchestrator.schedule_cron_task(cleanup_task, {'hour': '0', 'minute': '0'})

API Reference

AreionServer API

AreionServer

Represents the main server class.

Constructor Parameters:

Methods:

AreionServerBuilder

Builder class for constructing AreionServer instances.

Methods:

Router API

Router

Manages URL routes and their handlers.

Methods:

HttpRequest and HttpResponse

HttpRequest

Represents an HTTP request. These are injected into each route handler as the first argument via the HttpServer.

Attributes:

Methods:

HttpResponse

Represents an HTTP response.

Constructor Parameters:

Methods:

Response Utility Functions

Exception Handling

Areion provides custom exceptions for common HTTP errors. These exceptions can be raised in route handlers to return the corresponding error response and status code.

Easily create custom exceptions by subclassing the HttpError class.

Usage:

from areion import NotFoundError

def get_item(request, item_id):
    item = database.get(item_id)
    if not item:
        raise NotFoundError("Item not found")
    return item

Exceptions in routes and middleware are handled globally and converted to a proper HTTP response.

Best Practices

Responses

We recommend returning a HttpResponse object directly from route handlers. This allows for more control over the response status code, headers, and body. Additionally, it is recommended, but not required, to pass an explicit content_type during construction for performance reasons.

_Helpful wrappers now exist in the areion/utils/response_utils.py to make this easier for common use cases._

Make sure that objects are JSON serializable before returning them in the response body.

Example:

from areion import HttpResponse

@router.route("/user", methods=GET)
def get_user(request):
    user = {
        "name": "John Doe",
        "age": 30
    }
    return HttpResponse(body=user, content_type="application/json")

Missing HTTP/1.1 Features

Contributing

Contributions are welcome! For feature requests, bug reports, or questions, please open an issue. If you would like to contribute code, please open a pull request with your changes.

License

MIT License

Copyright (c) 2024 Joshua Caponigro

Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software.