import datetime
import itertools
import json
import os
import shutil

import numpy as np


def batch_iterable(iterable, n=128):
    if isinstance(iterable, list):
        iterable = iter(iterable)

    while True:
        batch = list(itertools.islice(iterable, n))
        if not batch:
            break
        yield batch


def extract_row_from_batch_dict(batch, index):
    return {k: v[index] for k, v in batch.items()}


class NpEncoder(json.JSONEncoder):
    """
    A useful thing to make dicts json compatible.
    """

    def default(self, obj):
        import torch

        if isinstance(obj, np.integer):
            return int(obj)
        if isinstance(obj, np.floating):
            return float(obj)
        if isinstance(obj, np.ndarray) or isinstance(obj, torch.Tensor):
            return obj.tolist()
        return super(NpEncoder, self).default(obj)


def json_valid_dict(obj):
    return json.loads(json.dumps(obj, cls=NpEncoder))


def utc_epoch_now():
    return datetime.datetime.now().replace(tzinfo=datetime.timezone.utc).timestamp()


def makedir(path: str, isfile: bool = False):
    """
    Creates a directory given a path to either a directory or file.
    If a directory is provided, creates that directory. If a file is provided (i.e. isfile == True),
    creates the parent directory for that file.
    :param path: Path to a directory or file.
    :param isfile: Whether the provided path is a directory or file.
    """
    if isfile:
        path = os.path.dirname(path)
    if path != "":
        os.makedirs(path, exist_ok=True)


def rmdir(path: str):
    """
    Creates a directory given a path to either a directory or file.
    If a directory is provided, creates that directory. If a file is provided (i.e. isfile == True),
    creates the parent directory for that file.
    :param path: Path to a directory or file.
    :param isfile: Whether the provided path is a directory or file.
    """
    try:
        shutil.rmtree(path)
    except Exception as Ex:
        print("rmdir failure", Ex)


def colored_background(r: int, g: int, b: int, text):
    """
    r,g,b integers between 0,255
    """
    return f"\033[48;2;{r};{g};{b}m{text}\033[0m"


def chunk_indexable(iterable, n=1, l=None):
    """
    Chunking when you can index.

    Args:
        iterable (_type_): Any iterable that supports indexing
        n (int, optional): Chunk size. Defaults to 1.
        l (int, optional): Length of iterable. Defaults to None.

    Yields:
        _type_: _description_
    """
    l = len(iterable) if l is None else l
    for ndx in range(0, l, n):
        yield iterable[ndx : min(ndx + n, l)]


def chunk_iterable(iterator, chunk_size: int, subsample: float = None):
    """Chunks an iterator that can't be indexed.
    If there's more left over, just returns the remainder.

    Optional subsampling.

    Args:
        iterator (_type_): Any iterable
        chunk_size (int): size of chunk
        subsample (float): fraction to subsample, if provided
    """

    chunk = []
    for elem in iterator:
        if not subsample is None:
            if np.random.random() > subsample:
                continue
        chunk.append(elem)
        if len(chunk) >= chunk_size:
            # yield and then reset
            yield chunk
            chunk = []
    if len(chunk):
        yield chunk
    return
