1# Authors: Karl MacMillan <kmacmillan@mentalrootkit.com> 2# 3# Copyright (C) 2006 Red Hat 4# see file 'COPYING' for use and warranty information 5# 6# This program is free software; you can redistribute it and/or 7# modify it under the terms of the GNU General Public License as 8# published by the Free Software Foundation; version 2 only 9# 10# This program is distributed in the hope that it will be useful, 11# but WITHOUT ANY WARRANTY; without even the implied warranty of 12# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 13# GNU General Public License for more details. 14# 15# You should have received a copy of the GNU General Public License 16# along with this program; if not, write to the Free Software 17# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA 18# 19import locale 20import sys 21 22 23PY3 = sys.version_info[0] == 3 24 25if PY3: 26 bytes_type=bytes 27 string_type=str 28else: 29 bytes_type=str 30 string_type=unicode 31 32 33class ConsoleProgressBar: 34 def __init__(self, out, steps=100, indicator='#'): 35 self.blocks = 0 36 self.current = 0 37 self.steps = steps 38 self.indicator = indicator 39 self.out = out 40 self.done = False 41 42 def start(self, message=None): 43 self.done = False 44 if message: 45 self.out.write('\n%s:\n' % message) 46 self.out.write('%--10---20---30---40---50---60---70---80---90--100\n') 47 48 def step(self, n=1): 49 self.current += n 50 51 old = self.blocks 52 self.blocks = int(round(self.current / float(self.steps) * 100) / 2) 53 54 if self.blocks > 50: 55 self.blocks = 50 56 57 new = self.blocks - old 58 59 self.out.write(self.indicator * new) 60 self.out.flush() 61 62 if self.blocks == 50 and not self.done: 63 self.done = True 64 self.out.write("\n") 65 66def set_to_list(s): 67 l = [] 68 l.extend(s) 69 return l 70 71def first(s, sorted=False): 72 """ 73 Return the first element of a set. 74 75 It sometimes useful to return the first element from a set but, 76 because sets are not indexable, this is rather hard. This function 77 will return the first element from a set. If sorted is True, then 78 the set will first be sorted (making this an expensive operation). 79 Otherwise a random element will be returned (as sets are not ordered). 80 """ 81 if not len(s): 82 raise IndexError("empty container") 83 84 if sorted: 85 l = set_to_list(s) 86 l.sort() 87 return l[0] 88 else: 89 for x in s: 90 return x 91 92def encode_input(text): 93 """Encode given text via preferred system encoding""" 94 # locale will often find out the correct encoding 95 encoding = locale.getpreferredencoding() 96 try: 97 encoded_text = text.encode(encoding) 98 except UnicodeError: 99 # if it fails to find correct encoding then ascii is used 100 # which may lead to UnicodeError if `text` contains non ascii signs 101 # utf-8 is our guess to fix the situation 102 encoded_text = text.encode('utf-8') 103 return encoded_text 104 105def decode_input(text): 106 """Decode given text via preferred system encoding""" 107 # locale will often find out the correct encoding 108 encoding = locale.getpreferredencoding() 109 try: 110 decoded_text = text.decode(encoding) 111 except UnicodeError: 112 # if it fails to find correct encoding then ascii is used 113 # which may lead to UnicodeError if `text` contains non ascii signs 114 # utf-8 is our guess to fix the situation 115 decoded_text = text.decode('utf-8') 116 return decoded_text 117 118class Comparison(): 119 """Class used when implementing rich comparison. 120 121 Inherit from this class if you want to have a rich 122 comparison within the class, afterwards implement 123 _compare function within your class.""" 124 125 def _compare(self, other, method): 126 return NotImplemented 127 128 def __eq__(self, other): 129 return self._compare(other, lambda a, b: a == b) 130 131 def __lt__(self, other): 132 return self._compare(other, lambda a, b: a < b) 133 134 def __le__(self, other): 135 return self._compare(other, lambda a, b: a <= b) 136 137 def __ge__(self, other): 138 return self._compare(other, lambda a, b: a >= b) 139 140 def __gt__(self, other): 141 return self._compare(other, lambda a, b: a > b) 142 143 def __ne__(self, other): 144 return self._compare(other, lambda a, b: a != b) 145 146if sys.version_info < (2,7): 147 # cmp_to_key function is missing in python2.6 148 def cmp_to_key(mycmp): 149 'Convert a cmp= function into a key= function' 150 class K: 151 def __init__(self, obj, *args): 152 self.obj = obj 153 def __lt__(self, other): 154 return mycmp(self.obj, other.obj) < 0 155 def __gt__(self, other): 156 return mycmp(self.obj, other.obj) > 0 157 def __eq__(self, other): 158 return mycmp(self.obj, other.obj) == 0 159 def __le__(self, other): 160 return mycmp(self.obj, other.obj) <= 0 161 def __ge__(self, other): 162 return mycmp(self.obj, other.obj) >= 0 163 def __ne__(self, other): 164 return mycmp(self.obj, other.obj) != 0 165 return K 166else: 167 from functools import cmp_to_key 168 169def cmp(first, second): 170 return (first > second) - (second > first) 171 172if __name__ == "__main__": 173 import time 174 p = ConsoleProgressBar(sys.stdout, steps=999) 175 p.start("computing pi") 176 for i in range(999): 177 p.step() 178 time.sleep(0.001) 179