import math
from functools import partial

import torch
from torch.optim.lr_scheduler import LambdaLR
import transformers


def get_scheduler(
        optimizer,
        *,
        scheduler_type,
        num_training_steps,
        warmup_steps,
        min_lr_ratio,
        cycle_length=None,
        restart_warmup_steps=None,
        adjust_step=0,
        last_epoch=-1,
        update_gap=None
):
    if adjust_step != 0 and scheduler_type != "cosine_restarts":
        raise ValueError("adjust_step is only supported for cosine_restarts scheduler")

    if scheduler_type == "linear":
        return transformers.get_linear_schedule_with_warmup(
            optimizer,
            num_warmup_steps=warmup_steps,
            num_training_steps=num_training_steps,
            last_epoch=last_epoch,
        )
    if scheduler_type in ["cosine", "exponential"]:
        return get_cyclical_cosine_schedule_with_min_lr(
            optimizer,
            num_warmup_steps=warmup_steps,
            num_training_steps=num_training_steps,
            cycle_length=cycle_length,
            min_lr_ratio=min_lr_ratio,
            last_epoch=last_epoch, scheduler_type=scheduler_type,
        )
    if scheduler_type == "cosine_restarts":
        restart_warmup_steps = round(0.1 * update_gap)
        cycle_length = update_gap
        assert restart_warmup_steps is not None, "restart_warmup_steps must be specified for cosine_restarts scheduler"
        return get_cosine_schedule_with_multiple_warmups(
            optimizer,
            num_training_steps=num_training_steps,
            first_warmup_steps=warmup_steps,
            restart_warmup_steps=restart_warmup_steps,
            restart_every=cycle_length,
            min_lr_ratio=min_lr_ratio,
            last_epoch=last_epoch,
            adjust_step=adjust_step,
        )

    raise NotImplementedError(f"Scheduler {scheduler_type} is not implemented")


def get_cyclical_cosine_schedule_with_min_lr(optimizer, num_warmup_steps, num_training_steps, cycle_length,
                                             min_lr_ratio=0.1, last_epoch=-1, scheduler_type='cosine'):
    assert cycle_length is not None or num_training_steps is not None, "You must specify either cycle_length or num_training_steps"

    if cycle_length is None:
        cycle_length = num_training_steps

    if num_training_steps % cycle_length != 0:
        raise ValueError(
            f"num_training_steps ({num_training_steps}) must be divisible by cycle_length ({cycle_length})")

    lr_lambda = partial(
        _get_cyclical_cosine_schedule_with_min_lr_lambda,
        num_warmup_steps=num_warmup_steps,
        cycle_length=cycle_length,
        min_lr_ratio=min_lr_ratio,
        scheduler_type=scheduler_type,
    )
    return LambdaLR(optimizer, lr_lambda, last_epoch)


def get_cosine_schedule_with_multiple_warmups(
        optimizer,
        *,
        num_training_steps,
        first_warmup_steps,
        restart_warmup_steps,
        restart_every,
        min_lr_ratio=0.1,
        adjust_step=0,
        last_epoch=-1,
):
    if restart_every is None:
        raise ValueError("restart_every must be specified for cosine_restarts scheduler")

    if num_training_steps % restart_every != 0:
        raise ValueError(
            f"num_training_steps ({num_training_steps}) must be divisible by restart_every ({restart_every})")

    lr_lambda = partial(
        _get_cosine_schedule_with_multiple_warmups_lambda,
        num_training_steps=num_training_steps,
        first_warmup_steps=first_warmup_steps,
        restart_warmup_steps=restart_warmup_steps,
        restart_every=restart_every,
        min_lr_ratio=min_lr_ratio,
        adjust_step=adjust_step,
    )
    return LambdaLR(optimizer, lr_lambda, last_epoch)


def _get_cyclical_cosine_schedule_with_min_lr_lambda(current_step, *, num_warmup_steps, cycle_length, min_lr_ratio,
                                                     scheduler_type):
    assert 0 < min_lr_ratio <= 1.0, "min_lr_ratio must be in (0,1]"

    # compute where we are in the current cycle
    cycle_step = current_step % cycle_length

    if cycle_step < num_warmup_steps:
        if current_step != cycle_step:
            if cycle_step < 2:
                return 1e-7
        return float(cycle_step) / float(max(1, num_warmup_steps))

    progress = float(cycle_step - num_warmup_steps) / float(max(1, cycle_length - num_warmup_steps))
    gamma = 0.999965  # this gamma ^ 10000 ~ 0.7
    # we do this here as this function is already calibrated for [0,1]->[0,1].
    decay = 0.5 * (1.0 + math.cos(math.pi * progress)) if scheduler_type == 'cosine' else gamma ** (10000 * progress)

    return min_lr_ratio + (1.0 - min_lr_ratio) * decay


