Skip to content

Commit a3e0983

Browse files
wip: partial dict import bootstrap
1 parent f62afad commit a3e0983

File tree

1 file changed

+98
-0
lines changed

1 file changed

+98
-0
lines changed

execnet/_dictimport.py

+98
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,98 @@
1+
# incomplete, to be completed later
2+
# this will replace concat based bootstrap
3+
from __future__ import annotations
4+
5+
import base64
6+
import json
7+
import sys
8+
import types
9+
import zlib
10+
from importlib import import_module
11+
from importlib.abc import Loader
12+
from importlib.metadata import Distribution
13+
from importlib.metadata import DistributionFinder
14+
from typing import Any
15+
from typing import IO
16+
from typing import Iterable
17+
from typing import Sequence
18+
19+
20+
class DictImporter(DistributionFinder, Loader):
21+
def __init__(self, sources: dict[str, str]):
22+
self.sources = sources
23+
24+
def find_distributions(
25+
self, context: DistributionFinder.Context = DistributionFinder.Context()
26+
) -> Iterable[Distribution]:
27+
return []
28+
29+
def find_module(
30+
self, fullname: str, path: Sequence[str | bytes] | None = None
31+
) -> Loader | None:
32+
if fullname in self.sources:
33+
return self
34+
if fullname + ".__init__" in self.sources:
35+
return self
36+
return None
37+
38+
def load_module(self, fullname):
39+
# print "load_module:", fullname
40+
from types import ModuleType
41+
42+
try:
43+
s = self.sources[fullname]
44+
is_pkg = False
45+
except KeyError:
46+
s = self.sources[fullname + ".__init__"]
47+
is_pkg = True
48+
49+
co = compile(s, fullname, "exec")
50+
module = sys.modules.setdefault(fullname, ModuleType(fullname))
51+
module.__loader__ = self
52+
if is_pkg:
53+
module.__path__ = [fullname]
54+
55+
exec(co, module.__dict__)
56+
return sys.modules[fullname]
57+
58+
def get_source(self, name: str) -> str | None:
59+
res = self.sources.get(name)
60+
if res is None:
61+
res = self.sources.get(name + ".__init__")
62+
return res
63+
64+
65+
def bootstrap(
66+
modules: dict[str, str],
67+
entry: str,
68+
args: dict[str, Any],
69+
set_argv: list[str] | None,
70+
) -> None:
71+
importer = DictImporter(modules)
72+
sys.meta_path.append(importer)
73+
module, attr = entry.split(":")
74+
loaded_module = import_module(module)
75+
entry_func = getattr(loaded_module, entry)
76+
if set_argv is not None:
77+
sys.argv[1:] = set_argv
78+
entry_func(**args)
79+
80+
81+
def bootstrap_stdin(stream: IO) -> None:
82+
bootstrap_args = decode_b85_zip_json(stream.readline())
83+
bootstrap(**bootstrap_args)
84+
85+
86+
def decode_b85_zip_json(encoded: bytes | str):
87+
packed = base64.b85decode(encoded)
88+
unpacked = zlib.decompress(packed)
89+
return json.loads(unpacked)
90+
91+
92+
def naive_pack_module(module: types.ModuleType, dist: Distribution):
93+
assert module.__file__ is not None
94+
assert module.__path__
95+
96+
97+
if __name__ == "__main__":
98+
bootstrap_stdin(sys.stdin)

0 commit comments

Comments
 (0)