import gradio as gr
import re
import inspect

from sentence_transformers import SentenceTransformer
from sentence_transformers.util import cos_sim


codes = """001 - Vehicle Registration (New)
002 - Vehicle Registration Renewal
003 - Vehicle Ownership Transfer
004 - Vehicle De-registration
005 - Lost Registration Certificate Replacement
006 - Address Change Update
007 - Vehicle Data Correction
008 - Ownership Name Correction
009 - Vehicle Tax Payment
010 - Late Payment Fee Processing
011 - Vehicle Type/Specification Update
012 - BBNKB (Transfer Fee of Vehicle Ownership)
013 - STNK Issuance (Vehicle Registration Certificate)
014 - STNK Renewal
015 - Motor Vehicle Roadworthiness Inspection
016 - Plate Number Renewal
017 - Lost Plate Replacement
018 - Vehicle Export Registration
019 - Vehicle Import Registration
020 - Fleet Vehicle Registration
021 - Bulk Vehicle Registration Update
022 - Vehicle Insurance Assistance
023 - Vehicle Accident Reporting
024 - Vehicle Usage Change Declaration (e.g., personal to commercial)
025 - Legal Document Verification
026 - Ownership Transfer for Inherited Vehicle
027 - STNK Temporary Suspension
028 - Proof of Ownership Document Update
029 - Vehicle Ownership History Check
030 - Vehicle Tax Recalculation Request
031 - Tax Exemption Application (for special cases)
032 - Deceased Owner’s Vehicle Ownership Transfer""".split("\n")

undetected = "099 - Other/Undetected"


# codes = """001 - Pendaftaran Kendaraan (Baru)
# 002 - Pembaruan Pendaftaran Kendaraan
# 003 - Alih Kepemilikan Kendaraan
# 004 - Pembatalan Pendaftaran Kendaraan
# 005 - Penggantian Sertifikat Pendaftaran Kendaraan yang Hilang
# 006 - Pembaruan Perubahan Alamat
# 007 - Koreksi Data Kendaraan
# 008 - Koreksi Nama Kepemilikan
# 009 - Pembayaran Pajak Kendaraan
# 010 - Proses Denda Keterlambatan Pembayaran
# 011 - Pembaruan Jenis/Spesifikasi Kendaraan
# 012 - Pembayaran Pajak Kendaraan Melalui E-Samsat
# 013 - Penerbitan STNK (Sertifikat Pendaftaran Kendaraan)
# 014 - Pembaruan STNK
# 015 - Pemeriksaan Kelayakan Jalan Kendaraan Bermotor
# 016 - Pembaruan Nomor Plat Kendaraan
# 017 - Penggantian Plat yang Hilang
# 018 - Pendaftaran Ekspor Kendaraan
# 019 - Pendaftaran Impor Kendaraan
# 020 - Pendaftaran Kendaraan Armada
# 021 - Pembaruan Pendaftaran Kendaraan Massal
# 022 - Bantuan Asuransi Kendaraan
# 023 - Pelaporan Kecelakaan Kendaraan
# 024 - Deklarasi Perubahan Penggunaan Kendaraan (misalnya, pribadi ke komersial)
# 025 - Verifikasi Dokumen Hukum
# 026 - Alih Kepemilikan Kendaraan Warisan
# 027 - Penangguhan Sementara STNK
# 028 - Pembaruan Dokumen Bukti Kepemilikan
# 029 - Pemeriksaan Riwayat Kepemilikan Kendaraan
# 030 - Permintaan Perhitungan Ulang Pajak Kendaraan
# 031 - Permohonan Pembebasan Pajak (untuk kasus khusus)
# 032 - Alih Kepemilikan Kendaraan Pemilik yang Meninggal""".split("\n")

codes = """001 - Pendaftaran Kendaraan
002 - Pembaruan Data Kendaraan
003 - Alih atau Pembatalan Kepemilikan
004 - Pelaporan Dokumen atau Plat yang Hilang
005 - Pembayaran dan Pengelolaan Pajak Kendaraan
006 - Pemeriksaan dan Verifikasi Kendaraan
007 - Pendaftaran Kendaraan Ekspor, Impor, atau Armada
008 - Pelaporan dan Bantuan Terkait Kendaraan
009 - Penangguhan atau Deklarasi Perubahan Penggunaan Kendaraan""".split("\n")


vehicle_tax_info = {
    "B 1234 BCA": {
        "no_rangka": "1237191234",
        "type": "SUV",
        "tanggal": "23 Desember 2024",
        "status": "Belum Bayar",
        "harga_jual": 150000  # In Rupiah
    },
    "B 5678 XYZ": {
        "no_rangka": "9876543210",
        "type": "Sedan",
        "tanggal": "15 Januari 2025",
        "status": "Belum Bayar",
        "harga_jual": 300000  # In Rupiah
    },
    "D 3456 DEF": {
        "no_rangka": "4561237890",
        "type": "MPV",
        "tanggal": "10 Februari 2025",
        "status": "Sudah Bayar",
        "harga_jual": 250000  # In Rupiah
    }
}

