SILENT KILLERPanel

Current Path: > > opt > cloudlinux > venv > lib64 > python3.11 > > site-packages > pip > _vendor > rich


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/cloudlinux/venv/lib64/python3.11//site-packages/pip/_vendor/rich

NameTypeSizeLast ModifiedActions
__pycache__ Directory - -
__init__.py File 6090 bytes April 17 2025 13:10:58.
__main__.py File 8477 bytes April 17 2025 13:10:58.
_cell_widths.py File 10209 bytes April 17 2025 13:10:58.
_emoji_codes.py File 140235 bytes April 17 2025 13:10:58.
_emoji_replace.py File 1064 bytes April 17 2025 13:10:58.
_export_format.py File 2128 bytes April 17 2025 13:10:58.
_extension.py File 265 bytes April 17 2025 13:10:58.
_fileno.py File 799 bytes April 17 2025 13:10:58.
_inspect.py File 9655 bytes April 17 2025 13:10:58.
_log_render.py File 3225 bytes April 17 2025 13:10:58.
_loop.py File 1236 bytes April 17 2025 13:10:58.
_null_file.py File 1394 bytes April 17 2025 13:10:58.
_palettes.py File 7063 bytes April 17 2025 13:10:58.
_pick.py File 423 bytes April 17 2025 13:10:58.
_ratio.py File 5471 bytes April 17 2025 13:10:58.
_spinners.py File 19919 bytes April 17 2025 13:10:58.
_stack.py File 351 bytes April 17 2025 13:10:58.
_timer.py File 417 bytes April 17 2025 13:10:58.
_win32_console.py File 22755 bytes April 17 2025 13:10:58.
_windows.py File 1925 bytes April 17 2025 13:10:58.
_windows_renderer.py File 2783 bytes April 17 2025 13:10:58.
_wrap.py File 3404 bytes April 17 2025 13:10:58.
abc.py File 890 bytes April 17 2025 13:10:58.
align.py File 10469 bytes April 17 2025 13:10:58.
ansi.py File 6921 bytes April 17 2025 13:10:58.
bar.py File 3263 bytes April 17 2025 13:10:58.
box.py File 10831 bytes April 17 2025 13:10:58.
cells.py File 5130 bytes April 17 2025 13:10:58.
color.py File 18211 bytes April 17 2025 13:10:58.
color_triplet.py File 1054 bytes April 17 2025 13:10:58.
columns.py File 7131 bytes April 17 2025 13:10:58.
console.py File 100156 bytes April 17 2025 13:10:58.
constrain.py File 1288 bytes April 17 2025 13:10:58.
containers.py File 5502 bytes April 17 2025 13:10:58.
control.py File 6630 bytes April 17 2025 13:10:58.
default_styles.py File 8159 bytes April 17 2025 13:10:58.
diagnose.py File 972 bytes April 17 2025 13:10:58.
emoji.py File 2501 bytes April 17 2025 13:10:58.
errors.py File 642 bytes April 17 2025 13:10:58.
file_proxy.py File 1683 bytes April 17 2025 13:10:58.
filesize.py File 2484 bytes April 17 2025 13:10:58.
highlighter.py File 9586 bytes April 17 2025 13:10:58.
json.py File 5031 bytes April 17 2025 13:10:58.
jupyter.py File 3252 bytes April 17 2025 13:10:58.
layout.py File 14004 bytes April 17 2025 13:10:58.
live.py File 14270 bytes April 17 2025 13:10:58.
live_render.py File 3666 bytes April 17 2025 13:10:58.
logging.py File 12458 bytes April 17 2025 13:10:58.
markup.py File 8451 bytes April 17 2025 13:10:58.
measure.py File 5305 bytes April 17 2025 13:10:58.
padding.py File 4908 bytes April 17 2025 13:10:58.
pager.py File 828 bytes April 17 2025 13:10:58.
palette.py File 3396 bytes April 17 2025 13:10:58.
panel.py File 11235 bytes April 17 2025 13:10:58.
pretty.py File 36391 bytes April 17 2025 13:10:58.
progress.py File 60357 bytes April 17 2025 13:10:58.
progress_bar.py File 8162 bytes April 17 2025 13:10:58.
prompt.py File 12447 bytes April 17 2025 13:10:58.
protocol.py File 1391 bytes April 17 2025 13:10:58.
py.typed File 0 bytes April 17 2025 13:10:58.
region.py File 166 bytes April 17 2025 13:10:58.
repr.py File 4431 bytes April 17 2025 13:10:58.
rule.py File 4602 bytes April 17 2025 13:10:58.
scope.py File 2843 bytes April 17 2025 13:10:58.
screen.py File 1591 bytes April 17 2025 13:10:58.
segment.py File 24743 bytes April 17 2025 13:10:58.
spinner.py File 4364 bytes April 17 2025 13:10:58.
status.py File 4424 bytes April 17 2025 13:10:58.
style.py File 27067 bytes April 17 2025 13:10:58.
styled.py File 1258 bytes April 17 2025 13:10:58.
syntax.py File 35763 bytes April 17 2025 13:10:58.
table.py File 40103 bytes April 17 2025 13:10:58.
terminal_theme.py File 3370 bytes April 17 2025 13:10:58.
text.py File 47552 bytes April 17 2025 13:10:58.
theme.py File 3771 bytes April 17 2025 13:10:58.
themes.py File 102 bytes April 17 2025 13:10:58.
traceback.py File 31797 bytes April 17 2025 13:10:58.
tree.py File 9451 bytes April 17 2025 13:10:58.

