
    @g'x              
          d Z ddlZddlZddlZddlZddlZddlZddlZ	 ddlZddl	Z	ddl
Z
dZddlmZmZmZmZ ddlmZ ddlmZ ddlmZmZ dd	lmZ d
ZdZ	 dZ	 dez  Z	 edeZ	 dez  Z 	 dez  Z!	  e"dddddddd      Z#	  e"dddddd      Z$	  ejJ                  d ejL                  e      z        Z'	 dZ(	 dZ)	 dZ*	 ejV                  jY                  dd       Z-	 d;d!Z.d" Z/d# Z0d$ Z1d% Z2d& Z3d<d'Z4ed(        Z5d) Z6d* Z7d+ Z8d<d,Z9ed-        Z:d. Z;d/ Z<d0 Z=d1 Z>e*fd2Z?d<d3Z@d4 ZAd5 ZB eeCd6d7d8d9:       y# e$ r dZY w xY w)=a	  
Interaction with interactive text terminals.

The :mod:`~humanfriendly.terminal` module makes it easy to interact with
interactive text terminals and format text for rendering on such terminals. If
the terms used in the documentation of this module don't make sense to you then
please refer to the `Wikipedia article on ANSI escape sequences`_ for details
about how ANSI escape sequences work.

This module was originally developed for use on UNIX systems, but since then
Windows 10 gained native support for ANSI escape sequences and this module was
enhanced to recognize and support this. For details please refer to the
:func:`enable_ansi_support()` function.

.. _Wikipedia article on ANSI escape sequences: http://en.wikipedia.org/wiki/ANSI_escape_code#Sequence_elements
    NTF)coerce_string
is_unicode
on_windowswhich)cached)define_aliases)concatenateformat)format_usage)"ANSI_COLOR_CODESANSI_CSIANSI_ERASE_LINEANSI_HIDE_CURSOR
ANSI_RESETANSI_SGRANSI_SHOW_CURSORANSI_TEXT_STYLESCLEAN_OUTPUT_PATTERNDEFAULT_COLUMNSDEFAULT_ENCODINGDEFAULT_LINESHIGHLIGHT_COLOR
ansi_strip
ansi_style
ansi_width	ansi_wrapauto_encodeclean_terminal_outputconnected_to_terminalenable_ansi_supportfind_terminal_sizefind_terminal_size_using_ioctlfind_terminal_size_using_sttyget_pager_command have_windows_native_ansi_supportmessageoutputreadline_stripreadline_wrap
show_pagerterminal_supports_colorsusagewarningz[mz%sK0z%s?25lz%s?25h                     )blackredgreenyellowbluemagentacyanwhite	   )boldfaintitalic	underlineinversestrike_throughz
(|
||%s)   P   zUTF-8HUMANFRIENDLY_HIGHLIGHT_COLORr9   c                     t        j                  t              dt        j                  t              }t        j                  |d|       } |rt        |       } | S )a  
    Strip ANSI escape sequences from the given string.

    :param text: The text from which ANSI escape sequences should be removed (a
                 string).
    :param readline_hints: If :data:`True` then :func:`readline_strip()` is
                           used to remove `readline hints`_ from the string.
    :returns: The text without ANSI escape sequences (a string).
    z.*? )reescaper   r   subr(   )textreadline_hintspatterns      T/var/www/openai/venv/lib/python3.12/site-packages/humanfriendly/terminal/__init__.pyr   r      sD     99X.		(0CDG66'2t$Dd#K    c                     | j                         D cg c]  \  }}|t        v s|st        |    }}}dD ][  }| j                  |      }t        |t        t
        f      rgt        |      dk7  rd}t        ||z        |j                  |dk(  rdnd       |j                  d       |j                  t        t        |             t        |t        j                        r%|j                  |dk(  rdndd	t        |      f       |s|t        vr6d
}t        ||t        t        t         t#        t                          fz        |dk(  r| j                  d      rdndn| j                  d      rdnd}|j                  |t        |   z          ^ |rKt$        dj'                  t        t(        |            z   t*        z   }| j                  d      rt-        |      S |S yc c}}w )a  
    Generate ANSI escape sequences for the given color and/or style(s).

    :param color: The foreground color. Three types of values are supported:

                  - The name of a color (one of the strings 'black', 'red',
                    'green', 'yellow', 'blue', 'magenta', 'cyan' or 'white').
                  - An integer that refers to the 256 color mode palette.
                  - A tuple or list with three integers representing an RGB
                    (red, green, blue) value.

                  The value :data:`None` (the default) means no escape
                  sequence to switch color will be emitted.
    :param background: The background color (see the description
                       of the `color` argument).
    :param bright: Use high intensity colors instead of default colors
                   (a boolean, defaults to :data:`False`).
    :param readline_hints: If :data:`True` then :func:`readline_wrap()` is
                           applied to the generated ANSI escape sequences (the
                           default is :data:`False`).
    :param kw: Any additional keyword arguments are expected to match a key
               in the :data:`ANSI_TEXT_STYLES` dictionary. If the argument's
               value evaluates to :data:`True` the respective style will be
               enabled.
    :returns: The ANSI escape sequences to enable the requested text styles or
              an empty string if no styles were requested.
    :raises: :exc:`~exceptions.ValueError` when an invalid color name is given.

    Even though only eight named colors are supported, the use of `bright=True`
    and `faint=True` increases the number of available colors to around 24 (it
    may be slightly lower, for example because faint black is just black).

    **Support for 8-bit colors**

    In `release 4.7`_ support for 256 color mode was added. While this
    significantly increases the available colors it's not very human friendly
    in usage because you need to look up color codes in the `256 color mode
    palette <https://en.wikipedia.org/wiki/ANSI_escape_code#8-bit>`_.

    You can use the ``humanfriendly --demo`` command to get a demonstration of
    the available colors, see also the screen shot below. Note that the small
    font size in the screen shot was so that the demonstration of 256 color
    mode support would fit into a single screen shot without scrolling :-)
    (I wasn't feeling very creative).

      .. image:: images/ansi-demo.png

    **Support for 24-bit colors**

    In `release 4.14`_ support for 24-bit colors was added by accepting a tuple
    or list with three integers representing the RGB (red, green, blue) value
    of a color. This is not included in the demo because rendering millions of
    colors was deemed unpractical ;-).

    .. _release 4.7: http://humanfriendly.readthedocs.io/en/latest/changelog.html#release-4-7-2018-01-14
    .. _release 4.14: http://humanfriendly.readthedocs.io/en/latest/changelog.html#release-4-14-2018-07-13
    )color
