I am planning to deploy a Dash web app through my Flask server that supports multiple “different” web apps with entirely unique layouts and callbacks. Each web app layout should be user specific. I’ve built out most of this functionality that I will provide thorough context about. My hope is that others have been faced with similar use cases or may be able to provide some suggestions about how I may be able to better design my system as production Dash applications are still very new to me and I come from a very Python focused background.
My Current Implementation and Use Case
I set up my Flask server with my single Dash application attached like so:
app = Flask(__name__)
app.secret_key = service_constants.SESSION_SECRET_KEY
dash_app = Dash(
__name__,
server=app,
url_base_pathname=C_GLOBAL_DASH_APP_PATH,
on_error=websmith_error_handler,
assets_folder=DASH_ASSETS_DIR,
)
# Initialize the layout with handling for stylesheet updates and placeholder content
dash_app.layout = html.Div(
children=[
create_default_layout(),
html.H1("Placeholder content."),
]
)
create_default_layout includes a couple components that all “different” possible web apps should include, like a Modal to display errors that may have occurred in callbacks.
Now I will explain what I mean by the “different” web apps I have. In my repo, I have a registry of Python objects abiding by an Abstract Class interface defined by my IDashApp class that has abstract methods layout(self) → html.Div and register_callbacks(app: Dash). register_callbacks has function definitions within it containing the @app.callback() definitions. I have several different implementations of the IDashApp class for things like tracking status of batch jobs that are running, analyzing results of models generated from those jobs, among a few others. Each of these concrete implementations of IDashApp has entirely different layouts and callbacks, and should also have user-specific states. So I may want to track the execution status of Job #1, but my coworker may want to track the execution status of Job #2… we both should be able to view this web app independently of each other at the same time.
Now to explain how these different IDashApp classes get initialized and tie back to the original dash_app on my Flask server. I have an @app.before_request handler on my Flask service that will check for get requests with paths containing my dash app base url of C_GLOBAL_DASH_APP_PATH. If it does, then I’ll pick off the relevant URL and query parameters to initialize the concrete IDashApp the user is requesting. Once I have the initialized object, I update the layout method of the dash_app and ensure the callbacks are registered from the concrete web app: dash_app.layout = concrete_web_app.layout(), and concrete_web_app.register_callbacks(app=dash_app).
This largely works as expected when testing locally and as a single user, but some problems quickly arise with more users and just with the underlying design of my system. A side point worth noting is that I have IdP authentication going on behind the scenes, so I have authenticated users “logged in” or maintaining an active valid session when these web apps are requested. I store this session info in my database using FlaskSQLAlchemy and flask_sessionstore.
My Concerns Needing Addressing with this Approach
- One of the main concerns I have here is the lack of statelessness of these web apps. I initialize Python objects in memory that I am expecting to be unique to users and persist across requests. I find that one user (or a single user with multiple browsers) can effect the callbacks and layouts of another when the “same” web app is requested.
- I am wondering if there may be able to store some of these session specific objects in my database to be sure the correct ones are referenced.
- Or maybe I need to switch to multi page layouts which it seems like may play nicer with session specific callbacks?
- Is the idea of initializing these objects from a registry based on the request entirely foolish and unnecessary if it could be solved with multi page applications?
I would very much appreciate any input or feedback others have about my use case. This community is so great and I appreciate any and all input from you guys!!
Noteworthy References I have Looked Into