import numpy as np

class ParkingPermitSolver:

    def __init__(self):
        self.costs = None
        self.durations = None
        self.number_permits = 0
        self.request_sequence = None
        self.time_duration = 0

    def setPermitTypes(self, costs: np.array, durations: np.array):
        """Sets the permit types with their associated costs and durations."""

        # Check that both arrays are 1D and of the same length
        if costs.ndim != 1:
            raise ValueError(f"'costs' must be a 1D array, but got shape {costs.shape}")
        if durations.ndim != 1:
            raise ValueError(f"'durations' must be a 1D array, but got shape {durations.shape}")
        if costs.shape[0] != durations.shape[0]:
            raise ValueError(
                f"'costs' and 'durations' must have the same length, "
                f"but got {costs.shape[0]} and {durations.shape[0]}"
            )
        
        for i in range(0, durations.shape[0]-1):
            if durations[i] <= 0:
                raise ValueError(f"All durations must be positive, but got {durations[i]} at index {i}")
            if costs[i] <= 0:
                raise ValueError(f"All costs must be positive, but got {costs[i]} at index {i}")
            if durations[i+1] <= durations[i]:
                raise ValueError(
                    f"Durations must be in strictly increasing order, "
                    f"but got {durations[i]} and {durations[i+1]} at indices {i} and {i+1}"
                )

        self.costs = costs
        self.durations = durations
        self.number_permits = self.costs.shape[0]

    def setInstance(self, request_sequence: np.array):
        """Sets the instance with the given request sequence."""

        # Check that request_sequence is a 1D array
        if request_sequence.ndim != 1:
            raise ValueError(f"'request_sequence' must be a 1D array, but got shape {request_sequence.shape}")

        self.request_sequence = request_sequence
        self.time_duration = self.request_sequence.shape[0] 

    def calculateOptimalCostLaminar(self):
        
        # verify laminar property
        for i in range(self.number_permits-1):
            if self.durations[i+1] % self.durations[i] != 0:
                raise ValueError("Require each duration to be a multiple of the previous one.")
            
        dual_vars = np.zeros(self.time_duration, dtype=float)
        primal_vars = np.zeros((self.number_permits, self.time_duration), dtype=float)

        # calculate dual variables
        objective_value = 0.0
        slacks = np.zeros(self.number_permits, dtype=float) # slack for each permit type

        for t in range(self.time_duration):
            for k in range(self.number_permits):
                if t % self.durations[k] == 0:
                    slacks[k] = self.costs[k] # reset slack when old permit finishes

            if self.request_sequence[t] != 0:
                # dual variable calculations
                k_min = np.argmin(slacks)
                dual_vars[t] = min(slacks)
                objective_value += dual_vars[t]
                slacks -= dual_vars[t] # reduce slack by dual variable

                # buy permit of type k_min at time t
                primal_vars[k_min, t - (t % self.durations[k_min])] = 1.0
                for k in range(k_min):
                    for s in range(t - (t % self.durations[k_min]), t):
                        primal_vars[k, s] = 0.0

        return objective_value, dual_vars, primal_vars
    
    def calculateOptimalCostNonLaminar(self):
                    
        dual_vars = np.zeros(self.time_duration, dtype=float)
        primal_vars = np.zeros((self.number_permits, self.time_duration), dtype=float)

        # calculate dual variables
        objective_value = 0.0
        slacks = np.zeros(self.number_permits, dtype=float) # slack for each permit type

        slacks = self.costs.copy().astype(float) # initialize slacks

        for t in range(self.time_duration):
            for k in range(self.number_permits):
                slacks[k] += dual_vars[t - self.durations[k]] if t - self.durations[k] >= 0 else 0.0

            if self.request_sequence[t] != 0:
                # dual variable calculations
                k_min = np.argmin(slacks)
                dual_vars[t] = min(slacks)
                objective_value += dual_vars[t]
                slacks -= dual_vars[t] # reduce slack by dual variable

                # buy permit of type k_min at time t
                for k in range(k_min+1):
                    for s in range(t - self.durations[k_min]+1, t):
                        primal_vars[k, s] = 0.0
                primal_vars[k_min, t - self.durations[k_min] + 1] = 1.0

        return objective_value, dual_vars, primal_vars

    def competitiveRatioNonLaminar(self):
        opt, _, _ = self.calculateOptimalCostNonLaminar()
        online_cost, _, _ = self.calculateOnlineCost()

        if opt == 0.0 and online_cost == 0.0:
            return 1.0
        return online_cost / opt if opt > 0.0 else float('inf')


    def calculateOnlineCost(self):
        raise NotImplementedError("This method should be implemented by subclasses.")

