import functools
import random
from typing import Optional, Tuple

from rdkit import Chem


def rdkit_version():
    """
    Get the version of RDKit.
    """
    from rdkit import __version__

    return __version__


def disable_logger():
    from rdkit import RDLogger

    RDLogger.DisableLog("rdApp.*")


def works_on_smiles(raise_on_failure):
    """
    converts any function mapping a mol to another mol
    to a function that also works with smiles.

    I had too much fun with this.
    """

    def decorator(mol_func):
        @functools.wraps(mol_func)
        def wrapped_func(*args, **kwargs):
            if isinstance(args[0], str):
                if mol := Chem.MolFromSmiles(args[0]):
                    new_args = list(args)
                    new_args[0] = mol
                    results = mol_func(*new_args, **kwargs)
                    # try to convert back to smiles...
                    if isinstance(results, Chem.Mol):
                        return Chem.MolToSmiles(results)
                    elif isinstance(results, tuple):  # TODO: support other iterables?
                        return tuple(
                            Chem.MolToSmiles(res) if isinstance(res, Chem.Mol) else res
                            for res in results
                        )
                    else:
                        return results
                else:
                    if raise_on_failure:
                        raise ValueError(f"{args[0]} could not be converted to mol.")
                    else:
                        return None
            else:
                return mol_func(*args, **kwargs)

        return wrapped_func

    return decorator


def permute_smiles(smiles):
    mol = Chem.MolFromSmiles(smiles)
    ans = list(range(mol.GetNumAtoms()))
    random.shuffle(ans)
    nm = Chem.RenumberAtoms(mol, ans)
    return Chem.MolToSmiles(nm, canonical=False)


# note that this won't handle charges, fragments, salts etc.
def simple_canonicalize(smi: str, allow_error: bool = False) -> str:
    if (mol := Chem.MolFromSmiles(smi)) is not None:
        return Chem.MolToSmiles(mol)
    elif allow_error:
        return None
    else:
        raise ValueError(f"{smi} cannot be parsed. ")


def inchi(smi: str) -> Tuple[str, str]:
    """
    Get the InChI key of a SMILES string.
    """
    mol = Chem.MolFromSmiles(smi)
    if mol is None:
        raise ValueError(f"{smi} cannot be parsed. ")
    return Chem.MolToInchiKey(mol), Chem.MolToInchi(mol)
