SILENT KILLERPanel

Current Path: > > opt > alt > python38 > lib64 > python3.8 >


Operation   : Linux premium131.web-hosting.com 4.18.0-553.44.1.lve.el8.x86_64 #1 SMP Thu Mar 13 14:29:12 UTC 2025 x86_64
Software     : Apache
Server IP    : 162.0.232.56 | Your IP: 216.73.216.111
Domains      : 1034 Domain(s)
Permission   : [ 0755 ]

Files and Folders in: //opt/alt/python38/lib64/python3.8/

NameTypeSizeLast ModifiedActions
__pycache__ Directory - -
asyncio Directory - -
collections Directory - -
concurrent Directory - -
config-3.8-x86_64-linux-gnu Directory - -
ctypes Directory - -
curses Directory - -
dbm Directory - -
distutils Directory - -
email Directory - -
encodings Directory - -
ensurepip Directory - -
html Directory - -
http Directory - -
importlib Directory - -
json Directory - -
lib-dynload Directory - -
lib2to3 Directory - -
logging Directory - -
multiprocessing Directory - -
pydoc_data Directory - -
site-packages Directory - -
sqlite3 Directory - -
unittest Directory - -
urllib Directory - -
venv Directory - -
wsgiref Directory - -
xml Directory - -
xmlrpc Directory - -
LICENSE.txt File 13937 bytes September 06 2024 20:41:55.
__future__.py File 5147 bytes September 06 2024 20:41:55.
__phello__.foo.py File 64 bytes September 06 2024 20:41:55.
_bootlocale.py File 1801 bytes September 06 2024 20:41:55.
_collections_abc.py File 26100 bytes September 06 2024 20:41:55.
_compat_pickle.py File 8749 bytes September 06 2024 20:41:55.
_compression.py File 5340 bytes September 06 2024 20:41:55.
_dummy_thread.py File 6027 bytes September 06 2024 20:41:55.
_markupbase.py File 14598 bytes September 06 2024 20:41:55.
_osx_support.py File 21774 bytes September 06 2024 20:41:55.
_py_abc.py File 6189 bytes September 06 2024 20:41:55.
_pydecimal.py File 228666 bytes September 06 2024 20:41:55.
_pyio.py File 93177 bytes September 06 2024 20:41:55.
_sitebuiltins.py File 3115 bytes September 06 2024 20:41:55.
_strptime.py File 25268 bytes September 06 2024 20:41:55.
_sysconfigdata__linux_x86_64-linux-gnu.py File 41679 bytes September 23 2024 11:25:15.
_sysconfigdata_d_linux_x86_64-linux-gnu.py File 41438 bytes September 23 2024 11:17:38.
_threading_local.py File 7220 bytes September 06 2024 20:41:55.
_weakrefset.py File 5735 bytes September 06 2024 20:41:55.
abc.py File 4489 bytes September 06 2024 20:41:55.
aifc.py File 32814 bytes September 06 2024 20:41:55.
antigravity.py File 477 bytes September 06 2024 20:41:55.
argparse.py File 96015 bytes September 06 2024 20:41:55.
ast.py File 19234 bytes September 06 2024 20:41:55.
asynchat.py File 11328 bytes September 06 2024 20:41:55.
asyncore.py File 20094 bytes September 06 2024 20:41:55.
base64.py File 20395 bytes September 06 2024 20:41:55.
bdb.py File 32056 bytes September 06 2024 20:41:55.
binhex.py File 13954 bytes September 06 2024 20:41:55.
bisect.py File 2214 bytes September 06 2024 20:41:55.
bz2.py File 12558 bytes September 06 2024 20:41:55.
cProfile.py File 7023 bytes September 06 2024 20:41:55.
calendar.py File 24832 bytes September 06 2024 20:41:55.
cgi.py File 33945 bytes September 06 2024 20:41:55.
cgitb.py File 12096 bytes September 06 2024 20:41:55.
chunk.py File 5435 bytes September 06 2024 20:41:55.
cmd.py File 14860 bytes September 06 2024 20:41:55.
code.py File 10622 bytes September 06 2024 20:41:55.
codecs.py File 36667 bytes September 06 2024 20:41:55.
codeop.py File 6330 bytes September 06 2024 20:41:55.
colorsys.py File 4064 bytes September 06 2024 20:41:55.
compileall.py File 13678 bytes September 06 2024 20:41:55.
configparser.py File 54374 bytes September 06 2024 20:41:55.
contextlib.py File 24995 bytes September 06 2024 20:41:55.
contextvars.py File 129 bytes September 06 2024 20:41:55.
copy.py File 8661 bytes September 06 2024 20:41:55.
copyreg.py File 7135 bytes September 06 2024 20:41:55.
crypt.py File 3610 bytes September 06 2024 20:41:55.
csv.py File 16144 bytes September 06 2024 20:41:55.
dataclasses.py File 49973 bytes September 06 2024 20:41:55.
datetime.py File 88287 bytes September 06 2024 20:41:55.
decimal.py File 320 bytes September 06 2024 20:41:55.
difflib.py File 84058 bytes September 06 2024 20:41:55.
dis.py File 20570 bytes September 06 2024 20:41:55.
doctest.py File 104543 bytes September 06 2024 20:41:55.
dummy_threading.py File 2815 bytes September 06 2024 20:41:55.
enum.py File 38136 bytes September 06 2024 20:41:55.
filecmp.py File 9830 bytes September 06 2024 20:41:55.
fileinput.py File 14709 bytes September 06 2024 20:41:55.
fnmatch.py File 4079 bytes September 06 2024 20:41:55.
formatter.py File 15143 bytes September 06 2024 20:41:55.
fractions.py File 24329 bytes September 06 2024 20:41:55.
ftplib.py File 35129 bytes September 06 2024 20:41:55.
functools.py File 37406 bytes September 06 2024 20:41:55.
genericpath.py File 4975 bytes September 06 2024 20:41:55.
getopt.py File 7489 bytes September 06 2024 20:41:55.
getpass.py File 5994 bytes September 06 2024 20:41:55.
gettext.py File 27138 bytes September 06 2024 20:41:55.
glob.py File 5697 bytes September 06 2024 20:41:55.
gzip.py File 21413 bytes September 06 2024 20:41:55.
hashlib.py File 9730 bytes September 06 2024 20:41:55.
heapq.py File 22877 bytes September 06 2024 20:41:55.
hmac.py File 6629 bytes September 06 2024 20:41:55.
imaplib.py File 53606 bytes September 06 2024 20:41:55.
imghdr.py File 3808 bytes September 06 2024 20:41:55.
imp.py File 10536 bytes September 06 2024 20:41:55.
inspect.py File 118550 bytes September 06 2024 20:41:55.
io.py File 3541 bytes September 06 2024 20:41:55.
ipaddress.py File 74899 bytes September 06 2024 20:41:55.
keyword.py File 945 bytes September 06 2024 20:41:55.
linecache.py File 5330 bytes September 06 2024 20:41:55.
locale.py File 78191 bytes September 06 2024 20:41:55.
lzma.py File 12983 bytes September 06 2024 20:41:55.
mailbox.py File 78661 bytes September 06 2024 20:41:55.
mailcap.py File 9067 bytes September 06 2024 20:41:55.
mimetypes.py File 21664 bytes September 06 2024 20:41:55.
modulefinder.py File 24430 bytes September 06 2024 20:41:55.
netrc.py File 5566 bytes September 06 2024 20:41:55.
nntplib.py File 43261 bytes September 06 2024 20:41:55.
ntpath.py File 27734 bytes September 06 2024 20:41:55.
nturl2path.py File 2887 bytes September 06 2024 20:41:55.
numbers.py File 10244 bytes September 06 2024 20:41:55.
opcode.py File 5808 bytes September 06 2024 20:41:55.
operator.py File 10711 bytes September 06 2024 20:41:55.
optparse.py File 60369 bytes September 06 2024 20:41:55.
os.py File 38995 bytes September 06 2024 20:41:55.
pathlib.py File 52610 bytes September 06 2024 20:41:55.
pdb.py File 62751 bytes September 06 2024 20:41:55.
pickle.py File 64467 bytes September 06 2024 20:41:55.
pickletools.py File 93486 bytes September 06 2024 20:41:55.
pipes.py File 8916 bytes September 06 2024 20:41:55.
pkgutil.py File 21500 bytes September 06 2024 20:41:55.
platform.py File 40438 bytes September 06 2024 20:41:55.
plistlib.py File 32220 bytes September 06 2024 20:41:55.
poplib.py File 15077 bytes September 06 2024 20:41:55.
posixpath.py File 15627 bytes September 06 2024 20:41:55.
pprint.py File 21484 bytes September 06 2024 20:41:55.
profile.py File 23559 bytes September 06 2024 20:41:55.
pstats.py File 27345 bytes September 06 2024 20:41:55.
pty.py File 4807 bytes September 06 2024 20:41:55.
py_compile.py File 8203 bytes September 23 2024 11:15:42.
pyclbr.py File 15255 bytes September 06 2024 20:41:55.
pydoc.py File 106700 bytes September 23 2024 11:26:08.
queue.py File 11356 bytes September 06 2024 20:41:55.
quopri.py File 7265 bytes September 06 2024 20:41:55.
random.py File 28802 bytes September 06 2024 20:41:55.
re.py File 15861 bytes September 06 2024 20:41:55.
reprlib.py File 5267 bytes September 06 2024 20:41:55.
rlcompleter.py File 7097 bytes September 06 2024 20:41:55.
runpy.py File 12052 bytes September 06 2024 20:41:55.
sched.py File 6442 bytes September 06 2024 20:41:55.
secrets.py File 2038 bytes September 06 2024 20:41:55.
selectors.py File 18561 bytes September 06 2024 20:41:55.
shelve.py File 8527 bytes September 06 2024 20:41:55.
shlex.py File 13325 bytes September 06 2024 20:41:55.
shutil.py File 51761 bytes September 06 2024 20:41:55.
signal.py File 2273 bytes September 06 2024 20:41:55.
site.py File 21877 bytes September 23 2024 11:15:42.
smtpd.py File 34722 bytes September 06 2024 20:41:55.
smtplib.py File 45014 bytes September 06 2024 20:41:55.
sndhdr.py File 7099 bytes September 06 2024 20:41:55.
socket.py File 35464 bytes September 06 2024 20:41:55.
socketserver.py File 27296 bytes September 06 2024 20:41:55.
sre_compile.py File 26695 bytes September 06 2024 20:41:55.
sre_constants.py File 7154 bytes September 06 2024 20:41:55.
sre_parse.py File 40230 bytes September 06 2024 20:41:55.
ssl.py File 52533 bytes September 06 2024 20:41:55.
stat.py File 5485 bytes September 06 2024 20:41:55.
statistics.py File 39690 bytes September 06 2024 20:41:55.
string.py File 10535 bytes September 06 2024 20:41:55.
stringprep.py File 12917 bytes September 06 2024 20:41:55.
struct.py File 257 bytes September 06 2024 20:41:55.
subprocess.py File 78250 bytes September 06 2024 20:41:55.
sunau.py File 18375 bytes September 06 2024 20:41:55.
symbol.py File 2109 bytes September 23 2024 11:18:30.
symtable.py File 8021 bytes September 06 2024 20:41:55.
sysconfig.py File 24893 bytes September 23 2024 11:15:42.
tabnanny.py File 11419 bytes September 06 2024 20:41:55.
tarfile.py File 106031 bytes September 06 2024 20:41:55.
telnetlib.py File 23254 bytes September 06 2024 20:41:55.
tempfile.py File 27822 bytes September 06 2024 20:41:55.
textwrap.py File 19407 bytes September 06 2024 20:41:55.
this.py File 1003 bytes September 06 2024 20:41:55.
threading.py File 50820 bytes September 06 2024 20:41:55.
timeit.py File 13493 bytes September 06 2024 20:41:55.
token.py File 2368 bytes September 06 2024 20:41:55.
tokenize.py File 25841 bytes September 06 2024 20:41:55.
trace.py File 29883 bytes September 06 2024 20:41:55.
traceback.py File 23611 bytes September 06 2024 20:41:55.
tracemalloc.py File 17076 bytes September 06 2024 20:41:55.
tty.py File 879 bytes September 06 2024 20:41:55.
types.py File 9713 bytes September 06 2024 20:41:55.
typing.py File 68962 bytes September 06 2024 20:41:55.
uu.py File 7277 bytes September 23 2024 11:26:07.
uuid.py File 30466 bytes September 06 2024 20:41:55.
warnings.py File 19688 bytes September 06 2024 20:41:55.
wave.py File 18230 bytes September 06 2024 20:41:55.
weakref.py File 21387 bytes September 06 2024 20:41:55.
webbrowser.py File 24096 bytes September 06 2024 20:41:55.
xdrlib.py File 5913 bytes September 06 2024 20:41:55.
zipapp.py File 7535 bytes September 06 2024 20:41:55.
zipfile.py File 88476 bytes September 06 2024 20:41:55.
zipimport.py File 30765 bytes September 06 2024 20:41:55.