backgroundr2   zCInvalid color value %r! (expected tuple or list with three numbers)rU   0   &   r1   '   r4   zFInvalid color value %r! (expected an integer or one of the strings %s)brightd   (   Z      ;rO   rJ   )itemsr   get
isinstancetuplelistlen
ValueErrorappendextendmapintnumbersNumberr   r	   reprsortedr   joinstrr   r)   )	kwkv	sequences
color_typecolor_valuemsgoffsetencodeds	            rQ   r   r      s   v 24[AqDT?TYZ!!$I[+
ffZ(kE4=1;1$[ {!233:#=R2FQSk23W^^4 L0b3{#  "22^ [T6RbKcAd5e'f!fgg
 - x(bFF8,b"  V&6{&CCD9 ,: SXXc#y&9::XE)+0@)A}W%NwNG \s   G%G%G%c                 *    t        t        |             S )a  
    Calculate the effective width of the given text (ignoring ANSI escape sequences).

    :param text: The text whose width should be calculated (a string).
    :returns: The width of the text without ANSI escape sequences (an
              integer).

    This function uses :func:`ansi_strip()` to strip ANSI escape sequences from
    the given string and returns the length of the resulting string.
    )rd   r   )rN   s    rQ   r   r     s     z$  rR   c                 t    t        di |}|r*t        }|j                  d      rt        |      }|| z   |z   S | S )a  
    Wrap text in ANSI escape sequences for the given color and/or style(s).

    :param text: The text to wrap (a string).
    :param kw: Any keyword arguments are passed to :func:`ansi_style()`.
    :returns: The result of this function depends on the keyword arguments:

              - If :func:`ansi_style()` generates an ANSI escape sequence based
                on the keyword arguments, the given text is prefixed with the
                generated ANSI escape sequence and suffixed with
                :data:`ANSI_RESET`.

              - If :func:`ansi_style()` returns an empty string then the text
                given by the caller is returned unchanged.
    rO    )r   r   r`   r)   )rN   rp   start_sequenceend_sequences       rQ   r   r     sE       %"%N!66"#(6L$|33rR   c                     t        |g|i |}	 | j                  |       y# t        $ r, | j                  t        j                  |t
                     Y yw xY w)af  
    Reliably write Unicode strings to the terminal.

    :param stream: The file-like object to write to (a value like
                   :data:`sys.stdout` or :data:`sys.stderr`).
    :param text: The text to write to the stream (a string).
    :param args: Refer to :func:`~humanfriendly.text.format()`.
    :param kw: Refer to :func:`~humanfriendly.text.format()`.

    Renders the text using :func:`~humanfriendly.text.format()` and writes it
    to the given stream. If an :exc:`~exceptions.UnicodeEncodeError` is
    encountered in doing so, the text is encoded using :data:`DEFAULT_ENCODING`
    and the write is retried. The reasoning behind this rather blunt approach
    is that it's preferable to get output on the command line in the wrong
    encoding then to have the Python program blow up with a
    :exc:`~exceptions.UnicodeEncodeError` exception.
    N)r
   writeUnicodeEncodeErrorcodecsencoder   )streamrN   argsrp   s       rQ   r   r   /  sP    $ $$$$D<T <V]]4)9:;<s   # 2AAc                 v   g }d}d}t         j                  |       D ]i  }|dk(  rd}|dk(  rt        d|dz
        } |dk(  r|j                  |       |dt        fv rd}d}E|sH|t        |      z   }|d| }||d }||z   |z   }|}k |j                  |       |r|d   s|j                  d       |r|d   s|S )	a  
    Clean up the terminal output of a command.

    :param text: The raw text with special characters (a Unicode string).
    :returns: A list of Unicode strings (one for each line).

    This function emulates the effect of backspace (0x08), carriage return
    (0x0D) and line feed (0x0A) characters and the ANSI 'erase line' escape
    sequence on interactive terminals. It's intended to clean up command output
    that was originally meant to be rendered on an interactive terminal and
    that has been captured using e.g. the :man:`script` program [#]_ or the
    :mod:`pty` module [#]_.

    .. [#] My coloredlogs_ package supports the ``coloredlogs --to-html``
           command which uses :man:`script` to fool a subprocess into thinking
           that it's connected to an interactive terminal (in order to get it
           to emit ANSI escape sequences).

    .. [#] My capturer_ package uses the :mod:`pty` module to fool the current
           process and subprocesses into thinking they are connected to an
           interactive terminal (in order to get them to emit ANSI escape
           sequences).

    **Some caveats about the use of this function:**

    - Strictly speaking the effect of carriage returns cannot be emulated
      outside of an actual terminal due to the interaction between overlapping
      output, terminal widths and line wrapping. The goal of this function is
      to sanitize noise in terminal output while preserving useful output.
      Think of it as a useful and pragmatic but possibly lossy conversion.

    - The algorithm isn't smart enough to properly handle a pair of ANSI escape
      sequences that open before a carriage return and close after the last
      carriage return in a linefeed delimited string; the resulting string will
      contain only the closing end of the ANSI escape sequence pair. Tracking
      this kind of complexity requires a state machine and proper parsing.

    .. _capturer: https://pypi.org/project/capturer
    .. _coloredlogs: https://pypi.org/project/coloredlogs
    rJ   r   r0   
N)r   splitmaxrf   r   rd   pop)rN   cleaned_linescurrent_linecurrent_positiontokennew_positionprefixsuffixs           rQ   r   r   H  s    R ML%++D1D= d]"1&6&:;}$$\2//!#$ /#e*<%&7'78%lm4%~6#/ + 2. &
b 1" b 1rR   c                 j    | t         j                  n| } 	 | j                         S # t        $ r Y yw xY w)aF  
    Check if a stream is connected to a terminal.

    :param stream: The stream to check (a file-like object,
                   defaults to :data:`sys.stdout`).
    :returns: :data:`True` if the stream is connected to a terminal,
              :data:`False` otherwise.

    See also :func:`terminal_supports_colors()`.
    F)sysstdoutisatty	Exception)r   s    rQ   r   r     s6     ">SZZvF}} s   & 	22c                     t               rddl} | j                  j                  j	                  | j                  j                  j                  d      d       | j                  j                  j	                  | j                  j                  j                  d      d       yt               r)dt        j                  v ry	 ddl	}|j                          yt               S # t        $ r Y yw xY w)	a  
    Try to enable support for ANSI escape sequences (required on Windows).

    :returns: :data:`True` if ANSI is supported, :data:`False` otherwise.

    This functions checks for the following supported configurations, in the
    given order:

    1. On Windows, if :func:`have_windows_native_ansi_support()` confirms
       native support for ANSI escape sequences :mod:`ctypes` will be used to
       enable this support.

    2. On Windows, if the environment variable ``$ANSICON`` is set nothing is
       done because it is assumed that support for ANSI escape sequences has
       already been enabled via `ansicon <https://github.com/adoxa/ansicon>`_.

    3. On Windows, an attempt is made to import and initialize the Python
       package :pypi:`colorama` instead (of course for this to work
       :pypi:`colorama` has to be installed).

    4. On other platforms this function calls :func:`connected_to_terminal()`
       to determine whether ANSI escape sequences are supported (that is to
       say all platforms that are not Windows are assumed to support ANSI
       escape sequences natively, without weird contortions like above).

       This makes it possible to call :func:`enable_ansi_support()`
       unconditionally without checking the current platform.

    The :func:`~humanfriendly.decorators.cached` decorator is used to ensure
    that this function is only executed once, but its return value remains
    available on later calls.
    r   Nir6   iTANSICONF)r%   ctypeswindllkernel32SetConsoleModeGetStdHandler   osenvironcoloramainitImportErrorr   )r   r   s     rQ   r    r      s    D ()--fmm.D.D.Q.QRU.VXYZ--fmm.D.D.Q.QRU.VXYZ	