class OnlineDeterministicSolver(ParkingPermitSolver):
    
    def calculateOnlineCost(self):
                    
        dual_vars = np.zeros(self.time_duration, dtype=float)
        primal_vars = np.zeros((self.number_permits, self.time_duration), dtype=float)

        # calculate dual variables
        objective_value = 0.0
        slacks = self.costs.copy().astype(float) # initialize slacks

        for t in range(self.time_duration):
            for k in range(self.number_permits):
                slacks[k] += dual_vars[t - self.durations[k]] if t - self.durations[k] >= 0 else 0.0

            if self.request_sequence[t] != 0 and all(slacks > 0.0):
                # dual variable calculations
                k_min = np.argmin(slacks)
                dual_vars[t] = min(slacks)
                slacks -= dual_vars[t] # reduce slack by dual variable

                # buy permit of type k_min at time t
                primal_vars[k_min, t] = 1.0
                objective_value += self.costs[k_min]
    

        return objective_value, dual_vars, primal_vars

class OnlineRandomizedSolver(ParkingPermitSolver):

    y_step = 0.0001

    def calculateOnlineCost(self):

        dual_vars = np.zeros(self.time_duration, dtype=float)
        primal_vars = np.zeros((self.number_permits, self.time_duration), dtype=float)
        K = self.number_permits

        # calculate dual variables
        objective_value = 0.0
        permit_quantity = np.zeros(self.number_permits, dtype=float) # initialize permit quantities


        for t in range(self.time_duration):
            for k in range(self.number_permits):
                permit_quantity[k] -= primal_vars[k][t - self.durations[k]] if t - self.durations[k] >= 0 else 0.0

            if self.request_sequence[t] != 0 and sum(permit_quantity) < 1.0:
                current_purchases = np.zeros(self.number_permits, dtype=float)
                y = 0.0

                while sum(permit_quantity)+sum(current_purchases) < 1.0:
                    y += self.y_step
                    current_purchases = (1.0/(K * np.log(K)) + permit_quantity) * (np.exp(y / self.costs) - 1.0)

                dual_vars[t] = y
                primal_vars[:, t] = current_purchases
                permit_quantity += current_purchases
                objective_value += np.dot(self.costs, current_purchases)

                

        return objective_value, dual_vars, primal_vars
    
class DualAugmentedSolver(ParkingPermitSolver):

    def __init__(self, alpha):
        super().__init__()
        self.dual_predictions = None
        self.alpha = alpha

    def setDualPredictions(self, dual_predictions: np.array):
        """Sets the dual predictions for the instance."""

        self.dual_predictions = dual_predictions

    def calculateOnlineCost(self):
        
        # simulate days which don't work
        solverRandomized = OnlineRandomizedSolver()
        uncovered_days = np.zeros(self.time_duration, dtype=int)

        objective_value = 0.0
        primal_vars = np.zeros((self.number_permits, self.time_duration), dtype=float)
        dual_vars = np.zeros(self.time_duration, dtype=float)
        permit_quantity = np.zeros(self.number_permits, dtype=float) # initialize permit quantities



        for t in range(self.time_duration):
            for k in range(self.number_permits):
                permit_quantity[k] -= primal_vars[k][t - self.durations[k]] if t - self.durations[k] >= 0 else 0.0


            if self.request_sequence[t] != 0 and sum(permit_quantity) < 1.0:

                # look for k-saturated permits
                k_saturated = -1
                for k in range(self.number_permits):
                    left_endpoint = t - (t%self.durations[k])
                    right_endpoint = min(left_endpoint + self.durations[k], self.time_duration)
                    if self.dual_predictions[left_endpoint:right_endpoint].sum() >= self.costs[k] * self.alpha:
                        k_saturated = k

                if k_saturated != -1:
                    # buy highest k-saturated permit
                    primal_vars[k_saturated, t] = 1.0
                    permit_quantity[k_saturated] += 1.0
                    objective_value += self.costs[k_saturated]
                else:
                    uncovered_days[t] = 1


        solverRandomized.setPermitTypes(self.costs, self.durations)
        solverRandomized.setInstance(uncovered_days)


        # simulate uncovered days with randomized algorithm
        objective_value_randomized, _, primal_vars_randomized = solverRandomized.calculateOnlineCost()
        return objective_value+objective_value_randomized, dual_vars, primal_vars + primal_vars_randomized

            

        

def main():
    # Example usage
    costs = np.array([10, 15, 25])
    durations = np.array([1, 2, 4])
    request_sequence = np.array([1, 1, 1, 1, 1, 1, 1, 1, 1, 1])

    solverRandomized = OnlineRandomizedSolver()
    solverRandomized.setPermitTypes(costs, durations)
    solverRandomized.setInstance(request_sequence)

    online_cost, dual_vars, primal_vars = solverRandomized.calculateOnlineCost()
    print("Online Cost:", online_cost)
    print("Dual Variables:", dual_vars)
    print("Primal Variables:\n", primal_vars)

    solverDual = DualAugmentedSolver(alpha=0.5)
    solverDual.setPermitTypes(costs, durations)
    solverDual.setInstance(request_sequence)
    solverDual.setDualPredictions(np.zeros(len(request_sequence), dtype=float))

    online_cost_dual, dual_vars_dual, primal_vars_dual = solverDual.calculateOnlineCost()
    print("Online Cost (Dual):", online_cost_dual)
    print("Dual Variables (Dual):", dual_vars_dual)
    print("Primal Variables (Dual):\n", primal_vars_dual)


if __name__ == "__main__":
    main()