|
| 1 | +from multiprocessing import Event, Pipe, Process, Queue |
| 2 | +from os import chroot, getgid, getlogin, getuid, setgid, setuid |
| 3 | + |
| 4 | +from .namespace import get_id_map, new_id_map, unshare_namespace |
| 5 | + |
| 6 | + |
| 7 | +class NamespaceProcess(Process): |
| 8 | + """Like process, but runs in a new namespace. |
| 9 | + Puts the target return value in a queue, and any exceptions in a pipe. |
| 10 | + """ |
| 11 | + |
| 12 | + def __init__(self, target=None, args=None, kwargs=None, **ekwargs): |
| 13 | + self.target_root = kwargs.pop("target_root", "/") |
| 14 | + namespace_user = kwargs.pop("namespace_user", getlogin()) |
| 15 | + self.subuid_start, self.subuid_count = get_id_map(namespace_user, "uid") |
| 16 | + self.subgid_start, self.subgid_count = get_id_map(namespace_user, "gid") |
| 17 | + self.orig_uid = getuid() |
| 18 | + self.orig_gid = getgid() |
| 19 | + self.uidmapped = Event() |
| 20 | + self.completed = Event() |
| 21 | + self.exception_recv, self.exception_send = Pipe() |
| 22 | + self.function_queue = Queue() |
| 23 | + super().__init__(target=target, args=args, kwargs=kwargs, **ekwargs) |
| 24 | + |
| 25 | + def map_ids(self): |
| 26 | + new_id_map("uid", self.pid, 0, self.orig_uid, 1, 1, self.subuid_start, self.subuid_count) |
| 27 | + new_id_map("gid", self.pid, 0, self.orig_gid, 1, 1, self.subgid_start, self.subgid_count) |
| 28 | + |
| 29 | + def map_unshare_uids(self): |
| 30 | + self.start() |
| 31 | + self.map_ids() |
| 32 | + self.uidmapped.set() |
| 33 | + |
| 34 | + def run(self): |
| 35 | + unshare_namespace() |
| 36 | + self.uidmapped.wait() |
| 37 | + setuid(0) |
| 38 | + setgid(0) |
| 39 | + chroot(self.target_root) |
| 40 | + try: |
| 41 | + self.function_queue.put(self._target(*self._args, **self._kwargs)) |
| 42 | + except Exception as e: |
| 43 | + self.exception_send.send(e) |
| 44 | + self.completed.set() |
0 commit comments