Coverage for /pythoncovmergedfiles/medio/medio/usr/local/lib/python3.8/site-packages/rich/markup.py: 64%
115 statements
« prev ^ index » next coverage.py v7.0.1, created at 2022-12-25 06:11 +0000
« prev ^ index » next coverage.py v7.0.1, created at 2022-12-25 06:11 +0000
1import re
2from ast import literal_eval
3from operator import attrgetter
4from typing import Callable, Iterable, List, Match, NamedTuple, Optional, Tuple, Union
6from ._emoji_replace import _emoji_replace
7from .emoji import EmojiVariant
8from .errors import MarkupError
9from .style import Style
10from .text import Span, Text
12RE_TAGS = re.compile(
13 r"""((\\*)\[([a-z#/@][^[]*?)])""",
14 re.VERBOSE,
15)
17RE_HANDLER = re.compile(r"^([\w.]*?)(\(.*?\))?$")
20class Tag(NamedTuple):
21 """A tag in console markup."""
23 name: str
24 """The tag name. e.g. 'bold'."""
25 parameters: Optional[str]
26 """Any additional parameters after the name."""
28 def __str__(self) -> str:
29 return (
30 self.name if self.parameters is None else f"{self.name} {self.parameters}"
31 )
33 @property
34 def markup(self) -> str:
35 """Get the string representation of this tag."""
36 return (
37 f"[{self.name}]"
38 if self.parameters is None
39 else f"[{self.name}={self.parameters}]"
40 )
43_ReStringMatch = Match[str] # regex match object
44_ReSubCallable = Callable[[_ReStringMatch], str] # Callable invoked by re.sub
45_EscapeSubMethod = Callable[[_ReSubCallable, str], str] # Sub method of a compiled re
48def escape(
49 markup: str,
50 _escape: _EscapeSubMethod = re.compile(r"(\\*)(\[[a-z#/@][^[]*?])").sub,
51) -> str:
52 """Escapes text so that it won't be interpreted as markup.
54 Args:
55 markup (str): Content to be inserted in to markup.
57 Returns:
58 str: Markup with square brackets escaped.
59 """
61 def escape_backslashes(match: Match[str]) -> str:
62 """Called by re.sub replace matches."""
63 backslashes, text = match.groups()
64 return f"{backslashes}{backslashes}\\{text}"
66 markup = _escape(escape_backslashes, markup)
67 return markup
70def _parse(markup: str) -> Iterable[Tuple[int, Optional[str], Optional[Tag]]]:
71 """Parse markup in to an iterable of tuples of (position, text, tag).
73 Args:
74 markup (str): A string containing console markup
76 """
77 position = 0
78 _divmod = divmod
79 _Tag = Tag
80 for match in RE_TAGS.finditer(markup):
81 full_text, escapes, tag_text = match.groups()
82 start, end = match.span()
83 if start > position:
84 yield start, markup[position:start], None
85 if escapes:
86 backslashes, escaped = _divmod(len(escapes), 2)
87 if backslashes:
88 # Literal backslashes
89 yield start, "\\" * backslashes, None
90 start += backslashes * 2
91 if escaped:
92 # Escape of tag
93 yield start, full_text[len(escapes) :], None
94 position = end
95 continue
96 text, equals, parameters = tag_text.partition("=")
97 yield start, None, _Tag(text, parameters if equals else None)
98 position = end
99 if position < len(markup):
100 yield position, markup[position:], None
103def render(
104 markup: str,
105 style: Union[str, Style] = "",
106 emoji: bool = True,
107 emoji_variant: Optional[EmojiVariant] = None,
108) -> Text:
109 """Render console markup in to a Text instance.
111 Args:
112 markup (str): A string containing console markup.
113 emoji (bool, optional): Also render emoji code. Defaults to True.
115 Raises:
116 MarkupError: If there is a syntax error in the markup.
118 Returns:
119 Text: A test instance.
120 """
121 emoji_replace = _emoji_replace
122 if "[" not in markup:
123 return Text(
124 emoji_replace(markup, default_variant=emoji_variant) if emoji else markup,
125 style=style,
126 )
127 text = Text(style=style)
128 append = text.append
129 normalize = Style.normalize
131 style_stack: List[Tuple[int, Tag]] = []
132 pop = style_stack.pop
134 spans: List[Span] = []
135 append_span = spans.append
137 _Span = Span
138 _Tag = Tag
140 def pop_style(style_name: str) -> Tuple[int, Tag]:
141 """Pop tag matching given style name."""
142 for index, (_, tag) in enumerate(reversed(style_stack), 1):
143 if tag.name == style_name:
144 return pop(-index)
145 raise KeyError(style_name)
147 for position, plain_text, tag in _parse(markup):
148 if plain_text is not None:
149 # Handle open brace escapes, where the brace is not part of a tag.
150 plain_text = plain_text.replace("\\[", "[")
151 append(emoji_replace(plain_text) if emoji else plain_text)
152 elif tag is not None:
153 if tag.name.startswith("/"): # Closing tag
154 style_name = tag.name[1:].strip()
156 if style_name: # explicit close
157 style_name = normalize(style_name)
158 try:
159 start, open_tag = pop_style(style_name)
160 except KeyError:
161 raise MarkupError(
162 f"closing tag '{tag.markup}' at position {position} doesn't match any open tag"
163 ) from None
164 else: # implicit close
165 try:
166 start, open_tag = pop()
167 except IndexError:
168 raise MarkupError(
169 f"closing tag '[/]' at position {position} has nothing to close"
170 ) from None
172 if open_tag.name.startswith("@"):
173 if open_tag.parameters:
174 handler_name = ""
175 parameters = open_tag.parameters.strip()
176 handler_match = RE_HANDLER.match(parameters)
177 if handler_match is not None:
178 handler_name, match_parameters = handler_match.groups()
179 parameters = (
180 "()" if match_parameters is None else match_parameters
181 )
183 try:
184 meta_params = literal_eval(parameters)
185 except SyntaxError as error:
186 raise MarkupError(
187 f"error parsing {parameters!r} in {open_tag.parameters!r}; {error.msg}"
188 )
189 except Exception as error:
190 raise MarkupError(
191 f"error parsing {open_tag.parameters!r}; {error}"
192 ) from None
194 if handler_name:
195 meta_params = (
196 handler_name,
197 meta_params
198 if isinstance(meta_params, tuple)
199 else (meta_params,),
200 )
202 else:
203 meta_params = ()
205 append_span(
206 _Span(
207 start, len(text), Style(meta={open_tag.name: meta_params})
208 )
209 )
210 else:
211 append_span(_Span(start, len(text), str(open_tag)))
213 else: # Opening tag
214 normalized_tag = _Tag(normalize(tag.name), tag.parameters)
215 style_stack.append((len(text), normalized_tag))
217 text_length = len(text)
218 while style_stack:
219 start, tag = style_stack.pop()
220 style = str(tag)
221 if style:
222 append_span(_Span(start, text_length, style))
224 text.spans = sorted(spans[::-1], key=attrgetter("start"))
225 return text
228if __name__ == "__main__": # pragma: no cover
230 MARKUP = [
231 "[red]Hello World[/red]",
232 "[magenta]Hello [b]World[/b]",
233 "[bold]Bold[italic] bold and italic [/bold]italic[/italic]",
234 "Click [link=https://www.willmcgugan.com]here[/link] to visit my Blog",
235 ":warning-emoji: [bold red blink] DANGER![/]",
236 ]
238 from rich import print
239 from rich.table import Table
241 grid = Table("Markup", "Result", padding=(0, 1))
243 for markup in MARKUP:
244 grid.add_row(Text(markup), markup)
246 print(grid)