import argparse


def str_to_bool(arg):
    """Convert an argument string into its boolean value.

    Args:
        arg: String representing a bool.

    Returns:
        Boolean value for the string.
    """
    if arg.lower() in ('yes', 'true', 'True', 't', 'y', '1'):
        return True
    elif arg.lower() in ('no', 'false', 'False', 'f', 'n', '0'):
        return False
    else:
        raise argparse.ArgumentTypeError('Boolean value expected.')


def str_to_list(arg):
    """Convert an argument string into a list.

    Args:
        arg: String representing a list.

    Returns:
        List value for the string.
    """
    if arg is None:
        return []
    else:
        return [int(x) for x in arg.split(',')]


def generate_parser():
    parser = argparse.ArgumentParser(description="Argument parser for the experiment.")

    # --------------------------------
    # General experiment parameters
    parser.add_argument('--data', type=str, default="GBM",
                        choices=["HFCR", "PBC", "GBM", "GBSG", "METABRIC", "NACD", "SUPPORT", "MIMIC-IV_all"],
                        help="Dataset name for the experiment.")
    parser.add_argument('--n_exp', type=int, default=10,
                        help="Number of experiments to run.")
    parser.add_argument('--seed', type=int, default=0,
                        help="Random seed.")
    parser.add_argument('--model', type=str, default="IWSG",
                        choices=["RSF", "GB", "DeepHit", "CoxTime", "Nnet-survival", "CQRNN", "SODEN", "DCSurvival", "IWSG",  #baselines
                                 "CoxPH", "MTLR", "WeibullAFT", "LogLogisticAFT",
                                 "CoxPH-2B", "MTLR-2B", "WeibullAFT-2B", "LogLogisticAFT-2B",
                                 "CoxPH-LDR", "MTLR-LDR", "WeibullAFT-LDR", "LogLogisticAFT-LDR",
                                 ],
                        help="Model name.")
    # --------------------------------
    # LDR parameters.
    parser.add_argument('--alpha', type=float, default=10,
                        help="Balance weight for the Orthogonality term.")
    parser.add_argument('--beta', type=float, default=0.01,
                        help="Balance weight for the IPM term.")
    parser.add_argument('--ipm', type=str, default='mmd-rbf',
                        help="IPM function. Possible values: 'mmd2-lin', 'mmd-lin', 'mmd2-rbf', 'mmd-rbf'")
    parser.add_argument('--e_dims', type=str_to_list, default=[],
                        help="Hidden neurons of the event distribution networks. No space between numbers.")
    parser.add_argument('--c_dims', type=str_to_list, default=[],
                        help="Hidden neurons of the censoring distribution networks. No space between numbers.")

    # --------------------------------
    # Network Structure parameters. Not used for RSF and GB.
    parser.add_argument('--neurons', type=str_to_list, default=[64, 64],
                        help="Hidden neurons in neural network. No space between numbers."
                             "For SALaD, these are the hidden dims in representation networks.")
    parser.add_argument('--norm', type=str_to_bool, default=True,
                        help="Whether to use batch norm in neural network.")
    parser.add_argument('--dropout', type=float, default=0.6,
                        help="Dropout rate.")
    parser.add_argument('--activation', type=str, default='ReLU',
                        help="Activation function. Possible values: 'Sigmoid', 'Tanh', 'ReLU', "
                             "'LeakyReLU', 'PReLU', 'ELU', 'SELU',"
                             "See https://pytorch.org/docs/stable/nn.html for more choices.")
    parser.add_argument('--n_quantiles', type=int, default=9,
                        help="Number of quantiles to use for the CQRNN.",
                        choices=[4, 9, 19, 39, 49, 99])
    parser.add_argument('--mono_method', type=str, default="bootstrap",
                        choices=["ceil", "floor", "bootstrap"],
                        help="Method to make the CQRNN prediction monotonic.")
    # --------------------------------
    # Training parameters. Not used for RSF and GB.
    parser.add_argument('--optimizer', type=str, default="AdamW",
                        help="Optimizer for training. Possible values: 'SGD', 'Adam', ''AdamW', etc.")
    parser.add_argument('--n_epochs', type=int, default=10000,
                        help="Maximum number of training epochs. ")
    parser.add_argument('--early_stop', type=str_to_bool, default=True,
                        help="Whether to use early stop during training.")
    parser.add_argument('--batch_size', type=int, default=256,
                        help="Batch size for training.")
    parser.add_argument('--lr', type=float, default=0.001,
                        help="Learning rate.")
    parser.add_argument('--weight_decay', type=float, default=0.01,
                        help="Term for weight decay (L2 penalty).")
    # --------------------------------
    # Display parameters.
    parser.add_argument('--verbose', type=str_to_bool, default=True,
                        help="Whether to print training information.")

    args = parser.parse_args()
    return args
