-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathtranslate_ast.py
120 lines (96 loc) · 3.73 KB
/
translate_ast.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
import ast
# Get the code
code = ""
# Get the built-ins functions
code += open("default_values.py").read() + "\n" * 5
code += open("functions.py").read() + "\n" * 5
code += open("prog.py").read() + "\n" * 5
# Remove import
code = code.replace("from classes import *","")
# Define a transformer class to replace all built-ins classes by ExtendedClass
class IntStrTransformer(ast.NodeTransformer):
# When it visits a number
def visit_Num(self, node):
return ast.Call(func=ast.Name(id="ExtendedInt", ctx=ast.Load()),
args=[node],
keywords=[],
)
# When it visits a string
def visit_Str(self, node):
return ast.Call(func=ast.Name(id="ExtendedString", ctx=ast.Load()),
args=[node],
keywords=[],
)
# When it visits a formated value
def visit_FormattedValue(self, node):
return ast.Call(func=ast.Name(id="ExtendedString", ctx=ast.Load()),
args=[node],
keywords=[],
)
# When it visits a joinedstr
def visit_JoinedStr(self, node):
elts = []
for value in node.values:
if isinstance(value, ast.Str):
elts.append(self.visit_Str(value))
elif isinstance(value, ast.FormattedValue):
elts.append(self.visit_FormattedValue(value))
elif isinstance(value, ast.Call):
elts.append(self.visit_Call(value))
else:
raise ValueError(f"Unexpected node inside JoinedStr, {value!r}")
return ast.Call(func=ast.Name(id="ExtendedString", ctx=ast.Load()),
args=[ast.List(elts=elts, ctx=ast.Load())],
keywords=[],
)
def visit_Call(self, node):
args = [self.visit(arg) for arg in node.args]
keywords = [self.visit(keyword) for keyword in node.keywords]
return ast.Call(func=self.visit(node.func),
args=args,
keywords=keywords,
)
# When it visits a list
def visit_List(self, node):
return ast.Call(func=ast.Name(id="ExtendedList", ctx=ast.Load()),
args=[node],
keywords=[],
)
# When it visits a range => list
def visit_Range(self, node):
args = [node.start, node.end]
if node.step is not None:
args.append(node.step)
return ast.Call(func=ast.Name(id="ExtendedList", ctx=ast.Load()),
args=[node],
keywords=[],
)
# Parse the code into an AST
# tree = ast.literal_eval(code)
tree = ast.parse(code)
# Use the transformer to modify the AST
transformer = IntStrTransformer()
new_tree = transformer.visit(tree)
# Write the new code to the prog.py
with open("tmp/prog.py", "wt") as file:
intial_code = open("prog.py").read()
default_code = f'''#!/https://github.com/aderepas/CustomCodinGameClient
"""
=========================================
/!\ THIS CODE IS NOT WRITTEN BY A BOT /!\\
=========================================
Some of you might think that everything
here has been generated by an AI, or a
bot but actually, if the code works, it's
because of a Human. Below is the code
written by the Human and the GitHub repo
that explains what is this.
"""
""" Initial code: {len(intial_code)} bytes
{intial_code}
"""
'''
unparsed_code = ast.unparse(new_tree)
file.write(default_code)
file.write(open("classes.py").read().replace("from functions import *",""))
file.write(unparsed_code)