def eval_function_from_str(str_expr, vars):
    var_dict = {}
    for i, var in enumerate(vars):
        var_dict.update({
            f"x{i}": var,
        })
    try:
        ret = eval(
            str_expr,
            var_dict
        )
    except:
        return -1e20
    return ret

def show_expand_variable(expand_variable):
    num_variables = len(expand_variable)
    result = ""
    for i in range(num_variables):
        for j in range(expand_variable[i]):
            if i == num_variables - 1:
                result += "y "
            else:
                result += f"x{i} "
    if result == "":
        result = "c"
    result = result.strip(' ').replace(' ', '*')
    return result

def expand_variables_funx(num_x, num_op, K, show_expand_variables=False):
    # num_x and y
    num_variables = num_x + 1

    expand_variables = [[]]
    i = 0
    while i < num_variables:
        new_expand_variables = []
        for expand_variable in expand_variables:
            used_K = sum(expand_variable)
            if len(expand_variable) <= i:
                for alloted_k in range(K - used_K + 1):
                    if i == num_variables - 1 and alloted_k > 1: continue
                    new_expand_variable = expand_variable + [alloted_k]
                    new_expand_variables.append(new_expand_variable)
        i += 1
        expand_variables = new_expand_variables

    final_expand_variables = []
    for expand_variable in expand_variables:
        if max(expand_variable[:-1]) <= max(1, num_op + 2 - num_x):
            final_expand_variables.append(expand_variable)
    expand_variables = final_expand_variables

    # print(len(expand_variables), expand_variables)
    if show_expand_variables:
        for expand_variable in expand_variables:
            print(show_expand_variable(expand_variable), )
    return expand_variables


def compute_expand_variable(x, num_op, K, y):
    results = []
    v = x + [y]
    expand_variables = expand_variables_funx(len(x), num_op, K)
    for expand_variable in expand_variables:
        num_variables = len(expand_variable)
        result = 1
        for i in range(num_variables):
            for j in range(expand_variable[i]):
                result *= v[i]
        results.append(result)
    return results


def show_computed_funx(x, expand_variables, precision):
    num_expand_variables = len(expand_variables)
    result = ""
    for i in range(num_expand_variables):
        if abs(x[i]) > precision:
            if result != "" and x[i] > 0: result = result + "+"
            result = result + str(round(x[i] / precision) * precision) + "*" + show_expand_variable(expand_variables[i])
    return result