Reading File: //opt/cloudlinux/venv/lib64/python3.11//site-packages/pip/_vendor/rich/markup.py

import re
from ast import literal_eval
from operator import attrgetter
from typing import Callable, Iterable, List, Match, NamedTuple, Optional, Tuple, Union

from ._emoji_replace import _emoji_replace
from .emoji import EmojiVariant
from .errors import MarkupError
from .style import Style
from .text import Span, Text

RE_TAGS = re.compile(
    r"""((\\*)\[([a-z#/@][^[]*?)])""",
    re.VERBOSE,
)

RE_HANDLER = re.compile(r"^([\w.]*?)(\(.*?\))?$")


class Tag(NamedTuple):
    """A tag in console markup."""

    name: str
    """The tag name. e.g. 'bold'."""
    parameters: Optional[str]
    """Any additional parameters after the name."""

    def __str__(self) -> str:
        return (
            self.name if self.parameters is None else f"{self.name} {self.parameters}"
        )

    @property
    def markup(self) -> str:
        """Get the string representation of this tag."""
        return (
            f"[{self.name}]"
            if self.parameters is None
            else f"[{self.name}={self.parameters}]"
        )


_ReStringMatch = Match[str]  # regex match object
_ReSubCallable = Callable[[_ReStringMatch], str]  # Callable invoked by re.sub
_EscapeSubMethod = Callable[[_ReSubCallable, str], str]  # Sub method of a compiled re


def escape(
    markup: str,
    _escape: _EscapeSubMethod = re.compile(r"(\\*)(\[[a-z#/@][^[]*?])").sub,
) -> str:
    """Escapes text so that it won't be interpreted as markup.

    Args:
        markup (str): Content to be inserted in to markup.

    Returns:
        str: Markup with square brackets escaped.
    """

    def escape_backslashes(match: Match[str]) -> str:
        """Called by re.sub replace matches."""
        backslashes, text = match.groups()
        return f"{backslashes}{backslashes}\\{text}"

    markup = _escape(escape_backslashes, markup)
    if markup.endswith("\\") and not markup.endswith("\\\\"):
        return markup + "\\"

    return markup


def _parse(markup: str) -> Iterable[Tuple[int, Optional[str], Optional[Tag]]]:
    """Parse markup in to an iterable of tuples of (position, text, tag).

    Args:
        markup (str): A string containing console markup

    """
    position = 0
    _divmod = divmod
    _Tag = Tag
    for match in RE_TAGS.finditer(markup):
        full_text, escapes, tag_text = match.groups()
        start, end = match.span()
        if start > position:
            yield start, markup[position:start], None
        if escapes:
            backslashes, escaped = _divmod(len(escapes), 2)
            if backslashes:
                # Literal backslashes
                yield start, "\\" * backslashes, None
                start += backslashes * 2
            if escaped:
                # Escape of tag
                yield start, full_text[len(escapes) :], None
                position = end
                continue
        text, equals, parameters = tag_text.partition("=")
        yield start, None, _Tag(text, parameters if equals else None)
        position = end
    if position < len(markup):
        yield position, markup[position:], None