"	MMO %&&  		s    C 	C+*C+c                  H   t         j                  t         j                  t         j                  fD ]   } 	 t	        |       }t        |      dk\  r|c S " 	 t               }t        |      dk\  r|S 	 t        t        fS # t        $ r Y Vw xY w# t        $ r Y t        t        fS w xY w)a  
    Determine the number of lines and columns visible in the terminal.

    :returns: A tuple of two integers with the line and column count.

    The result of this function is based on the first of the following three
    methods that works:

    1. First :func:`find_terminal_size_using_ioctl()` is tried,
    2. then :func:`find_terminal_size_using_stty()` is tried,
    3. finally :data:`DEFAULT_LINES` and :data:`DEFAULT_COLUMNS` are returned.

    .. note:: The :func:`find_terminal_size()` function performs the steps
              above every time it is called, the result is not cached. This is
              because the size of a virtual terminal can change at any time and
              the result of :func:`find_terminal_size()` should be correct.

              `Pre-emptive snarky comment`_: It's possible to cache the result
              of this function and use :mod:`signal.SIGWINCH <signal>` to
              refresh the cached values!

              Response: As a library I don't consider it the role of the
              :mod:`humanfriendly.terminal` module to install a process wide
              signal handler ...

    .. _Pre-emptive snarky comment: http://blogs.msdn.com/b/oldnewthing/archive/2008/01/30/7315957.aspx
    r0   )