def _get_cosine_schedule_with_multiple_warmups_lambda(
        current_step,
        *,
        num_training_steps,
        first_warmup_steps,
        restart_warmup_steps,
        restart_every,
        min_lr_ratio,
        adjust_step,
):
    """
    Args:
        adjust_step: useful when continuing training from a warmed up checkpoint,
            it allows to sync the resets by reducing the number of steps
            after the first warmup and before the first reset.
            Thus, your ReLoRA resets can be synced with the optimizer resets.
    """
    assert 0 < min_lr_ratio <= 1.0, "min_lr_ratio must be in (0,1]"
    assert restart_every > 0, "restart_every must be positive"
    assert adjust_step + first_warmup_steps < num_training_steps, "warmup + adjust_step is more than full training steps"
    # assert adjust_step + first_warmup_steps < restart_every, "the first reset will happen before the warmup is done"

    _current_step = current_step + adjust_step

    restart_step = _current_step % restart_every
    restart_number = _current_step // restart_every

    if current_step < first_warmup_steps:
        value = float(current_step) / float(max(1, first_warmup_steps))
    else:
        progress = float(_current_step - first_warmup_steps) / float(max(1, num_training_steps - first_warmup_steps))
        cosine_decay = 0.5 * (1.0 + math.cos(math.pi * progress))

        value = min_lr_ratio + (1.0 - min_lr_ratio) * cosine_decay

    if (restart_step < restart_warmup_steps) and (restart_number > 0):
        if current_step < first_warmup_steps:
            warmup_lr_multiplier = float(restart_number * restart_every) / float(max(1, first_warmup_steps))
        else:
            # get expected lr multipler at the end of the warmup
            end_of_warmup_progress = (
                    float(restart_number * restart_every) /
                    float(max(1, num_training_steps - first_warmup_steps))
            )

            _decay = 0.5 * (1.0 + math.cos(math.pi * end_of_warmup_progress))
            warmup_lr_multiplier = min_lr_ratio + (1.0 - min_lr_ratio) * _decay

        return float(restart_step) / float(max(1, restart_warmup_steps)) * warmup_lr_multiplier
    else:
        return value


# def _get_cosine_schedule_with_multiple_warmups_lambda(
#     current_step,
#     *,
#     num_training_steps,
#     first_warmup_steps,
#     restart_warmup_steps,
#     restart_every,
#     min_lr_ratio,
#     adjust_step,
# ):
#     """
#     Args:
#         adjust_step: useful when continuing training from a warmed up checkpoint,
#             it allows to sync the resets by reducing the number of steps
#             after the first warmup and before the first reset.
#             Thus, your ReLoRA resets can be synced with the optimizer resets.
#     """
#     assert 0 < min_lr_ratio <= 1.0, "min_lr_ratio must be in (0,1]"
#     assert restart_every > 0, "restart_every must be positive"
#     assert adjust_step + first_warmup_steps < num_training_steps, "warmup + adjust_step is more than full training steps"
#     # assert adjust_step + first_warmup_steps < restart_every, "the first reset will happen before the warmup is done"
#
#     if current_step < first_warmup_steps:
#         return float(current_step) / float(max(1, first_warmup_steps))
#
#     _current_step = current_step + adjust_step
#
#     restart_step = _current_step % restart_every
#     restart_number = _current_step // restart_every
#
#     if restart_step < restart_warmup_steps:
#         # get expected lr multipler at the end of the warmup
#         end_of_warmup_progress = (
#             float(restart_number * restart_every) /
#             float(max(1, num_training_steps - first_warmup_steps))
#         )
#
#         _cosine_decay = 0.5 * (1.0 + math.cos(math.pi * end_of_warmup_progress))
#         warmup_lr_multiplier = min_lr_ratio + (1.0 - min_lr_ratio) * _cosine_decay
#
#         return float(restart_step) / float(max(1, restart_warmup_steps)) * warmup_lr_multiplier
#
#     progress = float(_current_step - first_warmup_steps) / float(max(1, num_training_steps - first_warmup_steps))
#     cosine_decay = 0.5 * (1.0 + math.cos(math.pi * progress))
#
#     return min_lr_ratio + (1.0 - min_lr_ratio) * cosine_decay


def collate_fn(batch_list):
    batch = {
        "input_ids": torch.stack([torch.Tensor(example["input_ids"]).long() for example in batch_list]),
        "attention_mask": torch.stack([torch.Tensor(example["attention_mask"]).long() for example in batch_list]),
    }
    return batch


def batch_fn(dataset, batch_size):
    batch = []
    for example in dataset:
        batch.append(example)
        if len(batch) == batch_size:
            batch = collate_fn(batch)
            yield batch
            batch = []
    if len(batch) > 0:
        yield batch


def max_train_tokens_to_number(max_train_tokens):
    if max_train_tokens.endswith("M"):
        return int(max_train_tokens.rstrip("M")) * 1_000_000
    elif max_train_tokens.endswith("B"):
        return int(max_train_tokens.rstrip("B")) * 1_000_000_000
    else:
        return int(max_train_tokens)
