Context¶
The Context
is a beauty of an object unique to Lilya. The context
is a parameter that
can be used inside the function handlers and provides additional information
that you might need for any particular reason.
Very similar approach to Request in terms of implementation but not the same.
In fact, a context
contains also the request and information about the handler itself.
Importing is as simple as:
from lilya.context import Context
The Context
class¶
You can see the context
as the request context
of a given handler. This also means, when
a function handler (view) is declared all the information passed to it
is automatically accessible via context.handler
parameter.
from lilya.context import Context
from lilya.responses import make_response
def home(context: Context):
"""
Accessing the context object.
"""
return make_response(content=None, status_code=204)
The context
also provides access to the request
object as well as the
application settings and other functions.
This means, if you want to pass a request
and context
you actually only need the context
directly as the request is already available inside but you can still pass both anyway.
Example
from lilya.apps import Lilya
from lilya.context import Context
from lilya.routing import Path
def read_context(context: Context, id: str):
host = context.request.client.host
context_data = context.get_context_data()
context.add_to_context("name", "Lilya")
context_data = context.get_context_data()
context_data.update({
"host": host, "user_id": id
})
return context_data
app = Lilya(
routes=[
Path("/users/{id}", read_request)
]
)
The context
becomes especially valuable when you need to retrieve handler
information that is
unavailable after the handler's instantiation. For instance, it proves useful when accessing
context.settings
to retrieve application settings, offering a versatile approach to accessing them.
Attributes¶
Handler¶
The function handler that is responsible for the request.
from lilya.context import Context
from lilya.responses import Response
def home(context: Context):
handler = context.handler
return Response("Ok")
Request¶
The request being used for the scope of the handler.
from lilya.context import Context
from lilya.responses import Response
def home(context: Context):
request = context.request
return Response("Ok")
User¶
The user allocated in the request.
from lilya.context import Context
from lilya.responses import Response
def home(context: Context):
user = context.user
return Response("Ok")
User¶
The user allocated to the request.
from lilya.context import Context
from lilya.responses import Response
def home(context: Context):
user = context.user
return Response("Ok")
Settings¶
The settings being used by the Lilya application.
This can be the global settings or if a settings_module
was provided, returns that
same settings.
from lilya.context import Context
from lilya.responses import Response
def home(context: Context):
settings = context.settings
return Response("Ok")
Scope¶
The scope of the application handler.
from lilya.context import Context
from lilya.responses import Response
def home(context: Context):
scope = context.scope
return Response("Ok")
Application¶
The Lilya application.
from lilya.context import Context
from lilya.responses import Response
def home(context: Context):
app = context.app
return Response("Ok")
Methods¶
The context
provides also different functions to manipulate the object.
get_context_data¶
The context of the application. This can be particularly useful when working with templates.
from lilya.context import Context
from lilya.responses import Response
def home(context: Context):
context = context.get_context_data()
return Response("Ok")
add_to_context¶
Adding values into the current context.
from lilya.context import Context
from lilya.responses import Response
def home(context: Context):
context.add_to_context("call", "ok")
return Response("Ok")
path_for¶
Retrives the path of a specific handler.
from lilya.context import Context
from lilya.responses import Response
def home(context: Context):
url = context.path_for("/home")
return Response(url)
The G
object¶
This object is a pretty cool one, it is what allows you to have a global request context across the lifecycle.
Imagine the following. You have a deceorator that validates a login and wants to store arbitrary data inside a request global context to be used later on and once the request is closed, then its cleared.
This is where the G
object comes to play. Imagine the g
of Flask, same principle.
This object is globally used and on each request it will be "reset", meaning, it only lives in the request context of the call.
from lilya.apps import Lilya
from lilya.context import g
from lilya.middleware import DefineMiddleware
from lilya.middleware.global_context import GlobalContextMiddleware
from lilya.routing import Path
def populate_g():
g.name = "Lilya"
g.age = 25
async def show_g():
return g.store
app = Lilya(
routes=[Path("/show", show_g)]
)
populate_g()
This of course is a very simple example and probably not like how you would like to use but it
explains how to use the g
. After every request, the g
is cleared.
Imagine now you want to use it in a decorator
.
import functools
from lilya.apps import Lilya
from lilya.context import g
from lilya.middleware import DefineMiddleware
from lilya.middleware.global_context import GlobalContextMiddleware
from lilya.routing import Path
def user_check(func):
@functools.wraps(func)
async def wrapper(*args, **kwargs):
if not hasattr(g, "user"):
g.user = "not logged in"
return await func(*args, **kwargs)
return wrapper
@user_check
async def show_g():
return g.store
app = Lilya(
routes=[Path("/show", show_g)]
)
This example is closer to the reality of the use of a g
like validating a login against anything
in your APIs.
The RequestContext
object¶
This is a very useful object that acts a lazy loader of the request object without the need of explicitly declaring it inside an handler.
Danger
When using the request_context
object, you must install the RequestContextMiddleware
or an ImproperlyConfigured
exception is raised.
from lilya.apps import Lilya
from lilya.context import request_context
from lilya.middleware import DefineMiddleware
from lilya.middleware.request_context import RequestContextMiddleware
from lilya.routing import Path
async def show_request_context():
return {"url": str(request_context.url)}
app = Lilya(routes=[
Path('/show', show_request_context)
],
middleware=[DefineMiddleware(RequestContextMiddleware)],
)
The Session
object¶
This is also a very useful object that can be used in a lazy load mode across the request lifecycle. If you are familiar
with the flask session
, this one acts in the same fashion.
Danger
When using the session
object, you must install the SessionMiddleware
and the SessionContextMiddleware in this specific order as the order matters or an ImproperlyConfigured
exception is raised.
from lilya.apps import Lilya
from lilya.context import session
from lilya.middleware import DefineMiddleware
from lilya.middleware.session_context import SessionContextMiddleware
from lilya.middleware.sessions import SessionMiddleware
from lilya.routing import Path
async def home():
session["visits"] = session.get("visits", 0) + 1
return {"visits": session["visits"]}
app = Lilya(routes=[
Path('/show', home)
],
middleware=[
DefineMiddleware(SessionMiddleware, secret='my_secret'),
DefineMiddleware(SessionContextMiddleware),
],
)