Reading File: //opt/alt/python38/lib64/python3.8//textwrap.py

"""Text wrapping and filling.
"""

# Copyright (C) 1999-2001 Gregory P. Ward.
# Copyright (C) 2002, 2003 Python Software Foundation.
# Written by Greg Ward <gward@python.net>

import re

__all__ = ['TextWrapper', 'wrap', 'fill', 'dedent', 'indent', 'shorten']

# Hardcode the recognized whitespace characters to the US-ASCII
# whitespace characters.  The main reason for doing this is that
# some Unicode spaces (like \u00a0) are non-breaking whitespaces.
_whitespace = '\t\n\x0b\x0c\r '

class TextWrapper:
    """
    Object for wrapping/filling text.  The public interface consists of
    the wrap() and fill() methods; the other methods are just there for
    subclasses to override in order to tweak the default behaviour.
    If you want to completely replace the main wrapping algorithm,
    you'll probably have to override _wrap_chunks().

    Several instance attributes control various aspects of wrapping:
      width (default: 70)
        the maximum width of wrapped lines (unless break_long_words
        is false)
      initial_indent (default: "")
        string that will be prepended to the first line of wrapped
        output.  Counts towards the line's width.
      subsequent_indent (default: "")
        string that will be prepended to all lines save the first
        of wrapped output; also counts towards each line's width.
      expand_tabs (default: true)
        Expand tabs in input text to spaces before further processing.
        Each tab will become 0 .. 'tabsize' spaces, depending on its position
        in its line.  If false, each tab is treated as a single character.
      tabsize (default: 8)
        Expand tabs in input text to 0 .. 'tabsize' spaces, unless
        'expand_tabs' is false.
      replace_whitespace (default: true)
        Replace all whitespace characters in the input text by spaces
        after tab expansion.  Note that if expand_tabs is false and
        replace_whitespace is true, every tab will be converted to a
        single space!
      fix_sentence_endings (default: false)
        Ensure that sentence-ending punctuation is always followed
        by two spaces.  Off by default because the algorithm is
        (unavoidably) imperfect.
      break_long_words (default: true)
        Break words longer than 'width'.  If false, those words will not
        be broken, and some lines might be longer than 'width'.
      break_on_hyphens (default: true)
        Allow breaking hyphenated words. If true, wrapping will occur
        preferably on whitespaces and right after hyphens part of
        compound words.
      drop_whitespace (default: true)
        Drop leading and trailing whitespace from lines.
      max_lines (default: None)
        Truncate wrapped lines.
      placeholder (default: ' [...]')
        Append to the last line of truncated text.
    """

    unicode_whitespace_trans = {}
    uspace = ord(' ')
    for x in _whitespace:
        unicode_whitespace_trans[ord(x)] = uspace

    # This funky little regex is just the trick for splitting
    # text up into word-wrappable chunks.  E.g.
    #   "Hello there -- you goof-ball, use the -b option!"
    # splits into
    #   Hello/ /there/ /--/ /you/ /goof-/ball,/ /use/ /the/ /-b/ /option!
    # (after stripping out empty strings).
    word_punct = r'[\w!"\'&.,?]'
    letter = r'[^\d\W]'
    whitespace = r'[%s]' % re.escape(_whitespace)
    nowhitespace = '[^' + whitespace[1:]
    wordsep_re = re.compile(r'''
        ( # any whitespace
          %(ws)s+
        | # em-dash between words
          (?<=%(wp)s) -{2,} (?=\w)
        | # word, possibly hyphenated
          %(nws)s+? (?:
            # hyphenated word
              -(?: (?<=%(lt)s{2}-) | (?<=%(lt)s-%(lt)s-))
              (?= %(lt)s -? %(lt)s)
            | # end of word
              (?=%(ws)s|\Z)
            | # em-dash
              (?<=%(wp)s) (?=-{2,}\w)
            )
        )''' % {'wp': word_punct, 'lt': letter,
                'ws': whitespace, 'nws': nowhitespace},
        re.VERBOSE)
    del word_punct, letter, nowhitespace

    # This less funky little regex just split on recognized spaces. E.g.
    #   "Hello there -- you goof-ball, use the -b option!"
    # splits into
    #   Hello/ /there/ /--/ /you/ /goof-ball,/ /use/ /the/ /-b/ /option!/
    wordsep_simple_re = re.compile(r'(%s+)' % whitespace)
    del whitespace

    # XXX this is not locale- or charset-aware -- string.lowercase
    # is US-ASCII only (and therefore English-only)
    sentence_end_re = re.compile(r'[a-z]'             # lowercase letter
                                 r'[\.\!\?]'          # sentence-ending punct.
                                 r'[\"\']?'           # optional end-of-quote
                                 r'\Z')               # end of chunk

    def __init__(self,
                 width=70,
                 initial_indent="",
                 subsequent_indent="",
                 expand_tabs=True,
                 replace_whitespace=True,
                 fix_sentence_endings=False,
                 break_long_words=True,
                 drop_whitespace=True,
                 break_on_hyphens=True,
                 tabsize=8,
                 *,
                 max_lines=None,
                 placeholder=' [...]'):
        self.width = width
        self.initial_indent = initial_indent
        self.subsequent_indent = subsequent_indent
        self.expand_tabs = expand_tabs
        self.replace_whitespace = replace_whitespace
        self.fix_sentence_endings = fix_sentence_endings
        self.break_long_words = break_long_words
        self.drop_whitespace = drop_whitespace
        self.break_on_hyphens = break_on_hyphens
        self.tabsize = tabsize
        self.max_lines = max_lines
        self.placeholder = placeholder


    # -- Private methods -----------------------------------------------
    # (possibly useful for subclasses to override)

    def _munge_whitespace(self, text):
        """_munge_whitespace(text : string) -> string

        Munge whitespace in text: expand tabs and convert all other
        whitespace characters to spaces.  Eg. " foo\\tbar\\n\\nbaz"
        becomes " foo    bar  baz".
        """
        if self.expand_tabs:
            text = text.expandtabs(self.tabsize)
        if self.replace_whitespace:
            text = text.translate(self.unicode_whitespace_trans)
        return text


    def _split(self, text):
        """_split(text : string) -> [string]

        Split the text to wrap into indivisible chunks.  Chunks are
        not quite the same as words; see _wrap_chunks() for full
        details.  As an example, the text
          Look, goof-ball -- use the -b option!
        breaks into the following chunks:
          'Look,', ' ', 'goof-', 'ball', ' ', '--', ' ',
          'use', ' ', 'the', ' ', '-b', ' ', 'option!'
        if break_on_hyphens is True, or in:
          'Look,', ' ', 'goof-ball', ' ', '--', ' ',
          'use', ' ', 'the', ' ', '-b', ' ', option!'
        otherwise.
        """
        if self.break_on_hyphens is True:
            chunks = self.wordsep_re.split(text)
        else:
            chunks = self.wordsep_simple_re.split(text)
        chunks = [c for c in chunks if c]
        return chunks

    def _fix_sentence_endings(self, chunks):
        """_fix_sentence_endings(chunks : [string])

        Correct for sentence endings buried in 'chunks'.  Eg. when the
        original text contains "... foo.\\nBar ...", munge_whitespace()
        and split() will convert that to [..., "foo.", " ", "Bar", ...]
        which has one too few spaces; this method simply changes the one
        space to two.
        """
        i = 0
        patsearch = self.sentence_end_re.search
        while i < len(chunks)-1:
            if chunks[i+1] == " " and patsearch(chunks[i]):
                chunks[i+1] = "  "
                i += 2
            else:
                i += 1

    def _handle_long_word(self, reversed_chunks, cur_line, cur_len, width):
        """_handle_long_word(chunks : [string],
                             cur_line : [string],
                             cur_len : int, width : int)

        Handle a chunk of text (most likely a word, not whitespace) that
        is too long to fit in any line.
        """
        # Figure out when indent is larger than the specified width, and make
        # sure at least one character is stripped off on every pass
        if width < 1:
            space_left = 1
        else:
            space_left = width - cur_len

        # If we're allowed to break long words, then do so: put as much
        # of the next chunk onto the current line as will fit.
        if self.break_long_words:
            cur_line.append(reversed_chunks[-1][:space_left])
            reversed_chunks[-1] = reversed_chunks[-1][space_left:]

        # Otherwise, we have to preserve the long word intact.  Only add
        # it to the current line if there's nothing already there --
        # that minimizes how much we violate the width constraint.
        elif not cur_line:
            cur_line.append(reversed_chunks.pop())

        # If we're not allowed to break long words, and there's already
        # text on the current line, do nothing.  Next time through the
        # main loop of _wrap_chunks(), we'll wind up here again, but
        # cur_len will be zero, so the next line will be entirely
        # devoted to the long word that we can't handle right now.

    def _wrap_chunks(self, chunks):
        """_wrap_chunks(chunks : [string]) -> [string]

        Wrap a sequence of text chunks and return a list of lines of
        length 'self.width' or less.  (If 'break_long_words' is false,
        some lines may be longer than this.)  Chunks correspond roughly
        to words and the whitespace between them: each chunk is
        indivisible (modulo 'break_long_words'), but a line break can
        come between any two chunks.  Chunks should not have internal
        whitespace; ie. a chunk is either all whitespace or a "word".
        Whitespace chunks will be removed from the beginning and end of
        lines, but apart from that whitespace is preserved.
        """
        lines = []
        if self.width <= 0:
            raise ValueError("invalid width %r (must be > 0)" % self.width)
        if self.max_lines is not None:
            if self.max_lines > 1:
                indent = self.subsequent_indent
            else:
                indent = self.initial_indent
            if len(indent) + len(self.placeholder.lstrip()) > self.width:
                raise ValueError("placeholder too large for max width")

        # Arrange in reverse order so items can be efficiently popped
        # from a stack of chucks.
        chunks.reverse()

        while chunks:

            # Start the list of chunks that will make up the current line.
            # cur_len is just the length of all the chunks in cur_line.
            cur_line = []
            cur_len = 0

            # Figure out which static string will prefix this line.
            if lines:
                indent = self.subsequent_indent
            else:
                indent = self.initial_indent

            # Maximum width for this line.
            width = self.width - len(indent)

            # First chunk on line is whitespace -- drop it, unless this
            # is the very beginning of the text (ie. no lines started yet).
            if self.drop_whitespace and chunks[-1].strip() == '' and lines:
                del chunks[-1]

            while chunks:
                l = len(chunks[-1])

                # Can at least squeeze this chunk onto the current line.
                if cur_len + l <= width:
                    cur_line.append(chunks.pop())
                    cur_len += l

                # Nope, this line is full.
                else:
                    break

            # The current line is full, and the next chunk is too big to
            # fit on *any* line (not just this one).
            if chunks and len(chunks[-1]) > width:
                self._handle_long_word(chunks, cur_line, cur_len, width)
                cur_len = sum(map(len, cur_line))

            # If the last chunk on this line is all whitespace, drop it.
            if self.drop_whitespace and cur_line and cur_line[-1].strip() == '':
                cur_len -= len(cur_line[-1])
                del cur_line[-1]

            if cur_line:
                if (self.max_lines is None or
                    len(lines) + 1 < self.max_lines or
                    (not chunks or
                     self.drop_whitespace and
                     len(chunks) == 1 and
                     not chunks[0].strip()) and cur_len <= width):
                    # Convert current line back to a string and store it in
                    # list of all lines (return value).
                    lines.append(indent + ''.join(cur_line))
                else:
                    while cur_line:
                        if (cur_line[-1].strip() and
                            cur_len + len(self.placeholder) <= width):
                            cur_line.append(self.placeholder)
                            lines.append(indent + ''.join(cur_line))
                            break
                        cur_len -= len(cur_line[-1])
                        del cur_line[-1]
                    else:
                        if lines:
                            prev_line = lines[-1].rstrip()
                            if (len(prev_line) + len(self.placeholder) <=
                                    self.width):
                                lines[-1] = prev_line + self.placeholder
                                break
                        lines.append(indent + self.placeholder.lstrip())
                    break

        return lines

    def _split_chunks(self, text):
        text = self._munge_whitespace(text)
        return self._split(text)

    # -- Public interface ----------------------------------------------

    def wrap(self, text):
        """wrap(text : string) -> [string]

        Reformat the single paragraph in 'text' so it fits in lines of
        no more than 'self.width' columns, and return a list of wrapped
        lines.  Tabs in 'text' are expanded with string.expandtabs(),
        and all other whitespace characters (including newline) are
        converted to space.
        """
        chunks = self._split_chunks(text)
        if self.fix_sentence_endings:
            self._fix_sentence_endings(chunks)
        return self._wrap_chunks(chunks)

    def fill(self, text):
        """fill(text : string) -> string

        Reformat the single paragraph in 'text' to fit in lines of no
        more than 'self.width' columns, and return a new string
        containing the entire wrapped paragraph.
        """
        return "\n".join(self.wrap(text))