# Table for detail calculations (perhitungan)
detail_perhitungan = {
    "001": {
        "name": "Pendaftaran Kendaraan",
        "formula": lambda harga_jual: harga_jual * 0.1,
        # Example formula: 10% of harga_jual
    },
    "002": {
        "name": "Pembaruan Data Kendaraan",
        "formula": lambda harga_jual: harga_jual * 0.05,
        # Example formula: 5% of harga_jual
    },
    "003": {
        "name": "Alih Kepemilikan Kendaraan",
        "formula": lambda harga_jual: harga_jual * 0.1,  
        # Example formula: 10% of harga_jual
    },
    "004": {
        "name": "Penggantian Dokumen atau Plat yang Hilang",
        "formula": lambda harga_jual: harga_jual * 0.03,
        # Example formula: 3% of harga_jual
    },
    "005": {
        "name": "Pembayaran dan Pengelolaan Pajak Kendaraan",
        "formula": lambda harga_jual: harga_jual * 0.12,
        # Example formula: 12% of harga_jual
    },
    "006": {
        "name": "Pemeriksaan dan Verifikasi Kendaraan",
        "formula": lambda harga_jual: 100000,
        # Example formula: 2% of harga_jual
    },
    "007": {
        "name": "Pendaftaran Kendaraan Ekspor, Impor, atau Armada",
        "formula": lambda harga_jual: harga_jual * 0.15,
        # Example formula: 15% of harga_jual
    },
    "008": {
        "name": "Pelaporan dan Bantuan Terkait Kendaraan",
        "formula": lambda harga_jual: harga_jual * 0.04,
        # Example formula: 4% of harga_jual
    },
    "009": {
        "name": "Penangguhan atau Deklarasi Perubahan Penggunaan Kendaraan",
        "formula": lambda harga_jual: harga_jual * 0.06,
        # Example formula: 6% of harga_jual
    }
}

undetected = "099 - Lainnya/Tidak Terdeteksi"

model_ids = [
    "BAAI/bge-m3",
    "sentence-transformers/paraphrase-multilingual-mpnet-base-v2",
    "intfloat/multilingual-e5-small",
    "sentence-transformers/distiluse-base-multilingual-cased-v2",
    "sentence-transformers/paraphrase-multilingual-MiniLM-L12-v2",
    "Alibaba-NLP/gte-multilingual-base",
]
# model_id = "sentence-transformers/paraphrase-multilingual-MiniLM-L12-v2"
# model_id = "Alibaba-NLP/gte-multilingual-base"
# model_id = "BAAI/bge-m3"
# model_id = "sentence-transformers/paraphrase-multilingual-mpnet-base-v2"
# model_id = "intfloat/multilingual-e5-small"
# model_id = "sentence-transformers/distiluse-base-multilingual-cased-v2"
model_id = model_ids[-1]
model = SentenceTransformer(model_id, trust_remote_code=True)

codes_emb = model.encode([x[6:] for x in codes])


def get_calculation(request_code, plate_number):
    print(request_code, plate_number, "GET CALC")
    calc = detail_perhitungan.get(request_code)
    vehicle = vehicle_tax_info.get(plate_number)
    
    if vehicle == None or calc == None:
        return None, None, None
        
    harga_jual = vehicle.get("harga_jual")
    formula = calc.get("formula")
    result = formula(harga_jual)
    description = inspect.getsource(formula).split(":", 2)[-1].strip()
    result_detail = request_code + " - " + calc.get("name")

    # out = "=============================================="
    # out += "\nWajib Pajak ingin melakukan proses berikut:\n"
    # out = ""
    
    return result, str(description), result_detail


def build_output(result, description, result_detail, plate_number):
    return build_outputs([result], [description], [result_detail], plate_number)


def build_outputs(results, descriptions, result_details, plate_number):
    out = "--------------------------------------------------"
    vehicle = vehicle_tax_info.get(plate_number)
    out = "\nPlate Number: " + plate_number + "\n"
    out += "\n".join([k + " : " + str(v) for  k,v in vehicle.items()])

    out += "\n----------------------------------------------"
    out += f"\nWajib Pajak dengan NoPol {plate_number} ingin melakukan proses berikut:\n"

    for i, (res,desc,detail) in enumerate(zip(results, descriptions, result_details)):
        out += f"{i+1}. {detail}\nDetail perhitungan: {desc.replace('harga_jual', str(res))}\n"

    out += "Estimasi biaya: "

    if len(results) > 1:
        out += " + ".join([f"Rp{x}" for x in results])
        out += f" = {sum(results)}"
    else:
        out += str(results[0])
    return out
    

