# import atexit
# import os
# import sys
# import time
# import traceback
# import enum
# from functools import partial as bind
#
#
# class Parallel:
#     def __init__(self, ctor, strategy):
#         self.worker = Worker(bind(self._respond, ctor), strategy, state=True)
#         self.callables = {}
#
#     def __getattr__(self, name):
#         if name.startswith("_"):
#             raise AttributeError(name)
#         try:
#             if name not in self.callables:
#                 self.callables[name] = self.worker(PMessage.CALLABLE, name)()
#             if self.callables[name]:
#                 return bind(self.worker, PMessage.CALL, name)
#             else:
#                 return self.worker(PMessage.READ, name)()
#         except AttributeError:
#             raise ValueError(name)
#
#     def __len__(self):
#         return self.worker(PMessage.CALL, "__len__")()
#
#     def close(self):
#         self.worker.close()
#
#     @staticmethod
#     def _respond(ctor, state, message, name, *args, **kwargs):
#         state = state or ctor
#         if message == PMessage.CALLABLE:
#             assert not args and not kwargs, (args, kwargs)
#             result = callable(getattr(state, name))
#         elif message == PMessage.CALL:
#             result = getattr(state, name)(*args, **kwargs)
#         elif message == PMessage.READ:
#             assert not args and not kwargs, (args, kwargs)
#             result = getattr(state, name)
#         return state, result
#
#
# class PMessage(enum.Enum):
#     CALLABLE = 2
#     CALL = 3
#     READ = 4
#
#
# class Worker:
#     initializers = []
#
#     def __init__(self, fn, strategy="thread", state=False):
#         if not state:
#             fn = lambda s, *args, fn=fn, **kwargs: (s, fn(*args, **kwargs))
#         inits = self.initializers
#         self.impl = {
#             "process": bind(ProcessPipeWorker, initializers=inits),
#             "daemon": bind(ProcessPipeWorker, initializers=inits, daemon=True),
#         }[strategy](fn)
#         self.promise = None
#
#     def __call__(self, *args, **kwargs):
#         self.promise and self.promise()  # Raise previous exception if any.
#         self.promise = self.impl(*args, **kwargs)
#         return self.promise
#
#     def wait(self):
#         return self.impl.wait()
#
#     def close(self):
#         self.impl.close()
#
#
# class ProcessPipeWorker:
#     def __init__(self, fn, initializers=(), daemon=False):
#         import multiprocessing
#         import cloudpickle
#
#         self._context = multiprocessing.get_context("spawn")
#         self._pipe, pipe = self._context.Pipe()
#         fn = cloudpickle.dumps(fn)
#         initializers = cloudpickle.dumps(initializers)
#         self._process = self._context.Process(
#             target=self._loop, args=(pipe, fn, initializers), daemon=daemon
#         )
#         self._process.start()
#         self._nextid = 0
#         self._results = {}
#         assert self._submit(Message.OK)()
#         atexit.register(self.close)
#
#     def __call__(self, *args, **kwargs):
#         return self._submit(Message.RUN, (args, kwargs))
#
#     def wait(self):
#         pass
#
#     def close(self):
#         try:
#             self._pipe.send((Message.STOP, self._nextid, None))
#             self._pipe.close()
#         except (AttributeError, IOError):
#             pass  # The connection was already closed.
#         try:
#             self._process.join(0.1)
#             if self._process.exitcode is None:
#                 try:
#                     os.kill(self._process.pid, 9)
#                     time.sleep(0.1)
#                 except Exception:
#                     pass
#         except (AttributeError, AssertionError):
#             pass
#
#     def _submit(self, message, payload=None):
#         callid = self._nextid
#         self._nextid += 1
#         self._pipe.send((message, callid, payload))
#         return Future(self._receive, callid)
#
#     def _receive(self, callid):
#         while callid not in self._results:
#             try:
#                 message, callid, payload = self._pipe.recv()
#             except (OSError, EOFError):
#                 raise RuntimeError("Lost connection to worker.")
#             if message == Message.ERROR:
#                 raise Exception(payload)
#             assert message == Message.RESULT, message
#             self._results[callid] = payload
#         return self._results.pop(callid)
#     #[todo] start
#     # def _receive(self, callid):
#     #     while callid not in self._results:
#     #         try:
#     #             message, callid, payload = self._pipe.recv()
#     #         except (OSError, EOFError) as e:
#     #             # 记录子进程的 exitcode 和堆栈信息
#     #             proc_exit = None
#     #             try:
#     #                 proc_exit = self._process.exitcode
#     #             except Exception:
#     #                 print(f"[DEBUG] worker PID={self._process.pid}, exitcode={proc_exit}", flush=True)
#     #             if proc_exit is not None:
#     #                 raise RuntimeError(
#     #                     f"Lost connection to worker. Child process exitcode={proc_exit}. Original error: {e}")
#     #             else:
#     #                 raise RuntimeError(f"Lost connection to worker (pipe closed). Original error: {e}")
#     #         if message == Message.ERROR:
#     #             # 如果是子进程发送的错误，打印堆栈信息
#     #             raise Exception(f"Worker error:\n{payload}")
#     #         assert message == Message.RESULT, message
#     #         self._results[callid] = payload
#     #     return self._results.pop(callid)
#
#     # @staticmethod
#     # def _loop(pipe, function, initializers):
#     #     import cloudpickle, traceback, sys, os, time
#     #
#     #     pid = os.getpid()
#     #     log_file = f"./tmp/worker_{pid}.log"
#     #     err_file = f"./tmp/worker_{pid}_err.log"
#     #
#     #     def log(msg):
#     #         with open(log_file, "a") as f:
#     #             f.write(f"[{time.strftime('%H:%M:%S')}] {msg}\n")
#     #             f.flush()
#     #
#     #     try:
#     #         log("worker starting")
#     #
#     #         initializers = cloudpickle.loads(initializers)
#     #         log("initializers deserialized")
#     #
#     #         function = cloudpickle.loads(function)
#     #         log("function deserialized")
#     #
#     #         # 执行初始函数
#     #         [fn() for fn in initializers]
#     #         log("initializers executed")
#     #
#     #         state = None
#     #         while True:
#     #             if not pipe.poll(0.1):
#     #                 continue
#     #             message, callid, payload = pipe.recv()
#     #
#     #             if message.name == "OK":
#     #                 pipe.send((message.RESULT, callid, True))
#     #             elif message.name == "STOP":
#     #                 log("received STOP; exiting loop")
#     #                 return
#     #             elif message.name == "RUN":
#     #                 args, kwargs = payload
#     #                 state, result = function(state, *args, **kwargs)
#     #                 pipe.send((message.RESULT, callid, result))
#     #             else:
#     #                 raise KeyError(f"Invalid message: {message}")
#     #
#     #     except Exception:
#     #         stacktrace = "".join(traceback.format_exception(*sys.exc_info()))
#     #         log("EXCEPTION:\n" + stacktrace)
#     #         try:
#     #             with open(err_file, "w") as f:
#     #                 f.write(stacktrace)
#     #         except Exception:
#     #             pass
#     #         try:
#     #             pipe.send(("ERROR", None, stacktrace))
#     #         except Exception:
#     #             pass
#     #         return
#     #     finally:
#     #         try:
#     #             pipe.close()
#     #         except Exception:
#     #             pass
#     #         log("worker closed")
#     #[todo] end
#
#
#
#     @staticmethod
#     def _loop(pipe, function, initializers):
#         try:
#             callid = None
#             state = None
#             import cloudpickle
#
#             initializers = cloudpickle.loads(initializers)
#             function = cloudpickle.loads(function)
#             [fn() for fn in initializers]
#             while True:
#                 if not pipe.poll(0.1):
#                     continue  # Wake up for keyboard interrupts.
#                 message, callid, payload = pipe.recv()
#                 if message == Message.OK:
#                     pipe.send((Message.RESULT, callid, True))
#                 elif message == Message.STOP:
#                     return
#                 elif message == Message.RUN:
#                     args, kwargs = payload
#                     state, result = function(state, *args, **kwargs)
#                     pipe.send((Message.RESULT, callid, result))
#                 else:
#                     raise KeyError(f"Invalid message: {message}")
#         except (EOFError, KeyboardInterrupt):
#             return
#         except Exception:
#             stacktrace = "".join(traceback.format_exception(*sys.exc_info()))
#             print(f"Error inside process worker: {stacktrace}.", flush=True)
#             pipe.send((Message.ERROR, callid, stacktrace))
#             return
#         finally:
#             try:
#                 pipe.close()
#             except Exception:
#                 pass
#
#
# class Message(enum.Enum):
#     OK = 1
#     RUN = 2
#     RESULT = 3
#     STOP = 4
#     ERROR = 5
#
#
# class Future:
#     def __init__(self, receive, callid):
#         self._receive = receive
#         self._callid = callid
#         self._result = None
#         self._complete = False
#
#     def __call__(self):
#         if not self._complete:
#             self._result = self._receive(self._callid)
#             self._complete = True
#         return self._result
#
#
# class Damy:
#     def __init__(self, env):
#         self._env = env
#
#     def __getattr__(self, name):
#         return getattr(self._env, name)
#
#     def step(self, action):
#         return lambda: self._env.step(action)
#
#     def reset(self):
#         return lambda: self._env.reset()