# -- Convenience interface ---------------------------------------------

def wrap(text, width=70, **kwargs):
    """Wrap a single paragraph of text, returning a list of wrapped lines.

    Reformat the single paragraph in 'text' so it fits in lines of no
    more than 'width' columns, and return a list of wrapped lines.  By
    default, tabs in 'text' are expanded with string.expandtabs(), and
    all other whitespace characters (including newline) are converted to
    space.  See TextWrapper class for available keyword args to customize
    wrapping behaviour.
    """
    w = TextWrapper(width=width, **kwargs)
    return w.wrap(text)

def fill(text, width=70, **kwargs):
    """Fill a single paragraph of text, returning a new string.

    Reformat the single paragraph in 'text' to fit in lines of no more
    than 'width' columns, and return a new string containing the entire
    wrapped paragraph.  As with wrap(), tabs are expanded and other
    whitespace characters converted to space.  See TextWrapper class for
    available keyword args to customize wrapping behaviour.
    """
    w = TextWrapper(width=width, **kwargs)
    return w.fill(text)

def shorten(text, width, **kwargs):
    """Collapse and truncate the given text to fit in the given width.

    The text first has its whitespace collapsed.  If it then fits in
    the *width*, it is returned as is.  Otherwise, as many words
    as possible are joined and then the placeholder is appended::

        >>> textwrap.shorten("Hello  world!", width=12)
        'Hello world!'
        >>> textwrap.shorten("Hello  world!", width=11)
        'Hello [...]'
    """
    w = TextWrapper(width=width, max_lines=1, **kwargs)
    return w.fill(' '.join(text.strip().split()))


