Search code examples
pythonflaskmultiprocessingplotly-dash

AttributeError: Can't pickle local object 'Flask.__init__.<locals>.<lambda>' due to multiprocessing


I'm trying to run a webserver inside a process. (Follow up to this question why I need to use multiprocessing: Stopping a dash server inside a thread does not work)

This is a minor example:

import logging
from dash import html, Dash
from flask import Flask
from multiprocessing import Process

############################## setup logger #############################
logging.basicConfig(
    level=logging.INFO,
    format="%(asctime)s [%(threadName)s, %(levelname)s] %(message)s",
    handlers=[
        logging.FileHandler("logs.log"),
        logging.StreamHandler()
    ]
)
#########################################################################

class test:
    def __init__(self):
        flask_app = Flask(__name__)
        self.app = Dash(__name__, server=flask_app)

        # Define a simple layout
        self.app.layout = html.Div(children=[
            html.H1(children='Hello Dash'),
            html.P(children='This is a simple Dash app for testing.'),
            html.Button('Click Me', id='button')
        ])

    def prepare_webserver(self):      
        self.server_proc = Process(target=self.run_server, 
                                   name='dash_server', 
                                   daemon=True)
        self.server_proc.start()

    # Function to run the server
    def run_server(self):
        logging.info("Starting Dash server")
        self.app.run_server(debug=True, use_reloader=False)

if __name__ == "__main__":
    sim_manager = test()
    sim_manager.prepare_webserver()

And I get the following error: Traceback (most recent call last):

  File "C:\Users\username\Desktop\Projects\python_framework\optimization_framework\test2.py", line 42, in <module>
    sim_manager.prepare_webserver()
  File "C:\Users\username\Desktop\Projects\python_framework\optimization_framework\test2.py", line 33, in prepare_webserver
    self.server_proc.start()
  File "C:\Users\username\AppData\Local\Programs\Python\Python310\lib\multiprocessing\process.py", line 121, in start
    self._popen = self._Popen(self)
  File "C:\Users\username\AppData\Local\Programs\Python\Python310\lib\multiprocessing\context.py", line 224, in _Popen
    return _default_context.get_context().Process._Popen(process_obj)
  File "C:\Users\username\AppData\Local\Programs\Python\Python310\lib\multiprocessing\context.py", line 327, in _Popen
    return Popen(process_obj)
  File "C:\Users\username\AppData\Local\Programs\Python\Python310\lib\multiprocessing\popen_spawn_win32.py", line 93, in __init__
    reduction.dump(process_obj, to_child)
  File "C:\Users\username\AppData\Local\Programs\Python\Python310\lib\multiprocessing\reduction.py", line 60, in dump
    ForkingPickler(file, protocol).dump(obj)
AttributeError: Can't pickle local object 'Flask.__init__.<locals>.<lambda>'
Traceback (most recent call last):
  File "<string>", line 1, in <module>
  File "C:\Users\username\AppData\Local\Programs\Python\Python310\lib\multiprocessing\spawn.py", line 116, in spawn_main
    exitcode = _main(fd, parent_sentinel)
  File "C:\Users\username\AppData\Local\Programs\Python\Python310\lib\multiprocessing\spawn.py", line 126, in _main
    self = reduction.pickle.load(from_parent)
EOFError: Ran out of input

Solution

  • Almost certainly self isn't pickleable, so you can't use self.run_server as a target.

    Try making run_server() a static method.

        @staticmethod
        # Function to run the server
        def run_server():
            server = Test()
            logging.info("Starting Dash server")
            server.app.run_server(debug=True, use_reloader=False)
    

    There is no reason the new process needs to be passed a pickled existing server rather than creating its own.