#>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>
#[todo] start
import atexit
import os
import sys
import time
import traceback
import enum
from functools import partial as bind
import cloudpickle
import multiprocessing as mp

class PMessage(enum.Enum):
    CALLABLE = 2
    CALL = 3
    READ = 4

class Message(enum.Enum):
    OK = 1
    RUN = 2
    RESULT = 3
    STOP = 4
    ERROR = 5

class Parallel:
    def __init__(self, ctor_or_instance, strategy="process"):
        def make_worker_fn(ctor_or_instance):
            if callable(ctor_or_instance):
                def worker_fn(state, message, name, *args, **kwargs):
                    if state is None:
                        state = ctor_or_instance()
                    if message == PMessage.CALLABLE:
                        result = callable(getattr(state, name))
                    elif message == PMessage.CALL:
                        result = getattr(state, name)(*args, **kwargs)
                    elif message == PMessage.READ:
                        result = getattr(state, name)
                    else:
                        raise KeyError("Unknown PMessage: %r" % (message,))
                    return state, result
                return worker_fn
            else:
                instance = ctor_or_instance
                def worker_fn(state, message, name, *args, **kwargs):
                    if state is None:
                        state = instance
                    if message == PMessage.CALLABLE:
                        result = callable(getattr(state, name))
                    elif message == PMessage.CALL:
                        result = getattr(state, name)(*args, **kwargs)
                    elif message == PMessage.READ:
                        result = getattr(state, name)
                    else:
                        raise KeyError("Unknown PMessage: %r" % (message,))
                    return state, result
                return worker_fn

        worker_fn = make_worker_fn(ctor_or_instance)
        self.worker = Worker(worker_fn, strategy=strategy, state=True)
        self.callables = {}

    def __getattr__(self, name):
        if name.startswith("_"):
            raise AttributeError(name)
        try:
            if name not in self.callables:
                is_callable = self.worker(PMessage.CALLABLE, name)()
                self.callables[name] = is_callable
            if self.callables[name]:
                return bind(self.worker, PMessage.CALL, name)
            else:
                return self.worker(PMessage.READ, name)()
        except AttributeError:
            raise ValueError(name)

    def __len__(self):
        return self.worker(PMessage.CALL, "__len__")()

    def close(self):
        self.worker.close()

