#include <iostream>
#include <string>
#include <vector>
#include <map>
#include <memory>
#include <stdexcept>
#include "BankAccount.hpp"
#include "json.hpp"
#include <iostream>
#include <fstream>
#include <string>
#include <cmath>
#include <iomanip>
using json = nlohmann::json;

class InterestCalculator {
    public:
        // Calculate interest for a single account
        static double calculate_interest(const BankAccount& account, double annual_rate, int days = 30) {
            if (annual_rate <= 0) {
                throw std::runtime_error("Interest rate must be positive");
            }
            if (days <= 0) {
                throw std::runtime_error("Number of days must be positive");
            }
            
            double balance = account.get_balance();
            if (balance <= 0) {
                return 0.0; // No interest for zero or negative balance
            }
            
            // Simple interest calculation: interest = principal × annual rate × days / 365
            double interest = balance * annual_rate * days / 365;
            return interest;
        }
        
        // Apply interest to account (via deposit method)
        static std::string apply_interest(BankAccount& account, double annual_rate, int days = 30) {
            if (!account.get_login_status()) {
                throw std::runtime_error("Please log in to your account first");
            }
            
            double interest = calculate_interest(account, annual_rate, days);
            
            if (interest > 0) {
                double new_balance = account.deposit(interest);
                return "Interest calculated successfully! Interest earned: " + std::to_string(interest) + 
                       ", New balance: " + std::to_string(new_balance);
            } else {
                return "No interest to calculate";
            }
        }
        
        // Calculate interest for multiple accounts in batch
        static std::map<std::string, double> calculate_batch_interest(
            const std::map<std::string, std::shared_ptr<BankAccount>>& accounts, 
            double annual_rate, int days = 30) {
            
            std::map<std::string, double> interest_results;
            
            for (const auto& account_pair : accounts) {
                double interest = calculate_interest(*account_pair.second, annual_rate, days);
                interest_results[account_pair.first] = interest;
            }
            
            return interest_results;
        }
        
        // Display interest calculation details
        static void display_interest_details(const BankAccount& account, double annual_rate, int days = 30) {
            double balance = account.get_balance();
            double interest = calculate_interest(account, annual_rate, days);
            
            std::cout << "=== Interest Calculation Details ===" << std::endl;
            std::cout << "Account ID: " << account.get_account_id() << std::endl;
            std::cout << "Current balance: " << balance << std::endl;
            std::cout << "Annual interest rate: " << (annual_rate * 100) << "%" << std::endl;
            std::cout << "Interest days: " << days << " days" << std::endl;
            std::cout << "Expected interest: " << interest << std::endl;
            std::cout << "Total after interest: " << (balance + interest) << std::endl;
        }
        
        // New: Execute all operations at once
        static std::string process_complete_interest(BankAccount& account, double annual_rate, int days = 30, bool apply = true) {
            std::string result;
            
            // Display details
            display_interest_details(account, annual_rate, days);
            
            // If applying interest is requested
            if (apply) {
                try {
                    result = apply_interest(account, annual_rate, days);
                } catch (const std::runtime_error& e) {
                    result = "Failed to apply interest: " + std::string(e.what());
                }
            } else {
                double interest = calculate_interest(account, annual_rate, days);
                result = "Interest calculation completed (not applied), expected interest: " + std::to_string(interest);
            }
            
            return result;
        }
    };


int main() {
    try {
        std::ifstream file("testcase.json");
        if (!file.is_open()) {
            std::cerr << "Cannot open file testcase.json" << std::endl;
            return 1;
        }
        
        json test_script;
        file >> test_script;
        file.close();

        // Get test case array
        auto& tests = test_script["interest_tests"];
        
        std::cout << "=== Interest Calculation Function Test ===" << std::endl;
        std::cout << std::fixed << std::setprecision(5);
        
        int passed_tests = 0;
        int total_tests = tests.size();
        
        for (const auto& test : tests) {
            std::string account_id = test["account_id"];
            double current_balance = test["current_balance"];
            double annual_rate = test["annual_rate"];
            int days = test["days"];
            double expected_interest = test["expected_interest"];
            double expected_total = test["expected_total"];
            
            // Create test account
            BankAccount account(account_id, current_balance);
            
            // Calculate actual interest
            double actual_interest = InterestCalculator::calculate_interest(account, annual_rate, days);
            double actual_total = current_balance + actual_interest;
            
            // Set tolerance range
            double tolerance = 0.0001;
            
            // Check if interest is within tolerance range
            bool interest_correct = std::abs(actual_interest - expected_interest) < tolerance;
            // Check if total is within tolerance range
            bool total_correct = std::abs(actual_total - expected_total) < tolerance;
            
            std::cout << "\nTest account: " << account_id << std::endl;
            std::cout << "Current balance: " << current_balance << std::endl;
            std::cout << "Annual interest rate: " << (annual_rate * 100) << "%" << std::endl;
            std::cout << "Interest days: " << days << " days" << std::endl;
            std::cout << "Expected interest: " << expected_interest << std::endl;
            std::cout << "Actual interest: " << actual_interest << std::endl;
            std::cout << "Expected total: " << expected_total << std::endl;
            std::cout << "Actual total: " << actual_total << std::endl;
            
            if (interest_correct && total_correct) {
                std::cout << "Test passed!" << std::endl;
                passed_tests++;
            } else {
                std::cout << "Test failed!" << std::endl;
                if (!interest_correct) {
                    std::cout << "  Interest calculation error: expected " << expected_interest 
                              << ", actual " << actual_interest 
                              << ", difference " << (actual_interest - expected_interest) << std::endl;
                }
                if (!total_correct) {
                    std::cout << "  Total calculation error: expected " << expected_total 
                              << ", actual " << actual_total 
                              << ", difference " << (actual_total - expected_total) << std::endl;
                }
            }
            std::cout << "----------------------------------------" << std::endl;
        }
        
        // Output test summary
        std::cout << "\n=== Test Summary ===" << std::endl;
        std::cout << "Total tests: " << total_tests << std::endl;
        std::cout << "Passed tests: " << passed_tests << std::endl;
        std::cout << "Failed tests: " << (total_tests - passed_tests) << std::endl;
        std::cout << "Pass rate: " << (static_cast<double>(passed_tests) / total_tests * 100) << "%" << std::endl;
        
    } catch (const std::exception& e) {
        std::cerr << "Error: " << e.what() << std::endl;
        return 1;
    }
}