def respond(
    message,
    history: list[tuple[str, str]],
    threshold,
    is_multiple
):
    global codes_emb
    global undetected

    undetected_code = undetected[:3]

    if history and history[-1][-1][21:24] == undetected_code: 
        list_his = ""
        for his in history[::-1]:
            if his[-1][21:24] != undetected_code:
                break
            list_his = his[0] + "\n" + list_his

        message += "\n" + list_his
    
    # pattern = r'\b([A-Z]{1,2})\s?(\d{4})\s?([A-Z]{3})\b'
    # pattern = r'\b([A-Z]{1,2})\s?(\d{4})\s?([A-Z]{1,3})\b'
    pattern = r'\b([A-Za-z]{1,2})\s?(\d{4})\s?([A-Za-z]{1,3})\b'

    matches = re.findall(pattern, message)

    plates = [" ".join(x).upper() for i,x in enumerate(matches)]

    plate_numbers = ", ".join(plates)
    
    text_emb = model.encode(message)
    scores = cos_sim(codes_emb, text_emb)[:,0]

    if is_multiple:
        request_details = []
        request_numbers = []
        request_scores = []
        for i,score in enumerate(scores):
            if score > threshold:
                request_details.append(codes[i][6:])
                request_numbers.append(codes[i][:3])
                request_scores.append(str( round(score.tolist(), 3) ) )

        if not request_details:
            request_details.append(undetected[6:])
            request_numbers.append(undetected_code)

        request_numbers_copy = request_numbers

        request_numbers = "\n".join(request_numbers)
        request_details = "\n".join(request_details)
        request_scores = "\n".join(request_scores)
        
        out = "Request code number:\n" + request_numbers + "\n\nRequest detail:\n" + request_details + f"\n\nConfidence score:\n{request_scores}" + "\n\nPlate numbers: " + plate_numbers

        for plate in plates:
            results, descriptions, result_details = [], [], []
            for code in request_numbers_copy:
                result, description, result_detail = get_calculation(code, plate)

                if result != None:
                    results.append(result)
                    descriptions.append(description)
                    result_details.append(result_detail)
                else:
                    break

            if results:
                out += "\n\n" + build_outputs(results, descriptions, result_details, plate)

        return out

        # result, description, result_detailget_calculation(request_code, plate_number)



    s_max = scores.argmax()

    if scores[s_max] < threshold:
        # request_code = "033 - Other/Undetected"
        request_code = undetected
    else:
        request_code = codes[scores.argmax()]
        # "{:.2f}".format(a)

    out = "Request code number: " + request_code[:3] + "\nRequest detail: " + request_code[6:] + f"\nConfidence score: {round(scores[s_max].tolist(),3)}" + "\nPlate numbers: " + plate_numbers

    for plate in plates:
        results, descriptions, result_details = [], [], []
        result, description, result_detail = get_calculation(request_code[:3], plate)

        if result != None:
            results.append(result)
            descriptions.append(description)
            result_details.append(result_detail)
            out += "\n\n" + build_outputs(results, descriptions, result_details, plate)

    return out
    
    # if vehicle_tax_info.get(request_detail)
    
    # for val in history:
    #     if val[0]:
    #         messages.append({"role": "user", "content": val[0]})
    #     if val[1]:
    #         messages.append({"role": "assistant", "content": val[1]})

    # messages.append({"role": "user", "content": message})

    # response = ""

    # for message in client.chat_completion(
    #     messages,
    #     max_tokens=max_tokens,
    #     stream=True,
    #     temperature=temperature,
    #     top_p=top_p,
    # ):
    #     token = message.choices[0].delta.content

    #     response += token
    #     yield response


"""
For information on how to customize the ChatInterface, peruse the gradio docs: https://www.gradio.app/docs/chatinterface
"""
# demo = gr.ChatInterface(
#     respond,
# )


def reload(chosen_model_id):
    global model
    global model_id
    global codes_emb

    if chosen_model_id != model_id:
        model = SentenceTransformer(chosen_model_id, trust_remote_code=True)
        model_id = chosen_model_id
        codes_emb = model.encode([x[6:] for x in codes])
        return f"Model {chosen_model_id} has been succesfully loaded!"
    return f"Model {chosen_model_id} is ready!"
    


with gr.Blocks() as demo:
    # Add header title and description
    gr.Markdown("# List of Request Numbers")
    gr.Markdown("<br>".join(codes) + "<br>" + undetected)
    gr.Markdown("# Valid License Plate Number Criteria:")
    gr.Markdown("(1-2 letters) (4 numbers) (1-3 letters)")
    gr.Markdown("# Example Valid Plate Numbers")
    gr.Markdown("<br>".join(vehicle_tax_info.keys()))


    gr.Markdown("# Choose & Load Model:")
    reload_model = gr.Interface(
        fn=reload, 
        inputs=[gr.Dropdown(choices=model_ids, value=model_id)], 
        outputs="text",
        # gr.HighlightedText(
        #     label="status",
        #     combine_adjacent=True,
        #     show_legend=True,
        #     color_map={"+": "red", "-": "green"}
        # ),
    )

    gr.Markdown("# Chatbot Interface:")
    chat_interface = gr.ChatInterface(
        respond, 
        additional_inputs=[
            gr.Number(0.5, label="confidence threshold", show_label=True, minimum=0., maximum=1.0, step=0.1),
            gr.Checkbox(label="multiple", info="Allow multiple request code numbers"),
            
        ]
    )

if __name__ == "__main__":
    demo.launch()