rc.xsh 7.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240
  1. $VI_MODE = True
  2. $AUTO_PUSHD = True
  3. $XONSH_AUTOPAIR = True
  4. # tab selection: do not execute cmd when pressing enter
  5. $COMPLETIONS_CONFIRM = True
  6. $_Z_EXCLUDE_DIRS = ['/tmp']
  7. xontrib load vox z
  8. def _last_exit_status():
  9. try:
  10. exit_status = __xonsh_history__.rtns[-1]
  11. return exit_status if exit_status != 0 else None
  12. except IndexError:
  13. return None
  14. $PROMPT_FIELDS['last_exit_status'] = _last_exit_status
  15. $SHLVL = int($SHLVL) + 1 if 'SHLVL' in ${...} else 1
  16. $XONSH_STDERR_PREFIX = '{RED}'
  17. $XONSH_STDERR_POSTFIX = '{NO_COLOR}'
  18. $DYNAMIC_CWD_WIDTH = '30%'
  19. $DYNAMIC_CWD_ELISION_CHAR = '…'
  20. $PROMPT = ''.join([
  21. '{RED}{last_exit_status:[{}] }',
  22. '{BOLD_GREEN}{user}@{hostname} ',
  23. '{YELLOW}{cwd} ',
  24. '{{BLUE}}{} '.format('{prompt_end}' * $SHLVL),
  25. '{NO_COLOR}',
  26. ])
  27. $RIGHT_PROMPT = '{gitstatus}{env_name: {}}'
  28. $XONSH_APPEND_NEWLINE = True
  29. import datetime as dt
  30. import io
  31. import os
  32. import re
  33. import shutil
  34. import stat
  35. import subprocess
  36. import sys
  37. os.umask(stat.S_IWGRP | stat.S_IRWXO) # 027
  38. # default locale
  39. # will be used for all non-explicitly set LC_* variables
  40. $LANG = 'en_US.UTF-8'
  41. # fallback locales
  42. # GNU gettext gives preference to LANGUAGE over LC_ALL and LANG
  43. # for the purpose of message handling
  44. # https://www.gnu.org/software/gettext/manual/html_node/The-LANGUAGE-variable.html
  45. # cave: if this list contains 'de(_.*)?' at any (sic!) position
  46. # vim 7.4.1689 will switch to german
  47. $LANGUAGE = ':'.join(['en_US', 'en'])
  48. $LC_COLLATE = 'C.UTF-8'
  49. # char classification, case conversion & other char attrs
  50. $LC_CTYPE = 'de_AT.UTF-8'
  51. # $ locale currency_symbol
  52. $LC_MONETARY = 'de_AT.UTF-8'
  53. # $ locale -k LC_NUMERIC | head -n 3
  54. # decimal_point="."
  55. # thousands_sep=""
  56. # grouping=-1
  57. $LC_NUMERIC = 'C.UTF-8'
  58. # A4
  59. $LC_PAPER = 'de_AT.UTF-8'
  60. USER_BIN_PATH = os.path.join($HOME, '.local', 'bin')
  61. if os.path.isdir(USER_BIN_PATH):
  62. $PATH.insert(0, USER_BIN_PATH)
  63. $PAGER = 'less'
  64. $EDITOR = 'vim'
  65. # required by pinentry-tty when using gpg command:
  66. $GPG_TTY = $(tty)
  67. if shutil.which('gpgconf'):
  68. # required by scute
  69. $GPG_AGENT_INFO = $(gpgconf --list-dir agent-socket).rstrip() + ':0:1'
  70. if not 'SSH_CLIENT' in ${...}:
  71. # in gnupg 2.1.13 the location of agents socket changed
  72. $SSH_AUTH_SOCK = $(gpgconf --list-dir agent-ssh-socket).rstrip()
  73. # wrapper for termite required when launching termite from ranger:
  74. $TERMCMD = os.path.join(os.path.dirname(__file__), 'ranger-termite-termcmd')
  75. # https://docs.docker.com/engine/security/trust/content_trust/
  76. $DOCKER_CONTENT_TRUST = 1
  77. class DockerImage:
  78. def __init__(self, image):
  79. import json
  80. attrs, = json.loads(subprocess.check_output(['sudo', 'docker', 'image', 'inspect', image])
  81. .decode(sys.stdout.encoding))
  82. self._id = attrs['Id']
  83. self._tags = attrs['RepoTags']
  84. def __repr__(self):
  85. return '{}(id={!r}, tags={!r})'.format(type(self).__name__, self._id, self._tags)
  86. @classmethod
  87. def build(cls, dockerfile):
  88. out = io.BytesIO()
  89. with StdoutTee(out) as tee:
  90. p = subprocess.Popen(['sudo', 'docker', 'build', '-'],
  91. stdin=subprocess.PIPE, stdout=tee)
  92. p.stdin.write(dockerfile.encode())
  93. p.stdin.close()
  94. assert p.wait() == 0, 'docker build failed'
  95. image_id, = re.search(rb'^Successfully built (\S+)$', out.getvalue(), re.MULTILINE).groups()
  96. return cls(image_id.decode(sys.stdout.encoding))
  97. @classmethod
  98. def pull(cls, image):
  99. out = io.BytesIO()
  100. with StdoutTee(out) as tee:
  101. subprocess.run(['sudo', 'docker', 'image', 'pull', image], stdout=tee)
  102. repo_digest, = re.search(rb'^Digest: (sha\S+:\S+)$', out.getvalue(), re.MULTILINE).groups()
  103. return cls('{}@{}'.format(image, repo_digest.decode()))
  104. def run(self, args=[], name=None, detach=False, publish_ports=[], caps=[]):
  105. params = ['sudo', 'docker', 'run', '--rm']
  106. if name:
  107. params.extend(['--name', name])
  108. if detach:
  109. params.append('--detach')
  110. else:
  111. params.extend(['--interactive', '--tty'])
  112. params.extend('--publish=' + ':'.join([str(a) for a in p]) for p in publish_ports)
  113. params.extend(['--security-opt=no-new-privileges', '--cap-drop=all'])
  114. params.extend(['--cap-add={}'.format(c) for c in caps])
  115. params.append(self._id)
  116. params.extend(args)
  117. sys.stderr.write('{}\n'.format(shlex_join(params)))
  118. subprocess.run(params, check=True)
  119. class StdoutTee:
  120. def __init__(self, sink):
  121. self._sink = sink
  122. def __enter__(self):
  123. self._read_fd, self._write_fd = os.pipe()
  124. import threading
  125. self._thread = threading.Thread(target=self._loop)
  126. self._thread.start()
  127. return self
  128. def _loop(self):
  129. while True:
  130. try:
  131. data = os_read_non_blocking(self._read_fd)
  132. except OSError: # fd closed
  133. return
  134. if data:
  135. self._sink.write(data)
  136. sys.stdout.buffer.write(data)
  137. sys.stdout.flush()
  138. def fileno(self):
  139. return self._write_fd
  140. def __exit__(self, exc_type, exc_value, traceback):
  141. os.close(self._read_fd)
  142. os.close(self._write_fd)
  143. self._thread.join()
  144. def dpkg_listfiles(pkg_name):
  145. assert isinstance(pkg_name, str)
  146. paths = $(dpkg --listfiles @(pkg_name)).split('\n')[:-1]
  147. assert len(paths) > 0, 'pkg {!r} not installed'.format(pkg_name)
  148. return paths
  149. def dpkg_search(path_search_pattern):
  150. assert isinstance(path_search_pattern, str)
  151. return re.findall(
  152. '^(\S+): (.*)$\n',
  153. $(dpkg --search @(path_search_pattern)),
  154. flags=re.MULTILINE,
  155. )
  156. def dpkg_welse(cmd):
  157. pkg_name, cmd_path = dpkg_which(cmd)
  158. return dpkg_listfiles(pkg_name)
  159. def dpkg_which(cmd):
  160. cmd_path = shutil.which(cmd)
  161. assert cmd_path, 'cmd {!r} not found'.format(cmd)
  162. matches = dpkg_search(cmd_path)
  163. assert len(matches) != 0, '{!r} not installed via dpkg'.format(cmd_path)
  164. assert len(matches) == 1
  165. return matches[0]
  166. def locate(*patterns, match_all=True, ignore_case=True):
  167. params = []
  168. if match_all:
  169. params.insert(0, '--all')
  170. if ignore_case:
  171. params.insert(0, '--ignore-case')
  172. return $(locate @(params) -- @(patterns)).split('\n')[:-1]
  173. def os_read_non_blocking(fd, buffer_size_bytes=8*1024, timeout_seconds=0.1):
  174. import select
  175. if fd in select.select([fd], [], [], timeout_seconds)[0]:
  176. return os.read(fd, buffer_size_bytes)
  177. else:
  178. return None
  179. def shlex_join(params):
  180. import shlex
  181. return ' '.join(shlex.quote(p) for p in params)
  182. def timestamp_now_utc():
  183. return dt.datetime.utcnow().replace(tzinfo=dt.timezone.utc)
  184. def timestamp_now_local():
  185. # if called without tz argument astimezone() assumes
  186. # the system local timezone for the target timezone
  187. return timestamp_now_utc().astimezone()
  188. def timestamp_iso_local():
  189. # if called without tz argument astimezone() assumes
  190. # the system local timezone for the target timezone
  191. return timestamp_now_local().strftime('%Y%m%dT%H%M%S%z')
  192. def yaml_load(path):
  193. import yaml
  194. with open(path, 'r') as f:
  195. return yaml.load(f.read())
  196. def yaml_write(path, data):
  197. import yaml
  198. with open(path, 'w') as f:
  199. f.write(yaml.dump(data, default_flow_style=False))
  200. aliases['d'] = ['sudo', 'docker']
  201. aliases['dpkg-welse'] = lambda args: '\n'.join(dpkg_welse(args[0]))
  202. aliases['dpkg-which'] = lambda args: '\t'.join(dpkg_which(args[0]))
  203. aliases['g'] = ['git']
  204. aliases['ll'] = ['ls', '-l', '--all', '--indicator-style=slash',
  205. '--human-readable', '--time-style=long-iso', '--color=auto']
  206. if shutil.which('startx') and $(tty).rstrip() == '/dev/tty1':
  207. startx
  208. # vim: filetype=python