app.py 3.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117
  1. #!/usr/bin/env python
  2. import os
  3. from random import randint, sample
  4. from alchemical.aio import Alchemical, Model
  5. import sqlalchemy.orm as so
  6. from asyncache import cached
  7. from cachetools.keys import hashkey
  8. from microdot.asgi import Microdot
  9. from microdot.jinja import Template
  10. app = Microdot()
  11. Template.initialize('templates', enable_async=True)
  12. db = Alchemical(os.environ.get('DATABASE_URL', 'sqlite:///'))
  13. class World(Model):
  14. __tablename__ = "world"
  15. id: so.Mapped[int] = so.mapped_column(primary_key=True)
  16. randomnumber: so.Mapped[int]
  17. def to_dict(self):
  18. return {"id": self.id, "randomNumber": self.randomnumber}
  19. class CachedWorld(Model):
  20. __tablename__ = "cachedworld"
  21. id: so.Mapped[int] = so.mapped_column(primary_key=True)
  22. randomnumber: so.Mapped[int]
  23. def to_dict(self):
  24. return {"id": self.id, "randomNumber": self.randomnumber}
  25. class Fortune(Model):
  26. __tablename__ = "fortune"
  27. id: so.Mapped[int] = so.mapped_column(primary_key=True)
  28. message: so.Mapped[str]
  29. def get_num_queries(request, name="queries"):
  30. try:
  31. num_queries = request.args.get(name, 1, type=int)
  32. except ValueError:
  33. num_queries = 1
  34. if num_queries < 1:
  35. return 1
  36. if num_queries > 500:
  37. return 500
  38. return num_queries
  39. def generate_ids(num_queries):
  40. return sample(range(1, 10001), num_queries)
  41. @app.route("/json")
  42. async def test_json(request):
  43. return {"message": "Hello, World!"}
  44. @app.route("/db")
  45. async def test_db(request):
  46. id = randint(1, 10000)
  47. async with db.Session() as session:
  48. world = await session.get(World, id)
  49. return world.to_dict()
  50. @app.route("/queries")
  51. async def test_queries(request):
  52. async with db.Session() as session:
  53. worlds = [(await session.get(World, id)).to_dict() for id in generate_ids(get_num_queries(request))]
  54. return worlds
  55. @app.route("/fortunes")
  56. async def test_fortunes(request):
  57. async with db.Session() as session:
  58. fortunes = list(await session.scalars(Fortune.select()))
  59. fortunes.append(Fortune(id=0, message="Additional fortune added at request time."))
  60. fortunes.sort(key=lambda f: f.message)
  61. return (
  62. await Template("fortunes.html").render_async(fortunes=fortunes),
  63. {'Content-Type': 'text/html; charset=utf-8'},
  64. )
  65. @app.route("/updates")
  66. async def test_updates(request):
  67. worlds = []
  68. ids = generate_ids(get_num_queries(request))
  69. ids.sort() # to avoid deadlocks
  70. async with db.begin() as session:
  71. for id in ids:
  72. world = await session.get(World, id)
  73. world.randomnumber = (randint(1, 9999) + world.randomnumber - 1) % 10000 + 1
  74. worlds.append(world.to_dict())
  75. return worlds
  76. @app.route("/plaintext")
  77. async def test_plaintext(request):
  78. return b"Hello, World!"
  79. @cached(cache={}, key=lambda session, id: hashkey(id))
  80. async def get_cached_world(session, id):
  81. return (await session.get(World, id)).to_dict()
  82. @app.route("/cached-queries")
  83. async def test_cached_queries(request):
  84. async with db.Session() as session:
  85. worlds = [await get_cached_world(session, id) for id in generate_ids(get_num_queries(request, 'count'))]
  86. return worlds