r   stdinr   stderrr"   minr   r#   r   r   )r   results     rQ   r!   r!     s    < ))SZZ3	3F;F6{a   4.0v;!M 
 /))  		  /))s#   A;B
 ;	BB
	B! B!c                     t         st        d      t        j                  dt	        j
                  | t        j                  t        j                  ddddd                  \  }}}}||fS )a  
    Find the terminal size using :func:`fcntl.ioctl()`.

    :param stream: A stream connected to the terminal (a file object with a
                   ``fileno`` attribute).
    :returns: A tuple of two integers with the line and column count.
    :raises: This function can raise exceptions but I'm not going to document
             them here, you should be using :func:`find_terminal_size()`.

    Based on an `implementation found on StackOverflow <http://stackoverflow.com/a/3010495/788200>`_.
    z2It looks like the `fcntl' module is not available!HHHHr   )	
HAVE_IOCTLNotImplementedErrorstructunpackfcntlioctltermios
TIOCGWINSZpack)r   hwhpwps        rQ   r"   r"     s`     !"VWW==VW=O=OQWQ\Q\]cefhiklnoQp)qrLAq"ba4KrR   c                     t        j                  ddgt         j                  t         j                        } | j                         \  }}|j	                         }t        |      dk7  rt        d      t        t        t        |            S )ap  
    Find the terminal size using the external command ``stty size``.

    :param stream: A stream connected to the terminal (a file object).
    :returns: A tuple of two integers with the line and column count.
    :raises: This function can raise exceptions but I'm not going to document
             them here, you should be using :func:`find_terminal_size()`.
    sttysize)r   r   r1   z Invalid output from `stty size'!)

subprocessPopenPIPEcommunicater   rd   r   rb   rh   ri   )r   r   r   tokenss       rQ   r#   r#     sr     VV,#-??#-??4D %%'NFF\\^F
6{a:;;S&!""rR   c                     | rt         | v rddg}n!t        j                  j                  dd      g}t        j                  j                  |d         dk(  r"|j                  d       |j                  d       |S )a  
    Get the command to show a text on the terminal using a pager.

    :param text: The text to print to the terminal (a string).
    :returns: A list of strings with the pager command and arguments.

    The use of a pager helps to avoid the wall of text effect where the user
    has to scroll up to see where the output began (not very user friendly).

    If the given text contains ANSI escape sequences the command ``less
    --RAW-CONTROL-CHARS`` is used, otherwise the environment variable
    ``$PAGER`` is used (if ``$PAGER`` isn't set :man:`less` is used).

    When the selected pager is :man:`less`, the following options are used to
    make the experience more user friendly:

    - ``--quit-if-one-screen`` causes :man:`less` to automatically exit if the
      entire text can be displayed on the first screen. This makes the use of a
      pager transparent for smaller texts (because the operator doesn't have to
      quit the pager).

    - ``--no-init`` prevents :man:`less` from clearing the screen when it
      exits. This ensures that the operator gets a chance to review the text
      (for example a usage message) after quitting the pager, while composing
      the next command.
    lessz--RAW-CONTROL-CHARSPAGERr   z	--no-initz--quit-if-one-screen)r   r   r   r`   pathbasenamerf   )rN   command_lines     rQ   r$   r$   -  sp    8 D  56

