utils.py 5.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164
  1. #!/usr/bin/python2 -O
  2. # vim: fileencoding=utf-8
  3. #
  4. # The Qubes OS Project, https://www.qubes-os.org/
  5. #
  6. # Copyright (C) 2010-2015 Joanna Rutkowska <joanna@invisiblethingslab.com>
  7. # Copyright (C) 2013-2015 Marek Marczykowski-Górecki
  8. # <marmarek@invisiblethingslab.com>
  9. # Copyright (C) 2014-2015 Wojtek Porczyk <woju@invisiblethingslab.com>
  10. #
  11. # This program is free software; you can redistribute it and/or modify
  12. # it under the terms of the GNU General Public License as published by
  13. # the Free Software Foundation; either version 2 of the License, or
  14. # (at your option) any later version.
  15. #
  16. # This program is distributed in the hope that it will be useful,
  17. # but WITHOUT ANY WARRANTY; without even the implied warranty of
  18. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  19. # GNU General Public License for more details.
  20. #
  21. # You should have received a copy of the GNU General Public License along
  22. # with this program; if not, write to the Free Software Foundation, Inc.,
  23. # 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
  24. #
  25. import hashlib
  26. import os
  27. import re
  28. import subprocess
  29. import docutils
  30. import docutils.core
  31. import docutils.io
  32. import pkg_resources
  33. import qubes.exc
  34. def get_timezone():
  35. # fc18
  36. if os.path.islink('/etc/localtime'):
  37. return '/'.join(os.readlink('/etc/localtime').split('/')[-2:])
  38. # <=fc17
  39. elif os.path.exists('/etc/sysconfig/clock'):
  40. clock_config = open('/etc/sysconfig/clock', "r")
  41. clock_config_lines = clock_config.readlines()
  42. clock_config.close()
  43. zone_re = re.compile(r'^ZONE="(.*)"')
  44. for line in clock_config_lines:
  45. line_match = zone_re.match(line)
  46. if line_match:
  47. return line_match.group(1)
  48. else:
  49. # last resort way, some applications makes /etc/localtime
  50. # hardlink instead of symlink...
  51. tz_info = os.stat('/etc/localtime')
  52. if not tz_info:
  53. return None
  54. if tz_info.st_nlink > 1:
  55. p = subprocess.Popen(['find', '/usr/share/zoneinfo',
  56. '-inum', str(tz_info.st_ino), '-print', '-quit'],
  57. stdout=subprocess.PIPE)
  58. tz_path = p.communicate()[0].strip()
  59. return tz_path.replace('/usr/share/zoneinfo/', '')
  60. return None
  61. def format_doc(docstring):
  62. '''Return parsed documentation string, stripping RST markup.
  63. '''
  64. if not docstring:
  65. return ''
  66. # pylint: disable=unused-variable
  67. output, pub = docutils.core.publish_programmatically(
  68. source_class=docutils.io.StringInput,
  69. source=' '.join(docstring.strip().split()),
  70. source_path=None,
  71. destination_class=docutils.io.NullOutput, destination=None,
  72. destination_path=None,
  73. reader=None, reader_name='standalone',
  74. parser=None, parser_name='restructuredtext',
  75. writer=None, writer_name='null',
  76. settings=None, settings_spec=None, settings_overrides=None,
  77. config_section=None, enable_exit_status=None)
  78. return pub.writer.document.astext()
  79. # FIXME those are wrong, k/M/G are SI prefixes and means 10**3
  80. # maybe adapt https://code.activestate.com/recipes/578019
  81. def parse_size(size):
  82. units = [
  83. ('K', 1000), ('KB', 1000),
  84. ('M', 1000 * 1000), ('MB', 1000 * 1000),
  85. ('G', 1000 * 1000 * 1000), ('GB', 1000 * 1000 * 1000),
  86. ('Ki', 1024), ('KiB', 1024),
  87. ('Mi', 1024 * 1024), ('MiB', 1024 * 1024),
  88. ('Gi', 1024 * 1024 * 1024), ('GiB', 1024 * 1024 * 1024),
  89. ]
  90. size = size.strip().upper()
  91. if size.isdigit():
  92. return int(size)
  93. for unit, multiplier in units:
  94. if size.endswith(unit):
  95. size = size[:-len(unit)].strip()
  96. return int(size) * multiplier
  97. raise qubes.exc.QubesException("Invalid size: {0}.".format(size))
  98. def mbytes_to_kmg(size):
  99. if size > 1024:
  100. return "%d GiB" % (size / 1024)
  101. else:
  102. return "%d MiB" % size
  103. def kbytes_to_kmg(size):
  104. if size > 1024:
  105. return mbytes_to_kmg(size / 1024)
  106. else:
  107. return "%d KiB" % size
  108. def bytes_to_kmg(size):
  109. if size > 1024:
  110. return kbytes_to_kmg(size / 1024)
  111. else:
  112. return "%d B" % size
  113. def size_to_human(size):
  114. """Humane readable size, with 1/10 precision"""
  115. if size < 1024:
  116. return str(size)
  117. elif size < 1024 * 1024:
  118. return str(round(size / 1024.0, 1)) + ' KiB'
  119. elif size < 1024 * 1024 * 1024:
  120. return str(round(size / (1024.0 * 1024), 1)) + ' MiB'
  121. else:
  122. return str(round(size / (1024.0 * 1024 * 1024), 1)) + ' GiB'
  123. def urandom(size):
  124. rand = os.urandom(size)
  125. if rand is None:
  126. raise IOError('failed to read urandom')
  127. return hashlib.sha512(rand).digest()
  128. def get_entry_point_one(group, name):
  129. epoints = tuple(pkg_resources.iter_entry_points(group, name))
  130. if not epoints:
  131. raise KeyError(name)
  132. elif len(epoints) > 1:
  133. raise TypeError(
  134. 'more than 1 implementation of {!r} found: {}'.format(name,
  135. ', '.join('{}.{}'.format(ep.module_name, '.'.join(ep.attrs))
  136. for ep in epoints)))
  137. return epoints[0].load()