mirror of
https://github.com/langgenius/dify.git
synced 2025-07-03 23:28:18 +00:00
66 lines
1.9 KiB
Python
66 lines
1.9 KiB
Python
![]() |
import contextvars
|
||
|
from collections.abc import Iterator
|
||
|
from contextlib import contextmanager
|
||
|
from typing import TypeVar
|
||
|
|
||
|
from flask import Flask, g, has_request_context
|
||
|
|
||
|
T = TypeVar("T")
|
||
|
|
||
|
|
||
|
@contextmanager
|
||
|
def preserve_flask_contexts(
|
||
|
flask_app: Flask,
|
||
|
context_vars: contextvars.Context,
|
||
|
) -> Iterator[None]:
|
||
|
"""
|
||
|
A context manager that handles:
|
||
|
1. flask-login's UserProxy copy
|
||
|
2. ContextVars copy
|
||
|
3. flask_app.app_context()
|
||
|
|
||
|
This context manager ensures that the Flask application context is properly set up,
|
||
|
the current user is preserved across context boundaries, and any provided context variables
|
||
|
are set within the new context.
|
||
|
|
||
|
Note:
|
||
|
This manager aims to allow use current_user cross thread and app context,
|
||
|
but it's not the recommend use, it's better to pass user directly in parameters.
|
||
|
|
||
|
Args:
|
||
|
flask_app: The Flask application instance
|
||
|
context_vars: contextvars.Context object containing context variables to be set in the new context
|
||
|
|
||
|
Yields:
|
||
|
None
|
||
|
|
||
|
Example:
|
||
|
```python
|
||
|
with preserve_flask_contexts(flask_app, context_vars=context_vars):
|
||
|
# Code that needs Flask app context and context variables
|
||
|
# Current user will be preserved if available
|
||
|
```
|
||
|
"""
|
||
|
# Set context variables if provided
|
||
|
if context_vars:
|
||
|
for var, val in context_vars.items():
|
||
|
var.set(val)
|
||
|
|
||
|
# Save current user before entering new app context
|
||
|
saved_user = None
|
||
|
if has_request_context() and hasattr(g, "_login_user"):
|
||
|
saved_user = g._login_user
|
||
|
|
||
|
# Enter Flask app context
|
||
|
with flask_app.app_context():
|
||
|
try:
|
||
|
# Restore user in new app context if it was saved
|
||
|
if saved_user is not None:
|
||
|
g._login_user = saved_user
|
||
|
|
||
|
# Yield control back to the caller
|
||
|
yield
|
||
|
finally:
|
||
|
# Any cleanup can be added here if needed
|
||
|
pass
|