#!/usr/bin/env python import os from random import randint, sample from alchemical.aio import Alchemical, Model import sqlalchemy.orm as so from asyncache import cached from cachetools.keys import hashkey from microdot.asgi import Microdot from microdot.jinja import Template app = Microdot() Template.initialize('templates', enable_async=True) db = Alchemical(os.environ.get('DATABASE_URL', 'sqlite:///')) class World(Model): __tablename__ = "world" id: so.Mapped[int] = so.mapped_column(primary_key=True) randomnumber: so.Mapped[int] def to_dict(self): return {"id": self.id, "randomNumber": self.randomnumber} class CachedWorld(Model): __tablename__ = "cachedworld" id: so.Mapped[int] = so.mapped_column(primary_key=True) randomnumber: so.Mapped[int] def to_dict(self): return {"id": self.id, "randomNumber": self.randomnumber} class Fortune(Model): __tablename__ = "fortune" id: so.Mapped[int] = so.mapped_column(primary_key=True) message: so.Mapped[str] def get_num_queries(request, name="queries"): try: num_queries = request.args.get(name, 1, type=int) except ValueError: num_queries = 1 if num_queries < 1: return 1 if num_queries > 500: return 500 return num_queries def generate_ids(num_queries): return sample(range(1, 10001), num_queries) @app.route("/json") async def test_json(request): return {"message": "Hello, World!"} @app.route("/db") async def test_db(request): id = randint(1, 10000) async with db.Session() as session: world = await session.get(World, id) return world.to_dict() @app.route("/queries") async def test_queries(request): async with db.Session() as session: worlds = [(await session.get(World, id)).to_dict() for id in generate_ids(get_num_queries(request))] return worlds @app.route("/fortunes") async def test_fortunes(request): async with db.Session() as session: fortunes = list(await session.scalars(Fortune.select())) fortunes.append(Fortune(id=0, message="Additional fortune added at request time.")) fortunes.sort(key=lambda f: f.message) return ( await Template("fortunes.html").render_async(fortunes=fortunes), {'Content-Type': 'text/html; charset=utf-8'}, ) @app.route("/updates") async def test_updates(request): worlds = [] ids = generate_ids(get_num_queries(request)) ids.sort() # to avoid deadlocks async with db.begin() as session: for id in ids: world = await session.get(World, id) world.randomnumber = (randint(1, 9999) + world.randomnumber - 1) % 10000 + 1 worlds.append(world.to_dict()) return worlds @app.route("/plaintext") async def test_plaintext(request): return b"Hello, World!" @cached(cache={}, key=lambda session, id: hashkey(id)) async def get_cached_world(session, id): return (await session.get(World, id)).to_dict() @app.route("/cached-queries") async def test_cached_queries(request): async with db.Session() as session: worlds = [await get_cached_world(session, id) for id in generate_ids(get_num_queries(request, 'count'))] return worlds