Where other frameworks struggle with "coordinated omission" (shedding latency measurements during spikes), Sanic’s non-blocking design ensures that slow database queries don’t freeze unrelated endpoints. Hopkins probably includes a case study: a social media feed endpoint that calls three external APIs concurrently using asyncio.gather() . In Flask, this requires third-party libraries ( aiohttp + gevent ) and risks callback hell. In Sanic, it is native.
Adam Hopkins, the creator and lead maintainer of Sanic, did not write Sanic: Python Web Development as just another tutorial. Reading between the lines of the framework’s evolution, this hypothetical but authoritative PDF serves a singular, disruptive thesis: This essay argues that Hopkins’ work is a polemic against "async-washing" (bolting async onto sync frameworks) and a practical manifesto for building web services that finally leverage Python’s async/await without compromise. Part I: The WSGI Hangover – Why Sanic Exists The first third of Hopkins’ book is likely dedicated to a painful history lesson: the limitations of WSGI. While WSGI served Python faithfully for two decades, it is fundamentally synchronous. Frameworks like Flask and Django, even with async endpoints added later, are at their core WSGI applications. They must spawn multiple worker processes (via Gunicorn) to handle concurrency, leading to a linear scalability problem. python web development with sanic adam hopkins pdf
Consider this practical example from the implied text: In Sanic, it is native
Introduction: The Noisy Ecosystem of Python Web Frameworks The Python web development landscape is often described as a battleground of giants. On one side stands Django, the "batteries-included" behemoth ideal for monolithic applications. On the other, Flask offers minimalist microframework elegance, later refined by FastAPI’s marriage of performance and automatic OpenAPI documentation. Lost in this noise, yet critically important, is Sanic. Part I: The WSGI Hangover – Why Sanic
@app.before_server_start async def setup_db(app): app.ctx.db = await asyncpg.create_pool(...) @app.get("/user/<uid>") async def get_user(request): async with request.app.ctx.db.acquire() as conn: return json(await conn.fetchrow("SELECT * FROM users WHERE id=$1", uid))
The deep thesis of the PDF is this: Until the entire ecosystem—from ORMs to template engines—becomes natively async, frameworks like Sanic will remain a niche for the performance-obsessed. But within that niche, Hopkins has built a cathedral of clean, fast, and honest code.
This is trivial, but the depth comes from Hopkins’ insistence on . The essay within the PDF would highlight that most async Python crashes stem from unclosed connections. By coupling setup and shutdown listeners ( @app.before_server_stop ), Sanic enforces a discipline that many ad-hoc FastAPI applications lack. Part III: Performance as a Feature, Not an Accident The latter third of Hopkins’ book inevitably confronts benchmarks. Sanic routinely outperforms Flask by an order of magnitude and edges out FastAPI in raw request handling (by 10-20% in typical JSON benchmarks). But Hopkins is not interested in winning pointless hello world races. Instead, the PDF likely argues for predictable performance under load .