import multiprocessing
import os
import time
from tqdm import tqdm

TIMEOUT=1000

def parse_kissat_log(fname: str):
    with open(fname, "r") as f:
        lines = f.readlines()
    metadata = {}
    statistics = False
    result = None
    solution = []
    for line in lines:
        if statistics:
            line = line.strip().split()
            if "----" in line:
                statistics = False
            else:
                if len(line) > 3:
                    name = line[1][:-1]
                    value = int(line[2])
                    metadata[name] = value
        elif "[ statistics ]" in line:
            statistics = True
        elif line.startswith("s"):
            line = line.strip().split()
            result = (line[1] == "SATISFIABLE")
        elif "process-time" in line:
            line = line.strip().split()
            process_time = float(line[-2])
    metadata["result"] = result
    metadata["process_time"] = process_time
    return metadata

def solve_kissat(fname: str):
    if fname.endswith(".cnf"):
        log_fname = fname[:-4]+".log"
    elif fname.endswith(".cnf.xz"):
        log_fname = fname[:-7]+".log"
    elif fname.endswith(".cnf.bz2"):
        log_fname = fname[:-8]+".log"
    if os.path.isfile(log_fname):
        metadata = parse_kissat_log(log_fname)
        if metadata["result"] is not None:
            return metadata["result"]
    command = "./kissat/build/kissat --time={} {} > {}".format(
        TIMEOUT, fname, log_fname
    )
    os.system(command)
    metadata = parse_kissat_log(log_fname)
    return metadata["result"]

def solve_kissat_mp(fnames: list[str], mp: int|None=None):
    if mp is None:
        results = [solve_kissat(f) for f in fnames]
    else:
        assert mp > 1
        with multiprocessing.Pool(processes=mp) as pool:
            results = list(tqdm(pool.imap(solve_kissat, fnames),
                                total=len(fnames), desc="Solving"))
    return results
