import random
import string
from tqdm import tqdm
import json
# create two lists of integers, The number range of the list is [0, 100]
# generate 30 operations on one list, or between two lists, like list.pop(), list.pop(0), list.append(), list.insert(), list.remove(), list.reverse(), 'a = list1.pop(0)\nlist2.append(a)', 'list2.append(list1[-1])',  etc, as well as if conditions, like 'if list1[0] > list2[0]:', etc
# do not generate random number in the operations, please use operations like list.pop(*) to get the number
# then each time, sample 5-8 operations or if sentences to generate a code snippet

list1 = [random.randint(0, 100) for _ in range(10)]
list2 = [random.randint(0, 100) for _ in range(10)]

class RFFT_funcs:
    def func_0(list1, list2, val):
        if list1:
            updated_list1 = list1[:]
            updated_list1[-1] += val
            output = f'''```
if list1:
    list1[-1] += {val}
```
list1 = {list1}
enter if
list1[-1] = {list1[-1]} + {val} = {list1[-1] + val}
now,
list1 = {updated_list1}
'''
        else:
            output = f'''```
if list1:
    list1[-1] += {val}
```
list1 = {list1}
do not enter if
'''
        return output

    def func_1(list1, list2, val):
        if list1:
            updated_list1 = list1[:]
            updated_list1[0] += val
            output = f'''```
if list1:
    list1[0] += {val}
```
list1 = {list1}
enter if
list1[0] = {list1[0]} + {val} = {list1[0] + val}
now,
list1 = {updated_list1}
'''
        else:
            output = f'''```
if list1:
    list1[0] += {val}
```
list1 = {list1}
do not enter if
'''
        return output

    def func_2(list1, list2, val):
        
        updated_list1 = list1[1:]
        updated_list2 = list2[:]
        
        if list1:
            var = list1[0]
            updated_list2.append(var)
            output = f'''```
if list1:
    var = list1.pop(0)
    list2.append(var)
```
list1 = {list1}
enter if
list2 = {list2}
var = {var}
now,
list1 = {updated_list1}
list2 = {updated_list2}
'''
        else:
            output = f'''```
if list1:
    var = list1.pop(0)
    list2.append(var)
```
list1 = {list1}
do not enter if
'''
        return output

    def func_3(list1, list2, val):
        var = list1[-1] if list1 else None
        updated_list1 = list1[:-1]
        updated_list2 = list2[:]
        if var is not None:
            updated_list2.append(var)
            output = f'''```
if list1:
    var = list1.pop()
    list2.append(var)
```
list1 = {list1}
enter if
list2 = {list2}
var = {var}
now,
list1 = {updated_list1}
list2 = {updated_list2}
''' 
        else:
            output = f'''```
if list1:
    var = list1.pop()
    list2.append(var)
```
list1 = {list1}
do not enter if
'''
        return output

    def func_4(list1, list2, val):
        updated_list1 = list1[:]
        updated_list1.insert(0, val)
        output = f'''```
list1.insert(0, {val})
```
list1 = {list1}
now,
list1 = {updated_list1}
'''
        return output

    def func_5(list1, list2, val):
        updated_list1 = sorted(list1)
        output = f'''```
list1.sort()
```
list1 = {list1}
now,
list1 = {updated_list1}
'''
        return output

    def func_6(list1, list2, val):
        updated_list1 = list1[::-1]
        output = f'''```
list1.reverse()
```
list1 = {list1}
now,
list1 = {updated_list1}
'''
        return output

    def func_7(list1, list2, val):
        updated_list1 = list1[1:] if list1 else list1
        output = f'''```
list1 = list1[1:] if list1 else list1
```
list1 = {list1}
now,
list1 = {updated_list1}
'''
        return output

    def func_8(list1, list2, val):
        if list2:
            updated_list1 = list1[:]
            updated_list1.insert(0, list2[0])
            output = f'''```
if list2:
    list1.insert(0, list2[0])
```
list2 = {list2}
enter if
list1 = {list1}
list2[0] = {list2[0]}
now,
list1 = {updated_list1}
'''
        else:
            updated_list1 = list1[:]
            updated_list1.insert(0, val)
            output = f'''```
if list2:
    list1.insert(0, list2[0])
```
list2 = {list2}
do not enter if
```
else:
    list1.insert(0, {val})
```
list1 = {list1}
now,
list1 = {updated_list1}
'''
        return output

    def func_9(list1, list2, val):
        updated_list1 = list1[:]
        updated_list1.append(list2[-1] if list2 else val)
        output = f'''```
val = list2[-1] if list2 else {val}
list1.append(val)
```
list2 = {list2}
val = {list2[-1] if list2 else val}
list1 = {list1}
now,
list1 = {updated_list1}
'''
        return output

    def func_10(list1, list2, val):
        if list1 and list2 and list1[0] > list2[0]:
            updated_list1 = list1[:]
            updated_list1.pop(0)
            output = f'''```
if list1 and list2 and list1[0] > list2[0]:
    list1.pop(0)
```
list1 = {list1}
list2 = {list2}
list1[0] = {list1[0] if list1 else ''}
list2[0] = {list2[0] if list2 else ''}
enter if
now,
list1 = {updated_list1}
'''
        else:
            output = f'''```
if list1 and list2 and list1[0] > list2[0]:
    list1.pop(0)
```
list1 = {list1}
list2 = {list2}
list1[0] = {list1[0] if list1 else ''}
list2[0] = {list2[0] if list2 else ''}
do not enter if
'''
        return output

    def func_11(list1, list2, val):
        if list1 and list2 and list1[-1] < list2[-1]:
            updated_list1 = list1[:]
            updated_list1.pop()
            output = f'''```
if list1 and list2 and list1[-1] < list2[-1]:
    list1.pop()
```
list1 = {list1}
list2 = {list2}
list1[0] = {list1[0] if list1 else ''}
list2[0] = {list2[0] if list2 else ''}
enter if
now,
list1 = {updated_list1}
'''
        else:
            output = f'''```
if list1 and list2 and list1[-1] < list2[-1]:
    list1.pop()
```
list1 = {list1}
list2 = {list2}
list1[0] = {list1[0] if list1 else ''}
list2[0] = {list2[0] if list2 else ''}
do not enter if
'''
        return output

    def func_12(list1, list2, val): # TO CHECK
        if list1:
            updated_list1 = list1[:]
            updated_list1.pop(0)
            output = f'''```
if list1:
    list1.pop(0)
```
list1 = {list1}
enter if
now,
list1 = {updated_list1}
'''
        else:
            output = f'''```
if list1:
    list1.pop(0)
```
list1 = {list1}
do not enter if
'''
        return output

    def func_13(list1, list2, val):
        if list1 and list2:
            updated_list1 = list1[:]
            updated_list2 = list2[:]
            updated_list1.append(updated_list2.pop())
            output = f'''```
if list1 and list2:
    list1.append(list2.pop())
```
list1 = {list1}
list2 = {list2}
enter if
list1.append({list2[-1]})
now,
list1 = {updated_list1}
list2 = {updated_list2}
'''
        else:
            output = f'''```
if list1 and list2:
    list1.append(list2.pop())
```
list1 = {list1}
list2 = {list2}
do not enter if
'''
        return output

    def func_14(list1, list2, val):
        if list1 and list1[0] % 2 == 0:
            updated_list1 = list1[:]
            updated_list1.pop(0)
            output = f'''```
if list1 and list1[0] % 2 == 0:
    list1.pop(0)
```
list1 = {list1}
list1[0] % 2 = {list1[0]} % 2 == 0
enter if
now,
list1 = {updated_list1}
'''
        elif list1:
            output = f'''```
if list1 and list1[0] % 2 == 0:
    list1.pop(0)
```
list1 = {list1}
list1[0] % 2 = {list1[0]} % 2 != 0
do not enter if
'''
        else:
            output = f'''```
if list1 and list1[0] % 2 == 0:
    list1.pop(0)
```
list1 = {list1}
do not enter if
'''
        return output

    def func_15(list1, list2, val):
        if list1 and list1[0] % 2 == 1:
            updated_list1 = list1[:]
            updated_list1.pop(0)
            output = f'''```
if list1 and list1[0] % 2 == 1:
    list1.pop(0)
```
list1 = {list1}
list1[0] % 2 = {list1[0]} % 2 == 1
enter if
now,
list1 = {updated_list1}
'''
        elif list1:
            output = f'''```
if list1 and list1[0] % 2 == 1:
    list1.pop(0)
```
list1 = {list1}
list1[0] % 2 = {list1[0]} % 2 != 1
do not enter if
'''
        else:
            output = f'''```
if list1 and list1[0] % 2 == 1:
    list1.pop(0)
```
list1 = {list1}
do not enter if
'''
        return output

    def func_16(list1, list2, val):
        if len(list1) > len(list2):
            updated_list1 = list1[:]
            updated_list2 = list2[:]
            updated_list2.insert(0, updated_list1.pop())
            output = f'''```
if len(list1) > len(list2):
    list2.insert(0, list1.pop())
```
list1 = {list1}
list2 = {list2}
len(list1) = {len(list1)}
len(list2) = {len(list2)}
{len(list1)} > {len(list2)}
enter if
list1.pop() = {list1[-1]}
now,
list1 = {updated_list1}
list2 = {updated_list2}
'''
        else:
            output = f'''```
if len(list1) > len(list2):
    list2.insert(0, list1.pop())
```
list1 = {list1}
list2 = {list2}
len(list1) = {len(list1)}
len(list2) = {len(list2)}
{len(list1)} <= {len(list2)}
do not enter if
'''
        return output

    def func_17(list1, list2, val):
        if list1 and list1[0] > val:
            updated_list1 = list1[:]
            updated_list1.pop(0)
            output = f'''```
if list1 and list1[0] > {val}:
    list1.pop(0)
```
list1 = {list1}
list1[0] = {list1[0]} > {val}
enter if
now,
list1 = {updated_list1}
'''
        elif list1:
            output = f'''```
if list1 and list1[0] > {val}:
    list1.pop(0)
```
list1 = {list1}
list1[0] = {list1[0]} <= {val}
do not enter if
'''
        else:
            output = f'''```
if list1 and list1[0] > {val}:
    list1.pop(0)
```
list1 = {list1}
do not enter if
'''
        return output

    def func_18(list1, list2, val):
        if list1 and list1[0] < val:
            updated_list1 = list1[:]
            updated_list1.pop(0)
            output = f'''```
if list1 and list1[0] < {val}:
    list1.pop(0)
```
list1 = {list1}
list1[0] = {list1[0]} < {val}
enter if
now,
list1 = {updated_list1}
'''
        elif list1:
            output = f'''```
if list1 and list1[0] < {val}:
    list1.pop(0)
```
list1 = {list1}
list1[0] = {list1[0]} >= {val}
do not enter if
'''
        else:
            output = f'''```
if list1 and list1[0] < {val}:
    list1.pop(0)
```
list1 = {list1}
do not enter if
'''
        return output

    def func_19(list1, list2, val):
        if list2:
            updated_list1 = list1[:]
            updated_list1.append(list2[0])
            updated_list2 = list2[1:]
            output = f'''```
if list2:
    list1.append(list2.pop(0))
```
list2 = {list2}
enter if
list1 = {list1}
list2.pop(0) = {list2[0]}
now,
list1 = {updated_list1}
list2 = {updated_list2}
'''
        else:
            output = f'''```
if list2:
    list1.append(list2.pop(0))
```
list2 = {list2}
do not enter if
'''
        return output

    def func_20(list1, list2, val):
        if list1:
            updated_list1 = list1[:]
            updated_list1.pop()
            output = f'''```
if list1:
    list1.pop()
```
list1 = {list1}
enter if
now,
list1 = {updated_list1}
'''
        else:
            output = f'''```
if list1:
    list1.pop()
```
list1 = {list1}
do not enter if
'''
        return output

    def func_21(list1, list2, val):
        if list2:
            updated_list2 = list2[:]
            updated_list2.pop()
            output = f'''```
if list2:
    list2.pop()
```
list2 = {list2}
enter if
now,
list2 = {updated_list2}
'''
        else:
            output = f'''```
if list2:
    list2.pop()
```
list2 = {list2}
do not enter if
'''
        return output

