-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathcore.py
151 lines (112 loc) · 4.33 KB
/
core.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
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
"""
core functions of the string_to_code module
"""
import typing
import random
import functools
Strings: typing.TypeAlias = list[str]
class Atom(typing.NamedTuple):
"""represents a single character to be printed"""
atom_char: str
CalledListEntry: typing.TypeAlias = int | Atom
InitialCall: typing.TypeAlias = CalledListEntry | None
class SimpleFunction(typing.NamedTuple):
"""
represents a function with no arguments and no return value calling
other functions of such type or displaying single characters
"""
called_list: list[CalledListEntry]
SimpleFunctions: typing.TypeAlias = list[SimpleFunction]
def str_pieces(in_str: str, in_pieces_len: list[int]) -> Strings:
"""returns in_str split into pieces of lengths as in in_pieces_len"""
assert all(_ > 0 for _ in in_pieces_len) # nosec B101
assert sum(in_pieces_len) == len(in_str) # nosec B101
cur_str = in_str
res = []
for _ in in_pieces_len:
res.append(cur_str[:_])
cur_str = cur_str[_:]
assert "".join(res) == in_str # nosec B101
return res
def random_pieces_len(in_total_len: int) -> list[int]:
"""returns a list of positive numbers with their sum being in_total_len"""
cur_num = in_total_len
res = []
while cur_num > 0:
tmp_len = random.randint(1, max(cur_num, 1)) # nosec B311
res.append(tmp_len)
cur_num -= tmp_len
random.shuffle(res)
assert sum(res) == in_total_len # nosec B101
return res
def random_split(in_str: str) -> Strings:
"""randomply splits in_str"""
return str_pieces(in_str, random_pieces_len(len(in_str)))
def _interesting_random_split(in_str: str) -> Strings:
assert len(in_str) > 1 # nosec B101
res = random_split(in_str)
while len(res) == 1:
res = random_split(in_str)
return res
def get_function_namer(
in_name_prefix: str = "f_",
) -> typing.Callable[[int], str]:
"""returns a default function namer"""
def _inner(in_id: int) -> str:
return in_name_prefix + str(in_id)
return _inner
def _prepare_printer_program(
in_str: str,
) -> typing.Tuple[InitialCall, SimpleFunctions]:
needed_functions: SimpleFunctions = []
@functools.lru_cache(maxsize=None)
def _generate_code(in_str: str) -> CalledListEntry:
if len(in_str) == 1:
return Atom(in_str)
str_split = _interesting_random_split(in_str)
res = SimpleFunction([_generate_code(_) for _ in str_split])
needed_functions.append(res)
return len(needed_functions) - 1
initial_call = _generate_code(in_str) if in_str else None
return initial_call, needed_functions
class PrinterProgram:
"""
Represents a program printing a given string.
It consists only of SimpleFunctions and Atoms.
"""
def __init__(self, initial_call: InitialCall, needed_functions: SimpleFunctions):
self._initial_call = initial_call
self._needed_functions = needed_functions
self._check_data()
def _check_data(self) -> None:
if self.initial_call is not None and not isinstance(self.initial_call, Atom):
assert self.needed_functions # nosec B101
if self.needed_functions:
assert isinstance(self.initial_call, int) # nosec B101
assert self.initial_call + 1 == len(self.needed_functions) # nosec B101
for fun_id, fun in enumerate(self.needed_functions):
assert all(
_ < fun_id for _ in fun.called_list if not isinstance(_, Atom)
) # nosec B101
def needed_function_definitions_str_list(
self, in_function_to_str, **kwargs
) -> Strings:
"""
returns a list of string representations of the definition of
the needed functions
"""
return [
in_function_to_str(id, fun, **kwargs)
for id, fun in enumerate(self.needed_functions)
]
@property
def initial_call(self) -> InitialCall:
"""returns the 'entry point' of the program"""
return self._initial_call
@property
def needed_functions(self) -> SimpleFunctions:
"""returns the list of all needed functions"""
return self._needed_functions
def get_printer_program(in_str: str) -> PrinterProgram:
"""returns a PrinterProgram object displaying in_str"""
return PrinterProgram(*_prepare_printer_program(in_str))