# !/usr/bin/env python
# coding: utf-8

# Importing python packages
import numpy as np
from sklearn.preprocessing import PolynomialFeatures


# OFD problem instance fixed: dim control the space of item and agent vectors
def problem_instance_fixed1(base_dim, items, item_copies, agents, rho):
    # ######## Problem Instances ########
    agent_dim   = base_dim
    item_dim    = base_dim
    lambda_val  = 0.01
    l_value     = np.sqrt(agent_dim + item_dim)
    s_value     = np.sqrt(agent_dim + item_dim)
    sigma_val   = 0.1
    delta_val   = 0.05

    # Set the seed
    np.random.seed(0)
    
    # Generate items
    base_item_vectors = np.random.uniform(low=0, high=1, size=(items, item_dim))
    
    # Replicate items
    item_vectors = np.copy(base_item_vectors)
    for _ in range(1, item_copies):
        item_vectors = np.concatenate((item_vectors, base_item_vectors), axis=0)
    
    # Add one item for each agent 
    base_item_agents = np.random.uniform(low=0, high=1, size=(agents, item_dim))
    item_vectors = np.concatenate((base_item_agents, item_vectors), axis=0)
    
    # Generate agents
    agent_vectors = np.random.uniform(low=0, high=1, size=(agents, agent_dim))
    
    # Concating item and agent vectors
    item_agents_list = []
    for item in item_vectors:
        item_agents  = []
        for agent in agent_vectors:
            item_agents.append(np.append(item, agent))

        item_agents_list.append(item_agents)
        
    item_agents = np.array(item_agents_list)
    
    # Generate theta
    dim = agent_dim + item_dim
    theta = np.random.uniform(low=0, high=10, size=dim)
    theta = theta / np.linalg.norm(theta)  # Normalizing theta
    
    # Weights for each agent <=1
    agent_weights = np.array([(rho**a) for a in range(agents)])
     
    # Algorithm parameters: [dimension, lambda, L, S, sigma, delta, agent_weights]
    parameters = [item_agents, theta, dim, lambda_val, l_value, s_value, sigma_val, delta_val, agent_weights]

    return parameters


# OFD problem instance fixed: dim control the space of item and agent vectors
def problem_instance_fixed(base_dim, items, item_copies, agents, rho):
    # ######## Problem Instances ########
    agent_dim   = base_dim
    item_dim    = base_dim
    lambda_val  = 0.01
    l_value     = np.sqrt(agent_dim + item_dim)
    s_value     = np.sqrt(agent_dim + item_dim)
    sigma_val   = 0.1
    delta_val   = 0.05

    # Set the seed
    np.random.seed(0)
    
    # Generate items
    base_item_vectors = np.random.uniform(low=0, high=10, size=(items, item_dim))
    
    # Replicate items
    item_vectors = np.copy(base_item_vectors)
    for _ in range(1, item_copies):
        item_vectors = np.concatenate((item_vectors, base_item_vectors), axis=0)
    
    # Add one item for each agent 
    base_item_agents = np.random.uniform(low=0, high=10, size=(agents, item_dim))
    item_vectors = np.concatenate((base_item_agents, item_vectors), axis=0)
    
    # Generate agents
    agent_vectors = np.random.uniform(low=0, high=10, size=(agents, agent_dim))
    
    # Concating item and agent vectors
    item_agents_list = []
    for item in item_vectors:
        item_agents  = []
        for agent in agent_vectors:
            item_agents.append(np.append(item, agent))

        item_agents_list.append(item_agents)
        
    item_agents = np.array(item_agents_list)
    
    # Generate theta
    dim = agent_dim + item_dim
    theta = np.random.uniform(low=0, high=10, size=dim)
    theta = theta / np.linalg.norm(theta)  # Normalizing theta
    
    # Weights for each agent <=1
    agent_weights = np.array([(rho**a) for a in range(agents)])
     
    # Algorithm parameters: [dimension, lambda, L, S, sigma, delta, agent_weights]
    parameters = [item_agents, theta, dim, lambda_val, l_value, s_value, sigma_val, delta_val, agent_weights]

    return parameters