operations = {
    0: "if list1:\n    list1[-1] += {}",
    1: "if list1:\n    list1[0] += {}",
    2: "if list1:\n    var = list1.pop(0)\n    list2.append(var)",
    3: "if list1:\n    var = list1.pop()\n    list2.append(var)",
    4: "list1.insert(0, {})",
    5: "list1.sort()",
    6: "list1.reverse()",
    7: "list1 = list1[1:] if list1 else list1",
    8: "if list2:\n    list1.insert(0, list2[0])\nelse:\n    list1.insert(0, {})",
    9: "val = list2[-1] if list2 else {}\nlist1.append(val)",
    10: "if list1 and list2 and list1[0] > list2[0]:\n    list1.pop(0)",
    11: "if list1 and list2 and list1[-1] < list2[-1]:\n    list1.pop()",
    12: "if list1:\n    list1.pop(0)",
    13: "if list1 and list2:\n    list1.append(list2.pop())",
    14: "if list1 and list1[0] % 2 == 0:\n    list1.pop(0)",
    15: "if list1 and list1[0] % 2 == 1:\n    list1.pop(0)",
    16: "if len(list1) > len(list2):\n    list2.insert(0, list1.pop())",
    17: "if list1 and list1[0] > {}:\n    list1.pop(0)",
    18: "if list1 and list1[0] < {}:\n    list1.pop(0)",
    19: "if list2:\n    list1.append(list2.pop(0))",
    20: "if list1:\n    list1.pop()",
    21: "if list2:\n    list2.pop()"
}