w78 
wwQ(F2K(23rR   c                      t               r9	 t        d t        j                         j	                  d      D              } | dk\  S y# t
        $ r Y yw xY w)a  
    Check if we're running on a Windows 10 release with native support for ANSI escape sequences.

    :returns: :data:`True` if so, :data:`False` otherwise.

    The :func:`~humanfriendly.decorators.cached` decorator is used as a minor
    performance optimization. Semantically this should have zero impact because
    the answer doesn't change in the lifetime of a computer process.
    c              3   2   K   | ]  }t        |        y wN)ri   ).0cs     rQ   	<genexpr>z3have_windows_native_ansi_support.<locals>.<genexpr>f  s     M/L!s1v/Ls   .)
   r   i98  F)r   rb   platformversionr   r   )
componentss    rQ   r%   r%   U  s[     |	
 Mx/?/?/A/G/G/LMMJ//   		s   7A 	AAc                 X    t        t        j                  t        |       dz   g|i | y)a[  
    Print a formatted message to the standard error stream.

    For details about argument handling please refer to
    :func:`~humanfriendly.text.format()`.

    Renders the message using :func:`~humanfriendly.text.format()` and writes
    the resulting string (followed by a newline) to :data:`sys.stderr` using
    :func:`auto_encode()`.
    r   N)r   r   r   r   rN   r   rp   s      rQ   r&   r&   m  &     

M$/$6DDDrR   c                 X    t        t        j                  t        |       dz   g|i | y)a\  
    Print a formatted message to the standard output stream.

    For details about argument handling please refer to
    :func:`~humanfriendly.text.format()`.

    Renders the message using :func:`~humanfriendly.text.format()` and writes
    the resulting string (followed by a newline) to :data:`sys.stdout` using
    :func:`auto_encode()`.
    r   N)r   r   r   r   r   s      rQ   r'   r'   {  r   rR   c                 F    | j                  dd      j                  dd      S )z
    Remove `readline hints`_ from a string.

    :param text: The text to strip (a string).
    :returns: The stripped text.
    rJ   )replaceexprs    rQ   r(   r(     s"     <<#++FB77rR   c                     d| z   dz   S )z
    Wrap an ANSI escape sequence in `readline hints`_.

    :param text: The text with the escape sequence to wrap (a string).
    :returns: The wrapped text.

    .. _readline hints: http://superuser.com/a/301355
    r   r   r{   r   s    rQ   r)   r)     s     D=6!!rR   c                    t               rmt        |       }t        |d         rTt        j                  |t        j
                        }t        |       r| j                  |      } |j                  |        yt        |        y)a'  
    Print a large text to the terminal using a pager.

    :param formatted_text: The text to print to the terminal (a string).
    :param encoding: The name of the text encoding used to encode the formatted
                     text if the formatted text is a Unicode string (a string,
                     defaults to :data:`DEFAULT_ENCODING`).

    When :func:`connected_to_terminal()` returns :data:`True` a pager is used
    to show the text on the terminal, otherwise the text is printed directly
    without invoking a pager.

    The use of a pager helps to avoid the wall of text effect where the user
    has to scroll up to see where the output began (not very user friendly).

    Refer to :func:`get_pager_command()` for details about the command line
    that's used to invoke the pager.
    r   )r   )inputN)