def render(
    markup: str,
    style: Union[str, Style] = "",
    emoji: bool = True,
    emoji_variant: Optional[EmojiVariant] = None,
) -> Text:
    """Render console markup in to a Text instance.

    Args:
        markup (str): A string containing console markup.
        style: (Union[str, Style]): The style to use.
        emoji (bool, optional): Also render emoji code. Defaults to True.
        emoji_variant (str, optional): Optional emoji variant, either "text" or "emoji". Defaults to None.


    Raises:
        MarkupError: If there is a syntax error in the markup.

    Returns:
        Text: A test instance.
    """
    emoji_replace = _emoji_replace
    if "[" not in markup:
        return Text(
            emoji_replace(markup, default_variant=emoji_variant) if emoji else markup,
            style=style,
        )
    text = Text(style=style)
    append = text.append
    normalize = Style.normalize

    style_stack: List[Tuple[int, Tag]] = []
    pop = style_stack.pop

    spans: List[Span] = []
    append_span = spans.append

    _Span = Span
    _Tag = Tag

    def pop_style(style_name: str) -> Tuple[int, Tag]:
        """Pop tag matching given style name."""
        for index, (_, tag) in enumerate(reversed(style_stack), 1):
            if tag.name == style_name:
                return pop(-index)
        raise KeyError(style_name)

    for position, plain_text, tag in _parse(markup):
        if plain_text is not None:
            # Handle open brace escapes, where the brace is not part of a tag.
            plain_text = plain_text.replace("\\[", "[")
            append(emoji_replace(plain_text) if emoji else plain_text)
        elif tag is not None:
            if tag.name.startswith("/"):  # Closing tag
                style_name = tag.name[1:].strip()

                if style_name:  # explicit close
                    style_name = normalize(style_name)
                    try:
                        start, open_tag = pop_style(style_name)
                    except KeyError:
                        raise MarkupError(
                            f"closing tag '{tag.markup}' at position {position} doesn't match any open tag"
                        ) from None
                else:  # implicit close
                    try:
                        start, open_tag = pop()
                    except IndexError:
                        raise MarkupError(
                            f"closing tag '[/]' at position {position} has nothing to close"
                        ) from None

                if open_tag.name.startswith("@"):
                    if open_tag.parameters:
                        handler_name = ""
                        parameters = open_tag.parameters.strip()
                        handler_match = RE_HANDLER.match(parameters)
                        if handler_match is not None:
                            handler_name, match_parameters = handler_match.groups()
                            parameters = (
                                "()" if match_parameters is None else match_parameters
                            )

                        try:
                            meta_params = literal_eval(parameters)
                        except SyntaxError as error:
                            raise MarkupError(
                                f"error parsing {parameters!r} in {open_tag.parameters!r}; {error.msg}"
                            )
                        except Exception as error:
                            raise MarkupError(
                                f"error parsing {open_tag.parameters!r}; {error}"
                            ) from None

                        if handler_name:
                            meta_params = (
                                handler_name,
                                meta_params
                                if isinstance(meta_params, tuple)
                                else (meta_params,),
                            )

                    else:
                        meta_params = ()

                    append_span(
                        _Span(
                            start, len(text), Style(meta={open_tag.name: meta_params})
                        )
                    )
                else:
                    append_span(_Span(start, len(text), str(open_tag)))

            else:  # Opening tag
                normalized_tag = _Tag(normalize(tag.name), tag.parameters)
                style_stack.append((len(text), normalized_tag))

    text_length = len(text)
    while style_stack:
        start, tag = style_stack.pop()
        style = str(tag)
        if style:
            append_span(_Span(start, text_length, style))

    text.spans = sorted(spans[::-1], key=attrgetter("start"))
    return text


if __name__ == "__main__":  # pragma: no cover
    MARKUP = [
        "[red]Hello World[/red]",
        "[magenta]Hello [b]World[/b]",
        "[bold]Bold[italic] bold and italic [/bold]italic[/italic]",
        "Click [link=https://www.willmcgugan.com]here[/link] to visit my Blog",
        ":warning-emoji: [bold red blink] DANGER![/]",
    ]

    from pip._vendor.rich import print
    from pip._vendor.rich.table import Table

    grid = Table("Markup", "Result", padding=(0, 1))

    for markup in MARKUP:
        grid.add_row(Text(markup), markup)

    print(grid)

SILENT KILLER Tool