# -- Loosely related functionality -------------------------------------

_whitespace_only_re = re.compile('^[ \t]+$', re.MULTILINE)
_leading_whitespace_re = re.compile('(^[ \t]*)(?:[^ \t\n])', re.MULTILINE)

def dedent(text):
    """Remove any common leading whitespace from every line in `text`.

    This can be used to make triple-quoted strings line up with the left
    edge of the display, while still presenting them in the source code
    in indented form.

    Note that tabs and spaces are both treated as whitespace, but they
    are not equal: the lines "  hello" and "\\thello" are
    considered to have no common leading whitespace.

    Entirely blank lines are normalized to a newline character.
    """
    # Look for the longest leading string of spaces and tabs common to
    # all lines.
    margin = None
    text = _whitespace_only_re.sub('', text)
    indents = _leading_whitespace_re.findall(text)
    for indent in indents:
        if margin is None:
            margin = indent

        # Current line more deeply indented than previous winner:
        # no change (previous winner is still on top).
        elif indent.startswith(margin):
            pass

        # Current line consistent with and no deeper than previous winner:
        # it's the new winner.
        elif margin.startswith(indent):
            margin = indent

        # Find the largest common whitespace between current line and previous
        # winner.
        else:
            for i, (x, y) in enumerate(zip(margin, indent)):
                if x != y:
                    margin = margin[:i]
                    break

    # sanity check (testing/debugging only)
    if 0 and margin:
        for line in text.split("\n"):
            assert not line or line.startswith(margin), \
                   "line = %r, margin = %r" % (line, margin)

    if margin:
        text = re.sub(r'(?m)^' + margin, '', text)
    return text


def indent(text, prefix, predicate=None):
    """Adds 'prefix' to the beginning of selected lines in 'text'.

    If 'predicate' is provided, 'prefix' will only be added to the lines
    where 'predicate(line)' is True. If 'predicate' is not provided,
    it will default to adding 'prefix' to all non-empty lines that do not
    consist solely of whitespace characters.
    """
    if predicate is None:
        def predicate(line):
            return line.strip()

    def prefixed_lines():
        for line in text.splitlines(True):
            yield (prefix + line if predicate(line) else line)
    return ''.join(prefixed_lines())


if __name__ == "__main__":
    #print dedent("\tfoo\n\tbar")
    #print dedent("  \thello there\n  \t  how are you?")
    print(dedent("Hello there.\n  This is indented."))

SILENT KILLER Tool