r   r$   r   r   r   r   r   r   r   r'   )formatted_textencodingr   pagers       rQ   r*   r*     sj    & (8a!$$\IE.)!/!6!6x!@N3
>rR   c                     t               r5dt        j                  v }dt        j                  v }t               }|s|s|syt        |       S )a3  
    Check if a stream is connected to a terminal that supports ANSI escape sequences.

    :param stream: The stream to check (a file-like object,
                   defaults to :data:`sys.stdout`).
    :returns: :data:`True` if the terminal supports ANSI escape sequences,
              :data:`False` otherwise.

    This function was originally inspired by the implementation of
    `django.core.management.color.supports_color()
    <https://github.com/django/django/blob/master/django/core/management/color.py>`_
    but has since evolved significantly.
    r   r   F)r   r   r   r   modulesr%   r   )r   have_ansiconhave_coloramahave_native_supports       rQ   r+   r+     sD     | BJJ."ckk1>@1D ((rR   c                 b    t        t        j                        rt        |       } t	        |        y)a  
    Print a human friendly usage message to the terminal.

    :param text: The usage message to print (a string).

    This function does two things:

    1. If :data:`sys.stdout` is connected to a terminal (see
       :func:`connected_to_terminal()`) then the usage message is formatted
       using :func:`.format_usage()`.
    2. The usage message is shown using a pager (see :func:`show_pager()`).
    N)r+   r   r   r   r*   )
usage_texts    rQ   r,   r,     s"      

+!*-
zrR   c                     t        |       } t        t        j                        rt	        | d      } t        t        j                  | dz   g|i | y)a  
    Show a warning message on the terminal.

    For details about argument handling please refer to
    :func:`~humanfriendly.text.format()`.

    Renders the message using :func:`~humanfriendly.text.format()` and writes
    the resulting string (followed by a newline) to :data:`sys.stderr` using
    :func:`auto_encode()`.

    If :data:`sys.stderr` is connected to a terminal that supports colors,
    :func:`ansi_wrap()` is used to color the message in a red font (to make
    the warning stand out from surrounding text).
    r8   )rT   r   N)r   r+   r   r   r   r   r   s      rQ   r-   r-     sB     D

+U+

D4K5$5"5rR   z'humanfriendly.usage.find_meta_variablesz humanfriendly.usage.format_usagez(humanfriendly.terminal.html.html_to_ansiz)humanfriendly.terminal.html.HTMLConverter)module_namefind_meta_variablesr   html_to_ansiHTMLConverter)Tr   )D__doc__r   rj   r   r   rK   r   r   r   r   r   r   r   humanfriendly.compatr   r   r   r   humanfriendly.decoratorsr   humanfriendly.deprecationr   humanfriendly.textr	   r
   humanfriendly.usager   __all__r   r   r   r   r   r   dictr   r   compilerL   r   r   r   r   r   r`   r   r   r   r   r   r   r   r   r    r!   r"   r#   r$   r%   r&   r'   r(   r)   r*   r+   r,   r-   __name__r{   rR   rQ   <module>r      s  $   	  	  

J
 N M + 4 2 ,#J  8 >(" D (+
 ;h&  Bh&  BaQa1ST\]^  QaQZ[\  "rzz"2YRYY5O"OP   = ?  .**..!@'J"^B!4<2GT$ 0' 0'f-*`$#&%P  .EE8	" )9 >)0$6,  B3 <=
k  Js   D? ?E
	E
