One way is to replace the lambda functions with actual functions.
Seeing what they do in the compiler.py, we can replicate those functions in order to remove the lambda and print the flag.
Quick Method
Find the portion of the code which represents the flag.
Replace "Give it another try!" with the code so it prints out the flag.
win
Can be written in 4 lines of Python code.
Analysis
We can basically see that chall.py contains our obfuscated code. Running python3 chall.py, we are prompted for the flag.
We are also given the compiler that presumably creates the code in chall.py.
compiler.py
#!/usr/bin/env python3# Created by Jong Han, thank me laterimport random, stringfrom secret import flagdefeval_obf(x:int)->str:global c5; global c6 add_funcs={'+1':c5,'+2':c6} starters=[1,0] counter = random.choice(starters) result =str(counter)while counter < x: chosen_func = random.choice(list(add_funcs.values()))if chosen_func == add_funcs['+1']: counter +=1else: counter +=2if counter > x: tmp_list =list(add_funcs.values()) tmp_list.remove(chosen_func) chosen_func = tmp_list[0] result = chosen_func +'('+ result +')'return resultdefstr_to_chrs(x:str,func_of_chr:str,obf_method:callable)->str:return'+'.join(map(lambdaa:f'{func_of_chr}('+obf_method(ord(a))+')',x))definitialize(code:str,flag:str)->str:global c1; global c2; global c3; global c4; global c5; global c6; global c7; global c8; global c9 c1,c2,c3,c4,c5,c6,c7,c8,c9 = random.sample(string.ascii_lowercase,9) l ='lambda' char =str_to_chrs('chr','chr',str) ladd1 =str_to_chrs('lambda a:a+1',c2,str) ladd2 =str_to_chrs('lambda b:b+2',c2,str) code =str_to_chrs(code,c4,eval_obf) p =str_to_chrs('print',c4,str) ifstr =str_to_chrs('if',c4,str) flag_obf =str_to_chrs(flag,c4,eval_obf) result =f'({l}{c1}:({l}{c1},{c2}:({l}{c3},{c4},{c5},{c6}:(lambda {c7},{c8},{c9}: {c9}("You R l33t Python RE master.") if {c7}=={c8} else {c9}("Give it another try!"))({c3}({code}),{flag_obf},{c3}({p})))({c1},{c2},({c1}({ladd1})),({c1}({ladd2}))))({c1},{c1}({char})))(eval)'return resultcode ='input("Enter flag: ")'withopen('chall.py','w')as f: f.write(initialize(code,flag))
We can see in compiler.py that the flag and the input(...) code are given as arguments into the initalize function.
In the initialize function, we see that the result variable is the important one to focus on as it returns the code in chall.py. I have rearranged result so that it can be understood more easily.
({l}{c1}:({l}{c1},{c2}:({l}{c3},{c4},{c5},{c6}:(lambda {c7},{c8},{c9}: {c9}("You R l33t Python RE master.") if{c7}=={c8}else{c9}("Give it another try!"))({c3}({code}),{flag_obf},{c3}({p})))({c1},{c2},({c1}({ladd1})),({c1}({ladd2}))))({c1},{c1}({char})))(eval)
Solving Through The Long Method
Analyzing result, we see that the eval function will always be passed into c1.
So {c1}({char}) means it just evaluates char into the chr function. This is then passed into c2 in the next lambda function.
We then see that eval(ladd1) and eval(ladd2) is made, where ladd1 and ladd2 are just functions to add 1 and add 2 to the given number respectively.
So in the next lambda function, we can see c3=eval(), c4=chr(), c5=+1func, c6=+2func. We can verify that we have identified c5 and c6 correctly as it is provided in the eval_obf function as +1 and +2 respectively.
In the next lambda function, we can see that c7=eval(input("Enter flag: ")), c8=flag_obf, and c9=eval("print") (which is basically the print function))
We go back to looking at the flag_obf variable. It is provided as an argument into the last lambda function. We can filter and take in only the portion of the obfuscated code where ({c3}({code}),{flag_obf},{c3}({p}))), create our own functions for c3,c4,c5,c6, and evaluate print(flag_obf)