class Worker:
    initializers = []

    def __init__(self, fn, strategy="thread", state=False):
        if not state:
            fn = lambda s, *args, fn=fn, **kwargs: (s, fn(*args, **kwargs))
        inits = self.initializers
        impl_ctor = {
            "process": bind(ProcessPipeWorker, initializers=inits),
            "daemon": bind(ProcessPipeWorker, initializers=inits, daemon=True),
            # "thread": bind(ThreadWorker, initializers=inits)  # 作为备选
        }[strategy]
        self.impl = impl_ctor(fn)
        self.promise = None

    def __call__(self, *args, **kwargs):
        self.promise and self.promise()
        self.promise = self.impl(*args, **kwargs)
        return self.promise

    def wait(self):
        return self.impl.wait()

    def close(self):
        self.impl.close()

class ProcessPipeWorker:
    def __init__(self, fn, initializers=(), daemon=False):
        self._context = mp.get_context("spawn")
        self._pipe, pipe = self._context.Pipe()
        fn = cloudpickle.dumps(fn)
        initializers = cloudpickle.dumps(initializers)
        self._process = self._context.Process(
            target=self._loop, args=(pipe, fn, initializers), daemon=daemon
        )
        self._process.start()
        self._nextid = 0
        self._results = {}
        assert self._submit(Message.OK)()
        atexit.register(self.close)

    def __call__(self, *args, **kwargs):
        return self._submit(Message.RUN, (args, kwargs))

    def wait(self):
        if self._process is not None:
            self._process.join()
        return None

    def close(self):
        try:
            self._pipe.send((Message.STOP, self._nextid, None))
            self._pipe.close()
        except (AttributeError, IOError):
            pass
        try:
            self._process.join(0.1)
            if self._process.exitcode is None:
                try:
                    os.kill(self._process.pid, 9)
                    time.sleep(0.1)
                except Exception:
                    pass
        except (AttributeError, AssertionError):
            pass

    def _submit(self, message, payload=None):
        callid = self._nextid
        self._nextid += 1
        try:
            self._pipe.send((message, callid, payload))
        except (BrokenPipeError, OSError) as e:
            raise RuntimeError("Failed to send to worker pipe") from e
        return Future(self._receive, callid)

    def _receive(self, callid):
        while callid not in self._results:
            try:
                message, rcallid, payload = self._pipe.recv()
            except (OSError, EOFError):
                raise RuntimeError("Lost connection to worker.")
            if message == Message.ERROR:
                raise Exception(payload)
            assert message == Message.RESULT, message
            self._results[rcallid] = payload
        return self._results.pop(callid)

    @staticmethod
    def _loop(pipe, function_ser, initializers_ser):
        callid = None
        try:
            initializers = cloudpickle.loads(initializers_ser)
            function = cloudpickle.loads(function_ser)
            for fn in initializers:
                try:
                    fn()
                except Exception as e:
                    print(f"Initialization error: {e}", file=sys.stderr)
            state = None
            while True:
                message, callid, payload = pipe.recv()
                if message == Message.OK:
                    pipe.send((Message.RESULT, callid, True))
                elif message == Message.STOP:
                    return
                elif message == Message.RUN:
                    call_args, call_kwargs = payload
                    try:
                        state, result = function(state, *call_args, **call_kwargs)
                        pipe.send((Message.RESULT, callid, result))
                    except Exception as e:
                        stacktrace = "".join(traceback.format_exception(*sys.exc_info()))
                        print(f"Error inside process worker: {stacktrace}", flush=True)
                        pipe.send((Message.ERROR, callid, stacktrace))
                else:
                    raise KeyError(f"Invalid message: {message}")
        except Exception:
            stacktrace = "".join(traceback.format_exception(*sys.exc_info()))
            with open("./worker_error.log", "a") as f:
                f.write(stacktrace + "\n\n")
            try:
                pipe.send((Message.ERROR, callid, stacktrace))
            except Exception:
                pass
            return
        except (EOFError, KeyboardInterrupt):
            return
        except Exception as e:
            stacktrace = "".join(traceback.format_exception(*sys.exc_info()))
            print(f"Worker loop exception: {stacktrace}", file=sys.stderr)
            try:
                pipe.send((Message.ERROR, callid, stacktrace))
            except Exception:
                pass
            return
        finally:
            try:
                pipe.close()
            except Exception:
                pass

class Future:
    def __init__(self, receive, callid):
        self._receive = receive
        self._callid = callid
        self._result = None
        self._complete = False

    def __call__(self):
        if not self._complete:
            self._result = self._receive(self._callid)
            self._complete = True
        return self._result

class Damy:
    def __init__(self, env):
        self._env = env

    def __getattr__(self, name):
        return getattr(self._env, name)

    def step(self, action):
        return lambda: self._env.step(action)

    def reset(self):
        return lambda: self._env.reset()

#[todo] end