data_dict = {}
for i in range(1, 31):
    data_dict[i] = []
    with open(f"../datasets_lc_bb/synthetic/l={i}.json", "r") as f:
        data_dict[i] = json.load(f)
invalid = 0

data_size = 300000
# list_length = 3
sample_range = (2, 10)
for d in tqdm(range(data_size)):
    while True:
        name1 = ''.join(random.choices(string.ascii_lowercase + '_', k=random.randint(3, 7)))
        name2 = ''.join(random.choices(string.ascii_lowercase + '_', k=random.randint(3, 7)))
        
        if name1 != name2 and name1.count('_') <= 1 and name2.count('_') <= 1:
            mapping = {
                'list1': name1,
                'list2': name2,
            }
            break
    snippet = random.sample(list(operations.keys()), random.randint(sample_range[0], sample_range[1]))
    if 20 not in snippet:
        snippet.append(20)
    # if 21 not in snippet:
    #     snippet.append(21)
    res = []
    list1 = [random.randint(0, 100) for _ in range(random.randint(30, 55))]
    list2 = [random.randint(0, 100) for _ in range(random.randint(30, 55))]
    
    for id in snippet:
        reverse = False
        s = operations[id]
        if random.random() < 0.5:
            s = s.replace('list1', '_tmp_')
            s = s.replace('list2', 'list1')
            s = s.replace('_tmp_', 'list2')
            reverse = True
        if '{}' in s:
            val = random.randint(0, 100)
            res.append((s.format(val), val, id, reverse)) # s, val, id, reverse
        else:
            res.append((s, None, id, reverse))
    
    random.shuffle(res)
    
    code = ['while list1 and list2:'] + [x[0] for x in res]
    code = '\n'.join(code)
    code = code.replace('\n', '\n    ') + '\nreturn list1'
    code = 'def process_list(list1, list2):\n' + code
    code = code.replace('\n', '\n    ')
    input = 'Follow the given rule to solve the question.\nrule:\n\n' + code + '\n\nQ: Given two lists, list1 = {} and list2 = {}, what is the final value of list1?'.format(list1, list2)
    # snippet = "\n".join(res)

    init = "{} = {}\n{} = {}\n".format("list1", list1, "list2", list2)
    exec(init)
    output = '1 Initialize\n' + init + '\n2 Main loop\n'
    # code = snippet
    length = 0
    flag_continue = False
    while list1 and list2:
        output += '''```
while list1 and list2:
```
list1 = {}
list2 = {}
enter the loop
2.1 One iteration:
'''.format(list1, list2)
        
        for line in res:
            if line[3]:
                tmp = eval(f"RFFT_funcs.func_{line[2]}(list2, list1, {line[1]})")
                tmp = tmp.replace('list1', '_tmp_')
                tmp = tmp.replace('list2', 'list1')
                tmp = tmp.replace('_tmp_', 'list2')
            else:
                tmp = eval(f"RFFT_funcs.func_{line[2]}(list1, list2, {line[1]})")
            output += tmp
            exec(line[0])
        length += 1
        if length > 31:
            flag_continue = True
            invalid += 1
            break
    if flag_continue:
        continue
    output += '''```
while list1 and list2:
```
list1 = {}
list2 = {}
do not enter
'''.format(list1, list2)

    output += '''```
return list1
```
So the answer is {}
'''.format(list1)
    output = output.replace('list1', mapping['list1'])
    output = output.replace('list2', mapping['list2'])
    input = input.replace('list1', mapping['list1'])
    input = input.replace('list2', mapping['list2'])
    num = len(data_dict.get(length, []))
    if num > 10000:
        continue
    sample = {}
    sample['input'] = input
    sample['output'] = output
    sample['length'] = length
    sample['answer'] = str(list1)
    sample['idx'] = num
    data_dict[length] = data_dict.get(length, []) + [sample]
    # print(input)
    # print(output)
    # print(length)

res = []
for k, v in data_dict.items():
    res.append((k, len(v)))

res = sorted(res, key=lambda x: x[0])
for k, v in res:
    print(k, v)

print(invalid)

for k, v in data_dict.items():
    with open(f"../datasets_lc_bb/synthetic/l={k}.json", "w") as f:
        json.dump(v, f)