from tools import * 

from sage.all import macaulay2 



# s4 basis 
prime_number = 65521 

b1 = binary_form_class(4, 4, 'm',
                       modulo_prime = prime_number,
                       symbol_str = 'p')

b2 = binary_form_class(4, 4, 'n',
                       modulo_prime = prime_number,
                       symbol_str = 'q')

b3 = transvectant_class(b1, b1, 4, modulo_prime = prime_number, expl_symbol = 'z_1')
b4 = transvectant_class(b2, b2, 4, modulo_prime = prime_number, expl_symbol = 'z_2')
b5 = transvectant_class(b1, b2, 4, modulo_prime = prime_number, expl_symbol = 'z_3')
b6 = transvectant_class(b1, b2, 3, modulo_prime = prime_number, expl_symbol = 'z_4')
b7 = transvectant_class(b1, b1, 2, modulo_prime = prime_number, expl_symbol = 'z_5')
b8 = transvectant_class(b2, b2, 2, modulo_prime = prime_number, expl_symbol = 'z_6')
b9 = transvectant_class(b1, b2, 2, modulo_prime = prime_number, expl_symbol = 'z_7')
b10 = transvectant_class(b1, b2, 1, modulo_prime = prime_number, expl_symbol = 'z_8')
b11 = transvectant_class(b1, b7, 4, modulo_prime = prime_number, expl_symbol = 'z_9')
b12 = transvectant_class(b2, b8, 4, modulo_prime = prime_number, expl_symbol = 'z_10')
b13 = transvectant_class(b1, b8, 4, modulo_prime = prime_number, expl_symbol = 'z_11')
b14 = transvectant_class(b2, b7, 4, modulo_prime = prime_number, expl_symbol = 'z_12')

b15 = transvectant_class(b1, b8, 3, modulo_prime = prime_number, expl_symbol = 'z_13')
b16 = transvectant_class(b2, b7, 3, modulo_prime = prime_number, expl_symbol = 'z_14')
b17 = transvectant_class(b1, b8, 2, modulo_prime = prime_number, expl_symbol = 'z_15')
b18 = transvectant_class(b2, b7, 2, modulo_prime = prime_number, expl_symbol = 'z_16')
b19 = transvectant_class(b1, b7, 1, modulo_prime = prime_number, expl_symbol = 'z_17')
b20 = transvectant_class(b2, b8, 1, modulo_prime = prime_number, expl_symbol = 'z_18')
b21 = transvectant_class(b1, b8, 1, modulo_prime = prime_number, expl_symbol = 'z_19')
b22 = transvectant_class(b2, b7, 1, modulo_prime = prime_number, expl_symbol = 'z_20')
b23 = transvectant_class(b7, b8, 4, modulo_prime = prime_number, expl_symbol = 'z_21')
b24 = transvectant_class(b7, b8, 3, modulo_prime = prime_number, expl_symbol = 'z_22')
b25 = transvectant_class(b19, b2, 4, modulo_prime = prime_number, expl_symbol = 'z_23')
b26 = transvectant_class(b1, b20, 4, modulo_prime = prime_number, expl_symbol = 'z_24')
b27 = transvectant_class(b1**2, b20, 6, modulo_prime = prime_number, expl_symbol = 'z_25')
b28 = transvectant_class(b19, b2**2, 6, modulo_prime = prime_number, expl_symbol = 'z_26')

s4s4_min_set = [b3, b4, b5, b11, b12, b13, b14, b23, b1, b2, b6, b7, b8, b9, b10, b15, b16, b17, b18, b19, b20, b21, b22, b24, b25, b26, b27, b28]



gens_wo_invs = [i for i in s4s4_min_set if i.order > 0]
gens_invs = [i for i in s4s4_min_set if i.order == 0]

total = 0 

gens_polys = [] 

order = 4

# Covariant Module Generators 

for degree in range(1, 30): 

    gen_set_polys_d2, b = generate_homo_space(degree, order, gens_wo_invs)

    total += len(gen_set_polys_d2) 
    if len(gen_set_polys_d2) > 0:
        print(f"degree-{degree}.")
        print(f"Covariant Module Generators: {len(gen_set_polys_d2)}")

    gens_polys.extend(gen_set_polys_d2)

print("Total number of polys generated: ", total)
print("Total Number of Invariants: ", len(gens_invs))


# create the base field and the base rings
macaulay2('K = ZZ/65521;')
macaulay2('R = K[m0, m1, m2, m3, m4, n0, n1, n2, n3, n4, h0, w0, x, y];')

# load invariants 
ind =  1 
for poly in gens_invs: 
    inv_poly = f"I{ind} = " +  str(poly.evaluate().as_expr()).replace("**", "^") + ";" 
    ind += 1 
    # print(inv_poly)
    macaulay2(inv_poly) 


macaulay2("I9 = h0;")
macaulay2("I10 = w0;")


macaulay2("J = gb ideal(I1, I2, I3, I4, I5, I6, I7, I8, I9, I10);")



# In Linear case, no degree 1 invariants are found. Nature of harmonic tensors. 

# load module generators  
ind =  1 
for poly in gens_polys: 
    cov_poly = f"f{ind} = " +  str(poly.evaluate().as_expr()).replace("**", "^") + ";" 
    ind += 1 
    # print(cov_poly)
    macaulay2(cov_poly) 


# fi %  J 
for i in range(1, len(gens_polys) + 1): 
    cmd = f"mf{i} = f{i} % J"
    macaulay2(cmd)

# create a list of all fi % J 

cmd = "F = {{"
for i in range(1, len(gens_polys) + 1):
    cmd += f"mf{i}, "
cmd = cmd[:-2] + "}}"

# print(cmd)

# select only non zero polys 

macaulay2(cmd)
macaulay2("nonZeroF = select(flatten F, e -> e!=0);")
macaulay2("# nonZeroF")


def generate_mons_string(n):
    terms = []
    for i in range(n, -1, -1):
        if i == n:
            terms.append(f"x^{n}")
        elif i == 0:
            terms.append(f"y^{n}")
        else:
            terms.append(f"x^{i}*y^{n-i}")
    mons_str = "mons = {" + ", ".join(terms) + "}"
    return mons_str

def generate_subList_cmd(var_prefixes, degrees):
    terms = []
    idx = 0
    for prefix, deg in zip(var_prefixes, degrees):
        for i in range(deg + 1):
            terms.append(f"{prefix}{i} => rand_{idx}")
            idx += 1
    cmd = "subList = {" + ", ".join(terms) + "};"
    return cmd

cmd = generate_mons_string(4)
print(cmd)
macaulay2(cmd)


numCoeffs = int(str(macaulay2("numgens R"))) - 2 

print(f"There are {numCoeffs} coefficients in each polynomial.")

matrix = [] 

subListCmd = generate_subList_cmd(["m", "n"], [4, 4])
numPoints = int(str(macaulay2("# nonZeroF")))
for pt in range(numPoints): 
    macaulay2(f"rand = apply({numCoeffs}, i -> random(1, 65521));")
    macaulay2(subListCmd)
    macaulay2("Fsub = apply(nonZeroF, e -> substitute(e, subList))")
    macaulay2("evalMatrix = apply(Fsub, fsub -> flatten entries last coefficients(fsub, Monomials => mons))")
    mat = np.array(macaulay2("matrix evalMatrix").sage(), dtype = int)
    if len(matrix) == 0: 
        matrix = mat 
    else:
        matrix = np.concatenate((matrix, mat), axis=1)

min_rows = fast_rref(matrix, 65521, True)

print(f"There are {len(min_rows)} R-linearly independent polynomials.")