|
1 | | -from pathlib import Path |
2 | | -from typing import Union, Tuple |
3 | | -from types import CodeType |
4 | | -from pie import LoaderForBetterLife |
5 | | -from importlib import import_module |
6 | | -from purescripto import rts |
7 | | -from purescripto.utilities import import_from_path |
8 | | -from purescripto.workaround import suppress_cpy38_literal_is, workaround |
9 | | -import marshal |
10 | | -import functools |
11 | | - |
12 | | -RES = 'res' |
13 | | -"""generated code object is stored in global variable $RES""" |
14 | | - |
15 | | - |
16 | | -@functools.lru_cache() |
17 | | -def _import_module_to_dict(m: str): |
18 | | - with suppress_cpy38_literal_is(): |
19 | | - package, _, module = m.rpartition('.') |
20 | | - entry_path = import_module(package).__file__ |
21 | | - loc = entry_path[:-len('__init__.py')] + module + '.py' |
22 | | - return import_from_path(m, loc).__dict__ |
23 | | - |
24 | | - |
25 | | -RTS_TEMPLATE = { |
26 | | - 'zfsr32': rts.zfsr32, |
27 | | - 'Error': Exception, |
28 | | - 'import_module': _import_module_to_dict |
29 | | -} |
30 | | - |
31 | | - |
32 | | -class LoadPureScriptImplCode(LoaderForBetterLife[CodeType]): |
33 | | - def source_to_prog(self, src: bytes, path: Path) -> CodeType: |
34 | | - with workaround(): |
35 | | - mod = import_from_path(self.qualified_name + '$', |
36 | | - str(path.absolute())) |
37 | | - return getattr(mod, RES) |
38 | | - |
39 | | - def load_program(self, b: bytes) -> CodeType: |
40 | | - return marshal.loads(b) |
41 | | - |
42 | | - def dump_program(self, prog: CodeType) -> bytes: |
43 | | - return marshal.dumps(prog) |
44 | | - |
45 | | - def suffix(self) -> Union[str, Tuple[str, ...]]: |
46 | | - return '.src.py' |
47 | | - |
48 | | - |
49 | | -def LoadPureScript(file: str, name: str): |
50 | | - loader = LoadPureScriptImplCode(file, name) |
51 | | - code = loader.load() |
52 | | - man_made_globals = RTS_TEMPLATE.copy() |
53 | | - exec(code, man_made_globals) |
54 | | - return man_made_globals['exports'] |
| 1 | +from pathlib import Path |
| 2 | +from typing import Union, Tuple |
| 3 | +from types import CodeType |
| 4 | +from pie import LoaderForBetterLife |
| 5 | +from importlib import import_module |
| 6 | +from purescripto import rts |
| 7 | +from purescripto.utilities import import_from_path |
| 8 | +from purescripto.workaround import suppress_cpy38_literal_is, workaround |
| 9 | +from py_sexpr.stack_vm.emit import module_code |
| 10 | +from purescripto.topdown import load_topdown |
| 11 | +import marshal |
| 12 | +import functools |
| 13 | + |
| 14 | +RES = "res" |
| 15 | +"""generated code object is stored in global variable $RES""" |
| 16 | + |
| 17 | + |
| 18 | +@functools.lru_cache() |
| 19 | +def _import_module_to_dict(m: str): |
| 20 | + with suppress_cpy38_literal_is(): |
| 21 | + package, _, module = m.rpartition(".") |
| 22 | + entry_path = import_module(package).__file__ |
| 23 | + loc = entry_path[: -len("__init__.py")] + module + ".py" |
| 24 | + return import_from_path(m, loc).__dict__ |
| 25 | + |
| 26 | + |
| 27 | +RTS_TEMPLATE = { |
| 28 | + "zfsr32": rts.zfsr32, |
| 29 | + "Error": Exception, |
| 30 | + "import_module": _import_module_to_dict, |
| 31 | +} |
| 32 | + |
| 33 | + |
| 34 | +def _META_ENV(): |
| 35 | + with workaround(): |
| 36 | + import py_sexpr.terms as terms |
| 37 | + |
| 38 | + def make_pair(a, b): |
| 39 | + return a, b |
| 40 | + |
| 41 | + env = {each: getattr(terms, each) for each in terms.__all__} |
| 42 | + env[make_pair.__name__] = make_pair |
| 43 | + return env |
| 44 | + |
| 45 | + |
| 46 | +META_ENV = _META_ENV() |
| 47 | + |
| 48 | + |
| 49 | +class LoadPureScriptImplCode(LoaderForBetterLife[CodeType]): |
| 50 | + def source_to_prog(self, src: bytes, path: Path) -> CodeType: |
| 51 | + |
| 52 | + filename = str(path.absolute()) |
| 53 | + if path.name.endswith(".raw.py"): |
| 54 | + meta_code = compile(src, filename, "eval") |
| 55 | + sexpr = eval(meta_code, META_ENV) |
| 56 | + code = module_code(sexpr, name=self.qualified_name + "$", filename=filename) |
| 57 | + else: |
| 58 | + assert path.name.endswith(".zip.py") |
| 59 | + with path.open(mode="utf8") as f: |
| 60 | + code = load_topdown(f, META_ENV) |
| 61 | + |
| 62 | + return code |
| 63 | + |
| 64 | + def load_program(self, b: bytes) -> CodeType: |
| 65 | + return marshal.loads(b) |
| 66 | + |
| 67 | + def dump_program(self, prog: CodeType) -> bytes: |
| 68 | + return marshal.dumps(prog) |
| 69 | + |
| 70 | + def suffix(self) -> Union[str, Tuple[str, ...]]: |
| 71 | + return ".raw.py", ".zip.py" |
| 72 | + |
| 73 | + |
| 74 | +def LoadPureScript(file: str, name: str): |
| 75 | + loader = LoadPureScriptImplCode(file, name) |
| 76 | + code = loader.load() |
| 77 | + man_made_globals = RTS_TEMPLATE.copy() |
| 78 | + exec(code, man_made_globals) |
| 79 | + return man_made_globals["exports"] |
0 commit comments