from federatedscope.core.fed_runner import StandaloneRunner, DistributedRunner
from federatedscope.core.parallel.parallel_runner import \
    StandaloneMultiGPURunner


def get_runner(server_class,
               client_class,
               config,
               client_configs=None,
               data=None):
    """
    Instantiate a runner based on a configuration file

    Args:
        server_class: server class
        client_class: client class
        config: configurations for FL, see ``federatedscope.core.configs``
        client_configs: client-specific configurations

    Returns:
        An instantiated FedRunner to run the FL course.

    Note:
      The key-value pairs of built-in runner and source are shown below:
        =============================  ===============================
        Mode                                          Source
        =============================  ===============================
        ``standalone``                 ``core.fed_runner.StandaloneRunner``
        ``distributed``                ``core.fed_runner.DistributedRunner``
        ``standalone(process_num>1)``  ``core.auxiliaries.parallel_runner.``
                                       ``StandaloneMultiGPURunner``
        =============================  ===============================
    """

    mode = config.federate.mode.lower()
    process_num = config.federate.process_num

    if mode == 'standalone':
        if process_num <= 1:
            runner_cls = StandaloneRunner
        else:
            runner_cls = StandaloneMultiGPURunner
    elif mode == 'distributed':
        runner_cls = DistributedRunner

    # federated dataset might change the number of clients
    # thus, we allow the creation procedure of dataset to modify the global
    # cfg object
    if runner_cls is StandaloneMultiGPURunner:
        data = None

    return runner_cls(data=data,
                      server_class=server_class,
                      client_class=client_class,
                      config=config,
                      client_configs=client_configs)
