#include <iostream>
#include <string>
#include <vector>
#include <map>
#include <stdexcept>
#include <iostream>
#include <fstream>
#include <string>
#include <vector>
#include <map>
#include <memory>
#include <stdexcept>
#include "json.hpp"
#include "BankAccount.hpp"

// State serialization function
json serialize_account_state(const BankAccount& account) {
    return {
        {"account_id", account.get_account_id()},
        {"balance", account.get_balance()},
        {"is_logged_in", account.get_login_status()},
        {"login_attempts", account.get_login_attempts()},
        {"transaction_count", account.get_transaction_count()}
    };
}

// Main test function
json execute_json_test_script(const json& test_script_json) {
    // Create accounts dictionary
    std::map<std::string, std::shared_ptr<BankAccount>> accounts;
    
    // Initialize results
    json results = {
        {"test_suite", test_script_json.value("test_suite", "unknown")},
        {"total_cases", test_script_json.contains("test_cases") ? test_script_json["test_cases"].size() : 0},
        {"passed_cases", 0},
        {"failed_cases", 0},
        {"case_details", json::array()}
    };
    
    // Initialize accounts
    if (test_script_json.contains("setup") && test_script_json["setup"].contains("accounts")) {
        for (const auto& account_config : test_script_json["setup"]["accounts"]) {
            auto account = std::make_shared<BankAccount>(
                account_config["account_id"],
                account_config["initial_balance"],
                account_config["password"]
            );
            accounts[account_config["account_id"]] = account;
        }
    }
    
    // Execute test cases
    if (test_script_json.contains("test_cases")) {
        for (const auto& test_case : test_script_json["test_cases"]) {
            json case_result = {
                {"case_id", test_case["case_id"]},
                {"description", test_case["description"]},
                {"passed", true},
                {"operation_results", json::array()}
            };
            
            for (const auto& operation : test_case["operations"]) {
                json op_result = {
                    {"operation", operation},
                    {"status", "unknown"}
                };
                
                try {
                    std::string action = operation["action"];
                    
                    if (action == "login") {
                        auto account = accounts[operation["account_id"]];
                        std::string result = account->login(operation["password"]);
                        
                        // Handle expected result - could be string or boolean
                        if (operation.contains("expected_result")) {
                            if (operation["expected_result"].is_string()) {
                                std::string expected = operation["expected_result"];
                                op_result["status"] = (result == expected) ? "passed" : "failed";
                                op_result["actual"] = result;
                                op_result["expected"] = expected;
                            } else if (operation["expected_result"].is_boolean()) {
                                bool expected_success = operation["expected_result"];
                                bool actual_success = (result == "Login successful");
                                op_result["status"] = (actual_success == expected_success) ? "passed" : "failed";
                                op_result["actual"] = result;
                                op_result["expected"] = expected_success ? "Login successful" : "Password error";
                            }
                        }
                        
                    } else if (action == "logout") {
                        auto account = accounts[operation["account_id"]];
                        std::string result = account->logout();
                        
                        if (operation.contains("expected_result")) {
                            if (operation["expected_result"].is_string()) {
                                std::string expected = operation["expected_result"];
                                op_result["status"] = (result == expected) ? "passed" : "failed";
                                op_result["actual"] = result;
                                op_result["expected"] = expected;
                            } else if (operation["expected_result"].is_boolean()) {
                                bool expected_success = operation["expected_result"];
                                bool actual_success = (result == "Logout successful");
                                op_result["status"] = (actual_success == expected_success) ? "passed" : "failed";
                                op_result["actual"] = result;
                                op_result["expected"] = expected_success ? "Logout successful" : "Not logged in";
                            }
                        }
                        
                    } else if (action == "deposit") {
                        auto account = accounts[operation["account_id"]];
                        double result = account->deposit(operation["amount"]);
                        
                        if (operation.contains("expected_result")) {
                            if (operation["expected_result"].is_number()) {
                                double expected = operation["expected_result"];
                                op_result["status"] = (result == expected) ? "passed" : "failed";
                                op_result["actual"] = result;
                                op_result["expected"] = expected;
                            } else if (operation["expected_result"].is_boolean()) {
                                // If expected result is boolean, we assume operation is expected to succeed
                                op_result["status"] = "passed";
                                op_result["actual"] = result;
                                op_result["expected"] = "Operation successful";
                            }
                        }
                        
                    } else if (action == "withdraw") {
                        auto account = accounts[operation["account_id"]];
                        double result = account->withdraw(operation["amount"]);
                        
                        if (operation.contains("expected_result")) {
                            if (operation["expected_result"].is_number()) {
                                double expected = operation["expected_result"];
                                op_result["status"] = (result == expected) ? "passed" : "failed";
                                op_result["actual"] = result;
                                op_result["expected"] = expected;
                            } else if (operation["expected_result"].is_boolean()) {
                                // If expected result is boolean, we assume operation is expected to succeed
                                op_result["status"] = "passed";
                                op_result["actual"] = result;
                                op_result["expected"] = "Operation successful";
                            }
                        }
                        
                    } else if (action == "transfer") {
                        auto from_account = accounts[operation["from_account"]];
                        auto to_account = accounts[operation["to_account"]];
                        std::string result = from_account->transfer(to_account, operation["amount"]);
                        
                        if (operation.contains("expected_result")) {
                            if (operation["expected_result"].is_string()) {
                                std::string expected = operation["expected_result"];
                                op_result["status"] = (result == expected) ? "passed" : "failed";
                                op_result["actual"] = result;
                                op_result["expected"] = expected;
                            } else if (operation["expected_result"].is_boolean()) {
                                bool expected_success = operation["expected_result"];
                                bool actual_success = (result == "Transfer successful");
                                op_result["status"] = (actual_success == expected_success) ? "passed" : "failed";
                                op_result["actual"] = result;
                                op_result["expected"] = expected_success ? "Transfer successful" : "Transfer failed";
                            }
                        }
                        
                    } else if (action == "verify_state") {
                        auto account = accounts[operation["account_id"]];
                        double actual_balance = account->get_balance();
                        double expected_balance = operation["expected_balance"];
                        op_result["status"] = (actual_balance == expected_balance) ? "passed" : "failed";
                        op_result["actual"] = actual_balance;
                        op_result["expected"] = expected_balance;
                    }
                    
                    // Check for expected error
                    if (operation.contains("expected_error") && op_result["status"] == "passed") {
                        op_result["status"] = "failed";  // Should have thrown an error but didn't
                    }
                    
                } catch (const std::exception& e) {
                    // Handle exception cases
                    if (operation.contains("expected_error")) {
                        std::string expected_error = operation["expected_error"];
                        // Simplified handling: check if exception message contains expected error
                        if (std::string(e.what()).find(expected_error) != std::string::npos) {
                            op_result["status"] = "passed";
                        } else {
                            op_result["status"] = "failed";
                        }
                    } else {
                        op_result["status"] = "failed";
                    }
                    op_result["error"] = e.what();
                    op_result["error_type"] = typeid(e).name();
                }
                
                // If any operation fails, the entire test case fails
                if (op_result["status"] == "failed") {
                    case_result["passed"] = false;
                }
                
                case_result["operation_results"].push_back(op_result);
            }
            
            // Update statistics
            if (case_result["passed"]) {
                results["passed_cases"] = results["passed_cases"].get<int>() + 1;
            } else {
                results["failed_cases"] = results["failed_cases"].get<int>() + 1;
            }
            
            results["case_details"].push_back(case_result);
        }
    }
    
    // Add final account states
    json final_account_states;
    for (const auto& [account_id, account] : accounts) {
        final_account_states[account_id] = serialize_account_state(*account);
    }
    results["final_account_states"] = final_account_states;
    
    return results;
}

// Usage example
int main() {
    try {
        // Read JSON file
        std::ifstream file("testcase.json");
        if (!file.is_open()) {
            std::cerr << "Cannot open file data.json" << std::endl;
            return 1;
        }
        
        json test_script;
        file >> test_script;
        file.close();
        
        // Execute test
        json result = execute_json_test_script(test_script);
        
        // Print results
        std::cout << "Test results:" << std::endl;
        std::cout << result.dump(2) << std::endl;
        
    } catch (const std::exception& e) {
        std::cerr << "Error occurred: " << e.what() << std::endl;
        return 1;
    }
    
    return 0;
}