fastapi-g-context
is a Python module that provides a simple mechanism for managing global variables with context isolation in FastAPI
applications. It is designed to ensure that each request operates within its own isolated context, preventing data leakage between requests.
While this behaviour can be achieved in other ways as well, this module tries to mimic the flask
's global g
object, so that flask
users find it easier to migrate their logic when moving to FastAPI
.
- Context Isolation: Each request operates in a fresh context, ensuring that global variables do not interfere with each other.
- Easy-to-Use Interface: Get, set, and manage global variables with a simple and intuitive API, while resembling the usage of
flask
'sg
object to make it easier for users that are switching toFastAPI
. - Middleware Integration: Integrates seamlessly with FastAPI as a pure ASGI lightweight middleware.
To install fastapi-g-context
simply use pip:
pip install fastapi-g-context
To use fastapi-g-context
in your FastAPI application, follow these simple steps:
- Install the package: Ensure you have
fastapi-g-context
installed. You can add it to your requirements.txt or install it directly. - Import the necessary components from
fastapi_g_context
:GlobalsMiddleware
g
object (the global context manager) .
- Add the GlobalsMiddleware to your FastAPI application. This middleware ensures that each request operates with a fresh context for global variables.
- Set Global Variables: You can set and manipulate global variables using the g object. These variables are request-scoped and won't leak between requests.
- Most usual use case would be to use that
g
object to set dependencies for your app's handlers via theFastAPI
's dependency injection system.
from fastapi import FastAPI, Depends
from fastapi_g_context import GlobalsMiddleware, g
app = FastAPI()
app.add_middleware(GlobalsMiddleware)
async def set_globals() -> None:
g.username = "JohnDoe"
g.request_id = "123456"
g.is_admin = True
@app.get("/", dependencies=[Depends(set_globals)])
async def info():
return {"username": g.username, "request_id": g.request_id, "is_admin": g.is_admin}
The global object g
provides a mechanism to manage global variables with context isolation. Below are the methods and attributes available in this class:
key in g
Check whether an attribute is present in the global context.
- Parameters:
key
(str): The name of the attribute to check.
- Return Type:
bool
g.get(name, default=None)
Get an attribute by name, or return a default value if the attribute is not present. This works similarly to dict.get()
.
- Parameters:
name
(str): Name of the attribute to retrieve.default
(Any | None): Value to return if the attribute is not present (default isNone
).
- Return Type:
Any
g.pop(name, default=None)
Get and remove an attribute by name. This works similarly to dict.pop()
.
- Parameters:
name
(str): Name of the attribute to pop.default
(Any | None): Value to return if the attribute is not present (default isNone
).
- Return Type:
Any
g.keys()
Return an iterator over the attribute names in the global context.
- Return Type:
Iterator[str]
g.values()
Return an iterator over the attribute values in the global context.
- Return Type:
Iterator[Any]
g.items()
Return an iterator over the attribute name-value pairs in the global context.
- Return Type:
Iterator[tuple[str, Any]]
g.to_dict()
Return all attributes and their current values as a dictionary.
- Return Type:
Dict[str, Any]
g.clear()
Clear all attributes from the global context.
- Return Type:
None
g.__getattr__(name)
Retrieve the value of an attribute by name. Raises an AttributeError
if the attribute is not found.
- Parameters:
name
(str): Name of the attribute to retrieve.
- Return Type:
Any
g.__setattr__(name, value)
Set the value of an attribute by name. Creates the attribute if it does not already exist.
- Parameters:
name
(str): Name of the attribute to set.value
(Any): Value to set for the attribute.
- Return Type:
None
To demonstrate the capabilities of the fastapi_g_context module, here's an example FastAPI application.
from fastapi import FastAPI, Depends
from fastapi.responses import JSONResponse
from fastapi_g_context import GlobalsMiddleware, g
import logging
logging.basicConfig(level=logging.INFO)
app = FastAPI()
app.add_middleware(GlobalsMiddleware)
async def set_globals() -> None:
g.username = "JohnDoe"
g.request_id = "123456"
g.is_admin = True
g.to_pop = "dispensable"
@app.get("/", dependencies=[Depends(set_globals)])
async def info():
# Iterate over globals like a dictionary
for name, value in g.items():
logging.info(f"Global variable '{name}' has value: {value}")
# Check for attributes/variables existence in globals
logging.info(f"'username' is present in globals: {'username' in g}")
logging.info(f"'password' is present in globals: {'password' in g}")
# Usage of .get() with a default value or not
logging.info(f"'non_existing_with_default' is present in globals: {'non_existing_with_default' in g}")
non_existing_get_with_default = g.get("non_existing_with_default", "non_existing_default")
logging.info(f"'non_existing_with_default' key has value: {non_existing_get_with_default}")
# Default value defaults to None if not provided
logging.info(f"'non_existing_none' is present in globals: {'non_existing_none' in g}")
non_existing_get = g.get("non_existing_none")
logging.info(f"'non_existing_none' key has value: {non_existing_get}")
# Usage of .pop() with a default value or not
logging.info(f"'non_existing_with_default' is present in globals: {'non_existing_with_default' in g}")
non_existing_pop_with_default = g.pop("non_existing_with_default", "non_existing_default")
logging.info(f"'non_existing_with_default' key has value: {non_existing_pop_with_default}")
# Default value defaults to None if not provided
logging.info(f"'non_existing_none' is present in globals: {'non_existing_none' in g}")
non_existing_pop = g.pop("non_existing_none")
logging.info(f"'non_existing_none' key has value: {non_existing_pop}")
# Accessing a non-set variable directly like an attribute will lead to an AttributeError
try:
logging.info(f"'not_existing' is present in globals: {'not_existing' in g}")
logging.info(g.not_existing)
except AttributeError as e:
logging.exception(e)
# Retrieve an already set variable as an attribute
logging.info(f"'username' key has value: {g.username}")
# Set a new variable
logging.info(f"'new_key' is present in globals: {'new_key' in g}")
g.new_key = "new_value"
logging.info(f"'new_key' is present in globals: {'new_key' in g}")
# Pop an existing variable
logging.info(f"'to_pop' is present in globals: {'to_pop' in g}")
key_to_pop = g.pop("to_pop")
logging.info(f"'to_pop' key has value: {key_to_pop}")
logging.info(f"'to_pop' is present in globals: {'to_pop' in g}")
# List all global variable names
globals_keys = list(g.keys())
# List all global variable values
globals_values = list(g.values())
# Get all global variables as a dictionary
global_dict = g.to_dict()
return JSONResponse(
content={
"global_keys": globals_keys,
"global_values": globals_values,
"global_dict": global_dict
}
)
After running this example app with uvicorn example:app --host 0.0.0.0 --port 8080 --reload
and requesting the only endpoint that it serves:
- The logs that are printed out:
INFO:root:Global variable 'username' has value: JohnDoe
INFO:root:Global variable 'request_id' has value: 123456
INFO:root:Global variable 'is_admin' has value: True
INFO:root:Global variable 'to_pop' has value: dispensable
INFO:root:'username' is present in globals: True
INFO:root:'password' is present in globals: False
INFO:root:'non_existing_with_default' is present in globals: False
INFO:root:'non_existing_with_default' key has value: non_existing_default
INFO:root:'non_existing_none' is present in globals: False
INFO:root:'non_existing_none' key has value: None
INFO:root:'non_existing_with_default' is present in globals: False
INFO:root:'non_existing_with_default' key has value: non_existing_default
INFO:root:'non_existing_none' is present in globals: False
INFO:root:'non_existing_none' key has value: None
INFO:root:'not_existing' is present in globals: False
ERROR:root:'not_existing' variable does not exist in globals, make sure to set it before trying to use it
Traceback (most recent call last):
File "/path/to/example.py", line 53, in info
logging.info(g.not_existing)
^^^^^^^^^^^^^^
File "/path/to/fastapi_g_context.py", line 41, in __getattr__
raise AttributeError(
AttributeError: 'not_existing' variable does not exist in globals, make sure to set it before trying to use it
INFO:root:'username' key has value: JohnDoe
INFO:root:'new_key' is present in globals: False
INFO:root:'new_key' is present in globals: True
INFO:root:'to_pop' is present in globals: True
INFO:root:'to_pop' key has value: dispensable
INFO:root:'to_pop' is present in globals: False
- The response returned:
{
"global_keys": [
"username",
"request_id",
"is_admin",
"new_key"
],
"global_values": [
"JohnDoe",
"123456",
true,
"new_value"
],
"global_dict": {
"username": "JohnDoe",
"request_id": "123456",
"is_admin": true,
"new_key": "new_value"
}
}