# OFD problem instance fixed non-linear: dim control the space of item and agent vectors
def problem_instance_fixed_nlin(base_dim, items, item_copies, agents, rho):
    # ######## Problem Instances ########
    agent_dim   = base_dim
    item_dim    = base_dim
    lambda_val  = 0.01
    l_value     = np.sqrt(agent_dim + item_dim)
    s_value     = np.sqrt(agent_dim + item_dim)
    sigma_val   = 0.1
    delta_val   = 0.05

    # Set the seed
    np.random.seed(0)
    
    # Generate items
    base_item_vectors = np.random.uniform(low=0, high=10, size=(items, item_dim))
    
    # Replicate items
    item_vectors = np.copy(base_item_vectors)
    for _ in range(1, item_copies):
        item_vectors = np.concatenate((item_vectors, base_item_vectors), axis=0)

    # print(item_vectors.shape, items, item_copies)
    
    # Add one item for each agent 
    base_item_agents = np.random.uniform(low=0, high=10, size=(agents, item_dim))
    item_vectors = np.concatenate((base_item_agents, item_vectors), axis=0)

    # Generate agents
    agent_vectors = np.random.uniform(low=0, high=10, size=(agents, agent_dim))
    
    # Concating item and agent vectors
    item_agents_list = []
    for item in item_vectors:
        item_agents  = []
        for agent in agent_vectors:
            item_agents.append(np.append(item, agent))

        item_agents_list.append(item_agents)
        
    item_agents = np.array(item_agents_list)

    # Dataset pre-processing for non-linear kernel
    item_agents = item_agents.reshape((items + agents) * agents, -1)
    degree      = 2                                 # Polynomial kernel degree
    poly        = PolynomialFeatures(degree)        # Polynomial kernel
    hd_data     = poly.fit_transform(item_agents)   # Lifting the data
    hd_data_no1 = np.delete(hd_data, 0, axis=1)     # Removing first column as it is 1 only
    item_agents = np.delete(hd_data_no1,-1,axis=1)  # Removing last column as it is 1 only
    dim     = len(item_agents[0])  

    # Unflattening the data
    item_agents = item_agents.reshape(items + agents, agents, -1)
    
    # Generate theta  
    theta   = np.random.uniform(low=0, high=10, size=dim)
    theta   = theta / np.linalg.norm(theta)  # Normalizing theta
    
    # Weights for each agent <=1
    agent_weights = np.array([(rho**a) for a in range(agents)])
     
    # Algorithm parameters: [dimension, lambda, L, S, sigma, delta, agent_weights]
    parameters = [item_agents, theta, dim, lambda_val, l_value, s_value, sigma_val, delta_val, agent_weights]

    return parameters


# OFD problem instance type1: dim control the space of item and agent vectors
def problem_instance_type1(base_dim, items, item_copies, agents, rho):
    # ######## Problem Instances ########
    agent_dim   = base_dim
    item_dim    = base_dim
    lambda_val  = 0.01
    l_value     = np.sqrt(agent_dim + item_dim)
    s_value     = np.sqrt(agent_dim + item_dim)
    sigma_val   = 0.1
    delta_val   = 0.05

    # Set the seed
    np.random.seed(0)
    
    # Generate items
    base_item_vectors = np.random.uniform(low=0, high=base_dim, size=(items, item_dim))
    
    # Replicate items
    item_vectors = np.copy(base_item_vectors)
    for _ in range(1, item_copies):
        item_vectors = np.concatenate((item_vectors, base_item_vectors), axis=0)
    
    # Add one item for each agent 
    base_item_agents = np.random.uniform(low=0, high=base_dim, size=(agents, item_dim))
    item_vectors = np.concatenate((base_item_agents, item_vectors), axis=0)
    
    # Generate agents
    agent_vectors = np.random.uniform(low=0, high=base_dim, size=(agents, agent_dim))
    
    # Concating item and agent vectors
    item_agents_list = []
    for item in item_vectors:
        item_agents  = []
        for agent in agent_vectors:
            item_agents.append(np.append(item, agent))

        item_agents_list.append(item_agents)
        
    item_agents = np.array(item_agents_list)
    
    # Generate theta
    dim = agent_dim + item_dim
    theta = np.random.uniform(low=0, high=1, size=dim)
    theta = theta / np.linalg.norm(theta)  # Normalizing theta
    
    # Weights for each agent <=1
    agent_weights = np.array([(rho**a) for a in range(agents)])
     
    # Algorithm parameters: [dimension, lambda, L, S, sigma, delta, agent_weights]
    parameters = [item_agents, theta, dim, lambda_val, l_value, s_value, sigma_val, delta_val, agent_weights]

    return parameters


