import subprocess from .zfs import * import random import shutil import json def get_jid(path): data = json.loads(subprocess.check_output(['jls', '--libxo=json'])) lst = data['jail-information']['jail'] lst = list(filter(lambda a: a['path'] == path, lst)) if len(lst) == 0: raise ValueError('JID not found for path: ' + path) if len(lst) > 1: raise ValueError('Ambiguous JID for path: ' + path) return str(lst[0]['jid']) def jail_run(path, command): command = ['jail', '-c', 'host.hostname=' + os.path.split(path)[1], 'persist=1', 'mount.devfs=1', 'interface=lo1', 'ip4.addr=127.0.1.0', 'path=' + path, 'command', '/bin/sh', '-c', command] print('Running:', ' '.join(command)) try: shutil.copyfile('/etc/resolv.conf', os.path.join(path, 'etc/resolv.conf')) res = subprocess.run(command) finally: try: subprocess.run(['jail', '-r', get_jid(path)]) except ValueError: pass subprocess.run(['umount', '-f', os.path.join(path, 'dev')]) if res.returncode != 0: # subprocess.run(['umount', os.path.join(path, 'dev')]) raise RuntimeError('Command failed') def command_jail_run(args): base, _ = zfs_snapshot_by_tag_or_sha256(args.image) # root = '/'.join(base.split('/')[:-1]) for _ in range(10**6): name = bytes([ random.randint(0, 255) for _ in range(4) ]).hex()[:7] name = base.split('/')[0] + '/focker/jails/' + name if not zfs_exists(name): break zfs_run(['zfs', 'clone', base, name]) try: jail_run(zfs_mountpoint(name), args.command) # subprocess.check_output(['jail', '-c', 'interface=lo1', 'ip4.addr=127.0.1.0', 'path=' + zfs_mountpoint(name), 'command', command]) finally: # subprocess.run(['umount', zfs_mountpoint(name) + '/dev']) zfs_run(['zfs', 'destroy', '-f', name]) # raise