# OFD problem instance type2: number of agents control space of item and agent vectors
def problem_instance_type2(base_dim, items, item_copies, agents, rho):
    # ######## Problem Instances ########
    agent_dim   = base_dim
    item_dim    = base_dim
    lambda_val  = 0.01
    l_value     = np.sqrt(agent_dim + item_dim)
    s_value     = np.sqrt(agent_dim + item_dim)
    sigma_val   = 0.1
    delta_val   = 0.05

    # Set the seed
    np.random.seed(0)
    
    # Generate items
    base_item_vectors = np.random.uniform(low=0, high=agents, size=(items, item_dim))
    
    # Replicate items
    item_vectors = np.copy(base_item_vectors)
    for _ in range(1, item_copies):
        item_vectors = np.concatenate((item_vectors, base_item_vectors), axis=0)
    
    # Add one item for each agent 
    base_item_agents = np.random.uniform(low=0, high=agents, size=(agents, item_dim))
    item_vectors = np.concatenate((base_item_agents, item_vectors), axis=0)
    
    # Generate agents
    agent_vectors = np.random.uniform(low=0, high=agents, size=(agents, agent_dim))
    
    # Concating item and agent vectors
    item_agents_list = []
    for item in item_vectors:
        item_agents  = []
        for agent in agent_vectors:
            item_agents.append(np.append(item, agent))

        item_agents_list.append(item_agents)
        
    item_agents = np.array(item_agents_list)
    
    # Generate theta
    dim = agent_dim + item_dim
    theta = np.random.uniform(low=0, high=1, size=dim)
    theta = theta / np.linalg.norm(theta)  # Normalizing theta
    
    # Weights for each agent
    agent_weights = np.array([(rho**a) for a in range(agents)])

    # Algorithm parameters: [dimension, lambda, L, S, sigma, delta, agent_weights]
    parameters = [item_agents, theta, dim, lambda_val, l_value, s_value, sigma_val, delta_val, agent_weights]

    return parameters


# OFD problem instance type1: dim control the space of item and agent vectors
def problem_instance(type, base_dim, items, item_copies, agents, rho):
    # ######## Problem Instances ########
    agent_dim   = base_dim
    item_dim    = base_dim
    lambda_val  = 0.01
    l_value     = np.sqrt(agent_dim + item_dim)
    s_value     = np.sqrt(agent_dim + item_dim)
    sigma_val   = 0.1
    delta_val   = 0.05

    # Set the seed
    np.random.seed(0)
    
    # Generate items
    base_item_vectors = np.random.uniform(low=0, high=base_dim, size=(items, item_dim))
    
    # Replicate items
    item_vectors = np.copy(base_item_vectors)
    for _ in range(1, item_copies):
        item_vectors = np.concatenate((item_vectors, base_item_vectors), axis=0)
    
    # Add one item for each agent 
    base_item_agents = np.random.uniform(low=0, high=base_dim, size=(agents, item_dim))
    item_vectors = np.concatenate((base_item_agents, item_vectors), axis=0)
    
    # Generate agents
    agent_vectors = np.random.uniform(low=0, high=base_dim, size=(agents, agent_dim))
    
    # Concating item and agent vectors
    item_agents_list = []
    for item in item_vectors:
        item_agents  = []
        for agent in agent_vectors:
            item_agents.append(np.append(item, agent))

        item_agents_list.append(item_agents)
        
    item_agents = np.array(item_agents_list)
    
    # Generate theta
    dim = agent_dim + item_dim
    if type == "easy":
        theta = np.random.uniform(low=0, high=1, size=dim)
        theta = theta / np.linalg.norm(theta)  # Normalizing theta
        
    elif type == "medium":
        theta = np.random.uniform(low=0, high=10, size=dim)
        
    elif type == "hard":
        theta = np.random.uniform(low=0, high=100, size=dim)
        
    else:
        raise ValueError("Invalid problem type")
    
    # Weights for each agent
    agent_weights = np.array([(rho**a) for a in range(agents)])

    # Algorithm parameters: [dimension, lambda, L, S, sigma, delta, agent_weights]
    parameters = [item_agents, theta, dim, lambda_val, l_value, s_value, sigma_val, delta_val, agent_weights]

    return parameters

